《JS原理、方法与实践》- canvas游戏

简介: 《JS原理、方法与实践》- canvas游戏

游戏其实就是在动画的基础上添加了控制,也就是添加键盘和鼠标的事件监听。添加事件的方法前面已经学过,添加键盘事件和普通节点对象的键盘事件相同,只是canvas中的鼠标事件需要做一些处理。

#### 鼠标事件

在鼠标事件中,鼠标指针所处位置的坐标是非常重要的属性,鼠标事件中只能获取相对于屏幕左上角和相对浏览器文档左上角的坐标,而canvas中使用的是自己的坐标系,因此需要将获取的坐标转换为canvas中的坐标。在转换之前,首先获取canvas在浏览器中的位置,可以通过getBoundingClientRect方法获取。例如下面的方法将浏览器中的坐标转换为canvas中的坐标:

```javascript

       function convertToCanvas(canvas, x, y){

           var canvasElement = canvas.getBoundingClientRect();

           return {

               x: (x - canvasElement.left) * (canvas.width / canvasElement.width),

               y: (y - canvasElement.top) * (canvas.height / canvasElement.height)

           };

       }

```

转换逻辑:

* 先使用鼠标事件中相对于浏览器文档的坐标减去canvas左上角的坐标

* 然后进行相应的缩放

示例:

点中圆中5分,没点中扣3分。

```html

<!DOCTYPE html>

<html lang="en">

<head>

   <meta charset="UTF-8">

   <meta name="viewport" content="width=device-width, initial-scale=1.0">

   <title>Document</title>

</head>

<body>

   <canvas id='c2d' width="300" height="300">浏览器不支持canvas</canvas>

   <script>

       const canvas = document.getElementById('c2d');

     

       function convertToCanvas(canvas, x, y){

           var canvasElement = canvas.getBoundingClientRect();

           return {

               x: (x - canvasElement.left) * (canvas.width / canvasElement.width),

               y: (y - canvasElement.top) * (canvas.height / canvasElement.height)

           };

       }

     

       if (canvas.getContext) {

           let ctx = canvas.getContext('2d');

           let path, scoreArea = {w:300, h:50};

           function drawObj(){

               const offsetX = 0, offsetY = scoreArea.h;

             

               ctx.save();

               ctx.clearRect(offsetX, offsetY, 300,300);

               const r = 30;

               const x = r + offsetX + Math.round(Math.random() * (canvas.width - 2 * r - offsetX));

               const y = r + offsetY + Math.round(Math.random() * (canvas.height - 2 * r -offsetY));

               path = new Path2D();

               path.arc(x,y,r,0,2*Math.PI);

               ctx.stroke(path);

           }

           window.setInterval(drawObj, 2000);

           let score = 3;

           function drawScore(isTrue){

               score += isTrue ? 5 : -3;

               ctx.save();

               ctx.fillStyle = 'red';

               ctx.clearRect(0,0,scoreArea.w, scoreArea.h);

               ctx.fillText('得分:' + score, 30,30);

               ctx.restore();

           }

           drawScore(false);

           canvas.onclick = function(event){

               const p = convertToCanvas(canvas, event.pageX, event.pageY);

               drawScore(ctx.isPointInPath(path,p.x,p.y));

           }

       }

   </script>

</body>

</html>

```

![](https://upload-images.jianshu.io/upload_images/2789632-8b2703200006b4e9.gif?imageMogr2/auto-orient/strip)

目录
相关文章
|
7天前
|
JavaScript 前端开发
javascript操作BOM的方法
javascript操作BOM的方法
15 0
|
2天前
|
JavaScript 前端开发
JavaScript 提供了多种方法来操作 DOM(文档对象模型)
【5月更文挑战第11天】JavaScript 用于DOM操作的方法包括获取元素(getElementById, getElementsByClassName等)、修改内容(innerHTML, innerText, textContent)、改变属性、添加/删除元素(appendChild, removeChild)和调整样式。此外,addEventListener用于监听事件。注意要考虑兼容性和性能当使用这些技术。
6 2
|
3天前
|
JavaScript 前端开发
在JavaScript中实现模块化开发有多种方法
JavaScript模块化开发可通过CommonJS、AMD和ES6模块实现。CommonJS适用于服务器端,使用`require`和`module.exports`处理模块;AMD(如RequireJS)用于浏览器端,依赖`require`和`define`;ES6模块提供原生支持,使用`import`和`export`。选择方式需考虑项目环境、复杂度和技术栈。
10 4
|
3天前
|
JavaScript 前端开发
深入了解前端框架Vue.js的响应式原理
本文将深入探讨Vue.js前端框架的核心特性之一——响应式原理。通过分析Vue.js中的数据绑定、依赖追踪和虚拟DOM等机制,读者将对Vue.js的响应式系统有更深入的理解,从而能够更好地利用Vue.js构建灵活、高效的前端应用。
|
4天前
|
前端开发 JavaScript
JS长按保存canvas绘图
JS长按保存canvas绘图
7 0
|
4天前
|
JavaScript 前端开发
JS tostring()和join()方法
JS tostring()和join()方法
7 1
|
4天前
|
存储 JavaScript 前端开发
深入了解JavaScript中的indexOf()方法:实现数组元素的搜索和索引获取
深入了解JavaScript中的indexOf()方法:实现数组元素的搜索和索引获取
7 0
|
5天前
|
JavaScript 前端开发 索引
js添加、删除、替换、插入元素的方法
js添加、删除、替换、插入元素的方法
10 0
|
6天前
|
JavaScript 前端开发
JavaScript 循环方法详解
JavaScript 循环方法详解
18 1
|
6天前
|
JavaScript 前端开发
JavaScript数字方法详解
JavaScript数字方法详解
17 0