// Copyright (c) 2017 Cœur // // Permission is hereby granted, free of charge, to any person obtaining a copy of // this software and associated documentation files (the "Software"), to deal in // the Software without restriction, including without limitation the rights to // use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of // the Software, and to permit persons to whom the Software is furnished to do so, // subject to the following conditions: // // The above copyright notice and this permission notice shall be included in all // copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS // FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR // COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER // IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN // CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. import UIKit /** * Simple UICollectionViewFlowLayout that centers the cells rather than justify them * * Based on https://github.com/Coeur/UICollectionViewLeftAlignedLayout */ open class CollectionViewCenteredFlowLayout: UICollectionViewFlowLayout { open override func layoutAttributesForElements(in rect: CGRect) -> [UICollectionViewLayoutAttributes]? { guard let layoutAttributesForElements = super.layoutAttributesForElements(in: rect) else { return nil } guard let collectionView = collectionView else { return layoutAttributesForElements } // we group copies of the elements from the same row/column var representedElements: [UICollectionViewLayoutAttributes] = [] var cells: [[UICollectionViewLayoutAttributes]] = [[]] var previousFrame: CGRect? if scrollDirection == .vertical { for layoutAttributes in layoutAttributesForElements { guard layoutAttributes.representedElementKind == nil else { representedElements.append(layoutAttributes) continue } // copying is required to avoid "UICollectionViewFlowLayout cache mismatched frame" let currentItemAttributes = layoutAttributes.copy() as! UICollectionViewLayoutAttributes // if the current frame, once stretched to the full row doesn't intersect the previous frame then they are on different rows if previousFrame != nil && !currentItemAttributes.frame.intersects(CGRect(x: -.greatestFiniteMagnitude, y: previousFrame!.origin.y, width: .infinity, height: previousFrame!.size.height)) { cells.append([]) } cells[cells.endIndex - 1].append(currentItemAttributes) previousFrame = currentItemAttributes.frame } // we reposition all elements return representedElements + cells.flatMap { group -> [UICollectionViewLayoutAttributes] in guard let section = group.first?.indexPath.section else { return group } let evaluatedSectionInset = evaluatedSectionInsetForSection(at: section) let evaluatedMinimumInteritemSpacing = evaluatedMinimumInteritemSpacingForSection(at: section) let evaluatedCollectionView = (collectionView.bounds.width + evaluatedSectionInset.left - evaluatedSectionInset.right - group.reduce(0, { $0 + $1.frame.size.width }) - CGFloat(group.count - 1) * evaluatedMinimumInteritemSpacing) var origin = evaluatedCollectionView / 2 // we reposition each element of a group return group.map { $0.frame.origin.x = origin origin += $0.frame.size.width + evaluatedMinimumInteritemSpacing return $0 } } } else { for layoutAttributes in layoutAttributesForElements { guard layoutAttributes.representedElementKind == nil else { representedElements.append(layoutAttributes) continue } // copying is required to avoid "UICollectionViewFlowLayout cache mismatched frame" let currentItemAttributes = layoutAttributes.copy() as! UICollectionViewLayoutAttributes // if the current frame, once stretched to the full column doesn't intersect the previous frame then they are on different columns if previousFrame != nil && !currentItemAttributes.frame.intersects(CGRect(x: previousFrame!.origin.x, y: -.greatestFiniteMagnitude, width: previousFrame!.size.width, height: .infinity)) { cells.append([]) } cells[cells.endIndex - 1].append(currentItemAttributes) previousFrame = currentItemAttributes.frame } // we reposition all elements return representedElements + cells.flatMap { group -> [UICollectionViewLayoutAttributes] in guard let section = group.first?.indexPath.section else { return group } let evaluatedSectionInset = evaluatedSectionInsetForSection(at: section) let evaluatedMinimumInteritemSpacing = evaluatedMinimumInteritemSpacingForSection(at: section) let evaluatedCollectionView = (collectionView.bounds.height + evaluatedSectionInset.top - evaluatedSectionInset.bottom - group.reduce(0, { $0 + $1.frame.size.height }) - CGFloat(group.count - 1) * evaluatedMinimumInteritemSpacing) var origin = evaluatedCollectionView / 2 // we reposition each element of a group return group.map { $0.frame.origin.y = origin origin += $0.frame.size.height + evaluatedMinimumInteritemSpacing return $0 } } } } } extension UICollectionViewFlowLayout { internal func evaluatedSectionInsetForSection(at section: Int) -> UIEdgeInsets { (collectionView?.delegate as? UICollectionViewDelegateFlowLayout)?.collectionView?(collectionView!, layout: self, insetForSectionAt: section) ?? sectionInset } internal func evaluatedMinimumInteritemSpacingForSection(at section: Int) -> CGFloat { (collectionView?.delegate as? UICollectionViewDelegateFlowLayout)?.collectionView?(collectionView!, layout: self, minimumInteritemSpacingForSectionAt: section) ?? minimumInteritemSpacing } }