一、问题分析

  在许多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 <uikit uikit.h=""> @protocol WHcrollViewViewDelegate; @interface WHScrollAndPageView : UIView <uiscrollviewdelegate>
{
__unsafe_unretained id <whcrollviewviewdelegate> _delegate;
} @property (nonatomic, assign) id <whcrollviewviewdelegate> 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 <nsobject> @optional
- (void)didClickPage:(WHScrollAndPageView *)view atIndex:(NSInteger)index; @end</nsobject></whcrollviewviewdelegate></whcrollviewviewdelegate></uiscrollviewdelegate></uikit>

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(, , _viewWidth, _viewHeight)];
_scrollView.delegate = self;
_scrollView.contentSize = CGSizeMake(_viewWidth * , _viewHeight);
_scrollView.showsHorizontalScrollIndicator = NO;
_scrollView.pagingEnabled = YES;
_scrollView.backgroundColor = [UIColor blackColor];
_scrollView.delegate = self;
[self addSubview:_scrollView]; //设置分页
_pageControl = [[UIPageControl alloc] initWithFrame:CGRectMake(, _viewHeight-, _viewWidth, )];
_pageControl.userInteractionEnabled = NO;
_pageControl.currentPageIndicatorTintColor = [UIColor redColor];
_pageControl.pageIndicatorTintColor = [UIColor whiteColor];
[self addSubview:_pageControl]; //设置单击手势
_tap = [[UITapGestureRecognizer alloc]initWithTarget:self action:@selector(handleTap:)];
_tap.numberOfTapsRequired = ;
_tap.numberOfTouchesRequired = ;
[_scrollView addGestureRecognizer:_tap];
}
return self;
} #pragma mark 单击手势
-(void)handleTap:(UITapGestureRecognizer*)sender
{
if ([_delegate respondsToSelector:@selector(didClickPage:atIndex:)]) {
[_delegate didClickPage:self atIndex:_currentPage+];
}
} #pragma mark 设置imageViewAry
-(void)setImageViewAry:(NSMutableArray *)imageViewAry
{
if (imageViewAry) {
_imageViewAry = imageViewAry;
_currentPage = ; //默认为第0页 _pageControl.numberOfPages = _imageViewAry.count;
} [self reloadData];
} #pragma mark 刷新view页面
-(void)reloadData
{
[_firstView removeFromSuperview];
[_middleView removeFromSuperview];
[_lastView removeFromSuperview]; //从数组中取到对应的图片view加到已定义的三个view中
if (_currentPage==) {
_firstView = [_imageViewAry lastObject];
_middleView = [_imageViewAry objectAtIndex:_currentPage];
_lastView = [_imageViewAry objectAtIndex:_currentPage+];
}
else if (_currentPage == _imageViewAry.count-)
{
_firstView = [_imageViewAry objectAtIndex:_currentPage-];
_middleView = [_imageViewAry objectAtIndex:_currentPage];
_lastView = [_imageViewAry firstObject];
}
else
{
_firstView = [_imageViewAry objectAtIndex:_currentPage-];
_middleView = [_imageViewAry objectAtIndex:_currentPage];
_lastView = [_imageViewAry objectAtIndex:_currentPage+];
} //设置三个view的frame,加到scrollview上
_firstView.frame = CGRectMake(, , _viewWidth, _viewHeight);
_middleView.frame = CGRectMake(_viewWidth, , _viewWidth, _viewHeight);
_lastView.frame = CGRectMake(_viewWidth*, , _viewWidth, _viewHeight);
[_scrollView addSubview:_firstView];
[_scrollView addSubview:_middleView];
[_scrollView addSubview:_lastView]; //设置当前的分页
_pageControl.currentPage = _currentPage; //显示中间页
_scrollView.contentOffset = CGPointMake(_viewWidth, );
} #pragma mark scrollvie停止滑动
-(void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView
{
//手动滑动时候暂停自动替换
[_autoScrollTimer invalidate];
_autoScrollTimer = nil;
_autoScrollTimer = [NSTimer scheduledTimerWithTimeInterval: target:self selector:@selector(autoShowNextImage) userInfo:nil repeats:YES]; //得到当前页数
float x = _scrollView.contentOffset.x; //往前翻
if (x<=) {
if (_currentPage-<) {
_currentPage = _imageViewAry.count-;
}else{
_currentPage --;
}
} //往后翻
if (x>=_viewWidth*) {
if (_currentPage==_imageViewAry.count-) {
_currentPage = ;
}else{
_currentPage ++;
}
} [self reloadData];
} #pragma mark 自动滚动
-(void)shouldAutoShow:(BOOL)shouldStart
{
if (shouldStart) //开启自动翻页
{
if (!_autoScrollTimer) {
_autoScrollTimer = [NSTimer scheduledTimerWithTimeInterval: 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-) {
_currentPage = ;
}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 ()<whcrollviewviewdelegate>
{
WHScrollAndPageView *_whView;
} @end @implementation ViewController - (void)viewDidLoad
{
[super viewDidLoad]; //创建view (view中包含UIScrollView、UIPageControl,设置frame)
_whView = [[WHScrollAndPageView alloc] initWithFrame:CGRectMake(, , , )]; //把N张图片放到imageview上
NSMutableArray *tempAry = [NSMutableArray array];
for (int i=; i<num; _whview="" _whview.delegate="self;" animated="" imageview="[[UIImageView" imageview.image="[UIImage" index="" mark="" nsstring="" pragma="" pre="" self.view="" tempary="" uiimageview="" view="" whscrollandpageview=""><p> </p></num;></whcrollviewviewdelegate>

IOS实现自动循环滚动广告--ScrollView的优化和封装的更多相关文章

  1. UIScrollView实现自动循环滚动广告

    实现效果如下: 功能说明: 程序运行,图片自动循环播放,采用定时器实现; 当用户用手势触摸滑动时,定时器的自动播放取消,停止触摸时,自动无限播放; 代码如下 : 采用封装视图,外部进行调用即可: 1. ...

  2. IOS无限自动循环滚动banner(源码)

    本文转载至 http://blog.csdn.net/iunion/article/details/19080259  目前有很多APP都开始使用一些滚动banner,我自己也做了一个,部分算法没有深 ...

  3. 【转】Android循环滚动广告条的完美实现,封装方便,平滑过渡,从网络加载图片,点击广告进入对应网址

    Android循环滚动广告条的完美实现,封装方便,平滑过渡,从网络加载图片,点击广告进入对应网址 关注finddreams,一起分享,一起进步: http://blog.csdn.net/finddr ...

  4. iOS开发 - 循环滚动的ScrollView

    源码在这里,请多多指教. 由于开发需要,要用到循环自动滚动的scrollView,借鉴了前人的思路,重新设计了一个AutoSlideScrollView.先自吹自擂一翻吧: 借鉴UITableView ...

  5. 使用Recyclerview实现图片水平自动循环滚动

    简介: 本篇博客主要介绍的是如何使用RecyclerView实现图片水平方向自动循环(跑马灯效果) 效果图: 思路: 1.准备m张图片 1.使用Recyclerview实现,返回无数个(实际Inter ...

  6. ListView的自动循环滚动显示

    最近项目里需要做评价内容的循环滚动显示,一开始想到的就是定时器.后来查了资料才知道ListView里面有个函数smoothScrollToPosition(position),瞬间觉得简单了很多.首先 ...

  7. 自动循环滚动ScrollView

    // // SBCycleScrollView.h // SBCycleScrollView // // Created by luo.h on 15/7/12. // Copyright (c) 2 ...

  8. iOS之UIScrollView循环滚动

    #import "ViewController.h" #define SCREEN_WIDTH [UIScreen mainScreen].bounds.size.width #d ...

  9. item上下自动循环滚动显示

    //li 上下滚动 (function($){ $.fn.extend({ Scroll:function(opt,callback){ //参数初始化 if(!opt) var opt={}; va ...

随机推荐

  1. Node学习笔记(三):基于socket.io web版你画我猜(二)

    上一篇基础实现的功能是客户端canvas作图,导出dataURL从而实现图片信息推送,下面具体讲下服务端的配置及客户端的配置同步 首先先画一个流程图,讲下大概思路 <canvas id=&quo ...

  2. 发布在即!.NET Core 1.0 RC2已准备就绪!!

    先说点废话,从去年夏天就开始关注学习ASP.NET Core,那时候的版本还是beta5,断断续续不停踩坑.一路研究到11月份RC1发布. 在这个乐此不疲的过程里,学习了很多新的东西,对ASP.NET ...

  3. EC笔记:第三部分:16成对使用new和delete

    我们都知道,申请的资源,使用完毕后要释放.但是这个释放动作,一定要注意. 举个例子,很多人动态分配的资源,在使用之后,往往直接调用了delete,而不管申请资源的时候用的是new还是new[]. 如下 ...

  4. 静态代理和利用反射形成的动态代理(JDK动态代理)

    代理模式 代理模式的定义:为其他对象提供一种代理以控制对这个对象的访问.在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用. 静态代理 1.新建 ...

  5. Java面试题整理一(侧重多线程并发)

    1..是否可以在static环境中访问非static变量? 答:static变量在Java中是属于类的,它在所有的实例中的值是一样的.当类被Java虚拟机载入的时候,会对static变量进行初始化.如 ...

  6. JAVA的序列化和持久化的区别与联系

      持久化(Persistence) 即把数据(如内存中的对象)保存到可永久保存的存储设备中(如磁盘).持久化的主要应用是将内存中的对象存储在关系型的数据库中,当然也可以存储在磁盘文件中.XML数据文 ...

  7. HTML5 oninput实时监听输入框值变化的完美方案

    在网页开发中经常会碰到需要动态监听输入框值变化的情况,如果使用 onkeydown.onkeypress.onkeyup 这个几个键盘事件来监测的话,监听不了右键的复制.剪贴和粘贴这些操作,处理组合快 ...

  8. 什么是WebPack,为什么要使用它?

    1.什么是Webpack WebPack可以看做是模块打包机:它做的事情是,分析你的项目结构,找到JavaScript模块以及其它的一些浏览器不能直接运行的拓展语言(Scss,TypeScript等) ...

  9. 如何用代码读取Office Online Server2016的文档的备注信息

    前言 在一个项目上客户要求读取office online server 2016的对文档的备注信息,如下图: 以前思路老纠结在OOS这个在线上,总有以为这个信息存储在某个列表中,其实错了,这个备注信息 ...

  10. Atitit.工作流 与 规则引擎

    Atitit.工作流 与 规则引擎 1.1. 应用来说,通常分为三部分:界面.业务逻辑和存储1 1.2. 自定义操作系列1 1.3. 自定义按钮系列2 1.1. 应用来说,通常分为三部分:界面.业务逻 ...