// UICollectionViewExtensions.swift - Copyright 2024 SwifterSwift
|
|
#if canImport(UIKit) && !os(watchOS)
|
import UIKit
|
|
// MARK: - Properties
|
|
public extension UICollectionView {
|
/// SwifterSwift: Index path of last item in collectionView.
|
var indexPathForLastItem: IndexPath? {
|
return indexPathForLastItem(inSection: lastSection)
|
}
|
|
/// SwifterSwift: Index of last section in collectionView.
|
var lastSection: Int {
|
return numberOfSections > 0 ? numberOfSections - 1 : 0
|
}
|
}
|
|
// MARK: - Methods
|
|
public extension UICollectionView {
|
/// SwifterSwift: Number of all items in all sections of collectionView.
|
///
|
/// - Returns: The count of all rows in the collectionView.
|
func numberOfItems() -> Int {
|
var section = 0
|
var itemsCount = 0
|
while section < numberOfSections {
|
itemsCount += numberOfItems(inSection: section)
|
section += 1
|
}
|
return itemsCount
|
}
|
|
/// SwifterSwift: IndexPath for last item in section.
|
///
|
/// - Parameter section: section to get last item in.
|
/// - Returns: optional last indexPath for last item in section (if applicable).
|
func indexPathForLastItem(inSection section: Int) -> IndexPath? {
|
guard section >= 0 else {
|
return nil
|
}
|
guard section < numberOfSections else {
|
return nil
|
}
|
guard numberOfItems(inSection: section) > 0 else {
|
return IndexPath(item: 0, section: section)
|
}
|
return IndexPath(item: numberOfItems(inSection: section) - 1, section: section)
|
}
|
|
/// SwifterSwift: Reload data with a completion handler.
|
///
|
/// - Parameter completion: completion handler to run after reloadData finishes.
|
func reloadData(_ completion: @escaping () -> Void) {
|
UIView.animate(withDuration: 0, animations: {
|
self.reloadData()
|
}, completion: { _ in
|
completion()
|
})
|
}
|
|
/// SwifterSwift: Dequeue reusable UICollectionViewCell using class name.
|
///
|
/// - Parameters:
|
/// - name: UICollectionViewCell type.
|
/// - indexPath: location of cell in collectionView.
|
/// - Returns: UICollectionViewCell object with associated class name.
|
func dequeueReusableCell<T: UICollectionViewCell>(withClass name: T.Type, for indexPath: IndexPath) -> T {
|
guard let cell = dequeueReusableCell(withReuseIdentifier: String(describing: name), for: indexPath) as? T else {
|
fatalError(
|
"Couldn't find UICollectionViewCell for \(String(describing: name)), make sure the cell is registered with collection view")
|
}
|
return cell
|
}
|
|
/// SwifterSwift: Dequeue reusable UICollectionReusableView using class name.
|
///
|
/// - Parameters:
|
/// - kind: the kind of supplementary view to retrieve. This value is defined by the layout object.
|
/// - name: UICollectionReusableView type.
|
/// - indexPath: location of cell in collectionView.
|
/// - Returns: UICollectionReusableView object with associated class name.
|
func dequeueReusableSupplementaryView<T: UICollectionReusableView>(ofKind kind: String, withClass name: T.Type,
|
for indexPath: IndexPath) -> T {
|
guard let cell = dequeueReusableSupplementaryView(
|
ofKind: kind,
|
withReuseIdentifier: String(describing: name),
|
for: indexPath) as? T else {
|
fatalError(
|
"Couldn't find UICollectionReusableView for \(String(describing: name)), make sure the view is registered with collection view")
|
}
|
return cell
|
}
|
|
/// SwifterSwift: Register UICollectionReusableView using class name.
|
///
|
/// - Parameters:
|
/// - kind: the kind of supplementary view to retrieve. This value is defined by the layout object.
|
/// - name: UICollectionReusableView type.
|
func register<T: UICollectionReusableView>(supplementaryViewOfKind kind: String, withClass name: T.Type) {
|
register(T.self, forSupplementaryViewOfKind: kind, withReuseIdentifier: String(describing: name))
|
}
|
|
/// SwifterSwift: Register UICollectionViewCell using class name.
|
///
|
/// - Parameters:
|
/// - nib: Nib file used to create the collectionView cell.
|
/// - name: UICollectionViewCell type.
|
func register(nib: UINib?, forCellWithClass name: (some UICollectionViewCell).Type) {
|
register(nib, forCellWithReuseIdentifier: String(describing: name))
|
}
|
|
/// SwifterSwift: Register UICollectionViewCell using class name.
|
///
|
/// - Parameter name: UICollectionViewCell type.
|
func register<T: UICollectionViewCell>(cellWithClass name: T.Type) {
|
register(T.self, forCellWithReuseIdentifier: String(describing: name))
|
}
|
|
/// SwifterSwift: Register UICollectionReusableView using class name.
|
///
|
/// - Parameters:
|
/// - nib: Nib file used to create the reusable view.
|
/// - kind: the kind of supplementary view to retrieve. This value is defined by the layout object.
|
/// - name: UICollectionReusableView type.
|
func register(nib: UINib?, forSupplementaryViewOfKind kind: String,
|
withClass name: (some UICollectionReusableView).Type) {
|
register(nib, forSupplementaryViewOfKind: kind, withReuseIdentifier: String(describing: name))
|
}
|
|
/// SwifterSwift: Register UICollectionViewCell with .xib file using only its corresponding class.
|
/// Assumes that the .xib filename and cell class has the same name.
|
///
|
/// - Parameters:
|
/// - name: UICollectionViewCell type.
|
/// - bundleClass: Class in which the Bundle instance will be based on.
|
func register(nibWithCellClass name: (some UICollectionViewCell).Type, at bundleClass: AnyClass? = nil) {
|
let identifier = String(describing: name)
|
var bundle: Bundle?
|
|
if let bundleName = bundleClass {
|
bundle = Bundle(for: bundleName)
|
}
|
|
register(UINib(nibName: identifier, bundle: bundle), forCellWithReuseIdentifier: identifier)
|
}
|
|
/// SwifterSwift: Safely scroll to possibly invalid IndexPath.
|
///
|
/// - Parameters:
|
/// - indexPath: Target IndexPath to scroll to.
|
/// - scrollPosition: Scroll position.
|
/// - animated: Whether to animate or not.
|
func safeScrollToItem(at indexPath: IndexPath, at scrollPosition: UICollectionView.ScrollPosition, animated: Bool) {
|
guard indexPath.item >= 0,
|
indexPath.section >= 0,
|
indexPath.section < numberOfSections,
|
indexPath.item < numberOfItems(inSection: indexPath.section) else {
|
return
|
}
|
scrollToItem(at: indexPath, at: scrollPosition, animated: animated)
|
}
|
|
/// SwifterSwift: Check whether IndexPath is valid within the CollectionView.
|
///
|
/// - Parameter indexPath: An IndexPath to check.
|
/// - Returns: Boolean value for valid or invalid IndexPath.
|
func isValidIndexPath(_ indexPath: IndexPath) -> Bool {
|
return indexPath.section >= 0 &&
|
indexPath.item >= 0 &&
|
indexPath.section < numberOfSections &&
|
indexPath.item < numberOfItems(inSection: indexPath.section)
|
}
|
}
|
|
#endif
|