「簡単なパックマン風のゲーム」をUnityで作成する。第3回目です。
はじめに
Part1は下記になります。
Part1でプレイヤーの移動を作成、Part2ではステージを作成しました。Part3では敵キャラクターを作成していきます。
作成開始
ステージ内を動く敵キャラを作成していきます。基本的な動きはプレイヤーと同じです。
Enemyの作成
敵キャラは下記の画像を利用していきます。白色で作成してあるので見えにくいです。
data:image/s3,"s3://crabby-images/1ca3c/1ca3c66204512d0fb55f5cb5d71b2f2f51076158" alt=""
画像を6枚(16×16のサイズ)に分割して、Unityに取り込みます。体部分と目の画像を分けてあります。
data:image/s3,"s3://crabby-images/44052/44052178e12c1738830045a4797cfab8bb625a7b" alt=""
プレイヤーの時と同じように画像の設定します。
data:image/s3,"s3://crabby-images/4a31a/4a31ab51bd1a77dcd667fd7106fefaf06de52b7f" alt=""
CreateEmptyで空オブジェクトを追加、名前を「Enemy」にします。
data:image/s3,"s3://crabby-images/c7807/c7807e8dce35379ec62ea0c0e68638ed44b6282a" alt=""
Enemyの配下に「Body」と「Eye」の空オブジェクトを作成します。
data:image/s3,"s3://crabby-images/277e1/277e1aa193f283bb632cfa17ff9c0b6c6e5ab69f" alt=""
Bodyには「SpriteRendrer」と「Animator」をアタッチします。Colorは好きな色で大丈夫です。「Order in Layer」を変更して手前に表示します。
data:image/s3,"s3://crabby-images/bbdf2/bbdf285d42eb9a80a79a87cdda6deb1ad4de38aa" alt=""
アニメーションの設定
Playerの時と同じようにアニメーションを設定していきます。下記のように画像を指定して、良い感じに動けばOKです。
data:image/s3,"s3://crabby-images/56e20/56e2021140564b82136b17d8b6a25f94f1661c41" alt=""
Eyeは「SpriteRendrer」をアタッチして画像をセット、「Order in Layer」を変更しておきます。
data:image/s3,"s3://crabby-images/79266/79266b28b7a311cf1bade7bc05e95e2da3f2ab31" alt=""
Enemyオブジェクトには「Rigidbody2D」をアタッチして重力の影響を受けないようにします。
data:image/s3,"s3://crabby-images/b0fa8/b0fa8d39413ef21aa1869057fe6b40f284c61343" alt=""
次にPlayerと同様に「CirclrCollider2D」とEnemyスクリプトもアタッチしておきます。
data:image/s3,"s3://crabby-images/dd8b1/dd8b19c7dfa02e252addc0720abe7f86f8a31955" alt=""
ポイントの作成
敵キャラの場合、プレイヤーと違い自動で動きます。今回は分岐地点に敵キャラが来たらランダムに移動先を選択するようにします。例えば、下記の左上の赤四角の場所に敵キャラが来たら下か右に移動するという感じです。
data:image/s3,"s3://crabby-images/322ee/322ee59b3ceca77900df7604a6e3f1d5eba141a0" alt=""
Hierarchyで「2DObject」→「Sprites」→「Circle」を選択します。
data:image/s3,"s3://crabby-images/73fcb/73fcbb6f755f602afae031e969a6552d0f985785" alt=""
名前をPointに変更して、「CircleCollider2D」をアタッチ、「isTrigger」にチェックを入れ、Pointスクリプトをアタッチします。
data:image/s3,"s3://crabby-images/df474/df4747613b2a2110d6978732e4bd2095d71135f1" alt=""
Pointスクリプトの中身を下記のようにします。
using System.Collections.Generic; using UnityEngine; public class Point : MonoBehaviour { public List<Vector2> Directions; }
例えば、先ほどの左上の場合、下記のように行き先を指定します。
data:image/s3,"s3://crabby-images/c2c7f/c2c7fd66100aaa1e5a8b8ba6406d3076dd462b23" alt=""
全てのPointを入力するのは大変なので、スクリプト内でRayを投げて壁の有無を確認して行き先候補をセットするようにします。
using System.Collections.Generic; using UnityEngine; public class Point : MonoBehaviour { [SerializeField] private LayerMask stageLayer; public List<Vector2> Directions; private void Start() { Directions = new List<Vector2>(); CheckDirection(Vector2.up); CheckDirection(Vector2.down); CheckDirection(Vector2.left); CheckDirection(Vector2.right); } private void CheckDirection(Vector2 direction) { RaycastHit2D hit = Physics2D.BoxCast (this.transform.position, Vector2.one * 0.5f, 0.0f, direction, 1.0f, this.stageLayer); if (hit.collider == null) { Directions.Add(direction); } } }
Pointオブジェクトをプレハブ化しておきます。
data:image/s3,"s3://crabby-images/5e85b/5e85b76fafe3fc995b2cda39dfafd1f229f14dc6" alt=""
Gridの配下に空オブジェクトを追加して「Points」という名前に変更、配下にPointを設置していきます。
data:image/s3,"s3://crabby-images/d8b28/d8b287263954f113158ee462d26454f40f0ad1b1" alt=""
下記のように分岐点全てにPointを設置します。
data:image/s3,"s3://crabby-images/383a5/383a5ecb31f93c988554a22132f12f8dcb26c7f8" alt=""
敵キャラ移動スクリプトの作成
敵キャラの移動スクリプト(Enemy)は下記のようにします。ほぼPlayerスクリプトと同様です。方向転換が「キー入力」から「ポイントにぶつかる」に変わっているのが異なる点です。
using UnityEngine; public class Enemy : MonoBehaviour { [SerializeField] private LayerMask stageLayer; private Rigidbody2D rb; private float speed = 7.5f; private Vector2 _direction; private Vector2 _directionReserve; private void Start() { rb = GetComponent<Rigidbody2D>(); _direction = Vector2.left; } private void Update() { if (_directionReserve != Vector2.zero) { CheckDirection(_directionReserve); } } private void FixedUpdate() { Vector2 dist = _direction * speed * Time.fixedDeltaTime; rb.MovePosition(rb.position + dist); } private void CheckDirection(Vector2 direction) { RaycastHit2D hit = Physics2D.BoxCast (transform.position, Vector2.one * 0.5f, 0.0f, direction, 1.0f, stageLayer); if (hit.collider == null) { _direction = direction; _directionReserve = Vector2.zero; } } private void OnTriggerEnter2D(Collider2D other) { Point point = other.GetComponent<Point>(); if (point != null) { int index = Random.Range(0, point.Directions.Count); _directionReserve = point.Directions[index]; } } }
Enemyスクリプトのパラメータをセットします。
data:image/s3,"s3://crabby-images/0eed8/0eed8392406c3004460fad5d473702a8d14bdc64" alt=""
Pointのradiusで判定の範囲を調整します(Radiusが大きいと分岐点の手前で敵キャラが移動します)
data:image/s3,"s3://crabby-images/45f46/45f46b8098d4560439fc02a4a9d4dc515d77a8ec" alt=""
目の動きのスクリプト
次に敵キャラの目の位置を進行方向に変更するスクリプトです。下記のようにEyeスクリプトを作成します。
using UnityEngine; public class Eye : MonoBehaviour { [SerializeField] private Sprite eye_up; [SerializeField] private Sprite eye_down; [SerializeField] private Sprite eye_left; [SerializeField] private Sprite eye_right; private SpriteRenderer _sr; private void Start() { _sr = GetComponent<SpriteRenderer>(); } public void ChangeEye(Vector2 direction) { if (direction == Vector2.up) { _sr.sprite = eye_up; } else if (direction == Vector2.down) { _sr.sprite = eye_down; } else if (direction == Vector2.left) { _sr.sprite = eye_left; } else if (direction == Vector2.right) { _sr.sprite = eye_right; } } }
パラメータには各画像をセットします。
data:image/s3,"s3://crabby-images/ece24/ece24b7fd4323b94b957b7f75bc3b19406228527" alt=""
Enemyの「CheckDirection」内に「eye.ChangeEye(_direction)」を追加します。
[SerializeField] private Eye eye; private void CheckDirection(Vector2 direction) { RaycastHit2D hit = Physics2D.BoxCast (transform.position, Vector2.one * 0.5f, 0.0f, direction, 1.0f, stageLayer); if (hit.collider == null) { _direction = direction; eye.ChangeEye(_direction); _directionReserve = Vector2.zero; } }
パラメータをセットします。
data:image/s3,"s3://crabby-images/de3b1/de3b1017dc566f38b41329b004339c79cff401b9" alt=""
実行して下記のように敵キャラがランダムに動けばOK
data:image/s3,"s3://crabby-images/360f2/360f2c454ab0c6ffdf0dbaede0ef7e324ef13530" alt=""
最後にPointのSpriteを無効にしてポイントの表示を消しておきます。
data:image/s3,"s3://crabby-images/80174/8017483d00457afe98b3a55c38f98dac36cbbed2" alt=""