From 33058f2b292b3a581333bdfb21b8f671898c5060 Mon Sep 17 00:00:00 2001 From: Peter Bengtsson Date: Tue, 8 Dec 2020 14:40:17 -0500 Subject: initial commit --- .../games/techniques/control_mechanisms/index.html | 78 +++++++++++ .../index.html" | 152 +++++++++++++++++++++ 2 files changed, 230 insertions(+) create mode 100644 files/zh-cn/games/techniques/control_mechanisms/index.html create mode 100644 "files/zh-cn/games/techniques/control_mechanisms/\347\247\273\345\212\250\347\253\257\350\247\246\346\221\270\346\216\247\345\210\266/index.html" (limited to 'files/zh-cn/games/techniques/control_mechanisms') diff --git a/files/zh-cn/games/techniques/control_mechanisms/index.html b/files/zh-cn/games/techniques/control_mechanisms/index.html new file mode 100644 index 0000000000..07b6e0b6e1 --- /dev/null +++ b/files/zh-cn/games/techniques/control_mechanisms/index.html @@ -0,0 +1,78 @@ +--- +title: Implementing game control mechanisms +slug: Games/Techniques/Control_mechanisms +tags: + - Controls + - Desktop + - Gamepad + - Games + - JavaScript + - Laptop + - Mobile + - NeedsTranslation + - TopicStub + - keyboard + - mouse + - touch +translation_of: Games/Techniques/Control_mechanisms +--- +
{{GamesSidebar}}

One of HTML5's main advantages as a game development platform is the ability to run on various platforms and devices. Streamlining cross device differences creates multiple challenges, not least when providing appropriate controls for different contexts. In this series of articles we will show you how you can approach building a game that can be played using touchscreen smartphones, mouse and keyboard, and also less common mechanisms such as gamepads.

+ +

Case study

+ +

We'll be using the Captain Rogers: Battle at Andromeda demo as an example.

+ +

Captain Rogers: Battle at Andromeda - cover of the game containing Enclave Games and Blackmoon Design logos, Roger's space ship and title of the game.

+ +

Captain Rogers was created using the Phaser framework, the most popular tool for simple 2D game development in JavaScript right now, but it should be fairly easy to reuse the knowledge contained within these articles when building games in pure JavaScript or any other framework. If you're looking for a good introduction to Phaser, then check the 2D breakout game using Phaser tutorial.

+ +

In the following articles we will show how to implement various different control mechanisms for Captain Rogers to support different platforms — from touch on mobile, through keyboard/mouse/gamepad on desktop, to more unconventional ones like TV remote, shouting to or waving your hand in front of the laptop, or squeezing bananas.

+ +

Setting up the environment

+ +

Let's start with a quick overview of the game's folder structure, JavaScript files and in-game states, so we know what's happening where. The game's folders look like this:

+ +

Captain Rogers: Battle at Andromeda - folder structure of the games' project containing JavaScript sources, images and fonts.

+ +

As you can see there are folders for images, JavaScript files, fonts and sound effects. The src folder contains the JavaScript files split as a separate states — Boot.js, Preloader.js, MainMenu.js and Game.js — these are loaded into the index file in this exact order. The first one initializes Phaser, the second preloads all the assets, the third one controls the main menu welcoming the player, and the fourth controls the actual gameplay.

+ +

Every state has its own default methods: preload(), create(), and update(). The first one is needed for preloading required assets, create() is executed once the state had started, and update() is executed on every frame.

+ +

For example, you can define a button in the create() function:

+ +
create: function() {
+	// ...
+	var buttonEnclave = this.add.button(10, 10, 'logo-enclave', this.clickEnclave, this);
+	// ...
+}
+
+ +

It will be created once at the start of the game, and will execute this.clickEnclave() action assigned to it when clicked, but you can also use the mouse's pointer value in the update() function to make an action:

+ +
update: function() {
+	// ...
+	if(this.game.input.mousePointer.isDown) {
+	    // do something
+	}
+	// ...
+}
+
+ +

This will be executed whenever the mouse button is pressed, and it will be checked against the input's isDown boolean variable on every frame of the game.

+ +

That should give you some understanding of the project structure. We'll be playing mostly with the MainMenu.js and Game.js files, and we'll explain the code inside the create() and update() methods in much more detail in later articles.

+ +

Pure JavaScript demo

+ +

There's also a small online demo with full source code available on GitHub where the basic support for the control mechanisms described in the articles is implemented in pure JavaScript. It will be explained in the given articles themselves below, but you can play with it already, and use the code however you want for learning purposes.

+ +

The articles

+ +

JavaScript is the perfect choice for mobile gaming because of HTML5 being truly multiplatform; all of the following articles focus on the APIs provided for interfacing with different control mechanisms:

+ +
    +
  1. Mobile touch controls — The first article will kick off with touch, as the mobile first approach is very popular.
  2. +
  3. Desktop mouse and keyboard controls — When playing on a desktop/laptop computer, providing keyboard and mouse controls is essential to provide an acceptable level of accessibility for the game.
  4. +
  5. Desktop gamepad controls — The Gamepad API rather usefully allows gamepads to be used for controlling web apps on desktop/laptop, for that console feel.
  6. +
  7. Unconventional controls — The final article showcases some unconventional control mechanisms, from the experimental to the slightly crazy, which you might not believe could be used to play the game.
  8. +
diff --git "a/files/zh-cn/games/techniques/control_mechanisms/\347\247\273\345\212\250\347\253\257\350\247\246\346\221\270\346\216\247\345\210\266/index.html" "b/files/zh-cn/games/techniques/control_mechanisms/\347\247\273\345\212\250\347\253\257\350\247\246\346\221\270\346\216\247\345\210\266/index.html" new file mode 100644 index 0000000000..e9a9abaf15 --- /dev/null +++ "b/files/zh-cn/games/techniques/control_mechanisms/\347\247\273\345\212\250\347\253\257\350\247\246\346\221\270\346\216\247\345\210\266/index.html" @@ -0,0 +1,152 @@ +--- +title: 移动端触摸控制 +slug: Games/Techniques/Control_mechanisms/移动端触摸控制 +translation_of: Games/Techniques/Control_mechanisms/Mobile_touch +--- +
{{GamesSidebar}}
+ +

{{NextMenu("Games/Techniques/Control_mechanisms/Desktop_with_mouse_and_keyboard", "Games/Techniques/Control_mechanisms")}}

+ +

未来手游一定是Web的天下,许多开发在游戏开发过程中首先选择手游 — 既然如此,触摸控制是不可少的。我们将在本教程中了解怎样简单地在移动端H5游戏中实现触摸控制 ,只要移动端支持触摸,你就可以尽情的玩。

+ +

说明:游戏 Captain Rogers: Battle at Andromeda 是基于Phaser 和Phaser-based管理控制,但它也可以用纯JavaScript实现。使用Phaser的好处它提供了辅助变量和方法可以直接调用,有助于快速的开发游戏,这需要根据项目实际情况选择。

+ +

纯 JavaScript 方式实现

+ +

我们可以实现自己的触摸事件 — 给document添加事件监听,并传入自定义功能的方法,非常简单:

+ +
var el = document.getElementsByTagName("canvas")[0];
+el.addEventListener("touchstart", handleStart);
+el.addEventListener("touchmove", handleMove);
+el.addEventListener("touchend", handleEnd);
+el.addEventListener("touchcancel", handleCancel);
+ +

这样, 在移动设备上屏幕上触摸游戏的 {{htmlelement("canvas")}} 将触发这些事件,因为我们就可以随意操控游戏(如:移动太空船)。 事件如下所示:

+ + + +
+

说明: 这篇 touch events 参考文章提供了更多的实例和介绍。

+
+ +

纯JavaScript示例

+ +

这个实现了移动端触摸的little demo代码已经放到了GibHub上,我们下载这个示例就可以实现在移动端屏幕上移动飞船。

+ +

我们将两种事件:touchstart touchmove 放到一个方法里处理. 为什么呢? touchHandler 方法定义的飞船位置变量适合下面两种情况下: 当玩家触摸时,但不移动它(touchstart)和当手指在屏幕上开始移动 (touchmove):

+ +
document.addEventListener("touchstart", touchHandler);
+document.addEventListener("touchmove", touchHandler);
+ +

touchHandler 方法的代码如下:

+ +
function touchHandler(e) {
+    if(e.touches) {
+        playerX = e.touches[0].pageX - canvas.offsetLeft - playerWidth / 2;
+        playerY = e.touches[0].pageY - canvas.offsetTop - playerHeight / 2;
+        output.innerHTML = "Touch: "+ " x: " + playerX + ", y: " + playerY;
+        e.preventDefault();
+    }
+}
+ +

If the touch occurs (touches object is not empty), then we will have all the info we need in that object. We can get the first touch (e.touches[0], our example is not multitouch-enabled), extract the pageX and pageY variables and set the player's ship position on the screen by subtracting the Canvas offset (distance from the Canvas and the edge of the screen) and half the player's width and height.

+ +

Touch controls for the player's ship, with visible output of the x and y position.

+ +

To see if it's working correctly we can output the x and y positions using the output element. The preventDefault() function is needed to prevent the browser from moving — without it you'd have the default behaviour, and the Canvas would be dragged around the page, which would show the browser scroll bars and look messy.

+ +

Touch events in Phaser

+ +

We don't have to do this on our own; frameworks like Phaser offer systems for managing touch events for us — see managing the touch events.

+ +

Pointer theory

+ +

A pointer represents a single finger on the touch screen. Phaser starts two pointers by default, so two fingers can perform an action at once. Captain Rogers is a simple game — it can be controlled by two fingers, the left one moving the ship and the right one controlling the ship's gun. There's no multitouch or gestures — everything is handled by single pointer inputs.

