123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175 |
- //
- // CustomCollectionViewLayout.swift
- // Product Calculator
- //
- // Created by Suraj Kumar Mandal on 09/12/21.
- //
- import UIKit
- class CustomCollectionViewLayout: UICollectionViewLayout {
- let numberOfColumns = 5
- var shouldPinFirstColumn = true
- var shouldPinFirstRow = true
- var itemAttributes = [[UICollectionViewLayoutAttributes]]()
- var itemsSize = [CGSize]()
- var contentSize: CGSize = .zero
- override func prepare() {
- guard let collectionView = collectionView else {
- return
- }
- if collectionView.numberOfSections == 0 {
- return
- }
- if itemAttributes.count != collectionView.numberOfSections {
- generateItemAttributes(collectionView: collectionView)
- return
- }
- for section in 0..<collectionView.numberOfSections {
- for item in 0..<collectionView.numberOfItems(inSection: section) {
- if section != 0 && item != 0 {
- continue
- }
- let attributes = layoutAttributesForItem(at: IndexPath(item: item, section: section))!
- if section == 0 {
- var frame = attributes.frame
- frame.origin.y = collectionView.contentOffset.y
- attributes.frame = frame
- }
- if item == 0 {
- var frame = attributes.frame
- frame.origin.x = collectionView.contentOffset.x
- attributes.frame = frame
- }
- }
- }
- }
- override var collectionViewContentSize: CGSize {
- return contentSize
- }
- override func layoutAttributesForItem(at indexPath: IndexPath) -> UICollectionViewLayoutAttributes? {
- return itemAttributes[indexPath.section][indexPath.row]
- }
- override func layoutAttributesForElements(in rect: CGRect) -> [UICollectionViewLayoutAttributes]? {
- var attributes = [UICollectionViewLayoutAttributes]()
- for section in itemAttributes {
- let filteredArray = section.filter { obj -> Bool in
- return rect.intersects(obj.frame)
- }
- attributes.append(contentsOf: filteredArray)
- }
- return attributes
- }
- override func shouldInvalidateLayout(forBoundsChange newBounds: CGRect) -> Bool {
- return true
- }
- }
- // MARK: - Helpers
- extension CustomCollectionViewLayout {
- func generateItemAttributes(collectionView: UICollectionView) {
- if itemsSize.count != numberOfColumns {
- calculateItemSizes()
- }
- var column = 0
- var xOffset: CGFloat = 0
- var yOffset: CGFloat = 0
- var contentWidth: CGFloat = 0
- itemAttributes = []
- for section in 0..<collectionView.numberOfSections {
- var sectionAttributes: [UICollectionViewLayoutAttributes] = []
- for index in 0..<numberOfColumns {
- let itemSize = itemsSize[index]
- let indexPath = IndexPath(item: index, section: section)
- let attributes = UICollectionViewLayoutAttributes(forCellWith: indexPath)
- attributes.frame = CGRect(x: xOffset, y: yOffset, width: itemSize.width, height: itemSize.height).integral
- if section == 0 && index == 0 {
- // First cell should be on top
- attributes.zIndex = 1024
- } else if section == 0 || index == 0 {
- // First row/column should be above other cells
- attributes.zIndex = 1023
- }
- if section == 0 {
- var frame = attributes.frame
- frame.origin.y = collectionView.contentOffset.y
- attributes.frame = frame
- }
- if index == 0 {
- var frame = attributes.frame
- frame.origin.x = collectionView.contentOffset.x
- attributes.frame = frame
- }
- sectionAttributes.append(attributes)
- xOffset += itemSize.width
- column += 1
- if column == numberOfColumns {
- if xOffset > contentWidth {
- contentWidth = xOffset
- }
- column = 0
- xOffset = 0
- yOffset += itemSize.height
- }
- }
- itemAttributes.append(sectionAttributes)
- }
- if let attributes = itemAttributes.last?.last {
- contentSize = CGSize(width: contentWidth, height: attributes.frame.maxY)
- }
- }
- func calculateItemSizes() {
- itemsSize = []
- for index in 0..<numberOfColumns {
- itemsSize.append(sizeForItemWithColumnIndex(index))
- }
- }
- func sizeForItemWithColumnIndex(_ columnIndex: Int) -> CGSize {
- var text: NSString
- switch columnIndex {
- case 0:
- text = "MMM-99"
- default:
- text = "Content"
- }
- let size: CGSize = text.size(withAttributes: [NSAttributedString.Key.font: UIFont.systemFont(ofSize: 14.0)])
- let width: CGFloat = size.width + 16
- return CGSize(width: 150, height: 30)
- }
- }
|