美文网首页
无限循环轮播图

无限循环轮播图

作者: Coder007 | 来源:发表于2016-03-25 15:22 被阅读991次

    无限循环轮播图

    • 在工作的过程中,很多情况下会遇到要使用轮播图,相信大家也遇到过,使用轮播图的话分两种情况:
      • 不能无限循环的轮播图
        • 这种轮播图很简单,只要使用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
    

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

    相关文章

      网友评论

          本文标题:无限循环轮播图

          本文链接:https://www.haomeiwen.com/subject/exqdlttx.html