成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

記一次游戲H5開發(fā)經(jīng)驗(yàn)

GitChat / 482人閱讀

摘要:為了實(shí)現(xiàn)物體隨機(jī)出現(xiàn)的效果,讓每個(gè)物體隨機(jī)多少秒后開始出現(xiàn)最后一個(gè)物體出現(xiàn)完,多少秒后出現(xiàn)結(jié)束畫面等等,需要理清楚各個(gè)定時(shí)器的關(guān)系,并對(duì)其添加語(yǔ)義化良好的標(biāo)記,及時(shí)對(duì)未完結(jié)的定時(shí)器進(jìn)行清除,防止定時(shí)器帶來(lái)的意想不到的問題。

快到年終的時(shí)候做了一個(gè)以游戲形式展示的h5活動(dòng)頁(yè),第一次嘗試使用js寫小游戲,很有趣的過(guò)程,很寶貴的經(jīng)驗(yàn)。

效果圖

直接上個(gè)效果的gif圖,游戲的一小部分效果,錄出來(lái)有點(diǎn)卡

結(jié)果頁(yè):

起因

產(chǎn)品妹子突然給我拉進(jìn)來(lái)一個(gè)群,跟我們講做了這么久的制作平臺(tái)(用戶制作手機(jī)主題的平臺(tái)),我們是不是應(yīng)該反饋給用戶點(diǎn)什么東西,就像之前特別火的微信年終總結(jié)那樣。總之就是要打動(dòng)用戶,要特別酷。說(shuō)特別酷的時(shí)候她回頭朝我微微一笑,微笑中帶著一點(diǎn)點(diǎn),嗯,殺意。
活動(dòng)形式,展現(xiàn)方式,什么數(shù)據(jù)反正就是統(tǒng)統(tǒng)都沒想好,整個(gè)過(guò)程中大家討論的熱火朝天。當(dāng)時(shí)不知道我為啥腦子一熱,跟她說(shuō)了一句:“沒事兒,搞吧,你能想出來(lái)我就給你做出來(lái)?!倍乙惨?yàn)檫@句話把自己置身于水深火熱之中。。
討論的結(jié)果就是大家的idea感覺都不是特別酷,又不好玩兒,干脆就做個(gè)游戲形式的吧!所有人都轉(zhuǎn)頭看向我,我想了想之前說(shuō)的話,只吐出來(lái)一個(gè)字,“搞”。而內(nèi)心中五味雜陳,“游戲?有意思???搞!沒搞過(guò)啊?能搞定嗎?搞!”。最終敲定,兩周時(shí)間,游戲方式,展現(xiàn)用戶在魔秀的點(diǎn)點(diǎn)滴滴。

準(zhǔn)備工作

游戲的形式大概類似一個(gè)滑雪大冒險(xiǎn)和賽車的結(jié)合,以賽車的形式進(jìn)行偽3d效果的展現(xiàn),滑雪大冒險(xiǎn)的樣式作為我們的主題,同時(shí)大家還給我們的游戲起了個(gè)酷炫的名字----魔秀時(shí)光道。

游戲引擎

游戲的展現(xiàn)形式確定后,直覺告訴我,想要將游戲快速穩(wěn)定的呈現(xiàn),免去圖片加載控制,動(dòng)畫控制之類的復(fù)雜處理,我需要一個(gè)JS游戲引擎。最終在EgretPhaserPixiJS中選定了PixiJS,雖然不像Egret一樣有完善的中文文檔,但是它提供了清晰易懂的examples可快速上手,沒有復(fù)雜的生態(tài),簡(jiǎn)單的幾行代碼就可以用js實(shí)現(xiàn)我想要以下幾點(diǎn)功能:

容器渲染及背景描繪

我需要定制整個(gè)畫布的大小和背景,我需要使用不同的容器來(lái)承接不同的內(nèi)容,并且靈活控制每個(gè)容器的屬性:

//畫布
var app = new PIXI.Application(800, 600, {backgroundColor : 0x1099bb});
document.body.appendChild(app.view);

//定制container
var container = new PIXI.Container();

