ios – 如何使UICollectionView的中心单元与位于侧面的其他两个单元重叠?

前端之家收集整理的这篇文章主要介绍了ios – 如何使UICollectionView的中心单元与位于侧面的其他两个单元重叠?前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
我正在制作一个集合视图来生成轮播效果.

我需要中心单元格与左侧和右侧的其他两个单元格重叠.中心单元需要始终位于顶部.

但是,当我尝试将侧面的细胞与中心的细胞重叠时,它不起作用.相反,右侧单元格(蓝色单元格)与中心单元格(黑色单元格)重叠,如下图所示.

使用的代码是针对此效果如下:

以下是集合视图的视图控制器.

  1. import UIKit
  2.  
  3. private let reuseIdentifier = "Cell"
  4. let kRoomCellScaling: CGFloat = 0.6
  5.  
  6. class RoomsViewController: UICollectionViewController {
  7.  
  8. override func viewDidLoad() {
  9. super.viewDidLoad()
  10.  
  11. // This method sets up the collection view
  12. let layout = UPCarouselFlowLayout()
  13. layout.itemSize = CGSizeMake(250,250)
  14. layout.scrollDirection = .Horizontal
  15.  
  16. layout.sideItemAlpha = 1
  17. layout.sideItemScale = 0.8
  18. layout.spacingMode = UPCarouselFlowLayoutSpacingMode.overlap(visibleOffset: 60)
  19.  
  20. collectionView?.setCollectionViewLayout(layout,animated: false)
  21.  
  22. }
  23.  
  24. override func didReceiveMemoryWarning() {
  25. super.didReceiveMemoryWarning()
  26. // Dispose of any resources that can be recreated.
  27. }
  28.  
  29.  
  30.  
  31. // MARK: UICollectionViewDataSource
  32.  
  33. override func numberOfSectionsInCollectionView(collectionView: UICollectionView) -> Int {
  34. // #warning Incomplete implementation,return the number of sections
  35. return 1
  36. }
  37.  
  38.  
  39. override func collectionView(collectionView: UICollectionView,numberOfItemsInSection section: Int) -> Int {
  40. // #warning Incomplete implementation,return the number of items
  41. return 3
  42. }
  43.  
  44. override func collectionView(collectionView: UICollectionView,cellForItemAtIndexPath indexPath: NSIndexPath) -> UICollectionViewCell {
  45. let cell = collectionView.dequeueReusableCellWithReuseIdentifier(reuseIdentifier,forIndexPath: indexPath)
  46.  
  47.  
  48.  
  49. // Configure the cell
  50. switch indexPath.row%3 {
  51.  
  52. case 0:
  53. cell.backgroundColor = UIColor.redColor()
  54. case 1:
  55. cell.backgroundColor = UIColor.blackColor()
  56. case 2:
  57. cell.backgroundColor = UIColor.blueColor()
  58.  
  59. default:
  60. break
  61.  
  62. }
  63.  
  64. return cell
  65. }
  66.  
  67.  
  68. }

