canvas游戏小试:画一个按方向键移动的圆点

canvas游戏小试:画一个按方向键移动的圆点

自己对canvas,但又有一颗做游戏的心TT。然后记录一下对canvas的学习吧,用一个按方向键控制的小圆点来做练习。(编程时用了一些es6的语法)

示例的html很简单,只有一个canvas元素:

<html>
    <head>
        <link rel="stylesheet" href="css/base.css">
        <link rel="stylesheet" href="css/index.css">
        <script src="js/commons.js" charset="utf-8"></script>
        <script src="js/main.js"></script>
    </head>
    <body>
        <header></header>
        <canvas id="canvas" width=1000 height=500></canvas>
    </body>
</html>

这里可以看到我在canvas标签里直接定义了宽和高,这和在css里面定义是不同的,canvas元素其实有两套大小

1.元素本身大小

2.绘画表面大小

默认情况下canvas的绘画表面大小是300x150像素,在css设置宽和高只能修改元素本身大小,但绘画表面大小不变,这样就会使浏览器对绘画表面进行缩放来适应元素本身的大小。

所以要定义宽和高要定义在标签或者在js里面定义,如下。

var canvas=document.getElementById("canvas");
canvas.width=window.innerWidth;
canvas.height=window.innerHeight;

然后我们来说逻辑的部分,其实比较简单,但作为一个可继续发展的游戏雏形,我们利用面向对象编程的思想

定义engine类,来表示游戏的入口,sprite类表示游戏中的对象,listener类来监听游戏的事件

依照顺序逻辑,先看listener类:

class Listener{
  constructor(key,callback){
    this.key = key ;
    this.callback = callback ;
  }

  run(){
    this.callback() ;
  }

  getKey(){
    return this.key ;
  }
}

export {Listener}

主要有两个对象,一个是它的key值,用来说明它是干什么的监听器,另外是一个回调函数,用来触发事件

sprite类

import {Listener} from ‘./listener‘

class Sprite{

  constructor(context,x,y,imgUrl,speed){
    this.x = x ;
    this.y = y ;
    this.imgUrl = imgUrl ;
    this.speed = speed||10 ;
    this.listeners = [] ;
    this.context = context ;
    this.drawImage() ;
    this.initListener() ;
  }

  drawImage(){
    this.context.fillStyle = ‘black‘ ;
    this.context.beginPath();
    this.context.arc(this.x,this.y,5,0,2*Math.PI,true);//radius = 5
    this.context.closePath();
    this.context.fill();
  }

  update(x,y){
    this.context.clearRect(this.x-5,this.y-5,10,10);
    this.context.beginPath();
    this.context.arc(x,y,5,0,2*Math.PI,true);
    this.context.closePath();
    this.context.fill();
    this.x = x ;
    this.y = y ;
  }

  addListener(keyListener){
    this.keyListenerList.push(keyListener) ;
  }

  findKeyListener(key){
    for(let i in this.listeners){
      if(this.listeners[i].getKey()===key){
        return this.listeners[i] ;
      }
    }
    return null ;
  }
  //default listener
  initListener(){
    this.listeners[‘up‘] = new Listener(‘up‘,()=>{
      this.update(this.x,this.y-this.speed) ;
    });
    this.listeners[‘down‘] = new Listener(‘down‘,()=>{
      this.update(this.x,this.y+this.speed) ;
    });
    this.listeners[‘left‘] = new Listener(‘left‘,()=>{
      this.update(this.x-this.speed,this.y) ;
    });
    this.listeners[‘right‘] = new Listener(‘right‘,()=>{
      this.update(this.x+this.speed,this.y) ;
    });
  }

}

export {Sprite}

精灵类中引用了之前定义的监听类,然后定义了“上下左右”这是个默认监听对象来加入到这个精灵自身的监听列表中,正常游戏是用帧动画的,我们这先用一个圆来代替~。

drawImage是画圆,在构造函数中调用,来展示形象。update函数来更新圆的位置,其实是把原先的圆清掉重画一次,它被监听器触发。

findKeyListener这个函数是用来遍历自己的监听器列表的,里面值得说一下的是循环我用的for in,这是因为我在下面定义默认监听器的时候键值用的stirng而不是数字。如果是正常的[0.....n]这样以数字为索引的数组的话,建议用es6的for of来遍历

for (var value of Array) {
  console.log(value);//不是key,而是值
}

engine类

import {Sprite} from ‘./sprite‘

class Engine{

  constructor(canvasId){
    this.canvas = document.getElementById(canvasId) ;
    this.context = this.canvas.getContext(‘2d‘) ;
    this.spriteList = [] ;
    this.keyListenerList = [] ;
    //time
    this.startTime = 0 ;
    this.lastTime = 0 ;
    this.currentTime = 0 ;
    this.FPS = 30 ;
    //height and width
    this.bgHeight = this.canvas.height ;
    this.bgWidth = this.canvas.width ;
  }

  //sprite
  addSprite(x,y,imgUrl,speed){
    var sprite = new Sprite(this.context,x,y,imgUrl,speed)
    this.spriteList.push(sprite) ;
  }

