UICollectionView自定義佈局(一)

前言

最近看了www.raywenderlich.com的關於UICollectionView自定義佈局的的教程,做一下筆記,方便之後查閱。UICollectionView自定義佈局的基本概念能夠查看喵神的WWDC 2012 Session筆記——219 Advanced Collection Views and Building Custom Layouts這篇文章。git

wheel.gif

基本原理

下面是輪盤旋轉圖解,黃色的區域是手機屏幕,藍色的圓角矩形是UICollectionViewCells,cell以radius爲半徑,以手機屏幕外的一點爲圓心旋轉,每一個cell之間的夾角爲anglePerItemgithub

圖1

正如你看到的,不是全部的cell都在屏幕內,假設第0個cell的旋轉角度是angle,那麼第1個cell的旋轉角度就是angle + anglePerItem,以此類推能夠獲得第i個cell的旋轉角度:bash

CGFloat  angleFori = angle + anglePerItem *i
複製代碼

下面是角度座標是,0°表明的是屏幕中心,向左爲負方向,向右爲正方向,因此當cell的角度爲0°時是垂直居中的。app

圖2

自定義佈局屬性

由於系統的UICollectionViewLayoutAttributes沒有angleanchorPoint屬性,因此咱們繼承UICollectionViewLayoutAttributes自定義佈局屬性。佈局

@implementation WheelCollectionLayoutAttributes
- (instancetype)init{
    self = [super init];
    if (self) {
        self.anchorPoint = CGPointMake(0.5, 0.5);
        self.angle = 0;
    }
    return self;
}

- (id)copyWithZone:(NSZone *)zone{
    WheelCollectionLayoutAttributes *attribute = [super copyWithZone:zone];
    attribute.anchorPoint = self.anchorPoint;
    attribute.angle = self.angle;
    return attribute;
}
@end
複製代碼

初始化時錨點anchorPoint是CGPointMake(0.5, 0.5),angle是0,而且重寫- (id)copyWithZone:(NSZone *)zone方法,當對象被拷貝的時候,保證anchorPointangle屬性被賦值。ui

**注意:**對於自定義的佈局屬性,UICollectionViewCell必須實現下面👇這個方法,用來改變cell的錨點,改變錨點會引發layer的position的變化,關於anchorPointposition的關係請看這篇文章spa

- (void)applyLayoutAttributes:(UICollectionViewLayoutAttributes *)layoutAttributes{
    [super applyLayoutAttributes:layoutAttributes];
    //改變錨點(改變錨點會影響center的位置 參考http://www.jianshu.com/p/15f007f40209 )
    WheelCollectionLayoutAttributes *attribute = (WheelCollectionLayoutAttributes*)layoutAttributes;
    self.layer.anchorPoint = attribute.anchorPoint;
    CGFloat centerY = (attribute.anchorPoint.y - 0.5)*CGRectGetHeight(self.bounds);
    self.center = CGPointMake(self.center.x, centerY + self.center.y);
}
複製代碼

自定義佈局

  1. 首先進行初始化: 👇這個方法告訴CollectionView佈局屬性使用自定義的WheelCollectionLayoutAttributes屬性,而不是使用系統默認的UICollectionViewLayoutAttributes屬性。
+ (Class)layoutAttributesClass{
    return [WheelCollectionLayoutAttributes class];
}
複製代碼
- (instancetype)init{
    self = [super init];
    if (self) {
        /*圓形半徑*/
        self.radius = 500.0;
        /**itemCell的大小*/
        self.itermSize = CGSizeMake(133, 173);

        /*
        每兩個item之間的旋轉角度,由*圖一*可計算獲得
        self.anglePerItem能夠是任意值
        */
        self.anglePerItem = atan(self.itermSize.width / self.radius);
    }
    return self;
}
複製代碼
  1. - (void)prepareLayout當collection view第一次顯示在屏幕上時會被調用,這個方法初始化佈局屬性,並將佈局屬性保存下來。
- (void)prepareLayout{
    [super prepareLayout];
    [self.allAttributeArray removeAllObjects];
    NSUInteger numberOfItem = [self.collectionView numberOfItemsInSection:0];
    if (numberOfItem == 0) {
        return;
    }

    /*獲取總的旋轉的角度*/
    CGFloat angleAtExtreme = (numberOfItem - 1) * self.anglePerItem;

    /*隨着UICollectionView的移動,第0個cell初始時的角度*/
    CGFloat angle = -1 *angleAtExtreme * self.collectionView.contentOffset.x / (self.collectionView.contentSize.width - CGRectGetWidth(self.collectionView.bounds));

    /*當前的屏幕中心的的座標*/
    CGFloat centerX = self.collectionView.contentOffset.x + CGRectGetWidth(self.collectionView.bounds)/2.0;

    /*錨點的位置*/
    CGFloat anchorPointY = (self.itermSize.height/2.0 + self.radius) / self.itermSize.height;
    
    for (int i = 0; i < numberOfItem; i++) {
        WheelCollectionLayoutAttributes *attribute = [WheelCollectionLayoutAttributes layoutAttributesForCellWithIndexPath:[NSIndexPath indexPathForItem:i inSection:0]];
        attribute.anchorPoint = CGPointMake(0.5, anchorPointY);
        attribute.size = self.itermSize;
        attribute.center = CGPointMake(centerX, CGRectGetMidY(self.collectionView.bounds));
        attribute.angle = angle + self.anglePerItem *i;
        attribute.transform = CGAffineTransformMakeRotation(attribute.angle);
        attribute.zIndex = (int)(-1) *i *1000;
        [self.allAttributeArray addObject:attribute];
    }
}
複製代碼

起點和終點的狀態.png

由起點和終點的狀態,當self.collectionView.contentOffset.x = 0時,第0個cell的旋轉角度爲0,當self.collectionView.contentOffset.x = Max時,第0個cell的旋轉角度最大是(numberOfItem - 1) * self.anglePerItem,因此根據數學知識能夠獲得第0個cell的旋轉角度和偏移量之間的關係 (向左爲負值):3d

CGFloat angle = -1 *angleAtExtreme * self.collectionView.contentOffset.x / (self.collectionView.contentSize.width - CGRectGetWidth(self.collectionView.bounds));
複製代碼

因此第i個cell的旋轉的角度爲:code

attribute.angle = angle + self.anglePerItem *i;
複製代碼

錨點.png

由上圖能夠計算出cell的錨點,X軸方向的值不變,Y軸方向的值發生變化:orm

CGFloat anchorPointY = (self.itermSize.height/2.0 + self.radius) / self.itermSize.height;
CGFloat anchorPoint = CGPointMake(0.5, anchorPointY);
複製代碼

3.- (CGSize)collectionViewContentSize返回collectionView的內容大小

- (CGSize)collectionViewContentSize{
    CGFloat numberOfIterm = [self.collectionView numberOfItemsInSection:0];
    return CGSizeMake(numberOfIterm *self.itermSize.width, CGRectGetHeight(self.collectionView.bounds));
}
複製代碼

4.返回對應的佈局屬性

- (NSArray *)layoutAttributesForElementsInRect:(CGRect)rect{
    return self.allAttributeArray;
}

- (UICollectionViewLayoutAttributes *)layoutAttributesForItemAtIndexPath:(NSIndexPath *)indexPath{
    return self.allAttributeArray[indexPath.item];
}
複製代碼
  1. collectionView滑動時,從新對collectionView進行佈局
- (BOOL)shouldInvalidateLayoutForBoundsChange:(CGRect)newBounds{
    return YES;
}
複製代碼

最後你能夠到GitHub下載Demo。

相關文章
相關標籤/搜索