SVG 图像入门

1- 概述

SVG(可缩放矢量图)是一种基于 XML 语法的图像格式,它的图像是通过对图形的形状描述来绘制的,因此它本质上是文本文件,体积较小,且不管放大多少倍都不会失真。

1.1- SVG 的特点

  • 基于 XML 语法:SVG 文件是用 XML 语法编写的,可以直接在文本编辑器中查看和编辑。
  • 文本文件:SVG 文件是纯文本文件,可以方便地进行版本控制和分享。
  • 体积较小:由于 SVG 文件是基于矢量的,因此相对于位图文件,体积更小。
  • 不会失真:SVG 图像可以无限放大,不会出现像素化或失真现象。

1.2- SVG 的应用

  • 网页中的图像:SVG 图像广泛应用于网页设计,特别是在需要高分辨率和可缩放的图像时。
  • 印刷材料中的图像:SVG 图像适合用于印刷材料,因为它可以保证图像在任何分辨率下都能保持高质量。
  • 动画和交互:SVG 支持内嵌的动画效果和交互功能,非常适合用于创建动态和交互式的网页内容。

2- 语法

2.1- <svg>标签

<svg>标签是 SVG 代码的根元素,它定义了一个 SVG 图像。

<svg width="100%" height="100%">
  <circle id="mycircle" cx="50" cy="50" r="50" />
</svg>

2.2- <circle>标签

<circle>标签用于绘制圆形。

<svg width="300" height="180">
  <circle cx="30" cy="50" r="25" />
  <circle cx="90" cy="50" r="25" class="red" />
  <circle cx="150" cy="50" r="25" class="fancy" />
</svg>

2.3- <line>标签

<line>标签用来绘制直线。

<svg width="300" height="180">
  <line x1="0" y1="0" x2="200" y2="0" style="stroke:rgb(0,0,0);stroke-width:5" />
</svg>

2.4- <polyline>标签

<polyline>标签用于绘制一根折线。

<svg width="300" height="180">
  <polyline points="3,3 30,28 3,53" fill="none" stroke="black" />
</svg>

2.5- <rect>标签

<rect>标签用于绘制矩形。

<svg width="300" height="180">
  <rect x="0" y="0" height="100" width="200" style="stroke: #70d5dd; fill: #dd524b" />
</svg>

2.6- <ellipse>标签

<ellipse>标签用于绘制椭圆。

<svg width="300" height="180">
  <ellipse cx="60" cy="60" ry="40" rx="20" stroke="black" stroke-width="5" fill="silver"/>
</svg>

2.7- <polygon>标签

<polygon>标签用于绘制多边形。

<svg width="300" height="180">
  <polygon fill="green" stroke="orange" stroke-width="1" points="0,0 100,0 100,100 0,100 0,0"/>
</svg>

2.8- <path>标签

<path>标签用于绘制路径。

<svg width="300" height="180">
  <path d="M 18,3 L 46,3 L 46,40 L 61,40 L 32,68 L 3,40 L 18,40 Z" fill="blue" stroke="black" stroke-width="1" />
</svg>

2.9- <text>标签

<text>标签用于绘制文本。

<svg width="300" height="180">
  <text x="50" y="25">Hello World</text>
</svg>

2.10- <use>标签

<use>标签用于复制一个形状。

<svg viewBox="0 0 30 10" xmlns="http://www.w3.org/2000/svg">
  <circle id="myCircle" cx="5" cy="5" r="4"/>
  <use href="#myCircle" x="10" y="0" fill="blue" />
  <use href="#myCircle" x="20" y="0" fill="white" stroke="blue" />
</svg>

2.11- <g>标签

<g>标签用于将多个形状组成一个组(group),方便复用。

<svg width="300" height="100">
  <g id="myCircle">
    <text x="25" y="20">圆形</text>
    <circle cx="50" cy="50" r="20"/>
  </g>
  <use href="#myCircle" x="100" y="0" fill="blue" />
  <use href="#myCircle" x="200" y="0" fill="white" stroke="blue" />
</svg>

2.12- <defs>标签

<defs>标签用于自定义形状,它内部的代码不会显示,仅供引用。

<svg width="300" height="100">
  <defs>
    <g id="myCircle">
      <text x="25" y="20">圆形</text>
      <circle cx="50" cy="50" r="20"/>
    </g>
  </defs>
  <use href="#myCircle" x="0" y="0" />
  <use href="#myCircle" x="100" y="0" fill="blue" />
  <use href="#myCircle" x="200" y="0" fill="white" stroke="blue" />
</svg>

