4
4
記事投稿キャンペーン 「2024年!初アウトプットをしよう」

Unity で C# スクリプト:敵を撃破する3Dゲームを作る

Last updated at Posted at 2024-01-06

Unity で C# スクリプト:敵を撃破する3Dゲームを作る

こんにちは、@studio_meowtoon です。今回は Windows 11 の Unity アプリ開発環境で、C# 言語で原始的な3Dゲームを作成する方法を紹介します。
csharp_in_unity.png

目的

Unity エディターで C# 言語を用いて原始的な3Dゲームを作成します。

Unity を使った簡単な3Dゲーム開発でも、通常は3Dモデルが必要です。この記事では Unity エディターのみを使って基本的な3Dゲームの骨組みを作り、理解を深めます。

ダウンロード

この記事の内容の Unity プロジェクトは、GitHub からダウンロードできます。

開発環境

  • Windows 11 Home 22H2
  • Unity 2022.3.10f1 LTS
  • Visual Studio Community 2019 Version 16.11.19

Unity プロジェクトの作成

Unity Hub を起動します。

image.png

[新しいプロジェクト] ボタンを押します。

image.png

[3D コア] ボタンを押します。

image.png

プロジェクト名を入力して、[プロジェクトを作成] ボタンを押します。

image.png

空の Unity プロジェクトが立ち上がります。

image.png

ここまでの作業で空の Unity プロジェクトを作成することができました。

3Dゲームの実装

地面を作る

左の [ヒエラルキー] タブ内で右クリメニューを表示して、[3D オブジェクト] > [キューブ] と選択します。

image.png

ここでは、[Cube] を、[Ground] とリネームしました。

image.png

右の [インスペクター] タブ内の [Transform] のスケールで X: 100、Z: 100 と入力します。また、位置で Y: -0.5 と入力します。

image.png

Unity の単位 1 は 1m になります。ここまでの作業で、100m四方の地面を作成することができました。

地面に色を塗る

マテリアルの作成

下部の [プロジェクト] タブから [Assets] > [Materials] フォルダを作成します。

image.png

[Materials] フォルダで右クリメニューから、[作成] > [マテリアル] と選択します。

image.png

ここでは、[ground_green] という名前を付けました。

image.png

右の [インスペクター] タブ内の [Main Maps] 項目の [アルベド] をクリックします。

image.png

[色] 選択ダイアログで [HSV] に切り替えます。

image.png

ここでは、H: 120、S: 50、V: 60 と設定しました。

image.png

このような緑色のマテリアルが作成されました。

image.png

マテリアルを適用

[ground_green] マテリアルをつかみ、[シーン] タブ内の [Ground] オブジェクトにドロップします。

image.png

[Ground] オブジェクトに [ground_green] マテリアルが適用されました。

image.png

ここまでの作業で、緑色の地面を作成することができました。

自機を作る

左の [ヒエラルキー] タブ内で右クリメニューを表示して、[3D オブジェクト] > [キューブ] と選択し、[Player] オブジェクトを作成します。

image.png

右の [インスペクター] タブ内の [Transform] の位置で Y: 0.5 と入力します。

image.png

ここまでの作業で、Player オブジェクトを作成することができました。

自機を操作する

Rigidbody コンポーネントを追加

Rigidbody を使うと、ゲームオブジェクト を物理特性によって制御する事ができるようになります。

[Player] オブジェクトを選択します。右の [インスペクター] タブ内の [コンポーネントを追加] ボタンを押します。

image.png

検索から [Rigidbody] を選択して、[Rigidbody] ボタンを押します。

image.png

[Player] の [インスペクター] タブ内に [Rigidbody] コンポーネントが追加されました。回転を固定を全てチェックします。

image.png

カメラの設定

メインカメラを自機の子オブジェクトします。

[ヒエラルキー] の [Main Camera] をドラッグして、

image.png

[Player] オブジェクトの子オブジェクトに設定します。

image.png

カメラが映す角度を調整します。