container.x = (app.renderer.width - container.width) / 2;
container.y = (app.renderer.height - container.height) / 2;

app.stage.addChild(container);
圖片加載及動(dòng)畫處理

大家都知道,使用canvas進(jìn)行圖片繪制的時(shí)候,需要確定圖片已經(jīng)成功加載,而游戲中有著大量的圖片資源需要去維護(hù),PixiJS已經(jīng)為我們提供此項(xiàng)服務(wù):

var bunny = PIXI.Sprite.fromImage("required/assets/basics/bunny.png’);

bunny.x = app.renderer.width / 2;
bunny.y = app.renderer.height / 2;

app.stage.addChild(bunny);

同時(shí),我們需要一個(gè)動(dòng)畫控制器,來(lái)控制各Sprite的運(yùn)動(dòng)和重繪,而不是生硬的對(duì)各項(xiàng)屬性進(jìn)行重新修改:

app.ticker.add(function(delta) {
  bunny.rotation += 0.1 * delta;
});

需要注意的是,我們會(huì)發(fā)現(xiàn),此處的Sprite動(dòng)畫控制,相當(dāng)于添加了運(yùn)動(dòng)的動(dòng)畫隊(duì)列,并且實(shí)現(xiàn)了類似transformjs的效果,可直接對(duì)實(shí)例的屬性進(jìn)行操作。而我在寫項(xiàng)目的時(shí)候官方的例子是通過(guò)統(tǒng)一animate函數(shù)進(jìn)行操作,通過(guò)requestAnimationFrame進(jìn)行幀動(dòng)畫控制,更推薦新的方式而不是如下:

function animate () {
  requestAnimationFrame(animate);
  
  bunny.rotation += 1;
  renderer.render(stage);
}
事件處理

游戲最重要的部分相當(dāng)于用戶的交互了,也就是所謂的事件處理,為Sprite添加事件監(jiān)聽,很簡(jiǎn)單,如下所示:

//元素可點(diǎn)擊
sprite.interactive = true;

//鼠標(biāo)移入cursor
sprite.buttonMode = true;

sprite.on("click", onClick); // mouse-only
prite.on("tap", onClick); // touch-only

function onClick () {
  sprite.scale.x *= 1.25;
  sprite.scale.y *= 1.25;
}
設(shè)計(jì)圖

設(shè)計(jì)圖當(dāng)然也是很重要的,決定了我們?nèi)绾稳?shí)現(xiàn)這個(gè)游戲,當(dāng)我拿到設(shè)計(jì)圖的時(shí)候,他是長(zhǎng)成這樣的,我的內(nèi)心是崩潰的。我能怎么樣,我也很無(wú)奈呀~ 開搞吧!

實(shí)現(xiàn)思路

根據(jù)以上,PixiJS已經(jīng)基本滿足我們的需求,也就是說(shuō),工具準(zhǔn)備和素材準(zhǔn)備已經(jīng)都完成了。在動(dòng)手書寫之前,我們需要把實(shí)現(xiàn)思路想好,才能保證書寫過(guò)程的清晰,避免不必要的麻煩。

背景滑動(dòng)效果實(shí)現(xiàn)

就像我們平時(shí)玩兒賽車游戲一樣,我們感覺賽車在跑道上進(jìn)行比賽,實(shí)際上賽車只進(jìn)行左右移動(dòng)而已,而運(yùn)動(dòng)的則是背景,如何規(guī)劃好路線,讓背景按照既定的場(chǎng)景去運(yùn)動(dòng),并展現(xiàn)不同的視角,特意向央美的同胞咨詢了下,他們是用一個(gè)叫“攝像機(jī)”的東西實(shí)現(xiàn)的。對(duì)于我們來(lái)說(shuō),不需要那么復(fù)雜的場(chǎng)景,只需讓背景像前規(guī)律的“平移”,造成“樹動(dòng)我不動(dòng)”的視覺效果,同時(shí)我們利用“透視”的原理,讓背景以“近大遠(yuǎn)小”的方式進(jìn)行變化,就會(huì)產(chǎn)生一種low low的立體效果。

