2
4

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 5 years have passed since last update.

MagicOnion のストリーミング接続を複数シーンで持ち越す場合のTips

Last updated at Posted at 2019-07-08

はじめに

MagicOnion についてはこちらをご覧ください。
一般的なマルチプレイゲームでは、マッチング→メインのゲーム→リザルト といったふうに、複数のシーンでストリーミング接続を繋ぎっぱなしにする事が多いと思います。
その場合、1シーンで使う場合に比べ気をつける必要がある点がいくつかあるので、それぞれに対して自分がやっている対応を紹介します。

Receiver

接続全体を1つのインターフェースにまとめることになるので、シーンごとに一部のメソッドしか使わないことになります。
一方、IHogeHubReceiver を実装するクラスでは当然すべてのメソッドを定義しなければいけません。
このままでは非常に扱いづらいのですべてのメソッドを UniRx の Observable に変換してしまうと良さそうです。
インターフェイスの明示的実装を使えば同名のPublicなプロパティが生やせます。

public interface IHogeHubReceiver
{
    void OnHoge();
    void OnFuga(int i);
    void OnPiyo(int i, int j);
}

public class HogeHubReceiver : IHogeHubReceiver
{
    // 引数なしは UniRx.Unit に
    public IObservable<Unit> OnHoge => _onHoge;
    private Subject<Unit> _onHoge = new Subject<Unit>();
    void IHogeHubReceiver.OnHoge() => _onHoge.OnNext(Unit.Default);

    // 1引数はそのまま
    public IObservable<int> OnFuga => _onFuga;
    private Subject<int> _onFuga = new Subject<int>();
    void IHogeHubReceiver.OnFuga(int i) => _onFuga.OnNext(i);

    // 2引数以上はタプル化
    public IObservable<(int i, int j)> OnPiyo => _onPiyo;
    private Subject<(int i, int j)> _onPiyo = new Subject<(int i, int j)>();
    void IHogeHubReceiver.OnPiyo(int i, int j) => _onPiyo.OnNext((i, j));
}

これを手動で書くのはめんどくさくてしょうがないのでエディタ拡張で自動生成しましょう。

StreamingHubReceiverCreator.cs
using System;
using System.Linq;
using System.Text.RegularExpressions;
using System.IO;
using UnityEditor;

/// <summary>
/// StreamingHubReceiverのインターフェースから具象型を生成するエディタ拡張
/// </summary>
public static class StreamingHubReceiverCreator
{
    /// <summary>
    /// StreamingHubReceiverのインターフェースから具象型を生成する
    /// すべてのメソッドはIObservableに変換される
    /// </summary>
    [MenuItem("Assets/Create/StreamingHubReceiver")]
    private static void CreateStreamingHubReceiver()
    {
        var interfaceName = Selection.activeObject.name;

        var source = File.ReadAllText(AssetDatabase.GetAssetPath(Selection.activeObject));

        // using追加
        if (!source.Contains("using System;")) source = "using System;\r\n" + source;
        if (!source.Contains("using UniRx;")) source = "using UniRx;\r\n" + source;

        // クラス名変換
        source = Regex.Replace(source, "interface I(.*Receiver)", "class $1 : I$1");

        // 各メソッド変換
        source = Regex.Replace(source, "( *)void (.*)\\((.*)\\);", m =>
        {
            var space = m.Groups[1].Value;
            var name = m.Groups[2].Value;
            var privateName = "_" + Char.ToLower(name[0]) + name.Substring(1);

            var args = m.Groups[3].Value;
            var genericArgs = "Unit";
            var argNames = "Unit.Default";

            // 引数の個数に応じてデータ構造を変更
            var x = Regex.Matches(args, "(?:(.+?[^,]) (.+?)(?:,|$))").Cast<Match>().ToArray();
            if (x.Length == 1)
            {
                genericArgs = x[0].Groups[1].Value;
                argNames = x[0].Groups[2].Value;
            }
            else if (x.Length > 1)
            {
                // タプル化
                genericArgs = "(" + args + ")";
                argNames = "(" + string.Join(", ", x.Select(y => y.Groups[2].Value)) + ")";
            }

            return $"{space}public IObservable<{genericArgs}> {name} => {privateName};\r\n" +
                $"{space}private Subject<{genericArgs}> {privateName} = new Subject<{genericArgs}>();\r\n" +
                $"{space}void {interfaceName}.{name}({args}) => {privateName}.OnNext({argNames});";
        });

        var fileName = Selection.activeObject.name.Substring(1) + ".cs";
        var classPath = Directory.EnumerateFiles(".", fileName, SearchOption.AllDirectories).FirstOrDefault();
        File.WriteAllText(classPath ?? "Assets/" + fileName, source);
        AssetDatabase.Refresh();
    }

    /// <summary>
    /// 選択しているアイテムがスクリプトファイルかどうかを判別する
    /// </summary>
    /// <returns>スクリプトファイルかどうか</returns>
    [MenuItem("Assets/Create/StreamingHubReceiver", isValidateFunction: true)]
    private static bool ValidateCreateStreamingHubReceiver() => Selection.activeObject is MonoScript;
}

切断

接続が不要になったら切断するというは大事で、これをしないと Unity エディタがすぐフリーズします。
単独のシーンで使うならシーンで使うスクリプトの OnDestroy で切断すれば良いですが、
複数シーンの場合は、接続を管理するオブジェクトを作り DontDestroyOnLoad に設定しておくのが良いでしょう。

public class ConnectionHolder : MonoBehaviour
{
    public IHogeHub Client { get; private set; }
    public HogeHubReceiver Receiver { get; } = new HogeHubReceiver(); // さっき作ったやつ

    private Channel _channel;

    public static ConnectionHolder Create()
    {
        // DontDestroy化
        var gameObject = new GameObject("ConnectionHolder");
        DontDestroyOnLoad(gameObject);

        // 接続
        var holder = gameObject.AddComponent<ConnectionHolder>();
        holder._channel = new Channel("host:port", ChannelCredentials.Insecure);
        holder.Client = StreamingHubClient.Connect<IHogeHub, IHogeHubReceiver>(holder._channel, holder.Receiver);

        return holder;
    }

    private void OnDestroy()
    {
        Client.DisposeAsync();
        _channel.ShutdownAsync();
    }
}
  1. はじめのシーンでは Create を呼ぶ
  2. 以降のシーンでは Find で探してくる
  3. 最後のシーンで Destroy する

というイメージです。
ついでに参加者の保持管理もこのクラスでやってしまうと後のシーンで使いやすくて便利です。

2
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
2
4

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?