CollectionView FlowLayout 瀑布流(可同时存在不同列数)

最终效果

不定列数瀑布流

介绍

瀑布流的自定义的一般流程请参考另外一篇文章,比较详细,我也不继续解释。ios - 用UICollectionView实现瀑布流详解

网上能搜索到的瀑布流一般都是相同列数的文章,而因项目需求,需要在同一个collectionView 能实现不同列数的瀑布流。譬如:一个collectionView可以同时存在 1、2、3.……列的瀑布流。

分析

为了方便控制,不同列数的cell用section来区分,而同一个section的cell的布局就可以跟固定列数的瀑布流一样。

然后下面来了解一下,哪些内容是在计算cell的位置必须要知道的,方便作为属性或者是delegate的方式公开出去。(备注:自定义flowlayout发现了一件奇怪的事情:如果设置了collectionView.contentInset, viewController竟然会不调用dataSource的cellForItemAtIndexPath方法,导致collectionView一片空白,至今没有找到原因。)

必要项

  • numberOfSections: section的个数
  • numberOfColumnInSection: 每个section的列数
  • size: 每个cell的大小size,如果全部cell宽度相等的话,可以考虑只是获取高度。

可选项

  • contentInset: 为了解决上面说的情况,自己添加了一个属性来替代collectionView.contentInset。(可以在创建对象时直接设置)
  • lineSpacing: 每一行的间距
  • itemSpace: 每一列的间距
  • sectionInset: 代替collectionView 的sectionInset。

除了contentInset之外,其他属性对于不同section不一定相同,所以使用协议的方式比较好。

实现

.h 文件

根据上面的分析,可以定义layout相关的协议,只有实现该协议的,才能得到瀑布流布局。

#import <UIKit/UIKit.h>

@protocol WatchFlowLayoutDelegate <NSObject>

@optional

// 行间距
- (CGFloat)minimumLineSpacingForSectionAtIndex:(NSInteger)section;  
// 列间距
- (CGFloat)minimumInteritemSpacingForSectionAtIndex:(NSInteger)section; 
// sectionInset
- (UIEdgeInsets)contentInsetOfSectionAtIndex:(NSInteger)section;        

@required

// section的数量
- (NSInteger)numberOfSection;  
// cell的大小
- (CGSize)sizeForItemAtIndexPath:(NSIndexPath *)indexPath;  
 // section的列数
- (NSInteger)numberOfColumnInSectionAtIndex:(NSInteger)section;

@end

@interface WatchFlowLayout : UICollectionViewLayout

@property (nonatomic, weak) id<WatchFlowLayoutDelegate> flowDelegate;

// 代替collectionView.contentInset
@property (nonatomic, assign) UIEdgeInsets contentInset;    

@end

.m 文件

为了保存信息,设置了下面的几个属性。

  • maxYOfColumns: 保存section每一列的最大的Y值,然后获取到最短的一列,将下一个cell放在该列中。
  • layoutAttributes: 保存所有cell的frame等信息,决定cell的布局。
  • contentHeight:保存collectionView的bouns的高度,决定collectionView竖向滑动的长度。
#import "WatchFlowLayout.h"

@interface WatchFlowLayout()

// 保存section每一列的最大的Y值,然后获取到最短的一列,将下一个cell放在该列中。
@property (nonatomic, strong) NSMutableArray *maxYOfColumns;    
// 保存所有cell的位置信息
@property (nonatomic, strong) NSMutableArray *layoutAttributes; 
// 保存collectionView的bouns的高度。
@property (nonatomic, assign) CGFloat contentHeight;            

@end

prepareLayout方法中计算出所有cell的位置。

@implementation WatchFlowLayout

- (void)prepareLayout {
[super prepareLayout];

// 没有代理,没法布局。
if (_flowDelegate == nil) {
    NSLog(@"需要代理");
    
    return;
}

// 重新赋值,清除上一次计算的数据。
_contentHeight = self.contentInset.top;
_layoutAttributes = [NSMutableArray new];

// 使用delegate获取section的数量
NSInteger numberOfSection = [_flowDelegate numberOfSection];

for (int section = 0; section < numberOfSection; section++) {
    NSMutableArray *sectionLayoutAttributes = [self computeLayoutAttributesInSection:section];
    [_layoutAttributes addObjectsFromArray:sectionLayoutAttributes];
}
}

