无限循环轮播图

无限循环轮播图

  • 在工作的过程中,很多情况下会遇到要使用轮播图,相信大家也遇到过,使用轮播图的话分两种情况:
    • 不能无限循环的轮播图
      • 这种轮播图很简单,只要使用UIScrollView就能完成,也不需要什么计算什么的。设置好分页就差不多了!
    • 能无限循环的轮播图
      • 这种轮播图比较麻烦,新手估计很多都不会!

原理

无限循环不能每次都创建UIImageView,就好像不能每次都创建Cell一样,需要循环利用,那么这时候就要去想如何去设计,如何重复利用UIImageView

设计原理:
  • 在UIScrollView中添加三个UIImageView
  • 在给图片数组赋值的时候,设置pageControl的页码总数和当前页为0
  • 在layoutSubviews方法中,先设置UIScrollView的contentSize,然后设置三个UIImageView 和 pageControl的frame
  • 在自定义的updateContent方法中,需要对三个UIImageView中的图片进行更新,这个方法要在layoutSubviews方法中调用一次(最少一次)
  • 在updateContent方法中,需要去设置三个UIImageView的图片
    • for循环中每次取出一个子控件(也就是UIImageView)
    • 取出当前的pageControl的当前页码index
    • i == 0 index--;
    • i == 2 index++;
    • index < 0 index = 总页数 - 1;
    • index >= 总页数 index = 0;
    • 设置imageView的tag = index
    • 从数组中取出图片赋值给UIImageView
    • for循环结束后,需要设置偏移量在中间,也就是总是显示中间那张图片
  • scrollViewDidScroll:方法中,取出中间的那个UIImageView,然后设置pageControl的currnetPage
  • scrollViewDidEndDecelerating:方法中需要再次调用updateContent方法

如果有定时器

  • 在设置图片数组的时候需要开启定时器
  • scrollViewWillBeginDragging:方法中要停止定时器
  • scrollViewDidEndDragging:方法中要开启定时器
  • 定时器方法中使用setContentOffset: animated:方法将偏移量设置成两倍的宽度
  • scrollViewDidEndScrollingAnimation:方法中调用updateContent方法

需要点击图片做一些事情

  • 创建UIImageView的时候可以给每一个加上一个手势UITapGestureRecognizer
  • 在对应的方法中就可以通知代理点击的哪一个图片
        - (void)clickImage:(UITapGestureRecognizer *)tapGest
        {
            //tapGest.view.tag获取当前点击的UIImageView的tag
             NSLog(@"%ld",tapGest.view.tag);
        }
    

上面的方法做到了之后就可以写出一个完美的无限滚动轮播图了!当然了,设置图片轮播图的时候一般都是从右往左滚动的,也可以设置成从下往上滚动的,这个就需要在设置偏移量的时候进行判断,设置不同方向上的偏移量就可以实现不同方向上的滚动了!!

实现代码

//.h文件内容

#import <UIKit/UIKit.h>

@interface YWInfiniteScrollView : UIView

/**
 *  图片数组
 */
@property (strong, nonatomic) NSArray *images;
/**
 *  页码,可以自定义设置选中和非选中的颜色等
 */
@property (weak, nonatomic, readonly) UIPageControl *pageControl;

/**
 *  滚动方向:YES是垂直方向,NO是水平方向
 */
@property (assign, nonatomic, getter=isScrollDirectionPortrait) BOOL scrollDirectionPortrait;

@end

//.m文件内容

#import "YWInfiniteScrollView.h"
#import "UIImageView+WebCache.h"

//UIImageView总数
static int const ImageViewCount = 3;

#define kScrollViewHeigth   self.scrollView.frame.size.height
#define kScrollViewWidth    self.scrollView.frame.size.width
#define kBoundsWidth        self.bounds.size.width
#define kBoundsHeight       self.bounds.size.height

@interface YWInfiniteScrollView() <UIScrollViewDelegate>
@property (weak, nonatomic) UIScrollView *scrollView;
@property (weak, nonatomic) NSTimer *timer;

/** 第一次执行更新视图*/
@property(nonatomic, assign) BOOL isFirst;
@end

@implementation YWInfiniteScrollView

- (instancetype)initWithFrame:(CGRect)frame
{
    if (self = [super initWithFrame:frame]) {
        // 滚动视图
        UIScrollView *scrollView = [[UIScrollView alloc] init];
        scrollView.showsHorizontalScrollIndicator = NO;
        scrollView.showsVerticalScrollIndicator = NO;
        scrollView.pagingEnabled = YES;
        scrollView.bounces = NO;
        scrollView.delegate = self;
        [self addSubview:scrollView];
        self.scrollView = scrollView;

        // 图片控件
        for (int i = 0; i<ImageViewCount; i++) {
            UIImageView *imageView = [[UIImageView alloc] init];
            [scrollView addSubview:imageView];
        }

        // 页码视图
        UIPageControl *pageControl = [[UIPageControl alloc] init];
        [self addSubview:pageControl];
        _pageControl = pageControl;
    }
    return self;
}

