服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - IOS - IOS简单实现瀑布流UICollectionView

IOS简单实现瀑布流UICollectionView

2021-01-05 12:23tanhui_ui IOS

这篇文章主要为大家介绍了IOS简单实现瀑布流UICollectionView的相关资料,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

uicollectionview 比tableview 灵活,功能也强大很多。系统实现了流式布局,但用处还有很多限制。

要想实现更灵活的布局,就咬重写uicollectionviewlayout。
先看下实现效果:

IOS简单实现瀑布流UICollectionView

废话不多说,直接上代码:

先看waterfallcollectionlayout.m

?
1
2
3
4
5
6
7
8
9
10
#import "waterfallcollectionlayout.h"
#define colmargin 5
#define colcount 4
#define rolmargin 5
@interface waterfallcollectionlayout ()
//数组存放每列的总高度
@property(nonatomic,strong)nsmutablearray* colsheight;
//单元格宽度
@property(nonatomic,assign)cgfloat colwidth;
@end

该类要重写以下方法:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
//完成布局前的初始工作
-(void)preparelayout;
 
//collectionview的内容尺寸
-(cgsize)collectionviewcontentsize;
 
//为每个item设置属性
-(uicollectionviewlayoutattributes *)layoutattributesforitematindexpath:(nsindexpath *)indexpath;
 
//获取制定范围的所有item的属性
-(nsarray<uicollectionviewlayoutattributes *> *)layoutattributesforelementsinrect:(cgrect)rect;
 
-(bool)shouldinvalidatelayoutforboundschange:(cgrect)newbounds;

每次调用会清空colsheight数组里的信息:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
//完成布局前的初始工作
-(void)preparelayout{
  [super preparelayout];
  self.colwidth =( self.collectionview.frame.size.width - (colcount+1)*colmargin )/colcount;
  //让它重新加载
  self.colsheight = nil;
}
通过遍历colheight数组里的所有列来获得最长的那一列,返回contentsize
//collectionview的内容尺寸
-(cgsize)collectionviewcontentsize{
  nsnumber * longest = self.colsheight[0];
  for (nsinteger i =0;i<self.colsheight.count;i++) {
    nsnumber* rolheight = self.colsheight[i];
    if(longest.floatvalue<rolheight.floatvalue){
      longest = rolheight;
    }
  }
  return cgsizemake(self.collectionview.frame.size.width, longest.floatvalue);
}

每个cell要出来时这个方法会被调用,在此方法中设置该cell的frame。

注意heightblock是外部控制器传进来的block用以计算每个cell的高度,现在我只是设置了随机数。如果没有传block进来我这里直接让他崩溃了。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
//为每个item设置属性
-(uicollectionviewlayoutattributes *)layoutattributesforitematindexpath:(nsindexpath *)indexpath{
  uicollectionviewlayoutattributes* attr = [uicollectionviewlayoutattributes layoutattributesforcellwithindexpath:indexpath];
  nsnumber * shortest = self.colsheight[0];
  nsinteger shortcol = 0;
  for (nsinteger i =0;i<self.colsheight.count;i++) {
    nsnumber* rolheight = self.colsheight[i];
    if(shortest.floatvalue>rolheight.floatvalue){
      shortest = rolheight;
      shortcol=i;
    }
  }
  cgfloat x = (shortcol+1)*colmargin+ shortcol * self.colwidth;
  cgfloat y = shortest.floatvalue+colmargin;
  
  //获取cell高度
  cgfloat height=0;
  nsassert(self.heightblock!=nil, @"未实现计算高度的block ");
  if(self.heightblock){
    height = self.heightblock(indexpath);
  }
  attr.frame= cgrectmake(x, y, self.colwidth, height);
  self.colsheight[shortcol]=@(shortest.floatvalue+colmargin+height);
  
  return attr;
}
?
1
2
3
4
5
6
7
8
9
10
//获取所有item的属性
-(nsarray<uicollectionviewlayoutattributes *> *)layoutattributesforelementsinrect:(cgrect)rect{
  nsmutablearray* array = [nsmutablearray array];
  nsinteger items = [self.collectionview numberofitemsinsection:0];
  for (int i = 0; i<items;i++) {
    uicollectionviewlayoutattributes* attr = [self layoutattributesforitematindexpath:[nsindexpath indexpathforitem:i insection:0]];
    [array addobject:attr];
  }
  return array;
}

