ChatGPT解决这个技术问题 Extra ChatGPT

Maintain image aspect ratio when changing height

Using the CSS flex box model, how can I force an image to maintain its aspect ratio?

JS Fiddle: http://jsfiddle.net/xLc2Le0k/2/

Notice that the images stretch or shrink in order to fill the width of the container. That's fine, but can we also have it stretch or shrink the height to maintain the image proportions?

HTML

<div class="slider">
    <img src="http://www.placebacon.net/400/300" alt="Bacn">
    <img src="http://www.placebacon.net/400/300" alt="Bacn">
    <img src="http://www.placebacon.net/400/300" alt="Bacn">
    <img src="http://www.placebacon.net/400/300" alt="Bacn">
</div>

CSS

.slider {
    display: flex;
}
.slider img {
    margin: 0 5px;
}
As far as I know, the best solution is to use a <div> with the CSS background-image: url(...);background-size:contain; background-repeat:no-repeat
There are some interesting leads here, but what about when the images aren't already the same ratio? Adding and 'extra' div, is the last thing we need to worry about. Why not use a test case like this instead: jsfiddle.net/sheriffderek/999Lg9qv

M
Matty Balaam

To keep images from stretching in either axis inside a flex parent I have found a couple of solutions.

You can try using object-fit on the image which, e.g.

object-fit: contain;

http://jsfiddle.net/ykw3sfjd/

Or you can add flex-specfic rules which may work better in some cases.

align-self: center;
flex: 0 0 auto;

object-fit could be a good solution were it not missing support in IE and Edge even up to Edge 14
I've been using that fallback myself, works very well.
@daniels Edge now has object-fit in development: developer.microsoft.com/en-us/microsoft-edge/platform/status/…
object-fit: contain; did the trick for me, when I only had the problem in chrome, ff was fine.
S
Samuel Liew

For img tags if you define one side then other side is resized to keep aspect ratio and by default images expand to their original size.

Using this fact if you wrap each img tag into div tag and set its width to 100% of parent div then height will be according to aspect ratio as you wanted.

http://jsfiddle.net/0o5tpbxg/

* {
    margin: 0;
    padding: 0;
}
.slider {
    display: flex;
}
.slider .slide img {
    width: 100%;
}

I know there are two answers that suggest using divs, but I am marking this answer correct because it explains why the image heights are different than I expected. In fact, this is normal & correct behavior, and is unlikely to be "fixed", since it is not a bug.
But this answer doesn't talk anything about flexbox, because images do behave differently within flexbox container. Setting one side will not proportionally resize the other. Wrapping them in non flexbox container helps but that adds unnecessary markup losing semanticity.
It doesn't look like you need a wrapping div to make this work: jsfiddle.net/xLc2Le0k/120
But, what about this? jsfiddle.net/xLc2Le0k/15 I think that this solution is a fluke that only works because the images are all the same ratio. side note: there isn't any 'semanticity' lost by placing an image in a div for positioning / especially when dealing with responsive images.
Seriously the cleanest solution and it works with picture!
M
Marvin Herbold

No need to add a containing div.

The default for the css "align-items" property is "stretch" which is what is causing your images to be stretched to its full original height. Setting the css "align-items" property to "flex-start" fixes your issue.

.slider {
    display: flex;
    align-items: flex-start;  /* ADD THIS! */
}

Ahh... thanks for actually doing things the supported way. I was reading the answers until that one and couldn't believe all we had were dirty hacks..
This should be the answer
No other answer worked, you saved my day!
S
Stickers

Most of images with intrinsic dimensions, that is a natural size, like a jpeg image. If the specified size defines one of both the width and the height, the missing value is determined using the intrinsic ratio... - see MDN.

But that doesn't work as expected if the images that are being set as direct flex items with the current Flexible Box Layout Module Level 1, as far as I know.

See these discussions and bug reports might be related:

Flexbugs #14 - Chrome/Flexbox Intrinsic Sizing not implemented correctly.

Firefox Bug 972595 - Flex containers should use "flex-basis" instead of "width" for computing intrinsic widths of flex items

Chromium Issue 249112 - In Flexbox, allow intrinsic aspect ratios to inform the main-size calculation.

