canvas中的拖拽、缩放、旋转 (上) —— 数学知识准备

栏目: Html5 · 发布时间: 5年前

内容简介:本文首发于公众号:符合预期的CoyPan最近做了一个移动端活动页的需求,大概就是diy一个页面。用户可以对物料进行拖动、缩放、旋转,来达到diy的目的。用DOM来实现是不现实的,我采用了canvas来实现和用户的交互。开发过程中,涉及到了canvas中对物料元素的

写在前面

canvas中的拖拽、缩放、旋转 (上) —— 数学知识准备

本文首发于公众号:符合预期的CoyPan

最近做了一个移动端活动页的需求,大概就是diy一个页面。用户可以对物料进行拖动、缩放、旋转,来达到diy的目的。用DOM来实现是不现实的,我采用了canvas来实现和用户的交互。开发过程中,涉及到了canvas中对物料元素的 拖动、缩放、旋转 等。本文将详细介绍在 不使用任何第三方库 的情况下,如何实现这些功能。最终的效果demo,可以参考上面的gif图。demo体验地址在这里: 请用手机或浏览器模拟手机访问

本文先介绍整个需求中需要注意的数学知识。

需求分析

整个需求的大致流程是:

  1. 用户点击选择一个元素,则将该元素画在canvas中。
  2. 用户在canvas中对元素进行拖动、缩放、旋转等操作。
  3. 用户可以删除canvas中的某个元素。

在canvas中实现拖动、缩放、旋转等交互,最核心的两个点就是:

  1. 用户触摸时,判断用户是否触摸到了元素、是否触摸到了元素的缩放,旋转按钮。
  2. 用户移动手指时,根据手指的路劲,控制元素的运动。

我们知道,canvas中最基础的是坐标系统。本次的需求中,两个最关键的点是:

  1. 如何判断用户是否触摸到了某个元素,即触摸的 落点问题
  2. 如何通过坐标控制canvas中的元素的运动:移动、缩放、旋转。

落点问题:如何判断是否触摸到了某元素

首先提供一种简单的方法,canvas中有一个isPointInPath方法可以判断一个落点是否在某个 路径 中。不过如果canvas中的元素是图片,那么我们必须在画每一张图片时,为其加上一个路径包裹起来。这是一种解决落点问题的方案。这里不做深入介绍了,我采用的是下面的方案。

为了使问题简单化,可以大胆假设:

canvas中的所有元素(图片、各种图形等)都是长方形的。这已经能覆盖大部分情况了。长方形 四个顶点的坐标 就能确定一个元素的位置了。

而当用户触摸canvas时,通过触摸事件,可以拿到用户触摸的坐标。如果触摸坐标在长方形顶点坐标"内部",则表示触摸到了元素。于是,我们的问题可以抽象为:

已知长方形四个顶点坐标,某点的坐标,如何判断这个点是否在这个长方形内部。

如果长方形没有产生旋转,那么问题很简单,只用判断点的横纵坐标均在长方形的横纵坐标范围内即可。如果长方形产生了旋转,这种方法就没用了。要解决这个问题,先复习一下高中数学 。

向量

canvas中的拖拽、缩放、旋转 (上) —— 数学知识准备

我们可以将二维平面上的坐标都转化为向量来计算,可以将问题简化很多。

向量的叉积

两个向量的叉积运算结果是一个向量而不是一个标量。叉积的方向与这两个向量所在的平面垂直。

canvas中的拖拽、缩放、旋转 (上) —— 数学知识准备

canvas中的拖拽、缩放、旋转 (上) —— 数学知识准备

canvas中的拖拽、缩放、旋转 (上) —— 数学知识准备

对于平面中的两个向量,第三维方向上的值都为0,其叉积的值为:

canvas中的拖拽、缩放、旋转 (上) —— 数学知识准备

换句话说,我们可以很方便地判断:

一个平面中,在旋转角不超过180度的情况下,从一个向量到另外一个向量,是顺时针转动还是逆时针转动。

直接以canvas的二维坐标系统为例:

canvas中的拖拽、缩放、旋转 (上) —— 数学知识准备

可以得到这样的结论:

在canvas二维平面中,设向量A与向量B的叉积对应的二项式的值为m。如果m>0,则向量A顺时针转动一个角度(小于180度),就能够到达向量B的方向。如果m<0,则需要逆时针转动。

判断落点在长方形内

落点在长方形内的情景如下:

canvas中的拖拽、缩放、旋转 (上) —— 数学知识准备

于是,判断【是否触摸到了canvas中某元素】的函数就有了:

/**
 * 判断落点是否在长方形内
 * 
 * @param {Array} point 落点坐标。 数组:[x, y]
 * @param {Array} rect 长方形坐标, 按顺序分别是:左上、右上、左下、右下。 
 *                     数组:[[x1, y1], [x2, y2], [x3, y3], [x4, y4]]
 * 
 * @return {boolean} 
 */

function isPointInRect(point, rect) {
    const [touchX, touchY] = point;
    // 长方形四个点的坐标
    const [[x1, y1], [x2, y2], [x3, y3], [x4, y4]] = rect;
    // 四个向量
    const v1 = [x1 - touchX, y1 - touchY];
    const v2 = [x2 - touchX, y2 - touchY];
    const v3 = [x3 - touchX, y3 - touchY];
    const v4 = [x4 - touchX, y4 - touchY];
    if(
        (v1[0] * v2[1] - v2[0] * v1[1]) > 0 
        && (v2[0] * v4[1] -  v4[0] * v2[1]) > 0
        && (v4[0] * v3[1] - v3[0] * v4[1]) > 0
        && (v3[0] * v1[1] -  v1[0] * v3[1]) > 0
    ){
        return true;
    }
    return false;
}

