前端外描边完美实现

背景

最近在公司做画布相关的内容,涉及到了字体描边的方案选择,在三种方案对比和尝试下,最终选用了 SVG 作为字体描边的方案,今天就来讲讲这些方案的优缺点。

字体描边方案对比

text-stroke

优点: 实现效果好

缺点:

  • 兼容性一般,尤其是市面上的截图库都不支持这个属性,也就是截图后描边效果会丢失(尝试过 html2canvas、html-to-image、dom-to-image,以及公司内部的一些截图库)
  • 有描边吞字的现象:描边宽度变大时,描边会向内扩展把文本覆盖,
  • 宽度为0px的时候也依旧存在描边

效果:

text-shadow

优点: 兼容性好

缺点: 实现效果不好,怎么说呢,很难评,有种锯齿的美,毕竟人家不是干这行的 😅

效果:

SVG

优点: 兼容性好、实现效果好,整体上看比 text-stroke 效果还要好

缺点: iOS 上同样存在描边吞字的现象,但是它的缺点都可以解决,还请看下文

效果:

Canvas

优点: 兼容性好

缺点:

  • 字体整体比较模糊
  • 有描边吞字的现象
  • 需要通过 canvas api 来进行绘制

效果:

调试

上面四种方案都可以在 CodeSandBox 中自行尝试一下:

codesandbox.io/p/sandbox/s...

SVG 实现字体描边

通过 svg 的 paint-order 来实现字体描边,兼容性最好,并且实现效果也很不错,基本支持市面上所有浏览器,下面就来讲讲 SVG 字体描边方案的实现:

ini 复制代码
<svg 
    xmlns="http://www.w3.org/2000/svg" 
    width="400" height="200" 
    alignment-baseline="text-before-edge" 
    text-anchor="start"
>
  <text 
      x="0" 
      y="0" 
      alignment-baseline="text-before-edge" 
      text-anchor="start"
  >
      字体描边
  </text>
</svg>

text {
  font-size: 50px;
  font-weight: bold;
  stroke: red;
  stroke-width: 4px;
  paint-order: stroke;
}

通过 stroke-linejoin 属性,可以 对 svg 的描边有更灵活的控制:

但是在 iOS 中,使用 paint-order 有一个坑,会出现一种诡异的情况:当 stroke-width 被设置成不同值的时候,描边有可能向文字内部扩展,导致字体被吞没,最终字体的颜色变成跟描边的颜色一致。

解决这个问题当然也有一个办法:使用 svg 的 tspan

tspan 可以控制一个 text 标签中多行文本的展示,通过设置 dxdy 属性来控制与上一个 tspan 的距离。那么对于 iOS 描边展示异常这个问题,我们就有了一个解决办法:

  1. text 内添加两个 tspan
  2. 第一个 tspan 用来控制描边展示,设置 stroke-width
  3. 第二个 tspan 用户展示字体主体,覆盖在第一个 tspan 上面(设置 dx="0" dy="0"
html 复制代码
<svg xmlns="http://www.w3.org/2000/svg" width="400" height="200">
  <text>
    <tspan x="0" y="0" style="stroke-width: 5px">
        文本
    </tspan>
    <tspan dx="0" dy="0">
        文本
    </tspan>
  </text>
</svg>

兼容性如下:

总结

  • 整体上来看,通过 SVG 实现字体描边比其他三种方案效果都要好,并且兼容性也不错;
  • 同时,tspan 可以控制 text 中的文本换行,通过 tspan 可以解决字体被描边覆盖的问题
相关推荐
纸上的彩虹1 小时前
半年一百个页面,重构系统也重构了我对前端工作的理解
前端·程序员·架构
be or not to be1 小时前
深入理解 CSS 浮动布局(float)
前端·css
LYFlied2 小时前
【每日算法】LeetCode 1143. 最长公共子序列
前端·算法·leetcode·职场和发展·动态规划
老华带你飞2 小时前
农产品销售管理|基于java + vue农产品销售管理系统(源码+数据库+文档)
java·开发语言·前端·数据库·vue.js·spring boot·后端
小徐_23332 小时前
2025 前端开源三年,npm 发包卡我半天
前端·npm·github
GIS之路3 小时前
GIS 数据转换:使用 GDAL 将 Shp 转换为 GeoJSON 数据
前端
JIngJaneIL3 小时前
基于springboot + vue房屋租赁管理系统(源码+数据库+文档)
java·开发语言·前端·数据库·vue.js·spring boot·后端
天天扭码3 小时前
以浏览器多进程的角度解构页面渲染的整个流程
前端·面试·浏览器
你们瞎搞3 小时前
Cesium加载20GB航测影像.tif
前端·cesium·gdal·地图切片
南山安4 小时前
Tailwind CSS:顺风CSS
javascript·css·react.js