引言

最近我在学可视化的东西,借此来巩固一下学习的内容,向量运算是计算机图形学的基础,这个例子就是向量的一种应用,是利用向量来计算点到线段的距离,这个例子中可视化的展示采用Canvas2D来实现。

说起向量,当时一看到这个词,我是一种很模糊的记忆;这些是中学学的东西,感觉好像都还给老师了。然后又说起了向量的乘法,当看到点积、叉积这两个词,我才猛然想起点乘和叉乘;但整体上还是模模糊糊的,不太记得两者具体的定义了;就找资料快速过了一遍吧。

因为本文中不涉及向量的基础知识;如果有跟我一样遗忘的小伙伴,可以找点视频回忆一下,或者是找点资料看下。

题面

首先本次的例子中要获取两个值,一个是点到线段的距离,另一个是点到线段所在直线的距离。

假设存在一个线段AB,以及一个点C;则他们之前的位置可能有三种情况:

  • 点C在线段AB左侧

    left
  • 点C在线段AB的上方或下方

    top
  • 点C在线段AB的右侧

    right

在第一种和第三种情况下,点C到线段AB的距离为点C到点A或点B的距离,即向量AC或向量BC的长度。

在第二种情况下,点C到线段AB和到线段AB所在直线的距离是一样的,这个时候,我们就可以利用向量的乘法来解决这个距离的计算。

这个例子给的思路是利用向量的乘法,因为向量叉乘的几何意义就是平行四边形的面积,已知底边长度,也就是线段AB的长度,然后就可以得出点C到直线的距离;但因为要在页面上展示出来,所以我们需要求得点D的坐标。

d

思路

一开始我想的有点复杂,想要去求AB所在直线的函数方程,从而计算出点C是在直线的上方还是下方,虽然向量的叉乘我记得不太多了,但我依旧还记得,如果向量AB旋转到向量CD为顺时针,则向量AB叉乘向量CD的值就为正,如果是逆时针,就为负。

接着再利用叉乘和点乘,去计算点D的x坐标和y坐标;这其实有点把事情搞复杂了,另外还需要去特殊处理CD和X轴平行以及Y轴平行的特殊情况。

然后我看了别人的提示才反应过来,我们只要充分地利用向量的乘法就可以了,而不需要去求什么直线的函数方程,当然这也就不用考虑什么特殊情况。

d-2

由上图可知AD是AC在AB上的投影,然后我们知道投影可以通过点乘来求得,要求两个向量的点乘,有两种计算方式,一种是通过坐标来计算,另一种是通过向量的模和夹角来计算;分别对应以下两个公式:

  • AC · AB = AC.x * AB.x + AC.y * AB.y
  • AC · AB = |AC| * |AB| * cosθ

因为已知点A、点B和点C的坐标,所以我们可以利用以上两个公式计算点D的坐标。

具体实现

现在我们就来通过Canvas来实现以上效果。

HTML

首先我们在HTML中先放一个Canvas标签。

<canvas width="512" height="512"></canvas>

CSS

然后写一点简单的CSS样式。

canvas {
  margin: 0;
  width: 512px;
  height: 512px;
  border: 1px solid #eee;
}

JavaScript

最后我们来编写最重要的JavaScript代码。

这里预先定义了一个Vector2D的类用于表示二维向量。

/*
* 定义二维向量
* */
export default class Vector2D extends Array {
    constructor(x = 1, y = 0) {
        super(x, y);
    }
    get x() {
        return this[0];
    }
    set x(value) {
        this[0] = value;
    }
    get y() {
        return this[1];
    }
    set y(value) {
        this[1] = value;
    }
    // 获取向量的长度
    get len() {
        // x、y的平方和的平方根
        return Math.hypot(this.x, this.y);
    }
    // 获取向量与X轴的夹角
    get dir() {
        // 向量与X轴的夹角
        return Math.atan2(this.y, this.x);
    }
    // 复制向量
    copy() {
        return new Vector2D(this.x, this.y);
    }
    // 向量的加法
    add(v) {
        this.x += v.x;
        this.y += v.y;
        return this;
    }
    // 向量旋转
    rotate(rad) {
        const c = Math.cos(rad),
            s = Math.sin(rad);
        const [x, y] = this;

        this.x = x * c - y * s;
        this.y = x * s + y * c;

        return this;
    }
    scale(length) {
        this.x *= length;
        this.y *= length;

        return this;
    }
    // 向量的点乘
    dot(v) {
        return this.x * v.x + this.y * v.y;
    }
    // 向量的叉乘
    cross(v) {
        return this.x * v.y - v.x * this.y;
    }
    reverse() {
        return this.copy().scale(-1);
    }
    // 向量的减法
    minus(v) {
        return this.copy().add(v.reverse());
    }
    // 向量归一化
    normalize() {
        return this.copy().scale(1 / this.len);
    }
}

