UICollectionView系列API,屬性含義筆記。在UICollectionView筆記1中咱們瞭解了UICollectionView是什麼,以及能夠作什麼;在UICollectionView筆記2中咱們瞭解了UICollectionViewLayout相關內容。這一篇咱們就UICollectionView相關API作一個總體的瞭解。html
UICollectionView繼承自UIScrollView,是一種新的數據展現方式。內置的UICollectionViewFlowLayout提供了多行多列的展現方式,UICollectionViewDataSouce提供了數據源協議,UICollectionViewDelegate提供了UI交互的先關協議,使用UICollectionView咱們能夠用不多的代碼就能夠實現不少複雜的效果,下面是相關API以及屬性:數組
//直接初始化方法,須要提供一個Layout - (instancetype)initWithFrame:(CGRect)frame collectionViewLayout:(UICollectionViewLayout *)layout NS_DESIGNATED_INITIALIZER; //從nib文件直接初始化 - (nullable instancetype)initWithCoder:(NSCoder *)aDecoder NS_DESIGNATED_INITIALIZER; //當前Layout @property (nonatomic, strong) UICollectionViewLayout *collectionViewLayout; //用戶交互Delegate @property (nonatomic, weak, nullable) id <UICollectionViewDelegate> delegate; //UI的數據源DataSource @property (nonatomic, weak, nullable) id <UICollectionViewDataSource> dataSource; //iOS10新增長的Pre-Fetching預加載協議 @property (nonatomic, weak, nullable) id<UICollectionViewDataSourcePrefetching> prefetchDataSource NS_AVAILABLE_IOS(10_0); //是否容許預加載 @property (nonatomic, getter=isPrefetchingEnabled) BOOL prefetchingEnabled NS_AVAILABLE_IOS(10_0); //背景圖片,自動根據collectionView的大小調整 @property (nonatomic, strong, nullable) UIView *backgroundView; //註冊UICollectionViewCell,或其子類,而且提供一個複用標識符,在runtime的時候會初始化這個對象 - (void)registerClass:(nullable Class)cellClass forCellWithReuseIdentifier:(NSString *)identifier; //經過nib文件註冊複用cell - (void)registerNib:(nullable UINib *)nib forCellWithReuseIdentifier:(NSString *)identifier; //經過Class註冊補充視圖,而且指定補充視圖的種類 - (void)registerClass:(nullable Class)viewClass forSupplementaryViewOfKind:(NSString *)elementKind withReuseIdentifier:(NSString *)identifier; //經過nib文件註冊補充視圖 - (void)registerNib:(nullable UINib *)nib forSupplementaryViewOfKind:(NSString *)kind withReuseIdentifier:(NSString *)identifier; //從複用隊列經過指定Identifier獲取複用cell - (__kindof UICollectionViewCell *)dequeueReusableCellWithReuseIdentifier:(NSString *)identifier forIndexPath:(NSIndexPath *)indexPath; //從複用隊列經過指定Identifier獲取複用ReusableView - (__kindof UICollectionReusableView *)dequeueReusableSupplementaryViewOfKind:(NSString *)elementKind withReuseIdentifier:(NSString *)identifier forIndexPath:(NSIndexPath *)indexPath; //是否容許items選中,默認YES @property (nonatomic) BOOL allowsSelection; // default is YES //是否容許items多選,默認NO @property (nonatomic) BOOL allowsMultipleSelection; // default is NO //返回選中的index paths,返回nil或者NSArray #if UIKIT_DEFINE_AS_PROPERTIES @property (nonatomic, readonly, nullable) NSArray<NSIndexPath *> *indexPathsForSelectedItems; #else - (nullable NSArray<NSIndexPath *> *)indexPathsForSelectedItems; #endif //選中和解選indexPath對應item,可指定動畫和UICollectionViewScrollPosition - (void)selectItemAtIndexPath:(nullable NSIndexPath *)indexPath animated:(BOOL)animated scrollPosition:(UICollectionViewScrollPosition)scrollPosition; - (void)deselectItemAtIndexPath:(NSIndexPath *)indexPath animated:(BOOL)animated; //刷新數據 - (void)reloadData; //切換當前collectionView的layout - (void)setCollectionViewLayout:(UICollectionViewLayout *)layout animated:(BOOL)animated; //切換當前collectionView的layout,配置完成以後的block(iOS7以後可用) - (void)setCollectionViewLayout:(UICollectionViewLayout *)layout animated:(BOOL)animated completion:(void (^ __nullable)(BOOL finished))completion NS_AVAILABLE_IOS(7_0); //改變當前collectionView的layout,使用interactive transition effect. - (UICollectionViewTransitionLayout *)startInteractiveTransitionToCollectionViewLayout:(UICollectionViewLayout *)layout completion:(nullable UICollectionViewLayoutInteractiveTransitionCompletion)completion NS_AVAILABLE_IOS(7_0); //layout切換過渡完成 - (void)finishInteractiveTransition NS_AVAILABLE_IOS(7_0); //取消過渡切換layout - (void)cancelInteractiveTransition NS_AVAILABLE_IOS(7_0); //下面是UICollectionView的狀態信息 //返回section的數量 #if UIKIT_DEFINE_AS_PROPERTIES @property (nonatomic, readonly) NSInteger numberOfSections; #else - (NSInteger)numberOfSections; #endif //返回對應section中item的數量 - (NSInteger)numberOfItemsInSection:(NSInteger)section; //返回對應indexPath的item的LayoutAttributes - (nullable UICollectionViewLayoutAttributes *)layoutAttributesForItemAtIndexPath:(NSIndexPath *)indexPath; //返回對應indexPath的ReuseView的LayoutAttributes - (nullable UICollectionViewLayoutAttributes *)layoutAttributesForSupplementaryElementOfKind:(NSString *)kind atIndexPath:(NSIndexPath *)indexPath; //返回對應point的indexPath - (nullable NSIndexPath *)indexPathForItemAtPoint:(CGPoint)point; //返回對應cell的indexPath - (nullable NSIndexPath *)indexPathForCell:(UICollectionViewCell *)cell; //返回對應indexPath的cell - (nullable UICollectionViewCell *)cellForItemAtIndexPath:(NSIndexPath *)indexPath; //返回collectionView當前可見的item數組和可見item的indexPath數組 #if UIKIT_DEFINE_AS_PROPERTIES @property (nonatomic, readonly) NSArray<__kindof UICollectionViewCell *> *visibleCells; @property (nonatomic, readonly) NSArray<NSIndexPath *> *indexPathsForVisibleItems; #else - (NSArray<__kindof UICollectionViewCell *> *)visibleCells; - (NSArray<NSIndexPath *> *)indexPathsForVisibleItems; #endif //返回對應indexPath的ReuseView(iOS9以後可用) - (nullable UICollectionReusableView *)supplementaryViewForElementKind:(NSString *)elementKind atIndexPath:(NSIndexPath *)indexPath NS_AVAILABLE_IOS(9_0); //返回collectionView當前可見的ReuseView數組 - (NSArray<UICollectionReusableView *> *)visibleSupplementaryViewsOfKind:(NSString *)elementKind NS_AVAILABLE_IOS(9_0); //返回collectionView當前可見ReuseView的indexPath數組 - (NSArray<NSIndexPath *> *)indexPathsForVisibleSupplementaryElementsOfKind:(NSString *)elementKind NS_AVAILABLE_IOS(9_0); //下面是UICollectionView的交互相關信息 //滾動到指定的indexPath - (void)scrollToItemAtIndexPath:(NSIndexPath *)indexPath atScrollPosition:(UICollectionViewScrollPosition)scrollPosition animated:(BOOL)animated; //插入一個section - (void)insertSections:(NSIndexSet *)sections; //刪除一個section - (void)deleteSections:(NSIndexSet *)sections; //刷新一個section - (void)reloadSections:(NSIndexSet *)sections; //移動一個section到另一個section - (void)moveSection:(NSInteger)section toSection:(NSInteger)newSection; //插入一個section - (void)insertItemsAtIndexPaths:(NSArray<NSIndexPath *> *)indexPaths; //刪除一個section - (void)deleteItemsAtIndexPaths:(NSArray<NSIndexPath *> *)indexPaths; //刷新一個section - (void)reloadItemsAtIndexPaths:(NSArray<NSIndexPath *> *)indexPaths; //移動一個indexPath的item到另外一個indexPath的item - (void)moveItemAtIndexPath:(NSIndexPath *)indexPath toIndexPath:(NSIndexPath *)newIndexPath; //一次性操做插入,刪除,刷新,移動操做 //Animates multiple insert, delete, reload, and move operations as a group - (void)performBatchUpdates:(void (^ __nullable)(void))updates completion:(void (^ __nullable)(BOOL finished))completion; // allows multiple insert/delete/reload/move calls to be animated simultaneously. Nestable. // 排序相關 //是否容許排序,默認YES(iOS9以後有效) - (BOOL)beginInteractiveMovementForItemAtIndexPath:(NSIndexPath *)indexPath NS_AVAILABLE_IOS(9_0); //更新item的位置 - (void)updateInteractiveMovementTargetPosition:(CGPoint)targetPosition NS_AVAILABLE_IOS(9_0); //移動item到新的position - (void)endInteractiveMovement NS_AVAILABLE_IOS(9_0); //回覆item到原始的position - (void)cancelInteractiveMovement NS_AVAILABLE_IOS(9_0); //是否記住最後操做的indexPath,默認NO @property (nonatomic) BOOL remembersLastFocusedIndexPath NS_AVAILABLE_IOS(9_0); //UICollectionView 補充 的indexPath @interface NSIndexPath (UICollectionViewAdditions) //初始化indexPath + (instancetype)indexPathForItem:(NSInteger)item inSection:(NSInteger)section NS_AVAILABLE_IOS(6_0); //返回IndexPath的item @property (nonatomic, readonly) NSInteger item NS_AVAILABLE_IOS(6_0); @end //UICollectionViewDataSource數據源協議 @protocol UICollectionViewDataSource <NSObject> @required //配置UICollectionView的section對應item的數量 - (NSInteger)collectionView:(UICollectionView *)collectionView numberOfItemsInSection:(NSInteger)section; //配置UICollectionView的item,須要提早註冊 - (__kindof UICollectionViewCell *)collectionView:(UICollectionView *)collectionView cellForItemAtIndexPath:(NSIndexPath *)indexPath; @optional //配置UICollectionView的section數量 - (NSInteger)numberOfSectionsInCollectionView:(UICollectionView *)collectionView; //配置UICollectionView的ReuseView,須要提早註冊 - (UICollectionReusableView *)collectionView:(UICollectionView *)collectionView viewForSupplementaryElementOfKind:(NSString *)kind atIndexPath:(NSIndexPath *)indexPath; //是否能夠移動 - (BOOL)collectionView:(UICollectionView *)collectionView canMoveItemAtIndexPath:(NSIndexPath *)indexPath NS_AVAILABLE_IOS(9_0); //移動indexPath的item到另外一個indexPath - (void)collectionView:(UICollectionView *)collectionView moveItemAtIndexPath:(NSIndexPath *)sourceIndexPath toIndexPath:(NSIndexPath*)destinationIndexPath NS_AVAILABLE_IOS(9_0); @end //UICollectionViewDelegate UI交互協議 @protocol UICollectionViewDelegate <UIScrollViewDelegate> @optional //是否容許高亮 - (BOOL)collectionView:(UICollectionView *)collectionView shouldHighlightItemAtIndexPath:(NSIndexPath *)indexPath; //已經高亮 - (void)collectionView:(UICollectionView *)collectionView didHighlightItemAtIndexPath:(NSIndexPath *)indexPath; //取消高亮 - (void)collectionView:(UICollectionView *)collectionView didUnhighlightItemAtIndexPath:(NSIndexPath *)indexPath; //是否容許選中 - (BOOL)collectionView:(UICollectionView *)collectionView shouldSelectItemAtIndexPath:(NSIndexPath *)indexPath; //是否容許解選 - (BOOL)collectionView:(UICollectionView *)collectionView shouldDeselectItemAtIndexPath:(NSIndexPath *)indexPath; // called when the user taps on an already-selected item in multi-select mode //已經選中 - (void)collectionView:(UICollectionView *)collectionView didSelectItemAtIndexPath:(NSIndexPath *)indexPath; //已經解選 - (void)collectionView:(UICollectionView *)collectionView didDeselectItemAtIndexPath:(NSIndexPath *)indexPath; //將要顯示item(iOS8以後可用) - (void)collectionView:(UICollectionView *)collectionView willDisplayCell:(UICollectionViewCell *)cell forItemAtIndexPath:(NSIndexPath *)indexPath NS_AVAILABLE_IOS(8_0); //將要顯示SupplementaryView(iOS8以後可用) - (void)collectionView:(UICollectionView *)collectionView willDisplaySupplementaryView:(UICollectionReusableView *)view forElementKind:(NSString *)elementKind atIndexPath:(NSIndexPath *)indexPath NS_AVAILABLE_IOS(8_0); //已經顯示item - (void)collectionView:(UICollectionView *)collectionView didEndDisplayingCell:(UICollectionViewCell *)cell forItemAtIndexPath:(NSIndexPath *)indexPath; //已經顯示SupplementaryView - (void)collectionView:(UICollectionView *)collectionView didEndDisplayingSupplementaryView:(UICollectionReusableView *)view forElementOfKind:(NSString *)elementKind atIndexPath:(NSIndexPath *)indexPath; //是否容許展現copy/paste Menu - (BOOL)collectionView:(UICollectionView *)collectionView shouldShowMenuForItemAtIndexPath:(NSIndexPath *)indexPath; //是否能夠執行SEL - (BOOL)collectionView:(UICollectionView *)collectionView canPerformAction:(SEL)action forItemAtIndexPath:(NSIndexPath *)indexPath withSender:(nullable id)sender; //執行SEL - (void)collectionView:(UICollectionView *)collectionView performAction:(SEL)action forItemAtIndexPath:(NSIndexPath *)indexPath withSender:(nullable id)sender; @end
UICollectionView爲了優化性能,item和supplementView都採用複用隊列的形式來返回,普通的補充視圖直接就是UICollectionReusableView及其子類,item即UICollectionViewCell屬於UICollectionReusableView的子類,下面是UICollectionReusableView的頭文件聲明:app
//獲取複用標識符reuseIdentifier @property (nonatomic, readonly, copy, nullable) NSString *reuseIdentifier; //從複用隊列取到collectionView以前調用,子類在重寫這個方法,完成返回以前的須要的配置 - (void)prepareForReuse; //根據LayoutAttributes來佈局當前View,在view被添加到collectionView上可是尚未被服用隊列返回的時候調用 - (void)applyLayoutAttributes:(UICollectionViewLayoutAttributes *)layoutAttributes; // Override these methods to provide custom UI for specific layouts. - (void)willTransitionFromLayout:(UICollectionViewLayout *)oldLayout toLayout:(UICollectionViewLayout *)newLayout; - (void)didTransitionFromLayout:(UICollectionViewLayout *)oldLayout toLayout:(UICollectionViewLayout *)newLayout; - (UICollectionViewLayoutAttributes *)preferredLayoutAttributesFittingAttributes:(UICollectionViewLayoutAttributes *)layoutAttributes NS_AVAILABLE_IOS(8_0);
下面是UICollectionViewCell的頭文件聲明:ide
//內容視圖,子類圖須要粘貼在contentView上面 @property (nonatomic, readonly) UIView *contentView; //設置item的選中狀態 @property (nonatomic, getter=isSelected) BOOL selected; //設置item的高亮狀態 @property (nonatomic, getter=isHighlighted) BOOL highlighted; //背景圖片 @property (nonatomic, strong, nullable) UIView *backgroundView; //選中背景圖片 @property (nonatomic, strong, nullable) UIView *selectedBackgroundView;
UICollectionViewContoller繼承自UIViewController,默認實現了UICollectionViewDelegate和UICollectionViewDataSource,提供了一種簡單使用UICollectionViewContoller的方式,相似於UITableViewController函數
//用指定的layout初始化UICollectionViewController - (instancetype)initWithCollectionViewLayout:(UICollectionViewLayout *)layout NS_DESIGNATED_INITIALIZER; //當前的collectionView @property (nullable, nonatomic, strong) __kindof UICollectionView *collectionView; //在viewWillAppear:時是否清除選中效果,默認YES @property (nonatomic) BOOL clearsSelectionOnViewWillAppear; //在導航過渡的時候是否使用layout,在push以前設置,以後設置無效 @property (nonatomic, assign) BOOL useLayoutToLayoutNavigationTransitions NS_AVAILABLE_IOS(7_0); //當前的collectionViewLayout,只讀屬性 @property (nonatomic, readonly) UICollectionViewLayout *collectionViewLayout NS_AVAILABLE_IOS(7_0); //是否容許排序,默認是YES @property (nonatomic) BOOL installsStandardGestureForInteractiveMovement NS_AVAILABLE_IOS(9_0);
UICollectionViewFlowLayout是系統自帶的簡單的網格佈局,用來展現多行多列數據。能夠經過下面屬性的配置來控制全局item的大小佈局
//設置item行之間的最小間隔 @property (nonatomic) CGFloat minimumLineSpacing; //設置item之間的最小間隔 @property (nonatomic) CGFloat minimumInteritemSpacing; //設置item的大小 @property (nonatomic) CGSize itemSize; //設置item大小的近似值,默認是CGSizeZero(iOS8以後有效) @property (nonatomic) CGSize estimatedItemSize NS_AVAILABLE_IOS(8_0); // defaults to CGSizeZero - setting a non-zero size enables cells that self-size via -preferredLayoutAttributesFittingAttributes: //設置滾動方向 @property (nonatomic) UICollectionViewScrollDirection scrollDirection; // default is UICollectionViewScrollDirectionVertical //設置頭部引用視圖的大小,對於垂直滾動時,寬不起做用,高度起做用;水平滾動相反 @property (nonatomic) CGSize headerReferenceSize; ////設置尾部引用視圖的大小 @property (nonatomic) CGSize footerReferenceSize; //設置section的上下左右留白 @property (nonatomic) UIEdgeInsets sectionInset; //下面兩個屬性在iOS9以後可用,用來設置是否粘貼頭部在屏幕的頭部和尾部在屏幕的尾部 @property (nonatomic) BOOL sectionHeadersPinToVisibleBounds NS_AVAILABLE_IOS(9_0); @property (nonatomic) BOOL sectionFootersPinToVisibleBounds NS_AVAILABLE_IOS(9_0);
固然也能夠細緻的配置每個item大小,這時須要實現UICollectionViewDelegateFlowLayout協議性能
//設置具體indexPath的item大小 - (CGSize)collectionView:(UICollectionView *)collectionView layout:(UICollectionViewLayout*)collectionViewLayout sizeForItemAtIndexPath:(NSIndexPath *)indexPath; //設置具體section的上下左右留白 - (UIEdgeInsets)collectionView:(UICollectionView *)collectionView layout:(UICollectionViewLayout*)collectionViewLayout insetForSectionAtIndex:(NSInteger)section; //設置行最小間隔 - (CGFloat)collectionView:(UICollectionView *)collectionView layout:(UICollectionViewLayout*)collectionViewLayout minimumLineSpacingForSectionAtIndex:(NSInteger)section; //設置item間最小間隔 - (CGFloat)collectionView:(UICollectionView *)collectionView layout:(UICollectionViewLayout*)collectionViewLayout minimumInteritemSpacingForSectionAtIndex:(NSInteger)section; //設置頭部引用視圖的大小 - (CGSize)collectionView:(UICollectionView *)collectionView layout:(UICollectionViewLayout*)collectionViewLayout referenceSizeForHeaderInSection:(NSInteger)section; //設置尾部引用視圖的大小 - (CGSize)collectionView:(UICollectionView *)collectionView layout:(UICollectionViewLayout*)collectionViewLayout referenceSizeForFooterInSection:(NSInteger)section;
UICollectionViewLayout是UICollectionView的精髓,是區別於UITableView的一個重要特徵,UICollectionView經過 UICollectionViewLayout 返回的 UICollectionViewLayoutAttributes 來佈局每個item。子類經過重寫UICollectionViewLayout中的方法能夠實現各類各樣的佈局樣式,也可使用另一套佈局來更新當前佈局樣式,更可結合 UIDynamicBehavior 等UI動力學完成各類各樣的酷炫效果,下面是UICollectionViewLayoutAttributes的頭文件內容:fetch
//設置item的frame @property (nonatomic) CGRect frame; //設置item的center @property (nonatomic) CGPoint center; //設置item的size @property (nonatomic) CGSize size; //設置item的transform3D @property (nonatomic) CATransform3D transform3D; //設置item的bounds @property (nonatomic) CGRect bounds NS_AVAILABLE_IOS(7_0); //設置item的transform @property (nonatomic) CGAffineTransform transform NS_AVAILABLE_IOS(7_0); //設置item的alpha @property (nonatomic) CGFloat alpha; //設置item的zIndex,默認是0和其餘item在同一平面,設置小於0,在其餘item下面 @property (nonatomic) NSInteger zIndex; // default is 0 //設置item的hidden狀態,一般爲NO @property (nonatomic, getter=isHidden) BOOL hidden; //設置item的indexPath @property (nonatomic, strong) NSIndexPath *indexPath; //設置當前元素的類別,是個美劇變量 cell/supplementaryView/decorationView @property (nonatomic, readonly) UICollectionElementCategory representedElementCategory; //展現的元素類型,若是是cell 則爲nil @property (nonatomic, readonly, nullable) NSString *representedElementKind; // nil when representedElementCategory is UICollectionElementCategoryCell //指定indexPath的cell的layoutAttributes + (instancetype)layoutAttributesForCellWithIndexPath:(NSIndexPath *)indexPath; //指定indexPath的supplementaryView的layoutAttributes + (instancetype)layoutAttributesForSupplementaryViewOfKind:(NSString *)elementKind withIndexPath:(NSIndexPath *)indexPath; //指定indexPath的decorationView的layoutAttributes + (instancetype)layoutAttributesForDecorationViewOfKind:(NSString *)decorationViewKind withIndexPath:(NSIndexPath *)indexPath;
下面是UICollectionViewLayout相關的API:優化
//構造函數 - (instancetype)init NS_DESIGNATED_INITIALIZER; - (nullable instancetype)initWithCoder:(NSCoder *)aDecoder NS_DESIGNATED_INITIALIZER; //當前layout服務的collectionView對象 @property (nullable, nonatomic, readonly) UICollectionView *collectionView; //驗證當前layout,會觸發collectionView的reloadData - (void)invalidateLayout; //驗證當前layout,而且提供一個驗證的上下文,也會觸發collectionView的reloadData - (void)invalidateLayoutWithContext:(UICollectionViewLayoutInvalidationContext *)context NS_AVAILABLE_IOS(7_0); //註冊decorationView用Class/nib - (void)registerClass:(nullable Class)viewClass forDecorationViewOfKind:(NSString *)elementKind; - (void)registerNib:(nullable UINib *)nib forDecorationViewOfKind:(NSString *)elementKind;
自定義layout須要關注的API:動畫
//自定義layoutAttributesClass和invalidationContextClass在須要的時候 #if UIKIT_DEFINE_AS_PROPERTIES @property(class, nonatomic, readonly) Class layoutAttributesClass; // override this method to provide a custom class to be used when instantiating instances of UICollectionViewLayoutAttributes @property(class, nonatomic, readonly) Class invalidationContextClass NS_AVAILABLE_IOS(7_0); // override this method to provide a custom class to be used for invalidation contexts #else + (Class)layoutAttributesClass; // override this method to provide a custom class to be used when instantiating instances of UICollectionViewLayoutAttributes + (Class)invalidationContextClass NS_AVAILABLE_IOS(7_0); // override this method to provide a custom class to be used for invalidation contexts #endif //UICollectionView在第一次layout的時候會調用prepareLayout,而且在當前layout invalidated以後也會調用這個方法,子類須要重寫此方法完成相關配置 - (void)prepareLayout; //UICollectionView會調用下面四個方法完成相關的配置信息 //返回給定rect內的layoutAttributes數組 - (nullable NSArray<__kindof UICollectionViewLayoutAttributes *> *)layoutAttributesForElementsInRect:(CGRect)rect; //返回給定indexPath的item的layoutAttributes - (nullable UICollectionViewLayoutAttributes *)layoutAttributesForItemAtIndexPath:(NSIndexPath *)indexPath; //返回給定indexPath的supplementaryView的layoutAttributes,若是有 - (nullable UICollectionViewLayoutAttributes *)layoutAttributesForSupplementaryViewOfKind:(NSString *)elementKind atIndexPath:(NSIndexPath *)indexPath; //返回給定indexPath的decorationView的layoutAttributes,若是有 - (nullable UICollectionViewLayoutAttributes *)layoutAttributesForDecorationViewOfKind:(NSString*)elementKind atIndexPath:(NSIndexPath *)indexPath; //是否驗證給定的bounds,會引發UICollectionView刷新layout - (BOOL)shouldInvalidateLayoutForBoundsChange:(CGRect)newBounds; //返回給定bounds的無效上下文 - (UICollectionViewLayoutInvalidationContext *)invalidationContextForBoundsChange:(CGRect)newBounds NS_AVAILABLE_IOS(7_0); //是否驗證layoutAttributes - (BOOL)shouldInvalidateLayoutForPreferredLayoutAttributes:(UICollectionViewLayoutAttributes *)preferredAttributes withOriginalAttributes:(UICollectionViewLayoutAttributes *)originalAttributes NS_AVAILABLE_IOS(8_0); - (UICollectionViewLayoutInvalidationContext *)invalidationContextForPreferredLayoutAttributes:(UICollectionViewLayoutAttributes *)preferredAttributes withOriginalAttributes:(UICollectionViewLayoutAttributes *)originalAttributes NS_AVAILABLE_IOS(8_0); - (CGPoint)targetContentOffsetForProposedContentOffset:(CGPoint)proposedContentOffset withScrollingVelocity:(CGPoint)velocity; // return a point at which to rest after scrolling - for layouts that want snap-to-point scrolling behavior - (CGPoint)targetContentOffsetForProposedContentOffset:(CGPoint)proposedContentOffset NS_AVAILABLE_IOS(7_0); // a layout can return the content offset to be applied during transition or update animations //返回collectionView的contentSize #if UIKIT_DEFINE_AS_PROPERTIES @property(nonatomic, readonly) CGSize collectionViewContentSize; // Subclasses must override this method and use it to return the width and height of the collection view’s content. These values represent the width and height of all the content, not just the content that is currently visible. The collection view uses this information to configure its own content size to facilitate scrolling. #else - (CGSize)collectionViewContentSize; // Subclasses must override this method and use it to return the width and height of the collection view’s content. These values represent the width and height of all the content, not just the content that is currently visible. The collection view uses this information to configure its own content size to facilitate scrolling. #endif
刷新layout須要關注的 API:
- (void)prepareForCollectionViewUpdates:(NSArray<UICollectionViewUpdateItem *> *)updateItems; - (void)finalizeCollectionViewUpdates; // called inside an animation block after the update - (void)prepareForAnimatedBoundsChange:(CGRect)oldBounds; // UICollectionView calls this when its bounds have changed inside an animation block before displaying cells in its new bounds - (void)finalizeAnimatedBoundsChange; // also called inside the animation block // UICollectionView calls this when prior the layout transition animation on the incoming and outgoing layout - (void)prepareForTransitionToLayout:(UICollectionViewLayout *)newLayout NS_AVAILABLE_IOS(7_0); - (void)prepareForTransitionFromLayout:(UICollectionViewLayout *)oldLayout NS_AVAILABLE_IOS(7_0); - (void)finalizeLayoutTransition NS_AVAILABLE_IOS(7_0); // called inside an animation block after the transition // This set of methods is called when the collection view undergoes an animated transition such as a batch update block or an animated bounds change. // For each element on screen before the invalidation, finalLayoutAttributesForDisappearingXXX will be called and an animation setup from what is on screen to those final attributes. // For each element on screen after the invalidation, initialLayoutAttributesForAppearingXXX will be called and an animation setup from those initial attributes to what ends up on screen. - (nullable UICollectionViewLayoutAttributes *)initialLayoutAttributesForAppearingItemAtIndexPath:(NSIndexPath *)itemIndexPath; - (nullable UICollectionViewLayoutAttributes *)finalLayoutAttributesForDisappearingItemAtIndexPath:(NSIndexPath *)itemIndexPath; - (nullable UICollectionViewLayoutAttributes *)initialLayoutAttributesForAppearingSupplementaryElementOfKind:(NSString *)elementKind atIndexPath:(NSIndexPath *)elementIndexPath; - (nullable UICollectionViewLayoutAttributes *)finalLayoutAttributesForDisappearingSupplementaryElementOfKind:(NSString *)elementKind atIndexPath:(NSIndexPath *)elementIndexPath; - (nullable UICollectionViewLayoutAttributes *)initialLayoutAttributesForAppearingDecorationElementOfKind:(NSString *)elementKind atIndexPath:(NSIndexPath *)decorationIndexPath; - (nullable UICollectionViewLayoutAttributes *)finalLayoutAttributesForDisappearingDecorationElementOfKind:(NSString *)elementKind atIndexPath:(NSIndexPath *)decorationIndexPath; // These methods are called by collection view during an update block. // Return an array of index paths to indicate views that the layout is deleting or inserting in response to the update. - (NSArray<NSIndexPath *> *)indexPathsToDeleteForSupplementaryViewOfKind:(NSString *)elementKind NS_AVAILABLE_IOS(7_0); - (NSArray<NSIndexPath *> *)indexPathsToDeleteForDecorationViewOfKind:(NSString *)elementKind NS_AVAILABLE_IOS(7_0); - (NSArray<NSIndexPath *> *)indexPathsToInsertForSupplementaryViewOfKind:(NSString *)elementKind NS_AVAILABLE_IOS(7_0); - (NSArray<NSIndexPath *> *)indexPathsToInsertForDecorationViewOfKind:(NSString *)elementKind NS_AVAILABLE_IOS(7_0);
排序須要關注的layout API:
- (NSIndexPath *)targetIndexPathForInteractivelyMovingItem:(NSIndexPath *)previousIndexPath withPosition:(CGPoint)position NS_AVAILABLE_IOS(9_0); - (UICollectionViewLayoutAttributes *)layoutAttributesForInteractivelyMovingItemAtIndexPath:(NSIndexPath *)indexPath withTargetPosition:(CGPoint)position NS_AVAILABLE_IOS(9_0); - (UICollectionViewLayoutInvalidationContext *)invalidationContextForInteractivelyMovingItems:(NSArray<NSIndexPath *> *)targetIndexPaths withTargetPosition:(CGPoint)targetPosition previousIndexPaths:(NSArray<NSIndexPath *> *)previousIndexPaths previousPosition:(CGPoint)previousPosition NS_AVAILABLE_IOS(9_0); - (UICollectionViewLayoutInvalidationContext *)invalidationContextForEndingInteractiveMovementOfItemsToFinalIndexPaths:(NSArray<NSIndexPath *> *)indexPaths previousIndexPaths:(NSArray<NSIndexPath *> *)previousIndexPaths movementCancelled:(BOOL)movementCancelled NS_AVAILABLE_IOS(9_0);
UICollectionViewTransitionLayout繼承自UICollectionViewLayout,用於切換當前的layout,提供過渡時期的layout
//UICollectionView過渡的進度值 @property (assign, nonatomic) CGFloat transitionProgress; //當前UICollectionView的layout @property (readonly, nonatomic) UICollectionViewLayout *currentLayout; //切換以後的layout @property (readonly, nonatomic) UICollectionViewLayout *nextLayout; //初始化TransitionLayout對象 - (instancetype)initWithCurrentLayout:(UICollectionViewLayout *)currentLayout nextLayout:(UICollectionViewLayout *)newLayout NS_DESIGNATED_INITIALIZER; - (nullable instancetype)initWithCoder:(NSCoder *)aDecoder NS_DESIGNATED_INITIALIZER; - (instancetype)init NS_UNAVAILABLE; //下面的方法用於儲存過渡期LayoutAttributes的變化值 - (void)updateValue:(CGFloat)value forAnimatedKey:(NSString *)key; - (CGFloat)valueForAnimatedKey:(NSString *)key;