搜索

查看: 3094|回复: 11

[JavaScript] Vue.js中动态更改svg的相关属性详解

[复制链接]
发表于 2023-5-4 11:46:19 | 显示全部楼层 |阅读模式
Editor 2023-5-4 11:46:19 3094 11 看全部
目录
  • 引言
  • 怎样将一个远程的svg图标资源"下载"到本地
  • 怎样更改svgdom结构里面的相关属性
    引言
    公司项目中有一个关于图标库管理的需求,大致需要在页面能够动态去更改对应svg图标的大小、颜色等(这里的更改颜色限制线性图标)。在网上查找了相关资料,做了技术的预研及demo的编写,在此记录一下。

    怎样将一个远程的svg图标资源"下载"到本地
    首页我们可以利用XMLHttpRequest对象来请求对应的svg图标的远程资源链接地址,并监听实现XMLHttpRequest对象的load事件,将返回的资源进行dom对象的转换、string转换为xml。
    代码如下:
    const xhr = new XMLHttpRequest();
          xhr.open('GET', 'https://www.xx.com/img/xxx.svg', true);
          xhr.send();
          /* 监听xhr对象 */
          xhr.onreadystatechange = function () {
            if (xhr.readyState == 4 && xhr.status == 200) {
              console.log(xhr.responseXML, 'xhr.responseXML---------')
            }
          };
          xhr.addEventListener('load', () => {
            const resXML = stringToXml(xhr.response);
            this.svgDom = resXML.documentElement.cloneNode(true);
          });
    这里的工具函数stringToXml的完整代码如下:
    //将字符串转化成dom对象;string转换为xml
    function stringToXml (xmlString) {
      let xmlDoc;
      if (typeof xmlString == "string") {
        //FF
        if (document.implementation.createDocument) {
          const parser = new DOMParser();
          xmlDoc = parser.parseFromString(xmlString, "text/xml");
        } else if (window.ActiveXObject) {
          // eslint-disable-next-line no-undef
          xmlDoc = new ActiveXObject("Microsoft.XMLDOM");
          xmlDoc.async = false;
          xmlDoc.loadXML(xmlString);
        }
      }
      else {
        xmlDoc = xmlString;
      }
      return xmlDoc;
    }
    这样就可以获取到远程svg资源对应的dom结构了。

    怎样更改svgdom结构里面的相关属性
    产品的要求需要能够动态更改对应svg图标的宽、高、颜色值等。要实现这样的功能有以下几个小点:
  • 将svgDom对象转换成vue的虚拟dom,代码如下:const oSerializer = new XMLSerializer();
  • 根据序列化的对象提供的serializeToString方法将svgDom对象进行字符串化;
    通过svgDom对象提供的宽、高属性值,结合正则来遍历svgDom字符串化后的字符串,进行宽高值的替换。代码如下:
    let sXML = oSerializer.serializeToString(this.svgDom);
    sXML = sXML.replace(`width="${this.svgDom.width.baseVal.value}"`, 'width="40"').replace(`height="${this.svgDom.height.baseVal.value}"`, 'height="40"')
  • 根据sXML来截取svg结构表示的字符串里对应的颜色值,并结合is-color这个插件判断是否是一个真正的颜色,是的话,根据想要替换的颜色值进行全局替换就行。代码如下:
    let curColor = sXML.split('#')[1].substr(0, 6)
          if (!isColor(`#${curColor}`)) {
            curColor = sXML.split('#')[1].substr(0, 3)
          }
          sXML = sXML.replace(new RegExp(`#${curColor}`, "gm"), '#90EE90')
  • 通过Vue实例提供的extend方法创建实例并挂载到某个元素上,代码如下:
    const Profile = Vue.extend({
              template: "" + sXML + ''
            });
            // 创建实例,并挂载到元素上
            new Profile().$mount('#svgTemplate');
    处理前的效果图:

    20230223083636030.png

    20230223083636030.png


    处理后的效果图(将svg宽高由原来的20变为40,将颜色值改为"#90EE90"):

    20230223083636031.png

    20230223083636031.png


    最终完整的代码如下:
    testSvg () {
          const xhr = new XMLHttpRequest();
          xhr.open('GET', 'https://www.xx.com/img/xxx.svg', true);
          xhr.send();
          /* 监听xhr对象 */
          xhr.onreadystatechange = function () {
            if (xhr.readyState == 4 && xhr.status == 200) {
              console.log(xhr.responseXML, 'xhr.responseXML---------')
            }
          };
          xhr.addEventListener('load', () => {
            const resXML = stringToXml(xhr.response);
            this.svgDom = resXML.documentElement.cloneNode(true);
            /* 将svgDom对象转换成vue的虚拟dom */
            const oSerializer = new XMLSerializer();
            let sXML = oSerializer.serializeToString(this.svgDom);
            let curColor = sXML.split('#')[1].substr(0, 6)
            if (!isColor(`#${curColor}`)) {
              curColor = sXML.split('#')[1].substr(0, 3)
            }
            sXML = sXML.replace(`width="${this.svgDom.width.baseVal.value}"`, 'width="40"').replace(`height="${this.svgDom.height.baseVal.value}"`, 'height="40"').replace(new RegExp(`#${curColor}`, "gm"), '#90EE90')
            const Profile = Vue.extend({
              template: "" + sXML + ''
            });
            // 创建实例,并挂载到元素上
            new Profile().$mount('#svgTemplate');
          });
        },
    以上就是Vue.js中动态更改svg的相关属性详解的详细内容,更多关于Vue.js动态更改svg属性的资料请关注知鸟论坛其它相关文章!
  • 回复

    使用道具 举报

    发表于 2023-6-28 23:20:02 | 显示全部楼层
    执着等待等wc 2023-6-28 23:20:02 看全部
    楼主太厉害了!楼主,I*老*虎*U!我觉得知鸟论坛真是个好地方!
    回复

    使用道具 举报

    发表于 2023-6-29 16:16:27 | 显示全部楼层
    dxf17 2023-6-29 16:16:27 看全部
    论坛不能没有像楼主这样的人才啊!我会一直支持知鸟论坛
    回复

    使用道具 举报

    发表于 2023-6-29 17:23:16 | 显示全部楼层
    塞翁364 2023-6-29 17:23:16 看全部
    楼主,我太崇拜你了!我想我是一天也不能离开知鸟论坛
    回复

    使用道具 举报

    发表于 2023-6-29 19:26:48 | 显示全部楼层
    小妖花满楼满fx 2023-6-29 19:26:48 看全部
    我看不错噢 谢谢楼主!知鸟论坛越来越好!
    回复

    使用道具 举报

    发表于 2023-6-30 09:20:33 | 显示全部楼层
    我的苦恼冉 2023-6-30 09:20:33 看全部
    感谢楼主的无私分享!要想知鸟论坛好 就靠你我他
    回复

    使用道具 举报

    发表于 2023-6-30 09:40:40 | 显示全部楼层
    普通人物怨 2023-6-30 09:40:40 看全部
    楼主,我太崇拜你了!我想我是一天也不能离开知鸟论坛
    回复

    使用道具 举报

    发表于 2023-7-3 18:07:07 | 显示全部楼层
    井底燕雀傥 2023-7-3 18:07:07 看全部
    既然你诚信诚意的推荐了,那我就勉为其难的看看吧!知鸟论坛不走平凡路。
    回复

    使用道具 举报

    发表于 2023-7-4 22:18:10 | 显示全部楼层
    心随674 2023-7-4 22:18:10 看全部
    我看不错噢 谢谢楼主!知鸟论坛越来越好!
    回复

    使用道具 举报

    发表于 2023-7-5 23:38:21 | 显示全部楼层
    冀苍鸾 2023-7-5 23:38:21 看全部
    楼主,大恩不言谢了!知鸟论坛是最棒的!
    回复

    使用道具 举报

    • 您可能感兴趣
    点击右侧快捷回复 【请勿灌水】
    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则 返回列表

    RSS订阅| SiteMap| 小黑屋| 知鸟论坛
    联系邮箱E-mail:zniao@foxmail.com
    快速回复 返回顶部 返回列表