仿LOL项目开发第七天
by 草帽
不知不觉已经写到了第七篇这种类型的博客,但是回过头看看之前写的,发现都只能我自己能看懂。
我相信在看的童鞋云里雾里的,因为我基本上没怎么详细讲一个脚本怎么用?但是你们可以自己看下代码,很快你就知道怎么用!以后也可以用到自己的项目中。
所以说阅读别人的代码非常重要,因为你们从中发现他们代码的优点和缺点。
OK,废话不多说,我们继续上节课,上节课我们已经封装自己的UI框架,但是还少个没讲,可能有些童鞋也会遇到报错,怎么解决?
在Window下新建一个UILib,然后里面新建一个脚本:
WidgetFactory:
using UnityEngine; using System.Collections.Generic; namespace UILib { public class WidgetFactory { /// <summary> /// 查找节点下所有的UI组件,缓存到字典里面 /// </summary> /// <param name="trans"></param> /// <param name="parent"></param> /// <param name="dicAllUIObjects"></param> public static void FindAllUIObjects(Transform trans, IXUIObject parent, ref Dictionary<string, XUIObjectBase> dicAllUIObjects) { int i = 0; while (i < trans.childCount) { Transform child = trans.GetChild(i); XUIObjectBase component = child.GetComponent<XUIObjectBase>(); if (!(null != component)) { goto IL_85; } if (component.GetType().GetInterface("IXUIListItem") == null) { if (dicAllUIObjects.ContainsKey(component.name)) { Debug.Log(component.name); Debug.LogError("m_dicId2UIObject.ContainsKey:" + WidgetFactory.GetUIObjectId(component)); } dicAllUIObjects[component.name] = component; component.Parent = parent; goto IL_85; } else { Debug.Log("fdsfd"); } IL_8F: i++; continue; IL_85: WidgetFactory.FindAllUIObjects(child, parent, ref dicAllUIObjects); goto IL_8F; } } /// <summary> /// 取得该组件所在的id(包含父亲节点的id) /// </summary> /// <param name="uiObject"></param> /// <returns></returns> public static string GetUIObjectId(IXUIObject uiObject) { string result; if (null == uiObject) { result = string.Empty; } else { string text = uiObject.CacheGameObject.name; IXUIListItem iXUIListItem = uiObject as IXUIListItem; if (iXUIListItem != null) { text = iXUIListItem.Id.ToString(); } while (null != uiObject.Parent) { uiObject = uiObject.Parent; string arg = uiObject.CacheGameObject.name; iXUIListItem = (uiObject as IXUIListItem); if (null != iXUIListItem) { arg = iXUIListItem.Id.ToString(); } text = string.Format("{0}#{1}", arg, text); } result = text; } return result; } } }
这个脚本主要是用来找UI的子物体用的。
OK,我们正式进入到LoginWindow的完善。我们知道这个界面分为两块,一个是登陆,一个选择服务器。
我们先来写登陆的逻辑。
可以看到这个界面分为3个组件:
1.用户名输入框
2.密码输入框
3.确认登陆按钮
所以我们在LoginWindow初始化这些组件:
然后在InitWidget()里面初始化这些定义的组件:
这里的名字是我自己去的比如Login/UserNameInput,你们可以自己取自己的名字。
Login是LoginWindow下面的子物体,然后UserNameInput是Login下面的子物体,我们用/来区分紫武器。
OK,当我们点击确认登陆按钮的时候,肯定需要一个登陆事件给这个按钮,所以现在我们来写这个事件监听。
然后我们来编写OnLoginSumbit方法,主要处理一些登陆逻辑的事情。
public void OnLoginSumbit(GameObject go) { string username = this.m_Input_UsernameInput.value; string password = this.m_Input_PasswordInput.value; //如果用户名或者密码为空的话,就显示提示消息 if (string.IsNullOrEmpty(username) || string.IsNullOrEmpty(password)) { CEvent evt = new CEvent(EGameEvent.eGameEvent_ShowMessage); evt.AddParam("content", StringConfigManager.GetString("MessageWindow.EMT_SureTip.LoginNullUsernameOrPassword")); EventCenter.SendEvent<EMessageType, Action<bool>>(evt, EMessageType.EMT_SureTip, (isOk) => { EventCenter.Broadcast(EGameEvent.eGameEvent_HideMessage); }); } LoginCtrl.singleton.Login(username, password); }
可以看到我们搞了一个当确定消息提示,如果用户名和密码为空的话。
然后为了符合类的单一职责,我们把所有的逻辑放在LoginCtrl中。
新建文件夹取名为Controller,然后在里面新建一个LoginCtrl类:(我忘记之前是不是讲过这个类,如果有讲过这个类,因为我太久之前写的,都忘记了,读者自行改过)。
在写这个逻辑之前,我们先来想下我们登陆进去在选择服务器,所以登陆这个验证并没有进入到游戏服务器中,因为我们根本没有选择服务器,怎么能进到服务器中呢?
所以我认为就只是网页简单的验证用户名密码,然后选择服务器之后才真正的进入到游戏服务器。
所以这个登陆的逻辑步骤是:
1.开启协程访问一个网站,这个网站验证用户名和密码。(所以我决定用PHP来写这个验证)
2.如果验证通过之后,然后发送服务器列表给客户端,客户端再进入到选择服务器界面。
3.客户端选择某个服务器,然后进入到这个游戏服务器。
所以首先,我在网站文件夹下面新建一个check.php文件:
这个脚本就是通过数据库连接,然后进行验证,这里我简单一点,我就直接都放回成功。因为我们只是做下测试,不用每步都详细。
所以我这里只是简单放回一个success字符串,然后在LoginCtrl中:
public void Login(string account, string password) { this.username = account; this.password = password; LOLGameDriver.Instance.StartCoroutine(CheckUserPass()); } IEnumerator CheckUserPass() { if (string.IsNullOrEmpty(this.username) || string.IsNullOrEmpty(this.password)) { yield break; } WWW www = new WWW("http://127.0.0.1/LOLGameDemo/check.php"); bool success = false; for (int i = 0; i < 10; i++) { yield return new WaitForSeconds(0.5f); if (www.isDone) { if (string.IsNullOrEmpty(www.error)) { if (!string.IsNullOrEmpty(www.text)) { if (www.text == "success") { //加载服务器信息 SystemConfig.LoadServerList(); success = true; EventCenter.Broadcast(EGameEvent.eGameEvent_ShowSelectServerList); } else { //登陆失败 } } } else { this.m_log.Error(www.error.ToString()); } break; } } if (www != null) { www.Dispose(); www = null; } //如果不成功 if (!success) { } yield break; }
不知读者注意到没有,如果登陆成功的话,我们就开始加载服务器列表。
//加载服务器信息 SystemConfig.LoadServerList();
所以我们回到SystemConfig里面:
/// <summary> /// 加载服务器列表 /// </summary> public static void LoadServerList() { try { List<ServerInfo> servers; var url = GetCfgInfoUrlByName("ServerList"); string xmlSerList = ""; if (!string.IsNullOrEmpty(url)) { xmlSerList = DownloadMgr.Instance.DownLoadHtml(url); } servers = LoadXMLText<ServerInfo>(xmlSerList); if (servers.Count != 0) { ServerList = servers; } for (int i = 0; i < ServerList.Count; i++) { if (ServerList[i].id == LocalSetting.SelectedServer) { SelectedServerIndex = ServerList[i].id; break; } } } catch (Exception e) { m_log.Error(e.ToString()); } }
/// <summary> /// 根据名字取得服务端配置信息Url /// </summary> /// <param name="name"></param> /// <returns></returns> public static string GetCfgInfoUrlByName(string name) { string result = ""; foreach (var item in CfgInfoList) { if (item.name == name) { result = item.url; break; } } return result; }
然后我们到Cfg.xml文件添加这个url=>id是1,名字是ServerList:
<?xml version="1.0" encoding="utf-8"?> <root> <cfg> <id>0</id> <name>version</name> <url>http://127.0.0.1/LOLGameDemo/ServerVersion.xml</url> </cfg> <cfg> <id>1</id> <name>ServerList</name> <url>http://127.0.0.1/LOLGameDemo/ServerList.xml</url> </cfg> </root>
然后我们添加4个游戏服务器,分别是电信2个(type为0),联通两个(type为1):
<?xml version="1.0" encoding="utf-8"?> <root> <list> <id_i>0</id_i> <name_s>艾欧尼亚</name_s> <type_i>0</type_i> <flag_i>0</flag_i> <text_s>fe</text_s> </list> <list> <id_i>1</id_i> <name_s>战争学院</name_s> <type_i>0</type_i> <flag_i>0</flag_i> <text_s>csc</text_s> </list> <list> <id_i>2</id_i> <name_s>黑色玫瑰</name_s> <type_i>1</type_i> <flag_i>0</flag_i> <text_s>greger</text_s> </list> <list> <id_i>3</id_i> <name_s>洛克萨斯</name_s> <type_i>1</type_i> <flag_i>0</flag_i> <text_s>greger</text_s> </list> </root>
所以我们回到LoginWindow中,添加该事件:
protected override void OnAddListener() { EventCenter.AddListener(EGameEvent.eGameEvent_ShowSelectServerList, ShowSelectServer); }
protected override void OnRemoveListener() { EventCenter.RemoveListener(EGameEvent.eGameEvent_ShowSelectServerList, Show); }
在写ShowSelectServer方法之前,我们得先定义选择服务器界面组件:
#region 选择服务器界面 private UIPanel m_Panel_Select;//选择服务器界面整体Panel private UIButton m_Button_Select;//确认选择按钮 private UIButton m_Button_LeftButton; private UIButton m_BUtton_RightButton; private UIButton m_Button_ServerListButton;//服务器列表按钮 private XUIList m_List_Dianxin; private XUIList m_List_Wangtong; private XUISprite m_Sprite_IconBG; private XUISprite m_Sprite_ServerName; private XUISprite m_Sprite_Icon; private XUILabel m_Label_ServerName; private XUILabel m_Label_NetworkSpeed;//测试速度Label #endregion
InitWeight:
this.m_Button_ServerListButton = this.mRoot.FindChild("Select/Button/ServerListButton").GetComponent<UIButton>(); this.m_Button_Select = this.mRoot.FindChild("Select/Button/SelectButton").GetComponent<UIButton>(); this.m_Panel_Select = this.mRoot.FindChild("Select").GetComponent<UIPanel>(); this.m_List_Dianxin = this.mRoot.FindChild("Select/ServerList/Table/Dianxin/DianxinGrid").GetComponent<XUIList>(); this.m_List_Wangtong = this.mRoot.FindChild("Select/ServerList/Table/Wangtong/WangtongGrid").GetComponent<XUIList>(); this.m_Sprite_IconBG = this.mRoot.FindChild("Select/Signal/IconBG").GetComponent<XUISprite>(); this.m_Sprite_Icon = this.mRoot.FindChild("Select/Signal/IconAnim").GetComponent<XUISprite>(); this.m_Sprite_ServerName = this.mRoot.FindChild("Select/Signal/Name/ServerNameIcon").GetComponent<XUISprite>();
OK,初始化好界面后,我们回过头写ShowSelectServer:
/// <summary> /// 显示选择服务器界面 /// </summary> public void ShowSelectServer() { if (this.m_Panel_Select != null) { if (this.ShowServerList()) { //服务器列表按钮的sprite替换 if (this.m_bShowServerList) { this.m_Button_ServerListButton.normalSprite = "image 378"; } else { this.m_Button_ServerListButton.normalSprite = "image 383"; } this.m_Panel_Select.enabled = true; GameObject serverList = this.m_Panel_Select.transform.FindChild("ServerList").gameObject; //serverList. serverList.SetActive(this.m_bShowServerList); //如果显示服务器列表面板,播放偏移动画 if (this.m_bShowServerList) { serverList.transform.localPosition = new Vector3(14, 0, 0); TweenAlpha.Begin(serverList, 1, 1); TweenPosition.Begin(serverList, 1, Vector3.zero); } } else { Debug.LogError("服务器列表还没有初始化"); } } }
/// <summary> /// 显示服务器列表 /// </summary> public bool ShowServerList() { if (this.m_bHasLoadedServerList) { return true; } if (SystemConfig.ServerList != null) { this.m_selectedServerId = SystemConfig.SelectedServerIndex; this.m_lastSelectServerId = SystemConfig.SelectedServerIndex; int indexDianxin = 0; int indexWangtong = 0; foreach (var serverInfo in SystemConfig.ServerList) { IXUIListItem serverItem; switch (serverInfo.type) { case 0: if (serverInfo.flag == (int)ServerType.Recommend) { this.m_reDianxinServerId = serverInfo.id; } if (indexDianxin < this.m_List_Dianxin.Count) { serverItem = this.m_List_Dianxin.GetItemByIndex(indexDianxin); } else { serverItem = this.m_List_Dianxin.AddListItem(); } if (serverItem != null) { serverItem.SetText("ServerName",serverInfo.name); serverItem.SetVisible(true); serverItem.TipParam = new TipParam { Tip = serverItem.Tip }; serverItem.Id = serverInfo.id; } else { serverItem.SetVisible(false); } indexDianxin++; break; case 1: if (serverInfo.flag == (int)ServerType.Recommend) { this.m_reWangtongServerId = serverInfo.id; } if (indexWangtong < this.m_List_Wangtong.Count) { serverItem = this.m_List_Wangtong.GetItemByIndex(indexWangtong); } else { serverItem = this.m_List_Wangtong.AddListItem(); } if (serverItem != null) { serverItem.SetText("ServerName", serverInfo.name); serverItem.SetVisible(true); serverItem.TipParam = new TipParam { Tip = serverItem.Tip }; serverItem.Id = serverInfo.id; } else { serverItem.SetVisible(false); } indexWangtong++; break; } } this.m_bHasLoadedServerList = true; return true; } else { return false; } }
#region 变量定义 private bool m_bShowServerList = false;//是否显示服务器列表 private bool m_bHasLoadedServerList = false;//是否已经加载过服务器列表 private int m_selectedServerId = -1; private int m_lastSelectServerId = SystemConfig.SelectedServerIndex; private int m_reDianxinServerId = -1;//电信推荐服务器id private int m_reWangtongServerId = -1;//网通推荐服务器idn #endregion
然后在InitWeight中,让m_Button_ServerListButton监听显示服务器列表和不显示的事件监听:
UIEventListener.Get(this.m_Button_ServerListButton.gameObject).onClick = (x) => { this.m_bShowServerList = !this.m_bShowServerList; ShowSelectServer(); };
还有我们点击某个服务器,得把那个服务器的id,记录下来,所以我们还是注册一个事件:
this.m_List_Dianxin.RegisterListOnClickEventHandler(new ListOnClickEventHandler(this.OnServerListItemClick));
/// <summary> /// 服务器某个被点击 /// </summary> /// <param name="item"></param> /// <returns></returns> private bool OnServerListItemClick(IXUIListItem item) { if (null == item) { return false; } this.m_selectedServerId = item.Id; ServerInfo info = SystemConfig.GetServerInfoById(this.m_selectedServerId); bool active = true; if (info.flag == (int)ServerType.Close || info.flag == (int)ServerType.Maintain) { active = false; } ShowServerSignal(active, info); return true; } /// <summary> /// 播放动画 /// </summary> /// <param name="bActive"></param> /// <param name="info"></param> private void ShowServerSignal(bool bActive,ServerInfo info) { if (bActive) { if (info.id == 0) { this.m_Sprite_Icon.SetSprite("image 967", "Atlas/SelectAtlas/SelectServerAtlas"); this.m_Sprite_ServerName.SetSprite("image 975"); } else { this.m_Sprite_Icon.SetSprite("image 1015", "Atlas/SelectAtlas/SelectServerAtlas"); this.m_Sprite_ServerName.SetSprite("image 1012"); } if (this.m_Sprite_IconBG != null) { this.m_Sprite_IconBG.PlayFlash(false); this.m_Sprite_ServerName.Alpha = 0f; this.m_Sprite_Icon.Alpha = 0f; TweenAlpha.Begin(this.m_Sprite_ServerName.gameObject, 0.8f, 1f); TweenAlpha.Begin(this.m_Sprite_Icon.gameObject, 0.8f, 1f); } } }
OK,大致就是这么多的代码,具体步骤:
1.如果选择服务器Panel不激活的话,就显示服务器列表,然后激活。
2.注册一些按钮的事件
3.记录下来选择的服务器id
然后我们点击选择按钮,就发送这个服务器id给游戏网关服务器,网关服务器就把用户连接到这个游戏服务器中。
下节课,我们就开始进入游戏服务器连接,当然这里就涉及到NetworkManager的编写,主要是TCp的连接。