web游戏---canvas基础图形

news2025/1/9 2:28:36

基础

canvas标签

canvas是H5中新推出的标签,这个提供一块画布,可以在上面绘制图案,通过这种方式制作web游戏带来的性能消耗比操作DOM要小的多。

如果知做浏览器游戏,为了保证性能最好使用画布来制作。

坐标系

画布的坐标系和浏览器相同,以画布的左上角为原点。向左x坐标递增,向右y坐标递增,默认单位为像素。

绘图方式

canvas使用命令式编程的方式,使用语句说明出作图的路径和画笔的属性,最后进行着色并闭合路径。下面canvas的直线绘制流程。

// 获取画布
const canvas = document.querySelector('canvas');
// 获取画笔
const ctx = canvas.getContext('2d');
/ 开启路径
ctx.beginPath();
// 移动画笔到某点
ctx.moveTo(0, 0);
// 画直线到某点
ctx.lineTo(800, 600);
// 设置直线的颜色,设置颜色要在上色之前
ctx.strokeStyle = 'lightpink';
// 设置线条宽度,也要在上色之前设置
ctx.lineWidth = 5;
// 上色
ctx.stroke();
// 闭合路径
ctx.closePath();  

绘图函数

直线和曲线

上面举的例子我们绘制了一个直线,但我们可以发现上面的步骤很多都是重复的,没有必要重复得写,因此我们可以将其封装为一个函数。

// 可以画实线就可以画虚线,虚线本质上就是多条很短的直线,canvas内部提供了一个绘制虚线的函数,只需要我们将封装的函数进行一些修改
// 使用setLineDash() 函数可以设置一条虚线的长度和间距组合,这是对画笔的设置,会影响到之后的所有线条
// 使用getLineDash() 函数可以获取到当前路径下的虚线参数信息
function drawLine(x1, y1, x2, y2, color, width, dashed) {
  ctx.beginPath();
  ctx.moveTo(x1, y1);
  ctx.lineTo(x2, y2);
  if(dashed) ctx.setLineDash(dashed);
  ctx.strokeStyle = color;
  ctx.lineWidth = width;
  ctx.stroke();
  // 取消掉之前的曲线配置,防止影响下一条线
  ctx.setLineDash([0,0])
  ctx.closePath();
}

在连续调用4次之后我们可以绘制一个虚线组成的矩形边框

drawLine(100,100,400,100,'pink',5,[5,10]);
drawLine(400,100,400,400,'purple',5);
drawLine(400,400,100,400,'orangered',5,[3,4]);
drawLine(100,400,100,100,'blue',5,[4,6]);

在这里插入图片描述

矩形

上面我们绘制了一个四条边都不同的矩形,但如果是绘制普通的矩形,使用上面的方法就过于烦琐,我们可以使用canvas使用的矩形API来绘制。

绘制矩形有三个方法:

  • rect() 绘制空矩形,可以分别描边和填充
  • strokeRect() 绘制空心矩形,不能填充,会自动描边
  • fillRect() 绘制实心矩形,不能描边,会自动填充

下面的代码是一个矩形的绘制流程

ctx.beginPath();
ctx.rect(100,100,300,300);
// 设置填充颜色
ctx.fillStyle = 'skyblue';
ctx.fill();
// 设置描边颜色
ctx.strokeStyle = 'lightpink';
// 设置线宽
ctx.lineWidth = 5;
ctx.stroke();
ctx.closePath();

圆和圆弧

使用arc函数可以回执圆形或圆弧,共接收6个参数,具体的参数如下:

  • 圆心的x坐标和y坐标
  • 圆的半径
  • 圆的起点角度和终点角度(弧度制,三点钟方向为0)
  • 圆的绘制方向(选填,默认为逆时针)true表示逆时针,false表示顺时针

下面是一个例子,可以绘制一个半圆:

    ctx.strokeStyle = 'lightpink';
    ctx.lineWidth = 5;
    ctx.arc(250,250,200,0,pi);
    ctx.stroke();

在这里插入图片描述

画布清除

通过claerRect()可以清除一个矩形区域。

