中心事件模型、统一输入管理

单例基类

// file name : BaseManager.cs
using UnityEngine;
using UnityEngine.Assertions;
public class BaseManager<T> where T : new()
{
    private static T _instance;
    public static T Ins
    {
        get
        {
            if (_instance == null)
            {
                _instance = new T();
            }
            return _instance;
        }
    }
}

public class BaseManagerMonobehviour<T> : MonoBehaviour where T:MonoBehaviour
{
    private static T _instance;
    public static T Ins
    {
        get
        {
            if (_instance == null)
            {
                Assert.IsTrue(FindObjectsOfType<T>().Length <= 1);      
                _instance = FindObjectOfType<T>();
            }

            if (_instance == null)
            {
                Debug.LogWarning("Base Manager have create a gameobject named " + nameof(T));
                GameObject hc = Instantiate(new GameObject(nameof(T)));
                _instance = hc.AddComponent<T>() as T;
            }
            return _instance;
        }
    }
}

有两种单例的写法

  • 不需要继承 MonoBehaviour 的
  • 需要继承 MonoBehaviour 的

对于不需要继承 MonoBehaviour 的,也就是不参与 Unity 的脚本执行过程,也就是只需要被外部调用,不需要自己执行,比如说下面的中心事件管理类就是这样。
这种方式可以搭配公共 Mono 来实现跟下面的一样的效果,也就是参与到 Unity 脚本执行周期的单例类,但个人觉得更加麻烦,因为还需要在某个其他的地方把这个没有继承自 MonoBehaviour 的一个帧更新方法注册进公共 Mono 的 Update 里面,所以暂时不讨论。

对于需要继承 MonoBehaviour 的,也就是需要参与到脚本执行过程或者还需要在编辑器中给某个变量赋值的。因为是单例,所以场景中只需要有一个,先查找场景中的该脚本的个数,如果大于 1 ,那么直接 GG,如果等于 1 ,那么就是它了,如果等于 0,那么会新建一个空物体,然后把该脚本挂上去,但这样会有一个问题,需要在场景中赋值的脚本运行后就会报空指针,更严重的问题是不能在 OnDestroy 中引用这个单例,不然会在结束运行的时候报错(虽然并不影响什么,但看着很不舒服)

Some objects were not cleaned up when closing the scene. (Did you spawn new GameObjects from OnDestroy?)
The following scene GameObjects were found:
InputMrg

就是说你在 Destory 中创建了新物体。你可能不会直接在 Destory 中 new,但是只要在其中引用了这个单例实际上就会创建,因为单例被销毁的之后被判断为 null,然后就进入第二个判断,创建了新的 GameObject。

关于经常运行之后才发现 Inspector 面板上有 null 的情况,建议所有需要在场景中赋值的物体都在 OnValidate (其实我觉得在 OnDrawGizmos 中更好)中设置断言,如下所示:

public class BreathMaskCheckChapter : MonoBehaviour, IChapter
{

    public GameObject Mask;
    public Transform MaskPreparePosition;
    public GameObject CanvasStartOrReStudy;

    private void OnValidate()
    {
        Assert.IsNotNull(Mask);
        Assert.IsNotNull(MaskPreparePosition);
        Assert.IsNotNull(CanvasStartOrReStudy);
    }

这样不需要运行就可以在编辑器中检测这三个值是否为空。

事件中心

将所有的事件都用字符串索引的方式注册进事件中心,然后再用事件中心统一触发,这样可以在一定程度上解耦合,输入事件管理的解耦合主要也是依赖于此。

//file name : EventCenter.cs
using UnityEngine;
using UnityEngine.Events;
using System.Collections.Generic;
public interface IEventInfo
{
    //这是一个空接口
}
public class EventInfo<T> : IEventInfo
{
    public UnityAction<T> actions;

    public EventInfo(UnityAction<T> action)
    {
        actions += action;
    }
}

public class EventInfo : IEventInfo
{
    public UnityAction actions;

