Vue 3 + Canvas 实现图形拖拽、缩放、旋转

在很多图形编辑器、UI 设计器、应用绘图系统中,我们常常需要支持图形的 拖拽、缩放、旋转 三种基础操作。本文将分享如何利用 Vue 3 + Canvas 组合实现一个完整的图形操作示例,包括:

  • 项目结构介绍
  • Canvas 操作动作解析
  • 实现 Sprite 类和 Stage 管理器
  • Vue 3 与 Canvas 的组合实践

一、项目结构

我们使用 Vite 创建 Vue 3 项目,类目录结构如下:

css 复制代码
your-canvas-editor/
├─ index.html
├─ package.json
├─ vite.config.js
├─ src/
│  ├─ main.js
│  ├─ App.vue
│  └─ components/
│     └─ CanvasEditor.vue

index.html

指定页面根 DOM 节点

xml 复制代码
<div id="app"></div>
<script type="module" src="/src/main.js"></script>

package.json

指定 Vue 3 和 Vite 配置

perl 复制代码
"dependencies": {
  "vue": "^3.4.0"
},
"devDependencies": {
  "vite": "^5.0.0",
  "@vitejs/plugin-vue": "^5.0.0"
}

二、CanvasEditor.vue 主组件

这是一个基于 Canvas 实现拖拽、缩放、旋转图形的组件,使用 <script setup> 构建。

xml 复制代码
<template>
  <div class="canvas-container">
    <canvas ref="canvasRef" width="300" height="300"></canvas>
  </div>
  <div class="box-container">
    <div class="red-box" @click="addSprite"></div>
  </div>
</template>

<script setup>
import { ref, onMounted } from 'vue'

const canvasRef = ref(null)
let stage = null

onMounted(() => {
  stage = new Stage({ canvas: canvasRef.value })
})

function addSprite() {
  const randomX = Math.floor(Math.random() * 200)
  const randomY = Math.floor(Math.random() * 200)
  const sprite = new Sprite({
    pos: [randomX, randomY],
    size: [120, 60],
    minSize: [40, 20],
    maxSize: [240, 120]
  })
  stage.append(sprite)
}
</script>

三、核心类: Stage 和 Sprite

Stage 类

用于 Canvas 场景管理,包括:

  • 图形列表
  • 触摸事件分发
  • 拖拽 / 缩放 / 旋转解析与重绘

基本用法:

go 复制代码
stage.append(sprite)

重点方法:

  • handleTouchStart
  • handleTouchMove
  • handleTouchEnd
  • drawSprite()

Sprite 类

代表单个图形,包含:

  • 位置 pos
  • 大小 size
  • 四个顶点坐标 coordinate
  • 操作按钮:旋转/删除/缩放

操作方法:

scss 复制代码
sprite.resetPos(dx, dy) // 拖拽
sprite.resetSize(scale) // 缩放
sprite.setRotateAngle(angle) // 旋转

四、设计策略

1. 使用矩阵转换旋转坐标

Canvas 中旋转操作需要以中心点为基准:

scss 复制代码
ctx.translate(centerX, centerY)
ctx.rotate(angle)
ctx.translate(-centerX, -centerY)

2. 触摸区分别分类处理

  • 旋转区 -> rotateIcon
  • 缩放区 -> scaleIcon
  • 拖动区 -> sprite body
  • 删除区 -> delIcon

3. 各按钮位置用 sprite center + 向量 转换计算

arduino 复制代码
const vector = [x - centerX, y - centerY]
const scaled = [vector[0] * scale, vector[1] * scale]

五、效果展示

点击红色按钮创建新图形,可在 Canvas 中:

  • 按位置拖拽
  • 缩放经度
  • 四分之一圈旋转
  • 删除

支持多个 Sprite 同时存在


六、总结

通过本项目,我们用 Vue 3 的 <script setup> 构造组件,实现了 Canvas 图形的拖拽/缩放/旋转,其核心是:

  • Sprite 对应操作逻辑
  • Stage 分发 touch 操作
  • Canvas 2D 演练 transform 和 icon 操作

同时保持 Vue 3 简洁的编程风格


七、后续可扩展

  • 支持折线或多边形
  • 层级排序
  • 图形经纬系统定位
  • JSON 化导入导出
  • Vuex/Pinia 独立管理 state

帮助设计器类项目快速构建应用编辑器或场景应用。

相关推荐
^小桃冰茶1 小时前
CSS知识总结
前端·css
巴巴_羊2 小时前
yarn npm pnpm
前端·npm·node.js
chéng ௹4 小时前
vue2 上传pdf,拖拽盖章,下载图片
前端·css·pdf
嗯.~4 小时前
【无标题】如何在sheel中运行Spark
前端·javascript·c#
A_aspectJ6 小时前
【Bootstrap V4系列】学习入门教程之 组件-输入组(Input group)
前端·css·学习·bootstrap·html
兆。6 小时前
电子商城后台管理平台-Flask Vue项目开发
前端·vue.js·后端·python·flask
互联网搬砖老肖7 小时前
Web 架构之负载均衡全解析
前端·架构·负载均衡
sunbyte7 小时前
Tailwind CSS v4 主题化实践入门(自定义 Theme + 主题模式切换)✨
前端·javascript·css·tailwindcss
湛海不过深蓝8 小时前
【css】css统一设置变量
前端·css
程序员的世界你不懂9 小时前
tomcat6性能优化
前端·性能优化·firefox