目前正在构建一个基于浏览器的SVG应用程序。在这个应用程序中,用户可以设置各种形状的样式和位置,包括矩形。
当我将笔画宽度应用到SVG矩形元素(比如1px)时,不同的浏览器会以不同的方式将笔画应用到矩形的偏移和插入。这被证明是很麻烦的,特别是当我试图计算一个矩形的外部宽度和视觉位置,并将其放置在其他元素旁边时。
例如:
Firefox添加了1px的插入(底部和左侧)和1px的偏移(顶部和右侧)
Chrome添加了1px的插入(顶部和左侧),和1px的偏移(底部和右侧)
到目前为止,我唯一的解决方案是自己绘制实际的边界(可能是用路径工具),并将边界定位在描边元素的后面。但这种解决方案是一种令人不快的变通方法,如果可能的话,我宁愿不走这条路。
所以我的问题是,你能控制SVG的笔画宽度如何在元素上绘制吗?
下面是一个函数,它将计算你需要添加多少像素-使用给定的笔画-到顶部,右侧,底部和左侧,所有这些都基于浏览器:
var getStrokeOffsets = function(stroke){
var strokeFloor = Math.floor(stroke / 2), // max offset
strokeCeil = Math.ceil(stroke / 2); // min offset
if($.browser.mozilla){ // Mozilla offsets
return {
bottom: strokeFloor,
left: strokeFloor,
top: strokeCeil,
right: strokeCeil
};
}else if($.browser.webkit){ // WebKit offsets
return {
bottom: strokeCeil,
left: strokeFloor,
top: strokeFloor,
right: strokeCeil
};
}else{ // default offsets
return {
bottom: strokeCeil,
left: strokeCeil,
top: strokeCeil,
right: strokeCeil
};
}
};
我发现了一个简单的方法,虽然有一些限制,但对我来说很有效:
在defs中定义形状
定义一个引用形状的剪辑路径
使用它和双重中风与外部是修剪
下面是一个工作示例:
<svg width="240" height="240" viewBox="0 0 1024 1024">
< def >
<path id="ld" d="M256,0 L0,512 L384,512 L128,1024 L1024,384 L640,384 L896,0 L256,0 Z"/>
< clipPath id =“剪辑”>
<使用xlink: href = " # ld " / >
< / clipPath >
< / def >
< g >
<使用xlink:href="#ld" stroke="#0081C6" stroke-width="160" fill="#00D2B8" clip-path="url(#clip)" / >
< / g >
< / svg >
Xavier Ho的解决方案是将笔画宽度增加一倍,并改变着色顺序,尽管只有在填充颜色为纯色且不透明的情况下才有效。
我已经开发了其他方法,更复杂,但适用于任何填充。它也适用于椭圆或路径(后者存在一些具有奇怪行为的角落情况,例如交叉自己的开放路径,但并不多)。
诀窍是在两层中显示形状。一个没有笔画(只有填充),另一个只有两倍宽度的笔画(透明填充),并通过一个遮罩,显示整个形状,但隐藏了没有笔画的原始形状。
<svg width="240" height="240" viewBox="0 0 1024 1024">
<defs>
<path id="ld" d="M256,0 L0,512 L384,512 L128,1024 L1024,384 L640,384 L896,0 L256,0 Z"/>
<mask id="mask">
<use xlink:href="#ld" stroke="#FFFFFF" stroke-width="160" fill="#FFFFFF"/>
<use xlink:href="#ld" fill="#000000"/>
</mask>
</defs>
<g>
<use xlink:href="#ld" fill="#00D2B8"/>
<use xlink:href="#ld" stroke="#0081C6" stroke-width="160" fill="red" mask="url(#mask)"/>
</g>
</svg>