關(guān)鍵詞:透視 近大遠(yuǎn)小(偏移,大小,速度)

偏移路線

對(duì)于背景及物體的運(yùn)動(dòng),大概路線規(guī)劃如下:

確定視覺焦點(diǎn)后,我們只需隨機(jī)生成物體出現(xiàn)的位置,計(jì)算出a,b相對(duì)固定,使其y進(jìn)行相應(yīng)速度的增加,x根據(jù)運(yùn)動(dòng)軌跡進(jìn)行對(duì)應(yīng)偏移,則可實(shí)現(xiàn)往近跑的效果。針對(duì)運(yùn)動(dòng)軌跡, 假設(shè)物體向下偏移距離為N,則對(duì)應(yīng)水平針對(duì)中軸線的偏移為:

大小

同時(shí),我們還需對(duì)物體進(jìn)行近大遠(yuǎn)小的顯示,這個(gè)比較簡(jiǎn)單,以焦點(diǎn)為0,頁(yè)面最底端為1,進(jìn)行對(duì)應(yīng)比例放大即可:

scale = (curY - startY) / ( endY - startY);
運(yùn)動(dòng)速度

針對(duì)物體的運(yùn)動(dòng)速度,也應(yīng)在遠(yuǎn)近有不同的體現(xiàn)。

背景樹與碰撞物體的區(qū)別

針對(duì)背景樹,我們需在最初對(duì)所有的樹進(jìn)行展現(xiàn),鋪滿兩邊背景。每列樹對(duì)應(yīng)的運(yùn)動(dòng)路線一致,可直接讓其進(jìn)行循環(huán)展示,當(dāng)樹運(yùn)動(dòng)到最底時(shí),讓其出現(xiàn)在最頂點(diǎn)。因此只需確定一共有幾行幾列樹,并設(shè)定其邊界,根據(jù)行列確定初始唯一并對(duì)其進(jìn)行運(yùn)動(dòng)。同時(shí),可以讓樹進(jìn)行小范圍的隨機(jī)偏移,使樹錯(cuò)落有致。如下所示:

export default function Tree (row, col, direction) {
  this.cfg = {
    direction: direction, //方向
    col: col, //第幾列
    row: row, //第幾行
    MaxX: 440,
    minY: 210,
    maxY: 500,
    range: 10 //坐標(biāo)浮動(dòng)范圍
  }
};

而針對(duì)物體,則需要隨機(jī)生成它的初始x坐標(biāo),并計(jì)算出其對(duì)于的路線進(jìn)行運(yùn)動(dòng),在運(yùn)動(dòng)過(guò)程中,進(jìn)行碰撞檢測(cè),檢測(cè)是否與人物進(jìn)行相撞。

人物滑動(dòng)實(shí)現(xiàn)

人物滑動(dòng)的操作,用了最簡(jiǎn)單的實(shí)現(xiàn)方式:按鈕。當(dāng)用戶點(diǎn)擊不同方向時(shí),讓人物向?qū)?yīng)方向進(jìn)行偏移。同時(shí),為了讓人物滑動(dòng)不僵硬,在左右滑動(dòng)過(guò)程中,人也應(yīng)該隨著運(yùn)動(dòng)有對(duì)應(yīng)角度的傾斜,就像我們平時(shí)玩兒滑雪拐彎時(shí),會(huì)改變中心一樣。思路如下:

點(diǎn)擊按鈕時(shí),改變方向

運(yùn)動(dòng)時(shí)檢測(cè)方向,若向左,則x減小,向右,則增加

