Motions.cs 34 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014
  1. using System;
  2. using System.Collections.Generic;
  3. using CommonLang.Vector;
  4. using CommonLang;
  5. using CommonAI.Zone.Instance;
  6. using CommonAI.RTS.Manhattan;
  7. using CommonLang.IO;
  8. namespace CommonAI.Zone.Helper
  9. {
  10. public class MoveHelper
  11. {
  12. public static readonly int POS_PRECISION = 255;
  13. /// <summary>
  14. /// 更新速度
  15. /// </summary>
  16. public static void UpdateSpeed(int intervalMS, ref float speedSec, float addSec, float accSec)
  17. {
  18. float add = 0;
  19. float acc = 0;
  20. if (addSec != 0)
  21. {
  22. add = MoveHelper.GetDistance(intervalMS, addSec);
  23. }
  24. if (accSec != 0)
  25. {
  26. acc = MoveHelper.GetDistance(intervalMS, speedSec * accSec);
  27. }
  28. speedSec += (add + acc);
  29. }
  30. public static void UpdateSpeed(int intervalMS, ref float speedSec, float addSec, float accSec, float speedMin, float speedMax)
  31. {
  32. float add = 0;
  33. float acc = 0;
  34. if (addSec != 0)
  35. {
  36. add = MoveHelper.GetDistance(intervalMS, addSec);
  37. }
  38. if (accSec != 0)
  39. {
  40. acc = MoveHelper.GetDistance(intervalMS, speedSec * accSec);
  41. }
  42. speedSec += (add + acc);
  43. if (speedSec > speedMax) { speedSec = speedMax; }
  44. if (speedSec < speedMin) { speedSec = speedMin; }
  45. }
  46. public static void UpdateMoveDistance(int intervalMS, ref float moveDistance, float speed)
  47. {
  48. moveDistance += MoveHelper.GetDistance(intervalMS, speed);
  49. }
  50. #region IO
  51. public static byte ToDirectionD8(float d)
  52. {
  53. return (byte)(CMath.OpitimizeRadians(d) / CMath.PI_MUL_2 * POS_PRECISION);
  54. }
  55. public static float ToDirectionF32(byte d8)
  56. {
  57. return d8 * CMath.PI_MUL_2 / POS_PRECISION;
  58. }
  59. public static ushort ToPos16(float x)
  60. {
  61. return (ushort)(x * POS_PRECISION);
  62. }
  63. public static float ToPosF32(ushort x)
  64. {
  65. return ((float)x) / POS_PRECISION;
  66. }
  67. public static void WritePos(bool half, float x, IOutputStream output)
  68. {
  69. if (half)
  70. {
  71. output.PutU16(ToPos16(x));
  72. }
  73. else
  74. {
  75. output.PutF32(x);
  76. }
  77. }
  78. public static void WritePos(bool half, float x, float y, IOutputStream output)
  79. {
  80. if (half)
  81. {
  82. output.PutU16(ToPos16(x));
  83. output.PutU16(ToPos16(y));
  84. }
  85. else
  86. {
  87. output.PutF32(x);
  88. output.PutF32(y);
  89. }
  90. }
  91. public static void WriteDirection(bool half, float direction, IOutputStream output)
  92. {
  93. if (half)
  94. {
  95. output.PutU8(ToDirectionD8(direction));
  96. }
  97. else
  98. {
  99. output.PutF32(direction);
  100. }
  101. }
  102. public static void WritePosAndDirection(bool half, float x, float y, float direction, IOutputStream output)
  103. {
  104. if (half)
  105. {
  106. output.PutU16(ToPos16(x));
  107. output.PutU16(ToPos16(y));
  108. output.PutU8(ToDirectionD8(direction));
  109. }
  110. else
  111. {
  112. output.PutF32(x);
  113. output.PutF32(y);
  114. output.PutF32(direction);
  115. }
  116. }
  117. public static void ReadPos(bool half, out float x, IInputStream input)
  118. {
  119. if (half)
  120. {
  121. x = ToPosF32(input.GetU16());
  122. }
  123. else
  124. {
  125. x = input.GetF32();
  126. }
  127. }
  128. public static void ReadPos(bool half, out float x, out float y, IInputStream input)
  129. {
  130. if (half)
  131. {
  132. x = ToPosF32(input.GetU16());
  133. y = ToPosF32(input.GetU16());
  134. }
  135. else
  136. {
  137. x = input.GetF32();
  138. y = input.GetF32();
  139. }
  140. }
  141. public static void ReadPos(bool half, ref Vector2 pos, IInputStream input)
  142. {
  143. if (half)
  144. {
  145. pos.SetX(ToPosF32(input.GetU16()));
  146. pos.SetY(ToPosF32(input.GetU16()));
  147. }
  148. else
  149. {
  150. pos.SetX(input.GetF32());
  151. pos.SetY(input.GetF32());
  152. }
  153. }
  154. public static void ReadDirection(bool half, out float direction, IInputStream input)
  155. {
  156. if (half)
  157. {
  158. direction = ToDirectionF32(input.GetU8());
  159. }
  160. else
  161. {
  162. direction = input.GetF32();
  163. }
  164. }
  165. public static void ReadPosAndDirection(bool half, out float x, out float y, out float direction, IInputStream input)
  166. {
  167. if (half)
  168. {
  169. x = ToPosF32(input.GetU16());
  170. y = ToPosF32(input.GetU16());
  171. direction = ToDirectionF32(input.GetU8());
  172. }
  173. else
  174. {
  175. x = input.GetF32();
  176. y = input.GetF32();
  177. direction = input.GetF32();
  178. }
  179. }
  180. #endregion
  181. //------------------------------------------------------------------------------------
  182. public static float GetDistance(int timeMS, float speedSEC)
  183. {
  184. return speedSEC * timeMS / 1000f;
  185. }
  186. /// <summary>
  187. /// 转身算法
  188. /// </summary>
  189. /// <param name="dstD">目标角度</param>
  190. /// <param name="curD">当前角度</param>
  191. /// <param name="turnSpeedSEC">角速度</param>
  192. /// <param name="intervalMS">时间间隔</param>
  193. /// <returns></returns>
  194. public static float DirectionChange(float dstD, float curD, float turnSpeedSEC, int intervalMS)
  195. {
  196. if (turnSpeedSEC > 0)
  197. {
  198. float delta = turnSpeedSEC * intervalMS / 1000f;
  199. float dd = CMath.OpitimizeRadians(dstD) + CMath.PI_MUL_2;
  200. float cd = CMath.OpitimizeRadians(curD) + CMath.PI_MUL_2;
  201. float distanceL = Math.Abs(dd - cd);
  202. float distanceR = CMath.PI_MUL_2 - distanceL;
  203. if (distanceL > distanceR)
  204. {
  205. if (distanceR <= delta)
  206. {
  207. return dstD;
  208. }
  209. if (dd > cd)
  210. {
  211. curD -= delta;
  212. }
  213. else
  214. {
  215. curD += delta;
  216. }
  217. }
  218. else
  219. {
  220. if (distanceL <= delta)
  221. {
  222. return dstD;
  223. }
  224. if (dd > cd)
  225. {
  226. curD += delta;
  227. }
  228. else
  229. {
  230. curD -= delta;
  231. }
  232. }
  233. return curD;
  234. }
  235. return dstD;
  236. }
  237. /// <summary>
  238. /// 导弹一样飞向目标,并自动调整方向
  239. /// </summary>
  240. /// <param name="pos"></param>
  241. /// <param name="direction"></param>
  242. /// <param name="targetX"></param>
  243. /// <param name="targetY"></param>
  244. /// <param name="speedSEC"></param>
  245. /// <param name="turnSpeedSEC"></param>
  246. /// <param name="intervalMS"></param>
  247. public static void MoveToTargetTunning(ref Vector2 sPos, ref float direction, float targetX, float targetY, float speedSEC, float turnSpeedSEC, int intervalMS)
  248. {
  249. float dstD = MathVector.getDegree(targetX - sPos.X, targetY - sPos.Y);
  250. direction = DirectionChange(dstD, direction, turnSpeedSEC, intervalMS);
  251. MathVector.movePolar(ref sPos, direction, speedSEC, intervalMS);
  252. }
  253. /// <summary>
  254. /// 预估击飞到落下的时间
  255. /// </summary>
  256. /// <param name="z">要计算的高度</param>
  257. /// <param name="z_speed">Z速度</param>
  258. /// <param name="z_limit">Z坐标上限</param>
  259. /// <param name="gravity">重力加速度</param>
  260. /// <param name="intervalMS">时间精度,越小越精确,一般10足够精度</param>
  261. /// <returns></returns>
  262. public static int CalculateFlyTimeMS(float z, float z_speed, float z_limit, float gravity, int intervalMS = 10)
  263. {
  264. int time = 0;
  265. float tick_g = GetDistance(intervalMS, gravity);
  266. do
  267. {
  268. time += intervalMS;
  269. float sd = GetDistance(intervalMS, z_speed);
  270. z += sd;
  271. if (z < 0)
  272. {
  273. time -= (int)Math.Abs(intervalMS * (z / sd));
  274. }
  275. if (z_limit != 0 && z_speed > 0 && z > z_limit)
  276. {
  277. z_speed = 0;
  278. }
  279. z_speed -= tick_g;
  280. }
  281. while (z > 0);
  282. return time;
  283. }
  284. /// <summary>
  285. /// 给定最大距离和总时间,计算当前抛物线高度
  286. /// y = ax2 + bx + c;
  287. /// </summary>
  288. /// <param name="maxHeight"></param>
  289. /// <param name="totalTimeMS"></param>
  290. /// <param name="currentTimeMS"></param>
  291. /// <returns></returns>
  292. public static float CalulateParabolicHeight(float maxHeight, int totalTimeMS, int currentTimeMS)
  293. {
  294. float pct = currentTimeMS / (float)totalTimeMS;
  295. // 用正弦函数模拟 //
  296. float y = (float)(Math.Sin(pct * CMath.PI_F) * maxHeight);
  297. return y;
  298. }
  299. /// <summary>
  300. /// 计算单位击退方向
  301. /// </summary>
  302. /// <param name="damage">受击者</param>
  303. /// <param name="attacker">攻击者</param>
  304. /// <param name="mtype"></param>
  305. /// <returns></returns>
  306. public static float CalculateHitMoveDirection(InstanceUnit damage, InstanceZoneObject attacker, AttackProp.HitMoveType mtype)
  307. {
  308. switch (mtype)
  309. {
  310. case AttackProp.HitMoveType.BySenderPosition:
  311. return MathVector.getDegree(attacker.X, attacker.Y, damage.X, damage.Y);
  312. case AttackProp.HitMoveType.BySenderDirection:
  313. return attacker.Direction;
  314. case AttackProp.HitMoveType.BySenderLeftRight:
  315. float fx = attacker.X;
  316. float fy = attacker.Y;
  317. MathVector.movePolar(ref fx, ref fy, attacker.Direction, 10);
  318. if (CMath.PointOnLine(attacker.X, attacker.Y, fx, fy, damage.X, damage.Y) == CMath.PointOnLineResult.Left)
  319. {
  320. return attacker.Direction - CMath.PI_DIV_2;
  321. }
  322. else
  323. {
  324. return attacker.Direction + CMath.PI_DIV_2;
  325. }
  326. case AttackProp.HitMoveType.ToSenderCenter:
  327. case AttackProp.HitMoveType.ToSenderBodySize:
  328. return attacker.Direction + CMath.PI_F;
  329. }
  330. return damage.Direction + CMath.PI_F;
  331. }
  332. }
  333. public class PopupKeyFrames<K> : IComparer<K> where K : BaseKeyFrame
  334. {
  335. readonly private List<K> list = new List<K>();
  336. public int Count
  337. {
  338. get { return list.Count; }
  339. }
  340. public PopupKeyFrames()
  341. {
  342. }
  343. public void AddRange(ICollection<K> frames)
  344. {
  345. list.AddRange(frames);
  346. list.Sort(this);
  347. }
  348. /// <summary>
  349. /// 取出所有到时间的关键帧
  350. /// </summary>
  351. /// <param name="passTimeMS"></param>
  352. /// <param name="ret"></param>
  353. /// <returns></returns>
  354. public int PopKeyFrames(int passTimeMS, List<K> ret = null)
  355. {
  356. int count = 0;
  357. while (list.Count > 0)
  358. {
  359. int i = list.Count - 1;
  360. K kf = list[i];
  361. //不足一帧的,当帧处理
  362. if (kf.FrameMS <= passTimeMS + XmdsConstConfig.GAME_UPDATE_INTERVAL_MS)
  363. {
  364. list.RemoveAt(i);
  365. if (ret != null) { ret.Add(kf); }
  366. count++;
  367. }
  368. else
  369. {
  370. return count;
  371. }
  372. }
  373. return count;
  374. }
  375. public int Compare(K x, K y)
  376. {
  377. return y.FrameMS - x.FrameMS;
  378. }
  379. }
  380. public enum MoveResult : byte
  381. {
  382. MOVE_SMOOTH = 0,
  383. /// <summary>
  384. /// 标志状态,本次移动任何方向接触过地图
  385. /// </summary>
  386. MOVE_RESULT_TOUCH_MAP_ALL = 0x10,
  387. /// <summary>
  388. /// 标志状态,阻挡的单位同意让开
  389. /// </summary>
  390. MOVE_RESULT_TOUCH_OBJ_GETAWAY = 0x20,
  391. /// <summary>
  392. /// 标志状态,本次移动等等
  393. /// </summary>
  394. MOVE_RESULT_HOLD = 0x40,
  395. /// <summary>
  396. /// 标志状态,无法移动
  397. /// </summary>
  398. MOVE_RESULT_NO_WAY = 0x80,
  399. /// <summary>
  400. /// 标志判断,到达目标
  401. /// </summary>
  402. MOVE_RESULT_ARRIVED = 0x01,
  403. /// <summary>
  404. /// 标志判断,被单位阻挡
  405. /// </summary>
  406. MOVE_RESULT_BLOCK_OBJ = 0x02,
  407. /// <summary>
  408. /// 标志判断,被地图阻挡
  409. /// </summary>
  410. MOVE_RESULT_BLOCK_MAP = 0x04,
  411. /// <summary>
  412. /// 标志判断,没有达到预期距离或者移动距离太小
  413. /// </summary>
  414. MOVE_RESULT_MIN_STEP = 0x08,
  415. /// <summary>
  416. /// 【标志判断组】,被任何东西阻挡
  417. /// </summary>
  418. RESULTS_BLOCK_ANY = MOVE_RESULT_BLOCK_OBJ | MOVE_RESULT_BLOCK_MAP,
  419. /// <summary>
  420. /// 【标志判断】,移动结束(被阻挡或者到达目标)
  421. /// </summary>
  422. RESULTS_MOVE_END = MOVE_RESULT_ARRIVED | MOVE_RESULT_BLOCK_OBJ | MOVE_RESULT_BLOCK_MAP | MOVE_RESULT_NO_WAY,
  423. }
  424. public enum MoveImpactResult : byte
  425. {
  426. MOVE_SMOOTH = MoveResult.MOVE_SMOOTH,
  427. /// <summary>
  428. /// 标志状态,本次移动任何方向接触过地图
  429. /// </summary>
  430. MOVE_RESULT_TOUCH_MAP = MoveResult.MOVE_RESULT_TOUCH_MAP_ALL,
  431. /// <summary>
  432. /// 标志状态,阻挡的单位同意让开
  433. /// </summary>
  434. MOVE_RESULT_TOUCH_OBJ = MoveResult.MOVE_RESULT_TOUCH_OBJ_GETAWAY,
  435. /// <summary>
  436. /// 标志判断,被单位阻挡
  437. /// </summary>
  438. MOVE_RESULT_BLOCK_OBJ = MoveResult.MOVE_RESULT_BLOCK_OBJ,
  439. /// <summary>
  440. /// 标志判断,被地图阻挡
  441. /// </summary>
  442. MOVE_RESULT_BLOCK_MAP = MoveResult.MOVE_RESULT_BLOCK_MAP,
  443. /// <summary>
  444. /// 标志判断,没有达到预期距离或者移动距离太小
  445. /// </summary>
  446. MOVE_RESULT_MIN_STEP = MoveResult.MOVE_RESULT_MIN_STEP,
  447. /// <summary>
  448. /// 【标志判断组】,被任何东西阻挡
  449. /// </summary>
  450. RESULTS_CAN_NOT_MOVE = MOVE_RESULT_BLOCK_OBJ | MOVE_RESULT_BLOCK_MAP | MOVE_RESULT_MIN_STEP,
  451. }
  452. public struct MoveBlockResult
  453. {
  454. public MoveResult result;
  455. public InstanceZoneObject obj;
  456. public MoveBlockResult(MoveResult r)
  457. {
  458. this.result = r;
  459. this.obj = null;
  460. }
  461. public bool HasFlag(MoveResult flag)
  462. {
  463. return (result & flag) != 0;
  464. }
  465. }
  466. public struct TMoveTarget : IPositionObject
  467. {
  468. private float x;
  469. private float y;
  470. public TMoveTarget(float x, float y)
  471. {
  472. this.x = x;
  473. this.y = y;
  474. }
  475. public float X { get { return x; } }
  476. public float Y { get { return y; } }
  477. public float Direction { get { return 0; } }
  478. public float RadiusSize { get { return 0; } }
  479. }
  480. public class MoveAI
  481. {
  482. public readonly InstanceUnit unit;
  483. public readonly InstanceZone zone;
  484. private MoveTurnPosObject turnLR;
  485. private MoveTurnPosMapBlock turnMap;
  486. private AstarManhattan.MWayPoint next_path;
  487. private TimeExpire<int> holdTime;
  488. private bool mPause = true;
  489. private bool mOverrideActionStatus;
  490. private readonly int hold_time;
  491. private readonly int hold_max_time;
  492. private readonly int hold_min_time;
  493. public IPositionObject Target { get; private set; }
  494. public bool IsBypass { get; set; }
  495. public bool IsTurnLR { get { return turnLR != null; } }
  496. public float TargetX { get { return Target.X; } }
  497. public float TargetY { get { return Target.Y; } }
  498. public bool IsNoWay { get; private set; }
  499. public bool IsNoWayAutoFindNear { get; set; }
  500. public MoveAI(InstanceUnit owner, bool overrideActionStatus = true, int holdTimeMS = -1)
  501. {
  502. this.unit = owner;
  503. this.zone = owner.Parent;
  504. this.mOverrideActionStatus = overrideActionStatus;
  505. if (holdTimeMS < 0)
  506. {
  507. hold_time = unit.Templates.CFG.AI_MOVE_AI_HOLD_TIME_MS;
  508. }
  509. else
  510. {
  511. hold_time = holdTimeMS;
  512. }
  513. this.holdTime = new TimeExpire<int>(hold_time);
  514. this.turnLR = new MoveTurnPosObject(unit);
  515. this.turnMap = new MoveTurnPosMapBlock(unit);
  516. this.holdTime.End();
  517. this.hold_max_time = hold_time;
  518. this.hold_min_time = hold_time / 2;
  519. this.IsNoWayAutoFindNear = true;
  520. }
  521. public void Hold(int timeMS = 0)
  522. {
  523. if (timeMS <= 0)
  524. {
  525. timeMS = unit.RandomN.Next(hold_min_time, hold_max_time);
  526. }
  527. holdTime.Reset(timeMS);
  528. }
  529. public void FindPath(float x, float y)
  530. {
  531. this.FindPath(new TMoveTarget(x, y));
  532. }
  533. public void FindPath(IPositionObject target)
  534. {
  535. if (target != null)
  536. {
  537. this.Target = target;
  538. this.next_path = zone.findPath(unit.X, unit.Y, TargetX, TargetY);
  539. if (next_path == null)
  540. {
  541. holdTime.Reset(hold_time);
  542. IsNoWay = true;
  543. }
  544. this.mPause = false;
  545. }
  546. }
  547. public void Pause()
  548. {
  549. this.next_path = null;
  550. this.turnLR.Stop();
  551. this.turnMap.Stop();
  552. this.holdTime.End();
  553. this.mPause = true;
  554. }
  555. public bool IsDirectLookTarget()
  556. {
  557. //if ((Target is InstanceUnit) && CMath.includeRoundPoint(unit.X, unit.Y, unit.Info.GuardRangeLimit, TargetX, TargetY))
  558. if ((Target is InstanceUnit))
  559. {
  560. // 地图连线无阻挡 //
  561. if (!zone.RaycastMap(unit, unit.X, unit.Y, TargetX, TargetY))
  562. {
  563. return true;
  564. }
  565. }
  566. return false;
  567. }
  568. public MoveBlockResult Update()
  569. {
  570. MoveBlockResult result = new MoveBlockResult();
  571. if (Target == null)
  572. {
  573. result.result = MoveResult.MOVE_RESULT_HOLD;
  574. return result;
  575. }
  576. if (mPause)
  577. {
  578. result.result = MoveResult.MOVE_RESULT_HOLD;
  579. return result;
  580. }
  581. else if (IsNoWay)
  582. {
  583. if (IsNoWayAutoFindNear && holdTime.Update(zone.UpdateIntervalMS))
  584. {
  585. this.next_path = zone.findPath(
  586. unit.X + (float)(unit.Parent.GridCellW * unit.Parent.RandomN.NextDouble()),
  587. unit.Y + (float)(unit.Parent.GridCellH * unit.Parent.RandomN.NextDouble()),
  588. TargetX + (float)(unit.Parent.GridCellW * unit.Parent.RandomN.NextDouble()),
  589. TargetY + (float)(unit.Parent.GridCellH * unit.Parent.RandomN.NextDouble()));
  590. IsNoWay = next_path == null;
  591. }
  592. //else
  593. //{
  594. // unit.faceTo(Target.X, Target.Y);
  595. // result = unit.moveBlockTo(Target.X, Target.Y, unit.MoveSpeedSEC, zone.UpdateIntervalMS);
  596. // result.result = result.result | MoveResult.MOVE_RESULT_NO_WAY;
  597. // return result;
  598. //}
  599. if (IsNoWay)
  600. {
  601. if (this.unit.IsFollowMaster() && mOverrideActionStatus && unit.CurrentActionStatus != UnitActionStatus.Idle)
  602. {
  603. this.unit.Virtual.SendMsgToClient(CommonAI.XmdsConstConfig.TIPS_FOLLOW_FAIL);
  604. unit.SetActionStatus(UnitActionStatus.Idle);
  605. }
  606. result.result = result.result | MoveResult.MOVE_RESULT_NO_WAY;
  607. return result;
  608. }
  609. }
  610. //if (!holdTime.Update(zone.UpdateIntervalMS))
  611. //{
  612. //result.result = MoveResult.MOVE_RESULT_HOLD;
  613. //if (mOverrideActionStatus) unit.SetActionStatus(UnitActionStatus.Idle);
  614. //}
  615. //else
  616. //{
  617. if (mOverrideActionStatus) unit.SetActionStatus(UnitActionStatus.Move);
  618. if (!turnMap.IsEnd)
  619. {
  620. // 如果向目标移动过程中被地块阻挡,则向左或右调整一段距离 //
  621. if (turnMap.move(unit.MoveSpeedSEC, zone.UpdateIntervalMS, out result))
  622. {
  623. return result;
  624. }
  625. }
  626. else if (!turnLR.IsEnd)
  627. {
  628. // 如果向目标移动过程中被单位阻挡,则向左或右调整一段距离 //
  629. if (turnLR.move(unit.MoveSpeedSEC, zone.UpdateIntervalMS, out result))
  630. {
  631. if ((result.result & MoveResult.MOVE_RESULT_ARRIVED) != 0)
  632. {
  633. result.result ^= MoveResult.MOVE_RESULT_ARRIVED;
  634. }
  635. if ((result.result & MoveResult.MOVE_RESULT_TOUCH_OBJ_GETAWAY) != 0)
  636. {
  637. Hold();
  638. return result;
  639. }
  640. else if ((result.result & MoveResult.MOVE_RESULT_BLOCK_MAP) != 0)
  641. {
  642. Hold();
  643. }
  644. }
  645. }
  646. else if (next_path != null)
  647. {
  648. //if (IsDirectLookTarget())
  649. //{
  650. // next_path = null;
  651. //}
  652. //else
  653. //{
  654. // 如果向目标移动过程中被地图阻挡,则寻路 //
  655. result = unit.moveBlockTo(next_path.PosX, next_path.PosY, unit.MoveSpeedSEC, zone.UpdateIntervalMS);
  656. if ((result.result & MoveResult.MOVE_RESULT_MIN_STEP) != 0 ||
  657. (result.result & MoveResult.MOVE_RESULT_ARRIVED) != 0)
  658. {
  659. }
  660. else
  661. {
  662. unit.faceTo(next_path.PosX, next_path.PosY);
  663. }
  664. if ((result.result & MoveResult.MOVE_RESULT_ARRIVED) != 0)
  665. {
  666. next_path = next_path.Next;
  667. if (next_path != null)
  668. {
  669. result.result ^= MoveResult.MOVE_RESULT_ARRIVED;
  670. if (IsDirectLookTarget())
  671. {
  672. next_path = null;
  673. }
  674. }
  675. else
  676. {
  677. next_path = zone.findPath(unit.X, unit.Y, TargetX, TargetY);
  678. if (next_path == null)
  679. {
  680. Hold();
  681. IsNoWay = true;
  682. }
  683. }
  684. }
  685. else if ((result.result & MoveResult.MOVE_RESULT_TOUCH_MAP_ALL) != 0)
  686. {
  687. if ((result.result & MoveResult.MOVE_RESULT_MIN_STEP) != 0)
  688. {
  689. turnMap.Start(next_path.PosX, next_path.PosY);
  690. }
  691. }
  692. else if ((result.result & MoveResult.MOVE_RESULT_NO_WAY) != 0)
  693. {
  694. next_path = null;
  695. //unit.faceTo(TargetX, TargetY);
  696. //result = unit.moveBlockTo(TargetX, TargetY, unit.MoveSpeedSEC, zone.UpdateIntervalMS);
  697. }
  698. else if ((result.result & MoveResult.MOVE_RESULT_BLOCK_MAP) != 0)
  699. {
  700. Hold();
  701. next_path = null;
  702. return result;
  703. }
  704. //}
  705. }
  706. else
  707. {
  708. if (!IsDirectLookTarget())
  709. {
  710. next_path = zone.findPath(unit.X, unit.Y, TargetX, TargetY);
  711. if (next_path == null)
  712. {
  713. Hold();
  714. IsNoWay = true;
  715. }
  716. }
  717. else
  718. {
  719. unit.faceTo(TargetX, TargetY);
  720. result = unit.moveBlockTo(TargetX, TargetY, unit.MoveSpeedSEC, zone.UpdateIntervalMS);
  721. }
  722. }
  723. //}
  724. return result;
  725. }
  726. /// <summary>
  727. /// 如果向目标移动过程中被单位阻挡,则向左或右调整一段距离
  728. /// </summary>
  729. private class MoveTurnPosObject
  730. {
  731. public InstanceUnit Owner { get; private set; }
  732. public InstanceZoneObject Touched { get; private set; }
  733. public bool IsEnd { get; private set; }
  734. public float TargetX { get; private set; }
  735. public float TargetY { get; private set; }
  736. public bool FaceTo { get; private set; }
  737. public int TouchCount { get { return touched_list.Count; } }
  738. private Vector2 TargetPos;
  739. private HashMap<uint, InstanceZoneObject> touched_list = new HashMap<uint, InstanceZoneObject>();
  740. public MoveTurnPosObject(InstanceUnit owner)
  741. {
  742. this.Owner = owner;
  743. this.IsEnd = true;
  744. }
  745. public void Start(float targetX, float targetY, InstanceZoneObject touched)
  746. {
  747. this.Touched = touched;
  748. this.IsEnd = false;
  749. this.TargetPos = new Vector2(targetX, targetY);
  750. this.touched_list.Clear();
  751. TurnNearBypass();
  752. }
  753. public void Stop()
  754. {
  755. this.IsEnd = true;
  756. }
  757. /// <summary>
  758. /// 绕过某个单位移动
  759. /// </summary>
  760. /// <param name="speedSEC"></param>
  761. /// <param name="intervalMS"></param>
  762. /// <param name="result"></param>
  763. /// <returns>是否移动结束</returns>
  764. public bool move(float speedSEC, int intervalMS, out MoveBlockResult result)
  765. {
  766. Owner.faceTo(TargetX, TargetY);
  767. result = Owner.moveBlockTo(TargetX, TargetY, speedSEC, intervalMS);
  768. if (result.obj != null)
  769. {
  770. Touched = result.obj;
  771. }
  772. if ((result.result & (MoveResult.MOVE_RESULT_BLOCK_MAP)) != 0)
  773. {
  774. this.IsEnd = true;
  775. return true;
  776. }
  777. else if ((result.result & (MoveResult.MOVE_RESULT_ARRIVED)) != 0)
  778. {
  779. if (this.TargetX == Owner.X && this.TargetY == Owner.Y)
  780. {
  781. this.TargetX = TargetPos.X;
  782. this.TargetY = TargetPos.Y;
  783. this.IsEnd = true;
  784. return true;
  785. }
  786. }
  787. if (result.obj != null)
  788. {
  789. //换一种躲避方法//
  790. if (touched_list.ContainsKey(result.obj.ID))
  791. {
  792. TurnRandomBypass();
  793. }
  794. else
  795. {
  796. TurnNearBypass();
  797. touched_list.Put(Touched.ID, Touched);
  798. }
  799. }
  800. return false;
  801. }
  802. /// <summary>
  803. /// 获取绕过去,离目标最近的点
  804. /// </summary>
  805. /// <returns></returns>
  806. private Vector2 GetNearTurnPos(InstanceZoneObject obj)
  807. {
  808. float bodysize = Math.Max(Touched.BodyBlockSize, Owner.BodyBlockSize);
  809. float angle = MathVector.getDegree(obj.X - Owner.X, obj.Y - Owner.Y);
  810. Vector2 turnL = new Vector2(Owner.X, Owner.Y);
  811. Vector2 turnR = new Vector2(Owner.X, Owner.Y);
  812. MathVector.movePolar(turnL, angle + CMath.PI_DIV_2, bodysize);
  813. MathVector.movePolar(turnR, angle - CMath.PI_DIV_2, bodysize);
  814. float dl = MathVector.getDistanceSquare(turnL, TargetPos);
  815. float dr = MathVector.getDistanceSquare(turnR, TargetPos);
  816. if (dl < dr)
  817. {
  818. return turnL;
  819. }
  820. else
  821. {
  822. return turnR;
  823. }
  824. }
  825. /// <summary>
  826. /// 碰到障碍时,绕过去
  827. /// </summary>
  828. /// <param name="unit"></param>
  829. private void TurnRandomBypass()
  830. {
  831. float bodysize = (Touched.BodyBlockSize + Owner.BodyBlockSize); // 1~2个身位 //
  832. float distance = bodysize + (float)(Owner.RandomN.NextDouble() * bodysize);
  833. float angle = MathVector.getDegree(Touched.X, Touched.Y, Owner.X, Owner.Y);
  834. float factor = (float)(Owner.RandomN.NextDouble() * (CMath.PI_DIV_2 / 2f));
  835. if (Owner.RandomN.Next(0, 10) % 2 == 0)
  836. {
  837. angle += CMath.PI_DIV_2 - factor;
  838. }
  839. else
  840. {
  841. angle -= CMath.PI_DIV_2 + factor;
  842. }
  843. Vector2 turnLR = new Vector2(Owner.X, Owner.Y);
  844. MathVector.movePolar(turnLR, angle, distance);
  845. this.TargetX = turnLR.X;
  846. this.TargetY = turnLR.Y;
  847. }
  848. /// <summary>
  849. /// 从最近点绕过去
  850. /// </summary>
  851. private void TurnNearBypass()
  852. {
  853. float bodysize = Math.Max(Touched.BodyBlockSize, Owner.BodyBlockSize); // 1~2个身位 //
  854. float distance = bodysize + (float)(Owner.RandomN.NextDouble() * bodysize);
  855. float angle = MathVector.getDegree(Touched.X, Touched.Y, Owner.X, Owner.Y);
  856. Vector2 turnL = new Vector2(Owner.X, Owner.Y);
  857. Vector2 turnR = new Vector2(Owner.X, Owner.Y);
  858. MathVector.movePolar(turnL, angle + CMath.PI_DIV_2, distance);
  859. MathVector.movePolar(turnR, angle - CMath.PI_DIV_2, distance);
  860. float dl = MathVector.getDistanceSquare(turnL, TargetPos);
  861. float dr = MathVector.getDistanceSquare(turnR, TargetPos);
  862. if (dl < dr)
  863. {
  864. this.TargetX = turnL.X;
  865. this.TargetY = turnL.Y;
  866. }
  867. else
  868. {
  869. this.TargetX = turnR.X;
  870. this.TargetY = turnR.Y;
  871. }
  872. }
  873. }
  874. /// <summary>
  875. /// 如果向目标移动过程中被单位阻挡,则向左或右调整一段距离
  876. /// </summary>
  877. private class MoveTurnPosMapBlock
  878. {
  879. public InstanceUnit Owner { get; private set; }
  880. public bool IsEnd { get; private set; }
  881. private float tx;
  882. private float ty;
  883. public MoveTurnPosMapBlock(InstanceUnit owner)
  884. {
  885. this.Owner = owner;
  886. this.IsEnd = true;
  887. }
  888. public void Start(float target_x, float target_y)
  889. {
  890. float tdx = (target_x - Owner.X);
  891. float tdy = (target_y - Owner.Y);
  892. int ddx = (int)CMath.getDirect(tdx);
  893. int ddy = (int)CMath.getDirect(tdy);
  894. if (Math.Abs(tdx) < Math.Abs(tdy))
  895. {
  896. float tw = ddx * (float)(Owner.Parent.GridCellW + (Owner.RandomN.NextDouble() * Owner.Parent.GridCellW / 2f));
  897. tx = Owner.X + tw;
  898. ty = Owner.Y;
  899. }
  900. else
  901. {
  902. float th = ddy * (float)(Owner.Parent.GridCellH + (Owner.RandomN.NextDouble() * Owner.Parent.GridCellH / 2f));
  903. tx = Owner.X;
  904. ty = Owner.Y + th;
  905. }
  906. this.IsEnd = false;
  907. }
  908. public void Stop()
  909. {
  910. this.IsEnd = true;
  911. }
  912. /// <summary>
  913. /// 绕过某个单位移动
  914. /// </summary>
  915. /// <param name="speedSEC"></param>
  916. /// <param name="intervalMS"></param>
  917. /// <param name="result"></param>
  918. /// <returns>是否移动结束</returns>
  919. public bool move(float speedSEC, int intervalMS, out MoveBlockResult result)
  920. {
  921. Owner.faceTo(tx, ty);
  922. MoveImpactResult rst = Owner.moveImpactTo(tx, ty, speedSEC, intervalMS);
  923. result = new MoveBlockResult();
  924. result.result = (MoveResult)((byte)rst);
  925. if ((rst & MoveImpactResult.RESULTS_CAN_NOT_MOVE) != 0)
  926. {
  927. this.IsEnd = true;
  928. return true;
  929. }
  930. return false;
  931. }
  932. }
  933. }
  934. }