清除一个画布大小的矩形就可以清除整个画布

claerRect(0,0,width,height)

canvas动画

计时器动画

以一个逐渐出现的圆作为例子来演示canvas是如何制作动画的

    const pi = Math.PI;
    const deg = pi * 2 / 360;
    ctx.strokeStyle = 'lightpink';
    ctx.lineWidth = 5;
    let count = 0;
    const timer = setInterval(() => {
      count++;
      ctx.beginPath();
      ctx.arc(250,250,200,0,deg * count);
      ctx.stroke();
      ctx.closePath();
      if(count == 360) clearInterval(timer);
    }, 10)

边缘碰撞检测

现在我想做一个小球碰到边框后就转向的动画,我们先做简单的与边界的碰撞,之后在去做两个物体之间的碰撞。

下面为实现的代码,有状态法和速度法两种实现方法,推荐使用速度法,与物理的运动学一致,且代码更易懂:

	// 状态法通过修改状态的真值来决定运动方向
	// let statusX = true;
    // let statusY = true;
    
	// 速度法通过速度的正负来决定运动方向,这种方法与物理知识更契合
    let speedX = 5;
    let speedY = 5;
    const timer = setInterval(() => { 
      // 清空画布
      ctx.clearRect(0,0,w,h);
      // 修改小球方向
      if(x + r == 500) speedX = -5 // statusX = false;
      if(x - r == 0)   speedX = 5  // statusX = true;
      if(y - r == 0)   speedY = 5  // statusY = true;
      if(y + r == 500) speedY = -5 // statusY = false;

      // 根据小球的移动方向修改小球的坐标
      // if(statusX) {
      //   x = x + 10;
      // } else {
      //   x = x - 10;
      // }

      // if(statusY) {
      //   y = y + 10;
      // } else {
      //   y = y - 10;
      // }

      x += speedX;
      y += speedY;

      // 绘制小球
      ctx.beginPath();
      ctx.arc(x, y, r, 0, 2*pi);
      ctx.fill();
      ctx.closePath();
    },50)

使用速度法还可以使用取相反数反的方法来修改速度,或者更方便为小球设置不同的速度

// 设置小球速度
    let speedX = 5;
    let speedY = 8;
// 修改小球方向
    if(x + r >= 500 || x - r <= 0) speedX = -speedX 
    if(y - r <= 0 || y + r >= 500) speedY = -speedY  

面向对象

如果要制作多个小球,就需要用到面相对象的思想,将每个小球作为对象存储起来,这样就可以解决因为是一幅画而无法获取的问题。
代码如下:

	class Ball {
      constructor() {
        this.r = this.ran(90) + 10;
        // 防止出现边缘性问题
        this.x = this.ran(200) + this.r;
        this.y = this.ran(200) + this.r;
        
        this.color = `#${parseInt(Math.random() * 0xffffff).toString(16)}`;
        this.speedX = this.ran(5) + 2;
        this.speedY = this.ran(4) + 1;
        this.move();
      }

      ran(number) {
        return Math.random() * number;
      }

      show() {
        ctx.beginPath();
        ctx.arc(this.x, this.y, this.r, 0, pi*2);
        ctx.fillStyle = this.color;
        ctx.fill();
        ctx.closePath();
      }

      move() {
        if(this.x - this.r <= 0 || this.x + this.r >= w) this.speedX = -this.speedX;
        if(this.y - this.r <= 0 || this.y + this.r >= h) this.speedY = -this.speedY;
        this.x += this.speedX;
        this.y += this.speedY;
        this.show();
      }
    }
    
    // 生成并存储小球
    const ballArr = [];
    for(let i = 0; i < 100000; i++){
      ballArr.push(new Ball());
    }

    // 小球定时运动
    setInterval(() => {
      ctx.clearRect(0,0,w,h);
      ballArr.forEach(item => {
        item.move();
      })
    }, 10)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/179781.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

ThinkPadE540重装系统

