网页网站您现在的位置是:首页 > 博客日志 > 网页网站

CSS中内联SVG图片有比Base64更好的形式

<a href='mailto:'>微wx笑</a>的头像微wx笑2019-08-29 12:57:02网页网站人已围观关键字:CSS,SVG,Base64

简介一、之前SVG背景图的做法在写这篇文章之前,对于SVG图标或图形,如果作为CSS背景使用,只要尺寸不超过2K,很多时候我都是直接内联在CSS代码中的,采用的形式是base64格式。例如下面这

一、之前SVG背景图的做法

image.png29R编程技术_踩坑日志_进阶指南 - 无知人生

在写这篇文章之前,对于SVG图标或图形,如果作为CSS背景使用,只要尺寸不超过2K,很多时候我都是直接内联在CSS代码中的,采用的形式是base64格式。29R编程技术_踩坑日志_进阶指南 - 无知人生

例如下面这个截图中的下箭头小图标:29R编程技术_踩坑日志_进阶指南 - 无知人生

image.png29R编程技术_踩坑日志_进阶指南 - 无知人生

相关CSS代码如下:29R编程技术_踩坑日志_进阶指南 - 无知人生

.icon-arrow-down {
    width: 20px; height: 20px;
    background: url() no-repeat center/100%;
}

实时效果如下:29R编程技术_踩坑日志_进阶指南 - 无知人生

矢量,少了个文件请求,渲染几乎无延时,自认为性价比不错。29R编程技术_踩坑日志_进阶指南 - 无知人生

然而,今天我发现,对于SVG图形,使用Base64格式进行内联,并不是最佳的做法,还有更好的实现方式,就是直接使用SVG XML格式代码,无需进行base64转换。29R编程技术_踩坑日志_进阶指南 - 无知人生

二、SVG直接内联

方法就是把SVG代码直接内联在CSS的url()方法中,语法就是data:image/svg+xml;utf8,加上完整的SVG代码即可!例如比较常用的background-imageurl()方法,代码如下:29R编程技术_踩坑日志_进阶指南 - 无知人生

.icon-arrow-down {
    width: 20px; height: 20px;
    background: url('data:image/svg+xml;utf8,<svg version="1.1" xmlns="http://www.w3.org/2000/svg" width="200" height="200" viewBox="0 0 200 200"><path fill="#00A5E0" d="M145.659,68.949c-5.101-5.208-13.372-5.208-18.473,0L99.479,97.233 L71.772,68.949c-5.1-5.208-13.371-5.208-18.473,0c-5.099,5.208-5.099,13.648,0,18.857l46.18,47.14l46.181-47.14 C150.759,82.598,150.759,74.157,145.659,68.949z"/></svg>') no-repeat center;
    background-size: 100%;
}

您可以狠狠地点击这里:SVG代码直接写在CSS中demo29R编程技术_踩坑日志_进阶指南 - 无知人生

可以看到在Chrome浏览器下,小箭头图标显示出来了:29R编程技术_踩坑日志_进阶指南 - 无知人生

image.png29R编程技术_踩坑日志_进阶指南 - 无知人生

然后,不要高兴地太早,我们打开IE浏览器一看,心凉了半截,一片空白!29R编程技术_踩坑日志_进阶指南 - 无知人生

image.png29R编程技术_踩坑日志_进阶指南 - 无知人生

莫非IE浏览器不支持这种表示方法?29R编程技术_踩坑日志_进阶指南 - 无知人生

大家莫急莫慌且淡定,IE浏览器是支持直接内联的,只是,IE浏览器出于安全考虑,需要对一些字符进行安全转义。29R编程技术_踩坑日志_进阶指南 - 无知人生

使IE浏览器也支持的处理

首先,我们对所有SVG字符进行URL-encode肯定是OK的,例如执行:29R编程技术_踩坑日志_进阶指南 - 无知人生

encodeURIComponent('<svg version="1.1" ...</svg>')

得到如下CSS:29R编程技术_踩坑日志_进阶指南 - 无知人生

