viewForSupplementaryElementOfKind未在自定义UICollectionViewLayout上调用

时间:2016-03-09 00:47:33

标签: ios objective-c uicollectionview uicollectionviewlayout uicollectionreusableview

我创建了一个自定义的UICollectionViewLayout,并且我将它放在故事板中的Collection View Controller中。在显示项目/单元格时,我没有遇到太多问题。我的问题是没有调用viewForSupplementaryElementOfKind。我似乎无法确定为什么会这样。我试图回到默认布局,它确实调用它,但我需要我的自定义UICollectionViewLayout。我已经离开了NSLogs来检查并且没有调用viewForSupplementaryElementOfKind。

这是我的MyCollectionViewController.m

- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view.
viewArray = [NSMutableArray array];

//setup the delegate for vertical flow layout
[(VerticalFlowLayout*)self.collectionViewLayout setDelegate:self];

//register the headers
[self.collectionView registerNib:[ButtonHeaderCollectionReusableView nib] forSupplementaryViewOfKind:UICollectionElementKindSectionHeader withReuseIdentifier:[ButtonHeaderCollectionReusableView reusableID]];

//set the insets
[self.collectionView setContentInset:UIEdgeInsetsMake(23, 5, 10, 5)]; }

#pragma mark - CollectionView DataSource

-(NSInteger)collectionView:(UICollectionView *)collectionView numberOfItemsInSection:(NSInteger)section {
return [self.cardsArray count];
}

-(NSInteger)numberOfSectionsInCollectionView:(UICollectionView *)collectionView {
NSLog(@"Number of Sections");

return 1;
}

-(UICollectionViewCell*)collectionView:(UICollectionView *)collectionView cellForItemAtIndexPath:(NSIndexPath *)indexPath {

NSLog(@"Making Card");

CardCell *card = [collectionView dequeueReusableCellWithReuseIdentifier:@"CardCell" forIndexPath:indexPath];

//    UILabel *cardLabel = [card viewWithTag:101];
//    [cardLabel setText:[textArray objectAtIndex:[indexPath row]]];

return card;
}

-(UICollectionReusableView*)collectionView:(UICollectionView *)collectionView viewForSupplementaryElementOfKind:(NSString *)kind atIndexPath:(NSIndexPath *)indexPath {

NSLog(@"Making Header");

UICollectionReusableView *reusableView = nil; /*[[UICollectionReusableView alloc] initWithFrame:CGRectMake(0, 0, 100, 75)];*/
//    [reusableView setBackgroundColor:[UIColor blackColor]];

if (kind == UICollectionElementKindSectionHeader) {
    ButtonHeaderCollectionReusableView *headerview = [collectionView dequeueReusableSupplementaryViewOfKind:UICollectionElementKindSectionHeader withReuseIdentifier:[ButtonHeaderCollectionReusableView reusableID] forIndexPath:indexPath];

    return headerview;
}

//    if (kind == UICollectionElementKindSectionFooter) {
//        UICollectionReusableView *footerview = [collectionView dequeueReusableSupplementaryViewOfKind:UICollectionElementKindSectionFooter withReuseIdentifier:@"FooterView" forIndexPath:indexPath];
//    }

return reusableView;
}

-(CGSize)collectionView:collectionView layout:(nonnull UICollectionViewLayout *)collectionViewLayout referenceSizeForHeaderInSection:(NSInteger)section {
return CGSizeMake(180.0f, 72.0f);
}

现在我的MyFlowLayout.m

-(id)initWithCoder:(NSCoder *)aDecoder {
NSLog(@"Vertical Flow Layout init with coder");
if(self = [super initWithCoder:aDecoder]) {
    contentHeight = 0;
    cachedAttributes = [NSMutableArray array];
}

return self;
}

-(void)prepareLayout {
NSLog(@"Preparing Layout");
//    [super prepareLayout];

contentWidth = CGRectGetWidth(self.collectionView.bounds) - (self.collectionView.contentInset.left + self.collectionView.contentInset.right);

if([cachedAttributes count] == 0) {
    //compute for column width
    CGFloat columnWidth = contentWidth / NUMBER_OF_COLUMNS;
    NSMutableArray *xOffsets = [NSMutableArray array];
    for(int i = 0; i < NUMBER_OF_COLUMNS; i++) {
        [xOffsets addObject:[NSNumber numberWithFloat:(i * columnWidth)]];
    }

    //compute for height
    NSMutableArray *yOffsets = [NSMutableArray array];
    for(int i = 0; i < NUMBER_OF_COLUMNS; i++) {
        [yOffsets addObject:[NSNumber numberWithFloat:75]];
    }

    int column = 0;
    //loop through all the sections and items in the collectionview
    for(int i = 0; i < self.collectionView.numberOfSections; i++) {
        for(int j = 0; j < [self.collectionView numberOfItemsInSection:i]; j++) {
            NSIndexPath *indexPath = [NSIndexPath indexPathForItem:j inSection:i];

            //time to do some frame calculation
            ///let's start with the width
            CGFloat width = columnWidth - CELL_PADDING * 2;
            ///then the height
            CGFloat viewHeight = [self.delegate getViewHeightWithCollectionView:self.collectionView indexPath:indexPath withWidth:width];
            CGFloat height = CELL_PADDING + viewHeight + /*textHeight +*/ CELL_PADDING;

            CGFloat xOffset = [[xOffsets objectAtIndex:column] floatValue];
            CGFloat yOffset = [[yOffsets objectAtIndex:column] floatValue];

            CGRect frame = CGRectMake(xOffset, yOffset, columnWidth, height);
            CGRect insetFrame = CGRectInset(frame, CELL_PADDING, CELL_PADDING);

            //now that computation is done we shall make the attributes
            UICollectionViewLayoutAttributes *attributes = [UICollectionViewLayoutAttributes layoutAttributesForCellWithIndexPath:indexPath];
            [attributes setFrame:insetFrame];
            [cachedAttributes addObject:attributes];

            //time to increment the height and the column
            contentHeight = MAX(contentHeight, CGRectGetMaxY(frame));
            NSNumber *yOffSetColumn = [yOffsets objectAtIndex:column];
            yOffSetColumn = [NSNumber numberWithFloat:([yOffSetColumn floatValue] + height)];
            [yOffsets replaceObjectAtIndex:column withObject:yOffSetColumn];

            NSLog(@"Content Height: %f yOffSetColumn: %@ == %@", contentHeight, yOffSetColumn, [yOffsets objectAtIndex:column]);

            column = column >= (NUMBER_OF_COLUMNS - 1) ? 0 : ++column;
        }
    }
}
}

