`

从 Web 图标演进历史看最佳实践

阅读更多

一、使用独立图片

在过去有很长一段时间,前端是通过引入图片来承载图标。在没有 CSS 支持的时代,用 <img> 标签引入图标图片是唯一的可能。

 
<a href="/contact.html">  <img src="mail.jpg" alt="email"></a>
 
 
 
复制代码
 

到了 CSS 支持背景图以后,人们开始使用 background-image 来引入一个个小图片,但本质上没有改变每个图标都使用单独图片的问题。

显然,这样的方式在有很多图标的网页中将发起很多 HTTP 请求,占用浏览器的并行请求数量,导致整体加载时间缓慢,体验很差。对于有些鼠标悬浮后切换图标的设计,这种方式还会出现第一次切换时需要等待图标加载的问题。(但是令人沮丧的是,直到现在还有网站依然保留着这样的方式。)

二、CSS Sprite

后来在大约本世纪初的头几年,人们找到了一种新的技巧:通过将图片合并技术(image sprite)引入前端,将数量众多的图标图片进行巧妙拼合,并且在样式中通过 background-position 来通过不同位置匹配不同的图标进行显示。例如:

 
.toolbtn {  background: url(icons.png);  display: inline-block;  height: 20px;  width: 20px;}
#btn1 {  background-position: -20px 0px;}#btn2 {  background-position: -40px 0px;}
 
 
 
复制代码
 

虽然这种方式相较于每个小图标一个图片文件,只会发起一次 HTTP 请求,对性能更加友好,但是依然有着如下问题:

  • 拼合后的图片非常难以维护,需要手动精心调整。虽然也有一些自动生成“雪碧图”的工具,但由于 background-position 这种方式的限制,生成逻辑无法保证灵活适应各种可能的使用场景。

  • 图片来自 https://www.smashingmagazine.com/2012/04/css-sprites-revisited/

  • 当一个项目图标很多时,图片会在整体下载完以后才显示,可能会导致一段较长的时间内所有图标都无法显示。同时由于高昂的维护成本,很难做到按需加载图标,往往整站的图标都会全部合并到同一个“雪碧图”中。

  • 图标颜色是确定的,无法在前端根据内容上下文灵活调整图标的颜色。

  • 图片尺寸是固定的,进行缩放后很难保证图标的显示效果。

在这个时代,设计师和工程师协作的模式一般来说都是设计师将设计好的图标文件交付给工程师,由工程师来通过图片编辑工具或者一些雪碧图生成器来维护拼合后的图片,效率和可维护性都非常堪忧。

三、字体图标的崛起

由于图标从某种程度上来看可以被视为“象形文字”,所以当 CSS 开始支持 @font-face 引入 web font,人们立刻想到了用它来载入、显示图标。从 2012 年至今,提供大量免费图标的 FontAwesome 就取得了很大的成功(后来开始商业化的 FontAwesome 5 的甚至为他们在 Kickstarter 上筹集到了一百万美金),各种字体图标平台也层出不穷。阿里的 iconfont.cn 平台从多年前开始就已经成为国内最受欢迎的图标托管、共享、管理平台。可以说字体图标时至今日还是最热门的 web 图标方案之一。

字体图标的原理非常简单,通过占用一些 Unicode 字符编码(通常是私人使用区,U+E000-U+F8FFU+F0000-U+FFFFD 以及 U+100000-U+10FFFD 范围内)并为其绘制字形,同时生成好一堆预定义的图标名 class name,通过 web font 的方式加载资源,通过对应的 class name 来引用图标。由于各个浏览器对 web font 支持的字体格式兼容性有差异,往往需要生成多个格式的字体供浏览器进行选择性加载:

 
/* iconfont.cn 生成的样式文件大致如下: */@font-face {  font-family: "iconfont";  src: url('iconfont.eot'); /* IE9 */  src: url('iconfont.eot#iefix') format('embedded-opentype'), /* IE6-IE8 */  url('iconfont.woff2') format('woff2'),  url('iconfont.woff') format('woff'),  url('iconfont.ttf') format('truetype'), /* chrome, firefox, opera, Safari, Android, iOS 4.2+ */  url('iconfont.svg#iconfont') format('svg'); /* iOS 4.1- */}
.iconfont {  font-family: "iconfont" !important;}
.icon-flag:before {  content: "\e233";}
 
 
 
复制代码
 

在 HTML 中使用:

 
<i class="icon-flag"></i>
 
 
 
复制代码
 

字体图标虽然也很难维护,但是相比“雪碧图”还是有不少明显的优势:

  • 基于轮廓字体格式的字体图标是通过贝塞尔曲线描述的,可以任意伸缩并且保持显示效果不失真,这在移动端尤为重要。

  • 字体可以轻易地使用 CSS 设置颜色。

但我们可以看出,这个方案对使用者的工程能力已经有所要求。虽然在这个时代,多数业内前端团队已经都有了初步的工程化能力,开始使用诸如 Grunt/Gulp 甚至 webpack 等工具,基于 Node + npm 去定制各自团队的工程化方案了,但是编排每个图标的 Unicode 编码、生成对应的 CSS 代码就已经有比较大的工作量,更别说生成这么多格式的字体文件,普通工程师根本无从下手。这也是 iconfont.cn 吸引大量用户的重要原因。重度依赖第三方平台,自己建设成本又比较高,使得图标的可维护性依然存在一定的痛点。

另外,虽然字体图标解决了一些“雪碧图”的体验问题,它也带来了一些新问题:

  • 字体文件加载需要时间,在文件加载完成前,图标是无法显示的,内容就很容易发生闪烁。在某些浏览器下,处于私有使用区的图标在默认字体下甚至会显示为一个方块字符。

  • 图片来自 https://github.blog/2016-02-22-delivering-octicons-with-svg/

    这一点实际上和“雪碧图”有着很大的共同点。虽然我们可以使用 data URI 来将资源内联,事实上有很长时间我们也的确使用过将图片或者字体通过 data URI 编码后内联到 HTML 的方式来避免这个加载的时间差,但是编码本身会增加内容 1/3 左右的尺寸,实际上只能算是一种取舍和妥协。更别说字体图标需要生成如此多格式的字体,内联到 HTML 网页性能将大打折扣。

  • 可访问性问题:对于患有视力障碍使用读屏器的用户,由于字体图标实际由字符承载,无论字体是否加载完毕,读屏器都无法正常朗读其内容,在默认的状态下甚至会读出“unpronounceable”这样不符合预期的内容,可以想象如果一个网页大量使用字体图标却没有逐个标注 aria-hidden 这样的语义标记,会对读屏器用户产生多大的困惑。

四、SVG 图标

SVG 天生就带有可伸缩(SVG 中的 S)特性,非常适合用来实现图标。同时,SVG 是文本文件,同时诸多支持矢量编辑的设计工具都支持通过 SVG 导出,设计师可以直接交付给工程师使用,也不再需要生成字体文件,大大缓解了可维护性上的痛点。但如果将它当成图片,通过 <img> 或 CSS background-image 来引入,仅仅有这些优势还不足以撼动图标字体的地位。

4.1 内联 SVG

SVG 的真正强大之处在于,当将其内联入 HTML 内容,那么它的文档模型将可以被该页面的 JS/CSS 访问和操作。这为 web 图标开启了新的篇章:

  • 可以通过 CSS 控制图标的颜色甚至具体样式,使得受业务逻辑控制的动画图标成为可能。

  • 在显示效果上,字体图标由于本质上被视为文本,将受到浏览器的文字抗锯齿算法的影响,在特定操作系统、浏览器、字体设置下视觉效果可能会不那么“保真”。而 SVG 被视为图片进行渲染,不会受文字抗锯齿算法影响,渲染效果更加原汁原味。

  • SVG 内联入 HTML 内容并不需要进行编码,重复的 SVG 内容也是对 gzip 友好的,对 HTML 加载速度的性能损耗很小。

  • 不需要发起资源请求,可以随着 HTML 内容进行流式加载和渲染,不会产生任何闪动的体验问题。

  • 图标加载可以做到完全按需,当前页面没有用到的图标都不会输出。

  • SVG 可以通过 <title> 元素标记内容,对读屏器友好。

相比于通过图片资源加载或者图标字体,只有一个劣势:

  • 图标成为 HTML 内容的一部分,不再能在 CSS 中指定需要使用的图标了。当然这一点从我们的实践中来看,并不构成很大的阻碍。

虽然内联 SVG 有很多优势,但是在这个阶段,在开发时使用它们却不像字体图标那么简单直接(引入一个 CSS,前端就能任意使用),需要对工程有一定侵入性的处理。GitHub 在 2016 年全面启用了内联 SVG 的方案,他们的技术栈是 Ruby 的后端渲染,通过服务端脚本定义的 helper 函数来进行图标字体的调用:

 
<%= octicon(:symbol => "plus") %>
 
 
 
复制代码
 

输出:

 
<svg aria-hidden="true" class="octicon octicon-plus" width="12" height="16" role="img" version="1.1" viewBox="0 0 12 16">    <path d="M12 9H7v5H5V9H0V7h5V2h2v5h5v2z"></path></svg>
 
 
 
复制代码
 

4.2 SVG Sprite

由于 SVG 支持一个 <use> 元素,可以从内联的 SVG 中选取特定内容出来作为独立的 SVG 进行显示,所以人们受 CSS sprite 的启发,也设计了一个 SVG sprite 方案。引入整个 SVG sprite 的资源仅需要内联一个 <svg> 元素:

 
<svg>  <defs>    <symbol id="shape-icon-1">      <!-- icon paths and shapes -->    <symbol>    <symbol id="shape-icon-2">      <!-- icon paths and shapes -->    <symbol>    <!-- etc -->  </defs></svg>
 
 
 
复制代码
 

使用时:

 
<svg viewBox="0 0 16 16" class="icon">  <use xlink:href="#shape-icon-1"></use></svg>
 
 
 
复制代码
 

同时,也有不少基于 Grunt/Gulp/webpack 的构建方案,来快速生成 SVG sprite。

这种方式主要的问题在于:

  • 不容易按需引入图标。

  • 在各个场景使用时比较繁琐。

五、前端组件框架的时代

终于到了我们现在所处的时代,这是一个 web 端渲染逻辑被移到前端,前端工程方向被组件化框架主导的时代。在使用 React/Vue/Angular/Svelte/…… 等各种框架的过程中,我们已经习惯于将视图逻辑通过组件进行拆解和复用。那么我们很自然地就可以通过设计图标组件来对底层方案进行一层封装,暴露给前端更简单直接的 API 来使用图标。要注意的是,这并没有在根本上改变 web 图标渲染的方式,底层依然是基于前文提到的各种方案。在不使用这些视图层框架的项目中,我们依然仰赖使用上述 low-level 的实现来进行开发。

当然,从各方面综合比较,封装内联 SVG 应该是当前最佳的选择。上文 GitHub 后端 helper 的方案对应当前前端的技术方案,实际上就是基于内联 SVG 的图标组件。npm 上目前也有很多基于各个组件框架开发的图标组件,包括 FontAwesome 都已经内置了 SVG、React/Vue 组件等更现代化的方案。

既然体验问题已经由内联 SVG 得到了比较好的解决,那么在这个阶段我们就有更多的精力去更多地考虑研发效能、一致性、开发体验的问题了。从我们在百度内部以往的实践中来看,存在这如下的一些问题:

  • 工作流程缺乏最佳实践,由于长期各个团队有着较为独立的技术演变,使用的 web 图标方案并不统一。

  • 整个大体系下跨团队的设计师并没有很好地共享图标资源,存在一定的重复设计。

  • 有图标组件库,但是图标有限,业务需要新增图标时设计师往往还是将图标线下交付给工程师,前端通过一些类似 svg-icon-loader 的方案将图标引入项目,但方案往往各不相同。一旦引入这样的流程,相当于给图标在特定项目中新增了一个 fork 版本,日后想做设计风格的统一调整就需要业务跟进修改,成本很高。

  • 针对 SVG 图标组件,我们没有一个类似 iconfont.cn 的平台进行流程上的收拢,也没有自动化的代码包导出、发布能力。

理想情况下,我们希望达成如下目标:

  • 图标设计师维护图标源文件,发布以后没有任何人工干预造成流程分叉,有一个固定的图标库平台提供 single source of truth。

  • 每个团队能根据自身技术栈,选择需要导出的组件实现类型(React/Vue/San/...)。

  • 图标组件库中的图标数据会被自动优化、压缩。

  • 图标组件库应该是可以跟随图标库的数据更新升级的。

目前我们在推进百度设计语言系统的过程中,和工程效能团队一起,设计了如下整体方案:

图标平台整体流程

5.1 图标管理平台

这个平台可以视为是一个简单的图标 CMS,可以创建/管理图标库,图标设计师负责来在其中添加、管理图标。在完成数据的更新后,可以选择发布当前图标输出到 API。这个 API 返回图标库中图标的图形数据(SVG 源文件)和元数据,在整个流程中主要有两个消费者:给设计团队使用的 Sketch 插件,以及前端的编译/发布服务。我们允许图标库发布时通过 webhook 配置需要通知的编译服务,所以有必要的话,不同的使用方也可以选择自己自定义整套编译发布的流程。

5.2 Sketch 插件

我们给设计团队提供了联通图标管理平台的 Sketch 插件,设计师可以在插件中快速搜索需要的图标进行使用。通过我们的插件导出在线标注稿后,标注稿上就会自动标注图标在图标平台中的唯一标识符,这也是我们用来生成图标组件时用的标识符,前端工程师通过它就能直接从图标组件包中引入对应的图标组件。

5.3 优化/编译/发布服务

这个服务在图标库 API 触发更新时主要做了三件事:

  1. 优化。从 API 读取图标数据,并且将源文件通过 SVGO 进行初步优化。由于我们希望图标组件内联到 HTML 以后可以通过 CSS 灵活修改颜色,所以对于常见的单色图标,我们需要去除所有硬编码的颜色,在有必要时设置为 currentColor。在这一步我们通过 svgson 遍历 SVG 元素处理相关逻辑。

  2. 编译。得到了优化过的图标数据,我们需要根据他们来生成我们的图标组件包。在这里我们提供了多个框架的组件包模板,每个模板中都已经提供了对应各自框架的图标组件工厂函数,只需要通过脚本在模板中注入图标数据,即可根据平台数据灵活生成各个业务所需要的组件包。

  3. 发布。根据在 webhook 回调路径中的配置,我们可以指定需要发布的包的名称,描述等信息。版本号的逻辑也比较简单:

  • 删除/改名图标:major + 1

  • 新增图标:minor + 1

  • 修改图标内容:patch + 1

5.4 图标包模板

编译服务对包模板(boilerplate)仅有的约定是:

  1. 编译服务会在特定目录输出图标数据。

  2. 编译服务会依次调用特定的 npm script。

模板提供者需要提供图标组件的具体实现,以及将图标数据转换为前端代码的构建脚本。如果没有特殊的需求,直接使用我们提供的 React/Vue 等框架下的组件模板,就可以获得高质量的前端图标组件实现了。

通过编译服务发布完成以后,前端工程师只需要知道:1. 使用的图标来自哪个 npm 包 2. 这个图标叫什么名字,即可快速在前端项目中引入图标。同时,整个流程保证了设计师产出的设计稿、前端实现的一致,并且可以从图标平台中心化地控制升级。

 

from :https://xie.infoq.cn/article/87bb7aa279929310d063d1f67

分享到:
评论

相关推荐

    付钱拉金融云系统架构演进和最佳实践.pdf

    付钱拉云系统架构演进和最佳实践.pdf

    数据库分库分表的技术演进与最佳实践.docx

    本文将从技术演进和最佳实践两个方面来讨论数据库分库分表的技术演进和最佳实践。 一、 Why Not NoSQL/NewSQL? NoSQL/NewSQL 数据库的出现是为了解决传统关系数据库的局限性,但是在实际应用中,它们也存在一些...

    藏经阁-系统架构演进和最佳实践.pdf

    藏经阁-系统架构演进和最佳实践.pdf 系统架构演进和最佳实践是阿里云的技术分享,主要讲述了系统架构的演进过程和最佳实践。该分享分为四个部分:业务模型、架构演进过程、最佳实践和继续前进。 业务模型 业务...

    基于技术演进的Java web程序设计教学实践.pdf

    基于技术演进的Java web程序设计教学实践 本文主要讨论了基于技术演进的Java web程序设计教学实践,旨在提高学生对知识的理解能力,增加学生的学习兴趣,强化学生的动手意愿。 Java web程序设计是软件工程专业的...

    云原生技术架构与实践.pdf

    - 微服务架构的设计原则与最佳实践,包括服务拆分策略、服务间通信机制等。 - 中间件的选型与实践,例如消息队列RocketMQ、配置中心Nacos等。 3. **Serverless技术与实践** - Serverless的基础概念及其优势,...

    集团企业数据中心向云演进的策略与实践报告

    一年多来,国药集团协同神州信息基于云计算的理念和关键技术开展了对企业数据中心进行改造的积极探索,形成了一些企业数据中心向云演进的策略、实践。本文就这一专题进行总结和论述,跟大家进行探讨、分享和交流。

    [白皮书]LTE—UMTS长期演进理论与实践 中英文详细书签版

    《LTE—UMTS长期演进理论与实践》是一本深入探讨4G移动通信技术的权威著作,涵盖了LTE(Long Term Evolution)系统从概念到实际应用的全面知识。这本书旨在为读者提供一个理解LTE网络架构、协议栈、操作原理以及与...

    关于安全防御体系演进与零信任最佳实践之SDP的分析说明.zip

    关于安全防御体系演进与零信任最佳实践之SDP的分析说明.zip

    安全防御体系演进与零信任最佳实践之SDP.pdf

    综上所述,从OSI安全模型到CARTA模型,再到零信任和SDP的介绍,文章详细梳理了网络安全模型的发展演进和零信任的最佳实践。这些知识不仅丰富了我们对网络安全的认识,也为构建和实施现代网络安全防御体系提供了重要...

    微众银行网络架构演进及运维实践.pdf

    微众银行网络架构演进及运维实践.pdf

    NFS最佳实践.pdf

    本文档是一份关于NETAPP存储系统中NFS配置应用最佳实践的总结,涵盖了从架构设计、服务启动、性能优化到管理策略等关键实践。 首先,文档讲述了传统存储面临的一些业务挑战,并引出Clustered Data ONTAP(集群数据...

    SOA最佳实践

    Service-Oriented Architecture (SOA) 是一种...随着技术的进步和业务需求的变化,SOA最佳实践将持续演进,但核心目标始终是提升软件的灵活性、可维护性和业务价值。理解和应用这些最佳实践对于构建成功的SOA至关重要。

    饿了么推荐算法演进及在线学习实践(24页).pdf

    文章从推荐业务背景、算法演进路线、在线学习实践三个方面进行探讨。 推荐业务背景 推荐业务背景是指 ele.me 饿了么平台上的推荐业务,旨在提高用户体验和订单覆盖率。推荐产品形态包括搜索词营销推荐、类目入口...

    国产数据库发展演进历史.docx

    国产数据库发展演进历史.docx

    数据架构设计与实践MySQL在高可用演进之路V2.pdf

    【数据架构设计与实践:MySQL在高可用演进之路】主要探讨了如何通过不同技术手段提升MySQL数据库的高可用性,确保系统持续稳定地提供服务。高可用性(High Availability,HA)是衡量系统无中断运行能力的标准,通常...

    Web应用安全演进-从WAF到WAAP.pdf

    概述:Web应用安全演进-从WAF到WAAP.pdf文件主要讲述了Web应用安全的演进,从传统的WAF(Web Application Firewall)到WAAP(Web Application & API Protection)的架构演进。该文件还详细介绍了Web应用安全的挑战、...

    Unity3D脚本:修改webplayer发布后logo图标的方法1

    这篇内容将介绍如何通过修改Unity3D脚本来自定义Web Player发布的Logo图标。 首先,我们要明确的是,Unity3D提供了两种方法来替换Web Player加载时显示的Logo: 1. **方法一:通过JavaScript注入** 在HTML页面中...

Global site tag (gtag.js) - Google Analytics