一、canvas是什么?

canvas就是画布

canvas元素用于在网页上绘制2D图形和图像

二、canvas坐标体系

canvas的默认宽高是300*150,要在脚本中对画布进行操作。

踩坑注意:如果要对canvas画布的大小进行操作,不能在style上操作,要在内联样式上写或者js中操作,在style上改变宽高画布会被拉伸

三、canvas画直线、曲线和圆

(1)canvas画直线

<canvas id="myCanvas1">
    您的浏览器不支持Canvas,请升级浏览器
  </canvas>
<script>
    var canvas1 = document.getElementById('myCanvas1');
    var ctx1 = canvas1.getContext('2d');  //
    ctx1.canvas.width = 100; //js获取宽高
    ctx1.canvas.height = 100;
    ctx1.moveTo(0, 0);   //起点
    ctx1.lineTo(100, 100);  //直线到100,100
    ctx1.stroke(); //没有这个stroke()的话不会出现画布效果
</script>

(2)canvas画圆和矩形

var canvas = document.getElementById('myCanvas');
var ctx = canvas.getContext('2d');
     ctx.canvas.width = 600;
     ctx.canvas.height = 400;
     ctx.beginPath(); //重新开始新路径
     ctx.moveTo(0, 0);
     ctx.lineTo(100, 100);
     ctx.lineTo(100, 200);
     ctx.closePath(); //闭合路径(可以连接首尾)
     ctx.stroke(); //记得最后调用stroke才能画线
     // 画圆
     ctx.beginPath();
     //ctx.arc(开始原点x,开始原点y,半径,起始弧度,终止弧度,true表示逆时针动画)
     ctx.arc(200, 150, 100, 0, Math.PI * 2, true);
     ctx.closePath();
     ctx.stroke();
     //画矩形,封装一个stroke的rect(矩形,所以不用调用rect)
     //ctx.strokeRect(开始原点x,开始原点y,宽度,高度)
     ctx.strokeRect(100, 100, 100, 100);

(3)canvas描边和填充

//线条样式
    ctx.beginPath();
    ctx.strokeStyle = "#00FF00"; //设置线条的样式颜色
    ctx.lineWidth = 5; //设置描边的线宽
    ctx.moveTo(100, 200);
    ctx.lineTo(200, 200);
    ctx.stroke();
    //填充
    ctx.beginPath();
    ctx.moveTo(300, 100);
    ctx.lineTo(400, 100);
    ctx.lineTo(400, 200);
    ctx.closePath();
    //ctx.stroke();
    ctx.fillStyle = "#0000FF"; //设置填充的颜色
    ctx.fill(); //填充 (如果是首尾没有连接的话会自动连接填充里面)
    ctx.beginPath();
    ctx.moveTo(300, 150);
    ctx.lineTo(400, 150);
    ctx.lineTo(400, 250);
    ctx.closePath();
    ctx.stroke();

四、canvas图形变换

(1)、save()和restore()方法

save(),保存当前绘画环境,包括样式和变换

restroe(),恢复当前绘画环境,包括样式和变换

注意:ctx.save()和restore()方法,保存了上下文的环境包括图形变换和样式。并且一定要同时出现。restore之后的canvas样式会以save之前的样式重新开始

ctx.beginPath();
    //ctx.save()和restore()方法,保存了上下文的环境包括图形变换和样式。并且一定要同时出现
    ctx.save();
    //x轴的平移方向,translate平移变换
      ctx.translate(0, 20);
      ctx.moveTo(50, 50);
      ctx.lineTo(150, 50);
    ctx.restore(); //restore之后的canvas样式会以save之前的样式重新开始
    ctx.stroke();
    //ctx.fillRect(-5, -5, 10, 10);
    ctx.beginPath();

(2)偏移、旋转、缩放变换

ctx.beginPath();
    //x轴的平移方向,translate平移变换
    ctx.translate(300, 0);
    for(var i=0; i<10; i++) {
    //rotate旋转变换,Math.PI=180
      ctx.rotate(Math.PI / 5);
      ctx.moveTo(-50, 0);
      ctx.lineTo(50, 0);
    }
    ctx.stroke();
    //scale缩放变换,(x轴缩放,y轴缩放)
    ctx.sacle(1,0.5)

 五、canvas中的渐变

