「Netcode for GameObjects」の中でも特に強力なリレー機能のを使ってオンラインマルチプレイ機能を実装しましょう。リレー機能の基本から応用まで、ゲーム開発におけるマルチプレイヤー環境構築を一緒に学びましょう。その全てをわかりやすく、具体的な例を交えながら解説します。実際に動かすものに関しては、以前のNetcodeで作ったサンプルを利用しますので、こちらでプロジェクトを作ってから今回の記事をご確認ください。
data:image/s3,"s3://crabby-images/4ec12/4ec121f81470307c5bb894e6cbbc345371e95aef" alt=""
Relay機能を実装する
Relayパッケージを導入する
パッケージマネージャーを開いて、Relayを探してください。必要があればアップデートを行いつつ、インストールを行います。(下図はインストール済みのためRemoveボタンになっています)
data:image/s3,"s3://crabby-images/3433c/3433c58c41d02163fc2854e3481368b6d8780cce" alt=""
Project設定を行う・Gaming Serivces
Edit>Project Settingを選択して、ウインドを表示します。Serviceタブを選択してください。以下、始めてプロジェクトを使う人と、既存のプロジェクトをお持ちの方で分岐します。
始めてプロジェクトを作る人
Organizationsから作る組織を選択します。個人の場合はUnityのアカウント名などを選択してください。選択すると[Create project ID]のボタンが押せるようになります。Create project IDボタンを押すと、Product Nameを元にUnity Service上にプロジェクトが作られます。
data:image/s3,"s3://crabby-images/10a9e/10a9e160c93608271dd7b7329eaa8fb1267dc726" alt=""
既存のプロジェクトを利用する場合
すでにService上にプロジェクトがあり、それを利用する場合は[Use an existing Unity project ID]から選択することができます。
Organizations > Projectsの順番に選択してください。2つを選択した状態で[Link project ID]ボタンからリンクさせてください。
data:image/s3,"s3://crabby-images/fb7bb/fb7bb416a3530158a8174c091f14464d44f7c1e6" alt=""
UnityServiceでRelay機能を使えるようにする
プロジェクトのリンクができたら、DashboardボタンからUnityサービスのダッシュボードページへ移動してください。ページに入ると左上の[Back to Gaming Services]ボタンを押してください。
data:image/s3,"s3://crabby-images/2074e/2074ee6f689e3934eb5ee7654e4713744ce344e1" alt=""
下図のようなページに移動したら、編集したいプロジェクトが選択されていることを確認し、左側のタブメニューから[Multiplayer]を選択してください。
data:image/s3,"s3://crabby-images/eb0b2/eb0b2b38a84e40fa3a25ef2d39b12568565fbb8a" alt=""
「Multiplayer>Relay」を選択し、対応したいプロジェクトが選択されていることを確認した状態で[Get Started]ボタンを押してください。
data:image/s3,"s3://crabby-images/54d5b/54d5b27415ccde50519d06e8d6fe08d6e4a42364" alt=""
設定自体は以上で終了!この後Setup guideが始まりますが、無視しても大丈夫です。
data:image/s3,"s3://crabby-images/99160/9916065731a4e4500d61b2fdc3fea30ea5e34667" alt=""
先にこちらの設定から来るパターンの場合、Relayのパッケージなどを入れたりするのも教えてくれます!
RelayTestスクリプト
長いけど、全部載せておきます。下のスクリプトはHost/ClientいずれかでRelayに接続するための処理が書かれています。スクリプトのポイントに関していくつか補足説明を記載します。
using UnityEngine;
using Unity.Services.Core;
using Unity.Services.Authentication;
using Unity.Services.Relay;
using Unity.Services.Relay.Models;
using Unity.Netcode;
using Unity.Netcode.Transports.UTP;
using TMPro;
public class RelayTest : MonoBehaviour
{
[SerializeField] TMP_InputField joinCodeInput;
private async void Start()
{
await UnityServices.InitializeAsync();
AuthenticationService.Instance.SignedIn += () =>
{
Debug.Log("Signed In " + AuthenticationService.Instance.PlayerId);
};
await AuthenticationService.Instance.SignInAnonymouslyAsync();
}
public async void CreateRelayButton()
{
try
{
Allocation allocation = await RelayService.Instance.CreateAllocationAsync(3);
string joinCode = await RelayService.Instance.GetJoinCodeAsync(allocation.AllocationId);
NetworkManager.Singleton.GetComponent<UnityTransport>().SetHostRelayData(
allocation.RelayServer.IpV4,
(ushort)allocation.RelayServer.Port,
allocation.AllocationIdBytes,
allocation.Key,
allocation.ConnectionData
);
Debug.Log(joinCode);
NetworkManager.Singleton.StartHost();
}
catch (RelayServiceException e)
{
Debug.Log(e);
}
}
public void JoinRelayButton()
{
JoinRelay(joinCodeInput.text);
}
public async void JoinRelay(string joinCode)
{
try
{
Debug.Log("JoinRelay code = " + joinCode);
JoinAllocation joinAllocation = await RelayService.Instance.JoinAllocationAsync(joinCode);
NetworkManager.Singleton.GetComponent<UnityTransport>().SetClientRelayData(
joinAllocation.RelayServer.IpV4,
(ushort)joinAllocation.RelayServer.Port,
joinAllocation.AllocationIdBytes,
joinAllocation.Key,
joinAllocation.ConnectionData,
joinAllocation.HostConnectionData
);
NetworkManager.Singleton.StartClient();
}
catch (RelayServiceException e)
{
Debug.Log(e);
}
}
}
スクリプトコンセプト
Startメソッドで不特定ユーザーでログインを行います。このあとボタン2つとテキスト入力を1つ作成し、CreateRelayButton/JoinRelayButtonを呼び出せるようにします。
CreateRelayではホストとしてゲームに参加します。GetJoinCodeAsyncでいわゆるルームに入るためのコードを発行します。今回はデバッグログに表示させて、目視でログインを行います。
一方クライアント側ではホストがJoinCodeを発行したのを確認したら、InputTextにコードを入力してJoinRelayButtonメソッド経由でゲームに参加を行います。
Unityに配置
空のGameObjectを作成し、RelayTestスクリプトをアタッチします。GameObjectの名前もわかりやすくRelayTestとかに変えておきましょう。
前回のプロジェクトの続きであれば、画面左上が空いているはずなので、そこにものを配置していきましょう。下図を参考にボタンを2つ、InputField(TextMeshPro版)を追加してください。
data:image/s3,"s3://crabby-images/c184c/c184c6c752f41b1ed72d5934646e10176dd71d18" alt=""
data:image/s3,"s3://crabby-images/c184c/c184c6c752f41b1ed72d5934646e10176dd71d18" alt=""
各ボタンのOnClickイベントは以下を設定。
ボタン | メソッド |
---|---|
Hostボタン | RelayTest.CreateRelayButton |
Clientボタン | RelayTest.JoinRelayButton |
OnClickに設定するGameObjectは先程作成したRelayTestをアタッチしたGameObjectです!
data:image/s3,"s3://crabby-images/c4ff4/c4ff406339d47b1de5338948a89672d596aa25dd" alt=""
data:image/s3,"s3://crabby-images/c4ff4/c4ff406339d47b1de5338948a89672d596aa25dd" alt=""
RelayTestコンポーネントにJoinCodeInputが空になっていると思うので、追加したUIのテキストインプットをセットしてください。手順通りの場合、一つだけなので右の丸ポチからセットしてもOK。
data:image/s3,"s3://crabby-images/0b6f6/0b6f6967add4bc960a014ff2a82dbd0ac13a4dbe" alt=""
data:image/s3,"s3://crabby-images/0b6f6/0b6f6967add4bc960a014ff2a82dbd0ac13a4dbe" alt=""
Relay Unity Transportへの切り替え
NetworkManagerゲームオブジェクトに一緒に貼りつているUnityTransportコンポーネントのProtcol Typeを[Relay Unity Transport]に変更します。
data:image/s3,"s3://crabby-images/f0e9f/f0e9f23e479361504e6ae8559676e9b425bcc580" alt=""
data:image/s3,"s3://crabby-images/f0e9f/f0e9f23e479361504e6ae8559676e9b425bcc580" alt=""
わざわざ1セクション用意したけど、これだけ!
実際に動かしてみた
今回の実行ではUnityエディタ側がホストになる必要があります。JoinCodeをデバッグログに表示するようにしているためです。
- クライアント用アプリを立ち上げて待つ
- Unityエディタ側も起動(この2つは順序逆転してもOK)
- Unityエディタ側のHostボタンを押す
- デバッグログにJoinCodeが表示されるのを確認
- アプリ側のInputTextにJoinCodeを入力し、Clientボタンを押す
- お互いに接続されたことを確認。キャラクター動かしたりして見てください
data:image/s3,"s3://crabby-images/ebab5/ebab50aab94f4f73ef20d9fa77a9db246617cf6d" alt=""
data:image/s3,"s3://crabby-images/ebab5/ebab50aab94f4f73ef20d9fa77a9db246617cf6d" alt=""
コメント
コメント一覧 (1件)
[…] リレーRelay機能を使って、マルチプレイ環境を作る![Unity+Netcode for GameObject… 「Netcode for […]