最近作了一个 H5 小游戏,游戏的玩法是小汽车抢停车位,须要手指拖拽小汽车躲避障碍快速进入停车位。javascript
在线预览:go.163.com/web/2019051…html
扫码跳过视频体验小游戏java
游戏复杂的部分可能就是小红车和障碍物的碰撞检测,由于地图不是固定的,会有不一样的游戏地图。刚开始想把障碍物切成一块一块的,而后给几种排列方式,但须要测量不少数据,以为仍是有些复杂。git
以前 小建老师 和我提到过不少次用 getImageData()
实现的神奇功能,以后我用 getImageData()
实现了点击涂色,以为这个方法确实神奇且好用,因此此次碰撞检测也用 getImageData()
实现了一下。github
getImageData()
是 canvas
的一个方法,能够读取到图片指定矩形的数据,数据指的是 rgba
数据。web
有了 getImageData()
这个方法的助攻,就能够在 PS 中新建一个透明图层,给小车行驶的路上涂上一个特定的颜色,而后小车每次移动,读取小车位置所在像素的颜色,就知道小车在不在道路上了。canvas
在 PS 中新建一个透明图层,把小车能够行驶的路上,涂上白色,给停车位涂上红色,而后保存为 png 图片。下图左侧为游戏地图,右测为对应的涂色图片,两张图片尺寸保存相同: 布局
碰撞检测关键:ui
function areaJudge(x,y){ // 传入小车的位置
return ((resolve)=>{
const canvas = document.createElement('canvas'); // 建立一个canvas,将咱们涂好色的透明图片画到canvas
const ctx = canvas.getContext('2d');
canvas.width = WIDTH; // WIDTH 为图片的宽度
canvas.height = HEIGHT; // HEIGHT 为图片的高度
const path = new Image();
path.onload = ()=>{
ctx.drawImage(path,0,0);
// 参数说明:getImageData(小车x坐标,小车y坐标,获取1像素宽度的数据,获取1像素高度的数据)
// data:[r,g,b,a] 图像数据为 rgba
if(ctx.getImageData(x,y,1,1).data[3]===0){ // 获取当前像素的透明度,为 0 说明处于障碍区域
resolve(0);
}else if(ctx.getImageData(x,y,1,1).data[0]===255){// 获取当前像素的透明度,为 255 说明处于能够行驶的区域
resolve(1);
}else{ // 红色,停车位的位置
resolve(2);
}
};
path.src = psImgSrc; // psImgSrc 为咱们在 ps 涂色的透明图片
});
}
复制代码
而后在小车移动的过程当中,经过碰撞检测判断小车处于可行驶区域、故障区域仍是停车位:this
// 此处使用了 pixi.js 的 pointermove ,同理可使用 touchmove
car.on('pointermove',(e)=>{
// 此处 car.x 为小车中心的位置,省略了计算代码
this.areaJudge(car.x,car.y).then((resolve)=>{
if(resolve===1){
// 可行驶区域
}else if(resolve===0){
// 不可行驶区域
}else{
// 车位
}
});
});
复制代码
以上,就使用 getImageData()
实现了碰撞检测。
getImageData()
能够读取图片的颜色与透明度数据,这样节省了咱们页面障碍物的布局,能够少测量不少数据;getImageData()
的功能很强大,能够考虑扩展到其它应用场景。