(1)线性渐变 ctx.createLinearGradient(x轴起点,y轴起点,x轴终点,y轴终点)

(2)径向渐变 ctx.createRadialGradient(x,y起点,原点起点,x,y终点,原点终点)

(3)渐变中的颜色转换ctx.createLinearGradient(createRadialGradient).addColorStop(0-1百分比,rgba)

var canvas = document.getElementById('myCanvas');
    var ctx = canvas.getContext('2d');
    ctx.canvas.width = 600;
    ctx.canvas.height = 400;
    //线性渐变 ctx.createLinearGradient(x轴起点,y轴起点,x轴终点,y轴终点);
    var linearGradient = ctx.createLinearGradient(50, 50, 150, 150);
    //linearGradient.addColorStop(0-1百分比,rgba)
    linearGradient.addColorStop(0, 'rgb(255,0,0)');
    linearGradient.addColorStop(0.5, 'rgb(0,255,0)');
    linearGradient.addColorStop(1, 'rgb(0,0,255)');
    ctx.fillStyle = linearGradient;
    ctx.fillRect(0, 0, 200, 200);
    ctx.beginPath();
    ctx.arc(400, 150, 100, 0, Math.PI * 2, true);
    ctx.closePath();
    //径向渐变 ctx.createRadialGradient(x,y起点,原点起点,x,y终点,原点终点)
    var radialGradient = ctx.createRadialGradient(400, 150, 0, 400, 150, 100);
    radialGradient.addColorStop(0, 'rgb(255,0,0)');
    radialGradient.addColorStop(1, 'rgb(0,0,255)');
    ctx.fillStyle = radialGradient;
    ctx.fill();

六、canvas中文字和图片的绘制

1、canvas中的文字

var canvas = document.getElementById('myCanvas');
    var ctx = canvas.getContext('2d');
    ctx.canvas.width = 600;
    ctx.canvas.height = 400;
    var str = "hello world"
    //设置文本样式、比如大小,字体
    ctx.font = "bold 100px sans-serif";
    //水平对齐设置,left,center,top
    ctx.textAlign = "center";
    //垂直对齐设置,top,middle,bottom
    ctx.textBaseline = "top";
    ctx.fillStyle = "#FF0000";
    //填充文本
    ctx.fillText(str, 300, 100);
    ctx.strokeStyle = "#0000FF";
    //描边文本
    ctx.strokeText(str, 300, 300);
    //获取文本宽度
    console.log(ctx.measureText("慕课网").width);

2、canvas图片的绘制(图片的裁剪:要看图片的实际宽高)

注意:一定要在图像加载完成后的回调中会绘制图像,否则图像显示不出来

