From 33058f2b292b3a581333bdfb21b8f671898c5060 Mon Sep 17 00:00:00 2001 From: Peter Bengtsson Date: Tue, 8 Dec 2020 14:40:17 -0500 Subject: initial commit --- .../animations_and_tweens/index.html | 117 ++++++ .../bounce_off_the_walls/index.html | 49 +++ .../build_the_brick_field/index.html | 164 ++++++++ .../2d_breakout_game_phaser/buttons/index.html | 104 +++++ .../collision_detection/index.html | 59 +++ .../2d_breakout_game_phaser/extra_lives/index.html | 124 ++++++ .../2d_breakout_game_phaser/game_over/index.html | 53 +++ .../tutorials/2d_breakout_game_phaser/index.html | 57 +++ .../initialize_the_framework/index.html | 90 +++++ .../index.html | 71 ++++ .../move_the_ball/index.html | 49 +++ .../2d_breakout_game_phaser/physics/index.html | 99 +++++ .../player_paddle_and_controls/index.html | 122 ++++++ .../randomizing_gameplay/index.html | 61 +++ .../2d_breakout_game_phaser/scaling/index.html | 64 +++ .../2d_breakout_game_phaser/the_score/index.html | 76 ++++ .../win_the_game/index.html | 59 +++ .../bounce_off_the_walls/index.html | 99 +++++ .../build_the_brick_field/index.html | 106 +++++ .../collision_detection/index.html | 126 ++++++ .../create_the_canvas_and_draw_on_it/index.html | 114 ++++++ .../game_over/index.html | 81 ++++ .../2d_breakout_game_pure_javascript/index.html | 60 +++ .../move_the_ball/index.html | 143 +++++++ .../paddle_and_keyboard_controls/index.html | 122 ++++++ .../track_the_score_and_win/index.html | 92 +++++ .../index.html" | 113 ++++++ .../index.html" | 61 +++ .../index.html | 449 +++++++++++++++++++++ files/zh-cn/games/tutorials/index.html | 25 ++ 30 files changed, 3009 insertions(+) create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_phaser/animations_and_tweens/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_phaser/bounce_off_the_walls/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_phaser/build_the_brick_field/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_phaser/buttons/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_phaser/collision_detection/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_phaser/extra_lives/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_phaser/game_over/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_phaser/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_phaser/initialize_the_framework/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_phaser/load_the_assets_and_print_them_on_screen/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_phaser/move_the_ball/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_phaser/physics/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_phaser/player_paddle_and_controls/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_phaser/randomizing_gameplay/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_phaser/scaling/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_phaser/the_score/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_phaser/win_the_game/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/bounce_off_the_walls/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/build_the_brick_field/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/collision_detection/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/create_the_canvas_and_draw_on_it/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/game_over/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/move_the_ball/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/paddle_and_keyboard_controls/index.html create mode 100644 files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/track_the_score_and_win/index.html create mode 100644 "files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/\346\224\266\345\260\276\345\267\245\344\275\234/index.html" create mode 100644 "files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/\351\274\240\346\240\207\346\216\247\345\210\266/index.html" create mode 100644 files/zh-cn/games/tutorials/html5_gamedev_phaser_device_orientation/index.html create mode 100644 files/zh-cn/games/tutorials/index.html (limited to 'files/zh-cn/games/tutorials') diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_phaser/animations_and_tweens/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/animations_and_tweens/index.html new file mode 100644 index 0000000000..ff5aa23230 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/animations_and_tweens/index.html @@ -0,0 +1,117 @@ +--- +title: Animations and tweens +slug: Games/Tutorials/2D_breakout_game_Phaser/Animations_and_tweens +tags: + - 2D + - Animation + - Beginner + - Canvas + - Games + - JavaScript + - Phaser + - Tutorial + - tween +translation_of: Games/Tutorials/2D_breakout_game_Phaser/Animations_and_tweens +--- +
{{GamesSidebar}}
+ +
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Extra_lives", "Games/Workflows/2D_Breakout_game_Phaser/Buttons")}}

+ +
+

这是Gamedev Phaser教程 16 第14步您可以在Gamedev-Phaser-Content-Kit / demos / lesson14.html完成本课程后找到源代码

+
+ +

为了使游戏看起来更加多汁和活泼,我们可以使用动画和补间。这将导致更好,更有趣的体验。让我们来探讨如何在游戏中实现Phaser动画和补间。

+ +

动画

+ +

在Phaser,动画中,涉及从外部来源获取spritesheet并依次显示sprites。作为一个例子,当碰到一些东西时,我们会让球摇摆。

+ +

首先,从Github抓取spritesheet并将其保存在您的/img目录中。

+ +

接下来,我们将加载spritesheet - 将以下行放在preload()函数的底部

+ +
game.load.spritesheet('ball', 'img/wobble.png', 20, 20);
+
+ +

而不是加载单个图像的球,我们可以加载整个spritesheet - 不同图像的集合。我们将按顺序显示精灵,创造动画的幻觉。spritesheet()方法的两个额外的表格确定给定spritesheet文件中每个单个框架的宽度和高度,指示程序如何切割以获取单个框架。

+ +

加载动画

+ +

接下来,进入你的create()函数,找到加载球精灵的行,下面的调用animations.add()如下所示:

+ +
ball = game.add.sprite(50, 250, 'ball');
+ball.animations.add('wobble', [0,1,0,2,0,1,0,2,0], 24);
+
+ +

要向对象添加动画,我们使用该animations.add()方法,其中包含以下参数

+ + + +

当球击中桨时应用动画

+ +

arcade.collide()处理球和桨(第一行内部update(),见下文)之间的碰撞方法调用中,我们可以添加一个额外的参数,该参数指定每次发生碰撞时执行的功能,与该ballHitBrick()功能相同更新内部的第一行update(),如下所示:

+ +
function update() {
+    game.physics.arcade.collide(ball, paddle, ballHitPaddle);
+    game.physics.arcade.collide(ball, bricks, ballHitBrick);
+    paddle.x = game.input.x || game.world.width*0.5;
+}
+
+ +

然后我们可以创建ballHitPaddle()函数(具有ballpaddle作为默认参数),在调用时播放摆动动画。在结束</script>标签之前添加以下方法

+ +
function ballHitPaddle(ball, paddle) {
+    ball.animations.play('wobble');
+}
+
+ +

每次球击中桨时都会播放动画。你也可以animations.play()ballHitBrick()函数添加调用,如果你觉得它会使游戏看起来更好。

+ +

补间

+ +

而动画依次播放外部精灵,补间游戏中物体的属性平滑,如宽度或不透明度。

+ +

让我们在游戏中增加一个补间,使砖块在被球击中时顺利消失。转到您的ballhitBrick()功能,找到您的brick.kill();行,并将其替换为以下内容:

+ +
var killTween = game.add.tween(brick.scale);
+killTween.to({x:0,y:0}, 200, Phaser.Easing.Linear.None);
+killTween.onComplete.addOnce(function(){
+    brick.kill();
+}, this);
+killTween.start();
+
+ +

让我们来看看这里,看看这里发生了什么:

+ +
    +
  1. 当定义一个新的补间时,你必须指定哪些属性将被补间 - 在我们的例子中,而不是在被球击中时立即隐藏砖块,我们将把它们的宽度和高度缩放到零,所以它们将很好地消失。最后,我们使用该add.tween()方法,指定brick.scale为参数,因为这是我们想要补间。
  2. +
  3. to()方法定义补间结束时对象的状态。它需要一个包含所选参数的期望结束值的对象(比例取尺度值,1为大小的100%,0为大小的0%等),补间的时间(以毫秒为单位)以及使用的宽松类型补间。
  4. +
  5. 我们还将添加可选的onComplete事件处理程序,该处理程序定义了在补间程序完成时要执行的函数。
  6. +
  7. 最后一件事是立即开始补间start()
  8. +
+ +

这是补间定义的扩展版本,但是我们也可以使用速记语法:

+ +
game.add.tween(brick.scale).to({x:2,y:2}, 500, Phaser.Easing.Elastic.Out, true, 100);
+
+ +

这个补间将使用弹性宽松在半秒内将砖的比例翻倍,将自动启动,延迟100毫秒。

+ +

比较你的代码

+ +

您可以在下面的现场演示中查看本课程的完成代码,并使用它来更好地了解它的工作原理:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/9o4pakrb/","","400")}}

+ +

下一步

+ +

动画和tweens看起来很不错,但我们可以添加更多的我们的游戏 - 在下一节我们将看看处理按钮输入。

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Extra_lives", "Games/Workflows/2D_Breakout_game_Phaser/Buttons")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_phaser/bounce_off_the_walls/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/bounce_off_the_walls/index.html new file mode 100644 index 0000000000..2cc91361c6 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/bounce_off_the_walls/index.html @@ -0,0 +1,49 @@ +--- +title: Bounce off the walls +slug: Games/Tutorials/2D_breakout_game_Phaser/Bounce_off_the_walls +tags: + - 2D + - Beginner + - Canvas + - Games + - JavaScript + - Phaser + - Tutorial + - bouncing +translation_of: Games/Tutorials/2D_breakout_game_Phaser/Bounce_off_the_walls +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Physics", "Games/Workflows/2D_Breakout_game_Phaser/Player_paddle_and_controls")}}

+ +
+

这是Gamedev Phaser教程 6 Gamedev-Phaser-Content-Kit / demos / lesson06.html完成本课后,您可以找到源代码

+
+ +

现在已经介绍了物理引擎,我们可以开始在游戏中实现碰撞检测 - 首先我们来看看墙壁。

+ +

反弹边界

+ +

让我们的球从墙壁上弹起的最简单的方法是告诉框架,我们想要将<canvas>元素的边界视为墙壁,而不是让球移过它们。在Phaser中,可以使用该collideWorldsBound属性轻松实现在现有game.physics.enable()方法调用之后添加此行

+ +
ball.body.collideWorldBounds = true;
+
+ +

现在球将停在屏幕的边缘,而不是消失,但它不会弹起。为了使这种情况发生,我们必须设置它的bounciness。在上一行下面添加以下行:

+ +
ball.body.bounce.set(1);
+
+ +

再次尝试重新加载index.html - 现在您应该看到球从墙壁上弹起并在画布区域内移动。

+ +

比较你的代码

+ +

您可以在下面的现场演示中查看本课程的完成代码,并使用它来更好地了解它的工作原理:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/dcw36opz/","","400")}}

+ +

下一步

+ +

现在开始看起来更像是一个游戏,但是我们无法以任何方式控制它 - 现在是介绍玩家挡板和控制的时候了

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Physics", "Games/Workflows/2D_Breakout_game_Phaser/Player_paddle_and_controls")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_phaser/build_the_brick_field/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/build_the_brick_field/index.html new file mode 100644 index 0000000000..981b592469 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/build_the_brick_field/index.html @@ -0,0 +1,164 @@ +--- +title: Build the brick field +slug: Games/Tutorials/2D_breakout_game_Phaser/Build_the_brick_field +tags: + - 2D + - Beginner + - Canvas + - Games + - JavaScript + - Phaser + - Tutorial +translation_of: Games/Tutorials/2D_breakout_game_Phaser/Build_the_brick_field +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Game_over", "Games/Workflows/2D_Breakout_game_Phaser/Collision_detection")}}

+ +
+

这是Gamedev Phaser教程 16 第9步Gamedev-Phaser-Content-Kit / demos / lesson09.html完成本课后,您可以找到源代码

+
+ +

建立砖块比将单个对象添加到屏幕要复杂一点,尽管使用Phaser还是比纯JavaScript更容易。我们来探讨如何创建一组砖块,并使用循环在屏幕上打印。

+ +

定义新变量

+ +

首先,我们定义所需的变量 - 在以前的变量定义中添加以下内容:

+ +
var bricks;
+var newBrick;
+var brickInfo;
+
+ +

bricks变量将用于创建一个组,newBrick将在循环的每次迭代中添加到组中的新对象,brickInfo并将存储我们需要的所有数据。

+ +

渲染砖图像

+ +

接下来,我们加载砖的图像 - load.image()在其他地方添加以下调用:

+ +
function preload() {
+    // ...
+    game.load.image('brick', 'img/brick.png');
+}
+
+ +

您还需要从Github抓取砖图像并将其保存在您的/img目录中。

+ +

画砖

+ +

我们将将所有用于绘制砖块的代码放在一个initBricks函数中,以使其与其余代码分离。initBrickscreate()函数末尾添加一个调用

+ +
function create(){
+    // ...
+    initBricks();
+}
+
+ +

现在到函数本身。initBricks()在我们的游戏代码末尾添加功能,就在关闭</ script>标签之前,如下所示。首先我们已经包括了这个  brickInfo对象,因为这很快就会派上用场:

+ +
function initBricks() {
+    brickInfo = {
+        width: 50,
+        height: 20,
+        count: {
+            row: 7,
+            col: 3
+        },
+        offset: {
+            top: 50,
+            left: 60
+        },
+        padding: 10
+    };
+}
+
+ +

这个brickInfo对象将包含我们需要的所有信息:单个砖的宽度和高度,我们将在屏幕上看到的砖的行数和列数,顶部和左边的偏移量(画布上我们将开始绘制的位置)砖块)和每一列和砖块之间的填充。

+ +

现在,让我们开始创建砖块 - 首先添加一个空组来包含砖块,在initBricks()函数底部添加以下行

+ +
bricks = game.add.group();
+
+ +

我们可以循环遍历行和列,以便在每次迭代中创建新的砖块 - 在上一行代码下面添加以下嵌套循环:

+ +
for(c=0; c<brickInfo.count.col; c++) {
+    for(r=0; r<brickInfo.count.row; r++) {
+        // create new brick and add it to the group
+    }
+}
+
+ +

这样我们将创建我们需要的确切数量的砖,并将它们全部包含在一个组中。现在我们需要在嵌套循环结构中添加一些代码来绘制每个砖块。填写内容如下图所示:

+ +
for(c=0; c<brickInfo.count.col; c++) {
+    for(r=0; r<brickInfo.count.row; r++) {
+        var brickX = 0;
+        var brickY = 0;
+        newBrick = game.add.sprite(brickX, brickY, 'brick');
+        game.physics.enable(newBrick, Phaser.Physics.ARCADE);
+        newBrick.body.immovable = true;
+        newBrick.anchor.set(0.5);
+        bricks.add(newBrick);
+    }
+}
+
+ +

在这里,我们循环遍历行和列,创建新的砖块并将其放在屏幕上。新创建的砖块为Arcade物理引擎启用,它的身体被设置为不可移动(所以当球被击中时它不会移动),我们还将锚点放在中间并添加砖到集团。

+ +

目前的问题是,我们在一个地方绘制所有的砖,坐标(0,0)。我们需要做的是将每个砖块绘制在自己的x和y位置。更新brickXbrickY行如下:

+ +
var brickX = (r*(brickInfo.width+brickInfo.padding))+brickInfo.offset.left;
+var brickY = (c*(brickInfo.height+brickInfo.padding))+brickInfo.offset.top;
+
+ +

每个brickX位置都是brickInfo.widthbrickInfo.padding号乘以行号r,加上brickInfo.offset.left用于所述逻辑brickY是不同之处在于它使用的值列号相同cbrickInfo.heightbrickInfo.offset.top现在每个砖都可以放置在正确的位置,每个砖块之间填充,并从左侧和顶部画布边缘偏移绘制。

+ +

检查initBricks()代码

+ +

这是功能的完整代码initBricks()

+ +
function initBricks() {
+    brickInfo = {
+        width: 50,
+        height: 20,
+        count: {
+            row: 7,
+            col: 3
+        },
+        offset: {
+            top: 50,
+            left: 60
+        },
+        padding: 10
+    }
+    bricks = game.add.group();
+    for(c=0; c<brickInfo.count.col; c++) {
+        for(r=0; r<brickInfo.count.row; r++) {
+            var brickX = (r*(brickInfo.width+brickInfo.padding))+brickInfo.offset.left;
+            var brickY = (c*(brickInfo.height+brickInfo.padding))+brickInfo.offset.top;
+            newBrick = game.add.sprite(brickX, brickY, 'brick');
+            game.physics.enable(newBrick, Phaser.Physics.ARCADE);
+            newBrick.body.immovable = true;
+            newBrick.anchor.set(0.5);
+            bricks.add(newBrick);
+        }
+    }
+}
+
+ +

如果您现在重新加载index.html,您应该看到在屏幕上打印的砖块彼此相距甚远。

+ +

比较你的代码

+ +

您可以在下面的现场演示中查看本课程的完成代码,并使用它来更好地了解它的工作原理:

+ +

 

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/cck2b9e8/","","400")}}

+ +

下一步

+ +

有些东西丢失了 球不经停,经过砖块 - 我们需要适当的碰撞检测

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Game_over", "Games/Workflows/2D_Breakout_game_Phaser/Collision_detection")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_phaser/buttons/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/buttons/index.html new file mode 100644 index 0000000000..2b29b39714 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/buttons/index.html @@ -0,0 +1,104 @@ +--- +title: Buttons +slug: Games/Tutorials/2D_breakout_game_Phaser/Buttons +tags: + - 2D + - Beginner + - Buttons + - Canvas + - Games + - JavaScript + - Phaser + - Tutorial +translation_of: Games/Tutorials/2D_breakout_game_Phaser/Buttons +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Animations_and_tweens", "Games/Workflows/2D_Breakout_game_Phaser/Randomizing_gameplay")}}

