博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
IOS实现自动循环滚动广告--ScrollView的优化和封装
阅读量:6655 次
发布时间:2019-06-25

本文共 7321 字,大约阅读时间需要 24 分钟。

一、问题分析

  在许多App中,我们都会见到循环滚动的视图,比如广告,其实想实现这个功能并不难,用ScrollView就可以轻松完成,但是在制作的过程中还存在几个小问题,如果能够正确的处理好这些小问题,无论从效果还是性能上都会得到优化。

问题一

  第一个问题是如何用ScrollView来展示N个视图。想要实现这个效果,可以把N个视图依次按顺序添加到ScrollView上,然后把 ScrollView的contentSize设置为N个视图的尺寸,通过滑动ScrollView来查看加在上面的视图。

问题二

  第二个问题是如何完成图片的循环滚动,也就是展示完最后一张之后会接着展示第一张,形成图片的循环展示。想要实现这个效果,首先需要让ScrollView实现自动分页,这样可以保证滑动结束展示的是完整的视图;其次,需要根据当前展示的页数来设置ScrollView的contentOffset。

  对于第一个问题的解决是用的最简单的方式,但实际上忽略了一个很重要的问题,那就是如果要展示的视图数量N非常大的时候,我们该如何做呢?假设通过ScrollView来展示的每个视图的宽度恰好是屏幕的宽度,那么在展示的时候,其实能够呈现在我们眼前的最多只有两个视图,也就是要么是完整的一个视图,要么是两个不完整的视图。因此,我们只需要有三个视图,就能够完成循环的展示。

问题三

  第三个问题是在循环滚动的过程中,希望知道当前的页数,这个页数可以通过contentOffset.x来计算,通常会用UIPageControl来表示。此外,当点击某一个视图的时候,要能够知道当前点击的视图是哪一个。

问题四

  第四个问题是自动展示下一页的功能,这个需要写好跳到下一页的方法,然后通过NSTimer定时器来完成。

  除了上面的几个问题,大家也可以为其添加更多的功能。那么对于ScrollView自动翻页这样通用的功能,最好的方式是将其封装起来,这样可以大大的提高效率。下面的代码是把UIScrollView、UIPageControl封装到了一个UIView中,而其中的ScrollView用来循环展示多张图片。

二、功能实现

1、封装Scrollview代码.h:

//  WHScrollAndPageView.h//  循环滚动视图////  Created by jereh on 15-3-15.//  Copyright (c) 2015年 jereh. All rights reserved.//#import 
@protocol WHcrollViewViewDelegate;@interface WHScrollAndPageView : UIView
{ __unsafe_unretained id
_delegate;}@property (nonatomic, assign) id
delegate;@property (nonatomic, assign) NSInteger currentPage;@property (nonatomic, strong) NSMutableArray *imageViewAry;@property (nonatomic, readonly) UIScrollView *scrollView;@property (nonatomic, readonly) UIPageControl *pageControl;-(void)shouldAutoShow:(BOOL)shouldStart;@end@protocol WHcrollViewViewDelegate
@optional- (void)didClickPage:(WHScrollAndPageView *)view atIndex:(NSInteger)index;@end

2、封装Scrollview代码.m:

