systems/collision.system.js

  1. import {AABB, Actor, COLLIDERTYPES, Stage, System, Vector2} from '../index.js';
  2. /**
  3. * Creates collision systems.
  4. *
  5. * @example
  6. *
  7. * const system = new SystemCollision();
  8. */
  9. class SystemCollision extends System {
  10. /**
  11. * @typedef {Array<Actor>} TypePairActor A pair of actors.
  12. * @private
  13. */
  14. /**
  15. * Stores the current collision pairs.
  16. * @type {Array<TypePairActor>}
  17. * @private
  18. */
  19. $current;
  20. /**
  21. * Stores the previous collision pairs.
  22. * @type {Array<TypePairActor>}
  23. * @private
  24. */
  25. $previous;
  26. /**
  27. * Creates a new collision system.
  28. */
  29. constructor() {
  30. super();
  31. }
  32. /**
  33. * Checks if a collision previously existed between two given actors.
  34. * @param {Actor} $dynamic The first actor to check on.
  35. * @param {Actor} $inert The second actor to check with.
  36. * @returns {boolean}
  37. * @private
  38. */
  39. $hasCollisionPrevious($dynamic, $inert) {
  40. const result = this.$previous.find(([$dynamicPrevious, $inertPrevious]) => {
  41. return $dynamic === $dynamicPrevious
  42. && $inert === $inertPrevious;
  43. });
  44. return typeof result !== 'undefined';
  45. }
  46. /**
  47. * Called when the system is being initiated.
  48. * @public
  49. */
  50. onInitiate() {
  51. this.$current = [];
  52. this.$previous = [];
  53. }
  54. /**
  55. * Called when the system is being updated by one tick update.
  56. * @param {Object} $parameters The given parameters.
  57. * @param {Stage} $parameters.$stage The stage on which to execute the system.
  58. * @param {number} $parameters.$timetick The tick duration (in ms).
  59. * @public
  60. */
  61. onTick({$stage}) {
  62. /**
  63. * @typedef {Object} TypePairCollision A pair of candidates for collision.
  64. * @property {number} TypePairCollision.$distance The manhattan distance between the two actors.
  65. * @property {Actor} TypePairCollision.$dynamic The first actor.
  66. * @property {Actor} TypePairCollision.$inert The second actor.
  67. * @private
  68. */
  69. /**
  70. * @type {Array<TypePairCollision>}
  71. */
  72. const pairs = [];
  73. /**
  74. * @type {Array<Actor>}
  75. */
  76. const dynamics = [];
  77. /**
  78. * @type {Array<Actor>}
  79. */
  80. const kinetics = [];
  81. /**
  82. * @type {Array<Actor>}
  83. */
  84. const statics = [];
  85. $stage.actors.forEach(($actor) => {
  86. if ($actor.hasCollider() === false) {
  87. return;
  88. }
  89. switch ($actor.collider.type) {
  90. case COLLIDERTYPES.DYNAMIC: {
  91. dynamics.push($actor);
  92. break;
  93. }
  94. case COLLIDERTYPES.KINETIC: {
  95. kinetics.push($actor);
  96. break;
  97. }
  98. case COLLIDERTYPES.STATIC: {
  99. statics.push($actor);
  100. break;
  101. }
  102. }
  103. });
  104. const inerts = [...statics, ...kinetics];
  105. if (inerts.length === 0) {
  106. return;
  107. }
  108. dynamics.forEach(($dynamic) => {
  109. const boundariesDynamic = $dynamic.collider.boundaries.clone().translate($dynamic.translation);
  110. const centerBoundariesDynamic = new AABB(boundariesDynamic.center, boundariesDynamic.center);
  111. inerts.forEach(($inert) => {
  112. const boundariesInert = $inert.collider.boundaries.clone().translate($inert.translation);
  113. const distance = AABB.distanceManhattan(centerBoundariesDynamic, boundariesInert);
  114. pairs.push({
  115. $distance: distance,
  116. $dynamic: $dynamic,
  117. $inert: $inert
  118. });
  119. });
  120. });
  121. pairs.sort(($a, $b) => {
  122. return $a.$distance - $b.$distance;
  123. });
  124. pairs.forEach(($pair) => {
  125. const {$dynamic, $inert} = $pair;
  126. if ($stage.hasActor($dynamic) === false) {
  127. return;
  128. }
  129. if ($stage.hasActor($inert) === false) {
  130. return;
  131. }
  132. const boundariesDynamic = $dynamic.collider.boundaries.clone().translate($dynamic.translation);
  133. const boundariesInert = $inert.collider.boundaries.clone().translate($inert.translation);
  134. const overlapX = AABB.overlapX(boundariesDynamic, boundariesInert);
  135. if (overlapX <= 0) {
  136. return;
  137. }
  138. const overlapY = AABB.overlapY(boundariesDynamic, boundariesInert);
  139. if (overlapY <= 0) {
  140. return;
  141. }
  142. this.$current.push([$dynamic, $inert]);
  143. const directionX = Math.sign($inert.translation.x - $dynamic.translation.x);
  144. const directionY = Math.sign($inert.translation.y - $dynamic.translation.y);
  145. const checkMinimumX = (overlapX <= overlapY);
  146. const checkMinimumY = (overlapY <= overlapX);
  147. if ($dynamic.collider.traversable === false
  148. && $inert.collider.traversable === false) {
  149. const resolverDynamic = new Vector2(
  150. checkMinimumX ? - directionX * overlapX : 0,
  151. checkMinimumY ? - directionY * overlapY : 0
  152. );
  153. $dynamic.translate(resolverDynamic);
  154. }
  155. const originDynamicEast = checkMinimumX === true && directionX === 1;
  156. const originDynamicNorth = checkMinimumY === true && directionY === 1;
  157. const originDynamicSouth = checkMinimumY === true && directionY === -1;
  158. const originDynamicWest = checkMinimumX === true && directionX === -1;
  159. if (this.$hasCollisionPrevious($dynamic, $inert) === false) {
  160. $dynamic.onCollideEnter({
  161. $actor: $inert,
  162. $east: originDynamicEast,
  163. $north: originDynamicNorth,
  164. $south: originDynamicSouth,
  165. $west: originDynamicWest
  166. });
  167. $inert.onCollideEnter({
  168. $actor: $dynamic,
  169. $east: originDynamicWest,
  170. $north: originDynamicSouth,
  171. $south: originDynamicNorth,
  172. $west: originDynamicEast
  173. });
  174. }
  175. $dynamic.onCollide({
  176. $actor: $inert,
  177. $east: originDynamicEast,
  178. $north: originDynamicNorth,
  179. $south: originDynamicSouth,
  180. $west: originDynamicWest
  181. });
  182. $inert.onCollide({
  183. $actor: $dynamic,
  184. $east: originDynamicWest,
  185. $north: originDynamicSouth,
  186. $south: originDynamicNorth,
  187. $west: originDynamicEast
  188. });
  189. });
  190. this.$previous.filter(([$dynamicPrevious, $inertPrevious]) => {
  191. const result = this.$current.find(([$dynamic, $inert]) => {
  192. return $dynamic === $dynamicPrevious
  193. && $inert === $inertPrevious;
  194. });
  195. return typeof result === 'undefined';
  196. }).forEach(([$dynamicPrevious, $inertPrevious]) => {
  197. $dynamicPrevious.onCollideLeave($inertPrevious);
  198. $inertPrevious.onCollideLeave($dynamicPrevious);
  199. });
  200. this.$previous = [...this.$current];
  201. this.$current = [];
  202. }
  203. }
  204. export {
  205. SystemCollision
  206. };
  207. export default SystemCollision;