    public EventInfo(UnityAction action)
    {
        actions += action;
    }
}

public class EventCenter : BaseManager<EventCenter>
{
    //字典中,key对应着事件的名字,
    //value对应的是监听这个事件对应的委托方法们(重点圈住:们)
    private Dictionary<string, IEventInfo> eventDic
        = new Dictionary<string, IEventInfo>();

    //添加事件监听
    //第一个参数:事件的名字
    //第二个参数:处理事件的方法(有参数(类型为T)的委托)
    public void AddEventListener<T>(string name, UnityAction<T> action)
    {
        //有没有对应的事件监听
        //有的情况
        if (eventDic.ContainsKey(name))
        {
            (eventDic[name] as EventInfo<T>).actions += action;
        }
        //没有的情况
        else
        {
            eventDic.Add(name, new EventInfo<T>(action));
        }
    }

    public void AddEventListener(string name, UnityAction action)
    {
        //有没有对应的事件监听
        //有的情况
        if (eventDic.ContainsKey(name))
        {
            (eventDic[name] as EventInfo).actions += action;
        }
        //没有的情况
        else
        {
            eventDic.Add(name, new EventInfo(action));
        }
    }

    //通过事件名字进行事件触发
    public void EventTrigger<T>(string name, T info)
    {
        //有没有对应的事件监听
        //有的情况(有人关心这个事件)
        if (eventDic.ContainsKey(name))
        {
            //调用委托(依次执行委托中的方法)
            //?是一个C#的简化操作
            (eventDic[name] as EventInfo<T>).actions?.Invoke(info);
        }
    }

    public void EventTrigger(string name)
    {
        //有没有对应的事件监听
        //有的情况(有人关心这个事件)
        if (eventDic.ContainsKey(name))
        {
            //调用委托(依次执行委托中的方法)
            //?是一个C#的简化操作
            (eventDic[name] as EventInfo).actions?.Invoke();
        }
    }

    //移除对应的事件监听
    public void RemoveEventListener<T>(string name, UnityAction<T> action)
    {
        if (eventDic.ContainsKey(name))
        {
            //移除这个委托
            (eventDic[name] as EventInfo<T>).actions -= action;
        }
    }

    public void RemoveEventListener(string name, UnityAction action)
    {
        if (eventDic.ContainsKey(name))
        {
            //移除这个委托
            (eventDic[name] as EventInfo).actions -= action;
        }
    }

    //清空所有事件监听(主要用在切换场景时)
    public void Clear()
    {
        eventDic.Clear();
    }
}

这样乍一看没什么问题,在外面的调用方式如下:

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;
using UnityEngine.EventSystems;
using UnityEngine.Events;

public class NewBehaviourScript : MonoBehaviour
{
    private void Start()
    {
        //EventCenter.Ins.AddEventListener<int>("Test1", Test1);
        EventCenter.Ins.AddEventListener("Test1", Test2);
    }

    private void Update()
    {
        if (Input.GetKeyDown(KeyCode.Space))
        {
            EventCenter.Ins.EventTrigger<int>("Test1",1);
        }
        
    }

    private void Test1(int i)
    {
        Debug.Log("Test1" + " " + i);
    }

    private void Test2()
    {
        Debug.Log("Test2");
    }
}

对于有参数的调用方式如下:

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;
using UnityEngine.EventSystems;
using UnityEngine.Events;

public class NewBehaviourScript : MonoBehaviour
{
    private void Start()
    {
        EventCenter.Ins.AddEventListener<int>("Test1", Test1);
        //EventCenter.Ins.AddEventListener("Test1", Test2);
    }

    private void Update()
    {
        if (Input.GetKeyDown(KeyCode.Space))
        {
            EventCenter.Ins.EventTrigger<int>("Test1",1);
        }
        
    }

    private void Test1(int i)
    {
        Debug.Log("Test1" + " " + i);
    }

