《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)

目录
相关文章
|
14天前
|
存储 JavaScript 索引
JS中数组的相关方法介绍
JS中数组的相关方法介绍
|
16天前
|
JavaScript 前端开发 容器
AJAX载入外部JS文件到页面并让其执行的方法(附源码)
AJAX载入外部JS文件到页面并让其执行的方法(附源码)
17 0
|
17天前
|
JSON JavaScript 前端开发
JavaScript原生代码处理JSON的一些高频次方法合集
JavaScript原生代码处理JSON的一些高频次方法合集
|
1月前
|
JavaScript 前端开发
解释 JavaScript 中的`map()`、`filter()`和`reduce()`方法的用途。
解释 JavaScript 中的`map()`、`filter()`和`reduce()`方法的用途。
19 1
|
14天前
|
JavaScript 前端开发 索引
JavaScript中与字符串相关的方法
JavaScript中与字符串相关的方法
|
1月前
|
JavaScript
JS数组增删方法的原理,使用原型定义
JS数组增删方法的原理,使用原型定义
|
3天前
|
前端开发 JavaScript 编译器
深入解析JavaScript中的异步编程:Promises与async/await的使用与原理
【4月更文挑战第22天】本文深入解析JavaScript异步编程,重点讨论Promises和async/await。Promises用于管理异步操作,有pending、fulfilled和rejected三种状态。通过.then()和.catch()处理结果,但可能导致回调地狱。async/await是ES2017的语法糖,使异步编程更直观,类似同步代码,通过事件循环和微任务队列实现。两者各有优势,适用于不同场景,能有效提升代码可读性和维护性。
|
7天前
|
JavaScript 前端开发 Java
js 垃圾回收机制的方法
js 垃圾回收机制的方法
|
7天前
|
JavaScript 前端开发
js数据类型有几类?一共有几种?判断数据类型的方法是什么?
js数据类型有几类?一共有几种?判断数据类型的方法是什么?
|
8天前
|
JavaScript 前端开发
js绑定事件的方法
js绑定事件的方法
22 11