设置图像的宽度/高度以避免图像加载时重新流动

问题描述 投票:20回答:5

当我在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%,并且还可以确保在加载图像时内容不会被重排。

html image css3 responsive-design
5个回答
7
投票

我也在寻找这个问题的答案。使用max-widthwidth=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/


3
投票

来自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)。


2
投票

起初我想写一下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)到包装容器,但它不是那么放置。

这是我对这个问题的答案的结束。


有关详细信息,请阅读更多信息


2
投票

对于仅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>

2
投票

如果我理解了要求,那么您希望能够设置图像大小,此大小仅在内容(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>
© www.soinside.com 2019 - 2024. All rights reserved.