XmdsZoneNode.cs 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456
  1. using CommonLang.Log;
  2. using CommonAI.Zone;
  3. using CommonAI.Zone.Instance;
  4. using System;
  5. using System.Collections.Generic;
  6. using System.Web.Helpers;
  7. using XmdsServerNode.Node;
  8. using XmdsCommon.Message;
  9. using XmdsCommon.Plugin;
  10. using XmdsCommonServer.Plugin;
  11. using XmdsCommonServer.Plugin.Units;
  12. using CommonAI.Zone.ZoneEditor;
  13. using XmdsCommon.EditorData;
  14. using CommonLang.Vector;
  15. using CommonLang;
  16. using static CommonAI.XmdsConstConfig;
  17. using CommonAI.Data;
  18. namespace XmdsServerEdgeJS.Zone
  19. {
  20. public class XmdsZoneNode
  21. {
  22. private readonly Logger log;
  23. private readonly ZoneService service;
  24. private readonly string instanceID;
  25. private readonly ServerZoneNode node;
  26. //连斩生效时间
  27. private long continueKillsValidTime = 0;
  28. public string InstanceID { get { return instanceID; } }
  29. public ServerZoneNode Node { get { return node; } }
  30. public void SendToGameServer(string name, Object param)
  31. {
  32. node.SendToGameServer(name, param);
  33. }
  34. public string GetBindGameSrvId()
  35. {
  36. return node.GetBindGameSrvId();
  37. }
  38. internal XmdsZoneNode(ZoneService svc, string instanceId, string gameServerId)
  39. {
  40. this.log = LoggerFactory.GetLogger(string.Format("XmdsZoneNode-{0}", instanceId));
  41. this.service = svc;
  42. this.instanceID = instanceId;
  43. this.node = new ServerZoneNode(gameServerId);
  44. }
  45. internal void Start(int mapTemplateId, string data, Action<XmdsZoneNode> started)
  46. {
  47. dynamic config = Json.Decode(data);
  48. this.node.OnZoneStart += (z) => { started(this); };
  49. string monsterHard = (string)config.monsterHard;
  50. byte sceneType = (byte)config.sceneType;
  51. byte areaType = (byte)config.areaType;
  52. int averageLevel = Convert.ToInt32(config.averageLevel);
  53. double propRatio = 0;
  54. double floorRatio = 0;
  55. if (averageLevel > 0)
  56. {
  57. propRatio = Convert.ToDouble(config.propRatio);
  58. floorRatio = Convert.ToDouble(config.floorRatio);
  59. }
  60. bool canRiding = true;
  61. bool isteam = config.isTeam == null ? false : (bool)config.isTeam;
  62. if (config.canRiding != null)
  63. {
  64. canRiding = config.canRiding;
  65. }
  66. int killInterval = (config.killInterval != null) ? (int)(config.killInterval) : 0;
  67. int killFullNum = (config.killFullNum != null) ? (int)(config.killFullNum) : 0;
  68. int killFullCollSec = (config.killFullCollSec != null) ? (int)(config.killFullCollSec) : 0;
  69. GSCreateAreaData gsData = new GSCreateAreaData();
  70. gsData.value1 = (config.value1 != null) ? ((int)(config.value1)) : 0;
  71. gsData.value2 = (config.value2 != null) ? ((int)(config.value2)) : 0;
  72. this.node.Start(instanceID, mapTemplateId , config.monsterHard, (byte)config.allowAutoGuard, config.calPKValue,
  73. averageLevel, floorRatio, propRatio, killInterval, killFullNum, killFullNum, config.usespaceDiv,
  74. (CommonAI.Data.SceneType)sceneType, gsData, (AreaType)areaType, isteam, canRiding);
  75. this.node.Zone.OnUnitDead += zone_onUnitDead;
  76. //this.node.Zone.OnGameOver += zone_onGameOver;
  77. this.node.Zone.OnUnitGotInstanceItem += zone_onUnitGotInstanceItem;
  78. this.node.Zone.OnUnitPickUnit += zone_onUnitPickUnit;
  79. this.node.Zone.OnSendMessageToGS += zone_onSendMessageToGS;
  80. this.node.Zone.OnTryPickItem += zone_onTryPickItem;
  81. }
  82. internal void Stop(Action<XmdsZoneNode> stopped)
  83. {
  84. if (node.IsDisposed)
  85. {
  86. stopped(this);
  87. }
  88. else
  89. {
  90. node.OnZoneStop += (z) => { stopped(this); };
  91. node.Stop();
  92. }
  93. }
  94. //------------------------------------------------------------------------------------------------------
  95. private HashMap<int, TVector2> _BornPlace = null;
  96. private HashMap<String, TVector2> _BornPlaceExt = null;
  97. public TVector2 r2b_get_getBornPlace(int areaId, string pointId)
  98. {
  99. lock (this)
  100. {
  101. if (_BornPlace == null)
  102. {
  103. _BornPlace = new HashMap<int, TVector2>();
  104. _BornPlaceExt = new HashMap<string, TVector2>();
  105. List<DecorationData> Decorations = this.Node.SceneData.Decorations;
  106. foreach (DecorationData decoration in Decorations)
  107. {
  108. foreach (AbilityData ability in decoration.Abilities)
  109. {
  110. XmdsSceneTransportAbilityData sceneAbility = ability as XmdsSceneTransportAbilityData;
  111. if (sceneAbility != null)
  112. {
  113. if (!string.IsNullOrEmpty(sceneAbility.NextScenePosition))
  114. {
  115. string[] pos = sceneAbility.NextScenePosition.Split(new Char[] { ',' });
  116. TVector2 posInfo = new TVector2((float)Convert.ToSingle(pos[0]), (float)Convert.ToSingle(pos[1]));
  117. _BornPlace.Put(sceneAbility.NextSceneID, posInfo);
  118. _BornPlaceExt.Put(decoration.Name, posInfo);
  119. }
  120. }
  121. }
  122. }
  123. }
  124. TVector2 ret;
  125. if(pointId != null && _BornPlaceExt.TryGetValue(pointId, out ret))
  126. {
  127. return ret;
  128. }
  129. else if (_BornPlace.TryGetValue(areaId, out ret))
  130. {
  131. return ret;
  132. }
  133. return new TVector2(0, 0);
  134. }
  135. }
  136. //------------------------------------------------------------------------------------------------------
  137. #region _场景内事件_
  138. private bool zone_onTryPickItem(InstanceZone zone, InstanceUnit unit, InstanceItem item)
  139. {
  140. XmdsInstancePlayer player = unit as XmdsInstancePlayer;
  141. XmdsItemProperties prop = item.Info.Properties as XmdsItemProperties;
  142. if (player != null && prop != null && prop.IsTaskItem)
  143. {
  144. List<CommonAI.Zone.Helper.QuestData> quests = player.GetQuests();
  145. for (int i = 0; i < quests.Count; i++)
  146. {
  147. var q = quests[i];
  148. var t = XmdsQuestData.GetTaskType(q);
  149. var targetId = XmdsQuestData.GetTargetID(q);
  150. var is_complete = XmdsQuestData.IsTargetComplete(q);
  151. if (!is_complete && t == (int)XmdsQuestData.TaskType.InterActiveItem && targetId == item.Info.TemplateID)
  152. {
  153. return true;
  154. }
  155. }
  156. return false;
  157. }
  158. else
  159. {
  160. return true;
  161. }
  162. }
  163. private void zone_onSendMessageToGS(InstanceZone zone, CommonAI.ZoneServer.SendMessageB2R e)
  164. {
  165. var eventParam = new
  166. {
  167. instanceId = zone.UUID,
  168. eventName = "message",
  169. msg = e.Message
  170. };
  171. node.SendToGameServer(EventType.areaEvent.ToString(), eventParam);
  172. }
  173. private void zone_onUnitPickUnit(InstanceZone zone, InstanceUnit obj, InstanceUnit pickable)
  174. {
  175. if (obj is InstancePlayer)
  176. {
  177. string uuid = (obj as InstancePlayer).PlayerUUID;
  178. var eventParam = new
  179. {
  180. eventName = "interActiveItem",
  181. playerId = uuid,
  182. itemId = pickable.Info.TemplateID,
  183. x = obj.X,
  184. y = obj.Y
  185. };
  186. node.SendToGameServer(EventType.playerEvent.ToString(), eventParam);
  187. }
  188. }
  189. private void zone_onUnitGotInstanceItem(InstanceZone zone, InstanceUnit unit, InstanceItem item)
  190. {
  191. InstancePlayer player = (unit as InstancePlayer);
  192. string uuid = player.PlayerUUID;
  193. XmdsDropableInstanceItem it = item as XmdsDropableInstanceItem;
  194. XmdsItemProperties prop = it.Info.Properties as XmdsItemProperties;
  195. XmdsCommon.Message.DropItem dropitem = it.GenSyncInfo(false).ExtData as XmdsCommon.Message.DropItem;
  196. if (dropitem != null)
  197. {
  198. var eventParam = new
  199. {
  200. eventName = "pickItem",
  201. instanceId = zone.UUID,
  202. playerId = uuid,
  203. itemId = dropitem.ObjID,
  204. isGuard = player.IsGuard,
  205. objectId = item.Info.ID,
  206. //type = prop.ItemType,
  207. //templateID = it.Info.TemplateID
  208. };
  209. node.SendToGameServer(EventType.areaEvent.ToString(), eventParam);
  210. //u.queueEvent(new UnitGotInstanceItemEvent(u.ID, this.ID));
  211. }
  212. else
  213. {
  214. dynamic eventParam = new
  215. {
  216. eventName = "interActiveItem",
  217. playerId = uuid,
  218. itemId = item.Info.TemplateID,
  219. objId = item.ID,
  220. type = prop.ItemType,
  221. isRemove = item.IsPickAll(),
  222. x = item.X,
  223. y = item.Y
  224. };
  225. node.SendToGameServer(EventType.playerEvent.ToString(), eventParam);
  226. }
  227. }
  228. /// <summary>
  229. /// 单位死亡事件
  230. /// </summary>
  231. /// <param name="zone"></param>
  232. /// <param name="obj"></param>
  233. /// <param name="attacker"></param>
  234. private void zone_onUnitDead(InstanceZone zone, InstanceUnit obj, InstanceUnit attacker)
  235. {
  236. //玩家死亡为1,怪物死亡为0
  237. int type = 0;
  238. //玩家id
  239. string playerId = null;
  240. long pkTime = 0;
  241. int pkValue = 0;
  242. if (obj is XmdsInstancePlayer)
  243. {
  244. type = 1;//玩家.
  245. playerId = (obj as InstancePlayer).PlayerUUID;
  246. XmdsVirtual_Player virPlayer = obj.Virtual as XmdsVirtual_Player;
  247. pkTime = virPlayer.getPkTime();
  248. pkValue = virPlayer.getPkValue();
  249. }
  250. else if (obj is XmdsInstancePet && !(obj is XmdsInstanceSummonUnit))
  251. {
  252. type = 2;//宠物.
  253. var pet = obj as XmdsInstancePet;
  254. if (pet.Master != null && pet.Master is InstancePlayer)
  255. {
  256. playerId = (pet.Master as InstancePlayer).PlayerUUID;
  257. }
  258. }
  259. int attackType = 0;
  260. //如果最后一击是宠物,则找主人
  261. if (attacker is InstancePet)
  262. {
  263. attacker = (attacker as InstancePet).Master;
  264. attackType = 1;
  265. }
  266. else if (attacker is InstanceSummon)
  267. {
  268. attacker = (attacker as InstanceSummon).SummonerUnit;
  269. }
  270. else if (attacker is XmdsInstanceSummonUnit)
  271. {
  272. attacker = (attacker as XmdsInstanceSummonUnit).SummonerUnit;
  273. }
  274. InstancePlayer hitFinal = attacker as InstancePlayer;
  275. //击中列表
  276. List<string> playerIds = new List<string>(2);
  277. //玩家被杀,改玩家添加到列表
  278. if (type == 1)
  279. {
  280. playerIds.Add(playerId);
  281. }
  282. if (hitFinal != null)
  283. {
  284. playerIds.Add(hitFinal.PlayerUUID);
  285. }
  286. #region 怪物ID特殊逻辑.
  287. //怪物生成时会随机出更高级的怪物,能力ID都会变更,但模板ID却一样。游戏服取MonsterID查找掉落和经验.Edit by Alex.Yu.
  288. int tpID = obj.Info.TemplateID;
  289. //击杀助攻列表.
  290. List<string> atkAssistantList = null;
  291. if (type == 0 && obj is XmdsInstanceMonster)
  292. {
  293. tpID = (obj as XmdsInstanceMonster).MonsterID;
  294. if (tpID <= 0)
  295. {
  296. tpID = obj.Info.TemplateID;
  297. log.Info("zone_onUnitDead单位id异常:" + obj.Info.TemplateID + ", " + obj.Info.Name + ", obj: " + obj + ", " + zone.GetSceneID());
  298. }
  299. atkAssistantList = (obj.Virtual as XmdsVirtual).mHateSystem.GetAtkAssistantList();
  300. }
  301. else if(type == 1)
  302. {
  303. atkAssistantList = (obj.Virtual as XmdsVirtual).mHateSystem.GetAtkAssistantList();
  304. }
  305. if (type != 2 && playerIds.Count == 0 && (atkAssistantList == null || atkAssistantList.Count == 0))
  306. {
  307. return;
  308. }
  309. #endregion
  310. //log.Info("monster dead objId:" + obj.ID + " templateId:" + tpID + " type " + type);
  311. int aID = 0;
  312. int aQColor = 0;
  313. int aLevel = 0;
  314. string aName = "";
  315. string sceneType = "";
  316. int attackID = attacker.Info.ID;
  317. try
  318. {
  319. if (attacker is XmdsInstanceMonster)
  320. {
  321. MonsterVisibleDataB2C mvd = attacker.GenSyncUnitInfo().VisibleInfo as MonsterVisibleDataB2C;
  322. sceneType = (attacker.Virtual as XmdsVirtual).GetCurSceneType();
  323. aID = attacker.Info.TemplateID;
  324. aQColor = (int)mvd.Qcolor;
  325. aName = mvd.DisplayName;
  326. aLevel = attacker.GenSyncUnitInfo().Level;
  327. }
  328. else if (attacker.IsPlayerUnit && obj.IsMonster)
  329. {
  330. InstanceUnit attackerUnit = attacker.IsPet ? attacker.Virtual.GetMasterUnit() : attacker;
  331. if (attackerUnit != null)
  332. {
  333. attackerUnit.Virtual.DispatchKillOtherEvent(attackerUnit.Virtual, obj.Virtual);
  334. }
  335. else
  336. {
  337. log.Warn("zone_onUnitDead: 找不到人形攻击者:" + attacker.Info.ID + ", " + zone.GetSceneID());
  338. }
  339. }
  340. }
  341. catch(Exception e)
  342. {
  343. log.Error("zone_onUnitDead catch: " + e);
  344. return;
  345. }
  346. //连斩数计算
  347. if(zone is EditorScene)
  348. {
  349. EditorScene scene = zone as EditorScene;
  350. if(scene != null && scene.Data.killInterval > 0 && attacker != null)
  351. {
  352. long curTime = CommonLang.CUtils.localTimeMS;
  353. if (continueKillsValidTime < curTime && attacker.Statistic != null &&
  354. curTime - attacker.Statistic.preKillTime < scene.Data.killInterval)
  355. {
  356. attacker.updateContinueKills();
  357. //如果连斩有上限,并且达到上限后有一定冷却时间
  358. if (scene.Data.killMax > 0 && scene.Data.killMaxCoolTime > 0 && attacker.Statistic.continueKills >= scene.Data.killMax)
  359. {
  360. attacker.Statistic.continueKills = 0;
  361. this.continueKillsValidTime = curTime + scene.Data.killMaxCoolTime * 1000;
  362. }
  363. }
  364. else
  365. {
  366. attacker.resetContinueKills();
  367. }
  368. attacker.Statistic.preKillTime = curTime;
  369. }
  370. }
  371. XmdsVirtual vir = (obj.Virtual as XmdsVirtual);
  372. XmdsHateSystem zhs = vir.GetHateSystem() as XmdsHateSystem;
  373. InstancePlayer belongPlayer = zhs.GetHeirs() as InstancePlayer;
  374. //Console.Write("attacker objId:" + aID + " aQColor:" + aQColor + " aName " + aName + " aLevel " + aLevel);
  375. var eventParam = new
  376. {
  377. eventName = "unitDead",
  378. playerIds = playerIds,
  379. areaId = zone.TerrainSrc.TemplateID,
  380. instanceId = zone.UUID,
  381. hitFinal = (hitFinal == null) ? null : hitFinal.PlayerUUID,
  382. belongPlayerId = (belongPlayer == null) ? null : belongPlayer.PlayerUUID,
  383. unitType = type,
  384. unitPlayerId = playerId,
  385. unitTemplateId = tpID,
  386. unitLevel = obj.Level,
  387. objId = obj.ID,
  388. posX = obj.get_x(),
  389. posY = obj.get_y(),
  390. attackerTemplateId = aID,
  391. attackerQColor = aQColor,
  392. attackerName = aName,
  393. attackerLevel = aLevel,
  394. attackerSceneType = sceneType,
  395. attackType = attackType,
  396. atkAssistantList = atkAssistantList,
  397. awardPlayer = zhs.GetSharedList(),
  398. refreshPoint = obj.GetAttribute("RefreshMonsterTag"),
  399. pkTime = pkTime,
  400. pkValue = pkValue,
  401. continuekills = attacker == null ? 0 : attacker.Statistic.continueKills,
  402. gsFlag = obj.gameServerFlag
  403. };
  404. node.SendToGameServer(EventType.areaEvent.ToString(), eventParam);
  405. }
  406. #endregion
  407. }
  408. }