# 26 | 如何绘制带宽度的曲线? 你好,我是月影。 在可视化应用中,我们经常需要绘制一些带有特定宽度的曲线。比如说,在地理信息可视化中,我们会使用曲线来描绘路径,而在3D地球可视化中,我们会使用曲线来描述飞线、轮廓线等等。 在Canvas2D中,要绘制带宽度的曲线非常简单,我们直接设置上下文对象的lineWidth属性就行了。但在WebGL中,绘制带宽度的曲线是一个难点,很多开发者都在这一步被难住过。 那今天,我就来说说怎么用Canvas2D和WebGL分绘制曲线。我要特别强调一下,我们讲的曲线指广义的曲线,线段、折线和平滑曲线都包含在内。 ## 如何用Canvas2D绘制带宽度的曲线? 刚才我说了,用Canvas2D绘制曲线非常简单。这是为什么呢?因为Canvas2D提供了相应的API,能够绘制出不同宽度、具有特定**连线方式**和**线帽形状**的曲线。 这句话怎么理解呢?我们从两个关键词,“连线方式(lineJoin)”和“线帽形状(lineCap)”入手理解。 我们知道,曲线是由线段连接而成的,两个线段中间转折的部分,就是lineJoin。如果线宽只有一个像素,那么连接处没有什么不同的形式,就是直接连接。但如果线宽超过一个像素,那么连接处的缺口,就会有不同的填充方式,而这些不同的填充方式,就对应了不同的lineJoin。 比如说,你可以看我给出的这张图,上面就显示了四种不同的lineJoin。其中,miter是尖角,round是圆角,bevel是斜角,none是不添加lineJoin。很好理解,我就不多说了 [![](https://static001.geekbang.org/resource/image/45/9c/458c1d4c49519c2ac897fe89397a0b9c.jpeg "4种不同的lineJoin")](https://mapserver.org/mapfile/symbology/construction.html) 说完了lineJoin,那什么是lineCap呢?lineCap就是指曲线头尾部的形状,它有三种类型。第一种是square,方形线帽,它会在线段的头尾端延长线宽的一半。第二种round也叫圆弧线帽,它会在头尾端延长一个半圆。第三种是butt,就是不添加线帽。 [![](https://static001.geekbang.org/resource/image/60/c8/60cfd2020c014d88b7ac4b3b69e8e7c8.jpeg "3种不同的lineCap")](http://falcon80.com/HTMLCanvas/Attributes/lineCap.html) 理解了这两个关键词之后,我们接着尝试一下,怎么在Canvas的上下文中,通过设置lineJoin和lineCap属性,来实现不同的曲线效果。 首先,我们要实现一个drawPolyline函数。这个函数非常简单,就是设置lineWidth、lingJoin、lineCap,然后根据points数据的内容设置绘图指令执行绘制。 ``` function drawPolyline(context, points, {lineWidth = 1, lineJoin = 'miter', lineCap = 'butt'} = {}) { context.lineWidth = lineWidth; context.lineJoin = lineJoin; context.lineCap = lineCap; context.beginPath(); context.moveTo(...points[0]); for(let i = 1; i < points.length; i++) { context.lineTo(...points[i]); } context.stroke(); } ``` 在设置lingJoin、lineCap时候,我们要注意,Canvas2D的lineJoin只支持miter、bevel和round,不支持none。lineCap支持butt、square和round。 接着,我们就可以执行JavaScript代码绘制曲线了。比如,我们绘制两条线,一条宽度为10个像素的红线,另一条宽度为1个像素的蓝线,具体的代码: ``` const canvas = document.querySelector('canvas'); const ctx = canvas.getContext('2d'); const points = [ [100, 100], [100, 200], [200, 150], [300, 200], [300, 100], ]; ctx.strokeStyle = 'red'; drawPolyline(ctx, points, {lineWidth: 10}); ctx.strokeStyle = 'blue'; drawPolyline(ctx, points); ``` 因为我们把连接设置成miter、线帽设置成了butt,所以我们绘制出来的曲线,是尖角并且不带线帽的。 ![](https://static001.geekbang.org/resource/image/d9/eb/d9ed91c48f6a44bcaa26dbe2yya4d5eb.jpeg) 其实,我们还可以修改lineJoins和lineCap参数。比如,我们将线帽设为圆的,连接设为斜角。除此之外,你还可以尝试不同的组合,我就不再举例了。 ``` ctx.strokeStyle = 'red'; drawPolyline(ctx, points, {lineWidth: 10, lineCap: 'round', lineJoin: 'bevel'}); ``` ![](https://static001.geekbang.org/resource/image/fe/0e/fe56yy2032acef591b6051328331d10e.jpeg) 除了lineJoin和lineCap外,我们还可以设置Canvas2D上下文的miterLimit属性,来改变lineJoin等于miter时的连线形式,miterLimit属性等于miter和线宽的最大比值。当我们把lineJoin设置成miter的时候,miterLimit属性就会限制尖角的最大值。 那具体会产生什么效果呢?我们可以先修改drawPolyline代码添加miterLimit。代码如下: ``` function drawPolyline(context, points, {lineWidth = 1, lineJoin = 'miter', lineCap = 'butt', miterLimit = 10} = {}) { context.lineWidth = lineWidth; context.lineJoin = lineJoin; context.lineCap = lineCap; context.miterLimit = miterLimit; context.beginPath(); context.moveTo(...points[0]); for(let i = 1; i < points.length; i++) { context.lineTo(...points[i]); } context.stroke(); } ``` 然后,我们修改参数,把miterLimit:设置为1.5: ``` ctx.strokeStyle = 'red'; drawPolyline(ctx, points, {lineWidth: 10, lineCap: 'round', lineJoin: 'miter', miterLimit: 1.5}); ``` ![](https://static001.geekbang.org/resource/image/fe/0e/fe56yy2032acef591b6051328331d10e.jpeg) 你会发现,这样渲染出来的图形,它两侧的转角由于超过了miterLimit限制,所以表现为斜角,而中间的转角因为没有超过miterLimit限制,所以是尖角。 总的来说,Canvas2D绘制曲线的方法很简单,只要我们调用对应的API就可以了。但用WebGL来绘制同样的曲线会非常麻烦。在详细讲解之前,我希望你先记住lineJoin、lineCap以及miterLimit这些属性,在WebGL中我们需要自己去实现它们。接下来,我们一起来看一下WebGL中是怎么做的。 ## 如何用WebGL绘制带宽度的曲线 我们先从绘制宽度为1的曲线开始。因为WebGL本身就支持线段类的图元,所以我们直接用图元就能绘制出宽度为1的曲线。 下面,我结合代码来说说具体的绘制过程。与Canvas2D类似,我们直接设置position顶点坐标,然后设置mode为gl.LINE\_STRIP。这里的LINE\_STRIP是一种图元类型,表示以首尾连接的线段方式绘制。这样,我们就可以得到宽度为1的折线了。具体的代码和效果如下所示: ``` import {Renderer, Program, Geometry, Transform, Mesh} from '../common/lib/ogl/index.mjs'; const vertex = ` attribute vec2 position; void main() { gl_PointSize = 10.0; float scale = 1.0 / 256.0; mat3 projectionMatrix = mat3( scale, 0, 0, 0, -scale, 0, -1, 1, 1 ); vec3 pos = projectionMatrix * vec3(position, 1); gl_Position = vec4(pos.xy, 0, 1); } `; const fragment = ` precision highp float; void main() { gl_FragColor = vec4(1, 0, 0, 1); } `; const canvas = document.querySelector('canvas'); const renderer = new Renderer({ canvas, width: 512, height: 512, }); const gl = renderer.gl; gl.clearColor(1, 1, 1, 1); const program = new Program(gl, { vertex, fragment, }); const geometry = new Geometry(gl, { position: {size: 2, data: new Float32Array( [ 100, 100, 100, 200, 200, 150, 300, 200, 300, 100, ], )}, }); const scene = new Transform(); const polyline = new Mesh(gl, {geometry, program, mode: gl.LINE_STRIP}); polyline.setParent(scene); renderer.render({scene}); ``` ![](https://static001.geekbang.org/resource/image/ef/b3/ef7cdd7a0df26a237592d229bec226b3.jpeg) 你可能会问,我们不能直接修改gl\_PointSize,来给折线设置宽度吗?很遗憾,这是不行的。因为gl\_PointSize只能影响gl.POINTS图元的显示,并不能对线段图元产生影响。 那我们该怎么让线的宽度大于1个像素呢? ## 通过挤压(extrude)曲线绘制有宽度的曲线 我们可以用一种挤压(Extrude)曲线的技术,通过将曲线的顶点沿法线方向向两侧移出,让1个像素的曲线变宽。 那挤压曲线要怎么做呢?我们先看一张示意图: ![](https://static001.geekbang.org/resource/image/39/67/396bc27a64ec8cb608734b63cf44ee67.jpeg "挤压线段") 如上图所示,黑色折线是原始的1个像素宽度的折线,蓝色虚线组成的是我们最终要生成的带宽度曲线,红色虚线是顶点移动的方向。因为折线两个端点的挤压只和一条线段的方向有关,而转角处顶点的挤压和相邻两条线段的方向都有关,所以顶点移动的方向,我们要分两种情况讨论。 首先,是折线的端点。假设线段的向量为(x, y),因为它移动方向和线段方向垂直,所以我们只要沿法线方向移动它就可以了。根据垂直向量的点积为0,我们很容易得出顶点的两个移动方向为(-y, x)和(y, -x)。如下图所示: ![](https://static001.geekbang.org/resource/image/48/12/482af6fd993dbf91f3635bddffdcd412.jpeg "折线端点挤压方向") 端点挤压方向确定了,接下来要确定转角的挤压方向了,我们还是看示意图。 ![](https://static001.geekbang.org/resource/image/fa/e3/fa6b04dabfba6aaca02a66dde1d743e3.jpeg "转角的挤压方向示意图") 如上图,我们假设有折线abc,b是转角。我们延长ab,就能得到一个单位向量v1,反向延长bc,可以得到另一个单位向量v2,那么挤压方向就是向量v1+v2的方向,以及相反的-(v1+v2)的方向。 现在我们得到了挤压方向,接下来就需要确定挤压向量的长度。 首先是折线端点的挤压长度,它等于lineWidth的一半。而转角的挤压长度就比较复杂了,我们需要再计算一下。 ![](https://static001.geekbang.org/resource/image/d2/82/d2be1cd10fe7cdd0ab5611c13ab56882.jpeg "计算转角挤压长度示意图") 绿色这条辅助线应该等于lineWidth的一半,而它又恰好是v1+v2在绿色这条向量方向的投影,所以,我们可以先用向量点积求出红色虚线和绿色虚线夹角的余弦值,然后用lineWidth的一半除以这个值,得到的就是挤压向量的长度了。 具体用JavaScript实现的代码如下所示: ``` function extrudePolyline(gl, points, {thickness = 10} = {}) { const halfThick = 0.5 * thickness; const innerSide = []; const outerSide = []; // 构建挤压顶点 for(let i = 1; i < points.length - 1; i++) { const v1 = (new Vec2()).sub(points[i], points[i - 1]).normalize(); const v2 = (new Vec2()).sub(points[i], points[i + 1]).normalize(); const v = (new Vec2()).add(v1, v2).normalize(); // 得到挤压方向 const norm = new Vec2(-v1.y, v1.x); // 法线方向 const cos = norm.dot(v); const len = halfThick / cos; if(i === 1) { // 起始点 const v0 = new Vec2(...norm).scale(halfThick); outerSide.push((new Vec2()).add(points[0], v0)); innerSide.push((new Vec2()).sub(points[0], v0)); } v.scale(len); outerSide.push((new Vec2()).add(points[i], v)); innerSide.push((new Vec2()).sub(points[i], v)); if(i === points.length - 2) { // 结束点 const norm2 = new Vec2(v2.y, -v2.x); const v0 = new Vec2(...norm2).scale(halfThick); outerSide.push((new Vec2()).add(points[points.length - 1], v0)); innerSide.push((new Vec2()).sub(points[points.length - 1], v0)); } } ... } ``` 在这段代码中,v1、v2是线段的延长线,v是挤压方向,我们计算法线方向与挤压方向的余弦值,就能算出挤压长度了。你还要注意,我们要把起始点和结束点这两个端点的挤压也给添加进去,也就是两个if条件中的处理逻辑。 这样一来,我们就把挤压之后的折线顶点坐标给计算出来了。向内和向外挤压的点现在分别保存在innerSide和outerSide数组中。 接下来,我们就要构建对应的Geometry对象,所以我们继续添加extrudePolyline函数的后半部分。 ``` function extrudePolyline(gl, points, {thickness = 10} = {}) ... const count = innerSide.length * 4 - 4; const position = new Float32Array(count * 2); const index = new Uint16Array(6 * count / 4); // 创建 geometry 对象 for(let i = 0; i < innerSide.length - 1; i++) { const a = innerSide[i], b = outerSide[i], c = innerSide[i + 1], d = outerSide[i + 1]; const offset = i * 4; index.set([offset, offset + 1, offset + 2, offset + 2, offset + 1, offset + 3], i * 6); position.set([...a, ...b, ...c, ...d], i * 8); } return new Geometry(gl, { position: {size: 2, data: position}, index: {data: index}, }); } ``` 这一步骤就非常简单了,我们根据innerSide和outerSide中的顶点来构建三角网格化的几何体顶点数据,最终返回Geometry对象。 ![](https://static001.geekbang.org/resource/image/1c/01/1c423ae467bce04f49a46a4fed376d01.jpeg "构建折线的顶点数据") 最后,我们只要调用extrudePolyline,传入折线顶点和宽度,然后用返回的Geometry对象来构建三角网格对象,将它渲染出来就可以了。 ``` const geometry = extrudePolyline(gl, points, {lineWidth: 10}); const scene = new Transform(); const polyline = new Mesh(gl, {geometry, program}); polyline.setParent(scene); renderer.render({scene}); ``` 我们最终渲染出来的效果如下图: ![](https://static001.geekbang.org/resource/image/7f/d4/7fb7607de6d1396ba92f53ac18e9acd4.jpeg) 这样,我们就在WebGL中实现了,与Canvas2D一样带宽度的曲线。 当然,这里我们只实现了最基础的带宽度曲线,它对应于Canvas2D中的lineJoin为miter,lineCap为butt的曲线。不过,想要实现lineJoins为bevel或round,lineCap为square或round的曲线,也不会太困难。我们可以基于extrudePolyline函数,对它进行扩展,计算出相应属性下对应的顶点就行了。因为基本原理是一样的,我就不详细说了,我把扩展的任务留给你作为课后练习。 ## 要点总结 这节课,我们讲了绘制带宽度曲线的方法。 首先,在Canvas2D中,绘制这样的曲线比较简单,我们直接通过API设置lineWidth即可。而且,Canvas2D还支持不同的lineJoin、lineCap设置以及miterLimit设置。 在WebGL中,绘制带宽度的曲线则比较麻烦,因为没有现成的API可以使用。这个时候,我们可以使用挤压曲线的技术来得到带宽度的曲线,挤压曲线的具体步骤可以总结为三步: 1. 确定端点和转角的挤压方向,端点可以沿线段的法线挤压,转角则通过两条线段延长线的单位向量求和的方式获得。 2. 确定端点和转角挤压的长度,端点两个方向的挤压长度是线宽lineWidth的一半。求转角挤压长度的时候,我们要先计算方向向量和线段法线的余弦,然后将线宽lineWidth的一半除以我们计算出的余弦值。 3. 由步骤1、2计算出顶点后,我们构建三角网格化的几何体顶点数据,然后将Geometry对象返回。 这样,我们就可以用WebGL绘制出有宽度的曲线了。 ## 小试牛刀 1. 你能修改extrudePolyline函数,让它支持lineCap为square和round吗?或者让它支持lineJoin为round吗? 2. 我想让你试着修改一下extrudePolyline函数,让它支持lineJoin为bevel,以及miterLimit。并且,当lineJoin为miter的时候,如果转角挤压长度超过了miterLimit,我们就按照bevel处理向外的挤压。 那通过今天的学习,你是不是已经学会绘制带宽度曲线的方法。那不妨就把这节课分享给你的朋友,也帮助他解决这个难题吧。好了,今天的内容就到这里了,我们下节课再见 * * * ## 源码 课程完整示例代码详见[GitHub仓库](https://github.com/akira-cn/graphics/tree/master/polyline-curve)