專案 載浮載沉
版本 phaser3.55.2.js
1. 資源
phaser3官網 example
2. 閱讀
Move Objects According To The Mouse Position With Phaser 3
how-to-create-sprite-sheets-for-phaser3
Phaser 幫我撐個 30 天
3. 圖層順序
4. 圖片縮放
5. 設定container 包物件

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| function preload () { this.load.image('buttonBG', 'assets/sprites/button-bg.png'); this.load.image('buttonText', 'assets/sprites/button-text.png'); }
function create () { var bg = this.add.image(0, 0, 'buttonBG'); var text = this.add.image(0, 0, 'buttonText');
var container = this.add.container(400, 300, [ bg, text ]);
container.setSize(bg.width, bg.height); }
|
reference
6. 設定sprite切換

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
| function preload () { this.load.spritesheet('aliens', 'assets/sprites/bsquadron-enemies.png', { frameWidth: 192, frameHeight: 160 }); }
function create () { var alien = this.add.sprite(400, 300, 'aliens', 0).setInteractive();
alien.on('pointerover', function () {
this.setFrame(3);
});
alien.on('pointerout', function () {
this.setFrame(0);
}); }
|
reference
7. 世界座標中心

1 2 3
| const screenCenterX = this.cameras.main.worldView.x + this.cameras.main.width / 2; const screenCenterY = this.cameras.main.worldView.y + this.cameras.main.height / 2; const loadingText = this.add.text(screenCenterX, screenCenterY, 'Loading: 0%').setOrigin(0.5);
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
| for (let i = 0; i < 200; i++) { this.load.image('bg' + i, 'assets/undersea.jpg') }
let percentText = this.add.text(this.cameras.main.width / 2, this.cameras.main.height / 2, '', { font: '52px Open Sans', fill: '#ffffff' }).setOrigin(0.5, 0.5)
this.load.on('progress', value => { percentText.setText(parseInt(value * 100) + '%') })
this.load.on('complete', () => { percentText.destroy() })
|
reference
8. 設定可拖曳
1 2
| cube.setInteractive(); self.input.setDraggable(cube);
|
必須先設定互動,才能使用draggable 功能
9. 物件與物件之間 overlap
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| create() { self.RectangleToRectangle = Phaser.Geom.Intersects.RectangleToRectangle; self.GetBounds = Phaser.Display.Bounds.GetBounds; }
update() {
this.GetBounds(this.cube04, this.rect1); this.GetBounds(this.rect_water, this.rect2);
if (this.RectangleToRectangle(this.rect1, this.rect2)) { } }
|
reference
10. 設定游標
1 2
| this.systems.game.input.setCursor({ cursor: 'grabbing' }); this.systems.game.input.resetCursor({ cursor: 'true' });
|
11. 設定物件透明度
1 2
| object.alpha = 0.5; object.clearAlpha();
|
12. 設定物件顯示or隱藏
13. 設定動畫

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30
| self.tweens.add({ targets: [self.rect_water, self.rect_water_ruler], y: self.gSetting.water.y, ease: 'Power2', duration: 500, onComplete: self.onCompleteHandler, });
onCompleteHandler = (tween, targets, myImage) => {
tween.parent.killTweensOf(self.rect_water); tween.parent.killTweensOf(self.rect_water_ruler); }
self.tweens.add({
targets: [self.rect_water, self.rect_water_ruler], y: self.add_water_height(self, 50), ease: 'Power2', duration: 500, onComplete: self.onCompleteHandler, });
add_water_height = (self, add_h) => { return self.gSetting.water.y - add_h; }
|
14. 創建重力物件
1 2 3 4 5 6 7 8
| scene.create = function() { let ground = this.add.sprite(180, 400, 'tile') this.physics.add.existing(ground)
let ground2 = this.physics.add.sprite(180, 200, 'tile') }
|
15. 設定重力
1 2
| object.body.setAllowGravity(false); object.body.setAllowGravity(true);
|
16. 不受重力碰撞影響而變更位置
1
| ground.body.immovable = true
|

17. 設定碰撞
1 2 3 4 5 6
| this.physics.add.collider(ground, ground2);
self.physics.add.collider(cube, self.weigh, weight_event, null, self); self.physics.add.collider(cube, self.box_down_side, boxDownSide_event, null, self);
|
18. 切換場景

index.js
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
| import MainScene from './Scenes/MainScene.js'; import StartScene from './Scenes/StartScene.js';
var config = { type: Phaser.AUTO, scale: { mode: Phaser.Scale.FIT, autoCenter: Phaser.Scale.CENTER_BOTH, width: 1500, height: 700 }, physics: { default: 'arcade', arcade: { gravity: { y: 800 }, } }, scene: [StartScene,MainScene] }; var game = new Phaser.Game(config);
|
StartScene.js
1 2 3 4 5 6 7 8 9 10 11 12
| create() { btnStart.on('pointerover', function () { this.scene.sys.game.input.setCursor({ cursor: 'pointer' }); });
btnStart.on('pointerout', function () { this.scene.sys.game.input.resetCursor({ cursor: 'true' }); }); btnStart.on('pointerdown', function () { self.scene.start('MainScene'); }); }
|
19. 放大鏡效果