+ +
+

这是Gamedev Phaser教程 16 第15步您可以在Gamedev-Phaser-Content-Kit / demos / lesson15.html完成本课程后找到源代码

+
+ +

而不是立即开始游戏,我们可以通过添加他们可以按的开始按钮将该决定留给玩家。我们来调查如何做到这一点。

+ +

新变量

+ +

我们需要一个变量来存储表示游戏当前是否正在播放的布尔值,另一个代表我们的按钮。将以下行添加到其他变量定义之下:

+ +
var playing = false;
+var startButton;
+
+ +

加载按钮spritesheet

+ +

我们可以加载按钮spritesheet与我们加载球的摆动动画相同的方式。将以下内容添加到preload()函数底部

+ +
game.load.spritesheet('button', 'img/button.png', 120, 40);
+
+ +

单个按钮框架宽120像素,高40像素。

+ +

您还需要从Github抓取按钮spritesheet,并将其保存在您的/img目录中。

+ +

将按钮添加到游戏中

+ +

使用该add.button方法可以将新的按钮添加到游戏中将以下行添加到create()函数的底部

+ +
startButton = game.add.button(game.world.width*0.5, game.world.height*0.5, 'button', startGame, this, 1, 0, 2);
+startButton.anchor.set(0.5);
+
+ +

button()方法的参数如下:

+ + + +
+

注意:超越事件与悬停相同,当指针从按钮中移出时,当按下按钮时,向下移动。

+
+ +

现在我们需要定义startGame()上面代码中引用函数:

+ +
function startGame() {
+    startButton.destroy();
+    ball.body.velocity.set(150, -150);
+    playing = true;
+}
+
+ +

当按下按钮时,我们删除按钮,设置球的初始速度并将playing变量设置true

+ +

最后对于这一部分,回到你的create()函数,找到ball.body.velocity.set(150, -150);一行,并删除它。你只需要按下按钮时移动球,而不是之前!

+ +

在游戏开始之前仍然保持桨

+ +

它按预期工作,但是当游戏尚未开始时,我们仍然可以移动桨,这看起来有点愚蠢。为了阻止这一点,我们可以利用playing变量,使得桨只有在游戏开始时才能移动。要做到这一点,调整update()功能如下所示:

+ +
function update() {
+    game.physics.arcade.collide(ball, paddle, ballHitPaddle);
+    game.physics.arcade.collide(ball, bricks, ballHitBrick);
+    if(playing) {
+        paddle.x = game.input.x || game.world.width*0.5;
+    }
+}
+
+ +

这样一来,在所有的装载和准备之后,但在实际游戏开始之前,桨是不可移动的。

+ +

比较你的代码

+ +

您可以在下面的现场演示中查看本课程的完成代码,并使用它来更好地了解它的工作原理:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/1rpj71k4/","","400")}}

+ +

下一步

+ +

在本系列文章中我们将做的最后一件事情是,通过添加一些随机化的方式,球从球上弹起来,使游戏更有趣

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Animations_and_tweens", "Games/Workflows/2D_Breakout_game_Phaser/Randomizing_gameplay")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_phaser/collision_detection/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/collision_detection/index.html new file mode 100644 index 0000000000..1977d93fb9 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/collision_detection/index.html @@ -0,0 +1,59 @@ +--- +title: Collision detection +slug: Games/Tutorials/2D_breakout_game_Phaser/Collision_detection +tags: + - 2D + - Beginner + - Canvas + - Games + - JavaScript + - Phaser + - Tutorial + - collision detection +translation_of: Games/Tutorials/2D_breakout_game_Phaser/Collision_detection +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Build_the_brick_field", "Games/Workflows/2D_Breakout_game_Phaser/The_score")}}

+ +
+

这是Gamedev Phaser教程 16 第10步您可以在Gamedev-Phaser-Content-Kit / demos / lesson10.html完成本课程后找到源代码

+
+ +

现在接下来的挑战 - 球和砖块之间的碰撞检测。幸运的是,我们可以使用物理引擎来检查单个对象(如球和桨)之间的碰撞,也可以检测对象和组之间的碰撞。

+ +

砖/球碰撞检测

+ +

物理引擎使一切都变得更容易 - 我们只需要添加两个简单的代码。首先,在你的update()函数中添加一行,检查球和砖之间的碰撞检测,如下所示:

+ +
function update() {
+    game.physics.arcade.collide(ball, paddle);
+    game.physics.arcade.collide(ball, bricks, ballHitBrick);
+    paddle.x = game.input.x || game.world.width*0.5;
+}
+
+ +

球的位置是根据组中所有砖的位置计算的。第三个可选参数是发生冲突时执行的功能ballHitBrick()创建这个新功能作为代码的底部,就在结束</script>标签之前,如下所示:

+ +
function ballHitBrick(ball, brick) {
+    brick.kill();
+}
+
+ +

就是这样!重新加载你的代码,你应该看到新的碰撞检测工作正常。

+ +

感谢Phaser,有两个参数传递给函数 - 第一个是球,我们在碰撞方法中明确定义,第二个是球碰撞的砖组中的单个砖。在功能内部,我们从屏幕上删除所讨论的砖块,只需运行其kill()方法即可。

+ +

您将期望在使用纯JavaScript时编写更多自己的计算机来实现碰撞检测这是使用框架的好处 - 您可以为Phaser留下大量无聊的代码,并专注于制作游戏中最有趣和最有趣的部分。

+ +

比较你的代码

+ +

您可以在下面的现场演示中查看本课程的完成代码,并使用它来更好地了解它的工作原理:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/wwneakwf/","","400")}}

+ +

下一步

+ +

我们可以打砖块并删除它们,这已经是游戏的一个很好的补充。结果,更好地计算被毁砖增加得分

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Build_the_brick_field", "Games/Workflows/2D_Breakout_game_Phaser/The_score")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_phaser/extra_lives/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/extra_lives/index.html new file mode 100644 index 0000000000..81143d4c14 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/extra_lives/index.html @@ -0,0 +1,124 @@ +--- +title: Extra lives +slug: Games/Tutorials/2D_breakout_game_Phaser/Extra_lives +tags: + - 2D + - Beginner + - Canvas + - Games + - JavaScript + - Phaser + - Tutorial + - lives +translation_of: Games/Tutorials/2D_breakout_game_Phaser/Extra_lives +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Win_the_game", "Games/Workflows/2D_Breakout_game_Phaser/Animations_and_tweens")}}

+ +
+

这是Gamedev Phaser教程 16 第13步您可以在Gamedev-Phaser-Content-Kit / demos / lesson13.html完成本课程后找到源代码

+
+ +

我们可以通过增加生活使游戏更愉快。在这篇文章中,我们将实施一个生活系统,以便玩家可以继续玩,直到他们失去了三个生命,而不仅仅是一个人。

+ +

新变量

+ +

在代码中的现有添加下面添加以下新变量:

+ +
var lives = 3;
+var livesText;
+var lifeLostText;
+
+ +

这些分别将存储生命数,显示剩余生命数的文本标签,以及当玩家失去生命之后将在屏幕上显示的文本标签。

+ +

定义新的文本标签

+ +

定义文本看起来像我们已经在分数课上已经做scoreTextcreate()函数的现有定义下方添加以下行

+ +
livesText = game.add.text(game.world.width-5, 5, 'Lives: '+lives, { font: '18px Arial', fill: '#0095DD' });
+livesText.anchor.set(1,0);
+lifeLostText = game.add.text(game.world.width*0.5, game.world.height*0.5, 'Life lost, click to continue', { font: '18px Arial', fill: '#0095DD' });
+lifeLostText.anchor.set(0.5);
+lifeLostText.visible = false;
+
+ +

livesTextlifeLostText物体看起来非常相似的scoreText一个-它们定义在屏幕上的位置,显示实际文本和字体样式。前者被锚定在其右上边缘上,与屏幕正确对齐,后者位于中心位置,两者均使用anchor.set()

+ +

lifeLostText会表示,只有当生命消失,因此其知名度初始设置为false

+ +

使我们的文字造型干燥

+ +

正如你可能已经注意到,我们使用相同的造型为三种文本:scoreTextlivesTextlifeLostText如果我们想要更改字体大小或颜色,我们必须在多个地方进行。为了使我们更容易维护,将来我们可以创建一个单独的变量来保存我们的样式,让我们textStyle将其调用并放在文本定义之前:

+ +
textStyle = { font: '18px Arial', fill: '#0095DD' };
+
+ +

现在我们可以在使用文本标签的时候使用这个变量 - 更新你的代码,使文本样式的多个实例被替换为变量:

+ +
scoreText = game.add.text(5, 5, 'Points: 0', textStyle);
+livesText = game.add.text(game.world.width-5, 5, 'Lives: '+lives, textStyle);
+livesText.anchor.set(1,0);
+lifeLostText = game.add.text(game.world.width*0.5, game.world.height*0.5, 'Life lost, click to continue', textStyle);
+lifeLostText.anchor.set(0.5);
+lifeLostText.visible = false;
+
+ +

这样一来,改变一个变量中的字体将会将更改应用于每个使用的地方。

+ +

生活处理代码

+ +

为了在我们的游戏中实现生活,让我们先改变球对onOutOfBounds事件的影响。而不是执行匿名函数并立即显示警报:

+ +
ball.events.onOutOfBounds.add(function(){
+    alert('Game over!');
+    location.reload();
+}, this);
+
+ +

我们将分配一个所谓的新功能ballLeaveScreen删除以前的事件处理程序(如上所示),并将其替换为以下行:

+ +
ball.events.onOutOfBounds.add(ballLeaveScreen, this);
+
+ +

我们想减少每次球离开帆布的人数。ballLeaveScreen()在代码末尾添加函数定义:

+ +
function ballLeaveScreen() {
+    lives--;
+    if(lives) {
+        livesText.setText('Lives: '+lives);
+        lifeLostText.visible = true;
+        ball.reset(game.world.width*0.5, game.world.height-25);
+        paddle.reset(game.world.width*0.5, game.world.height-5);
+        game.input.onDown.addOnce(function(){
+            lifeLostText.visible = false;
+            ball.body.velocity.set(150, -150);
+        }, this);
+    }
+    else {
+        alert('You lost, game over!');
+        location.reload();
+    }
+}
+
+ +

而不是立即打印警报,当你失去了一生,我们首先从当前的数字减去一个生命,并检查它是否是一个非零值。如果是,那么玩家还是有一些生命剩下,可以继续玩 - 他们会看到生命中的消息,球和桨的位置将被重置在屏幕上和下一个输入(点击或触摸),消息将被隐藏球将再次开始移动。

+ +

当可用生活数量达到零时,游戏结束,并显示游戏过期警报消息。

+ +

事件

+ +

您可能已经注意到了add(),并addOnce()在上面的两个代码块的方法调用,并想知道它们的区别。不同之处在于,该add()方法绑定给定的函数,并使其在每次事件发生时执行,同时addOnce()当您希望绑定函数只执行一次,然后解除绑定时有用,因此不会再次执行。在我们的例子中,每个outOfBounds事件ballLeaveScreen都将被执行,但当球离开屏幕时,我们只想从屏幕上删除一次消息。

+ +

比较你的代码

+ +

您可以在下面的现场演示中查看本课程的完成代码,并使用它来更好地了解它的工作原理:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/yk1c5n0b/","","400")}}

+ +

下一步

+ +

生活让游戏更加宽容 - 如果你失去一个生命,你还剩下两个,可以继续玩。现在让我们通过添加动画和补间来扩展游戏的外观和感觉

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Win_the_game", "Games/Workflows/2D_Breakout_game_Phaser/Animations_and_tweens")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_phaser/game_over/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/game_over/index.html new file mode 100644 index 0000000000..d91bcc5e3b --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/game_over/index.html @@ -0,0 +1,53 @@ +--- +title: Game over +slug: Games/Tutorials/2D_breakout_game_Phaser/Game_over +tags: + - 2D + - Beginner + - Canvas + - Games + - JavaScript + - Phaser + - Tutorial + - game over +translation_of: Games/Tutorials/2D_breakout_game_Phaser/Game_over +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Player_paddle_and_controls", "Games/Workflows/2D_Breakout_game_Phaser/Build_the_brick_field")}}

+ +
+

这是Gamedev Phaser教程 16 第8步Gamedev-Phaser-Content-Kit / demos / lesson08.html完成本课后,您可以找到源代码

+
+ +

为了使游戏更有趣,我们可以引入失去的能力 - 如果在到达屏幕底部边缘之前没有击球,那么这个游戏将会结束。

+ +

如何输

+ +

为了提供丢失的能力,我们将禁用球与屏幕底部的碰撞。create()函数内添加下面的代码刚刚定义球的属性就好了:

+ +
game.physics.arcade.checkCollision.down = false;
+
+ +

这将使三个墙壁(顶部,左侧和右侧)弹回球,但是第四个(底部)将消失,如果桨错过,则球从屏幕上脱落。我们需要一种方法来检测并相应地采取行动。在以前的新的下方添加以下行:

+ +
ball.checkWorldBounds = true;
+ball.events.onOutOfBounds.add(function(){
+    alert('Game over!');
+    location.reload();
+}, this);
+
+ +

添加这些行将使得球检查世界(在我们的例子中是画布)边界并执行绑定到onOutOfBounds事件的函数当您点击生成的警报时,页面将被重新加载,以便您可以再次播放。

+ +

比较你的代码

+ +

您可以在下面的现场演示中查看本课程的完成代码,并使用它来更好地了解它的工作原理:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/436bckb7/","","400")}}

+ +

下一步

+ +

现在的基本游戏就是让我们通过引入砖块来更有趣的是 - 现在是建造砖块的时候了

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Player_paddle_and_controls", "Games/Workflows/2D_Breakout_game_Phaser/Build_the_brick_field")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_phaser/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/index.html new file mode 100644 index 0000000000..9f283f7774 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/index.html @@ -0,0 +1,57 @@ +--- +title: 使用Phaser开发2D breakout game +slug: Games/Tutorials/2D_breakout_game_Phaser +tags: + - Phaser + - 教程 + - 游戏 +translation_of: Games/Tutorials/2D_breakout_game_Phaser +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{Next("Games/Workflows/2D_Breakout_game_Phaser/Initialize_the_framework")}}

+ +

在这个手把手的教程中,我们将使用Phaser框架制作一个使用JavaScript构建简单的MDN消除游戏。

+ +

教程的每一步骤都会有可供修改的样品来玩,所以你可以看到开发的每一步中间步骤。 您将学到如何使用Phaser框架来实现基础游戏机制的基本知识,诸如渲染和移动图像,碰撞检测,控制机制,框架特定的帮助器功能,动画和补间,以及获胜和失败状态等。

+ +

为了充分理解这一系列的文章,您应该确保已有基本的中级JavaScript知识。学完本教程,您将有能力用Phaser构建简单的Web游戏。

+ +

Gameplay screen from the game MDN Breakout created with Phaser where you can use your paddle to bounce the ball and destroy the brick field, with keeping the points and lives.

+ +

教学清单

+ +

所有的课程 — 以及我们接下来将一起做的各个版本的 MDN Breakout game  都能在 GitHub上找到

+ +
    +
  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. +
+ +

学习路线的小提示 — 最好先熟悉使用原生JavaScript进行网页游戏开发,这样可以打下坚实的基础.如果你还不熟悉原生javascript开发,我们建议你先过一遍这个系列, 使用原生Javascript开发MDN消除游戏.

+ +

在那之后,你就能随意挑选框架并用在你的项目中;我们选择了Phaser这个稳定优越的框架,它有着好的支持和社区环境以及大量优秀的插件. 框架加速了开发并能帮你管理无趣的部分,让你专注于有意思的事务. 然而, 框架也有不好的地方, 所以当一些意想不到的事情发生了或者想实现一些框架没有提供的功能时,你就将需要原生的JavaScript知识了.

+ +
+

注意: 本系列文章可用作实际游戏开发的材料。 如果您想要使用Phaser讨论游戏开发,您还可以使用基于本教程的 Gamedev Phaser内容套件.

+
+ +

Next steps

+ +

好了,那我们就开始吧!前往系列第一部分 — 初始化框架.

+ +

