我设置了UICollectionView
,其设置如下:
它非常像UITableView
,在大多数情况下都可以正常工作,但动画不适用的情况除外。
在collectionView
中的堆叠单元格中,其中一个上部单元格会扩展其高度。然后下部单元必须向下移动以保持距离。如果此移动单元格的目标帧超出collectionView's
边界,则不应用任何动画,并且单元格将消失。
相反的案件以同样的方式运作;如果下部单元格的源帧超出屏幕边界(当前在边界之外)并且上部单元格应该缩小,则不应用动画,它只会出现在目标帧上。
这在由UICollectionView
控制的内存管理逻辑中似乎是合适的,但同时向用户显示某些内容只是出现或消失的蓝色并不自然。我用UITableView对此进行了测试,同样的事情发生了。
此问题是否有解决方法?
答案 0 :(得分:5)
您应该添加一些代码或至少是您的UI问题的gif。
我尝试使用基本的UICollectionViewLayout
子类来复制您的问题:
protocol CollectionViewLayoutDelegate: AnyObject {
func heightForItem(at indexPath: IndexPath) -> CGFloat
}
class CollectionViewLayout: UICollectionViewLayout {
weak var delegate: CollectionViewLayoutDelegate?
private var itemAttributes: [UICollectionViewLayoutAttributes] = []
override func prepare() {
super.prepare()
itemAttributes = generateItemAttributes()
}
override func targetContentOffset(forProposedContentOffset proposedContentOffset: CGPoint) -> CGPoint {
return collectionView?.contentOffset ?? .zero
return super.targetContentOffset(forProposedContentOffset: proposedContentOffset)
}
override func layoutAttributesForItem(at indexPath: IndexPath) -> UICollectionViewLayoutAttributes? {
return itemAttributes.first { $0.indexPath == indexPath }
}
override func layoutAttributesForElements(in rect: CGRect) -> [UICollectionViewLayoutAttributes]? {
return itemAttributes.filter { $0.frame.intersects(rect) }
}
private func generateItemAttributes() -> [UICollectionViewLayoutAttributes] {
var offset: CGFloat = 0
return (0..<numberOfItems()).map { index in
let indexPath = IndexPath(item: index, section: 0)
let frame = CGRect(
x: 0,
y: offset,
width: collectionView?.bounds.width ?? 0,
height: delegate?.heightForItem(at: indexPath) ?? 0
)
offset = frame.maxY
let attributes = UICollectionViewLayoutAttributes(forCellWith: indexPath)
attributes.frame = frame
return attributes
}
}
}
在简单的UIViewController
中,每次选择一个单元格时,我都会重新加载第一个单元格:
func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) {
updatedIndexPath = IndexPath(item: 0, section: 0)
collectionView.reloadItems(at: [updatedIndexPath])
}
在这种情况下,我遇到了这样的动画:
我认为您可以尝试调整super.finalLayoutAttributesForDisappearingItem(at: itemIndexPath)
返回的属性,以计算其正确的帧并使用z-index。
但是您也可以像这样简单地尝试使所有布局无效:
func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) {
layout = CollectionViewLayout()
layout.delegate = self
collectionView.setCollectionViewLayout(layout, animated: true)
}
并覆盖:
override func targetContentOffset(forProposedContentOffset proposedContentOffset: CGPoint) -> CGPoint {
return collectionView?.contentOffset ?? .zero
}
避免在布局无效时避免错误的目标内容偏移量计算。