如何制作一款HTML5 RPG游戏引擎——第二篇,烟雨+飞雪效果
今天我们来实现烟雨+飞雪效果。首先来说,一款经典的RPG游戏难免需要加入天气的变化。那么为了使我们的RPG游戏引擎更完美,我们就只好慢慢地实现它。
本文为该系列文章的第二篇,如果想了解以前的文章可以看看下面列表:
如何制作一款HTML5 RPG游戏引擎——第一篇,地图类的实现
http://blog.csdn.net/yorhomwang/article/details/8892305
该引擎是基于lufylegend开发的,学习时请先了解lufylegend。
官方网站地址:http://lufylegend.com/lufylegend
API地址:http://lufylegend.com/lufylegend/api
※获得本次开发的灵感来自lufy在lufylegend.js下载包中编写的一个实例。
一,LEffect类
首先来搞一个LEffect类,构造器如下:
function LEffect(){
var s = this;
base(s,LSprite,[]);
s.rainList = [];
s.snowList = [];
s.rainLayer = new LSprite();
s.addChild(s.rainLayer);
s.snowLayer = new LSprite();
s.addChild(s.snowLayer);
}
这个LEffect类也是继承自LSprite,rainList是用来装雨点的一个数组,snowList则是用来装雪的一个数组。rainLayer,snowLayer分别是一个LSprite,如果是雪就绘画在snowLayer上,雨就在rainLayer。这个构造器很简单,接下来就来看看下雨的效果是怎么样实现的。
二,烟雨纷纷
这里我用烟雨来形容这个效果,难免有一些过分,因为它的美感远远还没达到那种地步。不过能在屏幕上飘雨点子还是实现了的,那好,就来让我们来看看关于它的代码吧:
LEffect.prototype.raining = function(speed,size){
var s = this;
if(!speed)speed = 30;
if(!size)size = 5;
s.rainLayer.addEventListener(LEvent.ENTER_FRAME,function(){
s.onshow("rain",speed,size);
});
}
首先,它是LEffect一个成员类,叫raining,有两个参数,分别是下雨时雨点落下速度,另一个是雨点尺寸。当然,使用时可以不给它们赋值,如果不给它们赋值那就用默认的值。接着加入ENTER_FRAME事件,也就是说,当rainLayer刷新一次,就调用以下功能,相当于一个时间轴。ENTER_FRAME具体的使用方法可以去lufylegend API文档看看。也许有朋友心急了,想迫切地看看onshow是个什么东西。本文最后一节会讲它,因为它和下雪的效果也有些关系。
接下来就轮到飞雪效果了。
三,飞雪效果
这次直接上代码吧,很raining差不多,只是改了最后调用onshow时传的参数和参数赋值而已:
LEffect.prototype.snowing = function(speed,size){
var s = this;
if(!speed)speed = 10;
if(!size)size = 1;
s.snowLayer.addEventListener(LEvent.ENTER_FRAME,function(){
s.onshow("snow",speed,size);
});
}
四,onshow方法
在实现raining和snowing时,我们都用到了onshow方法,并且还给它几个参数,我们不妨先看看代码:
LEffect.prototype.onshow = function(thing,speed,size){
var s = this;
if(thing == "rain"){
s.rainLayer.graphics.clear();
var rainX = Math.random()*(LStage.width-10-10)+10;
var n = s.rainList.length;
while(n--){
var o = s.rainList[n];
o.y += o.s;
s.rainLayer.graphics.drawRect(1,"white",[o.x,o.y,1,size],true,"#f3f3f3");
}
s.rainList.push({x:rainX,y:0,s:speed});
}else if(thing == "snow"){
s.snowLayer.graphics.clear();
var snowX = Math.random()*(LStage.width-10-10)+10;
var n = s.snowList.length;
while(n--){
var o = s.snowList[n];
o.y += o.s;
s.snowLayer.graphics.drawArc(2,"white",[o.x,o.y,size,0,2*Math.PI],true,"white");
}
s.snowList.push({x:snowX,y:0,s:speed});
}
}
很显然,这个onshow方法是整个效果类的核心部分,光看代码长度就应该知道。。。首先我们来看看它的参数:
thing:效果类型,如果为snow就下雪,如果是rain就下雨
speed:下雨下雪的速度
size:下雨下雪的尺寸
speed,size和snowing、raining中的speed,size值是一样的,因此,如果你在用snowing、raining时,直接给它们的参数赋值就行了。
我们现在来解释一下代码:
if(thing == "rain"){
s.rainLayer.graphics.clear();
var rainX = Math.random()*(LStage.width-10-10)+10;
var n = s.rainList.length;
while(n--){
var o = s.rainList[n];
o.y += o.s;
s.rainLayer.graphics.drawRect(1,"white",[o.x,o.y,1,size],true,"#f3f3f3");
}
s.rainList.push({x:rainX,y:0,s:speed});
}
以上是下雨效果的实现,首先我们先清屏一次,接着随机取一个x坐标作为绘制的位置。接着我们遍历了一下我们的雨点数组,并且每遍历到一个,就把这个的y坐标加上速度的大小。然后用lufylegend中专门用来画矩形的函数画一个小矩形作为一个雨点。然后在雨点数组中记录下这些雨点的数据。由于我们用了时间轴事件,所以就能使雨点不断地画出来。
下雪的原理是一样的,就只有绘画方式不同,把画矩形改成了画圆而已。大家可以根据我刚才说的对应看看。
五,封装完成后应该怎么写?
刚刚我们把效果类封装好了,那么别人怎么用呢?我不妨给大家呈现一下代码,原本我们要写百十行才能实现的,封装完成后就只有这些了:
var backLayer = new LSprite();
var effect = new LEffect();
backLayer.addChild(effect);
//下雪
effect.snowing();
//下雨
effect.raining();
很简单是不是?这就是封装的力量!
六,效果演示
为了方便测试,我给它加了一个按钮和选择框用来显示不同的效果。
截图如下:
演示地址:http://www.cnblogs.com/yorhom/articles/3073050.html
代码很少,归纳后如下,感兴趣的朋友可以复制下来看看:
/**
*LEffect效果类
*/
function LEffect(){
var s = this;
base(s,LSprite,[]);
s.rainList = [];
s.snowList = [];
s.rainLayer = new LSprite();
s.addChild(s.rainLayer);
s.snowLayer = new LSprite();
s.addChild(s.snowLayer);
}
LEffect.prototype.raining = function(speed,size){
var s = this;
if(!speed)speed = 30;
if(!size)size = 5;
s.rainLayer.addEventListener(LEvent.ENTER_FRAME,function(){
s.onshow("rain",speed,size);
});
}
LEffect.prototype.snowing = function(speed,size){
var s = this;
if(!speed)speed = 10;
if(!size)size = 1;
s.snowLayer.addEventListener(LEvent.ENTER_FRAME,function(){
s.onshow("snow",speed,size);
});
}
LEffect.prototype.onshow = function(thing,speed,size){
var s = this;
if(thing == "rain"){
s.rainLayer.graphics.clear();
var rainX = Math.random()*(LStage.width-10-10)+10;
var n = s.rainList.length;
while(n--){
var o = s.rainList[n];
o.y += o.s;
s.rainLayer.graphics.drawRect(1,"white",[o.x,o.y,1,size],true,"#f3f3f3");
}
s.rainList.push({x:rainX,y:0,s:speed});
}else if(thing == "snow"){
s.snowLayer.graphics.clear();
var snowX = Math.random()*(LStage.width-10-10)+10;
var n = s.snowList.length;
while(n--){
var o = s.snowList[n];
o.y += o.s;
s.snowLayer.graphics.drawArc(2,"white",[o.x,o.y,size,0,2*Math.PI],true,"white");
}
s.snowList.push({x:snowX,y:0,s:speed});
}
}
如何制作一款HTML5 RPG游戏引擎——第二篇,烟雨+飞雪效果的更多相关文章
- 如何制作一款HTML5 RPG游戏引擎——第一篇,地图类的实现
一,话说天下大事 前不久看到lufy的博客上,有一位朋友想要一个RPG游戏引擎,出于兴趣准备动手做一做.由于我研究lufylegend有一段时间了,对它有一定的依赖性,因此就准备将这个引擎基于lufy ...
- 如何制作一款HTML5 RPG游戏引擎——第五篇,人物&人物特效
上一次,我们实现了对话类,今天就来做一个游戏中必不可少的——人物类. 当然,你完全是可以自己写一个人物类,但是为了方便起见,还是决定把人物类封装到这个引擎里. 为了使这个类更有意义,我还给人物类加了几 ...
- 如何制作一款HTML5 RPG游戏引擎——第四篇,情景对话
今天我们来实现情景对话.这是一个重要的功能,没有它,游戏将变得索然无味.所以我们不得不来完成它. 但是要知道,使用对话可不是一件简单的事,因为它内部的东西很多,比如说人物头像,人物名称,对话内容... ...
- 如何制作一款HTML5 RPG游戏引擎——第三篇,利用幕布切换场景
开言: 在RPG游戏中,如果有地图切换的地方,通常就会使用幕布效果.所谓的幕布其实就是将两个矩形合拢,直到把屏幕遮住,然后再展开直到两个矩形全部移出屏幕. 为了大家做游戏方便,于是我给这个引擎加了这么 ...
- HTML5 RPG游戏引擎 地图实现篇
一,话说全国年夜事 前没有暂看到lufy的专客上,有一名伴侣念要一个RPG游戏引擎,出于兴趣筹办入手做一做.因为我研讨lufylegend有冶时间了,对它有必然的依赖性,因而便筹办将那个引擎基于 ...
- 用Html5结合Qt制作一款本地化EXE游戏-太空大战(Space War)
本次来说一说如何利用lufylegend.js引擎制作一款html5游戏后将其通过Qt转换成EXE程序.步骤其实非常简单,接下来就一步步地做一下解释和说明. 首先我们来开发一个有点类似于太空大战的游戏 ...
- HTML5开源RPG游戏引擎lufylegendRPG 0.1发布
一,小小开篇 首先不得不先介绍一下这个引擎: lufylegendRPG是lufylegend的拓展引擎,使用它时,需要引入lufylegend.同时您也需要了解lufylegend语法,这样才能 ...
- 推荐一些好用的 HTML5 & JavaScript 游戏引擎开发库
推荐一些好用的 HTML5 & JavaScript 游戏引擎开发库 0. 引言 如果你是一个游戏开发者,并且正在寻找一个可以与 JavaScript 和 HTML5 无缝工作的游戏引擎.那么 ...
- 或许您还不知道的八款Android开源游戏引擎
很多初学Android游戏开发的朋友,往往会显得有些无所适从,他们常常不知道该从何处入手,每当遇到自己无法解决的难题时,又往往会一边羡慕于iPhone下有诸如Cocos2d-iphone之类的免费游戏 ...
随机推荐
- 关于Linux网卡调优之:RPS (Receive Packet Steering)
昨天在查LVS调度均衡性问题时,最终确定是 persistence_timeout 参数会使用IP哈希.目的是为了保证长连接,即一定时间内访问到的是同一台机器.而我们内部系统,由于出口IP相对单一,所 ...
- Golang 中操作 Mongo Update 的方法
Golang 和 MongoDB 中的 ISODate 时间交互问题 2018年02月27日 11:28:43 独一无二的小个性 阅读数:357 标签: GolangMongoDB时间交互时间转换 更 ...
- 一个 Map 函数、一个 Reduce 函数和一个 main 函数
MapReduce 最简单的 MapReduce应用程序至少包含 3 个部分:一个 Map 函数.一个 Reduce 函数和一个 main 函数.main 函数将作业控制和文件输入/输出结合起来.在这 ...
- 如果想要跨平台,在file类下有separtor(),返回锁出平台的文件分隔符
对于命令:File f2=new file(“d:\\abc\\789\\1.txt”) 这个命令不具备跨平台性,因为不同的OS的文件系统很不相同. 如果想要跨平台,在file类下有separtor( ...
- 【BZOJ】2555: SubString(后缀自动机)
http://www.lydsy.com/JudgeOnline/problem.php?id=2555 学到了如何快速维护right值orz (不过这仍然是暴力维护,可以卡到O(n) 首先我们在加一 ...
- 把本地仓库导入到Github
1, create a new rep 2, 进入到本地仓库: $ git remote set-url origin URL 3,上传 git push -v ref: https://www.do ...
- WORD里怎样能做到局部“分栏”就是一页里有的分有的不分
选中你要分的部分再分栏如果不想分的部分也被分了,那就可以选中不想分的那部分,选择“分栏”->“一栏” 转自:http://zhidao.baidu.com/question/9873268.ht ...
- maven 完整的settings.xml
maven 完整的settings.xml <?xml version="1.0" encoding="UTF-8"?> <!-- Licen ...
- 设置两个WdatePicker的开始时间小于结束时间,结束时间大于开始时间
contract_start_date_id为开始时间ID contract_end_date_id为结束时间ID $("#contract_start_date_id").bin ...
- python入门(十):XML和JSON解析
一.python解析XML 1.xml.dom.*模块,它是W3C DOM API的实现,若需要处理DOM API则该模块很适合,注意xml.dom包里面有许多模块,须区分它们间的不同: 2.xml. ...