本示例是源于项目的一个需求做的一个demo。初衷只是为了实现通过拖拽绘制用户所需大小的矩形,后面就继续完善了一下功能。目前实现了通过拖拽绘制矩形、移动画布上现有的矩形、保存当前画布、撤销与反撤销
canvas是HTML5新增的元素,是HTML5 的一大亮点,canvas翻译过来其实就是画布的意思,它可以替代flash,制作网页的很多动画效果以及游戏。渲染效率非常高,不像flash要在游览器安装flash adobe插件,canvas不需要安装任何插件即可渲染这个动画。目前所有主流游览器都支持canvas。
canvas常用api
canvas的基础用法大家可以去查看文档,相信会看文档的人都会用了。以下列出了常用的几个api。
// ctx为canvas上下文对象
ctx.beginPath() // 开始一个路径
ctx.moveTo(x,y) // 路径移到画布中的指定点 , 即起点
ctx.lineTo(x,y) //添加一个新点,画线
ctx.closePath() // 关闭绘制路径
ctx.fillStyle // 设置填充颜色
ctx.fill() // 填充区域
ctx.lineWidth // 设置线的宽度
ctx.strokeStyle // 设置描边颜色
ctx.stroke() // 填充描边
ctx.rect(x,y,w,h) // 绘制矩形 x、y为起始坐标,w、h为矩形的宽、高
ctx.fillRect(x,y,w,h) // 填充矩形
ctx.strokeRect(x,y,w,h) // 描边矩形
arc(x,y,r,sa,ea,true/false) // 绘制圆形 x、y为圆心坐标,r为半径,sa为起始角度,ea为结束角度,true是逆时针画圆,false是顺时针画圆
fillText(text,x,y,maxWidth) // 填充绘制 text表示文字,x、y为坐标,maxWidth可选,为文字最大宽度,防止文字溢出
strokeText(text,x,y,maxWidth) // 描边绘制 text表示文字,x、y为坐标,maxWidth可选,为文字最大宽度,防止文字溢出
ctx.clearRect(x,y,width,height) // x为清除起点横坐标, y为清除起点纵坐标,width为清除长度,height为清除高度
功能1: 鼠标拖拽绘制矩形
在鼠标按下时记录当前按下位置的坐标值,为起始坐标,松开鼠标时的坐标为结束坐标,以这两个坐标可以得到四个点,绘制出一个矩形
// 为canvas注册事件
window.onload = function () {
canvas = document.getElementById('canvas')
context = canvas.getContext('2d')
canvas.onmousedown = mouseDown
canvas.onmouseup = mouseUp
}
// 鼠标按下事件,记录起始坐标值
function mouseDown(e) {
startX = e.offsetX
startY = e.offsetY
}
// 鼠标松开事件, 记录结束坐标
function mouseUp(e) {
endX = e.offsetX
endY = e.offsetY
rectList.unshift(new Rect(startX, startY, endX, endY, color))
// 绘制矩形
context.beginPath()
context.globalAlpha = 0.3 // 透明度
context.moveTo(startX, startY)
context.lineTo(endX, startY)
context.lineTo(endX, endY)
context.lineTo(startX, endY)
context.lineTo(startX, startY)
context.fillStyle = 'yellow'
context.strokeStyle = 'black'
context.fill()
context.stroke()
}
以上代码就可以绘制出一个矩形,可能有人会问为什么不直接用rect()方法来绘制,如果用rect()方法的话就得计算宽高,并通过计算方向来判断鼠标按下时的点为起始坐标还是松开时的点为起始坐标,而使用路径的方法绘制矩形就不用那么麻烦了。
但是是有一个问题,就是鼠标点击后拖拽并没有达到我们想要的效果,从点击———>拖拽———>松开这3步期间应该有个过渡效果。
优化:在鼠标拖拽过程中不断的改变结束的坐标值,并绘制矩形
为canvas注册鼠标拖拽事件mouseMove
let isDrawing = false // 是否正在绘图
// 在鼠标按下时设置isDrawing=true, 松开时为false
function mouseMove(e) {
if (isDrawing) { // 判断是否正在绘图
endX = e.offsetX
endY = e.offsetY
context.globalAlpha = 0.3
context.beginPath()
context.moveTo(startX, startY)
context.lineTo(endX, startY)
context.lineTo(endX, endY)
context.lineTo(startX, endY)
context.lineTo(startX, startY)
context.fillStyle = color
context.strokeStyle = 'black'
context.fill()
context.stroke()
}
}
运行后就会发现出现以下情况:
解决方案:在每次绘制时都清空画布
为了能同时绘制出多个矩形,用一个数组存储每个矩形对象,每次松开鼠标后把当前的矩形对象存进数组的头部(因为每次绘制是按顺序从数组中取出矩形绘制,如果存进尾部后绘制的矩形就会被前面的矩形覆盖,所以这里用的是unshift而不是push),因为每次绘制都需要清空画布,所以每次在画布上绘制矩形时都要还原之前画布的状态
// 定义一个矩形的函数
function Rect(startX, startY, endX, endY, color) {
this.startX = startX // 起始横坐标
this.startY = startY // 起始纵坐标
this.endX = endX // 结束横坐标
this.endY = endY // 结束纵坐标
this.color = color // 填充颜色
this.isSelected = false // 是否被选中
}
let rectList = [] // 矩形对象数组
function mouseUp(e) {
rectList.unshift(new Rect(startX, startY, endX, endY, color))
isDrawing = false
}
// 还原画布状态,在mouseMove中每次绘制前先调用该函数
function drawRects() {
context.clearRect(0, 0, canvas.width, canvas.height)
for (let i = 0; i < rectList.length; i++) {
let rect = rectList[i]
context.globalAlpha = 0.3
context.beginPath()
context.moveTo(rect.startX, rect.startY)
context.lineTo(rect.endX, rect.startY)
context.lineTo(rect.endX, rect.endY)
context.lineTo(rect.startX, rect.endY)
context.lineTo(rect.startX, rect.startY)
context.fillStyle = rect.color
context.fill()
}
}