「自身が所有権を持つプレイヤー」だけを動かすようにする
目次
概要
プレイヤーキャラクタを操作している箇所を探す
PlayerにMonobitEngine.MonobitView.isMineを適用する
Recorderにも同様の処置を適用する
Actionにも同様の処置を適用する
DamageReceiverにも同様の処置を適用する
Player_Shootにも同様の処置を適用する
Karmaにも同様の処置を適用する
変更したプログラムの保存
なぜ「複数クライアントの動作」に不具合が出てしまったのか?
ここが「オフラインゲーム開発」から「オンラインゲーム開発」に移行する上での大きな障壁の1つなのですが。
前節までの流れで、単体テストで上手く動いていましたが、複数クライアントでのテストでは不具合が発生してしまいました。
なぜでしょう?
この原因については、以下の折り畳み箇所に記述します。
(難しい理屈ですので、ここは理解できなくても構いません。)
ネットワーク上のGameObjectは、MUNサーバから「オブジェクトの所有権」が与えられる
理由は分からなくとも構いませんが、対策方法は知っておく必要があります。
ネットワーク上の GameObject には、(厳密にいえば MonobitNetwork.Instantiate() を実行した瞬間に)
MUNサーバから「オブジェクトの所有権」が与えられます。
簡単に言えば「そのオブジェクトをプレイヤー自身が動かして良いかどうかの権限(フラグ)」です。
現在プレイヤーキャラクタを動かしている箇所では、まだ、その所有権(フラグ)をもとにした「操作の可否」を設定していません。
この「操作の可否判定」をプレイヤー制御のループ系メソッド(Update や OnGUI など)に組み込むことで、今回の不具合は解決します。
実際にどう対策するのか、具体的に触れていきましょう。
プレイヤーキャラクタを操作している箇所を探す▲
プレイヤーキャラクタのプレハブから「Dude」を選択する
まずは、プレイヤーキャラクタのプレハブから、
プレイヤーキャラクタの操作メソッドが含まれている「Dude」オブジェクトを選択しましょう。
この箇所は Unity 2018.2 以前と Unity2018.3 以降でインタフェースが異なりますので、
お使いの Unity のバージョン に合わせて、以下の項目を選択して進めてください。
PlayerにMonobitEngine.MonobitView.isMineを適用する▲
「Dude」のInspectorから「Player」を開く
Dude の Inspector には、プレイヤー制御に関するたくさんのコンポーネントが登録されています。
まずはこの中の、プレイヤーキャラクタの操作処理が含まれている「Player」コンポーネントの
スクリプトファイルをダブルクリックして開いてみましょう。
data:image/s3,"s3://crabby-images/999b9/999b942b3ae10ad44c5e49b34a021fbee65716fc" alt=""
該当するクラス内に、MonobitView コンポーネント用のフィールドを用意する
Player.cs を開き、「自身が所有権を持つプレイヤー」だけを動かすように、対策を施しましょう。
まず、該当するクラス内に、MonobitView コンポーネントを管理するための変数を用意します。
Player.cs の 12 行目付近に、以下のフィールドを追加してください。
// MonobitView コンポーネント
MonobitEngine.MonobitView m_MonobitView = null;
data:image/s3,"s3://crabby-images/a4036/a4036ee049d9f91ceef83e8e2c2ce2aede4e3116" alt=""
オブジェクト所有権のフラグ判定を行なうためには、MonobitView コンポーネント本体の参照が必要です。
そのため、事前に変数を用意します。
該当するクラスの Awake() で、MonobitView コンポーネントの情報を取得する
更に、Player.cs の 23 行目付近に、以下の Awake() メソッドのコードを追加します。
void Awake()
{
// すべての親オブジェクトに対して MonobitView コンポーネントを検索する
if (GetComponentInParent<MonobitEngine.MonobitView>() != null)
{
m_MonobitView = GetComponentInParent<MonobitEngine.MonobitView>();
}
// 親オブジェクトに存在しない場合、すべての子オブジェクトに対して MonobitView コンポーネントを検索する
else if (GetComponentInChildren<MonobitEngine.MonobitView>() != null)
{
m_MonobitView = GetComponentInChildren<MonobitEngine.MonobitView>();
}
// 親子オブジェクトに存在しない場合、自身のオブジェクトに対して MonobitView コンポーネントを検索して設定する
else
{
m_MonobitView = GetComponent<MonobitEngine.MonobitView>();
}
}
data:image/s3,"s3://crabby-images/62e28/62e28a900bd57ba0868070e464717f6786a615f2" alt=""
前述のMonobitTransformView のカスタマイズ でも説明いたしましたが、親オブジェクトに存在する MonobitView を検索し、
そのコンポーネントのデータを m_MonobitView に格納します。
該当するクラスの Update() について、MonobitView.isMineによる実行可否判定を入れる
更に、Player.cs の 50 行目付近に、以下のコードを書き加えます。
// オブジェクト所有権を所持しなければ実行しない
if( !m_MonobitView.isMine )
{
return;
}
data:image/s3,"s3://crabby-images/4c779/4c779797cb77dd5025a85aa000736d709a95b68a" alt=""
MonobitEngine.MonobitView.isMine が、「自身のクライアントはそのオブジェクトの所有権を持つか?」を示す、オブジェクト所有権のフラグです。
単純に、「自分が操作しているクライアントが所有権を持たない場合、そのキャラクタの Update() を処理させない」というロジックを適用させます。
これだけです。
Recorderにも同様の処置を適用する▲
「Dude」のInspector から「Recorder」を選択する
同様の処置を他のスクリプトにも適用させます。
続けて、プレイヤーキャラクタの行動記録制御処理が含まれている「Recorder」コンポーネントの
スクリプトファイルをダブルクリックしましょう。
data:image/s3,"s3://crabby-images/61625/616256fdc0cf83bff12572e1eb68b7f2ae9033b3" alt=""
該当するクラス内に、MonobitView コンポーネント用のフィールドを用意する
Recorder.cs の 10 行目付近に、MonobitView コンポーネントの変数を宣言します。
// MonobitView コンポーネント
MonobitEngine.MonobitView m_MonobitView = null;
data:image/s3,"s3://crabby-images/a417a/a417a48ce31998056b6a14eb6892794def10ebd8" alt=""
Player.csと同様、オブジェクト所有権のフラグ判定を行なうために
MonobitView コンポーネント本体の参照用変数を用意します。
該当するクラスの Awake() で、MonobitView コンポーネントの情報を取得する
更に、Recorder.cs の 23 行目付近に、以下の Awake() メソッドのコードを追加します。
void Awake()
{
// すべての親オブジェクトに対して MonobitView コンポーネントを検索する
if (GetComponentInParent<MonobitEngine.MonobitView>() != null)
{
m_MonobitView = GetComponentInParent<MonobitEngine.MonobitView>();
}
// 親オブジェクトに存在しない場合、すべての子オブジェクトに対して MonobitView コンポーネントを検索する
else if (GetComponentInChildren<MonobitEngine.MonobitView>() != null)
{
m_MonobitView = GetComponentInChildren<MonobitEngine.MonobitView>();
}
// 親子オブジェクトに存在しない場合、自身のオブジェクトに対して MonobitView コンポーネントを検索して設定する
else
{
m_MonobitView = GetComponent<MonobitEngine.MonobitView>();
}
}
data:image/s3,"s3://crabby-images/271dd/271ddf3db53e86dcab29a416aca9dcd14e9cf604" alt=""
これも前述同様、親オブジェクトに存在する MonobitView を検索し、
そのコンポーネントのデータを m_MonobitView に格納します。
該当するクラスの OnGUI() について、実行しないようにする。
更に、Recorder.cs の 57 行目付近に、以下のコードを追記します。
// このUIは必要ないので実行しない
return;
data:image/s3,"s3://crabby-images/4d2f0/4d2f042d33509166221603bd765357c8ab343948" alt=""
UIで処理できる内容を同期させないように変更を加えます。
該当するクラスの Update() について、MonobitView.isMineによる実行可否判定を入れる
Recorder.cs の 135 行目付近にも、以下のコードを追加しましょう。
// オブジェクト所有権を所持しなければ実行しない
if( !m_MonobitView.isMine )
{
return;
}
data:image/s3,"s3://crabby-images/8f382/8f382b71753709aacb6478d136249e1421775028" alt=""
ここも「所有権を持たないキャラクタは、Update() を処理させない」というロジックを適用させます。
Actionにも同様の処置を適用する▲
「Dude」のInspector から「Action」を選択する
同様に、プレイヤーキャラクタのアニメーション切り替え処理が含まれている「Action」コンポーネントの
スクリプトファイルをダブルクリックしましょう。
data:image/s3,"s3://crabby-images/b4b84/b4b84e37363499cd6b391752741f549034af6c68" alt=""
該当するクラス内に、MonobitView コンポーネント用のフィールドを用意する
Action.cs の 14 行目付近に、MonobitView コンポーネントの変数を宣言します。
// MonobitView コンポーネント
MonobitEngine.MonobitView m_MonobitView = null;
data:image/s3,"s3://crabby-images/2f1ea/2f1eaaf176c8f0c3e26c4db969a0c61369cf9c85" alt=""
前述と同様、オブジェクト所有権のフラグ判定を行なうために、MonobitView コンポーネント本体の参照用変数を用意します。
該当するクラスの Awake() で、MonobitView コンポーネントの情報を取得する
更に、Action.cs の 32 行目付近に、以下の Awake() メソッドのコードを追加します。
void Awake()
{
// すべての親オブジェクトに対して MonobitView コンポーネントを検索する
if (GetComponentInParent<MonobitEngine.MonobitView>() != null)
{
m_MonobitView = GetComponentInParent<MonobitEngine.MonobitView>();
}
// 親オブジェクトに存在しない場合、すべての子オブジェクトに対して MonobitView コンポーネントを検索する
else if (GetComponentInChildren<MonobitEngine.MonobitView>() != null)
{
m_MonobitView = GetComponentInChildren<MonobitEngine.MonobitView>();
}
// 親子オブジェクトに存在しない場合、自身のオブジェクトに対して MonobitView コンポーネントを検索して設定する
else
{
m_MonobitView = GetComponent<MonobitEngine.MonobitView>();
}
}
data:image/s3,"s3://crabby-images/88560/885606b47fa2c49bef6310521ed571bccab48f54" alt=""
これも前述同様、親オブジェクトに存在する MonobitView を検索し、そのコンポーネントのデータを m_MonobitView に格納します。
該当するクラスの Update() について、MonobitView.isMineによる実行可否判定を入れる
Action.cs の 59 行目についても、以下のコードを記述します。
// オブジェクト所有権を所持しなければ実行しない
if( !m_MonobitView.isMine )
{
return;
}
data:image/s3,"s3://crabby-images/28d67/28d67e455340ef80fc72fc1d26755807109bea17" alt=""
ここも「所有権を持たないキャラクタは、Update() を処理させない」というロジックを適用させます。
DamageReceiverにも同様の処置を適用する▲
「Dude」のInspector から「DamageReceiver」を選択する
同様に、プレイヤーキャラクタのダメージアクション制御が含まれている「DamageReceiver」コンポーネントの
スクリプトファイルをダブルクリックしましょう。
data:image/s3,"s3://crabby-images/e952a/e952a715086b469c01ace61d26cdaa7fff370582" alt=""
該当するクラス内に、MonobitView コンポーネント用のフィールドを用意する
DamageReceiver.cs の 11 行目付近に、MonobitView コンポーネントの変数を宣言します。
// MonobitView コンポーネント
MonobitEngine.MonobitView m_MonobitView = null;
data:image/s3,"s3://crabby-images/13605/13605282a731a882bb22c3a6bf437d179ca537a4" alt=""
前述と同様、オブジェクト所有権のフラグ判定を行なうために、MonobitView コンポーネント本体の参照用変数を用意します。
該当するクラスの Awake() で、MonobitView コンポーネントの情報を取得する
更に、DamageReceiver.cs の 32 行目付近に、以下の Awake() メソッドのコードを追加します。
void Awake()
{
// すべての親オブジェクトに対して MonobitView コンポーネントを検索する
if (GetComponentInParent<MonobitEngine.MonobitView>() != null)
{
m_MonobitView = GetComponentInParent<MonobitEngine.MonobitView>();
}
// 親オブジェクトに存在しない場合、すべての子オブジェクトに対して MonobitView コンポーネントを検索する
else if (GetComponentInChildren<MonobitEngine.MonobitView>() != null)
{
m_MonobitView = GetComponentInChildren<MonobitEngine.MonobitView>();
}
// 親子オブジェクトに存在しない場合、自身のオブジェクトに対して MonobitView コンポーネントを検索して設定する
else
{
m_MonobitView = GetComponent<MonobitEngine.MonobitView>();
}
}
data:image/s3,"s3://crabby-images/c406d/c406d22b9d6de1226275f91a516fadc311176b06" alt=""
これも前述同様、親オブジェクトに存在する MonobitView を検索し、そのコンポーネントのデータを m_MonobitView に格納します。
該当するクラスの Update() について、MonobitView.isMineによる実行可否判定を入れる
さらに、DamageReceiver.cs の 47 行目付近にも、以下のコードを追記します。
// オブジェクト所有権を所持しなければ実行しない
if( !m_MonobitView.isMine )
{
return;
}
data:image/s3,"s3://crabby-images/07f69/07f69dd75cb69fe5091772526a7f799d08727c38" alt=""
ここも「所有権を持たないキャラクタは、Update() を処理させない」というロジックを適用させます。
Player_Shootにも同様の処置を適用する▲
「Dude」のInspector から「Player_Shoot」を選択する
同様に、プレイヤーキャラクタの武器であるショット弾の発射制御が含まれている
「Player_Shoot」コンポーネントのスクリプトファイルをダブルクリックしましょう。
data:image/s3,"s3://crabby-images/0dec5/0dec5b9daf3d9acda3b79a5241f3694c5d8da044" alt=""
該当するクラス内に、MonobitView コンポーネント用のフィールドを用意する
Player_Shoot.cs の 10 行目付近に、MonobitView コンポーネントの変数を宣言します。
// MonobitView コンポーネント
MonobitEngine.MonobitView m_MonobitView = null;
data:image/s3,"s3://crabby-images/15383/153834d5fcf8465c000e00d022bb00787f0ce114" alt=""
前述と同様、オブジェクト所有権のフラグ判定を行なうために、MonobitView コンポーネント本体の参照用変数を用意します。
該当するクラスの Awake() で、MonobitView コンポーネントの情報を取得する
更に、Player_Shoot.cs の 23 行目付近に、以下の Awake() メソッドのコードを追加します。
void Awake()
{
// すべての親オブジェクトに対して MonobitView コンポーネントを検索する
if (GetComponentInParent<MonobitEngine.MonobitView>() != null)
{
m_MonobitView = GetComponentInParent<MonobitEngine.MonobitView>();
}
// 親オブジェクトに存在しない場合、すべての子オブジェクトに対して MonobitView コンポーネントを検索する
else if (GetComponentInChildren<MonobitEngine.MonobitView>() != null)
{
m_MonobitView = GetComponentInChildren<MonobitEngine.MonobitView>();
}
// 親子オブジェクトに存在しない場合、自身のオブジェクトに対して MonobitView コンポーネントを検索して設定する
else
{
m_MonobitView = GetComponent<MonobitEngine.MonobitView>();
}
}
data:image/s3,"s3://crabby-images/6ccc4/6ccc481b5969c9385a86773985d36083d90abd6c" alt=""
これも前述同様、親オブジェクトに存在する MonobitView を検索し、そのコンポーネントのデータを m_MonobitView に格納します。
該当するクラスの Update() について、MonobitView.isMineによる実行可否判定を入れる
さらに Player_Shoot.cs の 50 行目付近に、以下のコードを追記します。
// オブジェクト所有権を所持しなければ実行しない
if( !m_MonobitView.isMine )
{
return;
}
data:image/s3,"s3://crabby-images/c5446/c544608101211c21b00ed7e2d6482f011a0894f4" alt=""
ここも「所有権を持たないキャラクタは、Update() を処理させない」というロジックを適用させます。
Karmaにも同様の処置を適用する▲
「Dude」のInspector から「Karma」を選択する
同様に、プレイヤーキャラクタが地面から落下したあとの復帰処理が含まれている
「Karma」コンポーネントのスクリプトファイルをダブルクリックしましょう。
data:image/s3,"s3://crabby-images/e13aa/e13aaa4e402619fb70e52b2dcac39059c7bf19f1" alt=""
該当するクラス内に、MonobitView コンポーネント用のフィールドを用意する
Karma.cs の 10 行目付近に、MonobitView コンポーネントの変数を宣言します。
// MonobitView コンポーネント
MonobitEngine.MonobitView m_MonobitView = null;
data:image/s3,"s3://crabby-images/b3a40/b3a402d35068285db696d9a43e1f7cbeec9b4853" alt=""
前述と同様、オブジェクト所有権のフラグ判定を行なうために、MonobitView コンポーネント本体の参照用変数を用意します。
該当するクラスの Awake() で、MonobitView コンポーネントの情報を取得する
更に、Karma.cs の 13 行目付近に、以下の Awake() メソッドのコードを追加します。
void Awake()
{
// すべての親オブジェクトに対して MonobitView コンポーネントを検索する
if (GetComponentInParent<MonobitEngine.MonobitView>() != null)
{
m_MonobitView = GetComponentInParent<MonobitEngine.MonobitView>();
}
// 親オブジェクトに存在しない場合、すべての子オブジェクトに対して MonobitView コンポーネントを検索する
else if (GetComponentInChildren<MonobitEngine.MonobitView>() != null)
{
m_MonobitView = GetComponentInChildren<MonobitEngine.MonobitView>();
}
// 親子オブジェクトに存在しない場合、自身のオブジェクトに対して MonobitView コンポーネントを検索して設定する
else
{
m_MonobitView = GetComponent<MonobitEngine.MonobitView>();
}
}
data:image/s3,"s3://crabby-images/4aa20/4aa2054c429a11ce968e89a8085a3823fc5476e1" alt=""
これも前述同様、親オブジェクトに存在する MonobitView を検索し、そのコンポーネントのデータを m_MonobitView に格納します。
該当するクラスの Update() について、MonobitView.isMineによる実行可否判定を入れる
更に Karma.cs の 34 行目付近に、以下のコードを追記します。
// オブジェクト所有権を所持しなければ実行しない
if( !m_MonobitView.isMine )
{
return;
}
data:image/s3,"s3://crabby-images/da17c/da17ca495281b82699792ebd3fa27b0dad9ca8dd" alt=""
ここも「所有権を持たないキャラクタは、Update() を処理させない」というロジックを適用させます。
変更したプログラムの保存▲
ここまで変更を加えた全スクリプトを保存する
変更したスクリプトファイルを、すべて保存しましょう。
Visual Studio のメニューから [ファイル] > [すべて保存] を選択し、保存してください。
data:image/s3,"s3://crabby-images/ca956/ca95614a0c5368e1994f89c64f506b170a634ba5" alt=""