Files
Woka_Native_iOS/WOKA/CollectionViewCenteredFlowLayout.swift

113 lines
6.6 KiB
Swift
Raw Normal View History

// 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
}
}