欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Vue3+Canvas實現(xiàn)坦克大戰(zhàn)游戲(二)

 更新時間:2022年03月18日 10:00:40   作者:Ethan_Zhou  
本文主要給大家講解一下子彈擊中物體、物體銷毀、敵方坦克構(gòu)建生成、運(yùn)動算法、爆炸效果、以及障礙物的生成,感興趣的小伙伴可以了解一下

前言

接著上篇講,本篇主要給大家講解一下子彈擊中物體、物體銷毀、敵方坦克構(gòu)建生成、運(yùn)動算法、爆炸效果、以及障礙物的生成;了解了這些我相信你可以不依賴游戲引擎實現(xiàn)大部分小游戲的開發(fā)。

Es5版本: 在線游戲 源代碼

W/上 S/下 A/左 D/右 F/射擊

讓我們開始吧!

敵方坦克的生成

我們可以使用 for 循環(huán)和Tank 構(gòu)造函數(shù),批量制造多個敵方坦克,x,y 為其在畫布中的坐標(biāo),direction 為坦克當(dāng)前方向,type 為精靈圖中截取坦克圖片的信息,包含了坐標(biāo),尺寸,類型等

for(let t=0; t<20; t++) {
  let tank = new Tank(new TankConfig({
    x: 100 + t*30,
    y: 100,
    direction: DIRECTION.DOWN,
    type: TANK_TYPE.ENEMY0,
    is_player: 0
  }))
  tank.star();
}

ENEMY0: {
  type: 'ENEMY1',
  dimension: [30, 30],
  image_coordinates: [129, 1, 129, 33]
}

坦克移動的算法

emove 函數(shù)就是就是物體移動狀態(tài)下每幀都會執(zhí)行的函數(shù),將根據(jù)當(dāng)前方向修改下幀的坐標(biāo),當(dāng)下幀坐標(biāo)到達(dá)地圖邊緣時將執(zhí)行 dir 函數(shù)更新坦克方向,如果是子彈則將銷毀。

