Collection View Compositional Layout with estimated height not working
What solved the issue for me was to set the same height dimension to both the item and the group.
I know it's the case in the question's shared code, and @Que20's solution definitely helped. But in case the problem persists you might wanna check that.
Example
I'm trying to display a cell containing a single UIButton pinned to the cell's margins. The button has a 44 points high height constraint.
Original Attempt
static func mapSection() -> NSCollectionLayoutSection {
let itemSize = NSCollectionLayoutSize(
widthDimension: .fractionalWidth(1),
heightDimension: .fractionalHeight(1)) // Here: a fractional height to the item
let item = NSCollectionLayoutItem(layoutSize: itemSize)
let groupSize = NSCollectionLayoutSize(widthDimension: .fractionalWidth(1),
heightDimension: .estimated(100)) // There: estimated height to the button cell
let group = NSCollectionLayoutGroup.horizontal(layoutSize: groupSize, subitems: [item])
let section = NSCollectionLayoutSection(group: group)
return section
}
Despite everything being configured correctly, the button's height constraint breaks and the "estimation" becomes an absolute value.
Successful implementation
static func mapSection() -> NSCollectionLayoutSection {
/*
Notice that I estimate my button cell to be 500 points high
It's way too much. But since it's an estimation and we're doing things well,
it doesn't really matter to the end result.
*/
let heightDimension = NSCollectionLayoutDimension.estimated(500)
let itemSize = NSCollectionLayoutSize(
widthDimension: .fractionalWidth(1),
heightDimension: heightDimension)
let item = NSCollectionLayoutItem(layoutSize: itemSize)
let groupSize = NSCollectionLayoutSize(widthDimension: .fractionalWidth(1),
heightDimension: heightDimension)
let group = NSCollectionLayoutGroup.horizontal(layoutSize: groupSize, subitems: [item])
let section = NSCollectionLayoutSection(group: group)
return section
}
The only real difference lies in the fact that both the item and the group are set to the same estimated height dimension. But it works now!
I found that making my LayoutGroup horizontal instead of vertical fixed the issue. Here is my final layout :
let estimatedHeight = CGFloat(100)
let layoutSize = NSCollectionLayoutSize(widthDimension: .estimated(200), heightDimension: .estimated(estimatedHeight))
let item = NSCollectionLayoutItem(layoutSize: layoutSize)
let group = NSCollectionLayoutGroup.horizontal(layoutSize: layoutSize, subitem: item, count: 1)
let section = NSCollectionLayoutSection(group: group)
section.contentInsets = NSDirectionalEdgeInsets(top: 10, leading: 10, bottom: 10, trailing: 10)
section.interGroupSpacing = 10
section.orthogonalScrollingBehavior = .groupPaging
Hope it'll help ^^
On iOS13+ if you want to create layout where the cell's constraints determine the height of the cell - try this one:
private func configureCollectionView() {
collectionView.collectionViewLayout = createLayout()
}
private func createLayout() -> UICollectionViewLayout {
let layout = UICollectionViewCompositionalLayout {
(sectionIndex: Int, layoutEnvironment: NSCollectionLayoutEnvironment) -> NSCollectionLayoutSection? in
let size = NSCollectionLayoutSize(widthDimension: .fractionalWidth(1.0), heightDimension: .estimated(10))
let item = NSCollectionLayoutItem(layoutSize: size)
item.edgeSpacing = NSCollectionLayoutEdgeSpacing(leading: nil, top: NSCollectionLayoutSpacing.fixed(20), trailing: nil, bottom: NSCollectionLayoutSpacing.fixed(0))
let group = NSCollectionLayoutGroup.vertical(layoutSize: size, subitems: [item])
group.contentInsets = NSDirectionalEdgeInsets(top: 0, leading: 20, bottom: 0, trailing: 20)
let section = NSCollectionLayoutSection(group: group)
return section
}
return layout
}
Make sure your constraints and hugging and compression resistance in the cell have priority - 1000 (required), all of them: stackviews, labels etc.
You should get something like this:
Or if you want horizontal scrolling with pagination, just add:
section.orthogonalScrollingBehavior = .groupPaging
to your section property, on layout creation and you will get something like this:
Enjoy :)