{{Next("Games/Workflows/2D_Breakout_game_Phaser/Initialize_the_framework")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_phaser/initialize_the_framework/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/initialize_the_framework/index.html new file mode 100644 index 0000000000..ed6ef03718 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/initialize_the_framework/index.html @@ -0,0 +1,90 @@ +--- +title: Initialize the framework +slug: Games/Tutorials/2D_breakout_game_Phaser/Initialize_the_framework +tags: + - Canvas + - HTML + - JavaScript + - Phaser + - 二维 + - 入门 + - 教程 + - 游戏 +translation_of: Games/Tutorials/2D_breakout_game_Phaser/Initialize_the_framework +--- +
{{GamesSidebar}}
{{IncludeSubnav("/zh-CN/docs/Games")}}
+ +

{{PreviousNext("Games/Tutorials/2D_Breakout_game_Phaser", "Games/Tutorials/2D_Breakout_game_Phaser/Scaling")}}

+ +
+

这是Gamedev Phaser教程系列的第一课. 在课程完成之后,你可以在Gamedev-Phaser-Content-Kit/demos/lesson01.html找到源码.

+
+ +

在我们开始写游戏的功能之前,我们需要创建一个用来内部渲染游戏的基础架构.使用HTML就能做到 — Parser框架将生成所需的 {{htmlelement("canvas")}} 元素.

+ +

游戏的HTML

+ +

HTML文档结构非常的简单,这个游戏将整个被渲染在框架生成的{{htmlelement("canvas")}} 元素上. 拿起你最爱的编辑器,挑一个好目录,创建一个HTML文档,存成index.html,然后写下下面的代码:

+ +
<!DOCTYPE html>
+<html>
+<head>
+    <meta charset="utf-8" />
+    <title>Gamedev Phaser Workshop - lesson 01: Initialize the framework</title>
+    <style>* { padding: 0; margin: 0; }</style>
+    <script src="js/phaser.min.js"></script>
+</head>
+<body>
+<script>
+    var game = new Phaser.Game(480, 320, Phaser.AUTO, null, {
+      preload: preload, create: create, update: update
+    });
+    function preload() {}
+    function create() {}
+    function update() {}
+</script>
+</body>
+</html>
+
+ +

下载Phaser

+ +

下面我们将下载Phaser的代码,并应用到我们的HTML文档中.

+ +
    +
  1. 进入 Phaser 下载页面.
  2. +
  3. 选择最适合你的下载项 — 我们建议选择min.js,因为它最小,而且你不太可能想去看它的源码
  4. +
  5. 将Phaser的源码存到一个和index.html同级的 /js 的目录下
  6. +
  7. 在上面第一个 {{htmlelement("script")}} 标签里写下phaser的路径.
  8. +
+ +

捋一捋我们干了些啥

+ +

这个时候我们在 {{htmlelement("header")}} 里定义了 {{htmlelement("charset")}} ,{{htmlelement("title")}} 和一些基础的css来重置默认的margin和padding. 我们也用 {{htmlelement("script")}} 标签向页面引入了 Phaser 源码。{{htmlelement("body ")}} 里也有一个 {{htmlelement("script")}} 标签,我们将在里面写 JavaScript 代码来渲染和控制游戏。

+ +

{{htmlelement("canvas")}} 元素是由框架自动生成的。我们是通过 Phaser.Game 创建一个对象并赋给了 game 变量来完成初始化的。参数的含义是:

+ + + +

完整示例

+ +

以下是第一章的完整代码,可以直接在 JSFiddle 中运行:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/h6cwzv2b/","","400")}}

+ +

下一步

+ +

现在我们已经完成了一个简单的 HTML 页面,并且学习了如何安装 Phaser, 让我们继续学习第二章: scaling.

+ +

{{PreviousNext("Games/Tutorials/2D_Breakout_game_Phaser", "Games/Tutorials/2D_Breakout_game_Phaser/Scaling")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_phaser/load_the_assets_and_print_them_on_screen/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/load_the_assets_and_print_them_on_screen/index.html new file mode 100644 index 0000000000..135bcee403 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/load_the_assets_and_print_them_on_screen/index.html @@ -0,0 +1,71 @@ +--- +title: Load the assets and print them on screen +slug: >- + Games/Tutorials/2D_breakout_game_Phaser/Load_the_assets_and_print_them_on_screen +tags: + - 2D + - Beginner + - Canvas + - Games + - JavaScript + - Phaser + - Sprites + - Tutorial +translation_of: >- + Games/Tutorials/2D_breakout_game_Phaser/Load_the_assets_and_print_them_on_screen +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Scaling", "Games/Workflows/2D_Breakout_game_Phaser/Move the ball")}}

+ +
+

这是Gamedev Phaser教程 16 第三步您可以在Gamedev-Phaser-Content-Kit / demos / lesson03.html完成本课程后找到源代码

+
+ +

我们的游戏将围绕屏幕滚动,弹出一个桨,摧毁砖块赚取积分 - 熟悉吗?在本文中,我们将介绍如何将sprite添加到我们的gameworld中。

+ +

有一个球

+ +

我们开始创建一个JavaScript变量来表示我们的球 - 在游戏初始化代码(我们的var game...块)和preload()函数之间添加以下行

+ +
var ball;
+
+ +
+

注意:为了本教程,我们将使用全局变量。本教程的目的是教导Phaser特定的游戏开发方法,而不是主观的最佳方法。

+
+ +

加载球精灵

+ +

使用Phaser加载图像并将其打印在我们的画布上比使用纯JavaScript容易得多。要加载资产,我们将使用game由Phaser创建对象,执行其load.image()方法。preload()函数的底部添加以下新行

+ +
function preload() {
+    // ...
+    game.load.image('ball', 'img/ball.png');
+}
+
+ +

第一个参数是我们要提供资产的名称 - 这将在我们的游戏代码中使用,例如我们的ball变量名称,所以我们需要确保它是一样的。第二个参数是图形资源的相对路径。在我们的情况下,我们将加载我们的球的图像(请注意,文件名不一定是一致的,但我们建议,因为它使一切更容易遵循。)

+ +

当然,要加载图像,它需要在我们的代码目录中可用。从Github抓住球图像,并将其保存/img在与index.html文件相同位置目录中

+ +

现在,要在屏幕上显示,我们将使用另一种Phaser方法add.sprite()create()函数内添加以下新的代码行,如图所示:

+ +
function create() {
+    ball = game.add.sprite(50, 50, 'ball');
+}
+
+ +

这将添加球到游戏,并将其呈现在屏幕上。前两个参数是要添加的画布的x和y坐标,第三个是我们之前定义的资产的名称。就是这样 - 如果你加载你的index.html文件,你会看到已经加载并在画布上渲染的图像!

+ +

比较你的代码

+ +

您可以在下面的现场演示中查看本课程的完成代码,并使用它来更好地了解它的工作原理:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/98xrv9x5/","","400")}}

+ +

下一步

+ +

打出球很容易; 接下来我们将尝试在屏幕上移动球

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Scaling", "Games/Workflows/2D_Breakout_game_Phaser/Move the ball")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_phaser/move_the_ball/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/move_the_ball/index.html new file mode 100644 index 0000000000..331d02cfab --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/move_the_ball/index.html @@ -0,0 +1,49 @@ +--- +title: Move the ball +slug: Games/Tutorials/2D_breakout_game_Phaser/Move_the_ball +tags: + - 2D + - Beginner + - Canvas + - Games + - JavaScript + - Phaser + - Tutorial + - moving +translation_of: Games/Tutorials/2D_breakout_game_Phaser/Move_the_ball +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Load_the_assets_and_print_them_on_screen", "Games/Workflows/2D_Breakout_game_Phaser/Physics")}}

+ +
+

这是Gamedev Phaser教程 16 第4步Gamedev-Phaser-Content-Kit / demos / lesson04.html完成本课后,您可以找到源代码

+
+ +

我们在屏幕上打印了我们的蓝色球,但它什么都不做,这样做会很酷。本文介绍如何做到这一点。

+ +

在每个框架上更新球的位置

+ +

记住update()功能及其定义?其中的代码在每个框架上执行,所以它是一个完美的地方,将代码更新球的位置在屏幕上。在里面添加以下新行代码update(),如下所示:

+ +
function update() {
+    ball.x += 1;
+    ball.y += 1;
+}
+
+ +

上面的代码在每个框架上为表示画布上的球坐标的属性xy属性添加了一个。重新加载index.html,你应该看到球在屏幕上滚动。

+ +

比较你的代码

+ +

您可以在下面的现场演示中查看本课程的完成代码,并使用它来更好地了解它的工作原理:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/g1cfp0vv/","","400")}}

+ +

下一步

+ +

下一步是添加一些基本的碰撞检测,所以我们的球可以从墙壁反弹。这将需要几行代码 - 一个比我们迄今为止看到的更复杂的步骤,特别是如果我们也想添加桨和砖碰撞 - 但是幸运的是Phaser使我们比我们想要使用纯粹的方法更容易做到这一点JavaScript的。

+ +

无论如何,在我们做所有的事情之前,我们将首先介绍Phaser的物理引擎,并做一些设置工作。

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Load_the_assets_and_print_them_on_screen", "Games/Workflows/2D_Breakout_game_Phaser/Physics")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_phaser/physics/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/physics/index.html new file mode 100644 index 0000000000..4f908a9bc2 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/physics/index.html @@ -0,0 +1,99 @@ +--- +title: Physics +slug: Games/Tutorials/2D_breakout_game_Phaser/Physics +tags: + - 2D + - Beginner + - Canvas + - Games + - JavaScript + - Phaser + - Tutorial + - physics +translation_of: Games/Tutorials/2D_breakout_game_Phaser/Physics +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Move_the_ball", "Games/Workflows/2D_Breakout_game_Phaser/Bounce_off_the_walls")}}

+ +
+

这是Gamedev Phaser教程 16 第5步您可以在Gamedev-Phaser-Content-Kit / demos / lesson05.html完成本课程后找到源代码

+
+ +

为了在我们的游戏中的对象之间进行正确的碰撞检测,我们将需要物理学; 本文将向您介绍Phaser中的可用内容,以及演示典型的简单设置。

+ +

添加物理效果

+ +

Phaser与三个不同的物理引擎(Arcade Physics,P2和Ninja Physics)捆绑在一起,第四个选项Box2D可作为商业插件使用。对于像我们这样的简单游戏,我们可以使用Arcade Physics引擎。我们不需要任何重的几何计算 - 毕竟只是一个球从墙壁和砖块弹起来。

+ +

首先,让我们在游戏中初始化Arcade Physics引擎。physics.startSystem()create函数开头添加方法(使其成为函数内的第一行),如下所示:

+ +
game.physics.startSystem(Phaser.Physics.ARCADE);
+
+ +

接下来,我们需要为物理系统启用我们的球 - 默认情况下,Phaser对象物理不启用。create()函数底部添加以下行

+ +
game.physics.enable(ball, Phaser.Physics.ARCADE);
+
+ +

接下来,如果我们要在屏幕上移动我们的球,我们可以设置velocitybody再次添加以下行create()

+ +
ball.body.velocity.set(150, 150);
+
+ +

删除我们以前的更新说明

+ +

记得删除添加值的我们的老方法x,并yupdate()功能:

+ +
function update() {
+    ball.x += 1;
+    ball.y += 1;
+}
+
+ +

我们正在使用物理引擎正确处理。

+ +

最终代码检查

+ +

最新的代码应该如下所示:

+ +
var ball;
+
+function preload() {
+    game.scale.scaleMode = Phaser.ScaleManager.SHOW_ALL;
+    game.scale.pageAlignHorizontally = true;
+    game.scale.pageAlignVertically = true;
+    game.stage.backgroundColor = '#eee';
+    game.load.image('ball', 'img/ball.png');
+}
+
+function create() {
+    game.physics.startSystem(Phaser.Physics.ARCADE);
+    ball = game.add.sprite(50, 50, 'ball');
+    game.physics.enable(ball, Phaser.Physics.ARCADE);
+    ball.body.velocity.set(150, 150);
+}
+
+function update() {
+}
+
+ +

尝试重新加载index.html- 球应该在给定的方向上不断移动。目前,物理引擎的重力和摩擦力设定为零。增加重力将导致球落下,同时摩擦力最终会停止球。

+ +

物理效果趣味

+ +

你可以用物理学来做更多的事情,例如添加ball.body.gravity.y = 100;你将设置球的垂直重力。因此,它将向上发射,但是由于重力的作用而下降。

+ +

这种功能只是冰山一角 - 有各种功能和变量可以帮助您操纵物理对象。查看官方物理文档,并使用ArcadeP2物理系统查看大量示例

+ +

比较你的代码

+ +

您可以在下面的现场演示中查看本课程的完成代码,并使用它来更好地了解它的工作原理:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/bjto9nj8/","","400")}}

+ +

下一步

+ +

现在我们可以转到下一课,看看如何让球从墙上弹起

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Move_the_ball", "Games/Workflows/2D_Breakout_game_Phaser/Bounce_off_the_walls")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_phaser/player_paddle_and_controls/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/player_paddle_and_controls/index.html new file mode 100644 index 0000000000..07ee9e7ee7 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/player_paddle_and_controls/index.html @@ -0,0 +1,122 @@ +--- +title: Player paddle and controls +slug: Games/Tutorials/2D_breakout_game_Phaser/Player_paddle_and_controls +tags: + - 2D + - Beginner + - Canvas + - Games + - JavaScript + - Phaser + - Tutorial +translation_of: Games/Tutorials/2D_breakout_game_Phaser/Player_paddle_and_controls +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Bounce_off_the_walls", "Games/Workflows/2D_Breakout_game_Phaser/Game_over")}}

+ +
+

这是Gamedev Phaser教程 16 第7步您可以在Gamedev-Phaser-Content-Kit / demos / lesson07.html完成本课程后找到源代码

+
+ +

我们有球从墙上移动并弹跳,但它很快变得无聊 - 没有互动!我们需要一种介绍游戏的方法,所以在这篇文章中,我们将创建一个桨来移动并击中球。

+ +

渲染桨

+ +

从框架的角度看,桨非常类似于球 - 我们需要添加一个变量来表示它,加载相关的图像资源,然后做出魔法。

+ +

装载桨

+ +

首先,添加paddle我们将在我们的游戏中使用的ball变量,就在变量之后:

+ +
var paddle;
+
+ +

然后,在该preload功能中,paddle通过添加以下新load.image()调用来加载图像

+ +
function preload() {
+    // ...
+    game.load.image('ball', 'img/ball.png');
+    game.load.image('paddle', 'img/paddle.png');
+}
+
+ +

添加桨图形

+ +

所以我们不要忘记,在这一点上,你应该从Github 抓住这个图形,并保存在你的/img文件夹中。

+ +

渲染桨用物理引擎

+ +

接下来,我们将通过add.sprite()create()函数中添加以下调用来初始化我们的桨,将其添加到底部:

+ +
paddle = game.add.sprite(game.world.width*0.5, game.world.height-5, 'paddle');
+
+ +

我们可以使用world.widthworld.height值来将桨定位到我们想要的位置:game.world.width*0.5将在屏幕中间。在我们这个例子中,世界和画布是一样的,但是对于其他类型的游戏,例如侧滚滚,这个世界会变大,你可以修改它来创造有趣的效果。

+ +

你会注意到,如果你index.html在这一点上重新加载,那么桨是目前不完全在中间的。为什么?因为计算位置的锚总是从对象的左上角开始。我们可以改变它,使锚在桨的宽度的中间和它的高度的底部,所以更容易将其定位在底部边缘。添加以下新行以下的行:

+ +
paddle.anchor.set(0.5,1);
+
+ +

桨现在位于我们想要的地方。现在,为了使它与球碰撞,我们必须为桨提供物理效果。继续添加以下新行,再次在create()函数的底部

+ +
game.physics.enable(paddle, Phaser.Physics.ARCADE);
+
+ +

现在,魔法可以开始发生 - 该框架可以在每个框架上检查碰撞检测。要启用桨和球之间的碰撞检测,请将collide()方法添加到如下update()功能中:

+ +
function update() {
+    game.physics.arcade.collide(ball, paddle);
+}
+
+ +

第一个参数是我们感兴趣的对象之一 - 球 - 第二个是另一个,桨。这有效,但不如我们预期的那样 - 当球击中桨时,桨从屏幕上掉下来!我们想要的就是球从跳板上跳起来,而桨子停在同一个地方。我们可以将body设置成immovable球,所以当球击中它时不会移动。为此,请在create()函数底部添加以下行

+ +
paddle.body.immovable = true;
+
+ +

现在它按预期工作。

+ +

控制桨

+ +

接下来的问题是我们不能移动桨。要做到这一点,我们可以使用系统的默认输入(鼠标或触摸,取决于平台),并将桨位置设置到位置的input位置。将以下新行添加到update()函数中,如下所示:

+ +
function update() {
+    game.physics.arcade.collide(ball, paddle);
+    paddle.x = game.input.x;
+}
+
+ +

现在在每个新的一帧上,桨的x位置将根据输入的x位置进行调整,但是当我们开始游戏时,桨的位置不在中间。这是因为输入位置尚未定义。要修复,我们可以将默认位置(如果输入位置尚未定义)设置为屏幕中间。更新上一行如下:

+ +
paddle.x = game.input.x || game.world.width*0.5;
+
+ +

如果您还没有这样做,请重新加载index.html并尝试!

+ +

定位球

+ +

我们有桨按预期工作,所以我们把球放在上面。它非常类似于定位桨 - 我们需要将其放置在屏幕中间水平和垂直的底部,与底部有一点偏移。要按照我们想要的方式放置它,我们将把锚定位到球的正中间。找到现有的行,并将其替换为以下两行:ball = game.add.sprite( ... )

+ +
ball = game.add.sprite(game.world.width*0.5, game.world.height-25, 'ball');
+ball.anchor.set(0.5);
+ +

速度保持不变 - 我们只是将第二个参数的值从150改为-150,所以球将通过向上移动而不是下降来开始游戏。查找现有ball.body.velocity.set( ... )行并将其更新为以下内容:

+ +
ball.body.velocity.set(150, -150);
+
+ +

现在球将从桨的中间开始。

+ +

比较你的代码

+ +