向右(左)運(yùn)動(dòng)時(shí),人物對(duì)應(yīng)rotation也進(jìn)行增加(減?。?/p>

松開手時(shí),人物對(duì)應(yīng)rotation慢慢恢復(fù)成0;

碰撞檢測(cè)

由于人物有吃東西的環(huán)節(jié)(不然這還叫什么游戲呢),因此碰撞檢測(cè)肯定是必須的啦。我們可以通過(guò)兩種方式進(jìn)行碰撞檢測(cè)

人物檢測(cè)碰撞物體,需實(shí)時(shí)遍歷物體坐標(biāo)列表,進(jìn)行檢測(cè)

每個(gè)物體自身進(jìn)行碰撞檢測(cè),檢測(cè)自身與人物位置的對(duì)應(yīng)差

我很機(jī)智的選擇了第二個(gè),畢竟每個(gè)物體的位置都是實(shí)時(shí)變動(dòng)的,而每次碰撞檢測(cè)都進(jìn)行一次循環(huán)的方法,太笨重啦。在這里我們?cè)O(shè)置碰撞檢測(cè)的區(qū)域(寬高),在物體運(yùn)動(dòng)時(shí),針對(duì)人物的x,y坐標(biāo),與自身的x,y坐標(biāo)加減形成的四條邊界進(jìn)行比較即可,若進(jìn)行碰撞,則進(jìn)行對(duì)應(yīng)的操作即可,如播放音頻,得分+1等。

架構(gòu)設(shè)計(jì)

思路理清楚之后,后面的路就很明朗啦。接下來(lái)我們就可以著手設(shè)計(jì)下如何實(shí)現(xiàn)這個(gè)東西了,很顯然,游戲中我們擁有許許多多的“角色”,使用“面向?qū)ο蟆钡姆绞皆俸貌贿^(guò)了。大概的劃分如下

stage //舞臺(tái),進(jìn)行基本場(chǎng)景渲染,游戲整體控制(開始,停止)等

player //玩兒家,也就是對(duì)應(yīng)的人物

sprite //出現(xiàn)的物體,如蛋糕等,提供玩兒家吃。 包含碰撞檢測(cè)等,會(huì)自己運(yùn)動(dòng)

tree //因?yàn)閠ree自身會(huì)運(yùn)動(dòng),所以每個(gè)tree為一個(gè)類

score //進(jìn)行分?jǐn)?shù)控制及顯示

cfg.js //包含整體游戲配置

對(duì)象內(nèi)部劃分

每個(gè)對(duì)象包含以下幾點(diǎn)屬性及功能:

1. 對(duì)象配置

每個(gè)對(duì)象包含其內(nèi)部自身基本配置,包括位置,邊界,圖片等。直觀,便于調(diào)試

export default function People (stage) {
  this.cfg = {
    img: require("./img/people.png"),
    anchor: {
      x: 0.5,
      y: 0.5,
    },
    position: {
      x: cfg.width / 2,
      y: 500
    },
    speed: 5,
  }
}
2. 其他方法

每個(gè)對(duì)象都包含其自身方法,如下所示:

render //進(jìn)行圖片等渲染

animate //動(dòng)畫function

init //一些初始化配置

實(shí)現(xiàn)

通過(guò)以上思路的設(shè)計(jì)和結(jié)構(gòu)的設(shè)計(jì),我很快的將這個(gè)游戲?qū)崿F(xiàn)了。。。沒錯(cuò),理清思路和結(jié)構(gòu)的重要性就是這樣。當(dāng)然,在實(shí)現(xiàn)過(guò)程中,也有一些小的點(diǎn)可以記錄下:

資源加載器(圖片)

為了游戲的進(jìn)行效果,還是決定在加載完所有資源(尤其是圖片資源)后,才停止loading頁(yè)面。如何判斷所有內(nèi)容都加載完畢了呢?寫了個(gè)小loader

var pics = [
  require("./img/bg-start.png"),
  require("./img/btn-start.png"),
  ...
];

function loadImages (pics, callback) {
  if(pics.length) {
    var img = new Image(),
      pic = pics.shift();

    img.onload = callback;
    img.src = pic;
    loadImages(pics, callback);
  } else{
    return;
  }
}

$(function() {
  loadImages(pics, function () {
    if (!pics.length) {
      $(".loading").hide();
    };
  })
});
強(qiáng)制橫屏

游戲是橫屏展示的,那就強(qiáng)制橫屏好啦。這個(gè)當(dāng)時(shí)還糾結(jié)挺久,還是自己功底不扎實(shí)腦子走私了,還在想是監(jiān)聽resize事件還是旋轉(zhuǎn)屏幕事件,都沒有這些事兒啊好嗎!直接讓它旋轉(zhuǎn)就好。