过年这段时间&#xff0c;帮家里人重装了一下win10系统&#xff0c;在这里记录一下&#xff0c;方便今后还要使用。 先准备两个U盘&#xff0c;一个存储电脑的文件&#xff08;以防文件丢失&#xff09;&#xff0c;一个空u盘&#xff08;制作重装系统的&#xff09; 一.下载镜…

【5-卷积神经网络】北京大学TensorFlow2.0

课程地址&#xff1a;【北京大学】Tensorflow2.0_哔哩哔哩_bilibiliPython3.7和TensorFlow2.1六讲&#xff1a;神经网络计算&#xff1a;神经网络的计算过程&#xff0c;搭建第一个神经网络模型神经网络优化&#xff1a;神经网络的优化方法&#xff0c;掌握学习率、激活函数、损…

Junit单元测试框架【基础篇】

Junit单元测试框架【基础篇】&#x1f34e;一.Junit单元测试框架&#x1f352;1.1 注解&#x1f352;1.2 断言&#x1f352;1.3 用例执行顺序&#x1f352;1.4 测试套件&#x1f349;1.4.1 指定类&#x1f349;1.4.1 指定包&#x1f352;1.5 参数化&#x1f349;1.5.1 单参数&a…

VBA提高篇_07 Goto跳转 / Exit退出 /VBA错误处理

文章目录使用逻辑变量控制循环使用Goto语句任意跳转捷径:使用Exit语句跳出结构保险: 使用错误处理改善用户体验On Error Goto Lablex:On Error Resume Next使用逻辑变量控制循环 使用Goto语句任意跳转 经常在错误处理时使用 捷径:使用Exit语句跳出结构 注意: 避免使用while…w…

【C++】AVL树(插入)

文章目录AVL树的概念平衡化旋转右单旋转左单旋转先左后右双旋转先右后左双旋转AVL树的插入根据BST树规则进行节点插入平衡化处理重新连接节点完整的插入函数代码AVL树的验证AVL树的性能AVL树的概念 二叉搜索树虽然可以提高查找的效率&#xff0c;但是二叉搜索树有其自身的缺陷&…

Python与Matlab混合编程案例

前言因为项目需要&#xff0c;需要批处理很多Matlab的.m文件&#xff0c;从每个文件中提取结果合并到一个文件中。 很明显&#xff0c;如果手工统计&#xff0c;几百个文件会累死的。 因此立即想到了Python在批处理方面的优势&#xff0c;因此就在网上找了相关资料&#xff0c;…

C++初阶:vector

文章目录1 vector介绍2 实现vector2.1 类的定义2.2 默认成员函数2.2.1 构造函数2.2.2 析构函数2.2.3 拷贝构造2.2.4 赋值重载2.3访问接口2.4 容量接口2.5 修改接口2.5.1 尾插尾删2.5.2 任意位置插入2.5.3 任意位置删除2.6 其他接口1 vector介绍 1 vector是表示可变大小数组的序…

10+编程语言实现云笔记

目标 为编程初学者打造入门学习项目&#xff0c;使用各种主流编程语言来实现。让想学编程的&#xff0c;一个都不落下。 上述基本涵盖了当前编程开发所有主流语言。 左侧为前端版本&#xff1a;安卓、iOS、鸿蒙、Flutter、Vue、uni-app。 右侧为服务器端版本&#xff1a;Jav…

代码随想录算法训练营三期 day 27 - 回溯 (3) (补)

39. 组合总和 题目链接&#xff1a;39. 组合总和 原文链接&#xff1a;39. 组合总和 视频链接&#xff1a;39. 组合总和 本题和 77.组合 &#xff0c;216.组合总和III 的区别是&#xff1a;本题没有数量要求&#xff0c;可以无限重复&#xff0c;但是有总和的限制。 树形结构&…

【axios】axios的基础知识和使用

一、基础知识概念Axios 是专注于网络数据请求的库,只负责发请求、拿数据&#xff0c;不能操作DOM元素。相比于原生的 XMLHttpRequest 对象&#xff0c;axios 简单易用。相比于 jQuery&#xff0c;axios 更加轻量化&#xff0c;不能操作DOM元素&#xff0c;只专注于网络数据请求…