以下是用于集合视图的流程布局.

  1. import UIKit
  2.  
  3.  
  4. public enum UPCarouselFlowLayoutSpacingMode {
  5. case fixed(spacing: CGFloat)
  6. case overlap(visibleOffset: CGFloat)
  7. }
  8.  
  9. public class UPCarouselFlowLayout: UICollectionViewFlowLayout {
  10.  
  11. private struct LayoutState {
  12. var size: CGSize
  13. var direction: UICollectionViewScrollDirection
  14. func isEqual(otherState: LayoutState) -> Bool {
  15. return CGSizeEqualToSize(self.size,otherState.size) && self.direction == otherState.direction
  16. }
  17. }
  18.  
  19. @IBInspectable public var sideItemScale: CGFloat = 0.6
  20. @IBInspectable public var sideItemAlpha: CGFloat = 0.6
  21. public var spacingMode = UPCarouselFlowLayoutSpacingMode.fixed(spacing: 40)
  22.  
  23. private var state = LayoutState(size: CGSizeZero,direction: .Horizontal)
  24.  
  25.  
  26. override public func prepareLayout() {
  27. super.prepareLayout()
  28.  
  29. let currentState = LayoutState(size: self.collectionView!.bounds.size,direction: self.scrollDirection)
  30.  
  31. if !self.state.isEqual(currentState) {
  32. self.setupCollectionView()
  33. self.updateLayout()
  34. self.state = currentState
  35. }
  36. }
  37.  
  38. private func setupCollectionView() {
  39. guard let collectionView = self.collectionView else { return }
  40. if collectionView.decelerationRate != UIScrollViewDecelerationRateFast {
  41. collectionView.decelerationRate = UIScrollViewDecelerationRateFast
  42. }
  43. }
  44.  
  45. private func updateLayout() {
  46. guard let collectionView = self.collectionView else { return }
  47.  
  48. let collectionSize = collectionView.bounds.size
  49. let isHorizontal = (self.scrollDirection == .Horizontal)
  50.  
  51. let yInset = (collectionSize.height - self.itemSize.height) / 2
  52. let xInset = (collectionSize.width - self.itemSize.width) / 2
  53. self.sectionInset = UIEdgeInsetsMake(yInset,xInset,yInset,xInset)
  54.  
  55. let side = isHorizontal ? self.itemSize.width : self.itemSize.height
  56. let scaledItemOffset = (side - side*self.sideItemScale) / 2
  57. switch self.spacingMode {
  58. case .fixed(let spacing):
  59. self.minimumLineSpacing = spacing - scaledItemOffset
  60. case .overlap(let visibleOffset):
  61. let fullSizeSideItemOverlap = visibleOffset + scaledItemOffset
  62. let inset = isHorizontal ? xInset : yInset
  63. self.minimumLineSpacing = inset - fullSizeSideItemOverlap
  64. }
  65. }
  66.  
  67. override public func shouldInvalidateLayoutForBoundsChange(newBounds: CGRect) -> Bool {
  68. return true
  69. }
  70.  
  71. override public func layoutAttributesForElementsInRect(rect: CGRect) -> [UICollectionViewLayoutAttributes]? {
  72. guard let superAttributes = super.layoutAttributesForElementsInRect(rect),let attributes = NSArray(array: superAttributes,copyItems: true) as? [UICollectionViewLayoutAttributes]
  73. else { return nil }
  74. return attributes.map({ self.transformLayoutAttributes($0) })
  75. }
  76.  
  77. private func transformLayoutAttributes(attributes: UICollectionViewLayoutAttributes) -> UICollectionViewLayoutAttributes {
  78. guard let collectionView = self.collectionView else { return attributes }
  79. let isHorizontal = (self.scrollDirection == .Horizontal)
  80.  
  81. let collectionCenter = isHorizontal ? collectionView.frame.size.width/2 : collectionView.frame.size.height/2
  82. let offset = isHorizontal ? collectionView.contentOffset.x : collectionView.contentOffset.y
  83. let normalizedCenter = (isHorizontal ? attributes.center.x : attributes.center.y) - offset
  84.  
  85. let maxDistance = (isHorizontal ? self.itemSize.width : self.itemSize.height) + self.minimumLineSpacing
  86. let distance = min(abs(collectionCenter - normalizedCenter),maxDistance)
  87. let ratio = (maxDistance - distance)/maxDistance
  88.  
  89. let alpha = ratio * (1 - self.sideItemAlpha) + self.sideItemAlpha
  90. let scale = ratio * (1 - self.sideItemScale) + self.sideItemScale
  91. attributes.alpha = alpha
  92. attributes.transform3D = CATransform3DScale(CATransform3DIdentity,scale,1)
  93.  
  94. return attributes
  95. }
  96.  
  97. override public func targetContentOffsetForProposedContentOffset(proposedContentOffset: CGPoint,withScrollingVelocity velocity: CGPoint) -> CGPoint {
  98. guard let collectionView = collectionView where !collectionView.pagingEnabled,let layoutAttributes = self.layoutAttributesForElementsInRect(collectionView.bounds)
  99. else { return super.targetContentOffsetForProposedContentOffset(proposedContentOffset) }
  100.  
  101. let isHorizontal = (self.scrollDirection == .Horizontal)
  102.  
  103. let midSide = (isHorizontal ? collectionView.bounds.size.width : collectionView.bounds.size.height) / 2
  104. let proposedContentOffsetCenterOrigin = (isHorizontal ? proposedContentOffset.x : proposedContentOffset.y) + midSide
  105.  
  106. var targetContentOffset: CGPoint
  107. if isHorizontal {
  108. let closest = layoutAttributes.sort { abs($0.center.x - proposedContentOffsetCenterOrigin) < abs($1.center.x - proposedContentOffsetCenterOrigin) }.first ?? UICollectionViewLayoutAttributes()
  109. targetContentOffset = CGPoint(x: floor(closest.center.x - midSide),y: proposedContentOffset.y)
  110. }
  111. else {
  112. let closest = layoutAttributes.sort { abs($0.center.y - proposedContentOffsetCenterOrigin) < abs($1.center.y - proposedContentOffsetCenterOrigin) }.first ?? UICollectionViewLayoutAttributes()
  113. targetContentOffset = CGPoint(x: proposedContentOffset.x,y: floor(closest.center.y - midSide))
  114. }
  115.  
  116. return targetContentOffset
  117. }
  118. }

那么,我怎样才能使中心单元始终与其他两个单元格重叠?

解决方法

您可以根据距离中心的距离将项目转换为较小的负z值.

替换此行:

  1. attributes.transform3D = CATransform3DScale(CATransform3DIdentity,1)

  1. let visibleRect = CGRect(origin: self.collectionView!.contentOffset,size: self.collectionView!.bounds.size)
  2. let dist = CGRectGetMidX(attributes.frame) - CGRectGetMidX(visibleRect)
  3. var transform = CATransform3DScale(CATransform3DIdentity,1)
  4. transform = CATransform3DTranslate(transform,-abs(dist/1000))
  5. attributes.transform3D = transform

或者您可以根据与中心的距离围绕y轴旋转项目,并为transform.m34提供一个小的负值,这样它就会有透视和更真实的外观.

替换此行:

  1. attributes.transform3D = CATransform3DScale(CATransform3DIdentity,size: self.collectionView!.bounds.size)
  2. let dist = CGRectGetMidX(attributes.frame) - CGRectGetMidX(visibleRect)
  3. let currentAngle = dist / (CGRectGetWidth(visibleRect)/2)
  4. var transform = CATransform3DScale(CATransform3DIdentity,1)
  5. transform.m34 = -1.0 / 1000
  6. transform = CATransform3DRotate(transform,-currentAngle,1,0)
  7. attributes.transform3D = transform

猜你在找的iOS相关文章