右の [インスペクター] タブ内の [Transform] の位置で Y: 1.5、Z: -4 と入力します。また、回転で X: 15 と入力します。

image.png

[ゲーム] タブで確認すると、カメラが自機を後方から捉える視点となります。

image.png

Input System のインポート

操作したキーを入力値として扱います。

[ウィンドウ] > [パッケージマネージャー] を選択します。

image.png

[パッケージ:Unity レジストリ] から [Input System] を検索します。[Input System] を選択して [インポート] ボタンを押します。

image.png

[ヒエラルキー] タブ内で右クリメニューから、[UI] > [イベントシステム] を選択します。

image.png

[EventSystem] オブジェクトが追加されます。

image.png

UniRx ライブラリのインポート

イベント処理に特化した UniRx ライブラリを導入します。

[ウィンドウ] > [アセットストア] を選択します。

image.png

[Search online] ボタンを押します。

image.png

UniRx をマイアセットに追加します。

image.png

[ウィンドウ] > [パッケージマネージャー] を選択します。

[パッケージ:マイアセット] から、[UniRx] を選択して [インポート] ボタンを押します。

image.png

[Examples] フォルダのチェックを外して、[インポート] ボタンを押します。

image.png

[プロジェクト] タブ内の [Assets] > [Plugins] フォルダに [UniRX] がインストールされます。

image.png

スクリプトの実装

Player オブジェクトを操作する C# スクリプトを作成します。

[Assets] > [Scripts] フォルダを作成します。

image.png

