美文网首页swift学习Swift&Objective-CiOS-swift
iOS - Swift UICollectionView横向分页

iOS - Swift UICollectionView横向分页

作者: LinXunFeng | 来源:发表于2017-01-02 13:33 被阅读2678次

    情况

    最近在做表情键盘时遇到一个问题,我用UICollectionView来布局表情,使用横向分页滚动,但在最后一页出现了如图所示的情况


    只显示一半

    情况分析图

    是的,现在的item分布就是这个鬼样子


    从上到下,从左到右

    现在想要做的,就是将现在这个鬼样子变成另外一种样子,如图


    从左到右,从上到下
    那怎么办?只好重新布局item了

    解决方案

    我是自定了一个Layout(LXFChatEmotionCollectionLayout),让UICollectionView在创建的时候使用了它

    在 LXFChatEmotionCollectionLayout.swift 中

    添加一个属性来保存所有item的attributes

    // 保存所有item的attributes
    fileprivate var attributesArr: [UICollectionViewLayoutAttributes] = []
    

    重新布局

    // MARK:- 重新布局
    override func prepare() {
        super.prepare()
        
        let itemWH: CGFloat = kScreenW / CGFloat(kEmotionCellNumberOfOneRow)
        
        // 设置itemSize
        itemSize = CGSize(width: itemWH, height: itemWH)
        minimumLineSpacing = 0
        minimumInteritemSpacing = 0
        scrollDirection = .horizontal
        
        // 设置collectionView属性
        collectionView?.isPagingEnabled = true
        collectionView?.showsHorizontalScrollIndicator = false
        collectionView?.showsVerticalScrollIndicator = true
        let insertMargin = (collectionView!.bounds.height - 3 * itemWH) * 0.5
        collectionView?.contentInset = UIEdgeInsetsMake(insertMargin, 0, insertMargin, 0)
        
        
        /// 重点在这里
        var page = 0
        let itemsCount = collectionView?.numberOfItems(inSection: 0) ?? 0
        for itemIndex in 0..<itemsCount {
            let indexPath = IndexPath(item: itemIndex, section: 0)
            let attributes = UICollectionViewLayoutAttributes(forCellWith: indexPath)
            
            page = itemIndex / (kEmotionCellNumberOfOneRow * kEmotionCellRow)
            // 通过一系列计算, 得到x, y值
            let x = itemSize.width * CGFloat(itemIndex % Int(kEmotionCellNumberOfOneRow)) + (CGFloat(page) * kScreenW)
            let y = itemSize.height * CGFloat((itemIndex - page * kEmotionCellRow * kEmotionCellNumberOfOneRow) / kEmotionCellNumberOfOneRow)
            
            attributes.frame = CGRect(x: x, y: y, width: itemSize.width, height: itemSize.height)
            // 把每一个新的属性保存起来
            attributesArr.append(attributes)
        }
    }
    

    返回所有当前可见的Attributes

    override func layoutAttributesForElements(in rect: CGRect) -> [UICollectionViewLayoutAttributes]? {
        var rectAttributes: [UICollectionViewLayoutAttributes] = []
        _ = attributesArr.map({
            if rect.contains($0.frame) {
                rectAttributes.append($0)
            }
        })
        return rectAttributes
    }
    

    大功告成

    完整代码

    import UIKit
    
    let kEmotionCellNumberOfOneRow = 8
    let kEmotionCellRow = 3
    
    class LXFChatEmotionCollectionLayout: UICollectionViewFlowLayout {
        // 保存所有item
        fileprivate var attributesArr: [UICollectionViewLayoutAttributes] = []
        
        // MARK:- 重新布局
        override func prepare() {
            super.prepare()
            
            let itemWH: CGFloat = kScreenW / CGFloat(kEmotionCellNumberOfOneRow)
            
            // 设置itemSize
            itemSize = CGSize(width: itemWH, height: itemWH)
            minimumLineSpacing = 0
            minimumInteritemSpacing = 0
            scrollDirection = .horizontal
            
            // 设置collectionView属性
            collectionView?.isPagingEnabled = true
            collectionView?.showsHorizontalScrollIndicator = false
            collectionView?.showsVerticalScrollIndicator = true
            let insertMargin = (collectionView!.bounds.height - 3 * itemWH) * 0.5
            collectionView?.contentInset = UIEdgeInsetsMake(insertMargin, 0, insertMargin, 0)
            
            var page = 0
            let itemsCount = collectionView?.numberOfItems(inSection: 0) ?? 0
            for itemIndex in 0..<itemsCount {
                let indexPath = IndexPath(item: itemIndex, section: 0)
                let attributes = UICollectionViewLayoutAttributes(forCellWith: indexPath)
                
                page = itemIndex / (kEmotionCellNumberOfOneRow * kEmotionCellRow)
                // 通过一系列计算, 得到x, y值
                let x = itemSize.width * CGFloat(itemIndex % Int(kEmotionCellNumberOfOneRow)) + (CGFloat(page) * kScreenW)
                let y = itemSize.height * CGFloat((itemIndex - page * kEmotionCellRow * kEmotionCellNumberOfOneRow) / kEmotionCellNumberOfOneRow)
                
                attributes.frame = CGRect(x: x, y: y, width: itemSize.width, height: itemSize.height)
                // 把每一个新的属性保存起来
                attributesArr.append(attributes)
            }
            
        }
        
        override func layoutAttributesForElements(in rect: CGRect) -> [UICollectionViewLayoutAttributes]? {
            var rectAttributes: [UICollectionViewLayoutAttributes] = []
            _ = attributesArr.map({
                if rect.contains($0.frame) {
                    rectAttributes.append($0)
                }
            })
            return rectAttributes
        }
        
    }
    

    附上相关项目:Swift 3.0 高仿微信

    相关文章

      网友评论

      • pikacode:你好我想问一下,要实现竖向滚动,横向分页,用这个方法可以实现吗?
        SuDream:大哥,求这个demo啊,我在这里卡了,也要做这个功能,
        LinXunFeng:@pikacode 可以呀,先将collectionView的竖向滚动showsHorizontalScrollIndicator设置为true,然后设置行数,以及contentSize就可以了,能不能滚动主要看contentSize的高度够不够
      • 不会游泳的飞鱼:有demo吗
        LinXunFeng:最近自己写项目时遇到的问题,比较乱,所以专门分离出来太麻烦了,等写好会分享的

      本文标题:iOS - Swift UICollectionView横向分页

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