实现下列方法会在出现新的cell时重新布局并调用preparelayout方法

?
1
2
3
-(bool)shouldinvalidatelayoutforboundschange:(cgrect)newbounds{
  return yes;
}

每列高度的存放,初始高度可以改,我这里是0

?
1
2
3
4
5
6
7
8
9
10
11
-(nsmutablearray *)colsheight{
  if(!_colsheight){
    nsmutablearray * array = [nsmutablearray array];
    for(int i =0;i<colcount;i++){
      //这里可以设置初始高度
      [array addobject:@(0)];
    }
    _colsheight = [array mutablecopy];
  }
  return _colsheight;
}

再来看看控制器里就是这么简单

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
#pragma mark getter-setter
-(uicollectionview *)collectionview{
  if(!_collectionview){
    _collectionview = [[uicollectionview alloc]initwithframe:self.view.frame collectionviewlayout:self.layout];
    _collectionview.backgroundcolor = [uicolor whitecolor];
    _collectionview.delegate=self;
    _collectionview.datasource=self;
    [_collectionview registerclass:[collectionviewcell class] forcellwithreuseidentifier:identifer];
  }
  return _collectionview;
}
-(uicollectionviewlayout *)layout{
  if(!_layout){
    _layout = [[waterfallcollectionlayout alloc]initwithitemsheightblock:^cgfloat(nsindexpath *index) {
      return [self.heightarr[index.item] floatvalue];
    }];
    
  }
  return _layout;
}
-(nsarray *)heightarr{
  if(!_heightarr){
    //随机生成高度
    nsmutablearray *arr = [nsmutablearray array];
    for (int i = 0; i<100; i++) {
      [arr addobject:@(arc4random()%50+80)];
    }
    _heightarr = [arr copy];
  }
  return _heightarr;
}

关于瀑布流的文章特别多,本文就是为大家分享了ios简单实现瀑布流的方法,希望对大家的学习有所帮助。

延伸 · 阅读

精彩推荐
  • IOSIOS 屏幕适配方案实现缩放window的示例代码

    IOS 屏幕适配方案实现缩放window的示例代码

    这篇文章主要介绍了IOS 屏幕适配方案实现缩放window的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要...

    xiari5772021-06-01
  • IOSiOS 雷达效果实例详解

    iOS 雷达效果实例详解

    这篇文章主要介绍了iOS 雷达效果实例详解的相关资料,需要的朋友可以参考下...

    SimpleWorld11022021-01-28
  • IOSIOS开发之字典转字符串的实例详解

    IOS开发之字典转字符串的实例详解

    这篇文章主要介绍了IOS开发之字典转字符串的实例详解的相关资料,希望通过本文能帮助到大家,让大家掌握这样的方法,需要的朋友可以参考下...

    苦练内功5832021-04-01
  • IOSiOS通过逆向理解Block的内存模型

    iOS通过逆向理解Block的内存模型

    自从对 iOS 的逆向初窥门径后,我也经常通过它来分析一些比较大的应用,参考一下这些应用中某些功能的实现。这个探索的过程乐趣多多,不仅能满足自...

    Swiftyper12832021-03-03
  • IOSiOS布局渲染之UIView方法的调用时机详解

    iOS布局渲染之UIView方法的调用时机详解

    在你刚开始开发 iOS 应用时,最难避免或者是调试的就是和布局相关的问题,下面这篇文章主要给大家介绍了关于iOS布局渲染之UIView方法调用时机的相关资料...

    windtersharp7642021-05-04
  • IOSiOS中tableview 两级cell的展开与收回的示例代码

    iOS中tableview 两级cell的展开与收回的示例代码

    本篇文章主要介绍了iOS中tableview 两级cell的展开与收回的示例代码,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧...

    J_Kang3862021-04-22
  • IOS关于iOS自适应cell行高的那些事儿

    关于iOS自适应cell行高的那些事儿

    这篇文章主要给大家介绍了关于iOS自适应cell行高的那些事儿,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的...

    daisy6092021-05-17
  • IOS解析iOS开发中的FirstResponder第一响应对象

    解析iOS开发中的FirstResponder第一响应对象

    这篇文章主要介绍了解析iOS开发中的FirstResponder第一响应对象,包括View的FirstResponder的释放问题,需要的朋友可以参考下...

    一片枫叶4662020-12-25