- (void)layoutSubviews
{
    [super layoutSubviews];
    self.scrollView.frame = self.bounds;
    if (self.isScrollDirectionPortrait) {
        self.scrollView.contentSize = CGSizeMake(0, ImageViewCount * kBoundsHeight);
    } else {
        self.scrollView.contentSize = CGSizeMake(ImageViewCount * kBoundsWidth, 0);
    }

    for (int i = 0; i<ImageViewCount; i++) {
        UIImageView *imageView = self.scrollView.subviews[i];

        if (self.isScrollDirectionPortrait) {
            imageView.frame = CGRectMake(0, i * kScrollViewHeigth, kScrollViewWidth, kScrollViewHeigth);
        } else {
            imageView.frame = CGRectMake(i * kScrollViewWidth, 0, kScrollViewWidth, kScrollViewHeigth);
        }
    }
    //pageControl的frame需要调整,根据个数的多少动态调整,这里就不写了
    CGFloat pageW = 80;
    CGFloat pageH = 20;
    CGFloat pageX = kScrollViewWidth - pageW;
    CGFloat pageY = kScrollViewHeigth - pageH;
    self.pageControl.frame = CGRectMake(pageX, pageY, pageW, pageH);
    if(!self.isFirst){
        [self updateContent];
        self.isFirst = YES;
    }

}

- (void)setImages:(NSArray *)images
{
    _images = images;

    // 设置页码
    self.pageControl.numberOfPages = images.count;
    self.pageControl.currentPage = 0;

    // 开始定时器
    [self startTimer];
}

#pragma mark - <UIScrollViewDelegate>
- (void)scrollViewDidScroll:(UIScrollView *)scrollView
{
    // 找出最中间的那个图片控件
    NSInteger page = 0;
    CGFloat minDistance = MAXFLOAT;
    for (int i = 0; i<self.scrollView.subviews.count; i++) {
        UIImageView *imageView = self.scrollView.subviews[i];
        CGFloat distance = 0;
        if (self.isScrollDirectionPortrait) {
            distance = ABS(imageView.frame.origin.y - scrollView.contentOffset.y);
        } else {
            distance = ABS(imageView.frame.origin.x - scrollView.contentOffset.x);
        }
        if (distance < minDistance) {
            minDistance = distance;
            page = imageView.tag;
        }
    }
    self.pageControl.currentPage = page;
}

- (void)scrollViewWillBeginDragging:(UIScrollView *)scrollView
{
    [self stopTimer];
}

- (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate
{
    [self startTimer];
}

- (void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView
{
    [self updateContent];
}

- (void)scrollViewDidEndScrollingAnimation:(UIScrollView *)scrollView
{
    [self updateContent];
}

#pragma mark - 内容更新
- (void)updateContent
{
    // 设置图片
    for (int i = 0; i<self.scrollView.subviews.count; i++) {
        UIImageView *imageView = self.scrollView.subviews[i];
        NSInteger index = self.pageControl.currentPage;
        if (i == 0) {
            index--;
        } else if (i == 2) {
            index++;
        }
        if (index < 0) {
            index = self.pageControl.numberOfPages - 1;
        } else if (index >= self.pageControl.numberOfPages) {
            index = 0;
        }
        imageView.tag = index;
        //设置图片
       [imageView sd_setImageWithURL:[NSURL URLWithString:self.images[index]] placeholderImage:[UIImage imageNamed:@"11"]];
    }

    // 设置偏移量在中间
    if (self.isScrollDirectionPortrait) {
        self.scrollView.contentOffset = CGPointMake(0, kScrollViewHeigth);
    } else {
        self.scrollView.contentOffset = CGPointMake(kScrollViewWidth, 0);
    }
}

#pragma mark - 定时器处理
- (void)startTimer
{
    NSTimer *timer = [NSTimer timerWithTimeInterval:10 target:self selector:@selector(next) userInfo:nil repeats:YES];
    [[NSRunLoop mainRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];
    self.timer = timer;
}

- (void)stopTimer
{
    [self.timer invalidate];
    self.timer = nil;
}

- (void)next
{
    if (self.isScrollDirectionPortrait) {
        [self.scrollView setContentOffset:CGPointMake(0, 2 * kScrollViewHeigth) animated:YES];
    } else {
        [self.scrollView setContentOffset:CGPointMake(2 * kScrollViewWidth, 0) animated:YES];
    }
}
@end

代码中肯定还要许多可以改进的地方,希望大家一起交流,一起进步!

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

推荐阅读更多精彩内容