UICollectionview实现类似UITableview的sectionheader悬浮代码

UICollectionview实现类似UITableview的sectionheader悬浮代码:

#import <UIKit/UIKit.h>


@interface CSNFloatingHeaderViewFlowLayout : UICollectionViewFlowLayout


/** zIndex for header

 default value is 1024

 */

@property (nonatomic, assign) NSInteger headerViewZIndex;


/** take over property values

 @return new instance

 */

- (instancetype)initWithFlowLayout:(UICollectionViewFlowLayout *)layout;


@end

-----------------------------------


#import "CSNFloatingHeaderViewFlowLayout.h"


@implementation CSNFloatingHeaderViewFlowLayout


- (void)commonInit;

{

    _headerViewZIndex = 1024;

}


- (id)init

{

    self = [super init];

    if (self) {

        [self commonInit];

    }

    

    return self;

}


- (id)initWithCoder:(NSCoder *)aDecoder

{

    self = [super initWithCoder:aDecoder];

    if (self) {

        [self commonInit];

    }

    

    return self;

}


- (instancetype)initWithFlowLayout:(UICollectionViewFlowLayout *)layout

{

    self = [self init];

    if (self) {

        self.minimumLineSpacing = layout.minimumLineSpacing;

        self.minimumInteritemSpacing = layout.minimumInteritemSpacing;

        self.itemSize = layout.itemSize;

        self.scrollDirection = layout.scrollDirection;

        self.headerReferenceSize = layout.headerReferenceSize;

        self.footerReferenceSize = layout.footerReferenceSize;

        self.sectionInset = layout.sectionInset;

    }

    

    return self;

}


- (NSArray *)layoutAttributesForElementsInRect:(CGRect)rect

{

    NSMutableArray *answer = [[super layoutAttributesForElementsInRect:rect] mutableCopy];

    UICollectionView * const collectionView = self.collectionView;

    CGPoint const contentOffset = collectionView.contentOffset;

    

    NSMutableIndexSet *missingSections = [NSMutableIndexSet indexSet];

    for (UICollectionViewLayoutAttributes *layoutAttributes in answer) {

        if (layoutAttributes.representedElementCategory == UICollectionElementCategoryCell) {

            [missingSections addIndex:layoutAttributes.indexPath.section];

        }

    }

    for (UICollectionViewLayoutAttributes *layoutAttributes in answer) {

        if ([layoutAttributes.representedElementKind isEqualToString:UICollectionElementKindSectionHeader]&&layoutAttributes.indexPath.section > 0) {

            [missingSections removeIndex:layoutAttributes.indexPath.section];

        }

    }

    

    [missingSections enumerateIndexesUsingBlock:^(NSUInteger idx, BOOL *stop) {

        

        NSIndexPath *indexPath = [NSIndexPath indexPathForItem:0 inSection:idx];

        

        UICollectionViewLayoutAttributes *layoutAttributes = [self layoutAttributesForSupplementaryViewOfKind:UICollectionElementKindSectionHeader atIndexPath:indexPath];

        

        [answer addObject:layoutAttributes];

        

    }];

    

    for (UICollectionViewLayoutAttributes *layoutAttributes in answer) {

        

        if ([layoutAttributes.representedElementKind isEqualToString:UICollectionElementKindSectionHeader]&&layoutAttributes.indexPath.section > 0) {

            

            NSInteger section = layoutAttributes.indexPath.section;

            NSInteger numberOfItemsInSection = [collectionView numberOfItemsInSection:section];

            

            NSIndexPath *firstObjectIndexPath = [NSIndexPath indexPathForItem:0 inSection:section];

            NSIndexPath *lastObjectIndexPath = [NSIndexPath indexPathForItem:MAX(0, (numberOfItemsInSection - 1)) inSection:section];

            

            BOOL cellsExist;

            UICollectionViewLayoutAttributes *firstObjectAttrs;

            UICollectionViewLayoutAttributes *lastObjectAttrs;

            

            if (numberOfItemsInSection > 0) { // use cell data if items exist

                cellsExist = YES;

                firstObjectAttrs = [self layoutAttributesForItemAtIndexPath:firstObjectIndexPath];

                lastObjectAttrs = [self layoutAttributesForItemAtIndexPath:lastObjectIndexPath];

            } else { // else use the header and footer

                cellsExist = NO;

                firstObjectAttrs = [self layoutAttributesForSupplementaryViewOfKind:UICollectionElementKindSectionHeader

                                                                        atIndexPath:firstObjectIndexPath];

                lastObjectAttrs = [self layoutAttributesForSupplementaryViewOfKind:UICollectionElementKindSectionFooter

                                                                       atIndexPath:lastObjectIndexPath];

                

            }

            

            CGFloat topHeaderHeight = (cellsExist) ? CGRectGetHeight(layoutAttributes.frame) : 0;

            CGFloat bottomHeaderHeight = CGRectGetHeight(layoutAttributes.frame);

            CGRect frameWithEdgeInsets = UIEdgeInsetsInsetRect(layoutAttributes.frame,

                                                               collectionView.contentInset);

            

            CGPoint origin = frameWithEdgeInsets.origin;

            UIEdgeInsets sectionInset = self.sectionInset;

            

            origin.y = MIN(

                           MAX(

                               contentOffset.y + collectionView.contentInset.top,

                               (CGRectGetMinY(firstObjectAttrs.frame) - topHeaderHeight - sectionInset.top)

                               ),

                           (CGRectGetMaxY(lastObjectAttrs.frame) - bottomHeaderHeight + sectionInset.bottom)

                           );


            

            layoutAttributes.zIndex = self.headerViewZIndex;

            layoutAttributes.frame = (CGRect){

                .origin = origin,

                .size = layoutAttributes.frame.size

            };

            

        }

        

    }

    

    return answer;

}


- (BOOL) shouldInvalidateLayoutForBoundsChange:(CGRect)newBound

{

    return YES;

}

@end


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值