var canvas = document.getElementById('myCanvas');
    var ctx = canvas.getContext('2d');
    ctx.canvas.width = 600;
    ctx.canvas.height = 400;
    ctx.fillStyle = "#000000";
    ctx.fillRect(0, 0, ctx.canvas.width, ctx.canvas.height);
    var img = new Image();
    img.src = "logo.png";
    //一定要在图像加载完成后的回调中会绘制图像,否则图像显示不出来
    img.onload = function () {
      //如果是三个参数,表示以(0,0)在图像左上角开始绘制图像
      //如果是五个参数,表示在(0,0)点处绘制img图像,缩放(截取)成40*40
      //如果是九个参数,表示获取img图像(0,0)点处的40*40区域,绘制在(100,100)点处,缩放成80*80
      ctx.drawImage(img, 0, 0, 40, 40, 100, 100, 80, 80);

3、图形画刷

var canvas = document.getElementById('myCanvas');
    var ctx = canvas.getContext('2d');
    ctx.fillRect(0, 0, canvas.width, canvas.height);
    var img = new Image();
    img.src = "logo.png";
    img.onload = function () {
      //创建图形画刷,repeat,no-repeat,repeat-x,repeat-y
      var pattern = ctx.createPattern(img, "repeat");
      ctx.fillStyle = pattern;
      ctx.fillRect(0, 0,canvas.width,canvas.height);

七、canvas中剪辑、阴影以及曲线的绘制

 1、canvas中剪辑区域

var canvas = document.getElementById('myCanvas');
    var ctx = canvas.getContext('2d');
    //保存当前环境
    ctx.save();
    ctx.beginPath();
    ctx.arc(200, 150, 100, 0, Math.PI * 2, true);
    ctx.closePath();
    //进行区域剪辑
    ctx.clip();
    //恢复环境,释放了剪辑区域的作用
    ctx.restore();
    ctx.fillStyle = "#FF0000";
    ctx.fillRect(100, 100, 200, 200);
    ctx.fillStyle = "#0000FF";
    ctx.fillRect(200, 150, 200, 200);

2、canvas阴影绘制

//阴影的x轴偏移量
    ctx.shadowOffsetX = 10;
    //阴影的y轴偏移量
    ctx.shadowOffsetY = 10;
    //阴影的颜色
    ctx.shadowColor = 'rgba(0, 0, 0, 0.5)';
    //阴影的模糊程度
    ctx.shadowBlur = 1.5;
    ctx.fillStyle = 'rgba(255, 0, 0, 0.5)';
    ctx.fillRect(100, 100, 100, 100);

3、canvas绘制曲线

ctx.beginPath();
    ctx.arc(150, 150, 100, 0, Math.PI, true);
    ctx.stroke();
    // 二次样条曲线
    ctx.beginPath();
    ctx.moveTo(50, 350); //起始点 
    ctx.quadraticCurveTo(100, 250, 150, 350); // 前两个参数是可以调节的点,最后两个参数是终点
    ctx.stroke();
    showPoint(ctx, 50, 350);
    showPoint(ctx, 100, 250);
    showPoint(ctx, 150, 350);
    // 三次贝塞尔曲线
    ctx.beginPath();
    ctx.moveTo(200, 350);
    ctx.bezierCurveTo(200, 250, 300, 250, 300, 350); // 前四个参数是可以调节的点,最后两个参数是终点
    ctx.stroke();
    ctx.fill();
    showPoint(ctx, 200, 350);
    showPoint(ctx, 200, 250);
    showPoint(ctx, 300, 250);
    showPoint(ctx, 300, 350);

八、canvas动画(实现一个方块左右移动,鼠标放上去就停止的效果)

var canvas = document.getElementById('myCanvas');
    var ctx = canvas.getContext('2d');
    ctx.canvas.width = 600;
    ctx.canvas.height = 400;
    var posx = 0, posy = 0;
    var dir = 1; //判断方块是正向前进还是反向运动。正向就是1,反向就是-1
    var isMouseInReat //判断鼠标是否移动进了方块里面,是的话就停止运动
    canvas.onmousemove =function(e){
      var mouseX = e.offsetX;
      var mouseY = e.offsetY;
      if (mouseX>posx && mouseX<posx +50 && mouseY>posy && mouseY<posy +50){
        isMouseInReat = true;
      }else{
        isMouseInReat = false;
      }
    }
    setInterval(function() {
      if(!isMouseInReat){
        posx = posx + 10 * dir;
      }
      //clearRect清空画布的一个矩形区域
      ctx.clearRect(0, 0, ctx.canvas.width, ctx.canvas.height);
      ctx.fillRect(posx, posy, 50, 50);
      if (posx +50 >=ctx.canvas.width) {
        dir = -1;
      }else if(posx <= 0){
        dir = 1
      }
    }, 50);

九、canvas离屏技术(可以解决因为setInterval多次循环卡顿的问题)

// 要显示的canvas
    var canvas = document.getElementById('myCanvas');
    var ctx = canvas.getContext('2d');
    ctx.canvas.width = 600;
    ctx.canvas.height = 400;
    // 不显示的canvas(离屏)用dispaly:none隐藏即可
    var offCanvas = document.getElementById('offCanvas');
    var offCtx = offCanvas.getContext('2d');
    offCtx.canvas.width = 600;
    offCtx.canvas.height = 400;
    
    ctx.fillStyle = "#FF0000";
    ctx.fillRect(0, 0, ctx.canvas.width, ctx.canvas.height);
    offCtx.strokeStyle = "#00FF00";
    offCtx.lineWidth = 10;
    offCtx.moveTo(0, 0);
    offCtx.lineTo(offCtx.canvas.width, offCtx.canvas.height);
    offCtx.stroke();
    
    // 把离屏canvas的内容搬过来
    ctx.drawImage(offCanvas, 0, 0, ctx.canvas.width, ctx.canvas.height,
                  0, 0, offCtx.canvas.width, offCtx.canvas.height);