    private void Test2()
    {
        Debug.Log("Test2");
    }
}

这两段代码几乎是一样的,为什么我不在一段里面直接说完呢?是因为如果顺着执行这两句的话,是错的。一开始 eventDic["Test1"] 的 value 是有参的,然后第二次执行的时候,下面的 as 转化的结果其实是 null,所以就会报空指针异常。

报错原因

稍微改一下,将无参、各个不同的参数分开来存放

using UnityEngine;
using UnityEngine.Events;
using System.Collections.Generic;
public interface IEventInfo
{
    //这是一个空接口
}
public class EventInfo<T> : IEventInfo
{
    public UnityAction<T> actions;

    public EventInfo(UnityAction<T> action)
    {
        actions += action;
    }
}

public class EventInfo : IEventInfo
{
    public UnityAction actions;

    public EventInfo(UnityAction action)
    {
        actions += action;
    }
}

public class EventCenter : BaseManager<EventCenter>
{
    private Dictionary<string, List<IEventInfo>> eventDic
        = new Dictionary<string, List<IEventInfo>>();

    public void AddEventListener<T>(string name, UnityAction<T> action)
    {
        if (eventDic.ContainsKey(name))
        {
            foreach(var d in eventDic[name])
            {
                if(d is EventInfo<T>)
                {
                    (d as EventInfo<T>).actions += action;
                    return;
                }
            }
        }
        else
        {
            eventDic.Add(name, new List<IEventInfo>());
        }
        eventDic[name].Add(new EventInfo<T>(action));
    }


    public void AddEventListener(string name, UnityAction action)
    {
        if (eventDic.ContainsKey(name))
        {
            foreach (var d in eventDic[name])
            {
                if (d is EventInfo)
                {
                    (d as EventInfo).actions += action;
                    return;
                }
            }
        }
        else
        {
            eventDic.Add(name, new List<IEventInfo>());
        }
        eventDic[name].Add(new EventInfo(action));
    }

    public void EventTrigger<T>(string name, T info)
    {
        if (eventDic.ContainsKey(name))
        {
            foreach (var d in eventDic[name])
            {
                if (d is EventInfo<T>)
                {
                    (d as EventInfo<T>).actions?.Invoke(info);
                }

                if(d is EventInfo)
                {
                    (d as EventInfo).actions?.Invoke();
                }
            }
        }
    }

    public void EventTrigger(string name)
    {
        if (eventDic.ContainsKey(name))
        {
            foreach (var d in eventDic[name])
            {
                if (d is EventInfo)
                {
                    (d as EventInfo).actions?.Invoke();
                    return;
                }
            }
        }
    }

    public void RemoveEventListener<T>(string name, UnityAction<T> action)
    {
        if (eventDic.ContainsKey(name))
        {
            foreach (var d in eventDic[name])
            {
                if (d is EventInfo<T>)
                {
                    (d as EventInfo<T>).actions -= action;
                    return;
                }
            }
        }
    }

    public void RemoveEventListener(string name, UnityAction action)
    {
        if (eventDic.ContainsKey(name))
        {
            foreach (var d in eventDic[name])
            {
                if (d is EventInfo)
                {
                    (d as EventInfo).actions -= action;
                    return;
                }
            }
        }
    }

    //清空所有事件监听(主要用在切换场景时)
    public void Clear()
    {
        eventDic.Clear();
    }
}

测试代码如下:

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;
using UnityEngine.EventSystems;
using UnityEngine.Events;

public class NewBehaviourScript : MonoBehaviour
{
    private void Start()
    {
        EventCenter.Ins.AddEventListener<int>("Test1", Test1);
        EventCenter.Ins.AddEventListener("Test1", Test2);
        EventCenter.Ins.AddEventListener<GameObject>("Test1", Test3);
    }

    private void Update()
    {
        if (Input.GetKeyDown(KeyCode.Alpha1))
        {
            EventCenter.Ins.EventTrigger<int>("Test1",1);
        }

        if (Input.GetKeyDown(KeyCode.Alpha2))
        {
            EventCenter.Ins.EventTrigger("Test1");
        }

        if (Input.GetKeyDown(KeyCode.Alpha3))
        {
            EventCenter.Ins.EventTrigger<GameObject>("Test1",new GameObject("miao"));
        }

    }

