Collection Views are a powerful user interface technology for laying out collections of data. With fantastic support for high performance grid layouts, flexible line based layouts, and fully custom layouts, along with integrated animation capabilities, they make creating beautiful data-driven UIs a wonderful experience both for the developer and the end user.
iOS 8 adds some nice enhancements to Collection Views, including the ability to self-size cells without the need for using a delegate. Likewise, the layout attributes of individual cells can now be adjusted directly from within the cell class without having to use a delegate. This makes it easier to make small layout changes that are related to a particular cell.
Self-sizing Cells
First, let’s look at how to self-size cells. Before iOS 8, creating cells that sized to their content required working within a layout class directly or via a layout’s delegate, such as the UICollectionViewDelegateFlowLayout.
For example, the following code adjusts the cell size to fit cells comprised of simple UILabels, presented using a UICollectionViewFlowLayout:
class FlowLayoutDelegate : UICollectionViewDelegateFlowLayout { string[] items; UIStringAttributes attr; public FlowLayoutDelegate (string[] items) { this.items = items; attr = new UIStringAttributes { Font = new UILabel ().Font }; } public override SizeF GetSizeForItem (UICollectionView collectionView, UICollectionViewLayout layout, NSIndexPath indexPath) { string text = items [indexPath.Row]; return new NSString (text).GetSizeUsingAttributes (attr); } }
This code results in a Collection View where each cell fits the text it contains:
Creating cells that are self-sized is much easier in iOS 8. Simply set the EstimatedItemSize on the UICollectionViewFlowLayout and implement SizeThatFits or use AutoLayout in the cell class.
For example, to self-size the cells in the example above, set EstimatedItemSize as shown below:
flowLayout = new UICollectionViewFlowLayout (){ EstimatedItemSize = new SizeF (44, 144) };
Then, simply adding the following implementation of SizeThatFits in the cell class produces the same result, without the need for the UICollectionViewDelegateFlowLayout implementation:
public override SizeF SizeThatFits (SizeF size) { label.Frame = new RectangleF (new PointF (0, 0), label.AttributedText.Size); return label.AttributedText.Size; }
Adjusting Layout Attributes in Cells
iOS 8 also makes it easier to manipulate the layout attributes returned from within the cell class without resorting to a UICollectionViewDelegateFlowLayout. Just override PreferredLayoutAttributesFittingAttributes in the cell class.
The following code changes the font of every other label and adjusts the layout for each cell appropriately:
public override UICollectionViewLayoutAttributes PreferredLayoutAttributesFittingAttributes (UICollectionViewLayoutAttributes layoutAttributes) { var newLayoutAttributes = (UICollectionViewLayoutAttributes)layoutAttributes.Copy (); if (layoutAttributes.IndexPath.Row % 2 == 0) { // label.TextColor = UIColor.Red; ContentView.BackgroundColor = UIColor.LightGray; var attr = new NSAttributedString (Text, UIFont.SystemFontOfSize (28.0f)); label.Font = UIFont.SystemFontOfSize (28.0f); // newLayoutAttributes.Frame = new RectangleF (new PointF (0, 0), attr.Size); label.Frame = new RectangleF (new PointF (0, 0), attr.Size); } else { // newLayoutAttributes.Frame = new RectangleF (new PointF (0, 0), label.AttributedText.Size); label.Frame = new RectangleF (new PointF (0, 0), label.AttributedText.Size); } return newLayoutAttributes; }
This allows fine-grained control at the cell level, as shown below:
As you can see, iOS 8 adds flexibility to Collection Views, allowing granular control of cells directly from the cell class and making it easier to control cell layout and sizing when needed.
The code from this post is available here.
0 comments