web-dev-qa-db-ja.com

画像の読み込み時にリフローを回避するために、画像の幅/高さを設定します

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%を指定できるようにしたいのですが、画像の読み込み時にコンテンツがリフローされないようにする必要もあります。

25
gabor

私もこの問題の答えを探しています。 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/

8
Paul Pritchard

更新2:(2019年12月)

FirefoxとChromeはデフォルトでこれを処理するようになりました。通常どおり、width属性とheight属性を追加するだけです。 このブログ投稿 を参照してください。詳細については。


更新1:(2018年7月)

私はこれのはるかに賢い代替バージョンを見つけました: http://cssmojo.com/aspect-ratio-using-custom-properties-and-calc/ 。これにはまだラッパー要素が必要であり、CSSカスタムプロパティが必要ですが、はるかにエレガントだと思います。 Codepenの例は ここ (Chris Coyierのクレジット オリジナル )です。


オリジナル:

From 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を見ることができます ここ

6
wiiiiilllllll

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>
2
benvc

要件を理解している場合は、画像サイズを設定できるようにします。このサイズはコンテンツ(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>
2
vals

最初に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
Bharata