x和y分别是向量的坐标,len获取的是向量的长度、利用了Math对象上的方法,dot和cross方法分别对应的就是向量的点乘和叉乘。

接着就来编写功能代码。

  • 首先是获取canvas2d的上下文,并完成坐标的转换

    let canvas = document.querySelector('canvas'),
       ctx = canvas.getContext('2d');
    
    ctx.translate(canvas.width / 2, canvas.height / 2);
    ctx.scale(1, -1);

    因为画布原始的坐标系是以左上角为原点,X轴向左,Y轴向下,这不符合我们在数学中常用的配置。

    这里我们先通过translate方法把坐标挪到画布中心,再通过scale方法将坐标系绕X轴翻转;通过这样的转换,就可以按照我们在数学中常见的坐标系来操作了。

  • 然后我们来初始化三个点,也就是之前说的点A、点B和点C。

    坐标可以随便写,只要范围在-256到256之间就可以。

    我这里就简单定义三个在X轴上的点,并维护在一个Map中,方便后续在canvas上显示三个点的标识;后面会加一个事件监听来更新点C的坐标。

    let map = new Map();
    let v0 = new Vector2D(0, 0),
        v1 = new Vector2D(100, 0),
        v2 = new Vector2D(-100, 0);
    map.set('C', v0);
    map.set('A', v1);
    map.set('B', v2);
  • 然后就可以开始绘制

    这里我们定义一个draw函数,然后调用它。

    draw();
    
    function draw() {}
    • 首先,为了看上去更清晰,我们可以把坐标系绘制出来。

      因为接下去绘制的直线比较多,这里我简单封装一个绘制直线的方法。

      function drawLine(start, end, color) {
        ctx.beginPath();
        ctx.save();
        ctx.lineWidth = '4px';
        ctx.strokeStyle = color;
        ctx.moveTo(...start);
        ctx.lineTo(...end);
        ctx.stroke();
        ctx.restore();
        ctx.closePath();
      }

      然后我们来绘制坐标系。

      drawAxis();
      
      function drawAxis() {
        drawLine([-canvas.width / 2, 0], [canvas.width / 2, 0], "#333");
        drawLine([0, canvas.height / 2], [0, -canvas.height / 2], "#333");
      }
    • 接着我们把点绘制到画布上

      for(const p of map) {
        drawPoint(p[1], p[0]);
      }
      
      function drawPoint(v, name, color='#333') {
        ctx.beginPath();
        ctx.save();
        ctx.fillStyle = color;
        ctx.arc(v.x, v.y, 2, 0, Math.PI * 2);
        ctx.scale(1, -1);
        ctx.fillText(`${name}`, v.x, 16 - v.y);
        ctx.restore();
        ctx.fill();
      }

      这里我们想把点的标识通过fillText也绘制到画布上,但由于之前坐标被绕X轴翻转过一次,所以直接绘制表示会导致文本是倒过来的,所以我们这里临时把坐标系翻转回来,完成文本绘制后,再通过restore恢复回去。

    • 现在我们把线段AB也绘制出来

      drawBaseline();
      
      function drawBaseline() {
        drawLine(map.get('A'), map.get('B'), "blue");
      }
    • 最后就是最关键的一步,把点C到线段AB和直线的距离求出来并展示在canvas画布上

      d为点C到线段AB的距离,dLine为点C到直线的距离;

      result存储的是AC和AB的点乘结果;crossProduct存储的是AC和AB的叉乘结果。

      根据叉乘结果,我们就可以计算出dLine的值,也就是点C到直线的距离。

      drawLines();
      
      function drawLines() {
        let AC = map.get('C').minus(map.get('A'));
        let AB = map.get('B').minus(map.get('A'));
        let BC = map.get('C').minus(map.get('B'));
        let result = AC.dot(AB);
        let d, dLine; // distance
      
        let crossProduct = AC.cross(AB);
        dLine = Math.abs(crossProduct) / AB.len;
        let pd = getD();
        map.set('D', pd);
        if (result < 0) {
          // 角CAB为钝角
          drawLine(map.get('A'), map.get('C'), 'red');
          drawLine(map.get('C'), pd, 'green');
          d = AC.len;
        } else if (result > Math.pow(AB.len, 2)) {
          // 角CBA为钝角
          drawLine(map.get('B'), map.get('C'), 'red');
          drawLine(map.get('C'), pd, 'green');
          d = BC.len;
        } else {
          d = dLine;
          drawLine(map.get('C'), pd, 'red');
        }
      
        let text = `点C到线段AB的距离:${Math.floor(d)}, 点C到AB所在直线的距离为${Math.floor(dLine)}`;
        drawText(text);
      }
      
      function getD() {
        let AC = map.get('C').minus(map.get('A'));
        let AB = map.get('B').minus(map.get('A'));
        let A = map.get('A'); // 即:向量OA
        // 已知:AD为AC在AB上的投影
        // AD = (AB / |AB|) * (AC·AB / |AB|)
        //    = AB * (AC·AB / |AB|²) 
        // D.x - A.x = AD.x, D.y - A.y = AD.y
        let AD = AB.scale(AC.dot(AB) / AB.len**2);
        let D = new Vector2D(
          AD.x + A.x,
          AD.y + A.y
        );
        return D;
      }

      然后我们来计算点D的坐标:

      已知:AD是AC在AB上的投影。

      所以AD可以表示为这样:(AB / |AB|) * (AC·AB / |AB|)

      向量AB除以AB的模即代表和向量AB同一方向夹角的单位向量,单位向量可以简单理解为长度为1的向量;

      AC和AB的点积除以AB的模结果等于AC的模乘以两个向量夹角的余弦值

      所以这两个值相乘,就等于是向量AD。

      通过调整上面的公式,我们可以得到AD =AB * (AC·AB / |AB|²),因为A、B、C的坐标都已知,也就可以得到向量AD的坐标。

      然后我们又知道向量AD的坐标可以直接通过向量的减法得到,也就是:

      • AD.x = D.x - A.x
      • AD.y = D.y - A.y

      所以我们就可以得到点D的坐标,即(AD.x + A.x, AD.y + A.y)

      接着我们根据AC和AB的点乘结果result,来绘制相应的直线。

      • 当result为负数时,说明AC和AB夹角的余弦值大于90度

        即∠CAB为钝角,说明点C到线段AB的距离就是点C到点A的距离。

      • 而当result大于AC长度的平方,也就是AC的模乘以余弦值大于AB的模,也就是说,AC在向量AB上的投影大于AB的长度

        那么此时∠CBA是钝角,点C到线段AB的距离就是点C到点B的距离。

      • 当result为0时,说明两个向量互相垂直

        此时,点C在线段AB的上方或下方,点C到线段AB的距离就是点C到直线的距离。也就是我们前面求到的dLine的值。

      最后我们将结果通过fillText方法绘制到屏幕上。

      function drawText(distance) {
        ctx.beginPath();
        ctx.save();
        ctx.font = "16px serif";
        ctx.scale(1, -1);
        ctx.fillText(`${distance}`, -250, 240);
        ctx.restore();
      }
    • 最后我们加一个鼠标移动事件,动态地更新点C的坐标,以及点C到线段AB和直线的距离。

      initEvents();
      
      function initEvents() {
        canvas.addEventListener('mousemove', e => {
          const rect = canvas.getBoundingClientRect();
          ctx.clearRect(-canvas.width / 2, -canvas.height / 2, canvas.width, canvas.height);
          let x = e.pageX - rect.left - canvas.width / 2;
          let y = -(e.pageY - rect.top - canvas.height / 2);
          v0 = new Vector2D(x, y);
          map.set('C', v0);
          draw();
       });
      }

    好啦,到这里为止一个简单的距离展示就完成了;我们可以通过移动鼠标来查看最后的效果。