轮播图片 高效图片轮播,两个imageView实现

该轮播框架的优势:

  • 文件少,代码简洁
  • 不依赖任何其他第三方库,耦合度低
  • 同时支持本地图片及网络图片
  • 可修改分页控件位置,显示或隐藏
  • 自定义分页控件的图片,就是这么个性
  • 自带图片缓存,一次加载,永久使用
  • 性能好,占用内存少,轮播流畅

实际使用

我们先看demo,代码如下

运行效果

轮播实现步骤

接下来,笔者将从各方面逐一分析。

层级结构

最底层是一个UIView,上面有一个UIScrollView以及UIPageControl,scrollView上有两个UIImageView,imageView宽高 = scrollview宽高 = view宽高

轮播原理

假设轮播控件的宽度为x高度为y,我们设置scrollview的contentSize.width为3x,并让scrollview的水平偏移量为x,既显示最中间内容

  1. scrollView.contentSize = CGSizeMake(3x, y);
  2. scrollView.contentOffset = CGPointMake(x, 0);

将imageView添加到scrollview内容视图的中间位置

接下来使用代理方法scrollViewDidScroll来监听scrollview的滚动,定义一个枚举变量来记录滚动的方向

  1. typedef enum{
  2. DirecNone,
  3. DirecLeft,
  4. DirecRight
  5. } Direction;@property (nonatomic, assign) Direction direction;
  6. - (void)scrollViewDidScroll:(UIScrollView *)scrollView {  self.direction = scrollView.contentOffset.x >x? DirecLeft : DirecRight;
  7. }

使用KVO来监听direction属性值的改变

  1. [self addObserver:self forKeyPath:@"direction" options:NSKeyValueObservingOptionNew context:nil];

判断滚动的方向,当偏移量大于x,表示左移,则将otherImageView加在右边,偏移量小于x,表示右移,则将otherImageView加在左边

  1. - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context {   //self.currIndex表示当前显示图片的索引,self.nextIndex表示将要显示图片的索引
  2. //_images为图片数组
  3. if(change[NSKeyValueChangeNewKey] == change[NSKeyValueChangeOldKey]) return;  if ([change[NSKeyValueChangeNewKey] intValue] == DirecRight) {    self.otherImageView.frame = CGRectMake(0, 0, self.width, self.height);    selfself.nextIndex = self.currIndex - 1;    if (self.nextIndex < 0) self.nextIndex = _images.count – 1;
  4. } else if ([change[NSKeyValueChangeNewKey] intValue] == DirecLeft){    self.otherImageView.frame = CGRectMake(CGRectGetMaxX(_currImageView.frame), 0, self.width, self.height);    self.nextIndex = (self.currIndex + 1) % _images.count;
  5. }  selfself.otherImageView.image = self.images[self.nextIndex];
  6. }

通过代理方法scrollViewDidEndDecelerating来监听滚动结束,结束后,会变成以下两种情况:

此时,scrollview的偏移量为0或者2x,我们通过代码再次将scrollview的偏移量设置为x,并将currImageView的图片修改为otherImageView的图片

  1. - (void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView {
  2. [self pauseScroll];
  3. }
  4. - (void)pauseScroll {  self.direction = DirecNone;//清空滚动方向
  5. //判断最终是滚到了右边还是左边
  6. int index = self.scrollView.contentOffset.x / x;  if (index == 1) return; //等于1表示最后没有滚动,返回不做任何操作
  7. selfself.currIndex = self.nextIndex;//当前图片索引改变
  8. selfself.pageControl.currentPage = self.currIndex;  self.currImageView.frame = CGRectMake(x, 0, x, y);  selfself.currImageView.image = self.otherImageView.image;  self.scrollView.contentOffset = CGPointMake(x, 0);
  9. }

那么我们看到的还是currImageView,只不过展示的是下一张图片,如图,又变成了最初的效果

自动滚动

轮播的功能实现了,接下来添加定时器让它自动滚动,相当简单

  1. - (void)startTimer {   //如果只有一张图片,则直接返回,不开启定时器
  2. if (_images.count <= 1) return;   //如果定时器已开启,先停止再重新开启
  3. if (self.timer) [self stopTimer];   self.timer = [NSTimer timerWithTimeInterval:self.time target:self selector:@selector(nextPage) userInfo:nil repeats:YES];
  4. [[NSRunLoop currentRunLoop] addTimer:self.timer forMode:NSRunLoopCommonModes];
  5. }
  6. - (void)nextPage {    //动画改变scrollview的偏移量就可以实现自动滚动
  7. [self.scrollView setContentOffset:CGPointMake(self.width * 2, 0) animated:YES];
  8. }

注意:setContentOffset:animated:方法执行完毕后不会调用scrollview的scrollViewDidEndDecelerating方法,但是会调用scrollViewDidEndScrollingAnimation方法,因此我们要在该方法中调用pauseScroll

  1. - (void)scrollViewDidEndScrollingAnimation:(UIScrollView *)scrollView {
  2. [self pauseScroll];
  3. }

拖拽时停止自动滚动

当我们手动拖拽图片时,需要停止自动滚动,此时我们只需要让定时器失效就行了,当停止拖拽时,重新启动定时器

  1. - (void)scrollViewWillBeginDragging:(UIScrollView *)scrollView {
  2. [self.timer invalidate];
  3. }
  4. - (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate{
  5. [self startTimer];
  6. }

加载图片

实际开发中,我们很少会轮播本地图片,大部分都是服务器获取的,也有可能既有本地图片,也有网络图片,那要如何来加载呢?

定义4个属性

  • NSArray imageArray:暴露在.h文件中,外界将要加载的图片或路径数组赋值给该属性
  • NSMutableArray images:用来存放图片的数组
  • NSMutableDictionary imageDic:用来缓存图片的字典,key为URL
  • NSMutableDictionary operationDic:用来保存下载操作的字典,key为URL

判断外界传入的是图片还是路径,如果是图片,直接加入图片数组中,如果是路径,先添加一个占位图片,然后根据路径去下载图片

  1. _images = [NSMutableArray array];for (int i = 0; i < imageArray.count; i++) {    if ([imageArray[i] isKindOfClass:[UIImage class]]) {
  2. [_images addObject:imageArray[i]];//如果是图片,直接添加到images中
  3. } else if ([imageArray[i] isKindOfClass:[NSString class]]){
  4. [_images addObject:[UIImage imageNamed:@"placeholder"]];//如果是路径,添加一个占位图片到images中
  5. [self downloadImages:i];  //下载网络图片
  6. }
  7. }

下载图片,先从缓存中取,如果有,则替换之前的占位图片,如果没有,去沙盒中取,如果有,替换占位图片,并添加到缓存中,如果没有,开启异步线程下载

  1. - (void)downloadImages:(int)index {  NSString *key = _imageArray[index];  //从字典缓存中取图片
  2. UIImage *image = [self.imageDic objectForKey:key];  if (image) {
  3. _images[index] = image;//如果图片存在,则直接替换之前的占位图片
  4. }else{    //字典中没有从沙盒中取图片
  5. NSString *cache = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject];    NSString *path = [cache stringByAppendingPathComponent:[key lastPathComponent]];    NSData *data = [NSData dataWithContentsOfFile:path];    if (data) {             //沙盒中有,替换占位图片,并加入字典缓存中
  6. image = [UIImage imageWithData:data];
  7. _images[index] = image;
  8. [self.imageDic setObject:image forKey:key];
  9. }else{       //字典沙盒都没有,下载图片
  10. NSBlockOperation *download = [self.operationDic objectForKey:key];//查看下载操作是否存在
  11. if (!download) {//不存在
  12. //创建一个队列,默认为并发队列
  13. NSOperationQueue *queue = [[NSOperationQueue alloc] init];        //创建一个下载操作
  14. download = [NSBlockOperation blockOperationWithBlock:^{          NSURL *url = [NSURL URLWithString:key];          NSData *data = [NSData dataWithContentsOfURL:url];           if (data) {                        //下载完成后,替换占位图片,存入字典并写入沙盒,将下载操作从字典中移除掉
  15. UIImage *image = [UIImage imageWithData:data];
  16. [self.imageDic setObject:image forKey:key];            self.images[index] = image;                        //如果只有一张图片,需要在主线程主动去修改currImageView的值
  17. if (_images.count == 1) [_currImageView performSelectorOnMainThread:@selector(setImage:) withObject:image waitUntilDone:NO];
  18. [data writeToFile:path atomically:YES];
  19. [self.operationDic removeObjectForKey:key];
  20. }
  21. }];
  22. [queue addOperation:download];
  23. [self.operationDic setObject:download forKey:key];//将下载操作加入字典
  24. }
  25. }
  26. }
  27. }

监听图片点击

当图片被点击的时候,我们往往需要执行某些操作,因此需要监听图片的点击,思路如下

1.定义一个block属性暴露给外界void(^imageClickBlock)(NSInteger index)

(不会block的可以用代理,或者看这里)

2.设置currImageView的userInteractionEnabled为YES

3.给currImageView添加一个点击的手势

4.在手势方法里调用block,并传入图片索引

结束语

上面是笔者的主要思路以及部分代码,需要源码的请前往笔者的github下载:https://github.com/codingZero/XRCarouselView。

写了这么多年的轮播图片的功能,这次居然总结的这么到位,果断收藏了。

时间: 2024-09-27 04:31:43

轮播图片 高效图片轮播,两个imageView实现的相关文章

element-ui中轮播图自适应图片高度

哈哈,久违了各位.我又回来了,最近在做毕设,所以难免会遇到很多问题,需要解决很多问题,在万能的博友帮助下,终于解决了Element-ui中轮播图的图片高度问题,话不多说上代码. 那个axios的使用不重要,大致思路就是页面渲染前拿到当前窗口的宽度*图片比例给轮播图一个初始的高度,当窗后大小发生变化时,监听事件里将轮播图高度重新赋值即可,再次感谢两位博友的帮助 <template> <div v-if="imgurl"> <el-carousel :heig

iOS:实现图片的无限轮播(二)---之使用第三方库SDCycleScrollView

下载链接:github不断更新地址:https://github.com/gsdios/SDCycleScrollView 使用原理:采用UICollectionView的重用机制和循环滚动的方式实现图片的无限轮播,播放非常顺畅,解决了UISCrollView使用时从最后一张跳到第一张时的生硬状态. 主要类截图: SDCollectionViewCell:用来重用的item,即显示图片的视图: SDCycleScrollView: 对外提供的一个创建轮播器的接口类,使用者就是直接使用这个类来实现

换主页轮播的主题图片(4、删除)---轻开电子商务系统(企业入门级B2C网站)

接(  换主页轮播的主题图片1 ) 文件:site/links/img0.html中的表单(第11行最后一个td) <td><if x="@{sys:canDo}"><a href="javascript:doDel('@{links:WE_ID}');" x=true>[删除]</a></if></td> 及16到23行的删除提示js函数(参数为reUrl和图片we_id) <scrip

js 实现图片间隔循环轮播以及没有间隔的循环轮播

链接地址:http://blog.sina.com.cn/s/blog_75cf5f32010199dn.html 最近做了个图片循环轮播的功能.就是几张图片不断的循环滚动显示. 感觉这个方法不错所以把实现过程记录下来: 图片间隔循环轮播: 1.html里面把要进行轮播的图片用一个容器包起来,用js来控制这个容器滚动,当然最外面的容器要设置overflow:hidden;. 2.css里面控制这个容器的位置,实现滚动就需要用到定位. 3.js里面利用jquery的animate动画函数实现滚动.

利用jQuery实现图片无限循环轮播(不借助于轮播插件)

原来我主要是用Bootstrap来实现轮播图的功能,而这次是用javaScript和jQuery来实现图片无限循环轮播! 用到的技术有:html.css.JavaScript(少).jQuery(主要) 效果展示: html代码: <body> <div id="container"><!-- left:-600px 表示:页面加载出现的第一张图片是1.jp --> <div id="list" style="le

Jquery+css实现图片无缝滚动轮播

Today,在XX学院的教学视频中,偶尔看到了Jquery+css实现图片无缝滚动轮播视频教程,虽然以前已写过类似的,但是我感觉他学的比较精简.为了方便以后做项目时直接拷贝,特地写出来,顺便和大家分享一下 最终实现界面如下: 页面加载时,自动轮播,轮播鼠标悬停在整个banner容器的时候,两边会显示向左,向右按钮,鼠标悬停在中下方索引圆圈的上面,自动跳转到相应的图片. banner容器里面包含了图片列表img,索引圆圈 num,还有按钮两个btn <div class="banner&qu

jQuery图片插件自动轮播原理解析

经常看到项目要用到图片轮播效果,一般的操作流程都是先到网上找一个好看的JQuery图片轮播插件,然后看下demo,再配下参数.好了,关机下班回家 其余的就交给插件吧. 是不是感觉有了jQuery,世界变得那么美好呢. 本人最近用的一个插件是 jquery.carousel.js,官方网站是:http://richardscarrott.co.uk/posts/view/jquery-carousel-plugin 下面,我们来讨论一下图片轮播原理. 首先来个简单的demo 效果图如下: 这是个导

js无缝轮播 和淡入淡出轮播

<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Document</title> <style> *{margin:0;padding:0;} ul,li{ list-style: none; } #banner{ width: 800px; height: 400px; margin: 30

php如何防止图片盗用/盗链的两种方法

如今的互联网,采集网站非常多,很多网站都喜欢盗链/盗用别人网站的图片,这样不仅侵犯网权,还导致被盗链的网站消耗大量的流量,给服务器造成比较大的压力,本文章向大家介绍php如何防止图片盗用/盗链的两种方法,需要的朋友可以参考一下. 图片防盗链有什么用? 防止其它网站盗用你的图片,浪费你宝贵的流量.本文章向大家介绍php防止图片盗用/盗链的两种方法 Apache图片重定向方法 设置images目录不充许http访问 Apache服务器下防止图片盗链的办法 如果你的网站以图片为主,哪天发现月底没到流量