- (UICollectionViewLayoutAttributes *)layoutAttributesForItemAtIndexPath:(NSIndexPath *)indexPath {
// 返回每个cell的位置信息等
NSInteger section = indexPath.section;
NSArray *sectionLayoutAttributes = _layoutAttributes[section];

return sectionLayoutAttributes[indexPath.row];
}

- (NSArray<UICollectionViewLayoutAttributes *> *)layoutAttributesForElementsInRect:(CGRect)rect {
return _layoutAttributes;
}

- (CGSize)collectionViewContentSize {
// 返回collectionView滑动的大小,因为横向没有滑动,X值不重要,也可以返回0
return CGSizeMake(0.0, _contentHeight + self.contentInset.bottom);
}

自定义的方法来计算每个section所有cell的frame

/**
 计算每个section的位置信息

 @param section section
 @return 与位置相关的信息。
 */
- (NSMutableArray *)computeLayoutAttributesInSection:(NSInteger)section {
// 获取section的列数和cell的个数
NSInteger column = [_flowDelegate numberOfColumnInSectionAtIndex:section];
NSInteger itemCount = [self.collectionView numberOfItemsInSection:section];

NSMutableArray *attributesArr = [NSMutableArray new];
CGFloat itemSpace = 0.0;
CGFloat lineSpace = 0.0;
UIEdgeInsets sectionInset;

// 获取间距等信息,下面计算位置时需要用到
// 因为是可选的实现方法,在直接使用时需要判断是否已经实现了。
if ([_flowDelegate respondsToSelector:@selector(contentInsetOfSectionAtIndex:)]) {
    sectionInset = [_flowDelegate contentInsetOfSectionAtIndex:section];
}

if ([_flowDelegate respondsToSelector:@selector(minimumLineSpacingForSectionAtIndex:)]) {
    itemSpace = [_flowDelegate minimumInteritemSpacingForSectionAtIndex:section];
}

if ([_flowDelegate respondsToSelector:@selector(minimumLineSpacingForSectionAtIndex:)]) {
    lineSpace = [_flowDelegate minimumLineSpacingForSectionAtIndex:section];
}

// 留出每个section的顶部与上一个section的距离
_contentHeight += sectionInset.top;

if (column == 1) {
    // 一列,cell会占满屏幕
    for (int index = 0; index < itemCount; index++) {
        NSIndexPath *indexPath = [NSIndexPath indexPathForItem:index inSection:section];
        UICollectionViewLayoutAttributes *attributes = [UICollectionViewLayoutAttributes layoutAttributesForCellWithIndexPath: indexPath];
        
        // 获取cell的大小
        CGSize size = [_flowDelegate sizeForItemAtIndexPath:indexPath];
        
        // 为了让collectionView.contentInset和sectionInset有效果,需要将width减去这个两个inset的左右的数值
        attributes.frame = CGRectMake(self.contentInset.left + sectionInset.left, _contentHeight, size.width - self.contentInset.left - self.contentInset.right - sectionInset.left - sectionInset.right, size.height);
        
        [attributesArr addObject:attributes];
        
        // 保存下一个cell的Y轴的数值
        _contentHeight += attributes.size.height + lineSpace;
    }
    
    // 减去最后一行底部添加的lineSpace
    _contentHeight += (sectionInset.bottom - lineSpace);
    
    return attributesArr;
}

// 不止一列时
// 保存每一个最后一个Cell的底部Y轴的数值
_maxYOfColumns = [NSMutableArray new];

for (int i = 0; i < column; i++) {
    self.maxYOfColumns[i] = @(0);
}

CGSize size;
CGFloat x = 0.0;
CGFloat y = 0.0;
NSInteger currentColumn = 0;
CGFloat width = 0.0;

for (int index = 0; index < itemCount; index++) {
    NSIndexPath *indexPath = [NSIndexPath indexPathForItem:index inSection:section];
    size = [_flowDelegate sizeForItemAtIndexPath:indexPath];
    
    if (index < column) {
        // 第一行直接添加到当前的列
        currentColumn = index;
        
    } else {// 其他行添加到最短的那一列
        // 这里使用!会得到期望的值
        NSNumber *minMaxY = [_maxYOfColumns valueForKeyPath:@"@min.self"];
        currentColumn = [_maxYOfColumns indexOfObject:minMaxY];
    }
    
    // 根据列数计算出每个cell的宽度
    width = (self.collectionView.bounds.size.width - itemSpace * (column - 1) - self.contentInset.left - self.contentInset.right - sectionInset.left - sectionInset.right) / column;
    
    // 根据将cell放在那一列,来计算出x坐标
    x = self.contentInset.left + sectionInset.left + currentColumn * (width + itemSpace);
    // 每个cell的y坐标
    y = lineSpace + [_maxYOfColumns[currentColumn] floatValue];
    
    // 记录每一列的最后一个cell的最大Y
    _maxYOfColumns[currentColumn] = @(y + size.height);

    UICollectionViewLayoutAttributes *attributes = [UICollectionViewLayoutAttributes layoutAttributesForCellWithIndexPath: indexPath];
    
    // 设置用于瀑布流效果的attributes的frame
    attributes.frame = CGRectMake(x, y + _contentHeight, width, size.height);
    
    [attributesArr addObject:attributes];
}

// 将所有列最大的Y值作为整个collectionView.cententSize的高度
CGFloat maxY = [[_maxYOfColumns valueForKeyPath:@"@max.self"] floatValue];
_contentHeight += maxY + sectionInset.bottom;

return attributesArr;
}