//  WHScrollAndPageView.m//  循环滚动视图////  Created by jereh on 15-3-15.//  Copyright (c) 2015年 jereh. All rights reserved.//#import "WHScrollAndPageView.h"@interface WHScrollAndPageView (){    UIView *_firstView;    UIView *_middleView;    UIView *_lastView;        float _viewWidth;    float _viewHeight;        NSTimer *_autoScrollTimer;        UITapGestureRecognizer *_tap;}@end@implementation WHScrollAndPageView- (id)initWithFrame:(CGRect)frame{    self = [super initWithFrame:frame];    if (self) {            _viewWidth = self.bounds.size.width;        _viewHeight = self.bounds.size.height;                //设置scrollview        _scrollView = [[UIScrollView alloc] initWithFrame:CGRectMake(0, 0, _viewWidth, _viewHeight)];        _scrollView.delegate = self;        _scrollView.contentSize = CGSizeMake(_viewWidth * 3, _viewHeight);        _scrollView.showsHorizontalScrollIndicator = NO;        _scrollView.pagingEnabled = YES;        _scrollView.backgroundColor = [UIColor blackColor];        _scrollView.delegate = self;        [self addSubview:_scrollView];                //设置分页        _pageControl = [[UIPageControl alloc] initWithFrame:CGRectMake(0, _viewHeight-30, _viewWidth, 30)];        _pageControl.userInteractionEnabled = NO;        _pageControl.currentPageIndicatorTintColor = [UIColor redColor];        _pageControl.pageIndicatorTintColor = [UIColor whiteColor];        [self addSubview:_pageControl];                //设置单击手势        _tap = [[UITapGestureRecognizer alloc]initWithTarget:self action:@selector(handleTap:)];        _tap.numberOfTapsRequired = 1;        _tap.numberOfTouchesRequired = 1;        [_scrollView addGestureRecognizer:_tap];    }    return self;}#pragma mark 单击手势-(void)handleTap:(UITapGestureRecognizer*)sender{    if ([_delegate respondsToSelector:@selector(didClickPage:atIndex:)]) {        [_delegate didClickPage:self atIndex:_currentPage+1];    }}#pragma mark 设置imageViewAry-(void)setImageViewAry:(NSMutableArray *)imageViewAry{    if (imageViewAry) {        _imageViewAry = imageViewAry;        _currentPage = 0; //默认为第0页                _pageControl.numberOfPages = _imageViewAry.count;    }        [self reloadData];}#pragma mark 刷新view页面-(void)reloadData{    [_firstView removeFromSuperview];    [_middleView removeFromSuperview];    [_lastView removeFromSuperview];        //从数组中取到对应的图片view加到已定义的三个view中    if (_currentPage==0) {        _firstView = [_imageViewAry lastObject];        _middleView = [_imageViewAry objectAtIndex:_currentPage];        _lastView = [_imageViewAry objectAtIndex:_currentPage+1];    }    else if (_currentPage == _imageViewAry.count-1)    {        _firstView = [_imageViewAry objectAtIndex:_currentPage-1];        _middleView = [_imageViewAry objectAtIndex:_currentPage];        _lastView = [_imageViewAry firstObject];    }    else    {        _firstView = [_imageViewAry objectAtIndex:_currentPage-1];        _middleView = [_imageViewAry objectAtIndex:_currentPage];        _lastView = [_imageViewAry objectAtIndex:_currentPage+1];    }        //设置三个view的frame,加到scrollview上    _firstView.frame = CGRectMake(0, 0, _viewWidth, _viewHeight);    _middleView.frame = CGRectMake(_viewWidth, 0, _viewWidth, _viewHeight);    _lastView.frame = CGRectMake(_viewWidth*2, 0, _viewWidth, _viewHeight);    [_scrollView addSubview:_firstView];    [_scrollView addSubview:_middleView];    [_scrollView addSubview:_lastView];        //设置当前的分页    _pageControl.currentPage = _currentPage;        //显示中间页    _scrollView.contentOffset = CGPointMake(_viewWidth, 0);}#pragma mark scrollvie停止滑动-(void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView{    //手动滑动时候暂停自动替换    [_autoScrollTimer invalidate];    _autoScrollTimer = nil;    _autoScrollTimer = [NSTimer scheduledTimerWithTimeInterval:2 target:self selector:@selector(autoShowNextImage) userInfo:nil repeats:YES];        //得到当前页数    float x = _scrollView.contentOffset.x;        //往前翻    if (x<=0) {        if (_currentPage-1<0) {            _currentPage = _imageViewAry.count-1;        }else{            _currentPage --;        }    }        //往后翻    if (x>=_viewWidth*2) {        if (_currentPage==_imageViewAry.count-1) {            _currentPage = 0;        }else{            _currentPage ++;        }    }        [self reloadData];}#pragma mark 自动滚动-(void)shouldAutoShow:(BOOL)shouldStart{    if (shouldStart)  //开启自动翻页    {        if (!_autoScrollTimer) {            _autoScrollTimer = [NSTimer scheduledTimerWithTimeInterval:2 target:self selector:@selector(autoShowNextImage) userInfo:nil repeats:YES];        }    }    else   //关闭自动翻页    {        if (_autoScrollTimer.isValid) {            [_autoScrollTimer invalidate];            _autoScrollTimer = nil;        }    }}#pragma mark 展示下一页-(void)autoShowNextImage{    if (_currentPage == _imageViewAry.count-1) {        _currentPage = 0;    }else{        _currentPage ++;    }        [self reloadData];}@end

3、使用封装好的Scrollview代码.m:

//  ViewController.m//  循环滚动视图////  Created by jereh on 15-3-15.//  Copyright (c) 2015年 jereh. All rights reserved.//#import "ViewController.h"#import "WHScrollAndPageView.h"#define NUM 10@interface ViewController ()
{ WHScrollAndPageView *_whView;}@end@implementation ViewController- (void)viewDidLoad{ [super viewDidLoad]; //创建view (view中包含UIScrollView、UIPageControl,设置frame) _whView = [[WHScrollAndPageView alloc] initWithFrame:CGRectMake(0, 44, 320, 400)]; //把N张图片放到imageview上 NSMutableArray *tempAry = [NSMutableArray array]; for (int i=1; i

 

作者:
出处:
 
本文版权归和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。
 
你可能感兴趣的文章
JVM【第九回】:【OutOfMemoryError异常之本机直接内存溢出】
查看>>
Angular
查看>>
ANTLR Reference书摘
查看>>
我的友情链接
查看>>
索引失效
查看>>
DELL云计算解决方案【我身边的戴尔企业级解决方案】
查看>>
我的友情链接
查看>>
MyBatis学习总结(14)——Mybatis使用技巧总结
查看>>
SSH 密钥 认证 多台机器
查看>>
图解TCPIP协议
查看>>
《Effective C++第三版》读书笔记——构造/析构/赋值运算
查看>>
Mysql中文汉字转拼音的实现(每个汉字转换全拼)
查看>>
Redhat 5.3 Linux内核的升级!
查看>>
Ubuntu Mate:扩展存储到整张SD卡
查看>>
学习使用clojure(1)
查看>>
EXCEL 2010规划求解基础篇
查看>>
es学习5-slowlog
查看>>
nf_conntrack: table full, dropping packet
查看>>
Linux的五个查找命令:find,locate,whereis,which,type
查看>>
KK课表抓取教务系统
查看>>