環境
Unity 2018.3.0f2
UniRx 6.2.2
UniRxのUniTaskを使うため2018.3以上が好ましい
サンプル
背景
ハッカソンとかでSceneManager.LoadSceneを使っているのをよく見る
シーンが重いと固まるため、普段からLoadSceneAsync使った方が良いと思う
LoadSceneAsyncだとシーン遷移の間バリア作らないといけないのが面倒
=> async使えるようになったしサンプル作って公開しよう!
サンプルの設計について
サンプルではSampleTitle, SampleMain, SampleResultという3つのシーンを作りました。
そして、各シーンにそれぞれシーン名と同名のクラスを作って置いておきます。
本記事ではこれらをシーンクラスと呼びます。
このシーンクラスがそのシーンにおける処理の起点となります。
そして基本的にはAwake, Start関数の使用は禁止です。初期化は全てこのシーンクラスから行います。
ハッカソンでは複数人が一緒に作業するため、コードの場所、命名規則がぐちゃぐちゃになる、Start関数が乱立するといったことがあると思います。
上記のようなルールを作っておくことで、他の誰かが作ったシーンのコード読むときはとりあえずそのシーンクラスを読み解けばいいので統一感がでます。
各シーンクラスはSceneBase.cs
を継承します。
シーンクラスはOnLoad
をOverrideすることで、前のシーンからのパラメータの受け取りや、初期化を行うことができます。
画面遷移
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UniRx.Async; // 必須
namespace CA2 {
public class SampleTitle : SceneBase {
public void GoMain() => SimpleSceneNavigator.Instance.GoForwardAsync<SampleMain>().Forget();
}
}
タイトルはメインに行く機能のみを持つため、最小限の画面遷移コードが書かれています。
SimpleSceneNavigator.Instance.GoForwardAsync<遷移先のシーンクラス>().Forget();
ポイントはシーンクラスを渡すことでシーン遷移を行うことです。
SimpleSceneNavigatorについて
ここが本題です。
using System.Collections;
using System.Collections.Generic;
using UniRx.Async;
using UnityEngine;
using UnityEngine.SceneManagement;
namespace CA2 {
public class SimpleSceneNavigator : MonoBehaviour {
public static SimpleSceneNavigator Instance = null;
void Awake () {
if(Instance != null){
Destroy(this.gameObject);
return;
}
DontDestroyOnLoad (this.gameObject);
Instance = this;
goLoadingBarrier.SetActive(false);
}
[SerializeField] GameObject goLoadingBarrier;
public async UniTask GoForwardAsync<T> (object options = null)
where T : SceneBase {
goLoadingBarrier.SetActive(true);
await SceneManager.LoadSceneAsync(typeof(T).Name);
goLoadingBarrier.SetActive(false);
var nextScene = Component.FindObjectOfType<T>();
if(nextScene == null)
throw new System.Exception(typeof(T).Name + " is Null");
nextScene.OnLoad(options);
}
}
}
SimpleSceneNavigatorの機能は以下の3つです。
- LoadSceneAsyncでシーン遷移
- シーン遷移中はバリアを貼る
- 遷移先のシーンクラスにパラメータを渡す
子にCanvasおよびBarrierを持ちます。
Barrierはタッチを防ぐためのものです。
サンプルではバリアが表示されていることをわかりやすく示すために半透明の黒いRawImageにしていますが、実際にゲーム作るときは色はいらないので透明で良いと思います。
このSceneNavigatorプレハブをシーン上にあらかじめ配置しておくことでシーン遷移が可能になります。
パラメータを渡す
メインからリザルトへの遷移のときには, リザルトシーンでスコア(10)を表示するためにパラメータを渡しています。
public class SampleMain : SceneBase {
public void GoResult(){
SimpleSceneNavigator.Instance.GoForwardAsync<SampleResult>(new SampleResult.Options(10)).Forget();
}
}
public class SampleResult : SceneBase {
public Text label;
public class Options {
public Options (int score) {
this.Score = score;
}
public int Score { get; private set; }
}
public override void OnLoad (object options) {
var op = options as Options;
label.text = op.Score.ToString();
}
public void GoTitle () => SimpleSceneNavigator.Instance.GoForwardAsync<SampleTitle> ().Forget();
}
SampleMainからSampleResultへ遷移するときに引数としてパラメータを渡しています
new SampleResult.Options(10)
これはSampleResultのOnLoadの引数として渡ってきます。
public override void OnLoad (object options) {
// objectからOptionsにキャストする
var op = options as Options;
label.text = op.Score.ToString();
}
余談ですが、Options作るときにVisualStudioCodeでプロパティーからコンストラクターを自動生成する機能を発見しました。ほしかった機能!
まとめ
画面遷移のサンプルを作りました。
個人開発やハッカソンでの開発の参考になれば幸いです。