30

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

4

6 回答 6

10

更新 2:(2019 年 12 月)

Firefox 和 Chrome 现在默认处理这个问题。只需像往常一样添加widthheight属性。有关更多详细信息,请参阅此博客文章


更新 1:(2018 年 7 月)

我发现了一个更聪明的替代版本:http ://cssmojo.com/aspect-ratio-using-custom-properties-and-calc/ 。这仍然需要一个包装器元素,并且需要 CSS 自定义属性,但我认为它更优雅。Codepen 示例在这里(感谢 Chris Coyier 的原创)。


原来的:

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

工作 Codepen 可以在这里看到。

于 2018-07-24T10:36:59.903 回答
8

我也在寻找这个问题的答案。使用max-width,width=height=,浏览器有足够的数据,它应该能够为图像留出适量的空间,但它似乎并不能那样工作。

我现在使用 jQuery 解决方案解决了这个问题。它要求您为您的标签提供width=and 。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/

于 2013-10-28T21:42:18.620 回答
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) 放置到被包装的容器中,但它不是这样放置的。

我对这个问题的回答到此结束。


欲了解更多信息,请阅读更多关于

于 2018-07-25T08:57:14.303 回答
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>

于 2018-07-23T19:13:23.447 回答
2

如果我理解要求没问题,您希望能够设置图像大小,该大小仅在内容 (HTML) 生成时才知道,因此可以将其设置为内联样式。

但这必须独立于 CSS,并且在图像加载之前,所以也独立于这个图像大小。

我已经找到了一个解决方案,它涉及将图像包装在一个 div 中,并在这个 div 中包含一个 svg,该 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>

于 2018-07-27T17:23:29.947 回答
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>

于 2020-05-13T12:16:24.120 回答