InstanceSpell.cs 57 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. using CommonAI.RTS;
  5. using CommonLang.Vector;
  6. using CommonAI.ZoneClient;
  7. using CommonLang;
  8. using CommonAI.Zone.Formula;
  9. using CommonAI.Zone.Helper;
  10. using static CommonAI.Zone.Instance.InstanceUnit;
  11. using CommonAI.ZoneServer.JSGModule;
  12. using CommonAI.Data;
  13. namespace CommonAI.Zone.Instance
  14. {
  15. public class InstanceSpell : InstanceZoneObject
  16. {
  17. readonly protected SpellTemplate mInfo;
  18. readonly protected SyncSpellInfo mSyncInfo;
  19. readonly protected LaunchSpell mLaunchData;
  20. // 释放者
  21. readonly protected InstanceUnit mLauncherUnit;
  22. // 发出者
  23. readonly protected InstanceZoneObject mSender;
  24. readonly protected AttackRangeHelper AttackRange;
  25. protected Vector2 mStartPos;
  26. protected float mLaunchDirection;
  27. // 被跟踪目标
  28. protected InstanceUnit mTarget;
  29. protected Vector2 mTargetPos;
  30. protected TimeExpire<int> mSeekingCooldownTime;
  31. protected SpellChainLevelInfo mChainInfo;
  32. /// <summary>
  33. /// 如果是技能产生的伤害,就是该技能的类型(就算是法术,也要追根到技能)
  34. /// </summary>
  35. public readonly XmdsSkillType FromSkillType = XmdsSkillType.none;
  36. private float mBaseSize;
  37. private float mDistance;
  38. private float mSpeed;
  39. private PopupKeyFrames<SpellTemplate.KeyFrame> mKeyFrames;
  40. private Dictionary<uint, InstanceUnit> mHittedUnits;
  41. private bool Finish = false;
  42. private TimeInterval<SpellTemplate.KeyFrame> mHitIntervalTicker;
  43. private InstanceUnit.StateSkill mBindingSkill;
  44. private bool mClientVisible;
  45. private bool mSyncPos;
  46. private float mMoveDistance;
  47. //服务器专用的扩展字段
  48. private int actionIndex;
  49. private int mMaxAffectUnit;
  50. //法术创建时,玩家的AOI状态
  51. private byte aoiFlag = 0;
  52. //每隔N秒触发,触发次数
  53. private int mHitTotalTimes;
  54. //法术总时长
  55. private readonly int mSpellTotalTime; //法术的总时长
  56. // 飞弹各种奇葩数据记录, 法术数量可控
  57. private JSGCreateSpellData mCreateSpellData; //法术创建信息(可变的,由技能数据确定)
  58. // 飞弹不同的法术id
  59. public List<LaunchSpell> mCurveSpellList;
  60. //连续飞弹
  61. private byte mCurveSpellIndex;
  62. private TVector2 mPrvePos = new TVector2();
  63. public InstanceSpell(
  64. InstanceZone zone,
  65. SpellTemplate data,
  66. LaunchSpell launch,
  67. InstanceUnit launcher,
  68. InstanceZoneObject sender,
  69. Dictionary<uint, InstanceUnit> damageList,
  70. XmdsSkillType fromSkillType = XmdsSkillType.none,
  71. int spellIndex = 0,
  72. JSGCreateSpellData createData = null)
  73. : base(zone, false)
  74. {
  75. this.mSpellTotalTime = data.LifeTimeMS;
  76. this.mCurveSpellIndex = (byte)spellIndex;
  77. SenderID = 0;
  78. if(sender!=null)
  79. {
  80. if (sender is InstanceUnit)
  81. SenderID = sender.ID;
  82. else if(sender is InstanceSpell)
  83. SenderID = (sender as InstanceSpell).SenderID;
  84. }
  85. //记录AOI状态
  86. if(sender.AoiStatus != null)
  87. {
  88. this.aoiFlag = 1;
  89. }
  90. this.FromSkillType = fromSkillType;
  91. this.mLaunchData = launch;
  92. if(createData == null)
  93. {
  94. this.mCreateSpellData = new JSGCreateSpellData();
  95. this.mCreateSpellData.mMaxSpellCount = launch.Count;
  96. }
  97. else
  98. {
  99. this.mCreateSpellData = createData;
  100. }
  101. this.mKeyFrames = new PopupKeyFrames<SpellTemplate.KeyFrame>();
  102. this.mKeyFrames.AddRange(data.KeyFrames);
  103. this.mInfo = data;
  104. this.mLauncherUnit = launcher;
  105. this.mSender = sender;
  106. this.mBaseSize = data.BodySize;
  107. this.mDistance = data.Distance;
  108. this.faceTo(launcher.Direction);
  109. this.IsAffectNearChange = false;
  110. this.mSyncInfo = new SyncSpellInfo(zone.IsHalfSync);
  111. this.mSyncInfo.TemplateID = mInfo.ID;
  112. this.mSyncInfo.Force = launcher.Force;
  113. if (mInfo.HitIntervalMS <= 0)
  114. {
  115. this.mHittedUnits = (damageList == null ? new Dictionary<uint, InstanceUnit>() : damageList);
  116. }
  117. this.mClientVisible = mInfo.ClientVisible;
  118. this.mSyncPos = zone.SyncPos;
  119. this.mHitIntervalTicker = new TimeInterval<SpellTemplate.KeyFrame>(data.HitIntervalMS);
  120. this.mHitIntervalTicker.Tag = data.HitIntervalKeyFrame;
  121. this.mHitTotalTimes = 0;
  122. this.AttackRange = new AttackRangeHelper(launcher);
  123. this.mSpeed = mInfo.MSpeedSEC;
  124. this.mMoveDistance = 0;
  125. if(launcher.AoiStatus != null)
  126. {
  127. this.setAoiStatus(launcher.AoiStatus);
  128. }
  129. }
  130. public SpellTemplate Info { get { return mInfo; } }
  131. public LaunchSpell LaunchData { get { return mLaunchData; } }
  132. /// <summary>
  133. /// 最先技能的发起者
  134. /// </summary>
  135. public InstanceUnit Launcher { get { return mLauncherUnit; } }
  136. public uint LauncherID { get { return mLauncherUnit.ID; } }
  137. /// <summary>
  138. /// 技能的出口,比如技能触发技能,则Sender就是一个Spell
  139. /// </summary>
  140. public InstanceZoneObject Sender { get { return mSender; } }
  141. public InstanceUnit Target { get { return mTarget; } }
  142. public override bool IntersectMap { get { return false; } }
  143. public override bool IntersectObj { get { return false; } }
  144. public override bool Moveable { get { return true; } }
  145. public override float BodyBlockSize { get { return mBaseSize; } }
  146. public override float BodyHitSize { get { return mBaseSize; } }
  147. public override float BodyHeight { get { return mInfo.BodySize; } }
  148. public override float Weight { get { return 0; } }
  149. public override bool ClientVisible { get { return mClientVisible; } }
  150. public override bool SyncPos { get { return mSyncPos; } }
  151. public int BindActionIndex {get{ return actionIndex; } set { this.actionIndex = value; } }
  152. public int MaxAffectUnit { get { return mMaxAffectUnit; } set { this.mMaxAffectUnit = value; } }
  153. /** 获取触发次数 */
  154. public int GetHitTimes()
  155. {
  156. return this.mHitTotalTimes;
  157. }
  158. /// <summary>
  159. /// 连锁等级
  160. /// </summary>
  161. public int ChainLevel { get { return (mChainInfo != null) ? mChainInfo.Level : 0; } }
  162. public SpellChainLevelInfo ChainInfo { get { return mChainInfo; } }
  163. public uint SenderID;
  164. public uint TargetID { get { return (mTarget != null) ? mTarget.ID : 0; } }
  165. public Vector2 TargetPos { get { return mTargetPos; } }
  166. public float LaunchHeightZ
  167. {
  168. get
  169. {
  170. if (mSender != null && mSender is InstanceSpell)
  171. {
  172. var obj = mSender as InstanceSpell;
  173. if (obj.mInfo.MType == SpellTemplate.MotionType.Cannon)
  174. {
  175. return 0;
  176. }
  177. else
  178. {
  179. return mSender.Z;
  180. }
  181. }
  182. float h = mLaunchData.LaunchSpellHeight;
  183. if (mLaunchData.FromUnitBody)
  184. {
  185. h = Launcher.Info.LaunchSpellHeight;
  186. }
  187. return h;
  188. }
  189. }
  190. public override SyncObjectInfo GenSyncInfo(bool net)
  191. {
  192. return GenSyncSpellInfo(net);
  193. }
  194. public SyncSpellInfo GenSyncSpellInfo(bool net = false)
  195. {
  196. mSyncInfo.x = X;
  197. mSyncInfo.y = Y;
  198. mSyncInfo.direction = this.Direction;
  199. mSyncInfo.HasSpeed = mInfo.IsMoveable && mSpeed != mInfo.MSpeedSEC;
  200. mSyncInfo.CurSpeed = mSpeed;
  201. return mSyncInfo;
  202. }
  203. // 如果是导弹类型,则需要目标
  204. public void setTarget(InstanceUnit target)
  205. {
  206. this.mTarget = target;
  207. }
  208. public void setTargetPos(Vector2 targetPos)
  209. {
  210. this.mTargetPos = targetPos;
  211. }
  212. public void setChainInfo(SpellChainLevelInfo c)
  213. {
  214. this.mChainInfo = c;
  215. }
  216. protected override void onAdded(bool pointLv)
  217. {
  218. if (mLaunchData.IsAutoSeekingTarget && mLaunchData.PType != LaunchSpell.PosType.POS_TYPE_AREA)
  219. {
  220. //目标不存在,或者已死亡才重新寻找, 不能打开,弹射逻辑
  221. //if(mTarget == null || mTarget.IsDead())
  222. {
  223. mTarget = Parent.SeekSpellAttackable(this.Launcher, this.Info, this.Launcher.AoiStatus,
  224. X, Y, mLaunchData.SeekingTargetRange, Info.ExpectTarget, mLaunchData.SeekingTargetExpect, mChainInfo);
  225. //Console.WriteLine("重新索敌:" + mTarget.ID);
  226. }
  227. }
  228. this.mSyncInfo.ObjectID = base.ID;
  229. this.mLaunchDirection = this.Direction;
  230. float radius = mLaunchData.LaunchSpellRadius;
  231. if (mLaunchData.FromUnitBody && (Sender is InstanceUnit))
  232. {
  233. radius = (Sender as InstanceUnit).Info.LaunchSpellRadius;
  234. }
  235. mStartPos = new Vector2(X, Y);
  236. switch (mInfo.MType)
  237. {
  238. case SpellTemplate.MotionType.SelectTarget:
  239. if (mTarget != null)
  240. {
  241. mStartPos.SetX(mTarget.X);
  242. mStartPos.SetY(mTarget.Y);
  243. }
  244. MathVector.movePolar(mStartPos, mLaunchDirection, radius);
  245. break;
  246. case SpellTemplate.MotionType.SelectLauncher:
  247. if (mLauncherUnit != null)
  248. {
  249. mStartPos.SetX(mLauncherUnit.X);
  250. mStartPos.SetY(mLauncherUnit.Y);
  251. }
  252. MathVector.movePolar(mStartPos, mLaunchDirection, radius);
  253. break;
  254. case SpellTemplate.MotionType.Immovability:
  255. MathVector.movePolar(mStartPos, mLaunchDirection, radius);
  256. break;
  257. case SpellTemplate.MotionType.Cannon:
  258. if (mTargetPos != null)
  259. {
  260. MathVector.moveTo(mStartPos, mTargetPos.X, mTargetPos.Y, radius);
  261. }
  262. else if (mTarget != null)
  263. {
  264. mTargetPos = new Vector2(mTarget.X, mTarget.Y);
  265. MathVector.moveTo(mStartPos, mTargetPos.X, mTargetPos.Y, radius);
  266. }
  267. break;
  268. case SpellTemplate.MotionType.AOE:
  269. MathVector.movePolar(mStartPos, mLaunchDirection, radius);
  270. break;
  271. case SpellTemplate.MotionType.AOE_Binding:
  272. break;
  273. case SpellTemplate.MotionType.AOE_BindingTarget:
  274. break;
  275. case SpellTemplate.MotionType.Straight:
  276. MathVector.movePolar(mStartPos, mLaunchDirection, radius);
  277. break;
  278. case SpellTemplate.MotionType.Binding:
  279. break;
  280. case SpellTemplate.MotionType.BindingTarget:
  281. break;
  282. case SpellTemplate.MotionType.Missile:
  283. if (mTarget != null)
  284. {
  285. MathVector.moveTo(mStartPos, mTarget.X, mTarget.Y, radius);
  286. }
  287. else
  288. {
  289. MathVector.movePolar(mStartPos, mLaunchDirection, radius);
  290. }
  291. break;
  292. case SpellTemplate.MotionType.MissileAttackRoute:
  293. if (mTarget != null)
  294. {
  295. MathVector.moveTo(mStartPos, mTarget.X, mTarget.Y, radius);
  296. }
  297. else
  298. {
  299. MathVector.movePolar(mStartPos, mLaunchDirection, radius);
  300. }
  301. //设置索敌间隔
  302. if(this.Info.SeekingCooldownMS > XmdsConstConfig.GAME_UPDATE_INTERVAL_MS)
  303. {
  304. this.mSeekingCooldownTime = new TimeExpire<int>(mInfo.SeekingCooldownMS);
  305. }
  306. else
  307. {
  308. log.Error("法术:" + this.Info.ID + ", 索敌间隔配置异常!");
  309. }
  310. break;
  311. case SpellTemplate.MotionType.SeekerMissile:
  312. if (mTarget != null)
  313. {
  314. MathVector.moveTo(mStartPos, mTarget.X, mTarget.Y, radius);
  315. }
  316. else
  317. {
  318. MathVector.movePolar(mStartPos, mLaunchDirection, radius);
  319. if (mInfo.SeekingCooldownMS == 0)
  320. {
  321. mTarget = seekAttackable(mInfo.SeekingRange);
  322. }
  323. else
  324. {
  325. this.mSeekingCooldownTime = new TimeExpire<int>(mInfo.SeekingCooldownMS);
  326. }
  327. }
  328. break;
  329. case SpellTemplate.MotionType.SeekerSelectTarget:
  330. if (mTarget == null)
  331. {
  332. if (mInfo.SeekingCooldownMS == 0)
  333. {
  334. mTarget = seekAttackable(mInfo.SeekingRange);
  335. if (mTarget != null)
  336. {
  337. mStartPos.SetX(mTarget.X);
  338. mStartPos.SetY(mTarget.Y);
  339. }
  340. }
  341. else
  342. {
  343. this.mSeekingCooldownTime = new TimeExpire<int>(mInfo.SeekingCooldownMS);
  344. }
  345. }
  346. break;
  347. case SpellTemplate.MotionType.Chain:
  348. if (mSender != null)
  349. {
  350. mStartPos.SetX(mSender.X);
  351. mStartPos.SetY(mSender.Y);
  352. }
  353. break;
  354. case SpellTemplate.MotionType.Boomerang1:
  355. case SpellTemplate.MotionType.Boomerang2:
  356. MathVector.movePolar(mStartPos, mLaunchDirection, radius);
  357. break;
  358. case SpellTemplate.MotionType.CurveMissile:
  359. MathVector.movePolar(mStartPos, mLaunchDirection, radius);
  360. if (mInfo.SeekingCooldownMS > 0)
  361. {
  362. this.mSeekingCooldownTime = new TimeExpire<int>(mInfo.SeekingCooldownMS);
  363. }
  364. break;
  365. case SpellTemplate.MotionType.Foxfire:
  366. MathVector.movePolar(mStartPos, mLaunchDirection, radius);
  367. mTarget = null;
  368. foxfireSeekWait = 0;
  369. break;
  370. case SpellTemplate.MotionType.StraightAndStop:
  371. MathVector.movePolar(mStartPos, mLaunchDirection, radius);
  372. break;
  373. }
  374. setPos(mStartPos.X, mStartPos.Y, this.LaunchHeightZ);
  375. }
  376. //public long mDisPosingTime = 0;
  377. //public long mDisPosingTime1 = 0;
  378. protected override void Disposing()
  379. {
  380. this.mSeekingCooldownTime = null;
  381. this.mKeyFrames = null;
  382. //mDisPosingTime = CommonLang.CUtils.localTimeMS;
  383. //mDisPosingTime1 = System.DateTime.Now.Ticks;
  384. if (this.mHittedUnits != null)
  385. {
  386. this.mHittedUnits.Clear();
  387. this.mHittedUnits = null;
  388. }
  389. this.mHitIntervalTicker = null;
  390. this.mCreateSpellData = null;
  391. if(this.mCurveSpellList != null)
  392. {
  393. this.mCurveSpellList.Clear();
  394. this.mCurveSpellList = null;
  395. }
  396. base.Disposing();
  397. }
  398. protected override void onRemoved()
  399. {
  400. if (mInfo.StopBindingSkillOnRemoved)
  401. {
  402. if (this.Launcher != null && this.Launcher.CurrentState is InstanceUnit.StateSkill)
  403. {
  404. (this.Launcher.CurrentState as InstanceUnit.StateSkill).block();
  405. this.Launcher.doSomething();
  406. }
  407. }
  408. }
  409. override protected void onUpdate(bool slowRefresh)
  410. {
  411. if (IsPaused) { return; }
  412. if (Info.MaxMoveDistance == 0 || mMoveDistance < Info.MaxMoveDistance)
  413. {
  414. updateMotion();
  415. if (mInfo.IsMoveable)
  416. {
  417. MoveHelper.UpdateSpeed(Parent.UpdateIntervalMS, ref mSpeed, mInfo.MSpeedAdd, mInfo.MSpeedAcc, mInfo.MSpeed_MIN, mInfo.MSpeed_MAX);
  418. MoveHelper.UpdateMoveDistance(Parent.UpdateIntervalMS, ref mMoveDistance, mSpeed);
  419. }
  420. }
  421. if(this.mKeyFrames != null)
  422. {
  423. updateKeyFrames(slowRefresh);
  424. }
  425. if (mInfo.MType == SpellTemplate.MotionType.Foxfire && mTarget != null)
  426. {
  427. }
  428. else if(mInfo.MType == SpellTemplate.MotionType.CurveMissile && mTarget != null)
  429. {
  430. }
  431. else if (mInfo.MType == SpellTemplate.MotionType.Boomerang1 || mInfo.MType == SpellTemplate.MotionType.Boomerang2)
  432. {
  433. }
  434. else if (mInfo.MType == SpellTemplate.MotionType.Binding)//绑定自身的spell才在自己死亡时移除
  435. {
  436. if (!Sender.Enable || Sender.IsDead())
  437. {
  438. Parent.RemoveObject(this);
  439. }
  440. else if((Sender.AoiStatus == null && this.aoiFlag == 1) || (Sender.AoiStatus != null && this.aoiFlag == 0))
  441. {
  442. //玩家AOI状态改变了,也要清一下
  443. Parent.RemoveObject(this);
  444. }
  445. StateSkill skill = this.mLauncherUnit.CurrentState as StateSkill;
  446. if (skill != null && skill.Skill != null && skill.SkillData.ActionQueue.Count > 1)
  447. {
  448. //UnitActionData curSkillAction = skill.Skill.GetCurAction();
  449. if (skill.CurrentActionIndex != this.actionIndex
  450. && (0 <= this.actionIndex && this.actionIndex < skill.SkillData.ActionQueue.Count)
  451. && skill.SkillData.ActionQueue[this.actionIndex].IsCancelBySkillNext)
  452. {
  453. Parent.RemoveObject(this);
  454. }
  455. }
  456. }
  457. if (PassTimeMS >= this.mSpellTotalTime)
  458. {
  459. Parent.RemoveObject(this);//Parent.RemoveObjectByID(ID);
  460. }
  461. }
  462. /// <summary>
  463. /// 更新移动行为
  464. /// </summary>
  465. private void updateMotion()
  466. {
  467. this.mPrvePos.SetX(X);
  468. this.mPrvePos.SetY(Y);
  469. switch (mInfo.MType)
  470. {
  471. case SpellTemplate.MotionType.Immovability:
  472. case SpellTemplate.MotionType.SelectTarget:
  473. case SpellTemplate.MotionType.SelectLauncher:
  474. break;
  475. case SpellTemplate.MotionType.Cannon:
  476. if (mTargetPos != null)
  477. {
  478. if(flyToTarget(mTargetPos.X, mTargetPos.Y, mSpeed, Parent.UpdateIntervalMS))
  479. {
  480. this.Finish = true;
  481. }
  482. }
  483. else if(PassTimeMS + Parent.UpdateIntervalMS >= this.mSpellTotalTime)//没有目标,结束时触发下
  484. {
  485. if(Parent.UpdateIntervalMS > 0)
  486. {
  487. this.Finish = true;
  488. }
  489. }
  490. break;
  491. case SpellTemplate.MotionType.Straight:
  492. flyTo(mLaunchDirection, mSpeed, Parent.UpdateIntervalMS);
  493. break;
  494. case SpellTemplate.MotionType.AOE:
  495. updateAOE();
  496. break;
  497. case SpellTemplate.MotionType.AOE_Binding:
  498. updateAOE();
  499. if (mSender != null && mSender.Enable)
  500. {
  501. updateBinding(mSender);
  502. }
  503. else
  504. {
  505. Parent.RemoveObject(this);
  506. }
  507. break;
  508. case SpellTemplate.MotionType.AOE_BindingTarget:
  509. updateAOE();
  510. if (mTarget != null && mTarget.IsActive)
  511. {
  512. updateBinding(mTarget);
  513. }
  514. else
  515. {
  516. Parent.RemoveObject(this);
  517. }
  518. break;
  519. case SpellTemplate.MotionType.Binding:
  520. if (mSender != null && mSender.Enable)
  521. {
  522. updateBinding(mSender);
  523. }
  524. else
  525. {
  526. Parent.RemoveObject(this);
  527. }
  528. break;
  529. case SpellTemplate.MotionType.BindingTarget:
  530. if (mTarget != null && mTarget.IsActive)
  531. {
  532. updateBinding(mTarget);
  533. }
  534. else
  535. {
  536. Parent.RemoveObject(this);
  537. }
  538. break;
  539. case SpellTemplate.MotionType.Missile:
  540. case SpellTemplate.MotionType.MissileAttackRoute:
  541. if (mTarget != null)
  542. {
  543. if (mInfo.RotateSpeedSEC == 0)
  544. {
  545. faceTo(mTarget.X, mTarget.Y);
  546. }
  547. flyToTarget(mTarget.X, mTarget.Y, mSpeed, Parent.UpdateIntervalMS);
  548. }
  549. else
  550. {
  551. flyTo(mLaunchDirection, mSpeed, Parent.UpdateIntervalMS);
  552. }
  553. break;
  554. case SpellTemplate.MotionType.SeekerMissile:
  555. if (mTarget != null)
  556. {
  557. if (mInfo.SeekingTurningAngleSEC != 0)
  558. {
  559. flyToTargetTunning(mTarget.X, mTarget.Y, mSpeed, mInfo.SeekingTurningAngleSEC, Parent.UpdateIntervalMS);
  560. }
  561. else
  562. {
  563. faceTo(mTarget.X, mTarget.Y);
  564. flyToTarget(mTarget.X, mTarget.Y, mSpeed, Parent.UpdateIntervalMS);
  565. }
  566. }
  567. else
  568. {
  569. flyTo(mLaunchDirection, mSpeed, Parent.UpdateIntervalMS);
  570. if (mSeekingCooldownTime == null || mSeekingCooldownTime.Update(Parent.UpdateIntervalMS))
  571. {
  572. mTarget = seekAttackable(mInfo.SeekingRange);
  573. if (mTarget != null)
  574. {
  575. Parent.queueObjectEvent(this, new SpellLockTargetEvent(ID, mTarget.ID, X, Y));
  576. }
  577. }
  578. }
  579. break;
  580. case SpellTemplate.MotionType.SeekerSelectTarget:
  581. if (mTarget == null)
  582. {
  583. if (mSeekingCooldownTime == null || mSeekingCooldownTime.Update(Parent.UpdateIntervalMS))
  584. {
  585. mTarget = seekAttackable(mInfo.SeekingRange);
  586. if (mTarget != null)
  587. {
  588. this.setPos(mTarget.X, mTarget.Y);
  589. Parent.queueObjectEvent(this, new SpellLockTargetEvent(ID, mTarget.ID, X, Y));
  590. }
  591. }
  592. }
  593. break;
  594. case SpellTemplate.MotionType.Chain:
  595. if (mSender != null && mSender.Enable && mTarget != null && mTarget.IsActive)
  596. {
  597. setPos(mSender.X, mSender.Y);
  598. faceTo(mTarget.X, mTarget.Y);
  599. updateBinding(mTarget);
  600. }
  601. else
  602. {
  603. Parent.RemoveObject(this);
  604. }
  605. break;
  606. case SpellTemplate.MotionType.Boomerang1:
  607. if (PassTimeMS < mInfo.BoomerangFlyTime)
  608. {
  609. flyTo(mLaunchDirection, mSpeed, Parent.UpdateIntervalMS);
  610. }
  611. else if (PassTimeMS > mInfo.BoomerangFlyTime + mInfo.BoomerangHangtime)
  612. {
  613. if (flyToTarget(mLauncherUnit.X, mLauncherUnit.Y, mSpeed * 1.4f, Parent.UpdateIntervalMS))
  614. {
  615. this.Finish = true;
  616. }
  617. }
  618. break;
  619. case SpellTemplate.MotionType.Boomerang2:
  620. if (PassTimeMS < mInfo.BoomerangFlyTime)
  621. {
  622. flyTo(mLaunchDirection, mSpeed, Parent.UpdateIntervalMS);
  623. }
  624. else if (PassTimeMS > mInfo.BoomerangFlyTime + mInfo.BoomerangHangtime)
  625. {
  626. if (flyToTarget(mStartPos.X, mStartPos.Y, mSpeed * 1.4f, Parent.UpdateIntervalMS))
  627. {
  628. this.Finish = true;
  629. }
  630. }
  631. break;
  632. case SpellTemplate.MotionType.CurveMissile:
  633. //IPosBase targetPos = mTargetPos;
  634. IPosBase targetPos = (mTarget != null) ? (IPosBase)mTarget : (IPosBase)mTargetPos;
  635. if (targetPos == null)
  636. {
  637. flyTo(mLaunchDirection, mSpeed, Parent.UpdateIntervalMS);
  638. }
  639. else if (mInfo.SeekingTurningAngleSEC != 0)
  640. {
  641. flyToTargetTunning(targetPos.X, targetPos.Y, mSpeed, mInfo.SeekingTurningAngleSEC, Parent.UpdateIntervalMS);
  642. }
  643. else
  644. {
  645. faceTo(targetPos.X, targetPos.Y);
  646. flyToTarget(targetPos.X, targetPos.Y, mSpeed, Parent.UpdateIntervalMS);
  647. }
  648. if (mTarget == null && (mSeekingCooldownTime == null || mSeekingCooldownTime.Update(Parent.UpdateIntervalMS)))
  649. {
  650. mTarget = seekAttackable(mInfo.SeekingRange);
  651. if (mTarget != null)
  652. {
  653. Parent.queueObjectEvent(this, new SpellLockTargetEvent(ID, mTarget.ID, X, Y));
  654. }
  655. }
  656. //////创建后续个飞弹
  657. if (PassTimeMS >= XmdsConstConfig.GAME_UPDATE_INTERVAL_MS && mCurveSpellIndex > 0 && this.Launcher.IsActive)
  658. {
  659. int nextSpellIndex = Math.Max(0, mCurveSpellIndex - 1);
  660. int flag = (this.mCreateSpellData.mMaxSpellCount - mCurveSpellIndex) % 2 == 0 ? 2 : -2;
  661. //float angleOffset = (this.Target == null ? this.LaunchData.StartAngle / 3 : this.LaunchData.StartAngle);
  662. float angleOffset = JSGModule.GetCurveStartAngleOffect(mLauncherUnit, this.mTarget, this.LaunchData.StartAngle, this.Info.SeekingRange);
  663. float startDirection = this.mLaunchDirection + angleOffset * flag;
  664. float launchX = this.Launcher.X - (float)(1.0f * Math.Cos(this.Launcher.Direction));
  665. float launchY = this.Launcher.Y - (float)(1.0f * Math.Sin(this.Launcher.Direction));
  666. LaunchSpell launchSpell = this.LaunchData;
  667. if (mCurveSpellList != null && mCurveSpellList.Count > mCurveSpellIndex)
  668. {
  669. launchSpell = mCurveSpellList[mCurveSpellIndex];
  670. }
  671. InstanceSpell addSpell = this.Parent.AddSpell(this.FromSkillType, this.Info, launchSpell, this.Sender, this.mLauncherUnit, (this.mTarget == null ? 0 : this.mTarget.ID),
  672. JSGModule.GetCurveDefaultTargetPos(mLauncherUnit, this.Info.SeekingRange), launchX, launchY, startDirection, null, -1, 0, null, nextSpellIndex);
  673. addSpell.mCurveSpellList = this.mCurveSpellList;
  674. mCurveSpellIndex = 0;
  675. }
  676. break;
  677. case SpellTemplate.MotionType.Foxfire:
  678. if (mTarget == null)
  679. {
  680. foxfireRoundAngle = mLaunchDirection + PassTimeMS * 0.001f * 3;
  681. var x = mLauncherUnit.X + (float)(Math.Cos(foxfireRoundAngle + 0.3f) * mLaunchData.LaunchSpellRadius);
  682. var y = mLauncherUnit.Y + (float)(Math.Sin(foxfireRoundAngle + 0.3f) * mLaunchData.LaunchSpellRadius);
  683. flyToTarget(x, y, mSpeed, Parent.UpdateIntervalMS);
  684. if (mInfo.RotateSpeedSEC == 0)
  685. {
  686. faceTo(foxfireRoundAngle + (float)Math.PI * 0.25f);
  687. }
  688. if (foxfireSeekWait >= this.Info.SeekingCooldownMS && PassTimeMS < this.mSpellTotalTime - 500)
  689. {
  690. var tar = seekAttackable(mInfo.SeekingRange);
  691. if (tar != null && tar != mLauncherUnit)
  692. {
  693. changeBrotherSeekWait(-250);
  694. Parent.queueObjectEvent(this, new SpellLockTargetEvent(ID, tar.ID, X, Y));
  695. mTarget = tar;
  696. }
  697. }
  698. foxfireSeekWait += Parent.UpdateIntervalMS;
  699. }
  700. else
  701. {
  702. if (mInfo.SeekingTurningAngleSEC != 0)
  703. {
  704. flyToTargetTunning(mTarget.X, mTarget.Y, mSpeed, mInfo.SeekingTurningAngleSEC, Parent.UpdateIntervalMS);
  705. }
  706. else
  707. {
  708. faceTo(mTarget.X, mTarget.Y);
  709. flyToTarget(mTarget.X, mTarget.Y, mSpeed, Parent.UpdateIntervalMS);
  710. }
  711. }
  712. break;
  713. case SpellTemplate.MotionType.StraightAndStop:
  714. if (PassTimeMS < mInfo.BoomerangFlyTime)
  715. {
  716. flyTo(mLaunchDirection, mSpeed, Parent.UpdateIntervalMS);
  717. }
  718. //else if (PassTimeMS > mInfo.BoomerangFlyTime + mInfo.BoomerangHangtime)
  719. //{
  720. // if (flyToTarget(mLauncherUnit.X, mLauncherUnit.Y, mSpeed * 1.8f, Parent.UpdateIntervalMS))
  721. // {
  722. // this.Finish = true;
  723. // }
  724. //}
  725. break;
  726. }
  727. if (mInfo.BodyShape == SpellTemplate.Shape.LineToTarget)
  728. {
  729. if (mTarget != null)
  730. {
  731. this.faceTo(mTarget.X, mTarget.Y);
  732. }
  733. }
  734. else if (mInfo.BodyShape == SpellTemplate.Shape.LineToStart)
  735. {
  736. this.faceTo(mStartPos.X, mStartPos.Y);
  737. }
  738. else if (mInfo.RotateSpeedSEC != 0 && mInfo.MType != SpellTemplate.MotionType.SeekerMissile)
  739. {
  740. this.turn(MoveHelper.GetDistance(Parent.UpdateIntervalMS, mInfo.RotateSpeedSEC));
  741. }
  742. }
  743. public static double AngleBetween(Vector2 vector1, Vector2 vector2)
  744. {
  745. double sin = vector1.X * vector2.Y - vector2.X * vector1.Y;
  746. double cos = vector1.X * vector2.X + vector1.Y * vector2.Y;
  747. return Math.Atan2(sin, cos);
  748. }
  749. private void updateAOE()
  750. {
  751. switch (Info.BodyShape)
  752. {
  753. case SpellTemplate.Shape.LineToTarget:
  754. case SpellTemplate.Shape.LineToStart:
  755. case SpellTemplate.Shape.Strip:
  756. case SpellTemplate.Shape.StripRay:
  757. case SpellTemplate.Shape.StripRayTouchEnd:
  758. case SpellTemplate.Shape.RectStrip:
  759. case SpellTemplate.Shape.RectStripRay:
  760. updateAoeMotion(Info.Distance, ref mDistance);
  761. break;
  762. default:
  763. updateAoeMotion(Info.BodySize, ref mBaseSize);
  764. break;
  765. }
  766. }
  767. private void updateAoeMotion(float base_value, ref float value)
  768. {
  769. switch (Info.AOEMType)
  770. {
  771. case SpellTemplate.AoeMotionType.Sine:
  772. value = (float)Math.Sin(CMath.PI_F * PassTimeMS / (float)this.mSpellTotalTime) * base_value;
  773. break;
  774. case SpellTemplate.AoeMotionType.Linear:
  775. default:
  776. value += MoveHelper.GetDistance(Parent.UpdateIntervalMS, mSpeed);
  777. break;
  778. }
  779. }
  780. private void updateBinding(InstanceZoneObject target)
  781. {
  782. if ((target is InstanceUnit))
  783. {
  784. var target_unit = target as InstanceUnit;
  785. if (mInfo.RemoveOnBindingUncontrollable)
  786. {
  787. // 目标不可操控,停止法术 //
  788. if (target_unit.IsControllable == false)
  789. {
  790. Parent.RemoveObject(this);
  791. return;
  792. }
  793. }
  794. if (mInfo.RemoveOnBindingSkillOver)
  795. {
  796. // 目标非技能状态,停止法术 //
  797. if (target_unit.CurrentState is InstanceUnit.StateSkill)
  798. {
  799. if (mBindingSkill == null)
  800. {
  801. mBindingSkill = target_unit.CurrentState as InstanceUnit.StateSkill;
  802. }
  803. else if (mBindingSkill != target_unit.CurrentState)
  804. {
  805. Parent.RemoveObject(this);
  806. return;
  807. }
  808. }
  809. else
  810. {
  811. Parent.RemoveObject(this);
  812. return;
  813. }
  814. }
  815. }
  816. if (mInfo.IsBinding)
  817. {
  818. if (mInfo.IsBindingDirection)
  819. {
  820. this.faceTo(target.Direction);
  821. }
  822. if (mInfo.IsBindingOrbit)
  823. {
  824. float dadd = mInfo.OrbitDistance;
  825. float ox = (float)Math.Cos(Direction) * dadd;
  826. float oy = (float)Math.Sin(Direction) * dadd;
  827. this.setPos(target.X + ox, target.Y + oy);
  828. }
  829. else
  830. {
  831. float xChanage, yChange;
  832. this.LaunchData.GetXModify(this.Launcher, out xChanage, out yChange);
  833. this.setPos(target.X + xChanage, target.Y + yChange);
  834. }
  835. }
  836. }
  837. private void affectToSingle(InstanceUnit target, SpellTemplate.KeyFrame kf)
  838. {
  839. if (kf == null) return;
  840. if (kf.Effect != null)
  841. {
  842. Parent.queueObjectEvent(this, new UnitEffectEvent(this.ID, kf.Effect));
  843. }
  844. // 法术造成伤害
  845. if (kf.Attack != null)
  846. {
  847. Parent.unitAttackSingle(mLauncherUnit, new AttackSource(this, kf.Attack), target, mInfo.ExpectTarget);
  848. }
  849. // 法术产生法术
  850. if (kf.Spell != null)
  851. {
  852. Parent.spellLaunchSpell(this.FromSkillType, this, kf.Spell, this.X, this.Y, target.ID);
  853. //Parent.spellLaunchSpell(this, kf.Spell, mLauncherUnit.X + 5, mLauncherUnit.Y + 5, target.ID);
  854. }
  855. // 召唤
  856. if (kf.Summon != null)
  857. {
  858. Parent.spellSummonUnit(this, kf.Summon);
  859. }
  860. }
  861. private void affectToMulti(List<InstanceUnit> list, SpellTemplate.KeyFrame kf, bool effect)
  862. {
  863. if (kf == null) return;
  864. if (effect && kf.Effect != null)
  865. {
  866. Parent.queueObjectEvent(this, new UnitEffectEvent(this.ID, kf.Effect));
  867. }
  868. uint targetUnitID = 0;
  869. if (list.Count > 0)
  870. {
  871. targetUnitID = list[0].ID;
  872. // 法术造成伤害
  873. if (kf.Attack != null)
  874. {
  875. Parent.unitAttackDirect(mLauncherUnit, new AttackSource(this, kf.Attack), list);
  876. }
  877. }
  878. // 法术产生法术
  879. if (kf.Spell != null)
  880. {
  881. Parent.spellLaunchSpell(this.FromSkillType, this, kf.Spell, this.X, this.Y, targetUnitID);
  882. //System.Console.WriteLine("------法术产生法术: " + kf.Spell.SpellID + ", " + this.X + ", " + this.Y);
  883. }
  884. // 召唤
  885. if (kf.Summon != null)
  886. {
  887. Parent.spellSummonUnit(this, kf.Summon);
  888. }
  889. }
  890. #region __检测范围内目标__
  891. // 跟踪导弹查找附近单位 //
  892. public InstanceUnit seekAttackable(float range)
  893. {
  894. return Parent.SeekSpellAttackable(this.Launcher, this.Info, this.AoiStatus,
  895. X, Y, range, mInfo.ExpectTarget, mInfo.SeekingExpectTarget, mChainInfo);
  896. }
  897. // 范围伤害,碰撞检测 //
  898. private void getShapeAttackable(List<InstanceUnit> ret, AttackReason reason, bool slowRefresh)
  899. {
  900. AttackRange.Shape = (AttackShape)mInfo.BodyShape;
  901. AttackRange.BodySize = this.mBaseSize;
  902. AttackRange.Direction = this.Direction;
  903. AttackRange.Distance = this.mDistance;
  904. AttackRange.ExpectTarget = mInfo.ExpectTarget;
  905. AttackRange.FanAngle = this.mInfo.FanAngle;
  906. AttackRange.StripWide = this.mInfo.RectWide;
  907. float prv_x = this.X;
  908. float prv_y = this.Y;
  909. if (mInfo.MType == SpellTemplate.MotionType.Straight || mInfo.MType == SpellTemplate.MotionType.StraightAndStop)
  910. {
  911. prv_x = mPrvePos.X;
  912. prv_y = mPrvePos.Y;
  913. }
  914. AttackRange.GetShapeAttackable(ret, reason, this.Info, X, Y, prv_x, prv_y, mInfo.MaxAffectUnit, AoiStatus);
  915. }
  916. #endregion
  917. float foxfireRoundAngle = 0f;
  918. int foxfireSeekWait = 0;
  919. public InstanceSpell[] brothers = null;
  920. void changeBrotherSeekWait(int timeValue)
  921. {
  922. if (brothers == null)
  923. {
  924. return;
  925. }
  926. foreach (var b in brothers)
  927. {
  928. b.foxfireSeekWait += timeValue;
  929. }
  930. }
  931. /// <summary>
  932. /// 更新范围检测以及关键帧
  933. /// </summary>
  934. private void updateKeyFrames(bool slowRefresh)
  935. {
  936. switch (Info.MType)
  937. {
  938. case SpellTemplate.MotionType.Missile:
  939. case SpellTemplate.MotionType.SeekerMissile:
  940. //【战斗】Missile类技能,在目标消失后失效//
  941. if (mTarget != null && CMath.includeRoundPoint(mTarget.X, mTarget.Y, mTarget.BodyHitSize, this.X, this.Y))
  942. {
  943. if (mTarget.IsActive)
  944. {
  945. affectToSingle(mTarget, mInfo.HitOnExplosionKeyFrame);
  946. }
  947. Parent.RemoveObject(this);
  948. }
  949. break;
  950. case SpellTemplate.MotionType.CurveMissile:
  951. //【战斗】Missile类技能,在目标消失后失效//
  952. if (mTarget != null && CMath.includeRoundPoint(mTarget.X, mTarget.Y, mTarget.BodyHitSize + this.BodyHitSize, this.X, this.Y))
  953. {
  954. if (mTarget.IsActive)
  955. {
  956. affectToSingle(mTarget, mInfo.HitOnExplosionKeyFrame);
  957. }
  958. Parent.RemoveObject(this);
  959. }
  960. break;
  961. case SpellTemplate.MotionType.MissileAttackRoute:
  962. if (mTarget != null && CMath.includeRoundPoint(mTarget.X, mTarget.Y, mTarget.BodyHitSize, this.X, this.Y))
  963. {
  964. if (mTarget.IsActive)
  965. {
  966. affectToSingle(mTarget, mInfo.HitOnExplosionKeyFrame);
  967. }
  968. Parent.RemoveObject(this);
  969. }
  970. else if (mSeekingCooldownTime.Update(Parent.UpdateIntervalMS))
  971. {
  972. using (var enemy_list = ListObjectPool<InstanceUnit>.AllocAutoRelease())
  973. {
  974. getShapeAttackable(enemy_list, AttackReason.Attack, slowRefresh);
  975. if (enemy_list.Count > 0)
  976. {
  977. InstanceUnit target = enemy_list[0];
  978. if (target.IsActive)
  979. {
  980. affectToSingle(target, mInfo.HitOnExplosionKeyFrame);
  981. }
  982. mSeekingCooldownTime.Reset();
  983. }
  984. }
  985. }
  986. break;
  987. case SpellTemplate.MotionType.Cannon:
  988. if (Finish)
  989. {
  990. SpellTemplate.KeyFrame kf = mInfo.HitOnExplosionKeyFrame;
  991. if (kf != null)
  992. {
  993. using (var list = ListObjectPool<InstanceUnit>.AllocAutoRelease())
  994. {
  995. getShapeAttackable(list, AttackReason.Attack, slowRefresh);
  996. affectToMulti(list, kf, true);
  997. }
  998. }
  999. Parent.RemoveObject(this);
  1000. }
  1001. break;
  1002. case SpellTemplate.MotionType.Chain:
  1003. if (mTarget != null)
  1004. {
  1005. if (Collider.Object_Pos_IncludeInRound(mTarget, this.X, this.Y, this.mDistance))
  1006. {
  1007. updateKeyFrameSingleTarget(mTarget, true);
  1008. }
  1009. else
  1010. {
  1011. Parent.RemoveObject(this);
  1012. }
  1013. }
  1014. break;
  1015. case SpellTemplate.MotionType.Boomerang1:
  1016. case SpellTemplate.MotionType.Boomerang2:
  1017. if (Info.SpecialEffect == SpellTemplate.SpecialAdditionalEffect.DragPath && Info.SpecialEffectParam != 0f)
  1018. {
  1019. updateDragPathKeyFramesRanged(slowRefresh);
  1020. }
  1021. else if (Info.SpecialEffect == SpellTemplate.SpecialAdditionalEffect.DarkHole && Info.SpecialEffectParam != 0f)
  1022. {
  1023. updateDarkHoleKeyFramesRanged(slowRefresh);
  1024. }
  1025. else
  1026. {
  1027. updateKeyFramesRanged(slowRefresh);
  1028. }
  1029. if (Finish)
  1030. {
  1031. Parent.RemoveObject(this);
  1032. }
  1033. break;
  1034. case SpellTemplate.MotionType.Foxfire:
  1035. if (mTarget != null && CMath.includeRoundPoint(mTarget.X, mTarget.Y, mTarget.BodyHitSize, this.X, this.Y))
  1036. {
  1037. if (mTarget.IsActive)
  1038. {
  1039. affectToSingle(mTarget, mInfo.HitOnExplosionKeyFrame);
  1040. }
  1041. Parent.RemoveObject(this);
  1042. }
  1043. break;
  1044. default:
  1045. if (Info.BodyShape == SpellTemplate.Shape.LineToTarget || Info.BodyShape == SpellTemplate.Shape.LineToStart)
  1046. {
  1047. if (mTarget != null && Collider.Object_Pos_IncludeInRound(mTarget, this.X, this.Y, this.mDistance))
  1048. {
  1049. updateKeyFrameSingleTarget(mTarget, true);
  1050. }
  1051. else
  1052. {
  1053. updateKeyFrameSingleTarget(mTarget, false);
  1054. }
  1055. }
  1056. else
  1057. {
  1058. if (Info.SpecialEffect == SpellTemplate.SpecialAdditionalEffect.DragPath && Info.SpecialEffectParam != 0f)
  1059. {
  1060. updateDragPathKeyFramesRanged(slowRefresh);
  1061. }
  1062. else if (Info.SpecialEffect == SpellTemplate.SpecialAdditionalEffect.DarkHole && Info.SpecialEffectParam != 0f)
  1063. {
  1064. updateDarkHoleKeyFramesRanged(slowRefresh);
  1065. }
  1066. else
  1067. {
  1068. updateKeyFramesRanged(slowRefresh);
  1069. }
  1070. }
  1071. break;
  1072. }
  1073. }
  1074. private void updateKeyFrameSingleTarget(InstanceUnit enemy, bool affect)
  1075. {
  1076. using (var kfs = ListObjectPool<SpellTemplate.KeyFrame>.AllocAutoRelease())
  1077. {
  1078. int kfs_count = mKeyFrames.PopKeyFrames(PassTimeMS, kfs);
  1079. bool is_interval_test = mHitIntervalTicker.Update(Parent.UpdateIntervalMS);
  1080. if (affect)
  1081. {
  1082. if (kfs_count > 0 || is_interval_test || mInfo.HitIntervalMS == 0)
  1083. {
  1084. if (kfs_count > 0)
  1085. {
  1086. for (int i = 0; i < kfs.Count; i++)
  1087. {
  1088. affectToSingle(enemy, kfs[i]);
  1089. }
  1090. }
  1091. if (mInfo.HitOnExplosion)
  1092. {
  1093. // 击中后爆炸
  1094. affectToSingle(enemy, mInfo.HitOnExplosionKeyFrame);
  1095. Parent.RemoveObject(this);
  1096. }
  1097. else if (mInfo.HitIntervalKeyFrame != null)
  1098. {
  1099. if (mInfo.HitIntervalMS == 0)
  1100. {
  1101. // 只在接触后第一次产生效果
  1102. if (!mHittedUnits.ContainsKey(enemy.ID))
  1103. {
  1104. affectToSingle(enemy, mInfo.HitIntervalKeyFrame);
  1105. mHittedUnits.Add(enemy.ID, enemy);
  1106. }
  1107. }
  1108. else if (is_interval_test)
  1109. {
  1110. // 间隔产生效果
  1111. affectToSingle(enemy, mInfo.HitIntervalKeyFrame);
  1112. }
  1113. }
  1114. }
  1115. }
  1116. }
  1117. }
  1118. private void updateKeyFramesRanged(bool slowRefresh)
  1119. {
  1120. using (var kfs = ListObjectPool<SpellTemplate.KeyFrame>.AllocAutoRelease())
  1121. {
  1122. //if(mKeyFrames == null)
  1123. //{
  1124. // long ltime1 = CommonLang.CUtils.localTimeMS - mDisPosingTime;
  1125. // long ltime2 = System.DateTime.Now.Ticks - mDisPosingTime1;
  1126. //}
  1127. int kfs_count = mKeyFrames.PopKeyFrames(PassTimeMS, kfs);
  1128. bool is_interval_test = mHitIntervalTicker.Update(Parent.UpdateIntervalMS);
  1129. //此处修正 加入触发次数上限检测
  1130. if (is_interval_test && mInfo.HitIntervalMS > 0 && mHitIntervalTicker.TotalTickCount > (this.mSpellTotalTime / mInfo.HitIntervalMS))
  1131. {
  1132. is_interval_test = false;
  1133. }
  1134. if (kfs_count > 0 || is_interval_test || mInfo.HitIntervalMS == 0)
  1135. {
  1136. this.mHitTotalTimes++;
  1137. using (var enemy_list = ListObjectPool<InstanceUnit>.AllocAutoRelease())
  1138. {
  1139. getShapeAttackable(enemy_list, AttackReason.Attack, slowRefresh);
  1140. if (kfs_count > 0)
  1141. {
  1142. // 单Frame情况下,如果是Attack。进行唯一敌人判定(X型拼接法术, 需要伤害只生效一次)
  1143. //if (kfs_count == 1 && mInfo.HitIntervalMS == 0 && this.mInfo.KeyFrames.Count == 1 && kfs[0].Attack != null && this.mInfo.HitIntervalKeyFrame == null)
  1144. if(this.LaunchData.PType == LaunchSpell.PosType.POS_TYPE_X)
  1145. {
  1146. if(this.refreshHitUnit(enemy_list))
  1147. {
  1148. for (int i = 0; i < kfs.Count; i++)
  1149. {
  1150. affectToMulti(enemy_list, kfs[i], true);
  1151. }
  1152. }
  1153. }
  1154. else
  1155. {
  1156. for (int i = 0; i < kfs.Count; i++)
  1157. {
  1158. affectToMulti(enemy_list, kfs[i], true);
  1159. }
  1160. }
  1161. }
  1162. if (mInfo.HitOnExplosion)
  1163. {
  1164. if (enemy_list.Count > 0)
  1165. {
  1166. // 击中后爆炸
  1167. affectToMulti(enemy_list, mInfo.HitOnExplosionKeyFrame, true);
  1168. Parent.RemoveObject(this);
  1169. }
  1170. }
  1171. else if(mInfo.HitIntervalKeyFrame != null)
  1172. {
  1173. if (mInfo.HitIntervalMS == 0)
  1174. {
  1175. // 只在接触后第一次产生效果
  1176. if (enemy_list.Count > 0 && this.refreshHitUnit(enemy_list))
  1177. {
  1178. affectToMulti(enemy_list, mInfo.HitIntervalKeyFrame, true);
  1179. }
  1180. }
  1181. else if (is_interval_test)
  1182. {
  1183. // 间隔产生效果
  1184. affectToMulti(enemy_list, mInfo.HitIntervalKeyFrame, false);
  1185. }
  1186. }
  1187. }
  1188. }
  1189. }
  1190. }
  1191. private void updateDarkHoleKeyFramesRanged(bool slowRefresh)
  1192. {
  1193. using (var kfs = ListObjectPool<SpellTemplate.KeyFrame>.AllocAutoRelease())
  1194. {
  1195. using (var enemy_list = ListObjectPool<InstanceUnit>.AllocAutoRelease())
  1196. {
  1197. getShapeAttackable(enemy_list, AttackReason.Attack, slowRefresh);
  1198. for (int i = enemy_list.Count - 1; i >= 0; --i)
  1199. {
  1200. if(enemy_list[i].IsIgnoreControl)
  1201. {
  1202. continue;
  1203. }
  1204. float angle, distance, offset, dx, dy;
  1205. if (enemy_list[i].Weight < XmdsConstConfig.DEFAULT_WEIGHT)
  1206. {
  1207. offset = MoveHelper.GetDistance(Parent.UpdateIntervalMS, Info.SpecialEffectParam);
  1208. dx = this.X - enemy_list[i].X;
  1209. dy = this.Y - enemy_list[i].Y;
  1210. distance = dx * dx + dy * dy;
  1211. if (offset * offset < distance)
  1212. {
  1213. angle = MathVector.getDegree(dx, dy);
  1214. enemy_list[i].moveLinearMap(offset, angle);
  1215. }
  1216. else
  1217. {
  1218. enemy_list[i].moveLinearMap2(dx, dy);
  1219. }
  1220. enemy_list[i].SetStunTimeMS(100);
  1221. }
  1222. }
  1223. int kfs_count = mKeyFrames.PopKeyFrames(PassTimeMS, kfs);
  1224. bool is_interval_test = mHitIntervalTicker.Update(Parent.UpdateIntervalMS);
  1225. #region <<关键帧逻辑>>
  1226. if (kfs_count > 0 || is_interval_test || mInfo.HitIntervalMS == 0)
  1227. {
  1228. //getShapeAttackable(enemy_list, AttackReason.Attack);
  1229. if (kfs_count > 0)
  1230. {
  1231. for (int i = 0; i < kfs.Count; i++)
  1232. {
  1233. affectToMulti(enemy_list, kfs[i], false);
  1234. }
  1235. }
  1236. if (mInfo.HitOnExplosion)
  1237. {
  1238. if (enemy_list.Count > 0)
  1239. {
  1240. // 击中后爆炸
  1241. affectToMulti(enemy_list, mInfo.HitOnExplosionKeyFrame, true);
  1242. Parent.RemoveObject(this);
  1243. }
  1244. }
  1245. else if(mInfo.HitIntervalKeyFrame != null)
  1246. {
  1247. if (mInfo.HitIntervalMS == 0)
  1248. {
  1249. // 只在接触后第一次产生效果
  1250. if (enemy_list.Count > 0 && this.refreshHitUnit(enemy_list))
  1251. {
  1252. affectToMulti(enemy_list, mInfo.HitIntervalKeyFrame, true);
  1253. }
  1254. }
  1255. else if (is_interval_test)
  1256. {
  1257. // 间隔产生效果
  1258. affectToMulti(enemy_list, mInfo.HitIntervalKeyFrame, false);
  1259. }
  1260. }
  1261. }
  1262. #endregion
  1263. }
  1264. }
  1265. }
  1266. public override String GetTemplateData()
  1267. {
  1268. return "Spell:" + this.Info.ID;
  1269. }
  1270. private bool refreshHitUnit(CommonLang.ListObjectPool<InstanceUnit>.AutoReleaseList<InstanceUnit> enemy_list)
  1271. {
  1272. if(enemy_list.Count <= 0)
  1273. {
  1274. return true;
  1275. }
  1276. foreach (InstanceUnit u in mHittedUnits.Values)
  1277. {
  1278. enemy_list.Remove(u);
  1279. }
  1280. if (enemy_list.Count > 0)
  1281. {
  1282. for (int i = enemy_list.Count - 1; i >= 0; --i)
  1283. {
  1284. InstanceUnit u = enemy_list[i];
  1285. mHittedUnits.Add(u.ID, u);
  1286. }
  1287. return true;
  1288. }
  1289. return false;
  1290. }
  1291. public const float Pi = (float)Math.PI;
  1292. public const float PiOver2 = (float)(Math.PI / 2.0);
  1293. public const float PiOver36 = (float)(Math.PI / 36.0);
  1294. public const float TwoPi = (float)(Math.PI * 2.0);
  1295. private void updateDragPathKeyFramesRanged(bool slowRefresh)
  1296. {
  1297. using (var kfs = ListObjectPool<SpellTemplate.KeyFrame>.AllocAutoRelease())
  1298. {
  1299. using (var enemy_list = ListObjectPool<InstanceUnit>.AllocAutoRelease())
  1300. {
  1301. getShapeAttackable(enemy_list, AttackReason.Attack, slowRefresh);
  1302. float _a, _b, _angle, offset, dx, dy , dis;
  1303. for (int i = enemy_list.Count - 1; i >= 0; --i)
  1304. {
  1305. if (enemy_list[i].IsIgnoreControl)
  1306. {
  1307. continue;
  1308. }
  1309. if (enemy_list[i].Weight < XmdsConstConfig.DEFAULT_WEIGHT)
  1310. {
  1311. dx = enemy_list[i].X - this.X;
  1312. dy = enemy_list[i].Y - this.Y;
  1313. _a = MathVector.getDegree(dx, dy);
  1314. _b = this.Direction - _a;
  1315. if (_b < -Pi)
  1316. {
  1317. _b += TwoPi;
  1318. }
  1319. else if(_b > Pi)
  1320. {
  1321. _b -= TwoPi;
  1322. }
  1323. if (Math.Abs(_b) < PiOver36)
  1324. {
  1325. continue;
  1326. }
  1327. if(_b > 0)
  1328. {
  1329. if(_b > PiOver2)
  1330. {
  1331. continue;
  1332. }
  1333. _angle = this.Direction + PiOver2;
  1334. }
  1335. else
  1336. {
  1337. if (_b < -PiOver2)
  1338. {
  1339. continue;
  1340. }
  1341. _angle = this.Direction - PiOver2;
  1342. }
  1343. dis = Math.Abs((float)(dy * Math.Sin(_angle) / Math.Sin(_a)));
  1344. offset = MoveHelper.GetDistance(Parent.UpdateIntervalMS, Info.SpecialEffectParam);
  1345. if(offset > dis)
  1346. {
  1347. offset = dis;
  1348. }
  1349. enemy_list[i].moveLinearMap(offset, _angle);
  1350. enemy_list[i].SetStunTimeMS(100);
  1351. }
  1352. }
  1353. int kfs_count = mKeyFrames.PopKeyFrames(PassTimeMS, kfs);
  1354. bool is_interval_test = mHitIntervalTicker.Update(Parent.UpdateIntervalMS);
  1355. #region <<关键帧逻辑>>
  1356. if (kfs_count > 0 || is_interval_test || mInfo.HitIntervalMS == 0)
  1357. {
  1358. //getShapeAttackable(enemy_list, AttackReason.Attack);
  1359. if (kfs_count > 0)
  1360. {
  1361. for (int i = 0; i < kfs.Count; i++)
  1362. {
  1363. affectToMulti(enemy_list, kfs[i], false);
  1364. }
  1365. }
  1366. if (mInfo.HitOnExplosion)
  1367. {
  1368. if (enemy_list.Count > 0)
  1369. {
  1370. // 击中后爆炸
  1371. affectToMulti(enemy_list, mInfo.HitOnExplosionKeyFrame, true);
  1372. Parent.RemoveObject(this);
  1373. }
  1374. }
  1375. else if (mInfo.HitIntervalKeyFrame != null)
  1376. {
  1377. if (mInfo.HitIntervalMS == 0)
  1378. {
  1379. // 只在接触后第一次产生效果
  1380. if (enemy_list.Count > 0 && this.refreshHitUnit(enemy_list))
  1381. {
  1382. affectToMulti(enemy_list, mInfo.HitIntervalKeyFrame, true);
  1383. }
  1384. }
  1385. else if (is_interval_test)
  1386. {
  1387. // 间隔产生效果
  1388. affectToMulti(enemy_list, mInfo.HitIntervalKeyFrame, false);
  1389. }
  1390. }
  1391. }
  1392. #endregion
  1393. }
  1394. }
  1395. }
  1396. }
  1397. }