[Scripts] フォルダで右クリメニューから、[作成] > [C# スクリプト] と選択します。

image.png

ここでは、[Player] という名前のスクリプトを作成しました。

image.png

スクリプトを実装します。

Assets\Scripts\Player.cs
using UnityEngine;
using UnityEngine.InputSystem;
using UnityEngine.InputSystem.Controls;
using static UnityEngine.GameObject;
using UniRx;
using UniRx.Triggers;

namespace GameDev {
    /// <summary>
    /// player script.
    /// </summary>
    public class Player : MonoBehaviour {

        ///////////////////////////////////////////////////////////////////////////////////////////////
        // Fields [noun, adjectives] 

        ButtonControl _a_button, _b_button, _up_button, _down_button, _left_button, _right_button;

        // Start is called before the first frame update.
        void Start() {
            /// <summary>
            /// キー入力を取得します
            /// </summary>
            this.UpdateAsObservable()
                .Subscribe(onNext: _ => {
                    mapGamepad();
                }).AddTo(gameObjectComponent: this);

            /// <summary>
            /// Rigidbody コンポーネントを取得します
            /// </summary>
            Rigidbody rb = transform.GetComponent<Rigidbody>();

            /// <summary>
            /// 速度を取得します
            /// </summary>
            float speed = 0f;
            this.FixedUpdateAsObservable()
                .Subscribe(onNext: _ => {
                    speed = rb.velocity.magnitude;
                }).AddTo(gameObjectComponent: this);

            /// <summary>
            /// 前進します
            /// </summary>
            bool forward = false; // 前進フラグ
            float SPEED_LIMIT = 5.0f; // スピード制限フラグ
            this.UpdateAsObservable()
                .Where(predicate: _ => 
                    _b_button.isPressed && // 上ボタンが押されて かつ
                    speed < SPEED_LIMIT) // スピード制限以下の時
                .Subscribe(onNext: _ => {
                    forward = true; // 前進フラグをONにする
                }).AddTo(gameObjectComponent: this);

            float FORWARD_FORCE = 7.5f; // 前進力
            this.FixedUpdateAsObservable()
                .Where(predicate: _ => 
                    forward) // 前進フラグがONの場合
                .Subscribe(onNext: _ => {
                    const float ADJUST_VALUE = 10.0f; // 調整値
                    rb.AddFor​​ce( // Rigidbody コンポーネントに力を加えます
                        force: transform.forward * FORWARD_FORCE * ADJUST_VALUE, // 力のベクトル
                        mode: ForceMode.Acceleration // 力をかけるモード: アクセラレーション
                    );
                    forward = false; // 前進フラグをOFFにする
                }).AddTo(gameObjectComponent: this);

            /// <summary>
            /// 回転します
            /// </summary>
            float ROTATIONAL_SPEED = 7.5f; // 回転スピード
            this.UpdateAsObservable()
                .Subscribe(onNext: _ => {
                    const float ADJUST_VALUE = 10.0f; // 調整値
                    int axis = _right_button.isPressed ? // 右ボタンが押されているか?
                        1 : // Yes: axis は 1
                        _left_button.isPressed ? // No: 左ボタンが押されているか?
                            -1 : // Yes: axis は -1
                            0; // No: axis は 0 (※左右キーは押されていない)
                    transform.Rotate( // オブジェクトを回転します
                        xAngle: 0, 
                        yAngle: axis * (ROTATIONAL_SPEED * Time.deltaTime) * ADJUST_VALUE, // Y軸(ヨー)で回転する
                        zAngle: 0
                    );
                }).AddTo(gameObjectComponent: this);
        }

        ///////////////////////////////////////////////////////////////////////////////////////////////
        // private Methods [verb]

        /// <summary>
        /// キー入力を取得します
        /// </summary>
        void mapGamepad() {
            // PCのキーボード入力を取得
            if (Application.platform == RuntimePlatform.WindowsPlayer || Application.platform == RuntimePlatform.WindowsEditor) {
                _up_button = Keyboard.current.upArrowKey; // キーボードの上カーソルキー
                _down_button = Keyboard.current.downArrowKey; // キーボードの下カーソルキー
                _left_button = Keyboard.current.leftArrowKey; // キーボードの左カーソルキー
                _right_button = Keyboard.current.rightArrowKey; // キーボードの右カーソルキー
                _b_button = Keyboard.current.zKey; // キーボードのZキー
                _a_button = Keyboard.current.xKey; // キーボードのXキー
                return;
            }
        }
    }
}

オブジェクトにスクリプトを追加します。

[Player] スクリプトをつかみ、[シーン] タブの [Player] オブジェクトにドロップします。

image.png

[Player] の [インスペクター] タブ内に [Player] スクリプトが追加されました。

image.png

ゲームを起動する

[ゲーム] タブの [▶] ボタンを押すと、このゲームを遊ぶことができます。

image.png

ここまでの作業で、自機である Player オブジェクトをPCのキーボード入力から操作することができました。

障害物を作る

自機を操作している実感を得るために、地面に障害物を作ってみます。

マテリアルを作成

[object_black] マテリアルを作成します。

image.png
色を作成します。

image.png
このようなマテリアルを作成しました。

image.png

オブジェクトを作成

[3D オブジェクト] > [キューブ] から、[Object] オブジェクトを作成します。

image.png

右の [インスペクター] タブ内の [Transform] の位置で X:2、Y: 1、Z: 2 と入力します。また、スケールで X:2、Y: 2、Z: 2 と入力します。

image.png

[object_black] マテリアルをつかみ、[シーン] タブの [Object] オブジェクトにドロップします。

image.png

[Object] オブジェクトにマテリアルが適用されました。

image.png

プレハブに適用

Unity の Prefab (プレハブ) 機能で、プレハブと呼ばれるオブジェクトのテンプレートを作成することができます。

[Assets] > [Prefabs] フォルダを作成します。

image.png

[ヒエラルキー] タブの [Obuject] オブジェクトを、[Prefabs] フォルダにドロップします。

image.png

右の [インスペクター] タブ内の [Transform] の位置で X:0、Y: 0、Z: 0 と入力します。

image.png

ここまでの作業で障害物オブジェクトのプレハブを作成することができました。

オブジェクトを配置

[Prefabs] フォルダの [Obuject] オブジェクトを [シーン] タブの地面オブジェクトにドラッグ&ドロップします。

image.png

[シーン] にプレハブから作成したオブジェクトを配置することができました。

image.png

[Obuject_n] オブジェクトをリネームしました。

image.png

ここまでの作業で、地面に複数の障害物オブジェクトを配置することができました。

弾を発射する

マテリアルを作成

[bullet_yellow] という名前のマテリアルを作成します。 (HSL: 60, 80, 100)

image.png

オブジェクトを作成

[ヒエラルキー] タブ内で右クリメニューから、[3D オブジェクト] > [スフィア] を選択します。

image.png

ここでは、[Bullet] という名前のオブジェクトを作成しました。(スケール X: 0.75、Y: 0.75、Z: 0.75)

image.png

[bullet_yellow] マテリアルを [Bullet] オブジェクトにドロップします。

image.png

検索から [Rigidbody] を選択して、[Rigidbody] ボタンを押します。

image.png

[Bullet] オブジェクトに [Rigidbody] が追加されました。

image.png

プレハブに適用

[ヒエラルキー] タブの [Bullet] オブジェクトを、[Prefabs] フォルダにドロップして、プレハブを作成します。

image.png

また、[ヒエラルキー] タブの [Bullet] オブジェクトを削除します。

image.png

スクリプトの実装

以下の内容を [Player] スクリプトに追加しました。

Assets\Scripts\Player.cs
// ※省略

namespace GameDev {
    /// <summary>
    /// player script.
    /// </summary>
    public class Player : MonoBehaviour {

        ///////////////////////////////////////////////////////////////////////////////////////////////
        // References [bool => is+adjective, has+past participle, can+verb prototype, triad verb]

        [SerializeField] GameObject _bullet; // 弾のプレハブを設定します

        // ※省略

        // Start is called before the first frame update
        void Start() {
            // ※省略

            /// <summary>
            /// 弾を撃ちます
            /// </summary>
            this.UpdateAsObservable()
                .Where(predicate: _ =>  
                    _a_button.wasPressedThisFrame)
                .Subscribe(onNext: _ => {
                    shoot(); // 弾を撃つ
                }).AddTo(gameObjectComponent: this);
        }
        
        // ※省略

        ///////////////////////////////////////////////////////////////////////////////////////////////
        // private Methods [verb]

        // ※省略

        /// <summary>
        /// 弾を撃ちます
        /// </summary>
        void shoot() {
            float BULLET_SPEED = 7500.0f; // 弾の速度

            // 弾の複製
            GameObject bullet = Instantiate(original: _bullet);

            // 弾の位置
            Vector3 position = transform.position + (transform.forward * 1.0f); // ノズル前方
            bullet.transform.position = position;

            // 弾へ加える力
            Vector3 force = (transform.forward + Vector3.up * 0.01f) * BULLET_SPEED; // ごくわずかに上向き

            // 弾を発射
            bullet.GetComponent<Rigidbody>().AddForce(force: force, mode: ForceMode.Force);
        }
    }
}

[ヒエラルキー] タブ内で [Player] オブジェクトを選択して、[Player] スクリプトの Bullet 参照に [Bullet] プレハブをドロップします。

image.png

ここまでの作業で、Player オブジェクトから弾を発射することができました。

弾で障害物を破壊する

スクリプトの実装

破壊されるオブジェクトを操作するスクリプトを作成します。

ここでは、[Assets] > [Scripts] > [Destroyable] スクリプトを作成をしました。

image.png

Assets\Scripts\Destroyable.cs
using System.Linq;
using UnityEngine;
using UniRx;
using UniRx.Triggers;

namespace GameDev {
    /// <summary>
    /// destroyable script.
    /// </summary>
    public class Destroyable : MonoBehaviour {

        ///////////////////////////////////////////////////////////////////////////////////////////////
        // References [bool => is+adjective, has+past participle, can+verb prototype, triad verb]

        [SerializeField] int ENDURANCE = 3; // 耐久値

        ///////////////////////////////////////////////////////////////////////////////////////////////
        // Fields [noun, adjectives]

        bool _auto_destroy = false; // 自動消去フラグ

        ///////////////////////////////////////////////////////////////////////////////////////////////
        // Properties [noun, adjectives]

        public bool AutoDestroy { set => _auto_destroy = value; }

        ///////////////////////////////////////////////////////////////////////////////////////////////
        // update Methods

        // Start is called before the first frame update.
        void Start() {
            /// <summary>
            /// 弾としてヒットしたとき
            /// </summary>
            this.OnCollisionEnterAsObservable()
                .Where(predicate: x =>
                    gameObject.name.Contains(value: "Bullet") && // 自分の名前に "Bullet" が含まれる かつ
                    !x.gameObject.name.Contains(value: "Ground")) // 名前に "Ground" が含まれないオブジェクトに衝突したとき
                .Subscribe(onNext: x => {
                    Rigidbody rb = x.gameObject.GetComponent<Rigidbody>();
                    if (rb != null) {
                        rb.velocity = Vector3.zero; // ヒットした相手を一時停止
                    }
                    Destroy(obj: gameObject); // 自分を削除
                }).AddTo(gameObjectComponent: this);

            /// <summary>
            /// 弾がヒットしたとき
            /// </summary>
            this.OnCollisionEnterAsObservable()
                .Where(predicate: x => 
                    x.gameObject.name.Contains(value: "Bullet")) // 名前に "Bullet" が含まれるオブジェクトが衝突したとき
                .Subscribe(onNext: x => {
                    ENDURANCE--; // 耐久度を減らす
                }).AddTo(gameObjectComponent: this);

            /// <summary>
            /// 耐久度がゼロになったとき
            /// </summary>
            this.UpdateAsObservable()
                .Where(predicate: _ => 
                    ENDURANCE == 0 && // 耐久度がゼロ かつ
                    !name.Contains(value: "_Piece")) // 名前に "_Piece" が含まれない場合には
                .Subscribe(onNext: _ => {
                    explodePiece(); // 破片を生成する
                    ENDURANCE = -1; // 耐久度を無効化
                    Destroy(obj: gameObject); // 自分を削除
                }).AddTo(gameObjectComponent: this);

            /// <summary>
            /// 自動消去フラグがONになったとき
            /// </summary>
            this.UpdateAsObservable()
                .Where(predicate: _ => 
                    _auto_destroy == true && // 自動消去フラグがON かつ
                    name.Contains(value: "_Piece")) // 名前に "_Piece" が含まれる破片である場合には
                .Subscribe(onNext: _ => {
                    // 破片を消去する
                    Observable
                        .TimerFrame(dueTimeFrameCount: Mathf.FloorToInt(f: Random.Range(minInclusive: 2500, maxInclusive: 5000)))
                        .Subscribe(onNext: _ => {
                            Destroy(obj: gameObject); // 自分(破片)を削除
                        }).AddTo(gameObjectComponent: this);
                }).AddTo(gameObjectComponent: this);
        }

        ///////////////////////////////////////////////////////////////////////////////////////////////
        // private Methods [verb]

        /// <summary>
        /// 破片を生成します
        /// </summary>
        void explodePiece(int number = 12, float scale = 0.5f, int force = 5) {
            // 複数の破片を作成
            for (int i = 0; i < number; i++) {
                // 破片オブジェクトを作成
                GameObject piece = Instantiate(original: gameObject); // 自分を複製して破片とする
                piece.name += $"_Piece_{i + 1}"; // 破片には名前に "_Piece_n" を付加する
                piece.transform.localScale = new Vector3(x: scale, y: scale, z: scale); // 縮小する
                if (piece.GetComponent<Rigidbody>() == null) {
                    piece.AddComponent<Rigidbody>(); // Rigidbody コンポーネントを持たなければ追加する
                }
                piece.GetComponent<Rigidbody>().isKinematic = false;
                piece.GetComponent<Rigidbody>().mass = 1.0f;
                // ランダム値のベクトルを取得
                Vector3 random_force = new Vector3(
                    x: Random.Range(minInclusive: force / 2, maxInclusive: force * 2), 
                    y: Random.Range(minInclusive: force / 2, maxInclusive: force * 2), 
                    z: Random.Range(minInclusive: force / 2, maxInclusive: force * 2)
                );
                // 破片に力と回転を加える
                piece.GetComponent<Rigidbody>().AddForce(force: random_force, mode: ForceMode.Impulse);
                piece.GetComponent<Rigidbody>().AddTorque(torque: random_force, mode: ForceMode.Impulse);
                // 破片の子オブジェクトを削除
                piece.GetComponentsInChildren<Transform>().ToList().ForEach(x => {
                    if (piece.name != x.name) { // 破片も破片の子リストにいるので除外
                        x.parent = null;
                        Destroy(obj: x.gameObject); // 破片の子オブジェクトは最初に削除
                    }
                });
                // 破片の自動消去フラグをON
                piece.GetComponent<Destroyable>().AutoDestroy = true;
            }
        }
    }
}

ここまでの作業で、Player オブジェクトから発射した弾で、障害物オブジェクトを破壊することができました。

敵を作る

マテリアルを作成

[Assets] > [Materials] > [enemy_blue] マテリアルを作成します。(HSV: 210, 90, 90)

image.png

オブジェクトを作成

[ヒエラルキー] 右クリメニュー > [3D オブジェクトを] > [キューブ] を選択して [Enemy] オブジェクトを作成します。また、[enemy_blue] マテリアルを [Enemy] オブジェクトにドロップします。

image.png

プレハブに適用

[ヒエラルキー] タブ内の [Enemy] オブジェクトを、[Prefabs] フォルダにドロップして、プレハブを作成します。

image.png

Rigidbody コンポーネントを追加します。回転を固定を全てチェックします。

image.png

敵が自機を索敵するために使用する、スフィアコライダー コンポーネントを追加します。

image.png

[トリガーにする] にチェックを入れます。半径を 15 に設定します。

image.png

スクリプトを実装

敵オブジェクトを操作するスクリプトを作成します。

ここでは、[Assets] > [Scripts] > [Enemy] スクリプトを作成をしました。

image.png

Assets\Scripts\Enemy.cs
using UnityEngine;
using static UnityEngine.GameObject;
using UniRx;
using UniRx.Triggers;

namespace GameDev {
    /// <summary>
    /// enemy script.
    /// </summary>
    public class Enemy : MonoBehaviour {

        ///////////////////////////////////////////////////////////////////////////////////////////////
        // References [bool => is+adjective, has+past participle, can+verb prototype, triad verb]

        [SerializeField] GameObject _bullet; // 弾のプレハブを設定します

        ///////////////////////////////////////////////////////////////////////////////////////////////
        // Fields [noun, adjectives] 

        float SEARCH_ANGLE = 60.0f; // 索敵アングル(※実際の範囲は2倍になります)

        GameObject _player_object;

        ///////////////////////////////////////////////////////////////////////////////////////////////
        // Properties [noun, adjectives] 

        public bool NotPiece { get => !name.Contains(value: "_Piece"); } // 破片ではないフラグ

        ///////////////////////////////////////////////////////////////////////////////////////////////
        // update Methods

        // Awake is called when the script instance is being loaded.
        void Awake() {
            _player_object = Find(name: "Player"); // Player オブジェクトの参照を取得
        }

        // Start is called before the first frame update.
        void Start() {
            /// <summary>
            /// Rigidbody コンポーネントを取得します
            /// </summary>
            Rigidbody rb = transform.GetComponent<Rigidbody>();

            /// <summary>
            /// 速度を取得します
            /// </summary>
            float speed = 0f;
            this.FixedUpdateAsObservable()
                .Subscribe(onNext: _ => {
                    speed = rb.velocity.magnitude;
                }).AddTo(gameObjectComponent: this);

            /// <summary>
            /// 索敵中に回転します
            /// </summary>
            bool searching = true; // 索敵中フラグ
            bool rotate = false; // 回転中フラグ
            bool stop = false; // 停止フラグ
            this.UpdateAsObservable()
                .Where(predicate: _ => 
                    searching && !rotate && // 索敵中フラグONかつ回転中ではない場合
                    NotPiece)
                .Subscribe(onNext: _ => {
                    // ランダムで移動位置指定
                    int random1 = Mathf.FloorToInt(f: Random.Range(minInclusive: 0.0f, maxInclusive: 8.0f));
                    int random2 = Mathf.FloorToInt(f: Random.Range(minInclusive: 0.1f, maxInclusive: 4.0f));
                    int random3 = Mathf.FloorToInt(f: Random.Range(minInclusive: -3.0f, maxInclusive: 3.0f));
                    int random4 = Mathf.FloorToInt(f: Random.Range(minInclusive: -3.0f, maxInclusive: 3.0f));
                    rotate = true; // 回転中フラグON
                    // random1 秒後に
                    Observable.Timer(System.TimeSpan.FromSeconds(value: random1))
                        .Subscribe(onNext: _ => {
                            stop = true; // 一時停止して
                            // さらに random2 秒後に
                            Observable.Timer(System.TimeSpan.FromSeconds(value: random2))
                                .Subscribe(onNext: _ => {
                                    int random5 = Mathf.FloorToInt(f: Random.Range(minInclusive: 1.0f, maxInclusive: 6.0f));
                                    if (random5 % 2 == 1) { // 偶数なら
                                        // ランダムな方向を向く
                                        transform.LookAt(worldPosition: new Vector3(
                                            x: transform.position.x + random3,
                                            y: transform.position.y,
                                            z: transform.position.z + random4
                                        ));
                                        stop = false; rotate = false;
                                    } else { // 奇数ならそのままの方向を維持
                                        stop = false; rotate = false;
                                    }
                            }).AddTo(gameObjectComponent: this);
                        }).AddTo(gameObjectComponent: this);
                }).AddTo(gameObjectComponent: this);

            /// <summary>
            /// 一時停止します
            /// </summary>
            this.FixedUpdateAsObservable()
                .Where(predicate: _ => 
                    stop && // 停止フラグONの場合
                    NotPiece)
                .Subscribe(onNext: _ => {
                    rb.velocity = Vector3.zero;
                }).AddTo(gameObjectComponent: this);

            /// <summary>
            /// 索敵中に移動します
            /// </summary>
            float SPEED_LIMIT = 3.0f; // スピード制限フラグ
            float FORWARD_FORCE = 16.0f; // 前進力
            this.FixedUpdateAsObservable()
                .Where(predicate: _ => 
                    searching && !stop && // 索敵中フラグONかつ停止中でない場合
                    NotPiece)
                .Subscribe(onNext: _ => {
                    if (speed < SPEED_LIMIT) { // 速度リミットまで
                        rb.AddFor​​ce(force: transform.forward * FORWARD_FORCE, mode: ForceMode.Acceleration); // 前に力を加える
                    }
                }).AddTo(gameObjectComponent: this);

            /// <summary>
            /// 追跡中に回転します
            /// </summary>
            bool chasing = false; // 追跡中フラグ
            this.UpdateAsObservable()
                .Where(predicate: _ => 
                    chasing && // 追跡中フラグONの場合
                    NotPiece)
                .Subscribe(onNext: _ => {
                    // Player オブジェクトの方向に回転する
                    transform.LookAt(worldPosition: new Vector3(
                        x: _player_object.transform.position.x,
                        y: transform.position.y,
                        z: _player_object.transform.position.z
                    ));
                    shoot(); // 弾を撃つ
                }).AddTo(gameObjectComponent: this);

            /// <summary>
            /// 追跡中に異動します
            /// </summary>
            this.FixedUpdateAsObservable()
                .Where(predicate: _ => 
                    chasing && // 追跡中フラグONの場合
                    NotPiece)
                .Subscribe(onNext: _ => {
                    if (speed < SPEED_LIMIT) { // 速度リミットまで
                        rb.AddFor​​ce(force: transform.forward * FORWARD_FORCE, mode: ForceMode.Acceleration); // 前に力を加える
                    }
                }).AddTo(gameObjectComponent: this);

            /// <summary>
            /// 索敵用コライダーに接触したとき
            /// </summary>
            this.OnTriggerEnterAsObservable()
                .Where(predicate: x => 
                    x.name.Equals("Player"))
                .Subscribe(onNext: _ => {
                    chasing = discovered(target: _player_object); // 追跡フラグ判定
                }).AddTo(gameObjectComponent: this);

            /// <summary>
            /// 索敵用コライダーとの接触を維持しているとき
            /// </summary>
            this.OnTriggerStayAsObservable()
                .Where(predicate: x => 
                    x.name.Equals("Player"))
                .Subscribe(onNext: _ => {
                    chasing = discovered(target: _player_object); // 追跡フラグ判定
                }).AddTo(gameObjectComponent: this);

            /// <summary>
            /// 索敵用コライダーから離脱したとき
            /// </summary>
            this.OnTriggerExitAsObservable()
                .Where(predicate: x => 
                    x.name.Equals("Player"))
                .Subscribe(onNext: _ => {
                    chasing = false; // 追跡フラグOFF
                }).AddTo(gameObjectComponent: this);
        }

        ///////////////////////////////////////////////////////////////////////////////////////////////
        // private Methods [verb]

        /// <summary>
        /// 追跡フラグを判定します (※視界の角度内に収まっているか)
        /// </summary>
        bool discovered(GameObject target) {
            // 対象までのベクトルを取得
            Vector3 position_delta = target.transform.position - transform.position;
            // 前方ベクトルと対象までの角度を取得
            float target_angle = Vector3.Angle(from: transform.forward, to: position_delta);
            // 視界内に収まっている場合
            if (target_angle < SEARCH_ANGLE) { 
                return true; // 追跡フラグON
            }
            return false; // 追跡フラグOFF
        }

        /// <summary>
        /// 弾を撃ちます
        /// </summary>
        void shoot() {
            float BULLET_SPEED = 7500.0f; // 弾の速度

            // ランダム値を生成して3の時だけ弾を発射
            int random = Mathf.FloorToInt(f: Random.Range(minInclusive: 0.0f, maxInclusive: 1024.0f));
            if (random == 3.0f) {
                // 弾の複製
                GameObject bullet = Instantiate(original: _bullet);

                // 弾の位置
                Vector3 position = transform.position + (transform.forward * 1.0f); // ノズル前方
                bullet.transform.position = position;

                // 弾へ加える力
                Vector3 force = (transform.forward + Vector3.up * 0.01f) * BULLET_SPEED; // ごくわずかに上向き

                // 弾を発射
                bullet.GetComponent<Rigidbody>().AddForce(force: force, mode: ForceMode.Force);
            }
        }
    }
}

[Enemy] プレハブにスクリプトを設定します。

image.png

敵を配置する

[Enemy] プレハブを [シーン] に複数追加しました。

image.png

ここまでの作業で、敵オブジェクトをシーンに配置することができました。

自機の耐久度を設定

[Player] オブジェクトに [Destroyable] スクリプトを追加して、ENDURANCE (耐久度) を5に設定しました。

image.png

ここまでの作業で、障害物や敵オブジェクトを撃破するゲームを作成することができました。

このゲームをプレイした動画

Movie_002.gif

まだとても初歩的な段階ではありますが、敵を撃破する3Dゲームの実装に成功しました。

まとめ

Unity エディターで C# 言語を用いて原始的な3Dゲームを作成することができました。

今回の実装にはいくつかの問題が見受けられます。将来的な改善点については、別の機会に実装記事で紹介していく予定です。

この記事の実装例は一つのアプローチに過ぎず、必ずしも正しい方法とは限りません。他にも多様な方法がありますので、さまざまな情報を照らし合わせて検討してみてください。

どうでしたか? Window 11 の Unity で3Dゲームを開発する環境を手軽に構築することができます、ぜひお試しください。今後も Unity の開発トピックなどを紹介していきますので、ぜひお楽しみにしてください。

4
4
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
4
4