我刚刚发现 Chrome 不会缓存放置在 SVG 中的图像,如果它们的cache-control
标题设置为no-cache
. Firefox & IE10 似乎忽略了这个设置。
我用静态 SVG 创建了一个小测试页面:
HTML:
<div style="width: 500px; text-align: center;">
<input id="move-left-btn" type="button" value="<<">
<input id="move-right-btn" type="button" value=">>">
</div>
<div class="svgwrapper" style="width: 500px; height: 250px; background-color: lightgrey;">
<svg id="svg" version="1.1" xmlns="http://www.w3.org/2000/svg" width="500" height="250">
<g id="svggroup" class="transition-on" transform="matrix(0.2,0,0,0.2,80,35)">
<image width="1672" height="887" opacity="1" xlink:href="https://dl.dropboxusercontent.com/sh/q7htlj5h8qqfhjf/SVDuynM7R3/car.png"></image>
</g>
</svg>
</div>
Javascript:
$(document).ready(function() {
var curXPos = 80;
// Local test function which represent some server calls in my "real life" scenario
// Just updates the x-position in the transform matrix in this test case
function updateSvgText(svgText, posXDelta) {
curXPos += posXDelta;
if (curXPos < 0) {
curXPos = 160;
} else if (curXPos > 160) {
curXPos = 0;
}
return svgText.replace(/matrix\(.*\)/, 'matrix(0.2,0,0,0.2,' + curXPos + ',35)');
}
// Fetch the new SVG (in real life from server) and rerender it
function moveSvg(posXDelta) {
var svg = $('#svg'),
svgText = updateSvgText($('.svgwrapper').html(), posXDelta);
svg.empty();
svg.append($(svgText).children());
}
$('#move-left-btn').click($.proxy(moveSvg, this, -20));
$('#move-right-btn').click($.proxy(moveSvg, this, 20));
});
cache-control
源图像标题设置为no-cache
(每次按下“移动”按钮后在 chrome 中闪烁)的工作示例:http:
//jsfiddle.net/zF6NF/4/具有不同源图像的相同示例,
cache-control
标题设置为max-age=315360000,public
(无闪烁):http:
//jsfiddle.net/zF6NF/5/
在 Chrome 中,您可以在第一个示例中看到每个按钮单击时图像的重新加载(图像的“闪烁”并且在开发工具的网络选项卡中可见),而 Firefox 在两个示例中都顺利地重新渲染了 SVG,而无需任何重新加载。
一些附加信息:
这只是一个例子。在我的“现实生活场景”中,我从服务器接收到一个新的 SVG(而不是
updateSvgText
方法调用),这意味着我不能只通过更改变换矩阵属性的值来执行 SVG 的部分更新,而是必须每次都重新渲染整个 SVG(至少现在......)。我无法控制图像的来源,这意味着两件事:
- 我无法更改
cache-control
标题 - 我无法创建 Base64 编码的数据 uri,将它们保存在本地,只需在渲染前用这些数据 uri 替换 SVG 内的图像(由于“资源相同”策略,无法创建 Base64 编码的数据 uri... )
- 我无法更改
有什么办法要么...
cache-control
即使图像来自不受控制的远程位置,也可以在本地覆盖/覆盖标头?- 从从不同域加载的图像创建 Base64 编码的数据 uri 我对客户端没有任何控制权?
- 不知何故告诉 Chrome 总是在我的 SVG 中缓存图像?
不用说,其他解决方案也非常受欢迎!
谢谢