您可以在下面的现场演示中查看本课程的完成代码,并使用它来更好地了解它的工作原理:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/ogqza0ye/","","400")}}

+ +

下一步

+ +

我们可以移动桨,并将球反弹,但是如果球从屏幕的底部边缘反弹,那又有什么意义?我们来介绍丢失的可能性 - 也称为游戏逻辑。

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Bounce_off_the_walls", "Games/Workflows/2D_Breakout_game_Phaser/Game_over")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_phaser/randomizing_gameplay/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/randomizing_gameplay/index.html new file mode 100644 index 0000000000..8642ed4eb7 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/randomizing_gameplay/index.html @@ -0,0 +1,61 @@ +--- +title: Randomizing gameplay +slug: Games/Tutorials/2D_breakout_game_Phaser/Randomizing_gameplay +tags: + - 2D + - Beginner + - Canvas + - Games + - JavaScript + - Phaser + - Tutorial +translation_of: Games/Tutorials/2D_breakout_game_Phaser/Randomizing_gameplay +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{Previous("Games/Workflows/2D_Breakout_game_Phaser/Buttons")}}

+ +
+

这是Gamedev Phaser教程 16中的16步您可以在Gamedev-Phaser-Content-Kit / demos / lesson16.html完成本课程后找到源代码

+
+ +

我们的游戏似乎已经完成了,但是如果你看起来足够近,你会发现球在整个游戏中都以相同的角度从桨上弹起。这意味着每个游戏都非常相似。为了解决这个问题,提高可玩性,我们应该使反弹角度更加随机,在本文中我们将介绍一下如何。

+ +

让篮板更随机

+ +

我们可以根据撞击桨的确切位置来改变球的速度,通过使用沿着下方的线路运行功能来修改x速度ballHitPaddle()现在添加这一行到您的代码,并尝试。

+ +
function ballHitPaddle(ball, paddle) {
+    ball.animations.play('wobble');
+    ball.body.velocity.x = -1*5*(paddle.x-ball.x);
+}
+ +

这有点魔法 - 新的速度越高,桨的中心和球撞到的地方之间的距离就越大。此外,方向(左或右)由该值确定 - 如果球击中桨的左侧,则其将向左反弹,而击球右侧将向右反弹。最终这样做是因为对某些值进行了一些实验,您可以进行自己的实验,看看会发生什么。这当然不是完全随机的,但它确实使游戏玩法变得更加不可预测,因此更有趣。

+ +

比较你的代码

+ +

您可以在下面的现场演示中查看本课程的完成代码,并使用它来更好地了解它的工作原理:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/3yds5ege/","","400")}}

+ +

概要

+ +

你已经完成了所有的课程 - 恭喜你!在这一点上,您将了解到Phaser的基础知识和简单2D游戏背后的逻辑。

+ +

练习跟随

+ +

你可以在游戏中做更多的事情 - 添加任何你觉得最好的东西,使它更有趣和有趣。Phaser提供的无数有用的方法的基本介绍。以下是关于如何扩展我们的小游戏的一些建议,让您开始:

+ + + +

一定要检查越来越多的示例列表官方文档如果您需要任何帮助,请访问HTML5 Gamedevs论坛

+ +

您也可以返回本教程系列的索引页

+ +

{{Previous("Games/Workflows/2D_Breakout_game_Phaser/Buttons")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_phaser/scaling/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/scaling/index.html new file mode 100644 index 0000000000..063f95aba9 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/scaling/index.html @@ -0,0 +1,64 @@ +--- +title: Scaling +slug: Games/Tutorials/2D_breakout_game_Phaser/Scaling +tags: + - 2D + - Beginner + - Canvas + - Games + - JavaScript + - Phaser + - Tutorial +translation_of: Games/Tutorials/2D_breakout_game_Phaser/Scaling +--- +
{{GamesSidebar}}
{{IncludeSubnav("/zh-CN/docs/Games")}}
+ +

{{PreviousNext("Games/Tutorials/2D_Breakout_game_Phaser/Initialize_the_framework", "Games/Tutorials/2D_Breakout_game_Phaser/Load_the_assets_and_print_them_on_screen")}}

+ +
+

这是Gamedev Phaser教程系列的第二课. 在课程完成之后,你可以在Gamedev-Phaser-Content-Kit/demos/lesson02.html找到源码.

+
+ +

缩放是指游戏画布如何在不同的屏幕尺寸上进行显示。我们可以在预加载阶段自动使游戏规模适合任何屏幕尺寸,之后就可以不用再担心屏幕尺寸的问题了。

+ +

Phaser中scale对象

+ +

Phaser中有一个特殊的对象:scale它包含一些特别的方法和属性。让我们来更改一下上一节中创建的的preload()函数

+ +
function preload() {
+    game.scale.scaleMode = Phaser.ScaleManager.SHOW_ALL;
+    game.scale.pageAlignHorizontally = true;
+    game.scale.pageAlignVertically = true;
+}
+
+ +

scaleMode 有几个不同的选项来指定 Canvas 应该如何缩放:

+ + + +

preload()中的其他两行代码负责水平和垂直居中画布,所以它始终以屏幕为中心,无论大小如何。

+ +

设置背景颜色

+ +

我们还可以给画布设置背景颜色来替代磨人的黑色背景。通过更改stage对象的backgroundColor属性来添加,我们可以使用CSS颜色定义语法进行设置。我们在刚才的代码下面添加以下代码:

+ +
game.stage.backgroundColor = '#eee';
+
+ +

完整的代码

+ +

您可以在下面的现场演示中查看本课程的完成代码,并使用它来更好地了解它的工作原理:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/6a64vecL/","","400")}}

+ +

下一步

+ +

现在我们设置了我们游戏的缩放比例,让我们继续第三课,并设计出如何加载资源并将其显示在屏幕上

+ +

{{PreviousNext("Games/Tutorials/2D_Breakout_game_Phaser/Initialize_the_framework", "Games/Tutorials/2D_Breakout_game_Phaser/Load_the_assets_and_print_them_on_screen")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_phaser/the_score/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/the_score/index.html new file mode 100644 index 0000000000..f71ded071f --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/the_score/index.html @@ -0,0 +1,76 @@ +--- +title: The score +slug: Games/Tutorials/2D_breakout_game_Phaser/The_score +tags: + - 2D + - Beginner + - Canvas + - Games + - JavaScript + - Phaser + - Tutorial + - scoring +translation_of: Games/Tutorials/2D_breakout_game_Phaser/The_score +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Collision_detection", "Games/Workflows/2D_Breakout_game_Phaser/Win_the_game")}}

+ +
+

这是Gamedev Phaser教程 16 第11步您可以在Gamedev-Phaser-Content-Kit / demos / lesson11.html完成本课程后找到源代码

+
+ +

得分也可以使游戏更有趣 - 你可以尝试击败自己的高分,或者你的朋友。在这篇文章中,我们将为我们的游戏添加一个评分系统。

+ +

我们将使用一个单独的变量来存储分数和Phaser的text()方法将其打印到屏幕上。

+ +

新变量

+ +

在以前定义的之后添加两个新变量:

+ +
// ...
+var scoreText;
+var score = 0;
+
+ +

将得分文字添加到游戏显示

+ +

现在在create()函数末尾添加这一行

+ +
scoreText = game.add.text(5, 5, 'Points: 0', { font: '18px Arial', fill: '#0095DD' });
+
+ +

text()方法可以采用四个参数:

+ + + +

最后一个参数与CSS样式非常相似。在我们的例子中,乐谱文字将为蓝色,大小为18像素,并使用Arial字体。

+ +

当砖块被破坏时更新分数

+ +

每当球击中砖块时,我们将增加点数,更新scoreText显示当前得分。这可以使用setText()方法 - 添加以下两行新ballHitBrick()功能:

+ +
function ballHitBrick(ball, brick) {
+    brick.kill();
+    score += 10;
+    scoreText.setText('Points: '+score);
+}
+
+ +

这是现在 - 重新加载你的,index.html并检查得分更新每个砖击。

+ +

比较你的代码

+ +

您可以在下面的现场演示中查看本课程的完成代码,并使用它来更好地了解它的工作原理:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/n8o6rhrf/","","400")}}

+ +

下一步

+ +

我们现在有一个得分系统,但是如果你不能赢得,那么玩和保持分数是多少?让我们看看我们如何能够增加胜利的状态,让我们赢得比赛

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/Collision_detection", "Games/Workflows/2D_Breakout_game_Phaser/Win_the_game")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_phaser/win_the_game/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/win_the_game/index.html new file mode 100644 index 0000000000..acdcc8fba9 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_phaser/win_the_game/index.html @@ -0,0 +1,59 @@ +--- +title: Win the game +slug: Games/Tutorials/2D_breakout_game_Phaser/Win_the_game +tags: + - 2D + - Beginner + - Canvas + - Games + - JavaScript + - Phaser + - Tutorial + - winning +translation_of: Games/Tutorials/2D_breakout_game_Phaser/Win_the_game +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/The_score", "Games/Workflows/2D_Breakout_game_Phaser/Extra_lives")}}

+ +
+

这是Gamedev Phaser教程 16 第12步您可以在Gamedev-Phaser-Content-Kit / demos / lesson12.html完成本课程后找到源代码

+
+ +

在我们的游戏中实现获胜是相当容易的:如果你碰巧摧毁所有的砖块,那么你赢了。

+ +

如何取胜?

+ +

将以下新代码添加到您的ballHitBrick()函数中:

+ +
function ballHitBrick(ball, brick) {
+    brick.kill();
+    score += 10;
+    scoreText.setText('Points: '+score);
+
+    var count_alive = 0;
+    for (i = 0; i < bricks.children.length; i++) {
+      if (bricks.children[i].alive == true) {
+        count_alive++;
+      }
+    }
+    if (count_alive == 0) {
+      alert('You won the game, congratulations!');
+      location.reload();
+    }
+}
+
+ +

我们循环使用组中的砖块bricks.children,检查每个砖块的.alive() 方法的活力如果没有更多的砖块活着,那么我们会显示一个获胜的消息,一旦警报被关闭,重新启动游戏。

+ +

比较你的代码

+ +

您可以在下面的现场演示中查看本课程的完成代码,并使用它来更好地了解它的工作原理:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/u8waa4Lx/1/","","400")}}

+ +

下一步

+ +

失败和获胜都是实施的,所以我们的游戏的核心游戏就完成了。现在让我们添加一些额外的东西-我们会给玩家将3个生活的,而不是一个。

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_Phaser/The_score", "Games/Workflows/2D_Breakout_game_Phaser/Extra_lives")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/bounce_off_the_walls/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/bounce_off_the_walls/index.html new file mode 100644 index 0000000000..afbfbc1908 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/bounce_off_the_walls/index.html @@ -0,0 +1,99 @@ +--- +title: 反弹的墙壁 +slug: Games/Tutorials/2D_Breakout_game_pure_JavaScript/Bounce_off_the_walls +translation_of: Games/Tutorials/2D_Breakout_game_pure_JavaScript/Bounce_off_the_walls +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_pure_JavaScript/Move_the_ball", "Games/Workflows/2D_Breakout_game_pure_JavaScript/Paddle_and_keyboard_controls")}}

+ +
+

本篇是 Gamedev Canvas tutorial 10节教程中的第三节。如果你完成了本篇教程之后,你可以从 Gamedev-Canvas-workshop/lesson3.html 看到源码。

+
+ +

看到我们的球动起来很惊讶吧,但是它很快就从屏幕上消失了,当然我们是可以控制它的。我们会实现一些非常简单的碰撞检测(详细后面解释),使球在画布的四周反弹回来。

+ +

简单的碰撞

+ +

我们将检查球体是否与边缘接触,如果有接触我们将相应的改变它的运动方向。

+ +

为了运算方便我们定义一个名为 ballRadius 的变量,来存储球的半径。向代码中添加一下内容:

+ +
var ballRadius = 10;
+ +

现在更新绘制球的 drawBall() 函数:

+ +
ctx.arc(x, y, ballRadius, 0, Math.PI*2);
+ +

从顶部和底部弹起

+ +

有四面墙壁可以让它反弹回来,我们先来看上面的那面墙。我们需要判断球运动的每一帧,球体是否与画布的顶部边缘接触。如果有接触,我们将会改变球体的运动方向,使它向相反的方向移动,并保证它在画布的可见范围之内。记住坐标系统的左上角,让我们开始并加以下代码:

+ +
if(y + dy < 0) {
+    dy = -dy;
+}
+ +

如果球的纵坐标(y轴)值小于零,我们将在球体原有的运动方向上逆转。如果球体向上移动的速度是2像素/帧,现在就是向上移动速度是-2像素。这相当于此时向下移动的速度是2像素/帧。

+ +

上面的代码将处理球与画布顶部边缘的反射,现在让我们思考一下底部边缘如何处理:

+ +
if(y + dy > canvas.height) {
+    dy = -dy;
+}
+ +

如果球的y位置大于canvas的高度(记住,我们从左上角计算y值,所以顶部边缘从0开始,底部边缘在480像素),然后通过像以前那样反转y轴运动而离开底部边缘。

+ +

我们可以将这两句冗长的代码合二为一:

+ +
if(y + dy > canvas.height || y + dy < 0) {
+    dy = -dy;
+}
+ +

如果其中一个判断为 true, 则反转球的运动。

+ +

从左边和右边反弹

+ +

我们有顶部和底部的边缘,所以我们来考虑一下左边和右边的边缘。 实际上非常相似,你所要做的就是颠倒x而不是y:

+ +
if(x + dx > canvas.width || x + dx < 0) {
+    dx = -dx;
+}
+
+if(y + dy > canvas.height || y + dy < 0) {
+    dy = -dy;
+}
+ +

你应该把上面的代码块插入到draw()函数中,就在大括号之前。

+ +

球部分消失在墙上!

+ +

测试你的代码,你会看到我们的球碰到任一边缘都会反弹!然而,我们还发现了一个问题,当球碰撞到边缘,反弹之前:

+ +

+ +

这是因为我们正在计算墙和球的中心碰撞点,而我们应该围绕它的周长来做。 如果碰到墙壁,球应该会弹起来,而不是陷入墙壁一半时,所以让我们来调整一下我们的判断条件。 更新你之前添加的代码:

+ +
if(x + dx > canvas.width-ballRadius || x + dx < ballRadius) {
+    dx = -dx;
+}
+if(y + dy > canvas.height-ballRadius || y + dy < ballRadius) {
+    dy = -dy;
+}
+ +

当球的中心到墙的边缘之间的距离与球的半径完全相同时,它将改变运动的方向。

+ +

比较你的代码

+ +

让我们再次检查这个部分的代码与你之间有何差异:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/redj37dc/","","370")}}

+ +
+

练习: 尝试修改你的代码,在每次碰到墙壁时都要把球的颜色改成随机的颜色。

+
+ +

下一步

+ +

现在我们已经到了我们的球正在移动和留在游戏板上的阶段。 在第四章中,我们将看看如何实现一个可控制的paddle - 参见paddle和键盘控制

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_pure_JavaScript/Move_the_ball", "Games/Workflows/2D_Breakout_game_pure_JavaScript/Paddle_and_keyboard_controls")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/build_the_brick_field/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/build_the_brick_field/index.html new file mode 100644 index 0000000000..e91072bc68 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/build_the_brick_field/index.html @@ -0,0 +1,106 @@ +--- +title: Build the brick field +slug: Games/Tutorials/2D_Breakout_game_pure_JavaScript/Build_the_brick_field +translation_of: Games/Tutorials/2D_Breakout_game_pure_JavaScript/Build_the_brick_field +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_pure_JavaScript/Game_over", "Games/Workflows/2D_Breakout_game_pure_JavaScript/Collision_detection")}}

+ +
+

这是 Gamedev Canvas tutorial教程10节的第6节。您可以在完成本课程后在这里Gamedev-Canvas-workshop/lesson6.html找到源代码。

+
+ +

在修改游戏机制后,我们可以输了 — 这样这游戏看起来终于像是一个游戏了,这真是太好了。但是,如果你总是让球与墙、板碰撞的话,很快就会感到无聊的。 好游戏需要的是让球消灭砖,这就是我们即将要做的!

+ +

设置砖变量

+ +

本课题的总体目标是使用一个二维数组嵌套的循环,给出砖的几行代码。首先我们需要设置一些变量定义的砖,如宽度和高度信息, 行和列,等。在之前的变量声明处加入以下几行代码。

+ +
var brickRowCount = 3;
+var brickColumnCount = 5;
+var brickWidth = 75;
+var brickHeight = 20;
+var brickPadding = 10;
+var brickOffsetTop = 30;
+var brickOffsetLeft = 30;
+ +

在这里,我们定义了砖的行数和列,宽度和高度,砖块之间的填充物,这样它们就不会互相接触;有一个上、左偏移量,所以它们不会从画布的边缘开始绘制。 

+ +

我们将在一个二维数组容纳我们所有的砖。它将包含砖列(c),砖行(R),每一个包含一个对象,其中包含x和y位置,让每个砖显示在屏幕上。在变量下面添加以下代码:

+ +
var bricks = [];
+for(c=0; c<brickColumnCount; c++) {
+    bricks[c] = [];
+    for(r=0; r<brickRowCount; r++) {
+        bricks[c][r] = { x: 0, y: 0 };
+    }
+}
+ +