2.13- <pattern>标签

<pattern>标签用于自定义一个形状,该形状可以被引用来平铺一个区域。

<svg width="500" height="500">
  <defs>
    <pattern id="dots" x="0" y="0" width="100" height="100" patternUnits="userSpaceOnUse">
      <circle fill="#bee9e8" cx="50" cy="50" r="35" />
    </pattern>
  </defs>
  <rect x="0" y="0" width="100%" height="100%" fill="url(#dots)" />
</svg>

2.14- <image>标签

<image>标签用于插入图片文件。

<svg viewBox="0 0 100 100" width="100" height="100">
  <image xlink:href="path/to/image.jpg" width="50%" height="50%"/>
</svg>

2.15- <animate>标签

<animate>标签用于产生动画效果。

<svg width="500px" height="500px">
  <rect x="0" y="0" width="100" height="100" fill="#feac5e">
    <animate attributeName="x" from="0" to="500" dur="2s" repeatCount="indefinite" />
  </rect>
</svg>

2.16- <animateTransform>标签

<animateTransform>标签用于对 CSS 的 transform 属性进行动画效果。

<svg width="500px" height="500px">
  <rect x="250" y="250" width="50" height="50" fill="#4bc0c8">
    <animateTransform attributeName="transform" type="rotate" begin="0s" dur="10s" from="0 200 200" to="360 400 400" repeatCount="indefinite" />
  </rect>
</svg>

3- JavaScript 操作

3.1- DOM 操作

如果 SVG 代码直接写在 HTML 网页之中,它就成为网页 DOM 的一部分,可以直接用 DOM 操作。

<svg
  id="mysvg"
  xmlns="http://www.w3.org/2000/svg"
  viewBox="0 0 800 600"
  preserveAspectRatio="xMidYMid meet"
>
  <circle id="mycircle" cx="400" cy="300" r="50" />
</svg>

3.2- 获取 SVG DOM

使用 <object><iframe><embed> 标签插入 SVG 文件,可以获取 SVG DOM。

var svgObject = document.getElementById('object').contentDocument;
var svgIframe = document.getElementById('iframe').contentDocument;
var svgEmbed = document.getElementById('embed').getSVGDocument();

3.3- 读取 SVG 源码

由于 SVG 文件就是一段 XML 文本,因此可以通过读取 XML 代码的方式,读取 SVG 源码。

var svgString = new XMLSerializer()
  .serializeToString(document.querySelector('svg'));

3.4- SVG 图像转为 Canvas 图像

首先,需要新建一个 Image 对象,将 SVG 图像指定到该 Image 对象的 src 属性。

var img = new Image();
var svg = new Blob([svgString], {type: "image/svg+xml;charset=utf-8"});
var DOMURL = self.URL || self.webkitURL || self;
var url = DOMURL.createObjectURL(svg);
img.src = url;

然后,当图像加载完成后,再将它绘制到 <canvas> 元素。

img.onload = function () {
  var canvas = document.getElementById('canvas');
  var ctx = canvas.getContext('2d');
  ctx.drawImage(img, 0, 0);
};

4- 实例:折线图

下面将一张数据表格画成折线图。

Date Amount
2014-01-01 $10
2014-02-01 $20
2014-03-01 $40
2014-04-01 $80
<svg width="350" height="160">
  <g class="layer" transform="translate(60,10)">
    <circle r="5" cx="0"   cy="105" />
    <circle r="5" cx="90"  cy="90"  />
    <circle r="5" cx="180" cy="60"  />
    <circle r="5" cx="270" cy="0"   />
    <polyline points="0,105 90,90 180,60 270,0" stroke="black" stroke-width="2" fill="none" />
    <g class="y axis">
      <line x1="0" y1="0" x2="0" y2="120" />
      <text x="-40" y="105" dy="5">$10</text>
      <text x="-40" y="0"   dy="5">$80</text>
    </g>
    <g class="x axis" transform="translate(0, 120)">
      <line x1="0" y1="0" x2="270" y2="0" />
      <text x="-30"   y="20">January 2014</text>
      <text x="240" y="20">April</text>
    </g>
  </g>
</svg>

4.1- 解释

  • <circle>标签:用于绘制每个数据点。
  • <polyline>标签:用于绘制连接数据点的折线。
  • <line>标签:用于绘制坐标轴。
  • <text>标签:用于添加标签和注释。

通过这些标签的组合,可以创建出一个简单的折线图。你可以根据需要调整样式和数据点的位置,以适应不同的数据和设计要求。

希望这些改进能帮助你更好地理解和使用 SVG。