Unity2Dゲームで簡単なワープ機能を実装してみます。特定のオブジェクトに触れるとプレイヤーが移動する感じですね。
はじめに
Unityのバージョンは「2021.3.14f1」で作成しています。
2Dゲームで「プレイヤー(キャラクター)が特定のオブジェクト間を瞬間移動する」のを実装していきます。ポータルとかテレポートのようなイメージですね。
実装開始
簡単な画面作成→プレイヤー作成→ポータル作成の順番で実装していきます。
プレイヤーの作成
「2DObject」→「Sprites」→「Square」を追加します。
data:image/s3,"s3://crabby-images/b9457/b9457cbfae2ef17803120f66be74fa3be52269f5" alt=""
名前をPlayerに変更して、AddComponentから「Rigidbody2D」と「BoxCollider2D」を追加します。
data:image/s3,"s3://crabby-images/09043/09043cc7316745ff945c8266b13380e1b741e470" alt=""
「Rigidbody2D」はGravityScaleを0にします。
data:image/s3,"s3://crabby-images/e8076/e8076b783918a8eac08c23b7e30069d97ee07bf1" alt=""
BoxCollider2DはisTriggerにチェックを入れます。
data:image/s3,"s3://crabby-images/dee48/dee48dcd2539ecce7e92b3d4c41017bd5b5b792b" alt=""
プレイヤー移動のスクリプト
下記スクリプトを作成してPlayerオブジェクトに追加します。キー入力で上下左右に移動します。
using UnityEngine; public class Player : MonoBehaviour { private float speed = 10.0f; private Rigidbody2D rb; private Vector2 movement; private void Awake() { rb = GetComponent<Rigidbody2D>(); } void Update() { movement.x = Input.GetAxisRaw("Horizontal"); movement.y = Input.GetAxisRaw("Vertical"); } private void FixedUpdate() { rb.MovePosition(rb.position + movement.normalized * speed * Time.fixedDeltaTime); } }
実行してみると下記のような感じに動きます。
data:image/s3,"s3://crabby-images/8b557/8b557093dbf65f6a621d0adb5c9c8b007aa4c834" alt=""
ポータルの作成
「2DObject」→「Sprites」→「Capsule」を2つ追加します。
data:image/s3,"s3://crabby-images/1c17d/1c17d6b5e03f19ef82db8a435ff7ea1dd5bce27d" alt=""
名前をPortalにして、色を変更。AddComponentからCapsuleCollider2Dを追加します。
data:image/s3,"s3://crabby-images/a59cf/a59cf4530c36dae160bd8852bce5186db74f74d3" alt=""
コピーして色を変更しておきます。
data:image/s3,"s3://crabby-images/9efff/9efff7a73a1eb218013ec6a4999d72087f7eed8f" alt=""
場所は適当に変更して下記のような画面にします。黄色と赤色の間を移動するようにしていきます。
data:image/s3,"s3://crabby-images/409cf/409cf24d8c3c98d602c97373f564b6459d00e2ff" alt=""
ポータルスクリプトの作成
下記スクリプトを作成します。
using UnityEngine; public class Portal : MonoBehaviour { [SerializeField] private Transform destination; private float distance = 0.3f; private void OnTriggerEnter2D(Collider2D collision) { if (Vector2.Distance(transform.position, collision.transform.position) > distance) collision.transform.position = new Vector2(destination.position.x, destination.position.y); } }
ポータルに触れたオブジェクトを指定した位置に移動します。
ポータルにスクリプトを追加して、移動先をセットします。
data:image/s3,"s3://crabby-images/2ffb7/2ffb7591363e614267894c484245de810864a214" alt=""
実行してみると下記のような感じに。黄色に触れると赤色に移動して、赤色に触れると黄色に移動します。
data:image/s3,"s3://crabby-images/9c04e/9c04e0a4b43e1b6cbb7de39c9125e7096d80f18c" alt=""
スクリプトの下記部分で距離が近すぎる場合は移動しないようにしています。理由はお互いにトリガーが発動して無限に行き来するからです。
if (Vector2.Distance(transform.position, collision.transform.position) > distance)
出入口が異なる場合
以前作成したパックマン風ゲームで利用する場合のパターンを実装してみます。左の緑色に入ると、右の赤色から出るような感じで、入口と出口が異なります。
data:image/s3,"s3://crabby-images/1a1b5/1a1b5ccfbed6dc795ef55d7cf736fdc4713963c3" alt=""
この場合のスクリプトは下記になります。入口と出口が別々なので、距離のチェックは不要です。
using UnityEngine; public class Portal : MonoBehaviour { [SerializeField] private Transform destination; private void OnTriggerEnter2D(Collider2D collision) { collision.transform.position = new Vector2(destination.position.x, destination.position.y); } }
移動先(出口)のパラメータをセットします。
data:image/s3,"s3://crabby-images/83ffe/83ffe6bcb950a9f9d8fae22f4f0d97d6711104db" alt=""
実行してみると下記のような感じに。
data:image/s3,"s3://crabby-images/c047e/c047ee192b5bf57a55773fafac44afc7e355ce5f" alt=""