上面的代码将通过行和列的循环和创造新砖。注意,砖块对象稍后也将用于碰撞检测。

+ +

画砖的逻辑

+ +

现在让我们创建一个函数来遍历数组中的所有砖块并在屏幕上绘制它们。. 代码如下:

+ +
function drawBricks() {
+    for(c=0; c<brickColumnCount; c++) {
+        for(r=0; r<brickRowCount; r++) {
+            bricks[c][r].x = 0;
+            bricks[c][r].y = 0;
+            ctx.beginPath();
+            ctx.rect(0, 0, brickWidth, brickHeight);
+            ctx.fillStyle = "#0095DD";
+            ctx.fill();
+            ctx.closePath();
+        }
+    }
+}
+ +

再次,我们遍历的行和列,给每一块砖的位置设置XY,我们也画布上画砖,---brickwidth X brickheight 。问题是我们都画在一个地方坐标(0,0)处。我们需要做的是增加一些计算,计算每个循环迭代后的砖块的x和y位置:

+ +
var brickX = (c*(brickWidth+brickPadding))+brickOffsetLeft;
+var brickY = (r*(brickHeight+brickPadding))+brickOffsetTop;
+ +

每个brickX位置是 brickWidth + brickPadding,乘以列数C,再加上brickOffsetLeft;对于砖brickY的逻辑相同,除了名称不同,使用行数RbrickHeight,和brickOffsetTop。现在,每一块砖都可以放在正确的地方,排成一排,每一块砖之间都有填充物,从左上角和顶部的帆布边缘偏移。

+ +

在设置brickXbrickY作为对应砖的坐标之后,形成了 drawBricks()函数的最终版本。将以下代码加在drawPaddle()函数后面:

+ +
function drawBricks() {
+    for(c=0; c<brickColumnCount; c++) {
+        for(r=0; r<brickRowCount; r++) {
+            var brickX = (c*(brickWidth+brickPadding))+brickOffsetLeft;
+            var brickY = (r*(brickHeight+brickPadding))+brickOffsetTop;
+            bricks[c][r].x = brickX;
+            bricks[c][r].y = brickY;
+            ctx.beginPath();
+            ctx.rect(brickX, brickY, brickWidth, brickHeight);
+            ctx.fillStyle = "#0095DD";
+            ctx.fill();
+            ctx.closePath();
+        }
+    }
+}
+ +

到了展现真正画砖的时候了

+ +

最后一件事就是在draw()中调用drawBricks(), 位置最好在函数开始处,在清除画布和画球之间。直接将下面代码加在drawBall() 处:

+ +
drawBricks();
+
+ +

 比较你的代码

+ +

这样,游戏变得更有趣了 :

+ +

{{JSFiddleEmbed("https://jsfiddle.net/kundan333/myd4vbwg/2/","","320")}}

+ +
+

练习:尝试在行或列上改变砖块数量,或者它们的位置。

+
+ +

下一节

+ +

现在,我们有砖啦!但是球根本就没有和它们互动 —— 接下来的第七章我们将让球和砖产生碰撞: 碰撞检测

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_pure_JavaScript/Game_over", "Games/Workflows/2D_Breakout_game_pure_JavaScript/Collision_detection")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/collision_detection/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/collision_detection/index.html new file mode 100644 index 0000000000..fe3769ee79 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/collision_detection/index.html @@ -0,0 +1,126 @@ +--- +title: 撞击处理 +slug: Games/Tutorials/2D_Breakout_game_pure_JavaScript/Collision_detection +translation_of: Games/Tutorials/2D_Breakout_game_pure_JavaScript/Collision_detection +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_pure_JavaScript/Build_the_brick_field", "Games/Workflows/2D_Breakout_game_pure_JavaScript/Track_the_score_and_win")}}

+ +
+

本篇为Gamedev Canvas tutorial10节教程中的第7节。在你完成这篇课程之后,你可以在Gamedev-Canvas-workshop/lesson7.html.找到我们的源代码。

+
+ +

我们已经在屏幕上画出了砖块,但游戏仍然没有那么有趣,因为球通过它们。我们需要考虑增加碰撞检测,这样球就可以弹击砖块并打破它们。

+ +

当然,这是我们的决定如何实现的,但是计算球是否触及矩形是很困难的,因为在画布中没有辅助函数。为了这个教程,我们将尽可能地做到这一点。我们将检查球的中心是否与任何给定的砖块碰撞。这不会每次都给出一个完美的结果,而且有很多更复杂的方法来进行碰撞检测,但是这对指导你的基本概念很有效。

+ +

撞击侦测函数

+ +

踢掉这一切,我们想创建一个碰撞检测功能,将循环通过所有砖块,并比较每一个砖的位置与球的坐标,因为每个帧绘制。为了更好地理解代码,我们将定义用于在碰撞检测的每个循环中存储砖块对象的B变量:

+ +
function collisionDetection() {
+    for(c=0; c<brickColumnCount; c++) {
+        for(r=0; r<brickRowCount; r++) {
+            var b = bricks[c][r];
+            // calculations
+        }
+    }
+}
+ +

如果球的中心在我们的一块砖块的坐标内,我们将改变球的方向。对于球的中心在砖块内,以下四个陈述都必须是正确的:

+ + + +

让我们在代码中写下:

+ +
function collisionDetection() {
+    for(c=0; c<brickColumnCount; c++) {
+        for(r=0; r<brickRowCount; r++) {
+            var b = bricks[c][r];
+            if(x > b.x && x < b.x+brickWidth && y > b.y && y < b.y+brickHeight) {
+                dy = -dy;
+            }
+        }
+    }
+}
+ +

将上面的块添加到代码中,在 keyUpHandler() 函数下面。

+ +

让砖块在被撞击之后消失

+ +

上述代码将按需要工作,球改变其方向。问题是砖块留在原地。我们必须想出一个办法来摆脱那些我们已经用球打中的砖。我们可以通过添加一个额外的参数来指示我们是否想在屏幕上画每个砖块。在初始化砖块的代码的一部分中,让我们为每个砖块对象添加一个状态属性。更新代码的下面部分,如突出显示的行所示:

+ +
var bricks = [];
+for(c=0; c<brickColumnCount; c++) {
+    bricks[c] = [];
+    for(r=0; r<brickRowCount; r++) {
+        bricks[c][r] = { x: 0, y: 0, status: 1 };
+    }
+}
+ +

接下来,我们将在绘制之前在 drawBricks() 中检查每个砖块的 status属性的值-如果 status是 1,然后画它,但是如果它是 0,那么它被球击中,我们不再希望它在屏幕上。更新您的 drawBricks() 函数如下:

+ +
function drawBricks() {
+    for(c=0; c<brickColumnCount; c++) {
+        for(r=0; r<brickRowCount; r++) {
+            if(bricks[c][r].status == 1) {
+                var brickX = (c*(brickWidth+brickPadding))+brickOffsetLeft;
+                var brickY = (r*(brickHeight+brickPadding))+brickOffsetTop;
+                bricks[c][r].x = brickX;
+                bricks[c][r].y = brickY;
+                ctx.beginPath();
+                ctx.rect(brickX, brickY, brickWidth, brickHeight);
+                ctx.fillStyle = "#0095DD";
+                ctx.fill();
+                ctx.closePath();
+            }
+        }
+    }
+}
+ +

跟踪并更新在撞击侦测函数中的状态

+ +

现在我们需要将砖块 status 属性包含在 collisionDetection()函数中:如果砖块是活动的(它的状态是 1),我们将检查碰撞是否发生;如果发生碰撞,我们将给定砖块的状态设置为 0,这样它就不会被绘制在屏幕上。更新您的 collisionDetection() 函数,如下所示:

+ +
function collisionDetection() {
+    for(c=0; c<brickColumnCount; c++) {
+        for(r=0; r<brickRowCount; r++) {
+            var b = bricks[c][r];
+            if(b.status == 1) {
+                if(x > b.x && x < b.x+brickWidth && y > b.y && y < b.y+brickHeight) {
+                    dy = -dy;
+                    b.status = 0;
+                }
+            }
+        }
+    }
+}
+ +

调用我们的撞击侦测函数

+ +

最后要做的是向 collisionDetection() 函数添加一个调用到我们的主要draw() 函数。将下面的行添加到 draw() 函数中,就在drawPaddle() 调用的下面:

+ +
collisionDetection();
+
+ +

对比你的代码

+ +

现在,在每一帧,每一块砖上都检查了球的碰撞检测。现在我们可以毁掉砖头:-!

+ +

{{JSFiddleEmbed("https://jsfiddle.net/kundan333/myd4vbwg/5/","","320")}}

+ +
+

练习:当球碰到砖头时,改变球的颜色。

+
+ +

下一节

+ +

我们现在肯定到了,继续前进吧!在第八章中,我们将探讨如何跟踪得分和获胜。Track the score and win.

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_pure_JavaScript/Build_the_brick_field", "Games/Workflows/2D_Breakout_game_pure_JavaScript/Track_the_score_and_win")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/create_the_canvas_and_draw_on_it/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/create_the_canvas_and_draw_on_it/index.html new file mode 100644 index 0000000000..59b0b70392 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/create_the_canvas_and_draw_on_it/index.html @@ -0,0 +1,114 @@ +--- +title: 创建、绘制画布 +slug: >- + Games/Tutorials/2D_Breakout_game_pure_JavaScript/Create_the_Canvas_and_draw_on_it +translation_of: >- + Games/Tutorials/2D_Breakout_game_pure_JavaScript/Create_the_Canvas_and_draw_on_it +--- +
{{GamesSidebar}}
+ +
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_pure_JavaScript", "Games/Workflows/2D_Breakout_game_pure_JavaScript/Move_the_ball")}}

+ +
+

本篇是 Gamedev Canvas tutorial 10节教程中的第一节。如果你完成了本篇教程之后,你可以从 Gamedev-Canvas-workshop/lesson1.html 看到源码。

+
+ +

在我们开始编写游戏功能之前,我们可以通过HTML的canvas标签创建支撑游戏的基本结构。

+ +

页面部分

+ +

HTML文档的结构是非常简单的,我们的游戏将完全呈现在这个HTML的canvas标签中。你可以选择一款你最喜欢的文本编辑器,创建一个HTML文件,保存到你理想的位置,名称为index.html。并添加以下代码:

+ +
<!DOCTYPE html>
+<html>
+<head>
+    <meta charset="utf-8" />
+    <title>Gamedev Canvas Workshop</title>
+    <style>
+    	* { padding: 0; margin: 0; }
+    	canvas { background: #eee; display: block; margin: 0 auto; }
+    </style>
+</head>
+<body>
+
+<canvas id="myCanvas" width="480" height="320"></canvas>
+
+<script>
+	// JavaScript code goes here
+</script>
+
+</body>
+</html>
+
+ +

head 标签中我们需要定义字符集(charset),标题(title)和一些基本的样式。在 body 标签中包含 canvas 标签和 javascript 标签,我们将在 javascript 标签中使用 JavaScript 代码来控制 canvas 标签中的内容展现。在 canvas 标签有一个名为 myCanvas 的 Id,根据这个属性我们可以很容易的获取到这个元素并设置他的宽为 480 像素,高为 320 像素。 之后我们会将所有的 JavaScript 代码全部写到 javascript 标签中(<script>...</script>)。

+ +

画布的基本属性

+ +

要想在 canvas 标签中呈现图像内容,我们必须在 JavaScript 代码中获取到这个元素。在 javascript 标签中添加以下代码;

+ +
var canvas = document.getElementById("myCanvas");
+var ctx = canvas.getContext("2d");
+ +

在这里我们可以声明一个变量保存 canvas 标签,然后我们需要创建变量 ctx, 这个变量可用来存储 2d 的渲染上下文,我们实际上就用它在 canvas 标签上绘制内容。 

+ +

让我们来看一个例子,打印在画布上的红色正方形。添加以下代码到你的 JavaScript 里,然后在浏览器中打开 index.html 来看看效果。

+ +
ctx.beginPath();
+ctx.rect(20, 40, 50, 50);
+ctx.fillStyle = "#FF0000";
+ctx.fill();
+ctx.closePath();
+ +

把全部指令放到 ctx.beginPath() 方法 和 ctx.closePath() 方法之间。我们使用 ctx.rect() 方法创建了一个矩形。这个方法的第一组参数(20,40)是确定这个矩形的左上角在画布上的坐标,第二组参数(50, 50)是指定矩形的宽度和高度。例子中矩形左上角的 x 坐标为20像素,y 坐标为40像素;宽和高各为50像素,这样画出一个完美的正方形。ctx.fillStyle 属性是用于填充绘画的颜色,这里填充的颜色为红色。ctx.fill() 方法是填充路径的内容区域生成实心的图形。

+ +

我们不局限于绘制矩形 — 下面这段代码会绘制出一个绿色的圆;尝试添加到 JavaScript 代码的底部,保存并刷新:

+ +
ctx.beginPath();
+ctx.arc(240, 160, 20, 0, Math.PI*2, false);
+ctx.fillStyle = "green";
+ctx.fill();
+ctx.closePath();
+ +

正如你所见,我们再一次使用了 ctx.beginPath() 方法 和 ctx.closePath() 方法;在这两个方法中间,最重要的部分是 ctx.arc() 方法。它需要六个参数:

+ + + +

ctx.fillStyle 属性的值看起来和之前不大一样。这是因为它与 CSS 一样,颜色可以指定为十六进制值也可以指定为 rgba() 函数,或者任何其他可用的颜色值函数。

+ +

当然我们也可以不使用 ctx.fill() 函数来填充图形和颜色,用 ctx.stroke() 函数来绘制定义图形的路径。试着将这段代码添加到您的JavaScript:

+ +
ctx.beginPath();
+ctx.rect(160, 10, 100, 40);
+ctx.strokeStyle = "rgba(0, 0, 255, 0.5)";
+ctx.stroke();
+ctx.closePath();
+ +

上面的代码绘制出一个蓝色边框的空心矩形。由于 rgba()  函数的特性,边框为蓝色半透明的状态。

+ +

比对你的代码

+ +

以上是第一节的全部代码,可以运行在JSFiddle(在线JS代码调试工具)上:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/x62h15e2/","","370")}}

+ +
+

练习:尝试改变给定几何图形的大小和颜色。

+
+ +

下一节

+ +

现在我们已经创建了基本的HTML和关于画布的基本知识。我们继续第二节,如何让球在游戏中动起来 — 让球动起来

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_pure_JavaScript", "Games/Workflows/2D_Breakout_game_pure_JavaScript/Move_the_ball")}}

+ +

+ +

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/game_over/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/game_over/index.html new file mode 100644 index 0000000000..5256e9b632 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/game_over/index.html @@ -0,0 +1,81 @@ +--- +title: 游戏结束 +slug: Games/Tutorials/2D_Breakout_game_pure_JavaScript/Game_over +translation_of: Games/Tutorials/2D_Breakout_game_pure_JavaScript/Game_over +--- +
{{GamesSidebar}}
+ +
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_pure_JavaScript/Paddle_and_keyboard_controls", "Games/Workflows/2D_Breakout_game_pure_JavaScript/Build_the_brick_field")}}

+ +
+

这是 Gamedev Canvas tutorial教程的第五章。您可以在完成本课程后在这里Gamedev-Canvas-workshop/lesson5.html找到源代码。

+
+ +

看球从墙上反弹,并能够移动球盘是很有趣的。但除此之外,游戏什么都不做,也没有任何进展或最终目标。 从游戏的角度来看,我们需要一个game over。 失败的逻辑很简单。 如果你的球拍错过了球,并且球到达屏幕的底部边缘,那么游戏就结束了。

+ +

实现游戏失败

+ +

让我们在代码中实现,下面是第三章里的一段代码,让球从墙上反弹:

+ +
if(x + dx > canvas.width-ballRadius || x + dx < ballRadius) {
+    dx = -dx;
+}
+
+if(y + dy > canvas.height-ballRadius || y + dy < ballRadius) {
+    dy = -dy;
+}
+ +

我们不需要让球从四面墙上反弹,应该只允许三个 - 左,上,右。 击中底部墙将结束游戏。 我们将编辑第二个if代码块,这是一个if else块,当球碰撞到画布的底部边缘时,它会触发我们的“游戏结束”状态。 现在我们将保持简单,显示一条警告消息,并通过重新加载页面重新开始游戏。

+ +

第一步,把您最开始使用的 setInterval() 函数

+ +
setInterval(draw, 10);
+ +

替换成:

+ +
var interval = setInterval(draw, 10);
+ +

然后将第二个if块替换为以下内容:

+ +
if(y + dy < ballRadius) {
+    dy = -dy;
+} else if(y + dy > canvas.height-ballRadius) {
+    alert("GAME OVER");
+    document.location.reload();
+}
+ +

让球拍接住球

+ +

本课中最后要做的是在球和球拍之间创建一些碰撞检测,以便它可以反弹并返回到游戏区域。 最简单的方法是检查球的中心是否在球拍的左边和右边之间。 再次更新您修改的代码的最后一位(第二个if块),如下所示:

+ +
if(y + dy < ballRadius) {
+    dy = -dy;
+} else if(y + dy > canvas.height-ballRadius) {
+    if(x > paddleX && x < paddleX + paddleWidth) {
+        dy = -dy;
+    }
+    else {
+        alert("GAME OVER");
+        document.location.reload();
+    }
+}
+ +

如果球击中画布的底部边缘,我们需要检查它是否碰到球拍。 如果是的话,就像你所期望的那样反弹。 如果没有,那么游戏就像以前一样结束。

+ +

代码对比

+ +

这里是完整例子,对比一下代码吧!

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/z4zy79fo/","","320")}}

+ +
+

练习: 当球碰到球拍时,让球移动得更快

+
+ +

下一步

+ +

到目前为止,我们的表现相当不错,游戏变得更有趣,并且现在你可以输了! 但它仍然缺少一些东西。 让我们继续前进到第六章 - 建造砖块 - 并创造一些砖块来消灭它们。

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_pure_JavaScript/Paddle_and_keyboard_controls", "Games/Workflows/2D_Breakout_game_pure_JavaScript/Build_the_brick_field")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/index.html new file mode 100644 index 0000000000..68a7f25df9 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/index.html @@ -0,0 +1,60 @@ +--- +title: 2D breakout game using pure JavaScript +slug: Games/Tutorials/2D_Breakout_game_pure_JavaScript +tags: + - 2D + - Beginner + - Canvas + - Games + - JavaScript + - NeedsTranslation + - TopicStub + - Tutorial +translation_of: Games/Tutorials/2D_Breakout_game_pure_JavaScript +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{Next("Games/Workflows/2D_Breakout_game_pure_JavaScript/Create_the_Canvas_and_draw_on_it")}}

+ +

在这里我们将完全使用JavaScript语言基于HTML5的canvas标签,一步一步的绘制一个简单的MDN消除游戏。

+ +

过程中的每一步都会有例子可供体验,让你更清晰的了解他的制作过程。你将学习到如何使用canvas标签的基本语法去实现简单游戏的渲染、动画、碰撞、控制胜负。

+ +

为了更快速高效的学习本系列教程,你需要掌握 JavaScript 的一些基础知识。学习完本教程之后你就可以创建自己的网页小游戏了。

+ +

Gameplay screen from the game MDN Breakout where you can use your paddle to bounce the ball and destroy the brick field, with keeping the score and lives.

+ +

教程详情

+ +

 

+ +

全部教程 — MDN 消除小游戏 的各个版本我们正一起管理并托管到 GitHub 上:

+ +
    +
  1. 创建、绘制画布
  2. +
  3. 让球动起来
  4. +
  5. 反弹的墙
  6. +
  7. 键盘操作
  8. +
  9. 游戏结束
  10. +
  11. 创建砖块
  12. +
  13. 撞击处理
  14. +
  15. 统计得分、获得胜利
  16. +
  17. 鼠标控制
  18. +
  19. 完成
  20. +
+ +

对于实现一个网页游戏而言,最好的方式是从纯 JavaScirpt 着手,那样可以让我们建立更坚实的基础。之后你可以在你的项目中选择你喜欢的框架。框架也只是用JavaScript语言实现的工具;如果你想要在项目中使用框架,你必须先了解语言本身。框架可以帮你提高开发效率并生成一些基础的内容;但是如果没有达到你的预期,你只能慢慢调试或者使用原生JavaScript去实现解决方案。 

+ +
+

说明: 如果你对使用第三方游戏资源库开发2d网页游戏感兴趣,可以参考本系列教程的 2D breakout game using Phaser.

+
+ +
+

说明: 本系列教程可以用作游戏开发工作室的素材资源。如果你想探讨普通的游戏开发,你可以利用Gamedev Canvas Content Kit以及本教程的内容。

+
+ +

下一步

+ +

好,让我们开始吧。第一步 — 创建、绘制画布

+ +

{{Next("Games/Workflows/2D_Breakout_game_pure_JavaScript/Create_the_Canvas_and_draw_on_it")}} 

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/move_the_ball/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/move_the_ball/index.html new file mode 100644 index 0000000000..e035a28838 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/move_the_ball/index.html @@ -0,0 +1,143 @@ +--- +title: 让球动起来 +slug: Games/Tutorials/2D_Breakout_game_pure_JavaScript/Move_the_ball +tags: + - 2D + - Beginner + - Canvas + - Games + - JavaScript + - Loop + - Tutorial + - movement +translation_of: Games/Tutorials/2D_Breakout_game_pure_JavaScript/Move_the_ball +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +
{{PreviousNext("Games/Workflows/2D_Breakout_game_pure_JavaScript/Create_the_Canvas_and_draw_on_it", "Games/Workflows/2D_Breakout_game_pure_JavaScript/Bounce_off_the_walls")}}
+ +
+

本篇是 Gamedev Canvas tutorial 10节教程中的第二节。如果你完成了本篇教程之后,你可以从 Gamedev-Canvas-workshop/lesson2.html 看到源码。

+
+ +

从上一节中你已经知道如何去绘制一个球。现在让我们使它动起来。从技术上讲,我们将在画布上绘制一个球,之后让它消失,然后在一个稍微不用的位置上再绘制一个一样的球。就想电影里的每一帧动起来的感觉。

+ +

我们需要定义一个绘图函数,每次使用一组不同的变量改变球体的位置;循环调用以保持画布上每一帧不断更新。你可以使用JavaScript时间函数 {{domxref("WindowTimers.setInterval()", "setInterval()")}} 或者 {{domxref("window.requestAnimationFrame()")}}。 

+ +

在你的HTML文件只保留前两行,删除其他所有的JavaScript代码并在 draw() 函数中添加以下内容保证每10毫秒执行一次 draw() 函数:

+ +
function draw() {
+    // drawing code
+}
+setInterval(draw, 10);
+ +

得益于 setInterval 的无限性,使得 draw() 函数将每10毫秒就会被调用,除非我们停止它。现在,我们来绘制小球吧,在 draw() 函数中添加以下内容:

+ +
ctx.beginPath();
+ctx.arc(50, 50, 10, 0, Math.PI*2);
+ctx.fillStyle = "#0095DD";
+ctx.fill();
+ctx.closePath();
+
+ +

现在,尝试更新你的代码,球会在每一帧画面被绘制

+ +

让球动起来

+ +

你不会注意到球正在不停地被重新刷,因为它没有移动。让我们改变这种情况。首先,我们不再使用固定位置(50,50),而是用x和y的变量来定义画布底部的起始点,然后使用这些变量来定义圆被绘制的位置。

+ +

首先,在draw()函数上方添加以下两行,以定义xy

+ +
var x = canvas.width/2;
+var y = canvas.height-30;
+
+ +

接下来更新 draw() 函数,在 arc() 方法中使用 xy 变量,如下面高亮行所示:

+ +
function draw() {
+    ctx.beginPath();
+    ctx.arc(x, y, 10, 0, Math.PI*2);
+    ctx.fillStyle = "#0095DD";
+    ctx.fill();
+    ctx.closePath();
+}
+
+ +

现在到了最重要的部分:我们想要在每一帧都被绘制出来之后,给 xy 添加一个较小的值,让它看起来像是在移动。让我们将这些值定义为 dxdy,并将它们的值分别设为 2-2。在你的 x 和 y 变量声明下方添加以下内容:

+ +
var dx = 2;
+var dy = -2;
+
+ +

最后要做的是在每一帧上更新 xy,在每一次更新中,把球画在新的位置上。将下面的两条新线添加到你的 draw() 函数:

+ +
function draw() {
+    ctx.beginPath();
+    ctx.arc(x, y, 10, 0, Math.PI*2);
+    ctx.fillStyle = "#0095DD";
+    ctx.fill();
+    ctx.closePath();
+    x += dx;
+    y += dy;
+}
+ +

再次保存代码,并在浏览器中尝试。很好,尽管看起来球在后面留下了痕迹:

+ +

+ +

在每一帧更新之前清空画布

+ +

球移动时留下了轨迹,因为我们在每一帧上都画了一个新的圆,而没有去掉之前的一个圆。不要担心,因为有一个方法来清空画布的内容:clearRect()。该方法有四个参数:矩形左上角的 x 和 y 坐标,以及矩形的右下角的 xy 坐标。这个矩形覆盖的整个区域里,之前所画的任何内容将被清除。

+ +

将下列高亮显示行添加到 draw() 函数:

+ +
function draw() {
+    ctx.clearRect(0, 0, canvas.width, canvas.height);
+    ctx.beginPath();
+    ctx.arc(x, y, 10, 0, Math.PI*2);
+    ctx.fillStyle = "#0095DD";
+    ctx.fill();
+    ctx.closePath();
+    x += dx;
+    y += dy;
+}
+
+ +

保存您的代码并再次尝试,这次你将看到球移动后没有留下轨迹。每隔10毫秒,画布就会被清除,蓝色的圆圈(我们的球)将被绘制在一个给定的位置上,而 xy 的值将在下一个帧被更新。

+ +

保持代码整洁

+ +

在接下来的几篇文章中,我们将在 draw() 函数中添加越来越多的命令,因此尽可能保持简单和整洁是很好的。让我们从把绘制球的代码移至一个单独的函数。

+ +

用以下两个函数替换现有的 draw() 函数:

+ +
function drawBall() {
+    ctx.beginPath();
+    ctx.arc(x, y, 10, 0, Math.PI*2);
+    ctx.fillStyle = "#0095DD";
+    ctx.fill();
+    ctx.closePath();
+}
+
+function draw() {
+    ctx.clearRect(0, 0, canvas.width, canvas.height);
+    drawBall();
+    x += dx;
+    y += dy;
+}
+ +

比较你的代码

+ +

您可以在下面的实时演示中查看本文的代码,并使用它来更好地了解其工作原理:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/3x5foxb1/","","415")}}

+ +
+

练习:尝试改变移动球的速度,或者移动球的方向。

+
+ +

下一步

+ +

我们已经画了我们的球,并将其移动,但它仍然消失在画布的边缘。在第三章中,我们将探讨如何使其 从墙壁上反弹.

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_pure_JavaScript/Create_the_Canvas_and_draw_on_it", "Games/Workflows/2D_Breakout_game_pure_JavaScript/Bounce_off_the_walls")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/paddle_and_keyboard_controls/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/paddle_and_keyboard_controls/index.html new file mode 100644 index 0000000000..6cd024799d --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/paddle_and_keyboard_controls/index.html @@ -0,0 +1,122 @@ +--- +title: 球板及键盘控制 +slug: Games/Tutorials/2D_Breakout_game_pure_JavaScript/Paddle_and_keyboard_controls +translation_of: Games/Tutorials/2D_Breakout_game_pure_JavaScript/Paddle_and_keyboard_controls +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_pure_JavaScript/Bounce_off_the_walls", "Games/Workflows/2D_Breakout_game_pure_JavaScript/Game_over")}}

+ +
+

这是Gamedev Canvas tutorial中的第四章。完成本课程后,你可以在Gamedev-Canvas-workshop/lesson4.html找到源码.

+
+ +

你可以看到球自由的、无限次的在墙壁上反弹,但是没有和我们发生任何交互。如果我们没有对它的控制操作,这仍然不是一个游戏。下面,我们新增一些用户操作:一个可以控制球的球板。

+ +

定义一个球板去接球

+ +

我们需要添加一个球板去接球:为此需要先定义一些变量。在你的代码的顶部的其它变量下方添加下列代码:

+ +
var paddleHeight = 10;
+var paddleWidth = 75;
+var paddleX = (canvas.width-paddleWidth)/2;
+ +

然后定义球拍的长和宽,以及为了之后的处理同时定义x轴上的初始位置。新建一个方法来在页面上描绘球板。把下列代码添加到你的drawBall()方法里去

+ +
function drawPaddle() {
+    ctx.beginPath();
+    ctx.rect(paddleX, canvas.height-paddleHeight, paddleWidth, paddleHeight);
+    ctx.fillStyle = "#0095DD";
+    ctx.fill();
+    ctx.closePath();
+}
+ +

允许用户控制球板

+ +

我们可以如愿的描绘出球板,也需让它听从用户的控制。是时候实现用键盘控制它了。我们需要:

+ + + +

按键可以使用boolean变量来初始定义。在你的其它变量附近添加下列代码:

+ +
var rightPressed = false;
+var leftPressed = false;
+ +

这两个变量的默认值都是false,因为在开始时按键没有被按下。为了监听按键的按下动作,我们需要添加两个监听器。把下列代码添加到底部的setInterval()的上一列去:

+ +
document.addEventListener("keydown", keyDownHandler, false);
+document.addEventListener("keyup", keyUpHandler, false);
+ +

当你按下任何键盘上的按键,按下事件被激活时keyDownHandler()方法会被调用。对于松开时的处理也是类似的:当松开按键时keyUpHandler()方法会被调用。把下列代码添加到addEventListener()下方

+ +
function keyDownHandler(e) {
+    if(e.keyCode == 39) {
+        rightPressed = true;
+    }
+    else if(e.keyCode == 37) {
+        leftPressed = true;
+    }
+}
+
+function keyUpHandler(e) {
+    if(e.keyCode == 39) {
+        rightPressed = false;
+    }
+    else if(e.keyCode == 37) {
+        leftPressed = false;
+    }
+}
+ +

当按下一个按键,这个信息会被储存在一个变量中。每种情况下的相关变量都设置为true。 当松开按键时,对应变量被设置回false

+ +

两个函数都以一个事件作为参数,由e(event)变量表示。 从这里你可以得到有用的信息:keyCode属性是被按下的键的信息。 例如,keyCode为37是左箭头键,而39是右箭头键。 如果按下左键,那么leftPressed变量设置为true,当松开时,leftPressed变量设置为false。右键同理。

+ +

球拍移动逻辑

+ +

我们现在有用于存储按键,事件监听器和相关功能的信息的变量。 现在我们将看到实际的代码来使用这些变量,并在屏幕上移动球拍。 在draw()函数内部,我们将检查每一帧被渲染的同时是否按下左或右键。 我们的代码如下:

+ +
if(rightPressed) {
+    paddleX += 7;
+}
+else if(leftPressed) {
+    paddleX -= 7;
+}
+ +

如果按一下左键,球拍将向左移动7个像素,如果按一下右键,球拍将向右移动7个像素。 目前这个功能可以正常工作,但是如果我们按任意一个键的时间太长,球拍就会从画布的边缘消失。 我们可以通过改变代码来改善这种情况,并且只能在画布的边界内移动球拍,如下所示:

+ +
if(rightPressed && paddleX < canvas.width-paddleWidth) {
+    paddleX += 7;
+}
+else if(leftPressed && paddleX > 0) {
+    paddleX -= 7;
+}
+ +

我们使用在Canvas左侧的0和右侧的canvas.width-paddleWidth之间的paddleX位置移动,这会让球拍按预期的要求移动。

+ +

将上面的代码块添加到底部的draw()函数中,在右大括号的上方。

+ +

现在唯一要做的就是在draw()函数内调用drawPaddle()函数,将其实际渲染在屏幕上。 在draw()函数内添加下面一行,就在调用drawBall()的那一行的下面:

+ +
drawPaddle();
+
+ +

比较你的代码

+ +

以下是我们的示例代码,以便与您进行比较:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/end3r/tgn3zscj/","","320")}}

+ +
+

练习: 让球拍变快变慢,或者改变它的大小。

+
+ +

下一步

+ +

现在我们有一些类似于游戏的东西。 唯一的麻烦就是无论如何你都可以继续用球拍击球。 这一切都将在第五章中改变,游戏结束时,我们会为游戏添加一个最后的状态。

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_pure_JavaScript/Bounce_off_the_walls", "Games/Workflows/2D_Breakout_game_pure_JavaScript/Game_over")}}

diff --git a/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/track_the_score_and_win/index.html b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/track_the_score_and_win/index.html new file mode 100644 index 0000000000..fc39385f02 --- /dev/null +++ b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/track_the_score_and_win/index.html @@ -0,0 +1,92 @@ +--- +title: 跟踪得分和获胜 +slug: Games/Tutorials/2D_Breakout_game_pure_JavaScript/Track_the_score_and_win +translation_of: Games/Tutorials/2D_Breakout_game_pure_JavaScript/Track_the_score_and_win +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_pure_JavaScript/Collision_detection", "Games/Workflows/2D_Breakout_game_pure_JavaScript/Mouse_controls")}}

+ +
+

本篇为Gamedev Canvas tutorial10节教程中的第8节。在你完成这篇课程之后,你可以在Gamedev-Canvas-workshop/lesson8.html找到我们的源代码。

+
+ +

破坏砖块真的很酷,但更酷的是,游戏可以给每个用户击破的砖块奖励分数,并保持总分。

+ +

计算分数

+ +

如果你能在整个游戏中看到你的分数,最终你会给你的朋友留下深刻印象。你需要一个变量来记录分数。在变量的其余部分之后,将下面的内容添加到JavaScript中:

+ +
var score = 0;
+ +

您还需要一个 drawScore() 函数来创建和更新分数显示。在 collisionDetection() 函数之后添加以下内容:

+ +
function drawScore() {
+    ctx.font = "16px Arial";
+    ctx.fillStyle = "#0095DD";
+    ctx.fillText("Score: "+score, 8, 20);
+}
+ +

在画布上绘制文本类似于绘制形状。字体定义看起来与CSS中的字体定义完全一样——可以在{{domxref("CanvasRenderingContext2D.font","font()")}} 方法中设置大小和字体类型。然后使用{{domxref("CanvasRenderingContext2D.fillStyle()","fillStyle()")}} 来设置字体的颜色,{{domxref("CanvasRenderingContext2D.fillText","fillText()")}} 来设置将放置在画布上的实际文本,和其放置位置。第一个参数是文本本身——上面的代码显示当前点的数量——最后两个参数是文本将放置在画布上的坐标。

