【上手】用 Tiny 写个 DoodleJump 给你看
keyword: 碰撞
简单的运动系统
重力感应事件
键盘事件
ES5
no-wewbpack
no-babel
一、需求说明
- 有一个 Doodle 的小玩偶,他会负责来回在跳板上跳来跳去,移动端依靠重力感应,电脑端依靠方向键。
- 有各种各样的跳板,不同的跳板与小玩偶发生碰撞的时候会有不同的行为发生。
- 有三个面板:Start、Main、End。
二、来吧游戏跑起来
使用 tinyjs-cli 工具可以快速搭建一个适合 Tiny 游戏开发的项目框架。
$ npm install tinyjs-cli -g
$ tiny init
目录中你可以看到
+-res
| |
| +--images // 图片资源目录
| +--sounds // 音频资源目录
|
+-src
| |
| + Resource.js // 加载资源用
| + StartLayer.js // 起始页面,写代码就从这里开始
|
+-index.html //入口HTML
|
+-main.js //启动Tiny的JS
|
+-...
三、资源加载与游戏启动
1、资源加载
<body>
<div class="bar">
<i id="progress"></i>
</div>
<p id="percent">Loading..</p>
<script src='https://gw.alipayobjects.com/as/g/tiny/tiny/1.1.7/tiny.js'></script>
<script src='src/Resource.js'></script>
<script src='src/StartLayer.js'></script>
<script src='main.js'></script>
</body>
在Resource.js中可以看到
var RESOURCES = {
// images,
"s_Tiny.js_png": "res/images/Tiny.js.png",
// sounds
};
你可以在这里引入你想要使用的资源,也可以是 CDN 地址。当然你可能会用到雪碧图。请参考 tinyjs-cli 提供的工具tiny resource
。
main.js
2、游戏启动 游戏启动配置,配置项可以参考 这里
var config = {
showFPS: true,
width: window.innerWidth,
height: window.innerHeight,
renderOptions: {
antialias: true,
backgroundColor: 0xFEF3E8,
},
};
Tiny.app = new Tiny.Application(config);
这里的 Tiny.app
要保持单例。一个游戏只启动一个 Application
。
加载资源与完善游戏 loading 动画。如果想更改游戏加载的动画可以修改这里的代码。
var resources = [];
for (var key in RESOURCES) {
resources.push(RESOURCES[key]);
}
var progress = document.getElementById('progress');
var percent = document.getElementById('percent');
Tiny.Loader.run({
resources: resources,
onProgress: function (pre, res) { //每加载完成一个资源更新一次
console.log('percent:', pre + '%', res.name);
var num = ~~pre;
//更新UI
percent.innerHTML = num + '%';
progress.style.width = num + '%';
},
onAllComplete: function () { // 全部加载完调用
console.log('all complete');
//clear DOM
var body = document.body;
body.removeChild(percent);
body.removeChild(progress.parentNode);
Tiny.app.run(new StartLayer());
},
});
在 app 还没有启动的时候,选择一个Tiny.app.run(layer)
来启动游戏。这里layer
一定要是Container
的子类。可以通过下面的方式继承。
var StartLayer = function () {
Tiny.Container.call(this);
};
StartLayer.prototype = Object.create(Tiny.Container.prototype);
StartLayer.prototype.constructor = StartLayer;
游戏启动之后,切换场景使用Tiny.app.replaceScene
方法,如果再次调用run
会发生异常。
四、启动页面
BitmapText
Event
rotation
position
1、关于漂亮的字体
一般的素材我们采用贴图的方式,但这次带大家体验一次 BitmapText。 这个插件可以将如下所示的图片裁剪并贴在场景里。
制作这种图片可以使用这个 工具
在Resource中如此load一下。
var RESOURCES = {
"s_font_fnt": "res/fonts/font.fnt",
"s_font_png": "res/fonts/font.png"
};
之后就可以在代码中使用了。
var title = new Tiny.BitmapText('doodle jump', {
font: '54px font', // 这个font的名字来自你的 fnt 文件中的 info 字段。
tint: '0x5a5816'
});
title.setPosition(30, 130);
title.rotation = -3.14 / 7;
这里最后就出现了大家再游戏出息页面看到的title。
2、关于事件和场景切换
启动页还有一个功能是点击开始按钮以后开始游戏,那么如何处理点击事件。
Tiny.js 屏蔽了 click 和 touch 事件的区别,定义了pointerdown
的事件。但是一定要记得先开启 setEventEnabled;
那么开始游戏这个开始的动作怎么完成呢?我们可以利用 Tiny.app 来切换场景从而达到启动和结束游戏的效果。
restartBtn.setEventEnabled(true);
restartBtn.on('pointerdown', function () {
Tiny.app.replaceScene(new MainLayer());
});
注意: 在结束动画的场景中我们可能需要渲染用户的得分,正确的姿势是在new这个Layer的时候将参数传进入。
Tiny.app.replaceScene(new EndLayer(self.score));//传入参数
而不是去把参数写到全局变量,EndLayer 的 constructor 再拿回来。
五、主页面跳跳跳
Sprite
重力感应
键盘事件
自己实现运动系统
doodle jump 游戏的主人公是我们的 Doodle,为了保证代码的整洁,我打算采用继承 Sprite 的方式,为 Doodle 单独做一个类 Doodle.js。
1、解决 Doodle 的脸怎么向左向右
这里使用 Doodle 的时候没有使用 Doodle 的雪碧图插件,如果使用雪碧图会容易的多。
doodle 的所有形象全部在Tiny.TextureCache[RESOURCES['all.png']]
中,向左的图,向右的图,还有跳的时候的图。
我们采用直接对 Texture 进行裁剪的暴力方式来实现 Doodle 的样子变换,但是如果不 clone 的话,这个操作会应用在所有使用该素材的 Sprite 上。
var all = Tiny.TextureCache[RESOURCES['all.png']].clone();
var height = 81;
var contactheight = 85;
var width = 110;
var rightRect = new Tiny.Rectangle(0, 121, width, height);
var leftRect = new Tiny.Rectangle(0, 201, width - 20, height);
var rightContactRect = new Tiny.Rectangle(0, 289, width, contactheight);
var leftContactRect = new Tiny.Rectangle(0, 371, width, height);
all.frame = rightRect;
this.texture = all;
这样我们再不同的时候只要去更改all.frame
对应的rect
就好。
2、自己的运动系统
重力
速度
摩擦力
一个简单的牛顿运动系统,考虑摩擦力,重力,速度在内。
doodle 在最开始的时候要有一个向上运动的速度,所以 v 初始化的时候是向上的100。
/**
* @description 速度
* @type {{x: number, y: number}}
*/
this.v = {x: 0, y: -100};
/**
* @description 摩擦力
* @type {{x: number, y: number}}
*/
this.m = {x: 0, y: 0};
/**
* @description 重力
* @type {{x: number, y: number}}
*/
this.g = {x: 0, y: 10};
/**
* @description update 每一帧更新doodle的速度等参数
*/
this.on('update', function () {
self.y += self.v.y / CONFIG.speed; // 根据 速度 去改变doodle的位置,从而实现运动的效果
self.x += self.v.x / CONFIG.speed;
self.v.x += self.g.x; // 根据 加速度 去改变速度,实现掉落的效果
self.v.y += self.g.y;
if (self.m.x !== 0) { // 摩擦力
var tx = self.v.x + self.m.x;
if (tx * self.m.x >= 0) { // 速度减到0的时候就不要再减了 判断方式是判断摩擦力和运动方向同向
self.v.x = 0;
self.m.x = 0;
} else {
self.v.x = tx;
}
}
if (self.m.y !== 0) {
var ty = self.v.y + self.m.y;
if (ty * self.m.y >= 0) {
self.v.y = 0;
self.m.y = 0;
} else {
self.v.y = ty;
}
}
});
当我们希望 doodle 向右动
doodle.v.x = 100;
当我们希望 doodle 带减速的向右动
doodle.v.x = 100;
doodle.m.x = -10;
3、重力感应和键盘控制
event.gamma 标识了手机绕 Y 轴旋转的角度从 -90 到 90.
window.addEventListener('deviceorientation', function (event) {
var g = event.gamma;
if (g >= 5) {
self.goRight(g);
return;
}
if (g < 5) {
self.goLeft(g);
return;
}
// self.stopsX();
}, true);
键盘控制这里使用 Tiny.js 的Keyboard 插件
var keyLeft = new Tiny.Keyboard(37);//left
var keyRight = new Tiny.Keyboard(39);//right
keyLeft.press = function (e) {
self.goLeft();
};
keyRight.press = function (e) {
self.goRight();
};
if(Tiny.isMobile.phone) {
//这里可判断是不是移动端进而可以确定通过什么方式处理事件。
}
4、碰撞
没有使用碰撞系统的碰撞
碰撞的场景其实很多,但是真实的使用中,不要怕,我们对碰撞大部分场景可以做抽象。
比如在我们的这个场景里,我们把 doodle 的模型抽象成在 doodle 最底部的高度为1
的一个条
。
这样就可以直接判断这个条
和每个跳板的两个矩形是不是发生了接触,接触了那就是发生了所谓的碰撞
。
var doodleRect = new Tiny.Rectangle(doodle.x, doodle.y + doodle.height, doodle.width, 1);
for (var i = 0; i < sprites.length; i++) {
if (Tiny.rectIntersectsRect(doodleRect, sprites[i])) {
doodle.emit('contact', i, self);
break;
}
}
六、总结
爽。