Reusable layout with alignment. Could be extended with plugins.
- When creating collection layout use
PluginableFlowLayout
- Setup base
UICollectionViewFlowLayout
with your needs- properly setup
scrollDirection
- other settings
- properly setup
- If needed custom alignment, modify
UICollectionView
- turn offisPagingEnabled
(default value isfalse
) and setdecelerationRate
to.fast
.
Layout descendant of UICollectionViewFlowLayout
.
Has feature for alignment:
lineStart
- left alignmentlineCenter
- center alignmentlineEnd
- right alignment
If pass nil
- there will not be any modifications for targetContentOffset
(you can use default isPagingEnabled = true
property for items with equal sizes).
Supports items with different sizes.
Protocol for custom plugins
Realization of FlowLayoutPlugin
protocol for adding visibility percentage of items.
Each percentage value contains number in range [-1; 1] where
- value
-1
means that cell is in before position (left position respectfull to center cell) - value
0
means that cell is in center position (in case of alignment not in center -PercentageFlowLayoutPlugin
respect alignment fromPluginableFlowLayout
) - value
+1
means that cell is in after position (right position respectfull to center cell)
Percentage info contains 3 values:
collectionVisibility
- visibility factor respectfull to bound of collectionView - intersection frame of cell in collectionViewidealFrameVisibility
- visibility factor respectfull to "ideal frame" (respecfull to alignmentPluginableFlowLayout
) - intersection frame of cell and "ideal frame"idealFrameAndCollectionVisibility
- mixed visibility factor
Note: if cell has width of collectionView and layout has horizontal scrollDirection (height/vertical) - all values will be equal.
- pass
PercentageFlowLayoutPlugin
in constructor ofPluginableFlowLayout
let collectionView = UICollectionView(
frame: .zero,
collectionViewLayout: PluginableFlowLayout(
alignment: .lineStart,
plugins: [PercentageFlowLayoutPlugin()]
)
)
- override method
apply(_ layoutAttributes: UICollectionViewLayoutAttributes)
and use custom attribute values
override func apply(_ layoutAttributes: UICollectionViewLayoutAttributes) {
// convert `UICollectionViewLayoutAttributes` to `PluginableFlowLayoutAttributes`
guard let pluginableAttributes = layoutAttributes as? PluginableFlowLayoutAttributes else {
return super.apply(layoutAttributes)
}
// get custom percentage attribute value
let percentage = pluginableAttributes.percentageInfo.collectionVisibility ?? 0
// set UIImageView translation
let translationX = -percentage / 2 * imageView.bounds.width
imageView.transform = .init(
translationX: translationX,
y: 0
)
// set alpha for dimm view
dimmView.alpha = 1 - abs(percentage * 2)
}
To add custom plugin, you need to realize protocol FlowLayoutPlugin
and return modified PluginableFlowLayoutAttributes
.
Kind of sttributes, that store values in dictionary. You can write extension to these class and realize setter/getter for values (like in file PercentageFlowLayoutPlugin.swift
).