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

时间:2013-05-31 06:27:06

标签: html image css3 responsive-design

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

6 个答案:

答案 0 :(得分:6)

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

上看到的技术

答案 1 :(得分: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或您选择的任何其他容器。此解决方案依赖于CSS calc(),其具有pretty wide browser support

可以看到here在工作的Codepen。

更新:

我找到了一个更聪明的替代版本:http://cssmojo.com/aspect-ratio-using-custom-properties-and-calc/。这仍然需要一个wrapper元素,并且需要CSS自定义属性,但是我认为它要优雅得多。 Codepen的示例为here(归功于Chris Coyier的original)。

答案 2 :(得分:2)

对于仅使用CSS的解决方案,您可以将img包装在一个容器中,其中padding-bottom百分比会在页面上保留空间,直到图像加载为止,以防止回流。

不幸的是,这种方法确实需要您通过基于图像高度和图像的padding-bottom百分比(或让您为css计算)来在css中包括图像长宽比(但不需要内联样式)。宽度。

如果可以将许多图像分为几个标准的长宽比,则可以为每个长宽比创建一个类,以将适当的padding-bottom百分比应用于具有该长宽比的所有图像。如果您不处理各种图像宽高比,则可以节省一些时间和精力。

以下是高宽比为2:1的图像的html和css示例:

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>

答案 3 :(得分: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)放置在包装好的容器中,而不是这样放置。

这是我对这个问题的回答的结尾。


有关更多信息,请参见

答案 4 :(得分: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>

答案 5 :(得分:1)

我发现最好的解决方案是创建具有相应尺寸的透明base64 gif作为img标签的占位符,该标签在页面加载后通过js触发加载。

<img data-src="/image.png" src="">

对于博客文章,我使用此PHP函数自动创建它们

function CreatePreloadPlaceholderGif($width, $height) {
    $wHex = str_split(str_pad(dechex($width), 4, "0", STR_PAD_LEFT), 2);
    $hHex = str_split(str_pad(dechex($height), 4, "0", STR_PAD_LEFT), 2);
    $hex = "474946383961".$wHex[1].$wHex[0].$hHex[1].$hHex[0]."800100ffffff00000021f904010a0001002c00000000010001000002024c01003b";
    $base64= '';
    foreach(str_split($hex, 2) as $pair){
       $base64.= chr(hexdec($pair));
    }
    return base64_encode($base64);
}
echo CreatePreloadPlaceholderGif(300, 500);
//  R0lGODlhLAH0AYABAP///wAAACH5BAEKAAEALAAAAAABAAEAAAICTAEAOw==

在前端,结果是这样的

function loadimage() {
      elements = document.querySelectorAll('img[data-src]');
    elements.forEach( el => {
        el.setAttribute('src', el.getAttribute('data-src'))
    });
}
img {
  background-color:#696969;
}
<div>300x500 image placeholder</div>
<img data-src="https://albahaabazar.com/wp-content/uploads/2017/11/300x500.png" src="">
<div>After page load, run js command to replace src attribute with data-src</div>
<button onclick="loadimage()">Load image</button>