MathVector.cs 24 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. using CommonLang.IO;
  5. using CommonLang.ByteOrder;
  6. using CommonLang;
  7. namespace CommonLang.Vector
  8. {
  9. public interface IPosBase
  10. {
  11. float X { get; }
  12. float Y { get; }
  13. }
  14. public interface IVector2 : IPosBase
  15. {
  16. void SetX(float value);
  17. void SetY(float value);
  18. void AddX(float value);
  19. void AddY(float value);
  20. }
  21. public interface IRoundObject : IVector2
  22. {
  23. //float X { get; set; }
  24. //float Y { get; set; }
  25. /// <summary>
  26. /// 半径
  27. /// </summary>
  28. float RadiusSize { get; }
  29. }
  30. public interface IPositionObject : IPosBase
  31. {
  32. /// <summary>
  33. /// 方向
  34. /// </summary>
  35. float Direction { get; }
  36. /// <summary>
  37. /// 半径
  38. /// </summary>
  39. float RadiusSize { get; }
  40. }
  41. /// <summary>
  42. /// 2D向量
  43. /// </summary>
  44. public class Vector2 : IVector2, ICloneable
  45. {
  46. private float x;
  47. private float y;
  48. public float X { get { return this.x; } }
  49. public float Y { get { return this.y; } }
  50. public void SetX(float value)
  51. {
  52. this.x = value;
  53. }
  54. public void SetY(float value)
  55. {
  56. this.y = value;
  57. }
  58. public void AddX(float value)
  59. {
  60. this.x += value;
  61. }
  62. public void AddY(float value)
  63. {
  64. this.y += value;
  65. }
  66. public Vector2()
  67. {
  68. this.x = 0;
  69. this.y = 0;
  70. }
  71. public Vector2(float _x, float _y)
  72. {
  73. this.x = _x;
  74. this.y = _y;
  75. }
  76. public object Clone()
  77. {
  78. return new Vector2(x, y);
  79. }
  80. public override bool Equals(object obj)
  81. {
  82. if (obj is Vector2)
  83. {
  84. Vector2 v = (Vector2)obj;
  85. return X == v.X && Y == v.Y;
  86. }
  87. return false;
  88. }
  89. public bool Equals(Vector2 v)
  90. {
  91. return v.x == x && v.y == y;
  92. }
  93. public static bool Equals(Vector2 a, Vector2 b)
  94. {
  95. if (a != null && b != null)
  96. {
  97. return a.x == b.x && a.y == b.y;
  98. }
  99. return a == b;
  100. }
  101. public override int GetHashCode()
  102. {
  103. return X.GetHashCode() ^ Y.GetHashCode();
  104. }
  105. public override string ToString()
  106. {
  107. return X + ", " + Y;
  108. }
  109. public string ToString(bool rounded)
  110. {
  111. if (rounded)
  112. {
  113. return (int)Math.Round(X) + ", " + (int)Math.Round(Y);
  114. }
  115. else
  116. {
  117. return ToString();
  118. }
  119. }
  120. public static Vector2 operator *(Vector2 value1, float value2)
  121. {
  122. value1.SetX(value1.X * value2);
  123. value1.SetY(value1.Y * value2);
  124. return value1;
  125. }
  126. #region IExternalizable 成员
  127. public void WriteExternal(IOutputStream output)
  128. {
  129. output.PutF32(x);
  130. output.PutF32(y);
  131. }
  132. public void ReadExternal(IInputStream input)
  133. {
  134. this.x = input.GetF32();
  135. this.y = input.GetF32();
  136. }
  137. #endregion
  138. }
  139. /// <summary>
  140. /// 2D向量
  141. /// </summary>
  142. public struct TVector2 : IVector2
  143. {
  144. public static readonly TVector2 Zero = new TVector2(0, 0);
  145. private float x;
  146. private float y;
  147. public float X { get { return this.x; } }
  148. public float Y { get { return this.y; } }
  149. public void SetX(float value)
  150. {
  151. this.x = value;
  152. }
  153. public void SetY(float value)
  154. {
  155. this.y = value;
  156. }
  157. public void AddX(float value)
  158. {
  159. this.x += value;
  160. }
  161. public void AddY(float value)
  162. {
  163. this.y += value;
  164. }
  165. public TVector2(float _x, float _y)
  166. {
  167. this.x = _x;
  168. this.y = _y;
  169. }
  170. public override bool Equals(object obj)
  171. {
  172. if (obj is Vector2)
  173. {
  174. Vector2 v = (Vector2)obj;
  175. return X == v.X && Y == v.Y;
  176. }
  177. return false;
  178. }
  179. public bool Equals(IVector2 v)
  180. {
  181. return v.X == X && v.Y == Y;
  182. }
  183. public override int GetHashCode()
  184. {
  185. return X.GetHashCode() ^ Y.GetHashCode();
  186. }
  187. public override string ToString()
  188. {
  189. return X + ", " + Y;
  190. }
  191. public string ToString(bool rounded)
  192. {
  193. if (rounded)
  194. {
  195. return (int)Math.Round(X) + ", " + (int)Math.Round(Y);
  196. }
  197. else
  198. {
  199. return ToString();
  200. }
  201. }
  202. #region IExternalizable 成员
  203. public void WriteExternal(IOutputStream output)
  204. {
  205. output.PutF32(x);
  206. output.PutF32(y);
  207. }
  208. public void ReadExternal(IInputStream input)
  209. {
  210. this.x = input.GetF32();
  211. this.y = input.GetF32();
  212. }
  213. #endregion
  214. public float Magnitude
  215. {
  216. get
  217. {
  218. return (float)Math.Sqrt(X * X + Y * Y);
  219. }
  220. }
  221. public void Normalize()
  222. {
  223. float magnitude = Magnitude;
  224. this.x = X / magnitude;
  225. this.y = Y / magnitude;
  226. }
  227. public TVector2 GetNormalized()
  228. {
  229. float magnitude = Magnitude;
  230. return new TVector2(X / magnitude, Y / magnitude);
  231. }
  232. public float DotProduct(TVector2 vector)
  233. {
  234. return this.X * vector.X + this.Y * vector.Y;
  235. }
  236. public float DistanceTo(TVector2 vector)
  237. {
  238. return (float)Math.Sqrt(Math.Pow(vector.X - this.X, 2) + Math.Pow(vector.Y - this.Y, 2));
  239. }
  240. public static TVector2 operator +(TVector2 a, TVector2 b)
  241. {
  242. return new TVector2(a.X + b.X, a.Y + b.Y);
  243. }
  244. public static TVector2 operator -(TVector2 a)
  245. {
  246. return new TVector2(-a.X, -a.Y);
  247. }
  248. public static TVector2 operator -(TVector2 a, TVector2 b)
  249. {
  250. return new TVector2(a.X - b.X, a.Y - b.Y);
  251. }
  252. public static TVector2 operator *(TVector2 a, float b)
  253. {
  254. return new TVector2(a.X * b, a.Y * b);
  255. }
  256. public static TVector2 operator *(TVector2 a, int b)
  257. {
  258. return new TVector2(a.X * b, a.Y * b);
  259. }
  260. public static TVector2 operator *(TVector2 a, double b)
  261. {
  262. return new TVector2((float)(a.X * b), (float)(a.Y * b));
  263. }
  264. }
  265. /// <summary>
  266. /// 2D极坐标向量
  267. /// </summary>
  268. public class Polar2 : ICloneable
  269. {
  270. public float direction;
  271. public float distance;
  272. public Polar2()
  273. {
  274. }
  275. public Polar2(float _direction, float _distance)
  276. {
  277. this.direction = _direction;
  278. this.distance = _distance;
  279. }
  280. public bool Equals(Polar2 v)
  281. {
  282. return v.direction == this.direction && v.distance == this.distance;
  283. }
  284. public object Clone()
  285. {
  286. return new Polar2(direction, distance);
  287. }
  288. #region IExternalizable 成员
  289. public void WriteExternal(IOutputStream output)
  290. {
  291. output.PutF32(direction);
  292. output.PutF32(distance);
  293. }
  294. public void ReadExternal(IInputStream input)
  295. {
  296. this.direction = input.GetF32();
  297. this.distance = input.GetF32();
  298. }
  299. #endregion
  300. }
  301. public struct TPolar2
  302. {
  303. public float direction;
  304. public float distance;
  305. public TPolar2(float _direction, float _distance)
  306. {
  307. this.direction = _direction;
  308. this.distance = _distance;
  309. }
  310. public bool Equals(TPolar2 v)
  311. {
  312. return v.direction == this.direction && v.distance == this.distance;
  313. }
  314. #region IExternalizable 成员
  315. public void WriteExternal(IOutputStream output)
  316. {
  317. output.PutF32(direction);
  318. output.PutF32(distance);
  319. }
  320. public void ReadExternal(IInputStream input)
  321. {
  322. this.direction = input.GetF32();
  323. this.distance = input.GetF32();
  324. }
  325. #endregion
  326. }
  327. public class Line2 : ICloneable
  328. {
  329. readonly public Vector2 p = new Vector2();
  330. readonly public Vector2 q = new Vector2();
  331. public Line2()
  332. {
  333. }
  334. public Line2(float x0, float y0, float x1, float y1)
  335. {
  336. p.SetX(x0);
  337. p.SetY(y0);
  338. q.SetX(x1);
  339. q.SetY(y1);
  340. }
  341. public float getMinX()
  342. {
  343. return Math.Min(p.X, q.X);
  344. }
  345. public float getMaxX()
  346. {
  347. return Math.Max(p.X, q.X);
  348. }
  349. public float getMinY()
  350. {
  351. return Math.Min(p.Y, q.Y);
  352. }
  353. public float getMaxY()
  354. {
  355. return Math.Max(p.Y, q.Y);
  356. }
  357. public object Clone()
  358. {
  359. return new Line2(p.X, p.Y, q.X, q.Y);
  360. }
  361. #region IExternalizable 成员
  362. public void WriteExternal(IOutputStream output)
  363. {
  364. output.PutF32(p.X);
  365. output.PutF32(p.Y);
  366. output.PutF32(q.X);
  367. output.PutF32(q.Y);
  368. }
  369. public void ReadExternal(IInputStream input)
  370. {
  371. p.SetX(input.GetF32());
  372. p.SetY(input.GetF32());
  373. q.SetX(input.GetF32());
  374. q.SetY(input.GetF32());
  375. }
  376. #endregion
  377. }
  378. public struct TLine2
  379. {
  380. public TVector2 p;
  381. public TVector2 q;
  382. public TLine2(float x0, float y0, float x1, float y1)
  383. {
  384. p = new TVector2();
  385. q = new TVector2();
  386. p.SetX(x0);
  387. p.SetY(y0);
  388. q.SetX(x1);
  389. q.SetY(y1);
  390. }
  391. public float getMinX()
  392. {
  393. return Math.Min(p.X, q.X);
  394. }
  395. public float getMaxX()
  396. {
  397. return Math.Max(p.X, q.X);
  398. }
  399. public float getMinY()
  400. {
  401. return Math.Min(p.Y, q.Y);
  402. }
  403. public float getMaxY()
  404. {
  405. return Math.Max(p.Y, q.Y);
  406. }
  407. #region IExternalizable 成员
  408. public void WriteExternal(IOutputStream output)
  409. {
  410. output.PutF32(p.X);
  411. output.PutF32(p.Y);
  412. output.PutF32(q.X);
  413. output.PutF32(q.Y);
  414. }
  415. public void ReadExternal(IInputStream input)
  416. {
  417. p.SetX(input.GetF32());
  418. p.SetY(input.GetF32());
  419. q.SetX(input.GetF32());
  420. q.SetY(input.GetF32());
  421. }
  422. #endregion
  423. }
  424. public static class MathVector
  425. {
  426. /**
  427. * 移动指定偏移
  428. * @param v
  429. * @param dx x距离
  430. * @param dy y距离
  431. */
  432. public static void move(IVector2 v, float dx, float dy)
  433. {
  434. v.AddX(dx);
  435. v.AddY(dy);
  436. }
  437. /**
  438. * 通过极坐标来移动
  439. * @param v
  440. * @param degree 弧度
  441. * @param distance 距离
  442. */
  443. public static void movePolar(IVector2 v, float degree, float distance)
  444. {
  445. float dx = (float)(Math.Cos(degree) * distance);
  446. float dy = (float)(Math.Sin(degree) * distance);
  447. move(v, dx, dy);
  448. }
  449. public static void movePolar(ref float x, ref float y, float degree, float distance)
  450. {
  451. x += (float)(Math.Cos(degree) * distance);
  452. y += (float)(Math.Sin(degree) * distance);
  453. }
  454. public static void movePolarExt(ref Vector2 pos, float degree, float distance)
  455. {
  456. pos.AddX((float)(Math.Cos(degree) * distance));
  457. pos.AddY((float)(Math.Sin(degree) * distance));
  458. }
  459. public static void movePolarExt(ref TVector2 pos, float degree, float distance)
  460. {
  461. pos.AddX((float)(Math.Cos(degree) * distance));
  462. pos.AddY((float)(Math.Sin(degree) * distance));
  463. }
  464. /**
  465. * 通过极坐标来移动
  466. * @param v
  467. * @param degree 弧度
  468. * @param speed 速度 (单位距离/秒)
  469. * @param interval_ms 毫秒时间
  470. */
  471. public static void movePolar(IVector2 v, float degree, float speed, float interval_ms)
  472. {
  473. float distance = getDistanceSpeedTime(speed, interval_ms);
  474. movePolar(v, degree, distance);
  475. }
  476. public static void movePolar(ref float x, ref float y, float degree, float speed, float interval_ms)
  477. {
  478. float distance = getDistanceSpeedTime(speed, interval_ms);
  479. movePolar(ref x, ref y, degree, distance);
  480. }
  481. public static void movePolar(ref Vector2 pos, float degree, float speed, float interval_ms)
  482. {
  483. float distance = getDistanceSpeedTime(speed, interval_ms);
  484. movePolarExt(ref pos, degree, distance);
  485. }
  486. /**
  487. * 向目标移动
  488. * @param v
  489. * @param x 目标x
  490. * @param y 目标y
  491. * @return 是否到达目的地
  492. */
  493. public static bool moveTo(IVector2 v, float dx, float dy, float distance)
  494. {
  495. float ddx = dx - v.X;
  496. float ddy = dy - v.Y;
  497. if (Math.Abs(ddx) < distance && Math.Abs(ddy) < distance)
  498. {
  499. v.SetX(dx);
  500. v.SetY(dy);
  501. return true;
  502. }
  503. else
  504. {
  505. float angle = (float)Math.Atan2(ddy, ddx);
  506. movePolar(v, angle, distance);
  507. return false;
  508. }
  509. }
  510. public static bool moveTo(ref float x, ref float y, float dx, float dy, float distance)
  511. {
  512. float ddx = dx - x;
  513. float ddy = dy - y;
  514. if (Math.Abs(ddx) < distance && Math.Abs(ddy) < distance)
  515. {
  516. x = (dx);
  517. y = (dy);
  518. return true;
  519. }
  520. else
  521. {
  522. float angle = (float)Math.Atan2(ddy, ddx);
  523. movePolar(ref x, ref y, angle, distance);
  524. return false;
  525. }
  526. }
  527. public static bool moveToX(IVector2 v, float x, float distance)
  528. {
  529. float ddx = x - v.X;
  530. if (Math.Abs(ddx) < distance)
  531. {
  532. v.SetX(x);
  533. return true;
  534. }
  535. else
  536. {
  537. if (ddx > 0)
  538. {
  539. v.AddX(distance);
  540. }
  541. else
  542. {
  543. v.AddX(-distance);
  544. }
  545. return false;
  546. }
  547. }
  548. public static bool moveToY(IVector2 v, float y, float distance)
  549. {
  550. float ddy = y - v.Y;
  551. if (Math.Abs(ddy) < distance)
  552. {
  553. v.SetY(y);
  554. return true;
  555. }
  556. else
  557. {
  558. if (ddy > 0)
  559. {
  560. v.AddY(distance);
  561. }
  562. else
  563. {
  564. v.AddY(-distance);
  565. }
  566. return false;
  567. }
  568. }
  569. public static void scale(IVector2 v, float scale)
  570. {
  571. v.SetX(v.X * scale);
  572. v.SetY(v.Y * scale);
  573. }
  574. public static void scale(IVector2 v, float scale_x, float scale_y)
  575. {
  576. v.SetX(v.X * scale_x);
  577. v.SetY(v.Y * scale_y);
  578. }
  579. public static void scale(ref float x, ref float y, float scale_x, float scale_y)
  580. {
  581. x = (x * scale_x);
  582. y = (y * scale_y);
  583. }
  584. public static void rotate(IVector2 v, float degree)
  585. {
  586. float cos_v = (float)Math.Cos(degree);
  587. float sin_v = (float)Math.Sin(degree);
  588. float x = (v.X) * cos_v - (v.Y) * sin_v;
  589. float y = (v.Y) * cos_v + (v.X) * sin_v;
  590. v.SetX(x);
  591. v.SetY(y);
  592. }
  593. public static void rotate(IVector2 v, IVector2 p0, float degree)
  594. {
  595. float dx = v.X - p0.X;
  596. float dy = v.Y - p0.Y;
  597. float cos_v = (float)Math.Cos(degree);
  598. float sin_v = (float)Math.Sin(degree);
  599. float x = p0.X + dx * cos_v - dy * sin_v;
  600. float y = p0.Y + dy * cos_v + dx * sin_v;
  601. v.SetX(x);
  602. v.SetY(y);
  603. }
  604. public static void rotate(IVector2 v, float px, float py, float degree)
  605. {
  606. float dx = v.X - px;
  607. float dy = v.Y - py;
  608. float cos_v = (float)Math.Cos(degree);
  609. float sin_v = (float)Math.Sin(degree);
  610. float x = px + dx * cos_v - dy * sin_v;
  611. float y = py + dy * cos_v + dx * sin_v;
  612. v.SetX(x);
  613. v.SetY(y);
  614. }
  615. public static void rotate(ref float x, ref float y, float px, float py, float degree)
  616. {
  617. float dx = x - px;
  618. float dy = y - py;
  619. float cos_v = (float)Math.Cos(degree);
  620. float sin_v = (float)Math.Sin(degree);
  621. float rx = px + dx * cos_v - dy * sin_v;
  622. float ry = py + dy * cos_v + dx * sin_v;
  623. x = (rx);
  624. y = (ry);
  625. }
  626. public static float getDirection(float d)
  627. {
  628. if (d > 0)
  629. {
  630. return 1;
  631. }
  632. if (d < 0)
  633. {
  634. return -1;
  635. }
  636. return 0;
  637. }
  638. /**
  639. * 得到速度和时间产生的距离
  640. * @param speed 速度 (单位距离/秒)
  641. * @param interval_ms 毫秒时间
  642. * @return
  643. */
  644. public static float getDistanceSpeedTime(float speed, float interval_ms)
  645. {
  646. float rate = interval_ms / 1000f;
  647. return speed * rate;
  648. }
  649. public static float getDistance(float x1, float y1, float x2, float y2)
  650. {
  651. float r1 = x1 - x2;
  652. float r2 = y1 - y2;
  653. return (float)Math.Sqrt(r1 * r1 + r2 * r2);
  654. }
  655. public static float getDistanceSquare(float x1, float y1, float x2, float y2)
  656. {
  657. float r1 = x1 - x2;
  658. float r2 = y1 - y2;
  659. return r1 * r1 + r2 * r2;
  660. }
  661. public static float getDistance(IVector2 v1, IVector2 v2)
  662. {
  663. float r1 = v1.X - v2.X;
  664. float r2 = v1.Y - v2.Y;
  665. return (float)Math.Sqrt(r1 * r1 + r2 * r2);
  666. }
  667. public static float getDistanceSquare(IVector2 v1, IVector2 v2)
  668. {
  669. float r1 = v1.X - v2.X;
  670. float r2 = v1.Y - v2.Y;
  671. return (r1 * r1 + r2 * r2);
  672. }
  673. /// <summary>
  674. /// 得到弧度
  675. /// </summary>
  676. /// <param name="dx">x向量</param>
  677. /// <param name="dy">y向量</param>
  678. /// <returns></returns>
  679. public static float getDegree(float dx, float dy)
  680. {
  681. return (float)Math.Atan2(dy, dx);
  682. }
  683. /// <summary>
  684. ///
  685. /// </summary>
  686. /// <param name="x1"></param>
  687. /// <param name="y1"></param>
  688. /// <param name="x2"></param>
  689. /// <param name="y2"></param>
  690. /// <returns></returns>
  691. public static float getDegree(float x1, float y1, float x2, float y2)
  692. {
  693. return (float)Math.Atan2(y2 - y1, x2 - x1);
  694. }
  695. /**
  696. * 得到弧度
  697. * @param v 向量
  698. * @return
  699. */
  700. public static float getDegree(IVector2 v)
  701. {
  702. return (float)Math.Atan2(v.Y, v.X);
  703. }
  704. public static float getDegree(IVector2 a, IVector2 b)
  705. {
  706. return (float)Math.Atan2(b.Y - a.Y, b.X - a.X);
  707. }
  708. /**
  709. * 将2个向量相加得到一个新的向量
  710. * @param a
  711. * @param b
  712. * @return
  713. */
  714. public static Vector2 vectorAdd(IVector2 a, IVector2 b)
  715. {
  716. Vector2 v = new Vector2();
  717. v.SetX(a.X + b.X);
  718. v.SetY(a.Y + b.Y);
  719. return v;
  720. }
  721. /**
  722. * 将2个向量相减得到一个新的向量
  723. * @param a
  724. * @param b
  725. * @return
  726. */
  727. public static Vector2 vectorSub(IVector2 a, IVector2 b)
  728. {
  729. Vector2 v = new Vector2();
  730. v.SetX(a.X - b.X);
  731. v.SetY(a.Y - b.Y);
  732. return v;
  733. }
  734. /**
  735. * 将一个向量加上新的向量,得到一个新的向量
  736. * @param a
  737. * @param degree
  738. * @param distance
  739. * @return
  740. */
  741. public static Vector2 vectorAdd(IVector2 a, float degree, float distance)
  742. {
  743. Vector2 v = new Vector2();
  744. v.SetX(a.X);
  745. v.SetY(a.Y);
  746. movePolar(v, degree, distance);
  747. return v;
  748. }
  749. /**
  750. * 把一个向量向自己本身的方向相加,得到一个新的向量
  751. * @param a
  752. * @param distance
  753. * @return
  754. */
  755. public static Vector2 vectorAdd(IVector2 a, float distance)
  756. {
  757. Vector2 v = new Vector2();
  758. v.SetX(a.X);
  759. v.SetY(a.Y);
  760. movePolar(v, getDegree(v), distance);
  761. return v;
  762. }
  763. /**
  764. * 将一个向量缩放一定比率后,得到一个新的向量
  765. * @param a
  766. * @param scale
  767. * @return
  768. */
  769. public static Vector2 vectorScale(IVector2 a, float scale)
  770. {
  771. Vector2 v = new Vector2();
  772. v.SetX(a.X * scale);
  773. v.SetY(a.Y * scale);
  774. return v;
  775. }
  776. public static float vectorDot(IVector2 v1, IVector2 v2)
  777. {
  778. return v1.X * v2.X + v1.Y * v2.Y;
  779. }
  780. public static float vectorDot(float x1, float y1, float x2, float y2)
  781. {
  782. return x1 * x2 + y1 * y2;
  783. }
  784. /// <summary>
  785. /// 挤压移动单位,某个单位在集合中移动,碰撞并挤开其他单位
  786. /// </summary>
  787. /// <param name="vectors"></param>
  788. /// <param name="obj"></param>
  789. /// <param name="angle"></param>
  790. /// <param name="distance"></param>
  791. /// <param name="depth"></param>
  792. /// <param name="max_depth"></param>
  793. public static void moveImpact(ICollection<IRoundObject> vectors, IRoundObject obj, float angle, float distance, int depth, int max_depth)
  794. {
  795. float dx = (float)(Math.Cos(angle) * distance);
  796. float dy = (float)(Math.Sin(angle) * distance);
  797. obj.AddX(dx);
  798. obj.AddY(dy);
  799. if (depth < max_depth)
  800. {
  801. foreach (IRoundObject o in vectors)
  802. {
  803. if (!o.Equals(obj))
  804. {
  805. float dr = MathVector.getDistance(o, obj) - o.RadiusSize - obj.RadiusSize;
  806. if (dr < 0)
  807. {
  808. float ta = MathVector.getDegree(obj.X, obj.Y, o.X, o.Y);
  809. moveImpact(vectors, o, ta, -dr, depth + 1, max_depth);
  810. }
  811. }
  812. }
  813. }
  814. }
  815. }
  816. public static class VectorGroupHelper
  817. {
  818. public static void GetCenterOf<T>(ICollection<T> vectors, out float cx, out float cy) where T : IVector2
  819. {
  820. if (vectors.Count == 0)
  821. {
  822. cx = 0;
  823. cy = 0;
  824. }
  825. else
  826. {
  827. float min_x = float.MaxValue;
  828. float max_x = float.MinValue;
  829. float min_y = float.MaxValue;
  830. float max_y = float.MinValue;
  831. foreach (T a in vectors)
  832. {
  833. min_x = Math.Min(min_x, a.X);
  834. max_x = Math.Max(max_x, a.X);
  835. min_y = Math.Min(min_y, a.Y);
  836. max_y = Math.Max(max_y, a.Y);
  837. }
  838. cx = min_x + (max_x - min_x) / 2f;
  839. cy = min_y + (max_y - min_y) / 2f;
  840. }
  841. }
  842. public static void MoveImpactInner<T>(ICollection<T> vectors, T obj, float spacing_size, float angle, float distance, int depth, int max_depth) where T : IVector2
  843. {
  844. float dx = (float)(Math.Cos(angle) * distance);
  845. float dy = (float)(Math.Sin(angle) * distance);
  846. obj.AddX(dx);
  847. obj.AddY(dy);
  848. if (depth < max_depth)
  849. {
  850. float dr2 = spacing_size * 2;
  851. foreach (T o in vectors)
  852. {
  853. if (!o.Equals(obj))
  854. {
  855. float dr = MathVector.getDistance(o, obj) - dr2;
  856. if (dr < 0)
  857. {
  858. float ta = MathVector.getDegree(obj.X, obj.Y, o.X, o.Y);
  859. MoveImpactInner(vectors, o, spacing_size, ta, -dr, depth + 1, max_depth);
  860. }
  861. }
  862. }
  863. }
  864. }
  865. /// <summary>
  866. /// 随机调整每个点,
  867. /// 使得距离都最小保持在spacing_size
  868. /// </summary>
  869. public static void DistributeSpacingSizeRandom<T>(IVector2 center, ICollection<T> vectors, float spacing_size, Random random) where T : IVector2
  870. {
  871. foreach (T o in vectors)
  872. {
  873. MoveImpactInner<T>(vectors, o, spacing_size, (float)(random.NextDouble() * CMath.PI_MUL_2), 0, 0, 1);
  874. }
  875. }
  876. /// <summary>
  877. /// 按正方形调整每个点,
  878. /// 使得距离都最小保持在spacing_size
  879. /// </summary>
  880. public static void DistributeSpacingSizeSquare<T>(IVector2 center, ICollection<T> vectors, float spacing_size) where T : IVector2
  881. {
  882. float cx = center.X;
  883. float cy = center.Y;
  884. int row_count = (int)Math.Round(Math.Sqrt(vectors.Count));
  885. //GetCenterOf(vectors, out cx, out cy);
  886. float sx = cx - (row_count - 1) * spacing_size / 2;
  887. float sy = cy - (vectors.Count / row_count - 1) * spacing_size / 2;
  888. int i = 0;
  889. foreach (T o in vectors)
  890. {
  891. int x = i % row_count;
  892. int y = i / row_count;
  893. o.SetX(sx + x * spacing_size);
  894. o.SetY(sy + y * spacing_size);
  895. i++;
  896. }
  897. }
  898. /// <summary>
  899. /// 按圆形调整每个点,
  900. /// 使得距离都最小保持在spacing_size
  901. /// </summary>
  902. public static void DistributeSpacingSizeRound<T>(IVector2 center, ICollection<T> vectors, float spacing_size) where T : IVector2
  903. {
  904. float cx = center.X;
  905. float cy = center.Y;
  906. int count = vectors.Count;
  907. T[] array = new T[count];
  908. vectors.CopyTo(array, 0);
  909. int cycle = 0;
  910. int i = 0;
  911. while (i < count)
  912. {
  913. if (i == 0)
  914. {
  915. array[i].SetX(cx);
  916. array[i].SetY(cy);
  917. i++;
  918. }
  919. else
  920. {
  921. float cr = spacing_size * cycle;
  922. float clen = cr * 2 * CMath.PI_F;
  923. int ccount = (int)(clen / spacing_size);
  924. float cangle = CMath.PI_MUL_2 / ccount;
  925. for (int j = 0; (j < ccount) && (i < count); j++)
  926. {
  927. float da = cangle * j;
  928. array[i].SetX(cx + (float)Math.Cos(da) * cr);
  929. array[i].SetY(cy + (float)Math.Sin(da) * cr);
  930. i++;
  931. }
  932. }
  933. cycle++;
  934. }
  935. }
  936. /// <summary>
  937. /// 按环形调整每个点,
  938. /// 使得距离都最小保持在spacing_size
  939. /// </summary>
  940. public static void DistributeSpacingSizeCycle<T>(IVector2 center, ICollection<T> vectors, float spacing_size) where T : IVector2
  941. {
  942. float cx = center.X;
  943. float cy = center.Y;
  944. int count = vectors.Count;
  945. float total_len = count * spacing_size;
  946. float total_r = total_len / CMath.PI_F / 2f;
  947. float sangle = CMath.PI_MUL_2 / count;
  948. int i = 0;
  949. foreach (T o in vectors)
  950. {
  951. float da = sangle * i;
  952. o.SetX(cx + (float)Math.Cos(da) * total_r);
  953. o.SetY(cy + (float)Math.Sin(da) * total_r);
  954. i++;
  955. }
  956. }
  957. /// <summary>
  958. /// 按蜂窝状调整每个点,
  959. /// 使得距离都最小保持在spacing_size
  960. /// </summary>
  961. public static void DistributeSpacingSizeBeehive<T>(IVector2 center, ICollection<T> vectors, float spacing_size) where T : IVector2
  962. {
  963. float cx = center.X;
  964. float cy = center.Y;
  965. int count = vectors.Count;
  966. T[] array = new T[count];
  967. vectors.CopyTo(array, 0);
  968. int cycle = 0;
  969. int i = 0;
  970. float d_angle = CMath.PI_MUL_2 / 6;
  971. while (i < count)
  972. {
  973. if (i == 0)
  974. {
  975. array[i].SetX(cx);
  976. array[i].SetY(cy);
  977. i++;
  978. }
  979. else
  980. {
  981. float c_r = spacing_size * cycle;
  982. for (int j = 0; (j < 6) && (i < count); j++)
  983. {
  984. float s_angle = d_angle * j;
  985. float s_x = cx + (float)Math.Cos(s_angle) * c_r;
  986. float s_y = cy + (float)Math.Sin(s_angle) * c_r;
  987. float b_angle = s_angle + d_angle * 2;
  988. for (int aj = 0; (aj < cycle) && (i < count); aj++)
  989. {
  990. float blen = aj * spacing_size;
  991. array[i].SetX(s_x + (float)Math.Cos(b_angle) * blen);
  992. array[i].SetY(s_y + (float)Math.Sin(b_angle) * blen);
  993. i++;
  994. }
  995. }
  996. }
  997. cycle++;
  998. }
  999. }
  1000. /// <summary>
  1001. /// 直线
  1002. /// 使得距离都最小保持在spacing_size
  1003. /// </summary>
  1004. public static void DistributeSpacingSizeLine<T>(IVector2 center, ICollection<T> vectors, float spacing_size) where T : IVector2
  1005. {
  1006. int count = vectors.Count / 2;
  1007. float statX = center.X - spacing_size * count;
  1008. float startY = center.Y - spacing_size * count;
  1009. if (vectors.Count % 2 != 0)
  1010. {
  1011. statX += spacing_size / 2;
  1012. startY = spacing_size / 2;
  1013. }
  1014. int i = 0;
  1015. foreach (T o in vectors)
  1016. {
  1017. o.SetX(statX + i * spacing_size);
  1018. o.SetY(startY + i * spacing_size);
  1019. i++;
  1020. }
  1021. }
  1022. }
  1023. }