  //keylistener
  keyPressed(e){
    let listener = undefined ;
    let key = "" ;

    switch (e.keyCode){
      case 32: key = "space" ; break ;
      case 37: key = "left" ; break ;
      case 38: key = "up" ; break ;
      case 39: key = "right" ; break ;
      case 40: key = "down" ; break ;
      case 13: key = "enter" ; break ;
    }

    for(let sprite of this.spriteList){
      listener = sprite.findKeyListener(key) ;
      if(listener){
        listener.run() ;
      }
    }
  }
}

export {Engine}

在engine类里定义添加精灵的方法,并处理外界传来的事件,里面可能有一些定义了但没用到的变量,以后会用到的,不过engine就是整个游戏的入口,总而言之在mian.js中只要引入engine就能让整个效果跑起来。

最后的main.js

import {Engine} from ‘./gameEngine‘

$(function(){
  init() ;
});

function init(){
  initGame() ;
}

function initGame(){
  var engine = new Engine(‘canvas‘) ;
  engine.addSprite(10,10,null,10) ;
  $(document).keydown(function (e) {
    engine.keyPressed(e) ;
  });
}

时间: 2024-08-07 16:48:44

canvas游戏小试:画一个按方向键移动的圆点的相关文章

玩转html5(四)----使用canvas画一个时钟(可以动的哦!)

先给个效果图,我画的比较丑,大家可以自己美化一下, 直接上代码: <!DOCTYPE html> <meta charset="utf-8"> <html> <body> <canvas width="500" height="500" id="clock" > 您的浏览器不支持canvas </canvas> <script> //获取画布

canvas 画一个小时钟

<!DOCTYPE html><html lang="en"><head> <meta charset="UTF-8"> <title>Title</title></head><body> <canvas id="mycanvas" height="1000" width="1000"></ca

canvas之画一个三角形

1 <canvas id="canvas" width="500" height="500" style="background-color: yellow;"></canvas> 1 var canvas=document.getElementById("canvas"); 2 var cxt=canvas.getContext("2d"); 3 cxt.beg

10分钟,利用canvas画一个小的loading界面(顺便讨论下绘制效率问题)

首先利用定义下canvas得样式 <canvas width="1024" height="720" id="canvas" style="border: 1px solid #808080;display: block;margin: 100px auto;>你的游览器不支持canvas</canvas> 这里主要要说的就是宽高,不要在style里面定义,不然会被拉伸.(对于这点,建议大家看下W3c文档,不是很

10分钟,利用canvas画一个小的loading界面

首先利用定义下canvas得样式 <canvas width="1024" height="720" id="canvas" style="border: 1px solid #808080;display: block;margin: 100px auto;>你的游览器不支持canvas</canvas> 这里主要要说的就是宽高,不要在style里面定义,不然会被拉伸.(对于这点,建议大家看下W3c文档,不是很

利用canvas画一个动态时钟

目标:利用canvas画布画一个动态时钟,根据目前的时间可以实时更新的,可以在过程中添加一些效果,比如让时钟外围的一圈颜色渐变,时钟上的数字颜色改变,时钟的指针颜色改变... 设置一个定时器 先放上一张效果图,参考一下 先建一个画布,写好样式 <style type="text/css"> *{ margin: 0; padding: 0; } div{ //设置div的text-align为center,margin-top text-align: center; mar

手对手的教你用canvas画一个简单的海报

啦啦啦,首先说下需求,产品想让用户在我们app内,分享一张图片到微信.qq等平台.图片中包含用户的姓名.头像.和带着自己信息的二维码.然后,如何生成这张海报呢~~~首先我们老大告诉我有一个插件叫html2canvas.其作用就是可以将dom节点转化成图片,是个不错的东西.我试验了下.确实可以,但~这个插件有点大啊,为了满足海报的这个需求引入这么大的东西感觉很亏!!!所以,还是自己画一个~首先先上效果图 当当当当~~~最后生成的海报中包括头像和姓名与二维码,当然图上的二维码是百度的二维码~最后生成

Directx11学习笔记【十二】 画一个旋转的彩色立方体

上一次我们学习了如何画一个2D三角形,现在让我们进一步学习如何画一个旋转的彩色立方体吧. 具体流程同画三角形类似,因此不再给出完整代码了,不同的部分会再说明. 由于我们要画彩色的立方体,所以顶点结构体中加入颜色变量 struct Vertex { XMFLOAT3 pos; XMFLOAT4 color; }; 着色器代码 1 cbuffer cbPerObject 2 { 3 float4x4 gWorldViewProj; 4 }; 5 6 struct VertexIn 7 { 8 flo

HTML5 Canvas游戏开发(一)基础知识

一.绘制基本图形 在每次用canvas画布时,都有几步是“套路” 1.在HTML中创建Canvas画布: <canvas id="mycanvas" width="960px" height="580px"> 浏览器不支持canvas <!-- 如果不支持会显示这段文字 --> </canvas> 2.获取画布标签,并得到一个2D对象: var c = document.getElementById('myca