1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54
| class Example extends Phaser.Scene { constructor () { super(); }
preload () { this.load.image('pic', 'assets/pics/sword-art-online.jpg'); this.load.image('magnify-out', 'assets/sprites/magnify-glass-outside.png'); this.load.image('magnify-in', 'assets/sprites/magnify-glass-inside.png'); }
create () { this.add.image(400, 300, 'pic').setTint(0x2d2d2d);
const pic = this.add.image(400, 300, 'pic').setScale(1.05);
const lense = this.make.sprite({ x: 400, y: 300, key: 'magnify-in', add: false });
pic.mask = new Phaser.Display.Masks.BitmapMask(this, lense);
const magnify = this.add.image(400, 300, 'magnify-out');
this.input.on('pointermove', function (pointer) {
lense.x = pointer.x; lense.y = pointer.y;
magnify.x = pointer.x; magnify.y = pointer.y;
}); } }
const config = { type: Phaser.WEBGL, parent: 'phaser-example', width: 800, height: 600, scene: [ Example ] };
const game = new Phaser.Game(config);
|
reference
20. 這裡開始跟專案無關
21. 工具

22. 相機效果
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
| scene.gameOver = function() { this.cameras.main.shake(500) this.cameras.main.on('camerashakecomplete', () => { this.cameras.main.fade(500) }) this.cameras.main.on('camerafadeoutcomplete', () => { this.scene.restart() }) }
|
23. 設定動畫
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| scene.create = function() { this.man = this.add.sprite(50, 310, 'man', 0).setOrigin(0.5, 1).setInteractive()
this.man.scaleTween = this.tweens.add({ targets: this.man, scaleX: 1.2, scaleY: 1.2, duration: 1000 }) }
|
上面一開始載入完時,就會自動執行動畫,所以可以多添加 paused: true
1 2 3 4 5 6 7
| this.man.scaleTween = this.tweens.add({ targets: this.man, scaleX: 1.2, scaleY: 1.2, duration: 1000, paused: true })
|
1 2 3
| this.man.on('pointerdown', () => { this.man.scaleTween.restart() })
|
1 2 3 4 5 6 7 8 9
| scene.create = function() { this.anims.create({ key: 'walking', frames: this.anims.generateFrameNames('player', { frames: [1, 2 ]}), frameRate: 10, yoyo: true, repeat: -1 }) }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| scene.update = function() { if (this.cursors.left.isDown) { ..... this.player.anims.play('walking') } else if (this.cursors.right.isDown) { ..... this.player.anims.play('walking') } else { ..... this.player.anims.stop('walking') this.player.setFrame(0) } }
|
24. 遊戲重新開始
1 2 3
| this.registry.destroy(); this.events.off(); this.scene.restart();
|
25. 加載進度

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
| for (let i = 0; i < 200; i++) { this.load.image('bg' + i, 'assets/bg.png') }
let percentText = this.add.text(320, 160, '', { font: '24px Open Sans', fill: '#ffffff' }).setOrigin(0.5, 0.5)
this.load.on('progress', value => { percentText.setText(parseInt(value * 100) + '%') })
this.load.on('complete', () => { percentText.destroy() })
|
26. Follow
有時候,我們希望不侷限在特定的區域內,而是隨著遊戲者的位置來進行遊戲的移動。這時候可以調整 camera 的設定來達到這個目的( camera 跟隨著 player 的位置 )。
1 2 3 4 5 6 7
|
this.physics.world.bounds.width = 360 this.physics.world.bounds.height = 1000 ..... this.cameras.main.setBounds(0, 0, 360, 1000) this.cameras.main.startFollow(this.player)
|

這次我們學到了,如何設定 camera,依照 player 的位置,去移動 camera 的位置。可以朝著 x 軸移動的遊戲(馬力歐);或者 y 軸移動的遊戲(小朋友下樓梯)等遊戲劇本,來設定 camera 以及 physics.world 的邊界。
27. 燈光 light
首先你需要有 normal maps,可以使用工具來製作 SpriteIlluminator 工具來製作。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28
| scene.preload = function () { this.load.setPath('assets/'); this.load.image('character', ['01.png', '01_n.png']); }
scene.create = function() { let capguy = this.add.sprite(250, 300, 'character') capguy.setScale(0.5) capguy.setPipeline('Light2D')
let light = this.lights.addLight(350, 250, 200) this.lights.enable() this.lights.setAmbientColor(0x555555)
this.input.on('pointermove', function (pointer) { light.x = pointer.x; light.y = pointer.y; }); }
|
如果載入 sprite sheet 的話,也必須有 normal map sheet,我們可以使用 TexturePack 工具產生 json 檔,使用 load.multiatlas() 載入兩種 sheet,將 normal 連結到 sprite sheet 上。