    private void Test1(int i)
    {
        Debug.Log("Test1" + " " + i);
    }

    private void Test2()
    {
        Debug.Log("Test2");
    }
    private void Test3(GameObject o)
    {
        Debug.Log("Test3" + " " + o.ToString());
    }
}

当按 1 的时候,会调用 Test1Test2;当按 2 的时候,会调用 Test2;当按 3 的时候,会调用 Test2Test3
也就是有参的触发时会顺带触发无参的。
最后说一下最好不要在代码中直接出现 string,可以使用一个静态类来存放所有的 string。

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public static class ConstName
{
    // sounds
    public readonly static string sound_descript_guoShai= "GuoShai";
    public readonly static string sound_descript_liangShai = "LiangShai";
    public readonly static string sound_descript_maDuo = "MaDuo";

    // EventCenter Names
    public readonly static string breathMask_descript_reture = "BreathMask_Reture";
    public readonly static string breathMask_check_startOperate = "BreathMask_Start_Operate";

    // pico SDK Tree
    public readonly static string pico_sdk_controllerManager = "ControllerManager";
    public readonly static string pico_sdk_pvrController = "PvrController";
    public readonly static string pico_sdk_handPosition = "handPosition";

    // Resources
    public readonly static string resources_hand_right= "Prefabs/Hands/hand_right";
    public readonly static string resources_hand_left= "Prefabs/Hands/hand_left";

    // Hand Animation Name
    public readonly static string hand_state_idle = "Idle";
    public readonly static string hand_state_grab = "GrabLarge";
}

输入事件管理

可以通过中心事件来统一管理输入事件。因为是做 Pico 开发,所有需要检测两套输入,因为需要集合不重复,所以可以直接使用 HashSet。如果后面需要再监听别的输入事件也只需要修改 InputMrg ,对于外部已经调用过的代码完全不需要修改。

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class InputMrg : BaseManagerMonobehviour<InputMrg>
{
    public enum KeyType
    {
        Keyborad,
        Pico
    };
    public class InputDate
    {
        public KeyType type { get; private set; }
        public KeyCode keyCodeType;
        public Pvr_UnitySDKAPI.Pvr_KeyCode picoKeyCodeType;
        public int picoKeyHand;

        public InputDate(KeyCode k)
        {
            type = KeyType.Keyborad;
            keyCodeType = k;
        }

        public InputDate(Pvr_UnitySDKAPI.Pvr_KeyCode k,int hand)
        {
            type = KeyType.Pico;
            picoKeyCodeType = k;
            picoKeyHand = hand;
        }
    }

    
    private readonly HashSet<KeyCode> ListKeyCode = new HashSet<KeyCode>();
    private readonly HashSet<Pvr_UnitySDKAPI.Pvr_KeyCode> ListPicoKeyCode = new HashSet<Pvr_UnitySDKAPI.Pvr_KeyCode>();

    private void Update()
    {
        foreach (var k in ListKeyCode)
        {
            if (Input.GetKeyDown(k))
            {
                EventCenter.Ins.EventTrigger<InputDate>(GetKeyDownEventName(k),new InputDate(k));
            }

            if (Input.GetKeyUp(k))
            {
                EventCenter.Ins.EventTrigger<InputDate>(GetKeyUpEventName(k),new InputDate(k));
            }
        }

        foreach (var k in ListPicoKeyCode)
        {
            if (Pvr_UnitySDKAPI.Controller.UPvr_GetKeyDown(0, k))
            {
                EventCenter.Ins.EventTrigger<InputDate>(GetKeyDownEventName(k, 0),new InputDate(k,0));
            }

            if (Pvr_UnitySDKAPI.Controller.UPvr_GetKeyDown(1, k))
            {
                EventCenter.Ins.EventTrigger<InputDate>(GetKeyDownEventName(k, 1),new InputDate(k,1));
            }

            if (Pvr_UnitySDKAPI.Controller.UPvr_GetKeyUp(0, k))
            {
                EventCenter.Ins.EventTrigger<InputDate>(GetKeyUpEventName(k, 0),new InputDate(k,0));
            }

            if (Pvr_UnitySDKAPI.Controller.UPvr_GetKeyUp(1, k))
            {
                EventCenter.Ins.EventTrigger<InputDate>(GetKeyUpEventName(k, 1),new InputDate(k,1));
            }
        }
    }

    public string GetKeyDownEventName(KeyCode key)
    {
        ListKeyCode.Add(key);
        return "key_down_" + key.ToString();
    }

    public string GetKeyDownEventName(Pvr_UnitySDKAPI.Pvr_KeyCode key, int handId)
    {
        ListPicoKeyCode.Add(key);
        return "pico_key_down_" + key.ToString() + '_' + handId;
    }

    public string GetKeyUpEventName(KeyCode key)
    {
        ListKeyCode.Add(key);
        return "key_up_" + key.ToString();
    }

    public string GetKeyUpEventName(Pvr_UnitySDKAPI.Pvr_KeyCode key, int handId)
    {
        ListPicoKeyCode.Add(key);
        return "pico_key_up_" + key.ToString() + '_' + handId;
    }

}

外部调用方式,你可以像下面这样用有参数的回调函数:

