当我在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%
,并且还可以确保在加载图像时不会重排内容。
答案 0 :(得分:6)
我也在寻找这个问题的答案。使用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/
上看到的技术答案 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>