+ +

若要在每次击中砖块时评分,则在 collisionDetection()中添加计分规则,以在每次检测到碰撞时增加得分变量的值。将下面突出显示的行添加到代码中:

+ +
function collisionDetection() {
+    for(var c=0; c<brickColumnCount; c++) {
+        for(var r=0; r<brickRowCount; r++) {
+            var b = bricks[c][r];
+            if(b.status == 1) {
+                if(x > b.x && x < b.x+brickWidth && y > b.y && y < b.y+brickHeight) {
+                    dy = -dy;
+                    b.status = 0;
+                    score++;
+                }
+            }
+        }
+    }
+}
+ +

从 draw() 函数调用 drawScore() ,使每一个新帧的分数都保持最新,在 draw()中添加下面的行,在 drawPaddle() 下面调用:

+ +
drawScore();
+ +

当所有砖块被破坏时显示获胜消息

+ +

收集这些点很有效,但是你不会永远添加它们-当所有的砖头都被破坏的时候呢?毕竟这是游戏的主要目的,所以如果收集到所有可用的点,你应该显示一个获胜的消息。将下面突出显示的部分添加到 collisionDetection() 函数中:

+ +
function collisionDetection() {
+    for(var c=0; c<brickColumnCount; c++) {
+        for(var r=0; r<brickRowCount; r++) {
+            var b = bricks[c][r];
+            if(b.status == 1) {
+                if(x > b.x && x < b.x+brickWidth && y > b.y && y < b.y+brickHeight) {
+                    dy = -dy;
+                    b.status = 0;
+                    score++;
+                    if(score == brickRowCount*brickColumnCount) {
+                        alert("YOU WIN, CONGRATULATIONS!");
+                        document.location.reload();
+                    }
+                }
+            }
+        }
+    }
+}
+ +

谢谢你做的这些,你的用户可以真正赢得游戏时,他们也摧毁了所有的砖块。当用户来到游戏还有一点非常重要,用户一旦点击了警告按钮, document.location.reload()函数将重新加载页面并重新启动游戏。

+ +

比较你的代码

+ +

最新的代码是这样(和工程),如果你想比较和对比它与你写的:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/yumetodo/2m74vr9r/1/","","395")}}

+ +
+

练习:在每一个砖头击破后添加更多的分数,打印出收集到的点数在游戏结束警告框中。

+
+ +

下一节

+ +

游戏到这一步看起来相当不错。在下一课中,您将通过添加鼠标控件来扩大游戏的吸引力:Mouse controls.。

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_pure_JavaScript/Collision_detection", "Games/Workflows/2D_Breakout_game_pure_JavaScript/Mouse_controls")}}

diff --git "a/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/\346\224\266\345\260\276\345\267\245\344\275\234/index.html" "b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/\346\224\266\345\260\276\345\267\245\344\275\234/index.html" new file mode 100644 index 0000000000..baa5a514fc --- /dev/null +++ "b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/\346\224\266\345\260\276\345\267\245\344\275\234/index.html" @@ -0,0 +1,113 @@ +--- +title: 收尾工作 +slug: Games/Tutorials/2D_Breakout_game_pure_JavaScript/收尾工作 +tags: + - Canvas + - JavaScript + - requestAnimationFrame + - 入门 + - 教程 + - 游戏 + - 生命 +translation_of: Games/Tutorials/2D_Breakout_game_pure_JavaScript/Finishing_up +--- +
{{GamesSidebar}}
+ +
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{Previous("Games/Workflows/2D_Breakout_game_pure_JavaScript/Mouse_controls")}}

+ +
+

本篇为 Gamedev Canvas tutorial 10节教程中的第10节也是最后一节。完成这篇课程后,你可以在 Gamedev-Canvas-workshop/lesson10.html 找到我们的源代码。

+
+ +

不管我们做什么游戏,它总是存在优化的空间。例如,我们可以为玩家多提供几条命,让他们能在发生一两次失误的情况下顺利完成游戏。或者,我们也可以在渲染代码上下工夫。

+ +

加入生命机制

+ +

在游戏中实现生命机制的思路很直接。让我们先新增一个变量,用来存储其生命值。把下面这行代码和我们声明其它变量的代码放在一起:

+ +
var lives = 3;
+ +

在 canvas 上绘制生命值计数的做法几乎和绘制分数一样——把下面的函数添加到drawScore() 函数后面:

+ +
function drawLives() {
+    ctx.font = "16px Arial";
+    ctx.fillStyle = "#0095DD";
+    ctx.fillText("Lives: "+lives, canvas.width-65, 20);
+}
+ +

当玩家失误时,我们不立即结束游戏,而是减少生命计数,直到为零。在玩家用掉一条命后,我们也可以重置小球和球板位置。那么,在函数 draw() 中将下面三行:

+ +
alert("GAME OVER");
+document.location.reload();
+clearInterval(interval); // Needed for Chrome to end game
+
+ + + + + +

替换为下面的代码,注意到我们加入了一点点逻辑控制:

+ +
lives--;
+if(!lives) {
+    alert("GAME OVER");
+    document.location.reload();
+    clearInterval(interval); // Needed for Chrome to end game
+}
+else {
+    x = canvas.width/2;
+    y = canvas.height-30;
+    dx = 2;
+    dy = -2;
+    paddleX = (canvas.width-paddleWidth)/2;
+}
+ +

现在,当小球碰到屏幕底边时,我们让变量lives 的值减一。如果生命用尽,游戏就宣告结束;否则就重置小球与球板的位置,以及小球的速度。

+ +

渲染生命值

+ +

现在只需在 draw() 函数内调用drawLives() 即可。让我们把它加到drawScore() 的下一行:

+ +
drawLives();
+
+ +

用 requestAnimationFrame() 优化渲染

+ +

现在让我们处理一些与游戏机制无关,但与画面渲染相关的东西。和我们目前使用{{domxref("windowTimers.setInterval()", "setInterval()")}} 实现的固定帧率渲染相比,{{domxref("window.requestAnimationFrame", "requestAnimationFrame")}} 能让浏览器更好地渲染画面。让我们把下面这行代码:

+ +
var interval = setInterval(draw, 10);
+ +

替换为:

+ +
draw();
+ +

再把代码中的每一处

+ +
clearInterval(interval); // Needed for Chrome to end game
+
+ +

删除。然后,在 draw() 函数的最下方(右花括号之前)加入下面这行代码。 它的作用是使 draw() 函数递归调用自身:

+ +
requestAnimationFrame(draw);
+ +

现在 draw() 函数在 requestAnimationFrame() 的循环中被反复调用,之先前做法最大的不同是,我们将帧率的控制权交给浏览器,而不是固定的 10 毫秒。浏览器会在适当的时机同步帧率,并且只在必要的时候才刷新渲染的图形。这使得我们的动画比之前的 setInterval() 方法更加流畅且高效.

+ +

比较你的代码

+ +

我们的游戏的最终版本已经完成!以上。

+ +

{{JSFiddleEmbed("https://jsfiddle.net/raymondjplante/dfh2tpu1/","","395")}}

+ +
+

练习:试着改变生命的数目和球从球板上反弹的角度。

+
+ +

游戏结束——暂时看来!

+ +

祝贺你——你完成了本教程的所有小节!现在,你应该已经掌握 canvas 操纵的基础和 2D 游戏背后的逻辑了。是时候去学习一些框架,继续你的游戏开发之旅了!你可以看看本系列的姊妹篇:用 Phaser 制作 2D 打砖块游戏 或者 Cyber Orb built in Phaser 。或者,你也可以在 MDN 游戏区 中获得灵感和更多知识。

+ +

你也可以回到本教程的目录页。祝编程愉快!

+ +

{{Previous("Games/Workflows/2D_Breakout_game_pure_JavaScript/Mouse_controls")}}

diff --git "a/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/\351\274\240\346\240\207\346\216\247\345\210\266/index.html" "b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/\351\274\240\346\240\207\346\216\247\345\210\266/index.html" new file mode 100644 index 0000000000..cb90cb8773 --- /dev/null +++ "b/files/zh-cn/games/tutorials/2d_breakout_game_pure_javascript/\351\274\240\346\240\207\346\216\247\345\210\266/index.html" @@ -0,0 +1,61 @@ +--- +title: 鼠标控制 +slug: Games/Tutorials/2D_Breakout_game_pure_JavaScript/鼠标控制 +tags: + - Canvas + - JavaScript + - 入门 + - 操作控制 + - 教程 + - 游戏 + - 鼠标 +translation_of: Games/Tutorials/2D_Breakout_game_pure_JavaScript/Mouse_controls +--- +
{{GamesSidebar}}
+ +
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_pure_JavaScript/Track_the_score_and_win", "Games/Workflows/2D_Breakout_game_pure_JavaScript/Finishing_up")}}

+ +
+

本篇为 Gamedev Canvas tutorial 10节教程中的第9节。在你完成这篇课程之后,你可以在 Gamedev-Canvas-workshop/lesson9.html 找到我们的源代码。

+
+ +

这个游戏实际已经完成,现在让我们着手去润色。我们已经添加过键盘控制,而加入鼠标控制也同样简单。

+ +

监听鼠标移动

+ +

监听鼠标移动甚至比监听按键更简单:只需监听 {{event("mousemove")}} 这个事件即可。把下面这行代码和其它事件监听代码放在一起,在 keyup event 的下一行:

+ +
document.addEventListener("mousemove", mouseMoveHandler, false);
+ +

将球板移动绑定到鼠标移动

+ +

我们可以根据鼠标光标位置来更新球板位置——下面这个函数正是做这件事的。把这个函数加到你的代码中,接在你刚刚加入的那行后面:

+ +
function mouseMoveHandler(e) {
+    var relativeX = e.clientX - canvas.offsetLeft;
+    if(relativeX > 0 && relativeX < canvas.width) {
+        paddleX = relativeX - paddleWidth/2;
+    }
+}
+ +

在这个函数中,我们首先计算 relativeX 的值,它等于鼠标在视窗中的水平位置 (e.clientX) 减去 canvas 元素左边框到视窗左边框的距离 (canvas.offsetLeft) —— 这就得到了 canvas 元素左边框到鼠标的距离。若这个值大于零,且小于 canvas 的宽度,说明鼠标指针落在 canvas 边界内,这时就把 paddleX (等于球板左边缘的坐标)设为 relativeX 减速去球板宽度的一半。这样就确保位移是相对于球板中心进行的。

+ +

现在球板将跟随鼠标指针。不过由于我们将球板移动限制在 canvas 大小范围内,它不会从两边完全消失。

+ +

比较你的代码

+ +

以下是我们的示例代码,以便与您进行比较:

+ +

{{JSFiddleEmbed("https://jsfiddle.net/raymondjplante/vt7y5hcp/","","395")}}

+ +
+

练习:调整球板移动的范围,使得整个球板总是可见,而不是在移动到边缘时被遮住一半。

+
+ +

下一步

+ +

现在我们已经拥有一个完整的游戏。我们的系列教程将以一些细节上的调整作为结束。

+ +

{{PreviousNext("Games/Workflows/2D_Breakout_game_pure_JavaScript/Track_the_score_and_win", "Games/Workflows/2D_Breakout_game_pure_JavaScript/Finishing_up")}}

diff --git a/files/zh-cn/games/tutorials/html5_gamedev_phaser_device_orientation/index.html b/files/zh-cn/games/tutorials/html5_gamedev_phaser_device_orientation/index.html new file mode 100644 index 0000000000..e5af242f12 --- /dev/null +++ b/files/zh-cn/games/tutorials/html5_gamedev_phaser_device_orientation/index.html @@ -0,0 +1,449 @@ +--- +title: 2D maze game with device orientation +slug: Games/Tutorials/HTML5_Gamedev_Phaser_Device_Orientation +translation_of: Games/Tutorials/HTML5_Gamedev_Phaser_Device_Orientation +--- +
{{GamesSidebar}}
+ +
{{IncludeSubnav("/en-US/docs/Games")}}
+ +
+

在本教程中,我们将介绍构建 HTML5 移动游戏的过程。 本游戏使用 Device OrientationVibration APIs 来增强游戏玩法,并使用 Phaser 框架构建。 为了充分理解本教程建议您先学习基础的JavaScript 知识。

+
+ +

Example game

+ +

在本教程结束时,您将有一个功能齐全的游戏demo:Cyber Orb 。如下所示:

+ +

A 2D game board featuring a small yellow ball. There is a large black hole for the ball to escape down, and a number of barriers blocking the ball from escaping.

+ +

Phaser framework

+ +

Phaser 是构建桌面和移动 HTML5 游戏的框架。它非常新,但由于热情的社区参与开发过程它同时也是快速增长的。您能够在 GitHub 查看它的开放源代码,阅读 在线文档 并浏览大量 示例 。Phaser 框架为您提供了一组工具,这些工具将加快开发速度,并帮助处理完成游戏所需的一般任务,因此您可以专注于游戏创意本身。

+ +

Starting with the project

+ +

您能够在GitHub看到它的源代码 Cyber Orb。文件夹结构非常简单:起点是 index.html。 我们在该文件中初始化框架并设置 html 元素 {{htmlelement("canvas")}} 以呈现游戏。

+ +

Screenshot of the GitHub repository with the Cyber Orb game code, listing the folders and the files in the main structure.

+ + + +

您可以在您最喜爱的浏览器中打开index文件以启动并尝试游戏。目录中还有三个文件夹:

+ + + +

Setting up the Canvas

+ +

We will be rendering our game on Canvas, but we won't do it manually — this will be taken care of by the framework. Let’s set it up: our starting point is the index.html file with the following content. You can create this yourself if you want to follow along:

