html5中canvas图形变换transform、setTransform原理(变换矩阵)

本节目录

变换矩阵

canvas中,平移 translate()、缩放 scale() 和旋转 rotate() 这 3 种方法在本质上是用变换矩阵 transform()方法来实现的。

也就是说,我们仅通过 transform() 方法就可以实现平移、缩放和旋转这 3 种操作。

语法

javascript 复制代码
ctx.transform(a,b,c,d,e,f);

a :水平缩放绘图

b :水平倾斜绘图

c :垂直倾斜绘图

d :垂直缩放绘图

e :水平移动绘图

f :垂直移动绘图

原理

原理是很简单的线性代数的矩阵乘法

设置开始坐标系为(x1, y1), 经过变换后得到 (x2, y2)

javascript 复制代码
          
令矩阵 
    [a, c, e]
A = [b, d, f]
    [0, 0, 1]

则:
  [x2]       [x1]
  [y2] = A * [y1]
  [c ]       [1]

则有:
x2 = a*x1 + c*y1 + e
y2 = b*x1 + d*y1 + f
c = 1

矩阵乘法原则是:
· 只有左矩阵的列数与右矩阵的行数相同的两个矩阵才能相乘.
· 乘积矩阵第i行第j列处的元素等于左矩阵的第i行与右矩阵的第j列对应元素乘积之和.
· 乘积矩阵的行数等于左矩阵的行数,列数等于右矩阵的列数.

c为了凑矩阵运算用的,计算得到后没用作用,直接舍去, 我们只看x2,y2

如果你对数学敏感,那么现在就可以看出来,前面abcdef的取名为什么是这样取的了

复制代码
x2 = a*x1 + c*y1 + e
y2 = b*x1 + d*y1 + f

a :水平缩放绘图
b :水平倾斜绘图
c :垂直倾斜绘图
d :垂直缩放绘图
e :水平移动绘图
f :垂直移动绘图

实现平移

由于:

x2 = ax1 + c y1 + e

y2 = bx1 + dy1 + f

对于点(x1, y1)平移到点(x2, y2), 那么应该有:

x2 = x1 + ?

y2 = y1 + ?

则令:

a=d=1

c=b=0

e即为x轴平移距离, f为y轴平移距离

即:

x2 = 1 * x1 + 0 * y1 + e

x2 = x1 + e
y2 = 0 * x1 + 1 * y1 + f

y2 = y1 + f

javascript 复制代码
ctx.transform(1, 0, 0, 1, e, f)

实现缩放

由于:

x2 = ax1 + c y1 + e

y2 = bx1 + dy1 + f

对于(x1, y1)缩放后到(x2, y2), 有:

x2 = nx1
y2 = n
y2

则令:

c = e = b = f = 0

x2 = ax1
y2 = d
y1

javascript 复制代码
ctx.transform(a, 0, 0, d, 0, 0)

实现旋转

假设图形开始坐标为(x1,y1),旋转后的坐标为(x2,y2),图形旋转的角度为 θ,那么就有

以下公式:

x2 = x1 * cosθ - y1 * sinθ;

y2 = x1 * sinθ + y1 * cosθ;

由于:

x 2 = a * x1 + c * y1 + e

y2 = b * x1 + d * y1 + f

则令

e = f = 0

a = cosθ

b = sinθ

c = -sinθ

d = cosθ

javascript 复制代码
ctx.transform(cosθ, sinθ,- sinθ, cosθ, 0, 0)

transform与setTransform的区别

区别就是transform会在上一次的基础上变换原点与坐标系旋转角度

而setTransform则每次都从w3c坐标系的基础上也就是最原始的坐标

这意味着混合使用先transform顺时针30度再使用setTransform顺时针旋转30度, 当前绘制的话, 其实只在w3c坐标系的基础上旋转了30度。

但是使用两个transform分别顺时针旋转30度则在w3c坐标系的基础上旋转了60度

相关推荐
LIUAWEIO6 小时前
CSS 让鼠标呈现手型,鼠标悬浮变小手
css·html·css3·html5
七夜zippoe6 小时前
OpenClaw Canvas A2UI:AI驱动的交互式界面开发实战
人工智能·canvas·交互式·a2ui·openclaw
LIUAWEIO7 小时前
vue里面下载配置使用zepto vue中怎样使用zepto
javascript·vue.js·es6·zepto
lantian7 小时前
TypeScript 三斜线指令完全指南:从入门到理解为什么不再需要它
前端·javascript·vue.js
用户938515635077 小时前
从"用栈实现队列"说起:深入理解 JavaScript 原型式面向对象
javascript
ZengLiangYi7 小时前
AI 编程工具的数据格式为什么不能统一
javascript·后端·架构
陈_杨7 小时前
鸿蒙APP开发-带你走进旧物集的时间线与收藏管理
前端·javascript
尼斯湖皮皮怪7 小时前
iceCoder双模详解
javascript
小雨下雨的雨8 小时前
月相分析工具鸿蒙PC Electron框架技术实现详解
前端·javascript·华为·electron
布依前端8 小时前
基于 Vue 3 的 Tiptap 富文本编辑器实践:tiptap-editor-vue3 项目介绍
前端·javascript·vue.js