.icon-arrow-down {
    width: 20px; height: 20px;
    background: url(data:image/svg+xml,%3Csvg%20version%3D%221.1%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20width%3D%22200%22%20height%3D%22200%22%20viewBox%3D%220%200%20200%20200%22%3E%3Cpath%20fill%3D%22%2300A5E0%22%20d%3D%22M145.659%2C68.949c-5.101-5.208-13.372-5.208-18.473%2C0L99.479%2C97.233%20L71.772%2C68.949c-5.1-5.208-13.371-5.208-18.473%2C0c-5.099%2C5.208-5.099%2C13.648%2C0%2C18.857l46.18%2C47.14l46.181-47.14%20C150.759%2C82.598%2C150.759%2C74.157%2C145.659%2C68.949z%22%2F%3E%3C%2Fsvg%3E) no-repeat center;
    background-size: 100%;
}

结果IE浏览器下SVG图标出现了(包括IE9浏览器):29R编程技术_踩坑日志_进阶指南 - 无知人生

image.png29R编程技术_踩坑日志_进阶指南 - 无知人生

眼见为实,您可以狠狠地点击这里:CSS中SVG代码完全转义demo29R编程技术_踩坑日志_进阶指南 - 无知人生

但是,完全URL-encode后也会带来另外的问题,和转义前原始SVG相比,可读性差了很多,想改个颜色都找不到在哪里。那有没有可以进一步提升的空间呢?29R编程技术_踩坑日志_进阶指南 - 无知人生

有!29R编程技术_踩坑日志_进阶指南 - 无知人生

部分URL encode IE浏览器也兼容

实际上,完全URL encode是没有必要的,也就是,我们无需让所有的字符都encode也能让IE浏览器正确识别。根据前辈的实践,只需要对下面这些字符转义就能满足绝大多数的场景,它们是:"%#{}<>29R编程技术_踩坑日志_进阶指南 - 无知人生

其中,由于XML中属性值双引号单引号没有区别,因此我们可以改成单引号',省去转换还增加了可读性,然后url()中使用双引号"包起来。29R编程技术_踩坑日志_进阶指南 - 无知人生

于是,下面的CSS加内嵌SVG代码就能在IE浏览器下正常工作:29R编程技术_踩坑日志_进阶指南 - 无知人生

.icon-arrow-down {
    width: 20px; height: 20px;
    background: url("data:image/svg+xml,%3Csvg version='1.1' xmlns='http://www.w3.org/2000/svg' width='200' height='200' viewBox='0 0 200 200'%3E%3Cpath fill='%2300A5E0' d='M145.659,68.949c-5.101-5.208-13.372-5.208-18.473,0L99.479,97.233 L71.772,68.949c-5.1-5.208-13.371-5.208-18.473,0c-5.099,5.208-5.099,13.648,0,18.857l46.18,47.14l46.181-47.14 C150.759,82.598,150.759,74.157,145.659,68.949z'/%3E%3C/svg%3E") no-repeat center;
    background-size: 100%;
}

结果IE9+浏览器下SVG背景也能正常显示:29R编程技术_踩坑日志_进阶指南 - 无知人生

image.png29R编程技术_踩坑日志_进阶指南 - 无知人生

眼见为实,您可以狠狠地点击这里:SVG转义后的代码直接写在CSS显示demo29R编程技术_踩坑日志_进阶指南 - 无知人生

可以看到,部分转义后的SVG代码既满足的兼容性要求,同时可读性也有了进一步的提高。29R编程技术_踩坑日志_进阶指南 - 无知人生

例如,我们想改变此SVG图标的颜色,就简单了,直接把fill='%2300A5E0'这里的00A5E0改成我们希望的十六进制颜色值即可。要是以前的Base64格式,改颜色是不可能的,需要该原始SVG并重新生成一下,特麻烦。29R编程技术_踩坑日志_进阶指南 - 无知人生

而且,直接原始SVG内嵌,即使有部分转义,其字符大小也明显比Base64格式要小。因此,理论上,CSS中内联SVG背景图,没有使用Base64格式的理由。但,目前大规模使用还有一个阻碍,就是缺少转换的工具,总不可能每次我们都手动修改需要转义的字符吧。29R编程技术_踩坑日志_进阶指南 - 无知人生

不要担心,工具是会有的,我这就给大家写一个,自带压缩和转义。29R编程技术_踩坑日志_进阶指南 - 无知人生

三、在线SVG压缩和转义工具