+ +
<!DOCTYPE html>
+<html>
+<head>
+    <meta charset="utf-8" />
+    <title>Cyber Orb demo</title>
+    <style> body { margin: 0; background: #333; } </style>
+    <script src="src/phaser-arcade-physics.2.2.2.min.js"></script>
+    <script src="src/Boot.js"></script>
+    <script src="src/Preloader.js"></script>
+    <script src="src/MainMenu.js"></script>
+    <script src="src/Howto.js"></script>
+    <script src="src/Game.js"></script>
+</head>
+<body>
+<script>
+(function() {
+    var game = new Phaser.Game(320, 480, Phaser.CANVAS, 'game');
+    game.state.add('Boot', Ball.Boot);
+    game.state.add('Preloader', Ball.Preloader);
+    game.state.add('MainMenu', Ball.MainMenu);
+    game.state.add('Howto', Ball.Howto);
+    game.state.add('Game', Ball.Game);
+    game.state.start('Boot');
+})();
+</script>
+</body>
+</html>
+ +

So far we have a simple HTML website with some basic content in the <head> section: charset, title, CSS styling and the inclusion of the JavaScript files. The <body> contains initialization of the Phaser framework and the definitions of the game states.

+ +
var game = new Phaser.Game(320, 480, Phaser.CANVAS, 'game');
+ +

The line above will initialize the Phaser instance — the arguments are the width of the Canvas, height of the Canvas, rendering method (we're using CANVAS, but there are also WEBGL and AUTO options available) and the optional ID of the DOM container we want to put the Canvas in. If there's nothing specified in that last argument or the element is not found, the Canvas will be added to the <body> tag. Without the framework, to add the Canvas element to the page, you would have to write something like this inside the <body> tag:

+ +
<canvas id='game' width='320' height='480'></canvas>
+ +

The important thing to remember is that the framework is setting up helpful methods to speed up a lot of things like image manipulation or assets management, which would be a lot harder to do manually.

+ +
+

Note: You can read the Building Monster Wants Candy article for the in-depth introduction to the basic Phaser-specific functions and methods.

+
+ +

Back to game states: the line below is adding a new state called Boot to the game:

+ +
game.state.add('Boot', Ball.Boot);
+ +

The first value is the name of the state and the second one is the object we want to assign to it. The start method is starting the given state and making it active. Let's see what the states are actually.

+ +

Managing game states

+ +

The states in Phaser are separate parts of the game logic; in our case we’re loading them from independent JavaScript files for better maintainability. The basic states used in this game are: Boot, Preloader, MainMenu, Howto and Game. Boot will take care of initializing a few settings, Preloader will load all of the assets like graphics and audio, MainMenu is the menu with the start button, Howto shows the "how to play" instructions and the Game state lets you actually play the game. Let's quickly go though the content of those states.

+ +

Boot.js

+ +

The Boot state is the first one in the game.

+ +
var Ball = {
+    _WIDTH: 320,
+    _HEIGHT: 480
+};
+Ball.Boot = function(game) {};
+Ball.Boot.prototype = {
+    preload: function() {
+        this.load.image('preloaderBg', 'img/loading-bg.png');
+        this.load.image('preloaderBar', 'img/loading-bar.png');
+    },
+    create: function() {
+        this.game.scale.scaleMode = Phaser.ScaleManager.SHOW_ALL;
+        this.game.scale.pageAlignHorizontally = true;
+        this.game.scale.pageAlignVertically = true;
+        this.game.state.start('Preloader');
+    }
+};
+ +

The main Ball object is defined and we're adding two variables called _WIDTH and _HEIGHT that are the width and the height of the game Canvas — they will help us position the elements on the screen. We're loading two images first that will be used later in the Preload state to show the progress of loading all the other assets. The create function holds some basic configuration: we're setting up the scaling and alignment of the Canvas, and moving on to the Preload state when everything's ready.

+ +

Preloader.js

+ +

The Preloader state takes care of loading all the assets:

+ +
Ball.Preloader = function(game) {};
+Ball.Preloader.prototype = {
+    preload: function() {
+        this.preloadBg = this.add.sprite((Ball._WIDTH-297)*0.5, (Ball._HEIGHT-145)*0.5, 'preloaderBg');
+        this.preloadBar = this.add.sprite((Ball._WIDTH-158)*0.5, (Ball._HEIGHT-50)*0.5, 'preloaderBar');
+        this.load.setPreloadSprite(this.preloadBar);
+
+        this.load.image('ball', 'img/ball.png');
+        // ...
+        this.load.spritesheet('button-start', 'img/button-start.png', 146, 51);
+        // ...
+        this.load.audio('audio-bounce', ['audio/bounce.ogg', 'audio/bounce.mp3', 'audio/bounce.m4a']);
+    },
+    create: function() {
+        this.game.state.start('MainMenu');
+    }
+};
+ +

There are single images, spritesheets and audio files loaded by the framework. In this state the preloadBar is showing the progress on the screen. That progress of the loaded assets is visualized by the framework with the use of one image. With every asset loaded you can see more of the preloadBar image: from 0% to 100%, updated on every frame. After all the assets are loaded, the MainMenu state is launched.

+ + + +

The MainMenu state shows the main menu of the game, where you can start playing by clicking the button.

+ +
Ball.MainMenu = function(game) {};
+Ball.MainMenu.prototype = {
+    create: function() {
+        this.add.sprite(0, 0, 'screen-mainmenu');
+        this.gameTitle = this.add.sprite(Ball._WIDTH*0.5, 40, 'title');
+        this.gameTitle.anchor.set(0.5,0);
+        this.startButton = this.add.button(Ball._WIDTH*0.5, 200, 'button-start', this.startGame, this, 2, 0, 1);
+        this.startButton.anchor.set(0.5,0);
+        this.startButton.input.useHandCursor = true;
+    },
+    startGame: function() {
+        this.game.state.start('Howto');
+    }
+};
+ +

To create a new button there's add.button method with the following list of optional arguments:

+ + + +

The anchor.set is setting up the anchor point on the button for which all the calculations of the position are applied. In our case it's anchored half the way from the left edge and at the start of the top edge, so it can be easily horizontally centered on the screen without the need to know its width.

+ +

When the start button is pressed, instead of jumping directly into the action the game will show the screen with the information on how to play the game.

+ +

Howto.js

+ +
Ball.Howto = function(game) {
+};
+Ball.Howto.prototype = {
+    create: function() {
+        this.buttonContinue = this.add.button(0, 0, 'screen-howtoplay', this.startGame, this);
+    },
+    startGame: function() {
+        this.game.state.start('Game');
+    }
+};
+ +

The Howto state shows the gameplay instructions on the screen before starting the game. After clicking the screen the actual game is launched.

+ +

Game.js

+ +

The Game state from the Game.js file is where all the magic happens. All the initialization is in the create() function (launched once at the beginning of the game). After that some functionality will require further code to control — we will write our own functions to handle more complicated tasks. In particular, take note of the update() function (executed at every frame), which updates things such as the ball position.

+ +
Ball.Game = function(game) {};
+Ball.Game.prototype = {
+    create: function() {},
+    initLevels: function() {},
+    showLevel: function(level) {},
+    updateCounter: function() {},
+    managePause: function() {},
+    manageAudio: function() {},
+    update: function() {},
+    wallCollision: function() {},
+    handleOrientation: function(e) {},
+    finishLevel: function() {}
+};
+ +

The create and update functions are framework-specific, while others will be our own creations:

+ + + +

Adding the ball and its motion mechanics

+ +

First, let’s go to the create() function, initialize the ball object itself and assign a few properties to it:

+ +
this.ball = this.add.sprite(this.ballStartPos.x, this.ballStartPos.y, 'ball');
+this.ball.anchor.set(0.5);
+this.physics.enable(this.ball, Phaser.Physics.ARCADE);
+this.ball.body.setSize(18, 18);
+this.ball.body.bounce.set(0.3, 0.3);
+ +

Here we’re adding a sprite at the given place on the screen and using the 'ball' image from the loaded graphic assets. We’re also setting the anchor for any physics calculations to the middle of the ball, enabling the Arcade physics engine (which handles all the physics for the ball movement), and setting the size of the body for the collision detection. The bounce property is used to set the bounciness of the ball when it hits the obstacles.

+ +

Controlling the ball

+ +

It’s cool to have the ball ready to be thrown around in the play area, but it’s also important to be able to actually move it! Now we will add the ability to control the ball with the keyboard on the desktop devices, and then we will move to the implementation of the Device Orientation API. Let’s focus on the keyboard first by adding the following to the create() function :

+ +
this.keys = this.game.input.keyboard.createCursorKeys();
+ +

As you can see there’s a special Phaser function called createCursorKeys(), which will give us an object with event handlers for the four arrow keys to play with: up, down, left and right.

+ +

Next we will add the following code to the update() function, so it will be fired on every frame. The this.keys object will be checked against player input, so the ball can react accordingly with the predefined force:

+ +
if(this.keys.left.isDown) {
+    this.ball.body.velocity.x -= this.movementForce;
+}
+else if(this.keys.right.isDown) {
+    this.ball.body.velocity.x += this.movementForce;
+}
+if(this.keys.up.isDown) {
+    this.ball.body.velocity.y -= this.movementForce;
+}
+else if(this.keys.down.isDown) {
+    this.ball.body.velocity.y += this.movementForce;
+}
+ +

That way we can check which key is pressed at the given frame and apply the defined force to the ball, thus increase the velocity in the proper direction.

+ +

Implementing the Device Orientation API

+ +

Probably the most interesting part of the game is its usage of the Device Orientation API for control on mobile devices. Thanks to this you can play the game by tilting the device in the direction you want the ball to roll. Here’s the code from the create() function responsible for this:

+ +
window.addEventListener("deviceorientation", this.handleOrientation, true);
+ +

We’re adding an event listener to the "deviceorientation" event and binding the handleOrientation function which looks like this:

+ +
handleOrientation: function(e) {
+    var x = e.gamma;
+    var y = e.beta;
+    Ball._player.body.velocity.x += x;
+    Ball._player.body.velocity.y += y;
+},
+ +

The more you tilt the device, the more force is applied to the ball, therefore the faster it moves (the velocity is higher).

+ +

An explanation of the X, Y and Z axes of a Flame mobile device with the Cyber Orb game demo on the screen.

+ +
+

Note: To find more out about implementing device orientation and what raw code would look like, read Keep it level: responding to device orientation changes.

+
+ +

Adding the hole

+ +

The main objective in the game is to move the ball from the starting position to the ending position: a hole in the ground. Implementation looks very similar to the part where we created the ball, and it's also added in the create() function of our Game state:

+ +
this.hole = this.add.sprite(Ball._WIDTH*0.5, 90, 'hole');
+this.physics.enable(this.hole, Phaser.Physics.ARCADE);
+this.hole.anchor.set(0.5);
+this.hole.body.setSize(2, 2);
+ +

The difference is that our hole’s body will not move when we hit it with the ball and will have the collision detection calculated (which will be discussed later on in this article).

+ +

Building the block labyrinth

+ +

To make the game harder and more interesting we will add some obstacles between the ball and the exit. We could use a level editor, but for the sake of this tutorial let's create something on our own.

+ +

To hold the block information we'll use a level data array: for each block we'll store the top and left absolute positions in pixels (x and y) and the type of the block — horizontal or vertical (t with the 'w' value meaning width and 'h' meaning height). Then, to load the level we'll parse the data and show the blocks specific for that level. In the initLevels function we have:

+ +
this.levelData = [
+    [
+        { x: 96, y: 224, t: 'w' }
+    ],
+    [
+        { x: 72, y: 320, t: 'w' },
+        { x: 200, y: 320, t: 'h' },
+        { x: 72, y: 150, t: 'w' }
+    ],
+    // ...
+];
+ +

Every array element holds a collection of blocks with an x and y position and t value for each. After levelData, but still in the initLevels function, we're adding the blocks into an array in the for loop using some of the framework-specific methods:

+ +
for(var i=0; i<this.maxLevels; i++) {
+    var newLevel = this.add.group();
+    newLevel.enableBody = true;
+    newLevel.physicsBodyType = Phaser.Physics.ARCADE;
+    for(var e=0; e<this.levelData[i].length; e++) {
+        var item = this.levelData[i][e];
+        newLevel.create(item.x, item.y, 'element-'+item.t);
+    }
+    newLevel.setAll('body.immovable', true);
+    newLevel.visible = false;
+    this.levels.push(newLevel);
+}
+ +

First, add.group() is used to create a new group of items. Then the ARCADE body type is set for that group to enable physics calculations. The newLevel.create method creates new items in the group with starting left and top positions, and its own image. If you don't want to loop through the list of items again to add a property to every single one explicitly, you can use setAll on a group to apply it to all the items in that group.

+ +

The objects are stored in the this.levels array, which is by default invisible. To load specific levels, we make sure the previous levels are hidden, and show the current one:

+ +
showLevel: function(level) {
+    var lvl = level | this.level;
+    if(this.levels[lvl-2]) {
+        this.levels[lvl-2].visible = false;
+    }
+    this.levels[lvl-1].visible = true;
+}
+ +

Thanks to that the game gives the player a challenge - now he have to roll the ball across the play area and guide it through the labyrinth built from the blocks. It's just an example of loading the levels, and there are only 5 of them just to showcase the idea, but you can work on expanding that on your own.

+ +

Collision detection

+ +

At this point we've got the ball that is controlled by the player, the hole to reach and the obstacles blocking the way. There’s a problem though — our game doesn’t have any collision detection yet, so nothing happens when the ball hits the blocks — it just goes through. Let’s fix it! The good news is that the framework will take care of calculating the collision detection, we only have to specify the colliding objects in the update() function:

+ +
this.physics.arcade.collide(this.ball, this.borderGroup, this.wallCollision, null, this);
+this.physics.arcade.collide(this.ball, this.levels[this.level-1], this.wallCollision, null, this);
+ +

This will tell the framework to execute the wallCollision function when the ball hits any of the walls. We can use the wallCollision function to add any functionality we want like playing the bounce sound and implementing the Vibration API.

+ +

Adding the sound

+ +

Among the preloaded assets there was an audio track (in various formats for browser compatibility), which we can use now. It has to be defined in the create() function first:

+ +
this.bounceSound = this.game.add.audio('audio-bounce');
+ +

If the status of the audio is true (so the sounds in the game are enabled), we can play it in the wallCollision function:

+ +
if(this.audioStatus) {
+    this.bounceSound.play();
+}
+ +

That's all — loading and playing the sounds is easy with Phaser.

+ +

Implementing the Vibration API

+ +

When collision detection works as expected let's add some special effects with the help from the Vibration API.

+ +

A visualization of the vibrations of a Flame mobile device with the Cyber Orb game demo on the screen.

+ +

The best way to use it in our case is to vibrate the phone every time the ball hits the walls — inside the wallCollision function:

+ +
if("vibrate" in window.navigator) {
+    window.navigator.vibrate(100);
+}
+ +

If the vibrate method is supported by the browser and available in the window.navigator object, vibrate the phone for 100 miliseconds. That's it!

+ +

Adding the elapsed time

+ +

To improve replayability and give players the option to compete with each other we will store the elapsed time — players can then try to improve on their best game completion time. To implement this we have to create a variable for storing the actual number of seconds elapsed from the start of the game, and to show it for the player in the game. Let’s define the variables in the create function first:

+ +
this.timer = 0; // time elapsed in the current level
+this.totalTimer = 0; // time elapsed in the whole game
+ +

Then, right after that, we can initialize the necessary text objects to display this information to the user:

+ +
this.timerText = this.game.add.text(15, 15, "Time: "+this.timer, this.fontBig);
+this.totalTimeText = this.game.add.text(120, 30, "Total time: "+this.totalTimer, this.fontSmall);
+ +

We’re defining the top and left positions of the text, the content that will be shown and the styling applied to the text. We have this printed out on the screen, but it would be good to update the values every second:

+ +
this.time.events.loop(Phaser.Timer.SECOND, this.updateCounter, this);
+ +

This loop, also in the create function, will execute the updateCounter function every single second from the beginning of the game, so we can apply the changes accordingly. This is how the complete updateCounter function looks:

+ +
updateCounter: function() {
+    this.timer++;
+    this.timerText.setText("Time: "+this.timer);
+    this.totalTimeText.setText("Total time: "+(this.totalTimer+this.timer));
+},
+ +

As you can see we’re incrementing the this.timer variable and updating the content of the text objects with the current values on each iteration, so the player sees the elapsed time.

+ +

Finishing the level and the game

+ +

The ball is rolling on the screen, the timer is working and we have the hole created that we have to reach. Now let’s set up the possibility to actually finish the level! The following line in the update() function adds a listener that fires when the ball gets to the hole.

+ +
this.physics.arcade.overlap(this.ball, this.hole, this.finishLevel, null, this);
+ +

This works similarly to the collide method explained earlier. When the ball overlaps with the hole (instead of colliding), the finishLevel function is executed:

+ +
finishLevel: function() {
+    if(this.level >= this.maxLevels) {
+        this.totalTimer += this.timer;
+        alert('Congratulations, game completed!\nTotal time of play: '+this.totalTimer+' seconds!');
+        this.game.state.start('MainMenu');
+    }
+    else {
+        alert('Congratulations, level '+this.level+' completed!');
+        this.totalTimer += this.timer;
+        this.timer = 0;
+        this.level++;
+        this.timerText.setText("Time: "+this.timer);
+        this.totalTimeText.setText("Total time: "+this.totalTimer);
+        this.levelText.setText("Level: "+this.level+" / "+this.maxLevels);
+        this.ball.body.x = this.ballStartPos.x;
+        this.ball.body.y = this.ballStartPos.y;
+        this.ball.body.velocity.x = 0;
+        this.ball.body.velocity.y = 0;
+        this.showLevel();
+    }
+},
+ +

If the current level is equal to the maximum number of levels (in this case 5), then the game is finished — you'll get a congratulations message along with the number of seconds elapsed through the whole game, and a button to press that takes you to the main menu.

+ +

If the current level is lower than 5, all the neccesary variables are reset and the next level is loaded.

+ +

Ideas for new features

+ +

This is merely a working demo of a game that could have lots of additional features. We can for example add power-ups to collect along the way that will make our ball roll faster, stop the timer for a few seconds or give the ball special powers to go through obstacles. There’s also room for the traps which will slow the ball down or make it more difficult to reach the hole. You can create more levels of increasing difficulty. You can even implement achievements, leaderboards and medals for different actions in the game. There are endless possibilities — they only depend on your imagination.

+ +

Summary

+ +

I hope this tutorial will help you dive into 2D game development and inspire you to create awesome games on your own. You can play the demo game Cyber Orb and check out its source code on GitHub.

+ +

HTML5 gives us raw tools, the frameworks built on top of it are getting faster and better, so now is a great time get into web game development. In this tutorial we used Phaser, but there are a number of other frameworks worth considering too like ImpactJS, Construct 2 or PlayCanvas — it depends on your preferences, coding skills (or lack thereof), project scale, requirements and other aspects. You should check them all out and decide which one suits your needs best.

diff --git a/files/zh-cn/games/tutorials/index.html b/files/zh-cn/games/tutorials/index.html new file mode 100644 index 0000000000..63b38fa324 --- /dev/null +++ b/files/zh-cn/games/tutorials/index.html @@ -0,0 +1,25 @@ +--- +title: Tutorials +slug: Games/Tutorials +tags: + - Canvas + - Games + - JavaScript + - NeedsTranslation + - TopicStub + - Web + - Workflows +translation_of: Games/Tutorials +--- +
{{GamesSidebar}}
{{IncludeSubnav("/en-US/docs/Games")}}
+ +

This page contains multiple tutorial series that highlight different workflows for effectively creating different types of web games.

+ +
+
2D breakout game using pure JavaScript
+
In this step-by-step tutorial you'll implement a simple breakout clone using pure JavaScript. Along the way you will learn the basics of using the {{htmlelement("canvas")}} element to implement fundamental game mechanics like rendering and moving images, collision detection, control machanisms, and winning and losing states.
+
2D breakout game using Phaser
+
In this step-by-step tutorial you'll implement the same breakout clone as the previous tutorial series, except that this time you'll do it using thePhaser HTML5 game framework. This idea here is to teach some of the fundamentals (and advantages) of working with frameworks, along with fundamental game mechanics.
+
2D maze game with device orientation
+
This tutorial shows how to create a 2D maze game using HTML5, incorporating fundamentals such as collision detection and sprite placement on a {{htmlelement("canvas")}}. This is a mobile game that uses the Device Orientation and Vibration APIs to enhance the gameplay and is built using the Phaser framework.
+
-- cgit v1.2.3-54-g00ecf