The CSS2.1 spec mandates that overflow
other than visible
establish a new "block formatting context". This strikes me as odd, that a property whose obvious purpose is to hide overflow without affecting layout, actually does affect layout in a major way.
It seems like overflow values other than visible
combine two completely unrelated features: whether a BFC is created and whether the overflow is hidden. It’s not like "overflow:hidden" is completely meaningless without a BFC, because floats historically can overflow their parent element, hiding the overflow without changing the layout seems sensible.
What are the reasons behind this decision, assuming they are known? Have the people who worked on the spec described why this was decided to be the case?
I asked about this on the mailing list on your behalf; the thread can be found here. In summary, this has to do with scrolling content for the most part:
Most likely, it refers to scrollable content in a box that may occur outside of the float's parent but would intersect with the float. I don't think this is related to rewrapping content around a float within a scrollable container, as that already happens naturally, plus the float would clip into the container and scroll along with the rest of its content anyway.
Finally this makes sense to me. In fact, I'm going to provide an example here so hopefully it makes sense to you and anyone else who may be wondering. Consider a scenario involving two boxes with the same fixed height and
overflow: visible
(the default), of which the first contains a float that stretches beyond its parent's height:Notice the similarity to one of the examples given in section 9.5. The second box here is simply shown to have overflowing content for the purposes of this answer.
This is fine since the content will never be scrolled, but when
overflow
is set to something other thanvisible
, that causes the content to not only be clipped by the bounds of the box, but also to become scrollable. If the second box hasoverflow: auto
, this is what it would look like had a browser implemented the original CSS2 spec:Because of the float, attempting to scroll the content would cause the browser to have to rewrap it so it doesn't become obscured by the float (and what should happen to the part that scrolls out of the top edge?). It would probably look something like this when scrolled to the bottom:
The catch here is that the browser has to rewrap the content every time it repaints it during scrolling. For browsers that are capable of pixel-based smooth scrolling — which is to say, all of them — I can see why it would be a performance disaster! (And a user experience one, too.)
But that's for when the user can scroll the content, right? This would make sense for
overflow: auto
andoverflow: scroll
, but what aboutoverflow: hidden
?Well, a common misconception is that a container with
overflow: hidden
simply hides content by clipping and cannot be scrolled. This is not completely true:Indeed, this is what it'd look like if the second box was set to
overflow: hidden
and then scrolled to the bottom with the following JavaScript:Again, notice that the content would have to be rewrapped to avoid being obscured by the float.
Even though this wouldn't be as painful for performance as had scrolling UI been available, my best guess is that they made boxes with any
overflow
value other thanvisible
generate a new BFC mainly for the sake of consistency.And so, this change was brought about in CSS2.1, documented here. Now if you apply an
overflow
value other thanvisible
only to the second box, what a browser does is push the entire box aside to make way for the float, because the box now creates a new block formatting context that encloses its contents, instead of flowing around the float. This particular behavior is specified in the following paragraph:Here's what it looks like with
overflow: auto
for example:Note that there is no clearance; if the second box had
clear: left
orclear: both
it would be pushed down, not to the side, regardless of whether it established its own BFC.If you apply
overflow: auto
to the first box instead, the float is clipped into its containing box with the rest of the content due to its fixed height, which is set to80px
in the example code given above:If you revert the first box to
height: auto
(the default value), either by overriding or removing theheight: 80px
declaration from above, it then stretches to the height of the float:This happens to be new in CSS2.1 as well, in that an element with
height: auto
that generates a new block formatting context (i.e. a block formatting context root) will stretch vertically to the height of its floats, and not just enough to contain its in-flow content unlike a regular box. The changes are documented here and here. The change leading to the side-effect of shrinking the box so that it does not intersect the float is documented here.In both of these cases, no matter what you do to the second box, it will never be affected by the float because it has been restricted by the bounds of its container.
I know this will be a speculative answer, however after reading the specifications a few times here is my view on this:
What section 9.4.1 is talking about is any block element that does not fully contain or does not fill the containment space. For example when you float an element it is no longer filling 100% of the parent, like in-flow elements do. Inline blocks, table cells, and table captions are also elements that you can affect height and width but that are not intrinsically 100% of the parent (yes table>tr>td is one that would fill 100% of it's parent but it is designed to allow for multiple td's so the td doesn't count as it will automatically shrink to accommodate additional td's) this also applies to any overflow other than visible because it breaks the containment of the block element.
So if I am reading this correctly the way it works is the 9.4.1 section is referring to block elements that break the default containment rules of the block elements as specified by section 9.2.1