if(window.orientation==180||window.orientation==0) {
  $("#main").height(winW);
  $("#main").width(winH);

  $(‘#main’).css({
    "transform": "rotate(90deg)",
  });
} else{
  $(‘#main’).css("transform", "rotate(0)");
}
timer控制

理清思路后,最亂的還是各種定時(shí)器啦。 為了實(shí)現(xiàn)物體隨機(jī)出現(xiàn)的效果,讓每個(gè)物體隨機(jī)多少秒后開始出現(xiàn);最后一個(gè)物體出現(xiàn)完,多少秒后出現(xiàn)結(jié)束畫面等等,需要理清楚各個(gè)定時(shí)器的關(guān)系,并對(duì)其添加語(yǔ)義化良好的標(biāo)記,及時(shí)對(duì)未完結(jié)的定時(shí)器進(jìn)行清除,防止定時(shí)器帶來(lái)的意想不到的問題。

寫在最后

最終游戲的效果基本讓大家滿意啦,也是第一次嘗試這方面的開發(fā),周圍也完全沒有做過(guò)這東西的人。從開始的忐忑和一無(wú)所措,到過(guò)程中理清思路和結(jié)構(gòu),到書寫中的各種未知的坑,自己在這兩周感覺經(jīng)歷了很充實(shí)的一件事情。同時(shí)也對(duì)后續(xù)進(jìn)行一些未知事物的探索和學(xué)習(xí)有了更豐富的經(jīng)驗(yàn),找對(duì)路子才是王道呀!

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/50622.html

相關(guān)文章

  • 一次游戲H5開發(fā)經(jīng)驗(yàn)

    摘要:為了實(shí)現(xiàn)物體隨機(jī)出現(xiàn)的效果,讓每個(gè)物體隨機(jī)多少秒后開始出現(xiàn)最后一個(gè)物體出現(xiàn)完,多少秒后出現(xiàn)結(jié)束畫面等等,需要理清楚各個(gè)定時(shí)器的關(guān)系,并對(duì)其添加語(yǔ)義化良好的標(biāo)記,及時(shí)對(duì)未完結(jié)的定時(shí)器進(jìn)行清除,防止定時(shí)器帶來(lái)的意想不到的問題。 快到年終的時(shí)候做了一個(gè)以游戲形式展示的h5活動(dòng)頁(yè),第一次嘗試使用js寫小游戲,很有趣的過(guò)程,很寶貴的經(jīng)驗(yàn)。 效果圖 直接上個(gè)效果的gif圖,游戲的一小部分效果,錄出...

    xingpingz 評(píng)論0 收藏0
  • 一次使用Fiddler抓包工具抓取Https協(xié)議數(shù)據(jù)的踩坑過(guò)程

    摘要:直到今天,突然看到一個(gè)有意思的微信小游戲。后來(lái)試了幾次之后才發(fā)現(xiàn),這個(gè)小游戲比較刁,不僅做了微信的登錄授權(quán),而且做了手機(jī)端訪問的判斷,更甚至竟然用的還是協(xié)議的網(wǎng)頁(yè)。調(diào)用的目標(biāo)發(fā)生了異常。 記一次使用Fiddler抓包工具抓取Https協(xié)議數(shù)據(jù)的踩坑過(guò)程 前言 記得從剛?cè)腴T前端第一天開始,當(dāng)時(shí)的師傅就跟我介紹了一個(gè)可以抓取一些必須要在微信瀏覽器打開的鏈接的工具Fiddler,主要用來(lái)抓取...

    JackJiang 評(píng)論0 收藏0
  • 2017年3月份前端資源分享

    平日學(xué)習(xí)接觸過(guò)的網(wǎng)站積累,以每月的形式發(fā)布。2017年以前看這個(gè)網(wǎng)址:http://www.kancloud.cn/jsfron... 03月份前端資源分享 1. Javascript 175453545 Redux compose and middleware 源碼分析 深入 Promise(二)——進(jìn)擊的 Promise Effective JavaScript leeheys blog -...

    ermaoL 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<