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%
を指定できるようにしたいのですが、画像の読み込み時にコンテンツがリフローされないようにする必要もあります。
私もこの問題の答えを探しています。 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/
更新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を見ることができます ここ 。
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>
要件を理解している場合は、画像サイズを設定できるようにします。このサイズはコンテンツ(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>
最初に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
)ラップされたコンテナに移動しますが、配置されていません。
これで、この質問に対する私の答えは終わりです。
詳細については、の詳細をご覧ください。