EPFCollectionViewLayout.swift 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. //
  2. // EPFCollectionViewLayout.swift
  3. // Product Calculator
  4. //
  5. // Created by Suraj Kumar Mandal on 09/12/21.
  6. //
  7. import UIKit
  8. class EPFCollectionViewLayout: UICollectionViewLayout {
  9. let numberOfColumns = 13
  10. var shouldPinFirstColumn = true
  11. var shouldPinFirstRow = true
  12. var itemAttributes = [[UICollectionViewLayoutAttributes]]()
  13. var itemsSize = [CGSize]()
  14. var contentSize: CGSize = .zero
  15. override func prepare() {
  16. guard let collectionView = collectionView else {
  17. return
  18. }
  19. if collectionView.numberOfSections == 0 {
  20. return
  21. }
  22. if itemAttributes.count != collectionView.numberOfSections {
  23. generateItemAttributes(collectionView: collectionView)
  24. return
  25. }
  26. for section in 0..<collectionView.numberOfSections {
  27. for item in 0..<collectionView.numberOfItems(inSection: section) {
  28. if section != 0 && item != 0 {
  29. continue
  30. }
  31. let attributes = layoutAttributesForItem(at: IndexPath(item: item, section: section))!
  32. if section == 0 {
  33. var frame = attributes.frame
  34. frame.origin.y = collectionView.contentOffset.y
  35. attributes.frame = frame
  36. }
  37. if item == 0 {
  38. var frame = attributes.frame
  39. frame.origin.x = collectionView.contentOffset.x
  40. attributes.frame = frame
  41. }
  42. }
  43. }
  44. }
  45. override var collectionViewContentSize: CGSize {
  46. return contentSize
  47. }
  48. override func layoutAttributesForItem(at indexPath: IndexPath) -> UICollectionViewLayoutAttributes? {
  49. return itemAttributes[indexPath.section][indexPath.row]
  50. }
  51. override func layoutAttributesForElements(in rect: CGRect) -> [UICollectionViewLayoutAttributes]? {
  52. var attributes = [UICollectionViewLayoutAttributes]()
  53. for section in itemAttributes {
  54. let filteredArray = section.filter { obj -> Bool in
  55. return rect.intersects(obj.frame)
  56. }
  57. attributes.append(contentsOf: filteredArray)
  58. }
  59. return attributes
  60. }
  61. override func shouldInvalidateLayout(forBoundsChange newBounds: CGRect) -> Bool {
  62. return true
  63. }
  64. }
  65. // MARK: - Helpers
  66. extension EPFCollectionViewLayout {
  67. func generateItemAttributes(collectionView: UICollectionView) {
  68. if itemsSize.count != numberOfColumns {
  69. calculateItemSizes()
  70. }
  71. var column = 0
  72. var xOffset: CGFloat = 0
  73. var yOffset: CGFloat = 0
  74. var contentWidth: CGFloat = 0
  75. itemAttributes = []
  76. for section in 0..<collectionView.numberOfSections {
  77. var sectionAttributes: [UICollectionViewLayoutAttributes] = []
  78. for index in 0..<numberOfColumns {
  79. let itemSize = itemsSize[index]
  80. let indexPath = IndexPath(item: index, section: section)
  81. let attributes = UICollectionViewLayoutAttributes(forCellWith: indexPath)
  82. attributes.frame = CGRect(x: xOffset, y: yOffset, width: itemSize.width, height: itemSize.height).integral
  83. if section == 0 && index == 0 {
  84. // First cell should be on top
  85. attributes.zIndex = 1024
  86. } else if section == 0 || index == 0 {
  87. // First row/column should be above other cells
  88. attributes.zIndex = 1023
  89. }
  90. if section == 0 {
  91. var frame = attributes.frame
  92. frame.origin.y = collectionView.contentOffset.y
  93. attributes.frame = frame
  94. }
  95. if index == 0 {
  96. var frame = attributes.frame
  97. frame.origin.x = collectionView.contentOffset.x
  98. attributes.frame = frame
  99. }
  100. sectionAttributes.append(attributes)
  101. xOffset += itemSize.width
  102. column += 1
  103. if column == numberOfColumns {
  104. if xOffset > contentWidth {
  105. contentWidth = xOffset
  106. }
  107. column = 0
  108. xOffset = 0
  109. yOffset += itemSize.height
  110. }
  111. }
  112. itemAttributes.append(sectionAttributes)
  113. }
  114. if let attributes = itemAttributes.last?.last {
  115. contentSize = CGSize(width: contentWidth, height: attributes.frame.maxY)
  116. }
  117. }
  118. func calculateItemSizes() {
  119. itemsSize = []
  120. for index in 0..<numberOfColumns {
  121. itemsSize.append(sizeForItemWithColumnIndex(index))
  122. }
  123. }
  124. func sizeForItemWithColumnIndex(_ columnIndex: Int) -> CGSize {
  125. var text: NSString
  126. switch columnIndex {
  127. case 0:
  128. text = "MMM-99"
  129. default:
  130. text = "Content"
  131. }
  132. let size: CGSize = text.size(withAttributes: [NSAttributedString.Key.font: UIFont.systemFont(ofSize: 14.0)])
  133. let width: CGFloat = size.width + 16
  134. return CGSize(width: width, height: 30)
  135. }
  136. }