当我在html中使用图像标签时,我尝试在img
标签中指定其宽度和高度,以便浏览器甚至在加载图像之前为它们保留空间,因此当它们完成加载时,页面不会重排(元素不会移动)。例如:
<img width="600" height="400" src="..."/>
问题是现在我想创建一个更“响应”的版本,对于“单列案例”,我想这样做:
<img style="max-width: 100%" src="..."/>
但是,如果我将其与明确指定的宽度和高度混合使用,例如:
<img style="max-width: 100%" width="600" height="400" src="..."/>
并且图像比可用空间宽,然后忽略宽高比调整图像大小。我理解为什么会发生这种情况(因为我“修复了”图像的高度),我想解决这个问题,但我不知道怎么做。
总结一下:我希望能够指定max-width: 100%
,并且还可以确保在加载图像时内容不会被重排。
我也在寻找这个问题的答案。使用max-width
,width=
和height=
,浏览器有足够的数据,它应该能够为图像留出适当的空间,但它似乎不会那样工作。
我现在用jQuery解决方案解决了这个问题。它需要您为width=
标签提供height=
和<img>
。
CSS:
img { max-width: 100%; height: auto; }
HTML:
<img src="image.png" width="400" height="300" />
jQuery的:
$('img').each(function() {
var aspect_ratio = $(this).attr('height') / $(this).attr('width') * 100;
$(this).wrap('<div style="padding-bottom: ' + aspect_ratio + '%">');
});
这会自动应用以下技术:http://andmag.se/2012/10/responsive-images-how-to-prevent-reflow/
来自this blog post by Jonathan Hollin:添加图像的高度和宽度作为内联样式的一部分。这为图像保留了空间,防止了图像加载时的回流,但它也具有响应性。
HTML
<figure style="padding-bottom: calc((400/600)*100%)">
<img src="/images/kitten.jpg" />
</figure>
CSS
figure {
position: relative;
}
img {
max-width: 100%;
position: absolute;
}
figure
可以用div
或您选择的任何其他容器替换。这个解决方案依赖于具有pretty wide browser support的CSS calc()。
工作Codepen可以看到here。
更新:
我找到了一个更聪明的替代版本:http://cssmojo.com/aspect-ratio-using-custom-properties-and-calc/。这仍然需要一个包装元素,它需要CSS自定义属性,但我认为它更优雅。 Codepen的例子是here(归功于Chris Coyier的original)。
起初我想写一下2013年10月的答案。这是不完整的复制,因为它们是不正确的。不要使用它。为什么?我们可以在此代码段中看到它(将已执行的代码段滚动到底部):
$('img').each(function() {
var aspect_ratio = $(this).attr('height') / $(this).attr('width') * 100;
$(this).wrap('<div style="padding-bottom: ' + aspect_ratio + '%">');
});
img { max-width: 100%; height: auto; }
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<div style="width:300px;border:1px solid red">
<img width="400" height="300" src=""/>
Some text
</div>
我们可以从底部看到文字远远不够。此示例中的内容是不完整/不正确的?我将使用纯JavaScript(我们不需要为此下载jQuery)以正确的示例显示它。
使用纯JavaScript的正确示例
请将执行的代码段滚动到底部。
var imgs = document.querySelectorAll('img');
for(var i = 0; i < imgs.length; i++)
{
var aspectRatio = imgs[i].getAttribute('height') /
imgs[i].getAttribute('width') * 100;
var div = document.createElement('div');
div.style.paddingBottom = aspectRatio + '%';
imgs[i].parentNode.insertBefore(div, imgs[i]);
div.appendChild(imgs[i]);
}
.restrict-container div{position:relative}
img
{
position:absolute;
max-width:100%;
top:0; left:0;
height:auto
}
<div class="restrict-container" style="width:300px;border:1px solid red">
<img width="400" height="300" src=""/>
Some text<br>
<img width="400" height="300" src=""/>
Some text
</div>
2013年10月回答的错误:图像应该绝对放置(position:absolute
)到包装容器,但它不是那么放置。
这是我对这个问题的答案的结束。
有关详细信息,请阅读更多信息
对于仅css解决方案,您可以将img
包装在容器中,其中padding-bottom
百分比在页面上保留空间,直到图像加载,从而防止回流。
不幸的是,这种方法确实要求您在css中包含图像宽高比(但不需要内联样式),方法是根据图像的高度和宽度计算(或让css为你计算)padding-bottom
百分比。
如果您的许多图像可以分组为几个标准宽高比,那么您可以为每个宽高比创建一个类,以将适当的padding-bottom
百分比应用于具有该宽高比的所有图像。如果您不处理各种图像宽高比,这可以节省您一点时间和精力。
以下是一些示例html和css,用于具有2:1宽高比的图像:
HTML
<div class="container">
<img id="image" src="https://via.placeholder.com/300x150" />
</div>
CSS
.container {
display: block;
position: relative;
padding-bottom: 50%; /* calc(100%/(300/150)); */
height: 0;
}
.container img {
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
}
下面的代码片段添加了一些额外的html,css和javascript来创建一些可视的顶部和底部参考点,并模拟非常慢的加载图像,这样您就可以直观地看到这种方法如何防止回流。
const image = document.getElementById('image');
const source = 'https://via.placeholder.com/300x150';
const changeSource = () => image.src = source;
setTimeout(changeSource, 3000);
.container {
display: block;
position: relative;
padding-bottom: 50%; /* calc(100%/(300/150)); */
height: 0;
}
.container img {
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
}
.top, .bottom {
background-color: green;
width: 100%;
height: 20px;
}
<div class="top"></div>
<div class="container">
<img id="image" src="" />
</div>
<div class="bottom"></div>
如果我理解了要求,那么您希望能够设置图像大小,此大小仅在内容(HTML)生成时已知,因此可以将其设置为内联样式。
但这必须独立于CSS,并且在图像加载之前,因此也独立于此图像大小。
我已经找到了一个解决方案,涉及将图像包装在div中,并在此div中包含一个svg,可以将其设置为直接作为内联样式的比例。
显然这不是很多语义,但至少它是有效的
包含div有一个名为img的类,以表明它应该是一个img
为了尝试重现加载阶段,图像具有破坏的src
.container {
margin: 10px;
border: solid 1px black;
width: 200px;
height: 400px;
position: relative;
}
.img {
border: solid 1px red;
width: fit-content;
max-width: 100%;
position: relative;
}
svg {
max-width: 100%;
background-color: lightgreen;
opacity: 0.1;
}
#ct2 {
width: 500px;
}
.img img {
position: absolute;
width: 100%;
height: 100%;
max-height: 100%;
max-width: 100%;
top: 0px;
left: 0px;
box-shadow: inset 0px 0px 10px blue;
}
<div class="container" id="ct1">
<div class="img">
<svg xmlns="http://www.w3.org/2000/svg" preserveAspectRatio="xMinYMin meet" viewBox="0 0 400 300" width="400">
</svg>
<img width="400" height="300" src="missing.jpg">
</div>
</div>
<div class="container" id="ct2">
<div class="img">
<svg xmlns="http://www.w3.org/2000/svg" preserveAspectRatio="xMinYMin meet" viewBox="0 0 40 30" width="400">
</svg>
<img width="400" height="300" src="missing.jpg">
</div>
</div>