this.emove = function (d, obstacle_sprites) {
    this.direction = d
    switch (d) {
      case DIRECTION.UP:
        if (
          (obstacle_sprites && !this.checkRangeOverlap(obstacle_sprites)) ||
          !obstacle_sprites
        ) {
          this.y -= this.speed
    
          if (this.y <= 10) {
            this.dir()
          }
        } else {
          this.dir()
        }
        break
    ...

dir 函數(shù)用來隨機(jī)修改移動的方向,當(dāng)然這個函數(shù)不能每幀都調(diào)用,否則坦克將像無頭蒼蠅一樣了;那么什么時候需要修改方向呢,我們認(rèn)為當(dāng)坦克和物體相撞的時候或者到達(dá)地圖邊緣時修改方向是合理的;子彈也可以認(rèn)為是一個物體,所以子彈到達(dá)坦克周邊一定距離時也將引起坦克規(guī)避動作,也就是將觸發(fā)dir 函數(shù)。

this.dir = function () {
    if(Math.floor(Math.random()*10) === 0 || Math.floor(Math.random()*10) === 1) {
      this.direction = DIRECTION.UP;
    };
    if(Math.floor(Math.random()*10) === 2 || Math.floor(Math.random()*10) === 3) {
      this.direction = DIRECTION.DOWN;
    };
    if(Math.floor(Math.random()*10) === 4 || Math.floor(Math.random()*10) === 5) {
      this.direction = DIRECTION.LEFT;
    };
    if(Math.floor(Math.random()*10) === 6 || Math.floor(Math.random()*10) === 7) {
      this.direction = DIRECTION.RIGHT;
    };
    if(Math.floor(Math.random()*10) === 8 || Math.floor(Math.random()*10) === 9) {
      this.dir();
    }
}

子彈擊中物體的算法

我們定義 checkRangeOverlap 函數(shù)來判斷兩個物體是否相撞,uiobjs 為畫布中所有的元素對象列表,包含 坦克、子彈、障礙物等;

this.getFrontPoints() 函數(shù)將根據(jù)當(dāng)前方向返回包含三個頂端點坐標(biāo)數(shù)組,形成判斷區(qū)域;

uiobjs[o].getCornerPoints() 函數(shù)返回當(dāng)前元素的四個邊角坐標(biāo)數(shù)組,形成判斷區(qū)域;

然后getFrontPoints() 的三個點坐標(biāo) 將分別和 uiobjs[o].getCornerPoints() 的四邊坐標(biāo)進(jìn)行點對點判斷,根據(jù)x, y 數(shù)值滿足下方四個條件時則認(rèn)為此點坐標(biāo)在元素內(nèi)部。

 // 判斷點坐標(biāo)是否在區(qū)域內(nèi)部,以識別是否相撞
  CanvasSprite.prototype.checkRangeOverlap = function (uiobjs) {
    for (let o in uiobjs) {
      let obstacle_cp = uiobjs[o].getCornerPoints()
      let fp = this.getFrontPoints()

      for (let idx = 0; idx < fp.length; idx++) {
        if (
          fp[idx].x > obstacle_cp[0].x &&
          fp[idx].x < obstacle_cp[1].x &&
          fp[idx].y > obstacle_cp[0].y &&
          fp[idx].y < obstacle_cp[2].y
        ) {
          return true
        }
      }
    }
    return false
  }
// 返回當(dāng)前物體頂端三坐標(biāo)
  CanvasSprite.prototype.getFrontPoints = function () {
    let front_point
    switch (this.direction) {
      case DIRECTION.UP:
        front_point = [
          new Point(this.x, this.y),
          new Point((2 * this.x + this.width) / 2, this.y),
          new Point(this.x + this.width, this.y),
        ]
        break
      ... 縮略,下左右方向
    }
    return front_point
  }
// 返回元素四邊坐標(biāo)形成的矩形區(qū)域范圍
  CanvasSprite.prototype.getCornerPoints = function () {
    var corner_points
    switch (this.direction) {
      case DIRECTION.UP:
        corner_points = [
          new Point(this.x, this.y),
          new Point(this.x + this.width, this.y),
          new Point(this.x + this.width, this.y + this.height),
          new Point(this.x, this.y + this.height),
        ]
        break
        ... 縮略,下左右方向
    }
    return corner_points
  }

啊 坦克搞多了!但是可以看到子彈擊中效果是成功的,你發(fā)現(xiàn)了沒,擊中后有一個爆炸效果,這個是怎么實現(xiàn)的呢?

爆炸效果的實現(xiàn)

當(dāng)識別到擊中后將此坦克將觸發(fā)explode 爆炸效果,然后每幀 判斷 isDestroied 是否銷毀,后續(xù)每幀將 explosion_count++ 從 0 到 5,然后更改alive 狀態(tài)為0 代表已銷毀。

if (s instanceof Tank && s.alive && s.isDestroied()) {
    s.explode()
}
this.isDestroied = function () {
    return explosion_count > 0
  }
  
this.explode = function () {
    if (explosion_count++ === 5) {
      this.alive = 0
    }
  }

然后 explosion_count 的數(shù)值,從0 到 5 代表爆炸效果圖的5幀

  this.getImg = function () {
    if (explosion_count > 0) {
      return {
        width: TANK_EXPLOSION_FRAME[explosion_count].dimension[0],
        height: TANK_EXPLOSION_FRAME[explosion_count].dimension[1],
        offset_x: TANK_EXPLOSION_FRAME[explosion_count].image_coordinates[0],
        offset_y: TANK_EXPLOSION_FRAME[explosion_count].image_coordinates[1],
      }
    } else {
      return {
        width: width,
        height: height,
        offset_x: this.type.image_coordinates[0],
        offset_y: this.type.image_coordinates[1],
      }
    }
  }

到現(xiàn)在我們的坦克游戲已經(jīng)基本可玩了,只不過現(xiàn)在是一片大平原,毫無遮攔,我們該為畫布增加一些障礙物如墻體,石頭等,增加游戲可玩性。

生成障礙物(石墻、磚墻等)

有了之前tanke 和 子彈的構(gòu)建函數(shù),現(xiàn)在這個 Block 就簡單多了,只需要定義好基本的信息,如坐標(biāo),寬高、狀態(tài)、方向,然后借用 apply 來使用 CanvasSprite 上的通用方法。

    let Block = function (x, y, direction, type) {
      type = type || BLOCK_TYPE.BLOCK_BRICK
      let alive = 1
      let width = type.dimension[0]
      let height = type.dimension[1]
      let explosion_count = 0
      this.type = type

      this.x = x
      this.y = y
      this.genre = 'block'
      this.direction = direction || DIRECTION.UP

      CanvasSprite.apply(this, [
        {
          alive: 1,
          border_y: HEIGHT,
          border_x: WIDTH,
          speed: 0,
          direction: direction,
          x: x,
          y: y,
          width: width,
          height: height,
        },
      ])

      this.isDestroied = function () {
        return explosion_count > 0
      }

      this.explode = function () {
        if (explosion_count++ === 5) {
          this.alive = 0
        }
      }

      this.getImg = function () {
        if (explosion_count > 0) {
          return {
            width: TANK_EXPLOSION_FRAME[explosion_count].dimension[0],
            height: TANK_EXPLOSION_FRAME[explosion_count].dimension[1],
            offset_x: TANK_EXPLOSION_FRAME[explosion_count].image_coordinates[0],
            offset_y: TANK_EXPLOSION_FRAME[explosion_count].image_coordinates[1],
          }
        } else {
          return {
            width: width,
            height: height,
            offset_x: type.image_coordinates[0],
            offset_y: type.image_coordinates[1],
          }
        }
      }

      this._generateId = function () {
        return uuidv4()
      }

      sprites[this._generateId()] = this
    }

定義好后,使用時只需批量生成Block 的實例,將坐標(biāo),類型等信息傳遞進(jìn)去就可以在下一幀渲染出來。

  for(let i=0; i<=2; i++) {
    for(let j=0; j<=2; j++) {
      let block = new Block(j*16, 140 + i*16, DIRECTION.UP, BLOCK_TYPE.BLOCK_STONE)
    }
  }

好了我們看看最終的效果吧!

總結(jié)

ok 坦克大戰(zhàn)基本上完成了,你可以修改子彈發(fā)射速度,敵方坦克數(shù)量,障礙物也可以自己隨意畫,可玩性還是挺好的,當(dāng)然還有很多細(xì)節(jié)可以完善,如 預(yù)制幾種不同的地圖,做成通關(guān)類,預(yù)制幾條生命等。如果你想試一下,可以 star 下 github 倉庫自己來改造自己的坦克大戰(zhàn)吧。

以上就是Vue3+Canvas實現(xiàn)坦克大戰(zhàn)游戲(二)的詳細(xì)內(nèi)容,更多關(guān)于Vue3 Canvas坦克大戰(zhàn)的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • 在vue項目中引入scss并使用scss樣式詳解

    在vue項目中引入scss并使用scss樣式詳解

    SCSS是一種CSS預(yù)處理語言,定義了一種新的專門的編程語言,編譯后形成正常的css文件,為css增加一些編程特性,這篇文章主要給大家介紹了關(guān)于在vue項目中引入scss并使用scss樣式的相關(guān)資料,需要的朋友可以參考下
    2022-07-07
  • 使用vant-picker實現(xiàn)自定義內(nèi)容,根據(jù)內(nèi)容添加圖標(biāo)

    使用vant-picker實現(xiàn)自定義內(nèi)容,根據(jù)內(nèi)容添加圖標(biāo)

    這篇文章主要介紹了使用vant-picker實現(xiàn)自定義內(nèi)容,根據(jù)內(nèi)容添加圖標(biāo),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-12-12
  • Vue項目中引入外部文件的方法(css、js、less)

    Vue項目中引入外部文件的方法(css、js、less)

    本篇文章主要介紹了Vue項目中引入外部文件的方法(css、js、less),非常具有實用價值,需要的朋友可以參考下
    2017-07-07
  • vue實現(xiàn)下拉菜單樹

    vue實現(xiàn)下拉菜單樹

    這篇文章主要為大家詳細(xì)介紹了vue實現(xiàn)下拉菜單樹,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-10-10
  • vue實現(xiàn)列表無縫滾動

    vue實現(xiàn)列表無縫滾動

    這篇文章主要為大家詳細(xì)介紹了vue實現(xiàn)列表無縫滾動,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-06-06
  • vue項目在安卓低版本機(jī)顯示空白的原因分析(兩種)

    vue項目在安卓低版本機(jī)顯示空白的原因分析(兩種)

    本文給大家?guī)韛ue項目在安卓低版本機(jī)顯示空白的原因分析,根據(jù)各自需求給大家?guī)砹藘煞N原因分析,大家可以參考下
    2018-09-09
  • 手把手教你vue實現(xiàn)動態(tài)路由

    手把手教你vue實現(xiàn)動態(tài)路由

    動態(tài)路由可以根據(jù)不同用戶登錄獲取不一樣的路由層級,可隨時調(diào)配路由,下面這篇文章主要給大家介紹了關(guān)于vue實現(xiàn)動態(tài)路由的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2022-07-07
  • Vite處理html模板插件之vite-plugin-html插件使用

    Vite處理html模板插件之vite-plugin-html插件使用

    這篇文章主要給大家介紹了關(guān)于Vite處理html模板插件之vite-plugin-html插件使用的相關(guān)資料,Vite是一個現(xiàn)代化的前端構(gòu)建工具,而vite-plugin-html是Vite的一個插件,用于在構(gòu)建時自動生成HTML文件,需要的朋友可以參考下
    2023-10-10
  • 基于Vue3文件拖拽上傳功能實現(xiàn)

    基于Vue3文件拖拽上傳功能實現(xiàn)

    這篇文章主要介紹了Vue3文件拖拽上傳功能,支持拖拽到此區(qū)域上傳,支持選擇多個文件/文件夾,代碼很簡單,對vue3拖拽上傳功能感興趣的朋友一起看看吧
    2022-10-10
  • 一文帶你了解Vue數(shù)組的變異方法

    一文帶你了解Vue數(shù)組的變異方法

    Vue框架提供了一些便捷的數(shù)組變異方法,包括push、pop、shift、unshift、splice、sort和reverse等,Vue的數(shù)組變異方法可以自動觸發(fā)DOM更新,本文就詳細(xì)帶大家了解一下Vue.js數(shù)組的變異方法
    2023-06-06

最新評論