-(CGSize)collectionViewContentSize {
//    NSLog(@"Collection View Content Size");
return CGSizeMake(contentWidth, contentHeight + 75);
}

//called after preparelayout to determine which items are visible in the given rect
-(NSArray<UICollectionViewLayoutAttributes *> *)layoutAttributesForElementsInRect:(CGRect)rect {
NSMutableArray

<UICollectionViewLayoutAttributes *> *layoutAttributes = [NSMutableArray array];

NSInteger sectionsCount = [self.collectionView.dataSource numberOfSectionsInCollectionView:self.collectionView];

NSLog(@"Sections count: %ld", (long)sectionsCount);

for(int i = 0; i < sectionsCount; i++) {
    //for header
    UICollectionViewLayoutAttributes *headerAttributes = [self layoutAttributesForSupplementaryViewOfKind:UICollectionElementKindSectionHeader atIndexPath:[NSIndexPath indexPathForItem:0 inSection:i]];
//        if(CGRectIntersectsRect(headerAttributes.frame, rect)) {
//            NSLog(@"Adding Section Attribute");
        [layoutAttributes addObject:headerAttributes];
//        }

    for (UICollectionViewLayoutAttributes *attributes in cachedAttributes) {
        if(CGRectIntersectsRect(attributes.frame, rect)) {
            [layoutAttributes addObject:attributes];
        }
    }

//        NSInteger itemsCount = [self.collectionView numberOfItemsInSection:i];
//        for(int j = 0; j < itemsCount; j++) {
//            UICollectionViewLayoutAttributes *attributes = [self layoutAttributesForItemAtIndexPath:[NSIndexPath indexPathForItem:j inSection:i]];
//            if(CGRectIntersectsRect(attributes.frame, rect)) {
//                [layoutAttributes addObject:attributes];
//            }
//        }
    }


return layoutAttributes;
}

//-(UICollectionViewLayoutAttributes*)layoutAttributesForItemAtIndexPath:(NSIndexPath *)indexPath {
//    NSLog(@"Layout Attributes For Item: %ld %ld", [indexPath row], [indexPath section]);
//    
//    UICollectionViewLayoutAttributes *attributes = [cachedAttributes objectAtIndex:[indexPath row]];
//    return attributes;
//}

-(UICollectionViewLayoutAttributes *)layoutAttributesForSupplementaryViewOfKind:(NSString *)elementKind atIndexPath:(NSIndexPath *)indexPath {

NSLog(@"Layout Attributes For Header: %@ %ld %ld", elementKind, [indexPath row], [indexPath section]);

UICollectionViewLayoutAttributes *attributes = [UICollectionViewLayoutAttributes layoutAttributesForCellWithIndexPath:indexPath];
[attributes setFrame:CGRectMake(0, 0, contentWidth /*320*/, 75.0f)];

return attributes;

}

正如你在MyFlowLayout中看到的那样,我尝试了一些东西,但它仍然没有调用viewForSupplementaryElementOfKind。我尝试实现layoutAttributesForItemAtIndexPath,因为我有layoutAttributesForSupplementaryViewOfKind。我也尝试使用75和320这样的固定数字来检查是否会生成标题。您看到的代表,只是为了获得视图的高度。

由于我已经给了标题足够的空间可以看到并且还剩下NSLogs(UICollectionReusableView *)collectionView:(UICollectionView *)collectionView viewForSupplementaryElementOfKind:(NSString *)kind atIndexPath:(NSIndexPath *)indexPath is没被打电话。

以下是我查看并试过的一些内容:stackoverflow但您可以从我已包含的注释代码中看到。它仍然没有用。

感谢您的帮助。

2 个答案:

答案 0 :(得分:7)

我也有这个问题,我的解决方案是在prepareLayout

中添加部分的属性
UICollectionViewLayoutAttributes *attributes = [UICollectionViewLayoutAttributes layoutAttributesForSupplementaryViewOfKind:UICollectionElementKindSectionHeader withIndexPath:[NSIndexPath indexPathForItem:0 inSection:0]];
attributes.frame = CGRectMake(0, 0, self.collectionView.frame.size.width, 100);
[cachedAttributes addObject:attributes];

答案 1 :(得分:6)

添加标题部分的默认大小:

collectionViewLayout.headerReferenceSize = CGSize(width, height)

默认大小值为(0,0)。这就是为什么标题没有显示。

OR

还有另一种方法,实现UICollectionView的委托方法:

collectionView(_:layout:referenceSizeForFooterInSection:)