ZoneObject.Spell.cs 34 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868
  1. using CommonAI.RTS;
  2. using CommonLang.Vector;
  3. using CommonAI.Zone;
  4. using CommonAI.Zone.Helper;
  5. using CommonLang;
  6. using System;
  7. using System.Collections.Generic;
  8. using System.Linq;
  9. using System.Text;
  10. using CommonLang.Log;
  11. namespace CommonAI.ZoneClient
  12. {
  13. public class ZoneSpell : ZoneObject
  14. {
  15. public readonly SpellTemplate Info;
  16. public override string Name { get { return null; } }
  17. public override string DisplayName { get { return Info.Name; } }
  18. public override float X { get { return mLocalPos.X; } }
  19. public override float Y { get { return mLocalPos.Y; } }
  20. public override float Z { get { return mLocal_Z; } }
  21. public override float RadiusSize { get { return mDisplaySize; } }
  22. public ZoneObject Sender { get; internal set; }
  23. public ZoneUnit Launcher { get; internal set; }
  24. public ZoneUnit Target { get; internal set; }
  25. public Vector2 TargetPos { get; internal set; }
  26. public Vector2 StartPos { get { return mStartPos; } }
  27. public float LaunchHeight { get { return mStartHeight; } set { this.mStartHeight = value; this.mLocal_Z = value; } }
  28. public float BodySize { get { return mDisplaySize; } }
  29. public float Distance { get { return mDisplayDistance; } }
  30. public int PassTimeMS { get { return mPassTimeMS; } }
  31. private Vector2 mLocalPos = new Vector2();
  32. private float mLocal_Z = 0;
  33. private float mSizeLimit, mDisplaySize;
  34. private float mDistanceLimit, mDisplayDistance;
  35. private Vector2 mStartPos = new Vector2();
  36. private float mStartDirection;
  37. private float mStartHeight;
  38. private int mPassTimeMS;
  39. private float mSpeed;
  40. private float mMoveDistance;
  41. private int mCurveMissileIndex = -1;
  42. private PopupKeyFrames<SpellTemplate.KeyFrame> mKeyFrames;
  43. private TimeInterval<SpellTemplate.KeyFrame> mHitIntervalTicker;
  44. private AddSpellEvent mAddEvent;
  45. public System.Action OnTargetChanged = null;
  46. private static HashMap<string, int> hashCurveMissileCnt = new HashMap<string, int>();
  47. private string strCurveMissileIndexKey;
  48. public ZoneSpell(SpellTemplate info, SyncSpellInfo syn, ZoneLayer parent, AddSpellEvent add)
  49. : base(syn.ObjectID, parent)
  50. {
  51. this.Info = info;
  52. this.mStartPos.SetX(syn.x);
  53. this.mStartPos.SetY(syn.y);
  54. this.mPos.SetX(syn.x);
  55. this.mPos.SetY(syn.y);
  56. this.mLocalPos.SetX(syn.x);
  57. this.mLocalPos.SetY(syn.y);
  58. this.mDirection = this.mStartDirection = syn.direction;
  59. this.mDistanceLimit = this.mDisplayDistance = info.Distance;
  60. this.mSizeLimit = this.mDisplaySize = info.BodySize;
  61. this.mPassTimeMS = 0;
  62. this.mAddEvent = add;
  63. if (add != null && add.IsHeight)
  64. {
  65. this.mLocal_Z = add.height;
  66. }
  67. this.mKeyFrames = new PopupKeyFrames<SpellTemplate.KeyFrame>();
  68. this.mKeyFrames.AddRange(info.KeyFrames);
  69. this.mHitIntervalTicker = new TimeInterval<SpellTemplate.KeyFrame>(info.HitIntervalMS);
  70. this.mHitIntervalTicker.Tag = info.HitIntervalKeyFrame;
  71. this.mSpeed = info.MSpeedSEC;
  72. this.mMoveDistance = 0;
  73. if (syn.HasSpeed)
  74. {
  75. this.mSpeed = syn.CurSpeed;
  76. }
  77. if(info.MType == SpellTemplate.MotionType.CurveMissile)
  78. {
  79. int cnt;
  80. strCurveMissileIndexKey = add.sender_unit_id.ToString() + ":" + info.TemplateID.ToString();
  81. if (hashCurveMissileCnt.TryGetValue(strCurveMissileIndexKey, out cnt))
  82. {
  83. hashCurveMissileCnt[strCurveMissileIndexKey] = ++cnt;
  84. }
  85. else
  86. {
  87. hashCurveMissileCnt.Add(strCurveMissileIndexKey, 1);
  88. cnt = 1;
  89. }
  90. mCurveMissileIndex = cnt-1;
  91. }
  92. }
  93. protected override void Disposing()
  94. {
  95. if (!string.IsNullOrEmpty(strCurveMissileIndexKey))
  96. {
  97. int cnt;
  98. if (hashCurveMissileCnt.TryGetValue(strCurveMissileIndexKey, out cnt))
  99. {
  100. if (cnt <= 1)
  101. {
  102. hashCurveMissileCnt.Remove(strCurveMissileIndexKey);
  103. }
  104. else
  105. {
  106. hashCurveMissileCnt[strCurveMissileIndexKey] = cnt - 1;
  107. }
  108. }
  109. }
  110. base.Disposing();
  111. }
  112. protected internal override void OnAdded()
  113. {
  114. base.OnAdded();
  115. float radius = 0;
  116. if (mAddEvent != null)
  117. {
  118. if (mAddEvent.LaunchData.FromUnitBody && Sender != null && (Sender is ZoneUnit))
  119. {
  120. radius = (Sender as ZoneUnit).Info.LaunchSpellRadius;
  121. }
  122. else
  123. {
  124. radius = mAddEvent.LaunchData.LaunchSpellRadius;
  125. }
  126. }
  127. if (!Info.IsLaunchSpellEventSyncPos)
  128. {
  129. switch (Info.MType)
  130. {
  131. case SpellTemplate.MotionType.SelectTarget:
  132. if (Target != null)
  133. {
  134. mStartPos.SetX(Target.X);
  135. mStartPos.SetY(Target.Y);
  136. mLocal_Z = Target.Z;
  137. mStartDirection = Target.ServerDirection;
  138. }
  139. MathVector.movePolar(mStartPos, mStartDirection, radius);
  140. break;
  141. case SpellTemplate.MotionType.SelectLauncher:
  142. if (Launcher != null)
  143. {
  144. mStartPos.SetX(Launcher.X);
  145. mStartPos.SetY(Launcher.Y);
  146. mLocal_Z = Launcher.Z;
  147. mStartDirection = Launcher.ServerDirection;
  148. }
  149. MathVector.movePolar(mStartPos, mStartDirection, radius);
  150. break;
  151. case SpellTemplate.MotionType.Chain:
  152. if (Sender != null)
  153. {
  154. mStartPos.SetX(Sender.X);
  155. mStartPos.SetY(Sender.Y);
  156. mLocal_Z = Sender.Z;
  157. }
  158. break;
  159. case SpellTemplate.MotionType.AOE_Binding:
  160. case SpellTemplate.MotionType.Binding:
  161. if (Sender != null)
  162. {
  163. mStartPos.SetX(Sender.X);
  164. mStartPos.SetY(Sender.Y);
  165. mLocal_Z = Sender.Z;
  166. }
  167. break;
  168. case SpellTemplate.MotionType.AOE_BindingTarget:
  169. case SpellTemplate.MotionType.BindingTarget:
  170. if (Target != null)
  171. {
  172. mStartPos.SetX(Target.X);
  173. mStartPos.SetY(Target.Y);
  174. mLocal_Z = Target.Z;
  175. }
  176. break;
  177. }
  178. this.mPos.SetX(this.mStartPos.X);
  179. this.mPos.SetY(this.mStartPos.Y);
  180. this.mLocalPos.SetX(this.mStartPos.X);
  181. this.mLocalPos.SetY(this.mStartPos.Y);
  182. this.mDirection = this.mStartDirection;
  183. }
  184. if (Info.TargetEffect != null)
  185. {
  186. if (Target != null)
  187. {
  188. Parent.PreQueueEvent(new UnitEffectEvent(Target.ObjectID, Info.TargetEffect));
  189. }
  190. else if (TargetPos != null)
  191. {
  192. Parent.PreQueueEvent(new AddEffectEvent(0,TargetPos.X, TargetPos.Y, mDirection, Info.TargetEffect));
  193. }
  194. }
  195. }
  196. public override void ForceSyncPos(float x, float y)
  197. {
  198. this.mPos.SetX(x);
  199. this.mPos.SetY(y);
  200. this.mLocalPos.SetX(x);
  201. this.mLocalPos.SetY(y);
  202. }
  203. internal protected override void DoEvent(ObjectEvent e)
  204. {
  205. if (e is SpellLockTargetEvent)
  206. {
  207. doSpellLockTargetEvent(e as SpellLockTargetEvent);
  208. }
  209. else if (e is ObjectForceSyncPosEvent)
  210. {
  211. var oe = e as ObjectForceSyncPosEvent;
  212. this.ForceSyncPos(oe.X, oe.Y);
  213. this.Direction = oe.Direction;
  214. }
  215. }
  216. private void doSpellLockTargetEvent(SpellLockTargetEvent e)
  217. {
  218. if (Parent == null)
  219. {
  220. log.Error("doSpellLockTargetEvent -> Parent == null");
  221. }
  222. if (Info == null)
  223. {
  224. log.Error("doSpellLockTargetEvent -> Info == null");
  225. }
  226. if (e == null)
  227. {
  228. log.Error("doSpellLockTargetEvent -> e == null");
  229. }
  230. this.Target = Parent.GetUnit(e.target_obj_id);
  231. if(OnTargetChanged != null)
  232. {
  233. OnTargetChanged();
  234. }
  235. switch (Info.MType)
  236. {
  237. case SpellTemplate.MotionType.SeekerSelectTarget:
  238. if (Target != null)
  239. {
  240. this.mLocalPos.SetX(Target.X);
  241. this.mLocalPos.SetY(Target.Y);
  242. this.mLocal_Z = Target.Z;
  243. }
  244. break;
  245. }
  246. }
  247. protected internal override void UpdateAI()
  248. {
  249. }
  250. internal protected override void Update()
  251. {
  252. int intervalMS = Parent.CurrentIntervalMS;
  253. mPassTimeMS += intervalMS;
  254. if (Parent.ActorSyncMode != SyncMode.ForceByServer && (Info.MaxMoveDistance == 0 || mMoveDistance < Info.MaxMoveDistance))
  255. {
  256. updateMotion(intervalMS);
  257. }
  258. updateZPos();
  259. switch (Info.MType)
  260. {
  261. case SpellTemplate.MotionType.AOE:
  262. case SpellTemplate.MotionType.AOE_Binding:
  263. case SpellTemplate.MotionType.AOE_BindingTarget:
  264. updateAOE(intervalMS);
  265. break;
  266. }
  267. switch (Info.BodyShape)
  268. {
  269. case SpellTemplate.Shape.StripRayTouchEnd:
  270. updateRayTouchEnd();
  271. break;
  272. case SpellTemplate.Shape.LineToStart:
  273. case SpellTemplate.Shape.LineToTarget:
  274. updateLineToTarget();
  275. break;
  276. }
  277. if (Info.IsMoveable && (Info.MaxMoveDistance == 0 || mMoveDistance < Info.MaxMoveDistance))
  278. {
  279. MoveHelper.UpdateSpeed(intervalMS, ref mSpeed, Info.MSpeedAdd, Info.MSpeedAcc);
  280. MoveHelper.UpdateMoveDistance(intervalMS, ref mMoveDistance, mSpeed);
  281. }
  282. updateKeyFrames();
  283. }
  284. public override void SyncPos(ref SyncPosEvent.UnitPos pos)
  285. {
  286. base.SyncPos(ref pos);
  287. if (Parent.ActorSyncMode == SyncMode.ForceByServer)
  288. {
  289. mLocalPos.SetX(pos.X);
  290. mLocalPos.SetY(pos.Y);
  291. if (Parent.IsSyncZ) { this.Z = pos.Z; }
  292. }
  293. }
  294. //-----------------------------------------------------------------------
  295. #region _UpdateMotions_
  296. float foxfireRoundAngle = 0f;
  297. float foxfireSeekWait = 0f;
  298. private Logger log = LoggerFactory.GetLogger("ZoneClient");
  299. /// <summary>
  300. /// 更新移动行为
  301. /// </summary>
  302. private void updateMotion(int intervalMS)
  303. {
  304. float speed_distance = MoveHelper.GetDistance(intervalMS, mSpeed);
  305. try
  306. {
  307. switch (Info.MType)
  308. {
  309. case SpellTemplate.MotionType.SelectLauncher:
  310. case SpellTemplate.MotionType.SelectTarget:
  311. break;
  312. case SpellTemplate.MotionType.Cannon:
  313. case SpellTemplate.MotionType.Straight:
  314. MathVector.movePolar(mLocalPos, mStartDirection, speed_distance);
  315. break;
  316. case SpellTemplate.MotionType.StraightAndStop:
  317. if (PassTimeMS < Info.BoomerangFlyTime)
  318. {
  319. MathVector.movePolar(mLocalPos, mStartDirection, speed_distance);
  320. }
  321. //else if (PassTimeMS > Info.BoomerangFlyTime + Info.BoomerangHangtime)
  322. //{
  323. // PreFlyTo(Sender.X, Sender.Y, mSpeed * 1.8f, intervalMS);
  324. //}
  325. break;
  326. case SpellTemplate.MotionType.Boomerang1:
  327. if (PassTimeMS < Info.BoomerangFlyTime)
  328. {
  329. MathVector.movePolar(mLocalPos, mStartDirection, speed_distance);
  330. }
  331. else if (Sender != null && PassTimeMS > Info.BoomerangFlyTime + Info.BoomerangHangtime)
  332. {
  333. PreFlyTo(Sender.X, Sender.Y, mSpeed * 1.4f, intervalMS);
  334. }
  335. break;
  336. case SpellTemplate.MotionType.Boomerang2:
  337. if (PassTimeMS < Info.BoomerangFlyTime)
  338. {
  339. MathVector.movePolar(mLocalPos, mStartDirection, speed_distance);
  340. }
  341. else if (PassTimeMS > Info.BoomerangFlyTime + Info.BoomerangHangtime)
  342. {
  343. PreFlyTo(mStartPos.X, mStartPos.Y, mSpeed * 1.4f, intervalMS);
  344. }
  345. break;
  346. case SpellTemplate.MotionType.Immovability:
  347. case SpellTemplate.MotionType.AOE:
  348. mLocalPos.SetX(mPos.X);
  349. mLocalPos.SetY(mPos.Y);
  350. break;
  351. case SpellTemplate.MotionType.AOE_Binding:
  352. case SpellTemplate.MotionType.Binding:
  353. if (Sender != null)
  354. {
  355. updateBinding(Sender.X, Sender.Y, Sender.Direction, intervalMS);
  356. }
  357. else
  358. {
  359. adjustPos(speed_distance);
  360. }
  361. break;
  362. case SpellTemplate.MotionType.AOE_BindingTarget:
  363. case SpellTemplate.MotionType.BindingTarget:
  364. if (Target != null)
  365. {
  366. updateBinding(Target.X, Target.Y, Target.Direction, intervalMS);
  367. }
  368. else
  369. {
  370. adjustPos(speed_distance);
  371. }
  372. break;
  373. case SpellTemplate.MotionType.Missile:
  374. case SpellTemplate.MotionType.MissileAttackRoute:
  375. if (Target != null)
  376. {
  377. if (Info.RotateSpeedSEC == 0)
  378. {
  379. PreFaceTo(Target.X, Target.Y);
  380. }
  381. PreFlyTo(Target.X, Target.Y, mSpeed, intervalMS);
  382. }
  383. else
  384. {
  385. MathVector.movePolar(mLocalPos, mStartDirection, speed_distance);
  386. }
  387. break;
  388. case SpellTemplate.MotionType.SeekerMissile:
  389. if (Target != null)
  390. {
  391. if (Info.SeekingTurningAngleSEC != 0)
  392. {
  393. FlyToTargetTunning(Target.X, Target.Y, mSpeed, Info.SeekingTurningAngleSEC, intervalMS);
  394. }
  395. else
  396. {
  397. PreFaceTo(Target.X, Target.Y);
  398. PreFlyTo(Target.X, Target.Y, mSpeed, intervalMS);
  399. }
  400. }
  401. else
  402. {
  403. MathVector.movePolar(mLocalPos, mStartDirection, speed_distance);
  404. }
  405. break;
  406. case SpellTemplate.MotionType.SeekerSelectTarget:
  407. break;
  408. case SpellTemplate.MotionType.Chain:
  409. if (Sender != null && Sender.IsEnable && Target != null && Target.IsActive)
  410. {
  411. mLocalPos.SetX(Sender.X);
  412. mLocalPos.SetY(Sender.Y);
  413. PreFaceTo(Target.X, Target.Y);
  414. }
  415. break;
  416. case SpellTemplate.MotionType.CurveMissile:
  417. IPosBase targetPos = Target == null ? (IPosBase)TargetPos : (IPosBase)Target;
  418. if(targetPos == null)
  419. {
  420. PreFlyTo(mStartDirection, mSpeed, intervalMS);
  421. }
  422. else if (Info.SeekingTurningAngleSEC != 0)
  423. {
  424. FlyToTargetTunning(targetPos.X, targetPos.Y, mSpeed, Info.SeekingTurningAngleSEC, intervalMS);
  425. }
  426. else
  427. {
  428. PreFaceTo(targetPos.X, targetPos.Y);
  429. PreFlyTo(targetPos.X, targetPos.Y, mSpeed, intervalMS);
  430. }
  431. break;
  432. case SpellTemplate.MotionType.Foxfire:
  433. if (Target == null)
  434. {
  435. var mLaunchData = mAddEvent.LaunchData;
  436. foxfireRoundAngle = mStartDirection + PassTimeMS * 0.001f * 3;
  437. var x = (float)(Math.Cos(foxfireRoundAngle + 0.3f) * mLaunchData.LaunchSpellRadius);
  438. var y = (float)(Math.Sin(foxfireRoundAngle + 0.3f) * mLaunchData.LaunchSpellRadius);
  439. if (Sender != null)
  440. {
  441. x = Sender.X + x;
  442. y = Sender.Y + y;
  443. }
  444. else
  445. {
  446. x = mStartPos.X + x;
  447. y = mStartPos.Y + y;
  448. }
  449. PreFlyTo(x, y, mSpeed, intervalMS);
  450. if (Info.RotateSpeedSEC == 0)
  451. {
  452. PreFaceTo(foxfireRoundAngle + (float)Math.PI * 0.25f);
  453. }
  454. foxfireSeekWait += intervalMS;
  455. }
  456. else
  457. {
  458. if (Info.SeekingTurningAngleSEC != 0)
  459. {
  460. FlyToTargetTunning(Target.X, Target.Y, mSpeed, Info.SeekingTurningAngleSEC, intervalMS);
  461. }
  462. else
  463. {
  464. PreFaceTo(Target.X, Target.Y);
  465. PreFlyTo(Target.X, Target.Y, mSpeed, intervalMS);
  466. }
  467. }
  468. break;
  469. }
  470. if (Info.BodyShape == SpellTemplate.Shape.LineToTarget)
  471. {
  472. if (Target != null)
  473. {
  474. PreFaceTo(Target.X, Target.Y);
  475. }
  476. }
  477. else if (Info.BodyShape == SpellTemplate.Shape.LineToStart)
  478. {
  479. PreFaceTo(StartPos.X, StartPos.Y);
  480. }
  481. else if (Info.RotateSpeedSEC != 0 && Info.MType != SpellTemplate.MotionType.SeekerMissile)
  482. {
  483. this.mDirection += MoveHelper.GetDistance(intervalMS, Info.RotateSpeedSEC);
  484. }
  485. }
  486. catch (Exception e) {
  487. log.Error(e.Message + "\n" + "Info.MType = " + Info.MType + " Info.BodyShape = " + Info.BodyShape);
  488. }
  489. }
  490. private void updateZPos()
  491. {
  492. switch (Info.MType)
  493. {
  494. case SpellTemplate.MotionType.Immovability:
  495. case SpellTemplate.MotionType.SelectTarget:
  496. case SpellTemplate.MotionType.SelectLauncher:
  497. case SpellTemplate.MotionType.AOE:
  498. updateFloatingZ(mStartHeight);
  499. break;
  500. case SpellTemplate.MotionType.Straight:
  501. case SpellTemplate.MotionType.Boomerang1:
  502. case SpellTemplate.MotionType.Boomerang2:
  503. case SpellTemplate.MotionType.StraightAndStop:
  504. updateFloatingZ(mStartHeight);
  505. break;
  506. case SpellTemplate.MotionType.AOE_Binding:
  507. case SpellTemplate.MotionType.Binding:
  508. if (Sender != null)
  509. {
  510. this.mLocal_Z = Sender.Z;
  511. }
  512. break;
  513. case SpellTemplate.MotionType.AOE_BindingTarget:
  514. case SpellTemplate.MotionType.BindingTarget:
  515. if (Target != null)
  516. {
  517. this.mLocal_Z = Target.Z;
  518. }
  519. break;
  520. case SpellTemplate.MotionType.Cannon:
  521. if (TargetPos != null)
  522. {
  523. updateTargetZ(TargetPos, 0);
  524. }
  525. break;
  526. case SpellTemplate.MotionType.Missile:
  527. case SpellTemplate.MotionType.SeekerMissile:
  528. case SpellTemplate.MotionType.MissileAttackRoute:
  529. if (Target != null)
  530. {
  531. updateTargetZ(new TVector2(Target.X, Target.Y), Target.Z + Target.Info.BodyHeight * 0.6f);
  532. }
  533. break;
  534. case SpellTemplate.MotionType.CurveMissile:
  535. if (Target != null)
  536. {
  537. updata3DParabola(new TVector2(Target.X, Target.Y));
  538. }
  539. break;
  540. }
  541. }
  542. private void updata3DParabola(IVector2 targetPos)
  543. {
  544. TVector2 nowPos = new TVector2(mLocalPos.X, mLocalPos.Y);
  545. TVector2 vcStart = new TVector2(mStartPos.X, mStartPos.Y);
  546. TVector2 vcEnd = new TVector2(targetPos.X, targetPos.Y);
  547. TVector2 v1 = nowPos - vcStart;
  548. TVector2 v2 = vcEnd - vcStart;
  549. float a = (float)Math.Atan2(v1.Y, v1.X);
  550. float b = (float)Math.Atan2(v2.Y, v2.X);
  551. if (float.Equals(a, b) || float.Equals(Math.PI, Math.Abs(a - b)))
  552. {
  553. mLocal_Z = Target.Z;
  554. }
  555. else
  556. {
  557. float dis = nowPos.DistanceTo(vcStart) * (float)Math.Abs(Math.Sin(a - b));
  558. mLocal_Z = this.mStartHeight + dis *1.6f* (float)Math.Sin(Math.PI / 6 * mCurveMissileIndex);
  559. //ClientLog.LogWarning(">z:{0}, index:{1}, ({2},{3})->({4},{5})->({6},{7})", mLocal_Z, mCurveMissileIndex, mStartPos.X, mStartPos.Y, TargetPos.X, targetPos.Y, nowPos.X, nowPos.Y);
  560. }
  561. }
  562. /// <summary>
  563. /// 计算有固定目标Z坐标
  564. /// </summary>
  565. /// <param name="targetPos"></param>
  566. /// <param name="targetZ"></param>
  567. private void updateTargetZ(IVector2 targetPos, float targetZ)
  568. {
  569. float fdis = MathVector.getDistance(mStartPos, targetPos);
  570. float odis = MathVector.getDistance(mStartPos, mLocalPos);
  571. float pct = CMath.getInRange((fdis == 0 ? 0 : (odis / fdis)), 0, 1);//生命周期百分比//
  572. float dz = CMath.getMiddleValue(pct, mStartHeight, targetZ);//和目标高度差//
  573. float ph = 0;//抛物线相对高度//
  574. if (Info.ParabolaHeight != 0)
  575. {
  576. //计算抛物线//
  577. ph = (float)Math.Sin(pct * CMath.PI_F) * Info.ParabolaHeight;
  578. }
  579. this.mLocal_Z = dz + ph;
  580. }
  581. private void updateFloatingZ(float targetZ)
  582. {
  583. float pct = (float)mPassTimeMS / (float)Info.LifeTimeMS;//生命周期百分比//
  584. float dz = CMath.getMiddleValue(pct, mStartHeight, targetZ);//和目标高度差//
  585. float ph = 0;//抛物线相对高度//
  586. if (Info.ParabolaHeight != 0)
  587. {
  588. //计算抛物线//
  589. ph = (float)Math.Sin(pct * CMath.PI_F) * Info.ParabolaHeight;
  590. }
  591. this.mLocal_Z = dz + ph;
  592. }
  593. private void updateBinding(float x, float y, float direction, int intervalMS)
  594. {
  595. if (Info.IsBindingDirection)
  596. {
  597. mDirection = direction;
  598. }
  599. if (Info.IsBindingOrbit)
  600. {
  601. float dadd = Info.OrbitDistance;
  602. float ox = (float)Math.Cos(Direction) * dadd;
  603. float oy = (float)Math.Sin(Direction) * dadd;
  604. mLocalPos.SetX(x + ox);
  605. mLocalPos.SetY(y + oy);
  606. }
  607. else
  608. {
  609. mLocalPos.SetX(x);
  610. mLocalPos.SetY(y);
  611. }
  612. }
  613. #endregion
  614. //---------------------------------------------------------------------------------------------------
  615. #region _UpdateShape_
  616. private void updateAOE(int intervalMS)
  617. {
  618. switch (Info.BodyShape)
  619. {
  620. case SpellTemplate.Shape.LineToTarget:
  621. case SpellTemplate.Shape.LineToStart:
  622. case SpellTemplate.Shape.Strip:
  623. case SpellTemplate.Shape.StripRay:
  624. case SpellTemplate.Shape.StripRayTouchEnd:
  625. case SpellTemplate.Shape.RectStrip:
  626. case SpellTemplate.Shape.RectStripRay:
  627. updateAoeMotion(intervalMS, Info.Distance, ref mDistanceLimit);
  628. mDisplayDistance = mDistanceLimit;
  629. break;
  630. default:
  631. updateAoeMotion(intervalMS, Info.BodySize, ref mSizeLimit);
  632. mDisplaySize = mSizeLimit;
  633. break;
  634. }
  635. }
  636. private void updateAoeMotion(int intervalMS, float base_value, ref float value)
  637. {
  638. switch (Info.AOEMType)
  639. {
  640. case SpellTemplate.AoeMotionType.Sine:
  641. value = (float)Math.Sin(CMath.PI_F * mPassTimeMS / (float)Info.LifeTimeMS) * base_value;
  642. break;
  643. case SpellTemplate.AoeMotionType.Linear:
  644. default:
  645. value += MoveHelper.GetDistance(intervalMS, mSpeed);
  646. break;
  647. }
  648. }
  649. private void updateRayTouchEnd()
  650. {
  651. if (Launcher != null)
  652. {
  653. float d_width = Info.RectWide / 2f;
  654. Vector2 p1 = new Vector2(X, Y);
  655. MathVector.movePolar(p1, this.Direction, this.mDistanceLimit);
  656. using (var list = ListObjectPool<ZoneUnit>.AllocAutoRelease())
  657. {
  658. Parent.ForEachNearObjectsRect(this.X, this.Y, p1.X, p1.Y, (ZoneObject u, ref bool cancel) =>
  659. {
  660. if ((u is ZoneUnit))
  661. {
  662. ZoneUnit zu = u as ZoneUnit;
  663. if (Parent.IsAttackable(Launcher, zu, Info.ExpectTarget))
  664. {
  665. if ((CMath.intersectLineRound(this.X, this.Y, p1.X, p1.Y, zu.X, zu.Y, d_width + zu.BodyHitSize)))
  666. {
  667. list.Add(u as ZoneUnit);
  668. }
  669. }
  670. }
  671. });
  672. if (list.Count > 0)
  673. {
  674. list.Sort(new ObjectBodySorterNearest<ZoneUnit>(this.X, this.Y, 0));
  675. this.mDisplayDistance = CMath.getDistance(this.X, this.Y, list[0].X, list[0].Y);
  676. this.mDisplayDistance = Math.Min(mDisplayDistance, this.mDistanceLimit);
  677. }
  678. else
  679. {
  680. this.mDisplayDistance = this.mDistanceLimit;
  681. }
  682. }
  683. }
  684. }
  685. private void updateLineToTarget()
  686. {
  687. if (Info.BodyShape == SpellTemplate.Shape.LineToTarget)
  688. {
  689. if (Target != null)
  690. {
  691. this.mDisplayDistance = CMath.getDistance(this.X, this.Y, Target.X, Target.Y);
  692. this.mDisplayDistance = Math.Min(mDisplayDistance, this.mDistanceLimit);
  693. PreFaceTo(Target.X, Target.Y);
  694. }
  695. else
  696. {
  697. this.mDisplayDistance = 0;
  698. }
  699. }
  700. else if (Info.BodyShape == SpellTemplate.Shape.LineToStart)
  701. {
  702. this.mDisplayDistance = CMath.getDistance(this.X, this.Y, mStartPos.X, mStartPos.Y);
  703. this.mDisplayDistance = Math.Min(mDisplayDistance, this.mDistanceLimit);
  704. PreFaceTo(mStartPos.X, mStartPos.Y);
  705. }
  706. else
  707. {
  708. this.mDisplayDistance = 0;
  709. }
  710. }
  711. #endregion
  712. //---------------------------------------------------------------------------------------------------
  713. #region _UpdateKeyFrames_
  714. /// <summary>
  715. /// 更新范围检测以及关键帧
  716. /// </summary>
  717. private void updateKeyFrames()
  718. {
  719. switch (Info.MType)
  720. {
  721. case SpellTemplate.MotionType.Missile:
  722. case SpellTemplate.MotionType.SeekerMissile:
  723. case SpellTemplate.MotionType.Cannon:
  724. case SpellTemplate.MotionType.Chain:
  725. case SpellTemplate.MotionType.CurveMissile:
  726. break;
  727. default:
  728. if (Info.BodyShape == SpellTemplate.Shape.LineToTarget || Info.BodyShape == SpellTemplate.Shape.LineToStart)
  729. {
  730. }
  731. else
  732. {
  733. updateKeyFramesRanged();
  734. }
  735. break;
  736. }
  737. }
  738. private void updateKeyFramesRanged()
  739. {
  740. using (var kfs = ListObjectPool<SpellTemplate.KeyFrame>.AllocAutoRelease())
  741. {
  742. // int kfs_count = mKeyFrames.PopKeyFrames(PassTimeMS, kfs);
  743. bool is_interval_test = mHitIntervalTicker.Update(Parent.CurrentIntervalMS);
  744. //if (kfs_count > 0)
  745. //{
  746. // for (int i = 0; i < kfs.Count; i++)
  747. // {
  748. // if (kfs[i].Effect != null)
  749. // {
  750. // Parent.PreQueueEvent(new UnitEffectEvent(ObjectID, kfs[i].Effect));
  751. // }
  752. // }
  753. //}
  754. if (Info.HitOnExplosion)
  755. {
  756. }
  757. else if (Info.HitIntervalMS == 0)
  758. {
  759. }
  760. else if (is_interval_test)
  761. {
  762. if (Info.HitIntervalKeyFrame != null && Info.HitIntervalKeyFrame.Effect != null)
  763. {
  764. Parent.PreQueueEvent(new UnitEffectEvent(ObjectID, Info.HitIntervalKeyFrame.Effect));
  765. }
  766. }
  767. }
  768. }
  769. #endregion
  770. //---------------------------------------------------------------------------------------------------
  771. private void adjustPos(float min_distance)
  772. {
  773. float fdistance = MathVector.getDistance(mLocalPos, mPos);
  774. if (fdistance < min_distance)
  775. {
  776. MathVector.moveTo(mLocalPos, mPos.X, mPos.Y, min_distance);
  777. }
  778. else
  779. {
  780. MathVector.moveTo(mLocalPos, mPos.X, mPos.Y, fdistance / 2f);
  781. }
  782. }
  783. private void PreFaceTo(float angle)
  784. {
  785. this.mDirection = angle;
  786. }
  787. private void PreFaceTo(float x, float y)
  788. {
  789. if (this.X == x && this.Y == y)
  790. {
  791. return;
  792. }
  793. this.mDirection = (float)(Math.Atan2(y - this.Y, x - this.X));
  794. }
  795. private void PreTurnTo(float add)
  796. {
  797. this.mDirection += add;
  798. }
  799. private void PreFlyTo(float x, float y, float speedSEC, int intervalMS)
  800. {
  801. float distance = MoveHelper.GetDistance(intervalMS, speedSEC);
  802. MathVector.moveTo(mLocalPos, x, y, distance);
  803. }
  804. private void PreFlyTo(float direction, float speedSEC, int intervalMS)
  805. {
  806. float distance = MoveHelper.GetDistance(intervalMS, speedSEC);
  807. float dx = (float)(Math.Cos(direction) * distance);
  808. float dy = (float)(Math.Sin(direction) * distance);
  809. mLocalPos.AddX(dx);
  810. mLocalPos.AddY(dy);
  811. }
  812. public void FlyToTargetTunning(float x, float y, float speedSEC, float tunningSpeedSEC, int intervalMS)
  813. {
  814. MoveHelper.MoveToTargetTunning(ref mLocalPos, ref mDirection, x, y, speedSEC, tunningSpeedSEC, intervalMS);
  815. }
  816. }
  817. }