image.png29R编程技术_踩坑日志_进阶指南 - 无知人生

哎呀,不容易呀,周五晚周六晚加周日下午这个页面才弄好。29R编程技术_踩坑日志_进阶指南 - 无知人生

当当当当,隆重介绍下:SVG在线压缩合并工具29R编程技术_踩坑日志_进阶指南 - 无知人生

image.png29R编程技术_踩坑日志_进阶指南 - 无知人生

支持选择SVG文件上传,支持拖拽上传,也支持直接粘贴SVG代码进行压缩。29R编程技术_踩坑日志_进阶指南 - 无知人生

例如这个名为download.svg的SVG文件(下载自iconfont.cn),我们上传压缩下,结果有:29R编程技术_踩坑日志_进阶指南 - 无知人生

image.png29R编程技术_踩坑日志_进阶指南 - 无知人生

其中,左边的多行文本域中是压缩后的SVG代码,右边的文本域中是压缩并转义的SVG代码,可以直接用在CSS代码中,也就是本文介绍的这种内联方式,来,实地测试下,点击文本域右上方的复制图标按钮,复制转义代码,放在CSS中,如下:29R编程技术_踩坑日志_进阶指南 - 无知人生

.icon-download {
    width: 40px; height: 40px;
    background: url("data:image/svg+xml,%3Csvg class='icon' viewBox='0 0 1024 1024' xmlns='http://www.w3.org/2000/svg' width='128' height='128'%3E%3Cdefs%3E%3Cstyle/%3E%3C/defs%3E%3Cpath d='M332.32 464.112l144 160a47.952 47.952 0 0 0 71.36 0l144-160a48 48 0 0 0-71.36-64.224L560 466.912V176a48 48 0 1 0-96 0v290.912l-60.32-67.024a48 48 0 1 0-71.36 64.224zM880 560a48 48 0 0 0-48 48v192H192V608a48 48 0 1 0-96 0v240a48 48 0 0 0 48 48h736a48 48 0 0 0 48-48V608a48 48 0 0 0-48-48z'/%3E%3C/svg%3E") no-repeat center/100%;
}

效果棒棒哒!29R编程技术_踩坑日志_进阶指南 - 无知人生


本工具还支持实时SVG Sprites合并,有时候一些小项目不想搭笨重的Node任务流,就可以试试这个敏捷轻便在线小工具。29R编程技术_踩坑日志_进阶指南 - 无知人生

四、结束语

这篇文章从周一写到周日,大部分时间就在产出最后那个SVG在线压缩工具上,压缩底层使用的是著名的SVG优化工具svgo(万级别Star项目,很强),所以压缩的质量大家不用担心。29R编程技术_踩坑日志_进阶指南 - 无知人生

工具页面还有个设置按钮,点击会展开参数配置列表,可以对几十个优化参数进行配置,按钮位置参见下图:29R编程技术_踩坑日志_进阶指南 - 无知人生

image.png29R编程技术_踩坑日志_进阶指南 - 无知人生

感谢LuLu UI提供UI界面支持,另外点击按钮直接下载SVG文件是纯前端实现的,有兴趣可以参考这篇文章:“JS前端创建html或json文件并浏览器导出下载”。29R编程技术_踩坑日志_进阶指南 - 无知人生


最后,对本文内容总结下:CSS中内联SVG的时候,没有使用Base64格式的任何理由,请使用部分转义的SVG原始代码代替,如何得到转义SVG代码,可以使用这个SVG在线压缩合并工具29R编程技术_踩坑日志_进阶指南 - 无知人生

哦,对了,好像微信小程序中Base64的SVG背景并不支持,但是直接SVG原始代码是支持的。29R编程技术_踩坑日志_进阶指南 - 无知人生

转自:https://www.zhangxinxu.com/wordpress/2018/08/css-svg-background-image-base64-encode/29R编程技术_踩坑日志_进阶指南 - 无知人生

很赞哦! () 有话说 ()

点击排行

站点信息

  • 建站时间:2018-10-24
  • 服务期限阿里云ECS 2027年到期
  • 主题模板:基于《今夕何夕》修改
  • 文章统计:188篇
  • 文章评论:27条
  • 文章阅读:2117次
  • 文章点赞:874次
  • 微信公众号:扫描二维码,关注我们