collectionView实现无限循环滚动卡片

github源码地址

效果展示
show.gif

前言

去年因为项目中有个切换学校的功能,要求以卡片浮动效果展示,并且能够无限循环滚动。

之前找了个demo它是通过自定义view动画实现的,卡片数量多的时候会比较卡顿,所以研究了一下,自己造了一个,现在把实现的思路分享一下~

好,开始正题~~~
技术调研

在实现之前,首先我想到的是如何实现无限循环的问题,以及多数量时如何复用的问题。关于view的无限循环滚动和复用问题,最经典的就是轮播图了,现在比较常用的两种方法就是UIScrollView或者UICollectionView实现的。

然后我决定选用UICollectionView,因为我们只需要给它提供数据源驱动,它自己就可以复用,实现应该比较简单。并且UICollectionViewlayout的灵活性非常强,完全可以自定义出很多酷炫的效果。

横向滚动

设置滚动方向为水平,禁用分页属性(默认就是false),这样就可以横向流畅滑动了

let layout = UICollectionViewFlowLayout()
layout.scrollDirection = .horizontal
collectionView = UICollectionView(frame: frame, collectionViewLayout: layout)
collectionView.isPagingEnabled = false // default NO

无限循环

定义数组imageArr来存储图片作为数据源,在每次滚动结束后,都重新定位到第一张。

func scrollViewDidEndDecelerating(_ scrollView: UIScrollView) {
   collectionView.scrollToItem(at: IndexPath(item: 0, section: 0), at:.centeredHorizontally, animated: false)
}

但是每次重定位的时候都会出现明显的突兀效果,所以我们可以把这个数组中的图片复制100份、200份、1000份。。。这样就有足够多的数据来供用户滑动期间来展示了。

但是直接存储图片的话,这个数组肯定会占用很大的内存,所以我们开辟一个新的数组indexArr,来存储imageArr中图片的下标,因为只是存储的是int,所以占用的内存是非常小的。

// 初始化数据源
imageArr = ["num_1", "num_2", "num_3", "num_4", "num_5"]
for _ in 0 ..< 100 {
    for j in 0 ..< imageArr.count {
        indexArr.append(j)
    }
}

比如imageArr中是["pic1", "pic2", "pic3"],那么indexArr中就是[0,1,2,0,1,2,0,1,2...],这样就可以把数据源设置为indexArr。

func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
    return indexArr.count
}

代码初始时我们把collectionView定位到中间那组(第50组,假设共设置了100组),每次滚动结束后也再次定位到中间组的第N张(N:上次滑动结束时的那张)。

// 定位到 第50组(中间那组)
override func viewDidLoad() {
    super.viewDidLoad()
    collectionView.scrollToItem(at: IndexPath(item: 50 * imageArr.count, section: 0) , at: UICollectionViewScrollPosition.centeredHorizontally, animated: false)
}

每次动画停止时,也重新定位到 第50组(中间那组) 模型

func scrollViewDidEndDecelerating(_ scrollView: UIScrollView) {
    collectionView.scrollToItem(at: IndexPath(item: 50 * imageArr.count, section: 0) , at: UICollectionViewScrollPosition.centeredHorizontally, animated: false)
}

每张卡片(cell)的数据填充

func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
    
    let cell = self.collectionView.dequeueReusableCell(withReuseIdentifier: NSStringFromClass(CyclicCardCell.self), for: indexPath) as! CyclicCardCell
    let index = indexArr[indexPath.row]
    cell.index = index
    cell.cardImgView.image = UIImage(named: imageArr[index])
    cell.cardNameLabel.text = "奔跑吧,小蜗牛~"
    return cell
}

滚动动画的平滑过渡

在每次重定位的时候,虽然设置的是回到中间组的对应下标的那个cell,并且animated也是设置的false,但是依然可以看出动画有些不连贯、突兀,显得不流畅自然。

这就需要我们自定义UICollectionViewFlowLayout,来重写targetContentOffset方法,通过遍历目标区域中的cell,来计算出距离中心点最近的cell,把它调整到中间,实现平缓流畅的滑动结束的效果。

override func targetContentOffset(forProposedContentOffset proposedContentOffset: CGPoint, withScrollingVelocity velocity: CGPoint) -> CGPoint {
        
        let targetRect = CGRect(x: proposedContentOffset.x, y: 0.0, width:  self.collectionView!.bounds.size.width, height: self.collectionView!.bounds.size.height)
        // 目标区域中包含的cell
        let attriArray = super.layoutAttributesForElements(in: targetRect)! as [UICollectionViewLayoutAttributes]
        // collectionView落在屏幕中点的x坐标
        let horizontalCenterX = proposedContentOffset.x + (self.collectionView!.bounds.width / 2.0)
        var offsetAdjustment = CGFloat(MAXFLOAT)
        for layoutAttributes in attriArray {
            let itemHorizontalCenterX = layoutAttributes.center.x
            // 找出离中心点最近的
            if(abs(itemHorizontalCenterX-horizontalCenterX) < abs(offsetAdjustment)){
                offsetAdjustment = itemHorizontalCenterX-horizontalCenterX
            }
        }
        
        //返回collectionView最终停留的位置
        return CGPoint(x: proposedContentOffset.x + offsetAdjustment, y: proposedContentOffset.y)
    }

卡片的缩放动画

在自定义的UICollectionViewFlowLayout中重写layoutAttributesForElements方法,通过该方法,可以获取到可视范围内的cell。

然后在cell的滑动过程中,通过cell偏移的距离来进行尺寸的缩放系数的设置,处于最中心的系数为1,则为原本的大小,随着中心距离的偏移,系数会逐渐变小为0.98,0.95,0.8...

因此卡片也会随之变小,从而达到在滚动过程中,滚动至中间的卡片最大,旁边的变小的效果。

override func layoutAttributesForElements(in rect: CGRect) -> [UICollectionViewLayoutAttributes]? {
        
        let array = super.layoutAttributesForElements(in: rect)
        var visibleRect = CGRect()
        visibleRect.origin = self.collectionView!.contentOffset
        visibleRect.size = self.collectionView!.bounds.size
        
        for attributes in array!{
            let distance = visibleRect.midX - attributes.center.x
            let normalizedDistance = abs(distance/ActiveDistance)
            let zoom = 1 - ScaleFactor * normalizedDistance
            attributes.transform3D = CATransform3DMakeScale(1.0, zoom, 1.0)
            attributes.zIndex = 1
//            let alpha = 1 - normalizedDistance
//            attributes.alpha = alpha
        }
        return array
    }

OK,至此主要的核心实现就完成了,可能有些地方思路表述的可能不是太清楚,大家可以去github上下载源码看下,有不对的地方,欢迎指正~

9月28号更新:

  • 因为这个文章是第一次写的,所以之前表述的不大好,今天重新修改了一下,希望能对大家有所帮助😆

  • 刚写了一个类似的重叠卡片滚动的动画,有兴趣的可以看下:

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

推荐阅读更多精彩内容