在 ILRuntime 的基础上,搭建一个简单的UI系统(二)UIView

上一篇主要处理了一些UIPanel相关的逻辑,这一篇我们主要来讲讲UIView部分的理解。

目前个人理解下来UIView主要有两种情况

1.例如商城界面,会有很多的物品卡,每一个物品卡就是一个UIView,然后我们在UIView中进行物品的详情显示等逻辑。(这种情况一般是动态加载物品卡的prefab)

在 ILRuntime 的基础上,搭建一个简单的UI系统(二)UIView_第1张图片

2.在一些组件较多的复杂界面,例如个人信息界面可能会有好几个标签页,如果这些组件逻辑全部写在一个UIPanel中,那么这个UIPanel的代码将会很长,看着很头疼。因此我们可以将其分成几个UIView,然后在UIView当中再去写单独的逻辑。(这种情况一般组件已经挂载在UIPanel上了)

在 ILRuntime 的基础上,搭建一个简单的UI系统(二)UIView_第2张图片

 

注:由于代码量较大,改动较多,有兴趣的小伙伴还是看github上面的工程吧,有好的建议可以提出。

由于有的组件需要动态添加,有的是已经加载好挂在UIPanel中的,因此我们会在UIView中添加一个GameObject变量,用于存储UIView对应的GameObject。(考虑下来之前的url属性暂时只有在UIPanel中用到,所以将其移到UIPanel中)

namespace Hotfix.UI
{
    public class UIView : IView
    {
        public GameObject gameObject { private set; get; }
        public Transform transform { private set; get; }
        public RectTransform rectTransform { private set; get; }

        Transform m_parent;
        public Transform parent
        {
            get { return m_parent; }
            set
            {
                m_parent = value;
                if (m_parent != null)
                    transform?.SetParent(m_parent);
            }
        }

        ......


        public UIView()
        {
        }

        public UIView(GameObject go)
        {
            SetGameObject(go);
        }

        public UIView(GameObject go, Transform parent)
        {
            SetGameObject(go, parent);
        }

        protected void SetGameObject(GameObject go, Transform parent = null)
        {
            gameObject = go;
            if (gameObject == null)
                return;
            transform = gameObject.transform;
            rectTransform = gameObject.GetComponent();
            this.parent = parent;
        }

        public void SetActive(bool isActive)
        {
            if (isActive)
                Show();
            else
                Hide();
        }

        public virtual void Init()
        {
            GetChild();
        }

        public virtual void GetChild()
        {
        }

        ......
    }
}

我们在UIViewManager中来实例化UIView,并且管理对应的生命周期。

namespace Hotfix.Manager
{
    public class UIViewManager : ManagerBase
    {
        ......

        public T CreateView(GameObject go) where T : UIView
        {
            return CreateView(typeof(T), go) as T;
        }

        public T CreateView(GameObject go, RectTransform parent) where T : UIView
        {
            return CreateView(typeof(T), go, parent) as T;
        }

        //创建UIView
        public UIView CreateView(Type type, params object[] args)
        {
            UIView view = Activator.CreateInstance(type, args) as UIView;
            view.Init();
            AddUIView(view);
            return view;
        }
        
        ......

    }
}

商城的物品卡组件GoodsItemView,由于是动态加载的,所以我们可以先加载好Prefab,然后Instantiate好后将GameObject传递给UIView,同时传递一个父节点用于挂载。

需要注意的一点是,我们用Activator.CreateInstance实例化UIView子类的时候,父节点ugui的transform的Type为RectTransform,因此我们构造函数的parent类型也要为RectTransform,否则无法正常的实例化对象。

namespace Hotfix.UI
{
    public class GoodsItemView : UIView
    {
        Text m_nameText;
        Button m_buyBtn;

        //因为在Activator.CreateInstance实例化的时候,ugui.transform的Type为RectTransform
        public GoodsItemView(GameObject go, RectTransform parent) : base(go, parent)
        {
        }

        public override void Init()
        {
            base.Init();

            m_buyBtn.onClick.AddListener(OnBuyBtnClicked);
        }

        public override void GetChild()
        {
            base.GetChild();
            m_nameText = transform.Find("NameText").GetComponent();
            m_buyBtn = transform.Find("BuyButton").GetComponent

然后在商城界面MallPanel中实例化对象

namespace Hotfix.UI
{
    [UI("MallPanel")]
    public class MallPanel : UIPanel
    {
        ......

        public override void Init()
        {
            base.Init();
            
            m_closeBtn.onClick.AddListener(OnCloseBtnClick);

            m_mallList.Add("明装");
            m_mallList.Add("燕尔");
            m_mallList.Add("西狩获麟");
            m_mallList.Add("刹那生灭");
            m_mallList.Add("听冰");
            m_mallList.Add("琅嬛");
            m_mallList.Add("陌上花");

            Object itemAsset = Resources.Load("GoodsItem");
            for (int i = 0; i < m_mallList.Count; i++)
            {
                GameObject go = GameObject.Instantiate(itemAsset) as GameObject;
                //m_grid.transform is RectTransform
                GoodsItemView view = UIViewManager.Instance.CreateView(go, m_grid.transform as RectTransform);
                view.Setting(m_mallList[i]);
                view.Show();
            }
        }

        ......
    }
}

第二种情况,我们可以简单的搭建一个界面MessagePanel,里面的内容为三个Toggle,还有三部分内容,我们要实现的就是点击一个Toggle,只显示对应的内容即可。

namespace Hotfix.UI
{
    [UI("MessagePanel")]
    public class MessagePanel : UIPanel
    {
        Button m_closeBtn;

        Toggle m_informationToggle;
        Toggle m_albumToggle;
        Toggle m_journalToggle;

        InformationView m_InformationView;
        AlbumView m_albumView;
        JournalView m_journalView;

        public MessagePanel(string url) : base(url)
        {
        }

        public override void Init()
        {
            base.Init();

            m_informationToggle.onValueChanged.AddListener(OnInformationToggleClicked);
            m_albumToggle.onValueChanged.AddListener(OnAlbumToggleClicked);
            m_journalToggle.onValueChanged.AddListener(OnJournalToggleClicked);

            m_closeBtn.onClick.AddListener(OnCloseBtnClick);
        }

        public override void GetChild()
        {
            base.GetChild();
            m_closeBtn = transform.Find("CloseButton").GetComponent

然后部分内容对应一个UIView,例如InformationView

namespace Hotfix.UI
{
    public class InformationView : UIView
    {
        Text m_nameText;
        Text m_sexText;
        Text m_phoneText;

        public InformationView(GameObject go) : base(go)
        {
        }

        public override void Init()
        {
            base.Init();

            m_nameText.text = "七龙珠";
            m_sexText.text = "成男";
            m_phoneText.text = "158xxxxx094";
        }

        public override void GetChild()
        {
            base.GetChild();
            m_nameText = transform.Find("NameList/ContentText").GetComponent();
            m_sexText = transform.Find("SexList/ContentText").GetComponent();
            m_phoneText = transform.Find("PhoneList/ContentText").GetComponent();
        }
    }
}

这样就能实现一些简单的UI管理,后续功能将继续完善,欢迎大家提出宝贵的意见

你可能感兴趣的:(Unity,ILRuntime)