cubeIDE开发, stm32人工智能开发应用实践(Cube.AI).篇二

一、事有蹊跷 接篇一&#xff0c;前面提到在使用cube.AI生成的c语言神经网络模型API调用时&#xff0c;输入数据数量是24&#xff0c;输出数据数量是4&#xff0c;但上文设想采集了三轴加速度传感器的x/y/z三个各数据&#xff0c;按Jogging(慢跑),Walking(走了)两种态势采集了两…

Java链表OJ题

目录1. 删除链表中等于给定值val的所有结点2. 逆置单链表3. 链表的中间结点4. 链表中倒数第k个结点5. 将两个有序链表合并为一个新的有序链表6. 以给定值x为基准将链表分割成两部分7. 判断是否为回文链表8. 两个链表的第一个公共结点9. 判断链表中是否有环10. 链表开始入环的第…

【Linux】目录权限和默认权限

上期介绍了Linux的文件权限&#xff0c;这期我们仔细来说说Linux环境下目录权限和默认权限一、目录权限1.1 进入目录所需的权限我们在进入目录时需要什么样的权限呢&#xff1f;是r、w还是x呢&#xff1f;下面我们一起来验证一下&#xff1a;&#x1f4cb;如下我门拥有全部目录…

Day11 AOP介绍

1 前言AOP&#xff0c;Aspect Oriented Programming&#xff0c;面向切面编程&#xff0c;是对面向对象编程OOP的升华。OOP是纵向对一个事物的抽象&#xff0c;一个对象包括静态的属性信息&#xff0c;包括动态的方法信息等。而AOP是横向的对不同事物的抽象&#xff0c;属性与属…

【Python从入门到精通】第一阶段

文章目录前言python的起源打印hello world注释变量变量基本概念类型类型转换运算符字符串拓展字符串的三种定义方法字符串拼接字符串格式化数据输入input比较布尔类型和比较运算符if判断if elseif elif else嵌套循环while循环while循环嵌套for循环range()的使用函数的使用函数的…

3小时精通opencv(五) 利用TrackBar进行颜色检测

3小时精通opencv(五) 利用TrackBar进行颜色检测 参考视频资源:3h精通Opencv-Python 本章内容介绍如何利用TrackBar调节色域, 手动提取到我们需要的颜色 文章目录3小时精通opencv(五) 利用TrackBar进行颜色检测创建Trackbar色彩检测创建Trackbar 在opencv中使用createTrackbar函…

C语言:数组

往期文章 C语言&#xff1a;初识C语言C语言&#xff1a;分支语句和循环语句C语言&#xff1a;函数 目录往期文章前言1. 一维数组的创建和初始化1.1 数组的创建1.2 数组的初始化2. 一维数组的使用3. 一维数组在内存中的存储4. 二维数组的创建和初始化4.1 二维数组的创建4.2 二维…

大数据技术架构(组件)7——Hive:Filter PushDown Cases And Outer Join Behavior

1.2、Filter PushDown Cases And Outer Join Behavior前提:关闭优化器set hive.auto.convertjoinfalse; set hive.cbo.enablefalse;Inner Join:1、Join On中的谓词: 左表下推、右表下推2、Where谓词:左表下推、右表下推-- 第一种情况: join on 谓词 selectt1.user_id,t2.user_i…

C++函数定义和调用介绍

C函数定义和调用介绍 函数的意义&#xff1a;利用率高&#xff0c;可读性强&#xff0c;利于移植。 一个C程序中主函数有且只有一个&#xff0c;是程序的入口&#xff0c;而函数&#xff08;或称子函数&#xff09;可以有很多。 每个 C 程序都至少有一个函数&#xff0c;即主…

2021 XV6 8:locks

实验有两个任务&#xff0c;都是为了减少锁的竞争从而提高运行效率。Memory allocator一开始我们是有个双向链表用来存储空闲的内存块&#xff0c;如果很多个进程要竞争这一个链表&#xff0c;就会把效率降低很多。所以我们把链表拆成每个CPU一个&#xff0c;在申请内存的时候就…