        EventCenter.Ins.AddEventListener<InputMrg.InputDate>(InputMrg.Ins.GetKeyDownEventName(KeyCode.B), EventActionSideButtonDown);
        EventCenter.Ins.AddEventListener<InputMrg.InputDate>(InputMrg.Ins.GetKeyUpEventName(KeyCode.B), EventActionSideButtonUp);
        EventCenter.Ins.AddEventListener<InputMrg.InputDate>(InputMrg.Ins.GetKeyDownEventName(Pvr_UnitySDKAPI.Pvr_KeyCode.Right, 1), EventActionSideButtonDown);
        EventCenter.Ins.AddEventListener<InputMrg.InputDate>(InputMrg.Ins.GetKeyUpEventName(Pvr_UnitySDKAPI.Pvr_KeyCode.Right, 1), EventActionSideButtonUp);
        EventCenter.Ins.AddEventListener<InputMrg.InputDate>(InputMrg.Ins.GetKeyDownEventName(Pvr_UnitySDKAPI.Pvr_KeyCode.Left, 0), EventActionSideButtonDown);
        EventCenter.Ins.AddEventListener<InputMrg.InputDate>(InputMrg.Ins.GetKeyUpEventName(Pvr_UnitySDKAPI.Pvr_KeyCode.Left, 0), EventActionSideButtonUp);```

然后在回调函数里面去判断是否进行触发:

    private void EventActionSideButtonDown(InputMrg.InputDate date)
    {
        if (m_grabbingObject.IsGrabbing)
        {
            if (date.type == InputMrg.KeyType.Keyborad)
            {
                EventActionSideButtonDown1();
            }
            else if (date.type == InputMrg.KeyType.Pico)
            {
                if (m_grabbingObject.IsGrabbingHand == date.picoKeyHand)    // 是正在抓着的手按得侧边键
                {
                    EventActionSideButtonDown1();
                }
            }
        }
    }

你也可以直接使用无参数的回调函数:

        EventCenter.Ins.AddEventListener(InputMrg.Ins.GetKeyDownEventName(KeyCode.Space), KeyActionTriggerDown);
        EventCenter.Ins.AddEventListener(InputMrg.Ins.GetKeyDownEventName(Pvr_UnitySDKAPI.Pvr_KeyCode.TRIGGER, handId), KeyActionTriggerDown);

参考

https://blog.csdn.net/m0_46378049/article/details/106180451

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,406评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,732评论 3 393
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,711评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,380评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,432评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,301评论 1 301
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,145评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,008评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,443评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,649评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,795评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,501评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,119评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,731评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,865评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,899评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,724评论 2 354

推荐阅读更多精彩内容