I stumbled upon a difference in layout rendering between Safari and Chrome/Firefox and I don’t know which one is “right”.
You can check the jsfiddle here
On Firefox/Chrome the layout is as expected, the yellow div is right after the red ones. But on Safari, the yellow div is positioned under the red ones.
After investigating what I did wrong I found out the bug comes from the CSS class E whose property margin-right (value: -11px) is bigger than the width property (value: 10px) for my div.
I think I understand why Safari renders it this way. The width of div of class B is computed as being the sum of the widths of its children as they have the property float: left;.
Here it is widthB = widthB2*2 + widthE + marginRightE + widthC or marginRightE < -widthE so widthB is not large enough to contain each div next to each other.
So my questions are:
- Am I right in my understanding of what Safari does?
- Why do Chrome and Firefox render differently? Are they just not decreasing the width of the parent div based on a negative
margin-right? - Would the proper correction to always have a
margin-rightlesser or equal to the width of a div in this case?
Thank you!
HTML:
<div class="A">
<div class="C">
<div class="B">
<div class="B2"></div>
<div class="B2"></div>
<div class="E"></div>
<div class="C">
<div class="D"></div>
</div>
</div>
</div>
</div>
CSS:
.A {
background-color: blue;
height: 200px;
}
.B {
height:100px;
}
.B2 {
background-color: red;
height: 100px;
width: 100px;
float: left;
}
.C {
float: left;
}
.D {
height: 40px;
width: 40px;
float:left;
background-color: yellow;
}
.E {
height: 50px;
width: 10px;
position: relative;
left: -10px;
margin-right: -11px;
background-color: black;
float: left;
}
I’m not sure what you expect to happen with the CSS in the JS fiddle. You are delving into undefined behaviour. I say this because:
‘C’ is floated but does not have a defined width. This leads to issues in various browsers depending on the complexity of the layout.
None of the floated elements are ever cleared. When floating it is imperative that a clearfix of some description is used, whether it is
clear:both, etc.If you tweak the mark-up and add a clear-fix, you see that the content is always
239px. See http://jsfiddle.net/eaFn9/However, it seems like the relatively positioned item ‘E’ and margin is having a negative impact on the width calculation, as Chrome’s web inspector seems to always report oddly for the negative margin on this element.
If you play around with this in web inspector you can see it’s almost as if the negative margin is the cause of the drop. I think it may be due to a container that does not have a width, and isn’t position relative in itself.
How to fix?
Personally, I would want to re-write your layout to include fixed widths on all floats, reduce nesting of floats and clear where possible. It seems overly complex but without a real world use case it’s hard to rewrite.
However, It seemed to me that you can wrap ‘B2’ + ‘E’ elements in a wrapper that is floated and relatively positioned, then use absolute positioning on ‘E’ to give the same affect and remove the negative margin.
This is the JSFiddle I came up with: http://jsfiddle.net/jV3Ub/