As a workaround, you could wrap each <img> with a <div> or a <span>, or so.

jsFiddle

.slider { display: flex; } .slider>div { min-width: 0; /* why? see below. */ } .slider>div>img { max-width: 100%; height: auto; }

4.5 Implied Minimum Size of Flex Items To provide a more reasonable default minimum size for flex items, this specification introduces a new auto value as the initial value of the min-width and min-height properties defined in CSS 2.1.

Alternatively, you can use CSS table layout instead, which you'll get similar results as flexbox, it will work on more browsers, even for IE8.

jsFiddle

.slider { display: table; width: 100%; table-layout: fixed; border-collapse: collapse; } .slider>div { display: table-cell; vertical-align: top; } .slider>div>img { max-width: 100%; height: auto; }


The first snippet needs .slider > div{min-width:0} on new browsers that support min-width: auto.
The flexbox spec introduced a new auto value as the default value for min-width and min-height (previously it was 0). Chrome does not implement it yet, so your first snippet works. But new browsers need it in order to allow the flex items to shrink smaller than the default width of the images.
+1 this should be accepted answer because it talks about images in flexbox model which is the root problem in this case...
Table is great if your layout doesn't change at various break-points, but that is unlikely these days.
What about when the images aren't all the same ratio? jsfiddle.net/sheriffderek/5u7y21we
M
Muhammad Umer

I have been playing around flexbox lately and i came to solution for this through experimentation and the following reasoning. However, in reality I'm not sure if this is exactly what happens.

If real width is affected by flex system. So after width of elements hit max width of parent they extra width set in css is ignored. Then it's safe to set width to 100%.

Since height of img tag is derived from image itself then setting height to 0% could do something. (this is where i am unclear as to what...but it made sense to me that it should fix it)

DEMO

(remember saw it here first!)

.slider {
    display: flex;
}
.slider img {
    height: 0%;
    width: 100%;
    margin: 0 5px;
}

Works only in chrome yet


Interesting, but it seems a bug. According to the spec, "The percentage is calculated with respect to the height of the generated box's containing block. If the height of the containing block is not specified explicitly (i.e., it depends on content height), and this element is not absolutely positioned, the value computes to auto". That's what happens on Firefox.
jsfiddle.net/xLc2Le0k/8 explain the difference in ff and chrome. for this
width seems to be doing in ff what height is doing in chrome.
It is interesting, but I'm afraid that X-browser-only solutions are not really solutions. Your commented fiddle, here: jsfiddle.net/xLc2Le0k/8 is absolutely fascinating, though, in FF. This seems to show that FF is making the image height "proportionate" to the height of the image if it were truly 100%. In other words, the img doesn't "know" about the container's display: flex. This explains why, in FF, if you set the width of the img to 100%, the image is taller than if you set the width to auto.
This is kinda close... but only works in Chrome: jsfiddle.net/sheriffderek/xLc2Le0k/270
A
Asim K T

This behavior is expected. flex container will stretch all its children by default. Image have no exception. (ie, parent will have align-items: stretch property )

To keep the aspect ratio we've two solutions:

Either replace default align-items: stretch property to align-items: flex-start or align-self: center etc, http://jsfiddle.net/e394Lqnt/3/

or

Set align property exclusively to the child itself: like, align-self: center or align-self: flex-start etc. http://jsfiddle.net/e394Lqnt/2/


Z
ZungTa

Declare where display: flex; was given Element.
align-items: center;


k
kengreg

Actually I found out that the container of the image tag need to have a height in order to keep the ratio of the image. for example>

.container{display:flex; height:100%;} img{width:100%;height:auto;}

then in your html your container will wrap the tag image

<div class="container"><img src="image.jpg" alt="image" /></div>

this work for IE and other browsers


C
Clinton Green

I just had the same issue. Use the flex align to center your elements. You need to use align-items: center instead of align-content: center. This will maintain the aspect ratio, while align-content will not keep the aspect ratio.


关注公众号,不定期副业成功案例分享
Follow WeChat

Success story sharing

Want to stay one step ahead of the latest teleworks?

Subscribe Now