我遇到了一個問題,一個內聯<svg>
元素並不延伸到其原始尺寸,在其viewBox
屬性聲明,當它被包裹在一個絕對定位的父:SVG未能大小適當的絕對定位的父
- 使用
width: 100%
似乎只會強制SVG延伸到瀏覽器定義的默認大小300px
。這也會導致<img>
未按其原始尺寸調整大小。 - 使用
width: auto
使SVG由0像素完全摺疊成0像素的尺寸,但<img>
現在尺寸以它的天然尺寸
有趣的是,這種行爲可以通過使用SVG作爲data:image/svg+xml
爲被複制src
屬性的<img>
元素,所以看起來SVG不會將其原始尺寸「傳遞」到其包含的父元素(可能是<svg>
或<img>
元素)。
所以,我的問題是,如果在CSS中有任何可靠的方法可以強制SVG根據其視圖屬性來調整其原始大小。我大概可以用JS來強制我的方式,通過閱讀viewBox
屬性和use the fixed aspect ratio hack將我的SVG作爲背景圖像以固定寬高比<div>
元素顯示,但我盡力避免這種情況。我可能誤解了瀏覽器的SVG規範的實現,但我似乎無法找到解決方法。
我的問題可以在下面的代碼片段中重現。您可以:使用頁面
- 開啓/關閉絕對父定位在
<img>
或<svg>
元素選擇width
聲明(自動或100%)
// The JS logic below is only used to dynamically set styles based on checkbox/select changes, has nothing to do with SVG layout
// Change positioning strategy
document.getElementById('absPosToggle').addEventListener('change', function() {
var parents = document.querySelectorAll('.parent');
if (this.checked) {
for (var i = 0; i < parents.length; i++) {
parents[i].classList.remove('no-absolute-positioning');
}
} else {
for (var i = 0; i < parents.length; i++) {
parents[i].classList.add('no-absolute-positioning');
}
}
});
// Change width declaration of <svg>/<img> elements
document.getElementById('widthSetting').addEventListener('change', function() {
var images = document.querySelectorAll('img, svg');
var value = this.options[this.selectedIndex].value;
if (value === '100%') {
for (var i = 0; i < images.length; i++) {
images[i].classList.add('width--100');
}
} else {
for (var i = 0; i < images.length; i++) {
images[i].classList.remove('width--100');
}
}
});
body {
margin: 0;
padding: 50px 0 0 0;
}
form {
position: fixed;
top: 0;
left: 0;
right: 0;
display: block;
background-color: #fff;
z-index: 1;
padding: 5px;
}
.wrapper {
width: 100%;
height: 250px;
background-color: #eee;
margin-bottom: 10px;
position: relative;
text-align: center;
}
.parent {
position: absolute;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
}
.parent.no-absolute-positioning {
position: relative;
top: 0;
left: 0;
transform: none;
}
img,
svg {
display: block;
width: auto;
}
img.width--100,
svg.width--100 {
width: 100%;
}
<form>
<label><input type="checkbox" id="absPosToggle" checked />Toggle absolute positioning</label><br />
<label for="widthSetting">Set svg/img widths to:</label><select id="widthSetting">
<option value="auto">Auto</option>
<option value="100%">100%</option>
</select>
</form>
<!-- <img> -->
<div class="wrapper">
<div class="parent">
<img src="https://via.placeholder.com/500x150/173755/ffffff" />
</div>
<span>This is an <code><img></code> element</span>
</div>
<!-- Inline <svg> -->
<div class="wrapper">
<div class="parent">
<svg viewBox="0 0 500 150" xmlns="http://www.w3.org/2000/svg"><rect x="0" y="0" width="500" height="150" fill="#b13131" /><g transform="translate(250, 75)"><text fill="#ffffff" style="text-anchor: middle; font-size: 50; font-family: Arial;" dy="0.35em">500 x 150</text></g></svg>
</div>
<span>This is an inline <code><svg></code> element</span>
</div>
<!-- <img> with SVG as data:image -->
<div class="wrapper">
<div class="parent">
<img src="data:image/svg+xml;charset=utf8,%3Csvg%20viewBox=%220%200%20500%20150%22%20xmlns=%22http://www.w3.org/2000/svg%22%3E%3Crect%20x=%220%22%20y=%220%22%20width=%22500%22%20height=%22150%22%20fill=%22#b13131%22%20/%3E%3Cg%20transform=%22translate(250,%2075)%22%3E%3Ctext%20fill=%22#ffffff%22%20style=%22text-anchor:%20middle;%20font-size:%2050;%20font-family:%20Arial;%22%20dy=%220.35em%22%3E500%20x%20150%3C/text%3E%3C/g%3E%3C/svg%3E"
/>
</div>
<span>This is an <code><img></code> element with SVG as data:image</span>
</div>
提出的問題很好。將寬度和高度屬性添加到SVG不是一種選擇嗎?您是否已經通過https://css-tricks.com/scale-svg/查看是否有任何建議可以解決此問題?我認爲問題的根本原因在於,通過絕對定位,您可以使父級的寬度計算方法變爲縮小擬合,並且對於沒有合適固有寬度的_and_內容,這可能是一種捕獲22,可能會解釋回落到「標準」300px寬度。 – CBroe
@CBroe這實際上是一個很好的觀點。在問這個問題之前,我讀過那篇文章,並且建議不要設置'width'和'height'屬性,而是使用'viewBox'。然而,這種方法的缺點似乎是,SVG將不再保持正確的長寬比,或者在絕對定位時具有摺疊大小。解決方法似乎不得不聲明一個「寬度」和「高度」。 – Terry
_「這種方法的缺點似乎是,SVG將不再保持正確的縱橫比」_ - 您是否檢查過「preserveAspectRatio」,viewbox'「sidekick」,正如文章所稱的那樣,無法解決該問題要麼? https://css-tricks.com/scale-svg/#article-header-id-4 – CBroe