Получить динамическую высоту для UICollectionViewCell с пользовательским макетом - PullRequest
0 голосов
/ 19 марта 2019

Мне нужно создать UICollectionView с переменной высотой ячейки, но с фиксированным вертикальным зазором между ячейками. Что-то вроде это

Я нашел много примеров, учебных пособий и библиотек, но все они должны реализовывать метод делегата, обеспечивающий высоту для каждого элемента. В моих ячейках есть несколько меток, которые могут иметь одну или несколько строк, поэтому я не знаю заранее конечную высоту элемента после того, как ячейка разметит свои виды.

Есть ли способ добиться того, что мне нужно?

1 Ответ

2 голосов
/ 19 марта 2019

Вы можете рассчитать высоту ячейки с помощью функции systemLayoutSizeFitting.

https://developer.apple.com/documentation/uikit/uiview/1622623-systemlayoutsizefitting

Например: (UICollectionViewCell является подклассом UICollectionReusableView, поэтому вы можете использовать его дляячейка и верхний колонтитул тоже)

public extension UICollectionReusableView {
    static func autoResizingView<T: UICollectionReusableView>(type: T.Type) -> T {
        let nibViews = Bundle.main.loadNibNamed(T.identifier, owner: nil, options: nil)
        return nibViews?.first as? T ?? T()
    }

    static func autoLayoutSize<T: UICollectionReusableView>(type: T.Type, targetWidth: CGFloat, configure: ((T) -> Void)?) -> CGSize {
        let resizingView = UICollectionReusableView.autoResizingView(type: type)
        resizingView.prepareForReuse()
        configure?(resizingView)
        resizingView.setNeedsLayout()
        resizingView.layoutIfNeeded()

        let targetSize = CGSize(width: targetWidth, height: 0)
        let calculateView: UIView
        if let contentView = (resizingView as? UICollectionViewCell)?.contentView {
            calculateView = contentView
        } else {
            calculateView = resizingView
        }
        // Calculate the size (height) using Auto Layout
        let autoLayoutSize = calculateView.systemLayoutSizeFitting(
            targetSize,
            withHorizontalFittingPriority: .required,
            verticalFittingPriority: .defaultLow)
        return autoLayoutSize
    }
}
...