components/Node.js

  1. /**
  2. ## JSON Definition
  3. {
  4. "type": "NodeResident",
  5. "nodeId": "city-hall",
  6. // Optional. The id of the node that this entity should start on. Uses the entity's nodeId property if not set here.
  7. "nodes": ['path','sidewalk','road'],
  8. // Optional. This is a list of node types that this entity can reside on. If not set, entity can reside on any type of node.
  9. "shares": ['friends','neighbors','city-council-members'],
  10. // Optional. This is a list of entities that this entity can reside with on the same node. If not set, this entity cannot reside with any entities on the same node.
  11. "speed": 5,
  12. // Optional. Sets the speed with which the entity moves along an edge to an adjacent node. Default is 0 (instantaneous movement).
  13. "updateOrientation": true
  14. // Optional. Determines whether the entity's orientation is updated by movement across the NodeMap. Default is false.
  15. }
  16. */
  17. import {arrayCache, greenSplice} from '../utils/array.js';
  18. import Vector from '../Vector.js';
  19. import createComponentClass from '../factory.js';
  20. export default createComponentClass(/** @lends platypus.components.Node.prototype */{
  21. id: 'Node',
  22. properties: {
  23. /**
  24. * If provided, treats these property names as neighbors, assigning them to the neighbors object. For example, ["east", "west"] creates `entity.east` and `entity.west` entity properties that are pointers to those neighbors.
  25. *
  26. * @property neighborProperties
  27. * @type Array
  28. * @default null
  29. */
  30. neighborProperties: null
  31. },
  32. publicProperties: {
  33. x: 0,
  34. y: 0,
  35. z: 0
  36. },
  37. /**
  38. * This component causes an entity to be a position on a [[NodeMap]]. This component should not be confused with `NodeResident` which should be used on entities that move around on a NodeMap: `Node` simply represents a non-moving location on the NodeMap.
  39. *
  40. * @memberof platypus.components
  41. * @uses platypus.Component
  42. * @constructs
  43. * @param {*} definition
  44. */
  45. initialize: function (definition) {
  46. const owner = this.owner;
  47. this.nodeId = definition.nodeId || owner.nodeId || owner.id || String(Math.random());
  48. if ((typeof this.nodeId !== 'string') && (this.nodeId.length)) {
  49. this.nodeId = definition.nodeId.join('|');
  50. }
  51. owner.nodeId = this.nodeId;
  52. owner.isNode = true;
  53. this.map = owner.map = owner.map || owner.parent || null;
  54. this.contains = owner.contains = arrayCache.setUp();
  55. this.edgesContain = owner.edgesContain = arrayCache.setUp();
  56. Vector.assign(owner, 'position', 'x', 'y', 'z');
  57. this.neighbors = owner.neighbors = definition.neighbors || owner.neighbors || {};
  58. if (this.neighborProperties) {
  59. const properties = this.neighborProperties;
  60. for (let i = 0; i < properties.length; i++) {
  61. const
  62. propertyName = properties[i],
  63. value = owner[propertyName];
  64. if (value) {
  65. this.neighbors[propertyName] = value;
  66. }
  67. Object.defineProperty(owner, propertyName, {
  68. get: () => this.neighbors[propertyName],
  69. set: (value) => {
  70. if (value !== this.neighbors[propertyName]) {
  71. this.neighbors[propertyName] = value;
  72. for (let i = 0; i < this.contains.length; i++) {
  73. this.contains[i].triggerEvent('set-directions');
  74. }
  75. }
  76. }
  77. });
  78. }
  79. }
  80. },
  81. events: {
  82. "add-neighbors": function (neighbors) {
  83. for (const direction in neighbors) {
  84. if (neighbors.hasOwnProperty(direction)) {
  85. this.neighbors[direction] = neighbors[direction];
  86. }
  87. }
  88. for (let i = 0; i < this.contains.length; i++) {
  89. this.contains[i].triggerEvent('set-directions');
  90. }
  91. },
  92. "remove-neighbor": function (nodeOrNodeId) {
  93. var i = null,
  94. id = nodeOrNodeId;
  95. if (typeof id !== 'string') {
  96. id = id.nodeId;
  97. }
  98. for (i in this.neighbors) {
  99. if (this.neighbors.hasOwnProperty(i)) {
  100. if (typeof this.neighbors[i] === 'string') {
  101. if (this.neighbors[i] === id) {
  102. delete this.neighbors[i];
  103. break;
  104. }
  105. } else if (this.neighbors[i].nodeId === id) {
  106. delete this.neighbors[i];
  107. break;
  108. }
  109. }
  110. }
  111. }
  112. },
  113. methods: {
  114. destroy: function () {
  115. arrayCache.recycle(this.contains);
  116. this.contains = this.owner.contains = null;
  117. arrayCache.recycle(this.edgesContain);
  118. this.edgesContain = this.owner.edgesContain = null;
  119. }
  120. },
  121. publicMethods: {
  122. /**
  123. * Gets a neighboring node Entity.
  124. *
  125. * @memberof Node.prototype
  126. * @param {String} desc Describes the direction to check.
  127. * @returns {platypus.Entity}
  128. */
  129. getNode: function (desc) {
  130. var neighbor = null;
  131. //map check
  132. if (!this.map && this.owner.map) {
  133. this.map = this.owner.map;
  134. }
  135. if (this.neighbors[desc]) {
  136. neighbor = this.neighbors[desc];
  137. if (neighbor.isNode) {
  138. return neighbor;
  139. } else if (typeof neighbor === 'string') {
  140. neighbor = this.map.getNode(neighbor);
  141. if (neighbor) {
  142. this.neighbors[desc] = neighbor;
  143. return neighbor;
  144. }
  145. } else if (neighbor.length) {
  146. neighbor = this.map.getNode(neighbor.join('|'));
  147. if (neighbor) {
  148. this.neighbors[desc] = neighbor;
  149. return neighbor;
  150. }
  151. }
  152. return null;
  153. } else {
  154. return null;
  155. }
  156. },
  157. /**
  158. * Puts an entity on this node.
  159. *
  160. * @memberof Node.prototype
  161. * @param {platypus.Entity} entity
  162. * @returns {platypus.Entity}
  163. */
  164. addToNode: function (entity) {
  165. var i = 0;
  166. for (i = 0; i < this.contains.length; i++) {
  167. if (this.contains[i] === entity) {
  168. return false;
  169. }
  170. }
  171. this.contains.push(entity);
  172. return entity;
  173. },
  174. /**
  175. * Removes an entity from this node.
  176. *
  177. * @memberof Node.prototype
  178. * @param {platypus.Entity} entity
  179. * @returns {platypus.Entity}
  180. */
  181. removeFromNode: function (entity) {
  182. var i = 0;
  183. for (i = 0; i < this.contains.length; i++) {
  184. if (this.contains[i] === entity) {
  185. return greenSplice(this.contains, i);
  186. }
  187. }
  188. return false;
  189. },
  190. /**
  191. * Adds an entity to this node's edges.
  192. *
  193. * @memberof Node.prototype
  194. * @param {platypus.Entity} entity
  195. * @returns {platypus.Entity}
  196. */
  197. addToEdge: function (entity) {
  198. var i = 0;
  199. for (i = 0; i < this.edgesContain.length; i++) {
  200. if (this.edgesContain[i] === entity) {
  201. return false;
  202. }
  203. }
  204. this.edgesContain.push(entity);
  205. return entity;
  206. },
  207. /**
  208. * Removes an entity from this node's edges.
  209. *
  210. * @memberof Node.prototype
  211. * @param {platypus.Entity} entity
  212. * @returns {platypus.Entity}
  213. */
  214. removeFromEdge: function (entity) {
  215. var i = 0;
  216. for (i = 0; i < this.edgesContain.length; i++) {
  217. if (this.edgesContain[i] === entity) {
  218. return greenSplice(this.edgesContain, i);
  219. }
  220. }
  221. return false;
  222. }
  223. }
  224. });