省钱达人

XLPlainFlowLayout.m 7.0KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137
  1. //
  2. // XLPlainFlowLayout.m
  3. // XLPlainFlowLayout
  4. //
  5. // Created by hebe on 15/7/30.
  6. // Copyright (c) 2015年 ___ZhangXiaoLiang___. All rights reserved.
  7. //
  8. // 工作邮箱E-mail: k52471@126.com
  9. #import "XLPlainFlowLayout.h"
  10. @implementation XLPlainFlowLayout
  11. -(instancetype)init
  12. {
  13. self = [super init];
  14. if (self)
  15. {
  16. _naviHeight = 64.0;
  17. }
  18. return self;
  19. }
  20. - (NSArray *) layoutAttributesForElementsInRect:(CGRect)rect
  21. {
  22. //UICollectionViewLayoutAttributes:我称它为collectionView中的item(包括cell和header、footer这些)的《结构信息》
  23. //截取到父类所返回的数组(里面放的是当前屏幕所能展示的item的结构信息),并转化成不可变数组
  24. NSMutableArray *superArray = [[super layoutAttributesForElementsInRect:rect] mutableCopy];
  25. //创建存索引的数组,无符号(正整数),无序(不能通过下标取值),不可重复(重复的话会自动过滤)
  26. NSMutableIndexSet *noneHeaderSections = [NSMutableIndexSet indexSet];
  27. //遍历superArray,得到一个当前屏幕中所有的section数组
  28. for (UICollectionViewLayoutAttributes *attributes in superArray)
  29. {
  30. //如果当前的元素分类是一个cell,将cell所在的分区section加入数组,重复的话会自动过滤
  31. if (attributes.representedElementCategory == UICollectionElementCategoryCell)
  32. {
  33. [noneHeaderSections addIndex:attributes.indexPath.section];
  34. }
  35. }
  36. //遍历superArray,将当前屏幕中拥有的header的section从数组中移除,得到一个当前屏幕中没有header的section数组
  37. //正常情况下,随着手指往上移,header脱离屏幕会被系统回收而cell尚在,也会触发该方法
  38. for (UICollectionViewLayoutAttributes *attributes in superArray) {
  39. //如果当前的元素是一个header,将header所在的section从数组中移除
  40. if ([attributes.representedElementKind isEqualToString:UICollectionElementKindSectionHeader]) {
  41. [noneHeaderSections removeIndex:attributes.indexPath.section];
  42. }
  43. }
  44. //遍历当前屏幕中没有header的section数组
  45. [noneHeaderSections enumerateIndexesUsingBlock:^(NSUInteger idx, BOOL *stop) {
  46. //取到当前section中第一个item的indexPath
  47. NSIndexPath *indexPath = [NSIndexPath indexPathForItem:0 inSection:idx];
  48. //获取当前section在正常情况下已经离开屏幕的header结构信息
  49. UICollectionViewLayoutAttributes *attributes = [self layoutAttributesForSupplementaryViewOfKind:UICollectionElementKindSectionHeader atIndexPath:indexPath];
  50. //如果当前分区确实有因为离开屏幕而被系统回收的header
  51. if (attributes) {
  52. //将该header结构信息重新加入到superArray中去
  53. [superArray addObject:attributes];
  54. }
  55. }];
  56. //遍历superArray,改变header结构信息中的参数,使它可以在当前section还没完全离开屏幕的时候一直显示
  57. for (UICollectionViewLayoutAttributes *attributes in superArray) {
  58. //如果当前item是header
  59. if ([attributes.representedElementKind isEqualToString:UICollectionElementKindSectionHeader]) {
  60. //得到当前header所在分区的cell的数量
  61. NSInteger numberOfItemsInSection = [self.collectionView numberOfItemsInSection:attributes.indexPath.section];
  62. //得到第一个item的indexPath
  63. NSIndexPath *firstItemIndexPath = [NSIndexPath indexPathForItem:0 inSection:attributes.indexPath.section];
  64. //得到最后一个item的indexPath
  65. NSIndexPath *lastItemIndexPath = [NSIndexPath indexPathForItem:MAX(0, numberOfItemsInSection-1) inSection:attributes.indexPath.section];
  66. //得到第一个item和最后一个item的结构信息
  67. UICollectionViewLayoutAttributes *firstItemAttributes, *lastItemAttributes;
  68. if (numberOfItemsInSection>0) {
  69. //cell有值,则获取第一个cell和最后一个cell的结构信息
  70. firstItemAttributes = [self layoutAttributesForItemAtIndexPath:firstItemIndexPath];
  71. lastItemAttributes = [self layoutAttributesForItemAtIndexPath:lastItemIndexPath];
  72. }else{
  73. //cell没值,就新建一个UICollectionViewLayoutAttributes
  74. firstItemAttributes = [UICollectionViewLayoutAttributes new];
  75. //然后模拟出在当前分区中的唯一一个cell,cell在header的下面,高度为0,还与header隔着可能存在的sectionInset的top
  76. CGFloat y = CGRectGetMaxY(attributes.frame)+self.sectionInset.top;
  77. firstItemAttributes.frame = CGRectMake(0, y, 0, 0);
  78. //因为只有一个cell,所以最后一个cell等于第一个cell
  79. lastItemAttributes = firstItemAttributes;
  80. }
  81. //获取当前header的frame
  82. CGRect rect = attributes.frame;
  83. /**
  84. * 算法
  85. */
  86. //当前的滑动距离 + 因为导航栏产生的偏移量,默认为64(如果app需求不同,需自己设置)
  87. CGFloat offset = self.collectionView.contentOffset.y + _naviHeight;
  88. //第一个cell的y值 - 当前header的高度 - 可能存在的sectionInset的top
  89. CGFloat headerY = firstItemAttributes.frame.origin.y - rect.size.height - self.sectionInset.top;
  90. //哪个大取哪个,保证header悬停
  91. //针对当前header基本上都是offset更加大,针对下一个header则会是headerY大,各自处理
  92. CGFloat maxY = MAX(offset,headerY);
  93. //最后一个cell的y值 + 最后一个cell的高度 + 可能存在的sectionInset的bottom - 当前header的高度
  94. //当当前section的footer或者下一个section的header接触到当前header的底部,计算出的headerMissingY即为有效值
  95. CGFloat headerMissingY = CGRectGetMaxY(lastItemAttributes.frame) + self.sectionInset.bottom - rect.size.height;
  96. //给rect的y赋新值,因为在最后消失的临界点要跟谁消失,所以取小
  97. rect.origin.y = MIN(maxY,headerMissingY);
  98. //给header的结构信息的frame重新赋值
  99. attributes.frame = rect;
  100. //如果按照正常情况下,header离开屏幕被系统回收,而header的层次关系又与cell相等,如果不去理会,会出现cell在header上面的情况
  101. //通过打印可以知道cell的层次关系zIndex数值为0,我们可以将header的zIndex设置成1,如果不放心,也可以将它设置成非常大,这里随便填了个7
  102. attributes.zIndex = 7;
  103. }
  104. }
  105. //转换回不可变数组,并返回
  106. return [superArray copy];
  107. }
  108. //return YES;表示一旦滑动就实时调用上面这个layoutAttributesForElementsInRect:方法
  109. - (BOOL) shouldInvalidateLayoutForBoundsChange:(CGRect)newBound {
  110. return YES;
  111. }
  112. @end