@end

使用方法 (示例在ViewController)

@interface ViewController () <UICollectionViewDelegate, UICollectionViewDataSource, WatchFlowLayoutDelegate>
@end

@implementation ViewController

- (void)viewDidLoad {
[super viewDidLoad];

// 创建layout对象,并设置delegate和 collectionView.contentInset
WatchFlowLayout *layout = [WatchFlowLayout new];
layout.flowDelegate = self;
layout.contentInset = UIEdgeInsetsMake(20, 10, 40, 10);

_collectionView = [[UICollectionView alloc] initWithFrame:CGRectMake(0, 100, self.view.bounds.size.width, self.view.bounds.size.height - 100)
                                     collectionViewLayout:layout];
}

实现layout的协议——WatchFlowLayoutDelegate

#pragma mark - WatchFlowLayoutDelegate

- (NSInteger)numberOfSection {
return [self numberOfSectionsInCollectionView:_collectionView];
}

- (NSInteger)numberOfColumnInSectionAtIndex:(NSInteger)section {
if (section == 1) {
    return 2;
}
else if (section == 3) {
    return 3;
}

return 1;
}

- (CGSize)sizeForItemAtIndexPath:(NSIndexPath *)indexPath {
if (indexPath.section == 1 || indexPath.section == 3) {
    // [40,160)随机数
    CGFloat height = 40 + arc4random() % (160 - 40 + 1);
    
    return CGSizeMake(0, height);
}

return CGSizeMake(_collectionView.bounds.size.width, 100);
}

- (CGFloat)minimumLineSpacingForSectionAtIndex:(NSInteger)section {
if (section == 1 || section == 3) {
    return 10;
}

return 4.0;
}

- (CGFloat)minimumInteritemSpacingForSectionAtIndex:(NSInteger)section {
if (section == 1 || section == 3) {
    return 10;
}

return 0.0;
}

- (UIEdgeInsets)contentInsetOfSectionAtIndex:(NSInteger)section {
if (section == 1 || section == 3) {
    return UIEdgeInsetsMake(0, 0, 10, 0);
}

return UIEdgeInsetsZero;
}

疑问

重点标注一下我发现的一个问题,如果简友们知道是什么原因,请评论一下或者私信我,谢谢!
问题:自定义flowlayout发现了一件奇怪的事情:如果设置了collectionView.contentInset, viewController竟然会不调用dataSource的cellForItemAtIndexPath方法,导致collectionView一片空白,至今没有找到原因。

扩展

简友可以试一下,如果多行多列没什么规律,就是每一行和每一列的宽或者高都不一致的时候,如何自定义瀑布流。
毕竟,这篇文章中每个section的宽度还是相等的。

demo 地址

demo百度云链接

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

推荐阅读更多精彩内容