旋转角度

用户可以对canvas中的元素进行旋转,那么如何通过用户前后两次的触摸落点坐标求旋转角度呢?

canvas中的拖拽、缩放、旋转 (上) —— 数学知识准备

从上面的等式可以看出,点积和叉积都能求夹角。选用哪一个呢?

在旋转的场景中,旋转的方向(逆时针or顺时针)是很重要的,而点积最终得到的只是一个标量,是没有方向。叉积是一个向量,是有方向的。我们选择叉积来计算旋转角度。

1、一般以canvas中的元素的中心为旋转原点,用户在canvas中触摸移动时,通过事件监听函数得到的前后两次触摸点的位移是很小的,与旋转中心形成的向量夹角必然是小于90度的。

2、向量的叉积正负值可以确定旋转方向。

3、反正弦函数是在负90度到90度之间单调递增的。

通过以上三点,可以得到:

canvas中的拖拽、缩放、旋转 (上) —— 数学知识准备

于是,在canvas中,可以用以下函数来计算连续两次触摸落点与旋转中心形成的旋转角度:

/**
 * 计算旋转角度
 * 
 * @param {Array} centerPoint 旋转中心坐标
 * @param {Array} startPoint 旋转起点
 * @param {Array} endPoint 旋转终点
 * 
 * @return {number} 旋转角度
 */

function getRotateAngle(centerPoint, startPoint, endPoint) {
    const [centerX, centerY] = centerPoint;
    const [rotateStartX, rotateStartY] = startPoint;
    const [touchX, touchY] = endPoint;
    // 两个向量
    const v1 = [rotateStartX - centerX, rotateStartY - centerY];
    const v2 = [touchX - centerX, touchY - centerY];
    // 公式的分子
    const numerator =  v1[0] * v2[1] - v1[1] * v2[0];
    // 公式的分母
    const denominator = Math.sqrt(Math.pow(v1[0], 2) + Math.pow(v1[1], 2)) 
        * Math.sqrt(Math.pow(v2[0], 2) + Math.pow(v2[1], 2));
    const sin = numerator / denominator;
    return Math.asin(sin);
}

已知旋转角度和初始坐标,求旋转后坐标

canvas中的拖拽、缩放、旋转 (上) —— 数学知识准备

已知旋转起点、旋转中心以及旋转角度,求旋转终点坐标的函数如下:

/**
 * 
 * 根据旋转起点、旋转中心和旋转角度计算旋转终点的坐标
 * 
 * @param {Array} startPoint  起点坐标
 * @param {Array} centerPoint  旋转点坐标
 * @param {number} angle 旋转角度
 * 
 * @return {Array} 旋转终点的坐标
 */

function getEndPointByRotate(startPoint, centerPoint, angle) {
    const [centerX, centerY] = centerPoint;
    const [x1, y1] = [startPoint[0] - centerX, startPoint[1] - centerY];
    const x2 = x1 * Math.cos(angle) - y1 * Math.sin(angle);
    const y2 = x1 * Math.sin(angle) + y1 * Math.cos(angle);
    return [x2 + centerX, y2 + centerY];
}

拖拽、缩放

拖拽和缩放在本次需求中,对于数学上的要求并不高。

拖拽需要计算好触摸点横纵坐标的差值,加到canvas中的元素上即可。

canvas中的拖拽、缩放、旋转 (上) —— 数学知识准备

canvas中的拖拽、缩放、旋转 (上) —— 数学知识准备

写在后面

本文主要介绍了在canvas中实现拖拽、缩放、旋转等交互时,所需要的一些数学知识。如有不对的地方,欢迎指正。同时,如果有其他解决需求的思路,欢迎交流。

下一篇文章将介绍【使用本文介绍的数学知识,来实现文章开头的demo】的过程。

符合预期。

canvas中的拖拽、缩放、旋转 (上) —— 数学知识准备


以上所述就是小编给大家介绍的《canvas中的拖拽、缩放、旋转 (上) —— 数学知识准备》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

Java并发编程实战

Java并发编程实战

Brian Goetz、Tim Peierls、Joshua Bloch、Joseph Bowbeer、David Holmes、Doug Lea / 童云兰 / 机械工业出版社华章公司 / 2012-2 / 69.00元

本书深入浅出地介绍了Java线程和并发,是一本完美的Java并发参考手册。书中从并发性和线程安全性的基本概念出发,介绍了如何使用类库提供的基本并发构建块,用于避免并发危险、构造线程安全的类及验证线程安全的规则,如何将小的线程安全类组合成更大的线程安全类,如何利用线程来提高并发应用程序的吞吐量,如何识别可并行执行的任务,如何提高单线程子系统的响应性,如何确保并发程序执行预期任务,如何提高并发代码的性......一起来看看 《Java并发编程实战》 这本书的介绍吧!

JS 压缩/解压工具
JS 压缩/解压工具

在线压缩/解压 JS 代码

RGB转16进制工具
RGB转16进制工具

RGB HEX 互转工具

在线进制转换器
在线进制转换器

各进制数互转换器