+ +

You can add more pointers to the game by using; this.game.input.addPointer up to ten pointers can be managed simultaneously. The most recently used pointer is available in the this.game.input.activePointer object — the most recent finger active on the screen.

+ +

If you need to access a specific pointer, they are all available at, this.game.input.pointer1this.game.input.pointer2, etc. They are assigned dynamically, so if you put three fingers on the screen, then, pointer1pointer2, and pointer3 will be active. Removing the second finger, for example, won't affect the other two, and setting it back again will use the first available property, so pointer2 will be used again.

+ +

You can quickly get the coordinates of the most recently active pointer via the this.game.input.x and this.game.input.y variables.

+ +

Input events

+ +

Instead of using the pointers directly it is also possible to listen for this.game.input events, like onDown, onUp, onTap and onHold:

+ +
this.game.input.onDown.add(itemTouched, this);
+
+function itemTouched(pointer) {
+    // do something
+}
+ +

The itemTouched() function will be executed when the onDown event is dispatched by touching the screen. The pointer variable will contain the information about the pointer that activated the event.

+ +

This approach uses the generally available this.game.input object, but you can also detect the actions on any game objects like sprites or buttons by using onInputOver, onInputOut, onInputDown, onInputUp, onDragStart, or onDragStop:

+ +
this.button.events.onInputOver.add(itemTouched, this);
+
+function itemTouched(button, pointer) {
+    // do something
+}
+ +

That way you'll be able to attach an event to any object in the game, like the player's ship, and react to the actions performed by the user.

+ +

An additional advantage of using Phaser is that the buttons you create will take any type of input, whether it's a touch on mobile or a click on desktop — the framework sorts this out in the background for you.

+ +

Implementation

+ +

The easiest way to add an interactive object that will listen for user input is to create a button:

+ +
var buttonEnclave = this.add.button(10, 10, 'logo-enclave', this.clickEnclave, this);
+ +

This one is formed in the MainMenu state — it will be placed ten pixels from the top left corner of the screen, use the logo-enclave image, and execute the clickEnclave() function when it is touched. This will work on mobile and desktop out of the box. There are a few buttons in the main menu, including the one that will start the game.

+ +

For the actual gameplay, instead of creating more buttons and covering the small mobile screen with them, we can use something a little bit different: we'll create invisible areas which respond to the given action. From a design point of view, it is better to make the field of activity bigger without covering half of the screen with button images. For example, tapping on the right side of the screen will fire the weapon:

+ +
this.buttonShoot = this.add.button(this.world.width*0.5, 0, 'button-alpha', null, this);
+this.buttonShoot.onInputDown.add(this.goShootPressed, this);
+this.buttonShoot.onInputUp.add(this.goShootReleased, this);
+ +

The code above will create a new button using a transparent image that covers the right half of the screen. You can assign functions on input down and input up separately if you'd like to perform more complicated actions, but in this game touching the right side of the screen will simply fire the bullets to the right — this is all we need in this case.

+ +

Moving the player could be managed by creating the four directional buttons, but we can take the advantage of touch screens and drag the player's ship around:

+ +
var player = this.game.add.sprite(30, 30, 'ship');
+player.inputEnabled = true;
+player.input.enableDrag();
+player.events.onDragStart.add(onDragStart, this);
+player.events.onDragStop.add(onDragStop, this);
+
+function onDragStart(sprite, pointer) {
+    // do something when dragging
+}
+ +

We can pull the ship around and do something in the meantime, and react when the drag is stopped. Hauling in Phaser, if enabled, will work out of the box — you don't have to set the position of the sprite yourself manually, so you could leave the onDragStart() function empty, or place some debug output to see if it's working correctly. The pointer element contains the x and y variables storing the current position of the dragged element.

+ +

Dedicated plugins

+ +

You could go even further and use dedicated plugins like Virtual Joystick — this is a paid, official Phaser plugin, but you can find free and open source alternatives. The initialization of Virtual Joystick looks like this:

+ +
this.pad = this.game.plugins.add(Phaser.VirtualJoystick);
+this.stick = this.pad.addStick(30, 30, 80, 'generic');
+ +

In the create() function of the Game state we're creating a virtual pad and a generic stick that has four directional virtual buttons by default. This is placed 30 pixels from the top and left edges of the screen and is 80 pixels wide.

+ +

The stick being pressed can be handled during the gameplay in the update function like so:

+ +
if(this.stick.isDown) {
+    // move the player
+}
+ +

We can adjust the player's velocity based on the current angle of the stick and move him appropriately.

+ +

摘要

+ +

这篇文章主要讲解如何在移动端实现触摸控制; 下一篇文章我们将看到怎样添加键盘和鼠标支持。

+ +

{{NextMenu("Games/Techniques/Control_mechanisms/Desktop_with_mouse_and_keyboard", "Games/Techniques/Control_mechanisms")}}

-- cgit v1.2.3-54-g00ecf