Swift: How to refresh UICollectionView layout after rotation of the device

后端 未结 14 867
闹比i
闹比i 2020-12-01 02:38

I used UICollectionView (flowlayout) to build a simple layout. the width for each cell is set to the width of screen using self.view.frame.width

but whe

相关标签:
14条回答
  • 2020-12-01 02:55

    I solve this by setting notification when screen orientation changes and reloading cell which set itemsize according to screen orientation and setting indexpath to previous cell. This does work with flowlayout too. Here is the code i wrote:

    var cellWidthInLandscape: CGFloat = 0 {
        didSet {
            self.collectionView.reloadData()
        }
    }
    
    var lastIndex: Int = 0
    
    override func viewDidLoad() {
        super.viewDidLoad()
    
        collectionView.dataSource = self
        collectionView.delegate = self
        NotificationCenter.default.addObserver(self, selector: #selector(rotated), name: NSNotification.Name.UIDeviceOrientationDidChange, object: nil)
        cellWidthInLandscape = UIScreen.main.bounds.size.width
    
    }
    deinit {
        NotificationCenter.default.removeObserver(self)
    }
    @objc func rotated() {
    
            // Setting new width on screen orientation change
            cellWidthInLandscape = UIScreen.main.bounds.size.width
    
           // Setting collectionView to previous indexpath
            collectionView.scrollToItem(at: IndexPath(item: lastIndex, section: 0), at: .right, animated: false)
    }
        override func viewWillTransition(to size: CGSize, with coordinator: UIViewControllerTransitionCoordinator) {
            NotificationCenter.default.addObserver(self, selector: #selector(rotated), name: NSNotification.Name.UIDeviceOrientationDidChange, object: nil)
    
    }
    
    func scrollViewDidEndDecelerating(_ scrollView: UIScrollView) {
    
       // Getting last contentOffset to calculate last index of collectionViewCell
        lastIndex = Int(scrollView.contentOffset.x / collectionView.bounds.width)
    }
    
    func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, sizeForItemAt indexPath: IndexPath) -> CGSize {
            // Setting new width of collectionView Cell
            return CGSize(width: cellWidthInLandscape, height: collectionView.bounds.size.height)
    
    }
    
    0 讨论(0)
  • 2020-12-01 03:00

    Calling viewWillLayoutSubviews is not optimal. Try calling the invalidateLayout() method first.

    If you experience the The behaviour of the UICollectionViewFlowLayout is not defined error, you need to verify if all the elements within your view's have changed its sizes, according to a new layout. (see the optional steps within the example code)

    Here is the code, to get you started. Depending on the way your UI is created, you may have to experiment to find the right view to call the recalculate method, yet that should guide you towards your first steps.

    override func viewWillTransition(to size: CGSize, with coordinator: UIViewControllerTransitionCoordinator) {
    
        super.viewWillTransition(to: size, with: coordinator)
    
        /// (Optional) Additional step 1. Depending on your layout, you may have to manually indicate that the content size of a visible cells has changed
        /// Use that step if you experience the `the behavior of the UICollectionViewFlowLayout is not defined` errors.
    
        collectionView.visibleCells.forEach { cell in
            guard let cell = cell as? CustomCell else {
                print("`viewWillTransition` failed. Wrong cell type")
                return
            }
    
            cell.recalculateFrame(newSize: size)
    
        }
    
        /// (Optional) Additional step 2. Recalculate layout if you've explicitly set the estimatedCellSize and you'll notice that layout changes aren't automatically visible after the #3
    
        (collectionView.collectionViewLayout as? CustomLayout)?.recalculateLayout(size: size)
    
    
        /// Step 3 (or 1 if none of the above is applicable)
    
        coordinator.animate(alongsideTransition: { context in
            self.collectionView.collectionViewLayout.invalidateLayout()
        }) { _ in
            // code to execute when the transition's finished.
        }
    
    }
    
    /// Example implementations of the `recalculateFrame` and `recalculateLayout` methods:
    
        /// Within the `CustomCell` class:
        func recalculateFrame(newSize: CGSize) {
            self.frame = CGRect(x: self.bounds.origin.x,
                                y: self.bounds.origin.y,
                                width: newSize.width - 14.0,
                                height: self.frame.size.height)
        }
    
        /// Within the `CustomLayout` class:
        func recalculateLayout(size: CGSize? = nil) {
            estimatedItemSize = CGSize(width: size.width - 14.0, height: 100)
        }
    
        /// IMPORTANT: Within the `CustomLayout` class.
        override func shouldInvalidateLayout(forBoundsChange newBounds: CGRect) -> Bool {
    
            guard let collectionView = collectionView else {
                return super.shouldInvalidateLayout(forBoundsChange: newBounds)
            }
    
            if collectionView.bounds.width != newBounds.width || collectionView.bounds.height != newBounds.height {
                return true
            } else {
                return false
            }
        }
    
    0 讨论(0)
  • 2020-12-01 03:01

    Also you can invalidate it in this way.

    - (void)viewWillTransitionToSize:(CGSize)size withTransitionCoordinator:(id<UIViewControllerTransitionCoordinator>)coordinator {
        [super viewWillTransitionToSize:size withTransitionCoordinator:coordinator];
    
        [self.collectionView.collectionViewLayout invalidateLayout]; 
    }
    
    0 讨论(0)
  • 2020-12-01 03:01

    Please understand this

    traitCollectionDidChange(previousTraitCollection :) won't be called on iPad when it is rotated, because size class is .regular in both portrait and landscape orientations. There's viewWillTransition(to:with:) that will be called whenever collection view size changes.

    Also you shouldn't use UIScreen.mainScreen().bounds if your app supports multitasking as it might not occupy the whole screen, better use collectionView.frame.width for that.

    0 讨论(0)
  • 2020-12-01 03:01

    I solved the issue using below method

    override func viewDidLayoutSubviews() {
            if let flowLayout = collectionView.collectionViewLayout as? UICollectionViewFlowLayout {
                collectionView.collectionViewLayout.invalidateLayout()
                collectionView.collectionViewLayout = flowLayout
            }
        }
    
    0 讨论(0)
  • 2020-12-01 03:02

    When UICollectionLayout detects a bounds change, it asks if it needs to reroute the Invalidate layout. You can rewrite the method directly.UICollectionLayout can call invalidateLayout method at the right time

    class CollectionViewFlowLayout: UICollectionViewFlowLayout{
    
        /// The default implementation of this method returns false.
        /// Subclasses can override it and return an appropriate value
        /// based on whether changes in the bounds of the collection
        /// view require changes to the layout of cells and supplementary views.
        /// If the bounds of the collection view change and this method returns true,
        /// the collection view invalidates the layout by calling the invalidateLayout(with:) method.
        override func shouldInvalidateLayout(forBoundsChange newBounds: CGRect) -> Bool {
    
            return (self.collectionView?.bounds ?? newBounds) == newBounds
        }
    }
    
    0 讨论(0)
提交回复
热议问题