resize superview after subviews change dynamically using autolayout resize superview after subviews change dynamically using autolayout ios ios

resize superview after subviews change dynamically using autolayout


If you're using Auto Layout, here's what you need to do:

  1. Make sure you aren't adding fixed width and/or height constraints to any of your subviews (depending on which dimension(s) you want to dynamically size). The idea is to let the intrinsic content size of each subview determine the subview's height. UILabels come with 4 automatic implicit constraints which will (with less than Required priority) attempt to keep the label's frame at the exact size required to fit all the text inside.

  2. Make sure that the edges of each label are connected rigidly (with Required priority constraints) to the edges of each other and their superview. You want to make sure that if you imagine one of the labels growing in size, this would force the other labels to make room for it and most importantly force the superview to expand as well.

  3. Only add constraints to the superview to set its position, not size (at least, not for the dimension(s) you want it to size dynamically). Remember that if you set the internal constraints up correctly, its size will be determined by the sizes of all the subviews, since its edges are connected to theirs in some fashion.

That's it. You don't need to call sizeToFit or systemLayoutSizeFittingSize: to get this to work, just load your views and set the text and that should be it. The system layout engine will do the calculations for you to solve your constraints. (If anything, you might need to call setNeedsLayout on the superview...but this shouldn't be required.)


Use container views

In the following example I have a 30x30 image, and the UILabel is smaller than the containing view with the placeholder text. I needed the containing view to be at least as big as the image, but it needed to grow to contain multi-line text.

In visual format the inner container looks like this:

H:|-(15.0)-[image(30.0)]-(15.0)-[label]-(15.0)-|V:|[image(30.0)]|V:|[label(>=30.0)]|

Then, set the containing view to match the height of the label. Now the containing view will ride the size of the label.

As @smileyborg pointed out in his answer, connecting the content rigidly to the superview informs the layout engine that the simple container view should cause it to grow.

Yellow alignment rectangles

If you want the yellow alignment rectangles add -UIViewShowAlignmentRects YES in your scheme's list of run arguments.

UILabel that is multi-line


This was made dramatically easier with the introduction of Stack Views in iOS 9. Use a stack view inside your view to contain all your content that resizes, and then simply call

view.setNeedsUpdateConstraints()view.updateConstraintsIfNeeded()view.setNeedsLayout()view.layoutIfNeeded()

after changing your content. Then you can get your new size by calling

view.systemLayoutSizeFittingSize(UILayoutFittingCompressedSize)

if you ever need to calculate the exact size required for a view.