使用 foreignObject 在微信小程序实现 svg 头像生成

前言

这段时间想要自己做一个玩具小程序,在写首页时,我就已经做到了需要展示小程序的模块。开始我随便找了一张图片进行占位。但是当我涉及到开发后端接口,头像就不能单纯使用同一张图了,虽然小程序官方提供了获取用户头像的能力,但却需要用户自己授权。在用户还未授权的这段时间,我当然需要去提供一个好看(✧∀✧)的默认头像。

寻求头像生成的方法

我先是在搜索引擎寻找提供生成头像的一些网站:

1.

等等还是很多的,但是要不就是我觉得不符合我的审美爱好,要不就是不提供生成服务,要不就是太复杂,虽然我有找到一些提供api接口的,但是 emm 国内访问不了。自己搭建服务也过于麻烦了。

眼前一亮

我搜到找了开源库react-nice-avatar,而且我也相对比较喜欢他的样式,可以根据字符串(可以使用用户的唯一 openid)哈希来生成唯一头像。

可行性研究

我先将项目进行clone下来,运行下来很顺利,项目基于 React,生成的方式是基于对 svg 的替换以及同一个 div 的 position 来控制 svg 的位置。下面的是项目结构。

解决在微信中显示 svg

因为之前遇到过需要在微信中显示 svg 图表,是通过转换成 background-image 来显示 svg,并且还需要对 svghtml 结构的一些字符进行替换。但是其中我发现头像的结构并不单纯涉及到 svg,其中还存在一些 div 用于控制 div 的位置。

解决办法

这时刚好我同事就在我旁边,我把我的疑惑告诉了他,他立刻就对我提到了一个 html 标签 foreignObject

对于兼容性( ie 哒咩)

方案测试

然后我将 div 包裹在了foreignObject中,显示正常!

项目改造

现在的项目是基于 react 来进行运行的,但是我其实需要的只是最终的头像 html 文本。

其实我只要将组件全部改造成返回 svg 文本的一个函数,最终在 index.ts 中进行拼接,就可以在保证原本的逻辑下得到我需要的结果。

接下来就是无聊的改造时间.....

替换过程

这里举一个替换的例子:

chroma-js 的 替换,项目中依赖了这个库来进行颜色处理,但是其实用到的只有两处。所以我的想法就是在 utils 中来实现所用到的两个方法,减少库的引入。 继续寻求 Gpt 的帮助

然后再引入处理最后需要用于 background-image 的 data:image/svg+xml ,改造就 ok 了。

进行测试

最后用产出的粘贴到小程序进行测试。意料之中的

嗯,显示失败了,一片空白....

错误排查

控制变量

同一个配置,切换到最开始时正常的分支,以及现在改造后错误的分支。 通过同一个配置生成图像。

js 复制代码
export function transformToBgImg(svgHTML: string): string {
  const txt = svgHTML
    .replace(
      "<svg",
      ~svgHTML.indexOf("xmlns")
        ? "<svg"
        : "<svg xmlns='http://www.w3.org/2000/svg'"
    )
    .replace(/\s+/g, " ");
  return `data:image/svg+xml,${txt}`
}

通过上面的函数来得到去除尽量多的连续空白字符。便于后续对比。

对于产生的区别,借助文本 diff 工具。可以很明显的看出来。

错误原因

过程就不具体讲了,这里直接贴出来排除的结果

  1. react 中使用的 props,在替换时忘记给变量包上双引号""
  2. props 中的驼峰写法的转换成 xxx-xxxx
  3. 将自闭和标签进行替换 <path/> => <path></path>

运行结果

项目地址

github.com/ayuan-gy/ni...

相关推荐
正小安1 小时前
如何在微信小程序中实现分包加载和预下载
前端·微信小程序·小程序
_.Switch2 小时前
Python Web 应用中的 API 网关集成与优化
开发语言·前端·后端·python·架构·log4j
一路向前的月光2 小时前
Vue2中的监听和计算属性的区别
前端·javascript·vue.js
长路 ㅤ   2 小时前
vite学习教程06、vite.config.js配置
前端·vite配置·端口设置·本地开发
长路 ㅤ   2 小时前
vue-live2d看板娘集成方案设计使用教程
前端·javascript·vue.js·live2d
Fan_web3 小时前
jQuery——事件委托
开发语言·前端·javascript·css·jquery
安冬的码畜日常3 小时前
【CSS in Depth 2 精译_044】第七章 响应式设计概述
前端·css·css3·html5·响应式设计·响应式
莹雨潇潇3 小时前
Docker 快速入门(Ubuntu版)
java·前端·docker·容器
Jiaberrr4 小时前
Element UI教程:如何将Radio单选框的圆框改为方框
前端·javascript·vue.js·ui·elementui
Tiffany_Ho4 小时前
【TypeScript】知识点梳理(三)
前端·typescript