如何在重新启动后重新绘制不可见的UICollectionViewCell?

时间:2015-10-21 12:23:30

标签: ios swift uicollectionview uicollectionviewcell prepareforreuse

如何在重用发生时重绘不可见的UICollectionViewCell?

我想到的一种方法是使用Layout Cell prepareForReuse函数中的代码,但是它的工作原理并非最佳,因为它会导致需要更多的重新绘制。

背景:在方向更改之后需要为单元格触发drawRect,这些方向更改当前不可见,但弹出使用且尚未重绘,所以到目前为止我只能看到{{1}是合适的。问题是我正在重新绘制所有“重用”单元格,而我真的只想重绘最初弹出的那些在设备的前一个方向位置创建的单元。

附加信息:所以目前我正在这样做:

在ViewController中:

prepareForReuse

在布局单元格类中:

override func viewWillLayoutSubviews() {
    // Clear cached layout attributes (to ensure new positions are calculated)
    (self.cal.collectionViewLayout as! GCCalendarLayout).resetCache()
    self.cal.collectionViewLayout.invalidateLayout()

    // Trigger cells to redraw themselves (to get new widths etc)
    for cell in self.cal?.visibleCells() as! [GCCalendarCell] {
        cell.setNeedsDisplay()
    }

    // Not sure how to "setNeedsDisplay" on non visible cells here?
}

没有上面的prepareForReuse中的代码会发生什么的示例。方向更改后拍摄快照,稍稍向上滚动后拍摄:

enter image description here

3 个答案:

答案 0 :(得分:4)

我想我现在在这里:

import UIKit

@IBDesignable class GCCalendarCell: UICollectionViewCell {
    var prevBounds : CGRect?

    override func layoutSubviews() {
        if let prevBounds = prevBounds {
            if !( (prevBounds.width == bounds.width) && (prevBounds.height == bounds.height) ) {
                self.setNeedsDisplay()
            }
        }
    }

    override func drawRect(rect: CGRect) {
        // Do Stuff
        self.prevBounds = self.bounds
    }

}

注意到此检查在“prepareForReuse”中不起作用,因为此时单元格未应用旋转。似乎可以在“layoutSubviews”中使用。

答案 1 :(得分:1)

您可以在单元格和持有集合视图的视图控制器(协议和委托或传递块或甚至直接引用VC)之间实现某种通信。然后您可以要求视图控制器进行旋转更改。

它有点乱,但如果您在视图控制器中有某种​​旋转跟踪您可以使用简单的if语句过滤setNeedsDisplay。

答案 2 :(得分:0)

我有类似的挑战更新已经显示并离开屏幕的单元格。虽然循环通过ALLL细胞可能是不可能的 - 刷新/循环通过不可见的细胞是。 如果这是您的用例 - 请继续阅读。预警 - 如果您要添加此类代码 - 请说明您执行此操作的原因。这是一种反模式 - 但可以帮助修复该错误并帮助发布您的应用程序,尽管增加了不必要的复杂性。不要在应用程序的多个位置使用它。

任何已取消初始化(在屏幕外并被重新录制)的集合视图单元都应自动取消订阅。

通知模式

let kUpdateButtonBarCell = NSNotification.Name("kUpdateButtonBarCell")

class Notificator {
     static func fireNotification(notificationName: NSNotification.Name) {
          NotificationCenter.default.post(name: notificationName, object: nil)
     }
}

extension UICollectionViewCell{
    func listenForBackgroundChanges(){
         NotificationCenter.default.removeObserver(self, name: kUpdateButtonBarCell, object: nil)
        NotificationCenter.default.addObserver(forName:kUpdateButtonBarCell, object: nil, queue: OperationQueue.main, using: { (note) in

            print( " contentView: ",self.contentView)

        })
    }
}



override func collectionView(collectionView: UICollectionView!, cellForItemAtIndexPath indexPath: NSIndexPath!) -> UICollectionViewCell! {
    let cell =  collectionView.dequeueReusableCellWithReuseIdentifier("die", forIndexPath: indexPath) as UICollectionViewCell
    cell.listenForBackgroundChanges()
    return cell
}


 // Where appropriate broadcast notification to hook into all cells past and present
 Notificator.fireNotification(notificationName: kUpdateButtonBarCell) 

委托模式

可以简化这个......为读者练习。只是不保留单元格(使用弱链接) - 否则你会有内存泄漏。