| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  | import { arrayDeleteValue, newEmptyMap, fastArrayDeleteValue } from "../core/utils"; | 
					
						
							|  |  |  | import { Component } from "./component"; | 
					
						
							|  |  |  | import { GameRoot } from "./root"; | 
					
						
							|  |  |  | import { Entity } from "./entity"; | 
					
						
							|  |  |  | import { BasicSerializableObject, types } from "../savegame/serialization"; | 
					
						
							|  |  |  | import { createLogger } from "../core/logging"; | 
					
						
							|  |  |  | import { globalConfig } from "../core/config"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const logger = createLogger("entity_manager"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Manages all entities
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // NOTICE: We use arrayDeleteValue instead of fastArrayDeleteValue since that does not preserve the order
 | 
					
						
							|  |  |  | // This is slower but we need it for the street path generation
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  | /** @typedef {number} EntityUid */ | 
					
						
							|  |  |  | /** @typedef {string} ComponentId */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  | export class EntityManager extends BasicSerializableObject { | 
					
						
							|  |  |  |     constructor(root) { | 
					
						
							|  |  |  |         super(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /** @type {GameRoot} */ | 
					
						
							|  |  |  |         this.root = root; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  |         /** @type {Set<Entity>} */ | 
					
						
							|  |  |  |         this.entities = new Set(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /** @type {Map<EntityUid, Entity>} */ | 
					
						
							|  |  |  |         this.entitiesByUid = new Map(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /** @type {Map<ComponentId, Set<Entity>>} */ | 
					
						
							|  |  |  |         this.entitiesByComponent = new Map(); | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-19 05:49:17 +00:00
										 |  |  |         // We store a separate list with entities to destroy, since we don't destroy
 | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  |         // them instantly
 | 
					
						
							|  |  |  |         /** @type {Array<Entity>} */ | 
					
						
							|  |  |  |         this.destroyList = []; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Store a map from componentid to entities - This is used by the game system
 | 
					
						
							|  |  |  |         // for faster processing
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  |         ///** @type {Object.<string, Array<Entity>>} */
 | 
					
						
							|  |  |  |         //this.componentToEntity = newEmptyMap();
 | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Store the next uid to use
 | 
					
						
							|  |  |  |         this.nextUid = 10000; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     static getId() { | 
					
						
							|  |  |  |         return "EntityManager"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     static getSchema() { | 
					
						
							|  |  |  |         return { | 
					
						
							|  |  |  |             nextUid: types.uint, | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     getStatsText() { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  |         return this.entities.size + " entities [" + this.destroyList.length + " to kill]"; | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Main update
 | 
					
						
							|  |  |  |     update() { | 
					
						
							|  |  |  |         this.processDestroyList(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param {Entity} entity | 
					
						
							|  |  |  |      * @param {ComponentId} componentId | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     addToComponentMap(entity, componentId) { | 
					
						
							|  |  |  |         let set; | 
					
						
							|  |  |  |         if ((set = this.entitiesByComponent.get(componentId))) { | 
					
						
							|  |  |  |             set.add(entity); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             this.entitiesByComponent.set(componentId, new Set([entity])); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Registers a new entity | 
					
						
							|  |  |  |      * @param {Entity} entity | 
					
						
							|  |  |  |      * @param {number=} uid Optional predefined uid | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     registerEntity(entity, uid = null) { | 
					
						
							|  |  |  |         if (G_IS_DEV && !globalConfig.debug.disableSlowAsserts) { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  |             assert(!this.entities.has(entity), `RegisterEntity() called twice for entity ${entity}`); | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |         assert(!entity.destroyed, `Attempting to register destroyed entity ${entity}`); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-18 18:41:40 +00:00
										 |  |  |         if (G_IS_DEV && !globalConfig.debug.disableSlowAsserts && uid !== null) { | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  |             assert(!this.findByUid(uid, false), "Entity uid already taken: " + uid); | 
					
						
							|  |  |  |             assert(uid >= 0 && uid < Number.MAX_SAFE_INTEGER, "Invalid uid passed: " + uid); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  |         // Give each entity a unique id
 | 
					
						
							|  |  |  |         entity.uid = uid ? uid : this.generateUid(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         this.entities.add(entity); | 
					
						
							|  |  |  |         this.entitiesByUid.set(uid, entity); | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Register into the componentToEntity map
 | 
					
						
							|  |  |  |         for (const componentId in entity.components) { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  |             this.addToComponentMap(entity, componentId); | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         entity.registered = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         this.root.signals.entityAdded.dispatch(entity); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Generates a new uid | 
					
						
							|  |  |  |      * @returns {number} | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     generateUid() { | 
					
						
							|  |  |  |         return this.nextUid++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Call to attach a new component after the creation of the entity | 
					
						
							|  |  |  |      * @param {Entity} entity | 
					
						
							|  |  |  |      * @param {Component} component | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     attachDynamicComponent(entity, component) { | 
					
						
							|  |  |  |         entity.addComponent(component, true); | 
					
						
							|  |  |  |         const componentId = /** @type {typeof Component} */ (component.constructor).getId(); | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         this.addToComponentMap(entity, componentId); | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  |         this.root.signals.entityGotNewComponent.dispatch(entity); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Call to remove a component after the creation of the entity | 
					
						
							|  |  |  |      * @param {Entity} entity | 
					
						
							|  |  |  |      * @param {typeof Component} component | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     removeDynamicComponent(entity, component) { | 
					
						
							|  |  |  |         entity.removeComponent(component, true); | 
					
						
							|  |  |  |         const componentId = /** @type {typeof Component} */ (component.constructor).getId(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  |         this.entitiesByComponent.get(componentId).delete(entity); | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  |         this.root.signals.entityComponentRemoved.dispatch(entity); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Finds an entity buy its uid, kinda slow since it loops over all entities | 
					
						
							|  |  |  |      * @param {number} uid | 
					
						
							|  |  |  |      * @param {boolean=} errorWhenNotFound | 
					
						
							|  |  |  |      * @returns {Entity} | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     findByUid(uid, errorWhenNotFound = true) { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  |         const entity = this.entitiesByUid.get(uid); | 
					
						
							|  |  |  |         if (entity) { | 
					
						
							|  |  |  |             if (entity.queuedForDestroy || entity.destroyed) { | 
					
						
							|  |  |  |                 if (errorWhenNotFound) { | 
					
						
							|  |  |  |                     logger.warn("Entity with UID", uid, "not found (destroyed)"); | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  |                 return null; | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  |             return entity; | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |         if (errorWhenNotFound) { | 
					
						
							|  |  |  |             logger.warn("Entity with UID", uid, "not found"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return null; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-19 06:51:28 +00:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Returns a map which gives a mapping from UID to Entity. | 
					
						
							|  |  |  |      * This map is not updated. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @returns {Map<number, Entity>} | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     getFrozenUidSearchMap() { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  |         return this.entitiesByUid; | 
					
						
							| 
									
										
										
										
											2020-09-19 06:51:28 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Returns all entities having the given component | 
					
						
							|  |  |  |      * @param {typeof Component} componentHandle | 
					
						
							|  |  |  |      * @returns {Array<Entity>} entities | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     getAllWithComponent(componentHandle) { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  |         const set = this.entitiesByComponent.get(componentHandle.getId()); | 
					
						
							|  |  |  |         if (!set) return []; | 
					
						
							|  |  |  |         else return [...set.values()]; | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  |     // Deprecated lol
 | 
					
						
							|  |  |  |     // /**
 | 
					
						
							|  |  |  |     //  * Return all of a given class. This is SLOW!
 | 
					
						
							|  |  |  |     //  * @param {object} entityClass
 | 
					
						
							|  |  |  |     //  * @returns {Array<Entity>} entities
 | 
					
						
							|  |  |  |     //  */
 | 
					
						
							|  |  |  |     // getAllOfClass(entityClass) {
 | 
					
						
							|  |  |  |     //     // FIXME: Slow
 | 
					
						
							|  |  |  |     //     //  Fine! I will!
 | 
					
						
							|  |  |  |     //     const result = [];
 | 
					
						
							|  |  |  |     //     const entities = [...this.entities.values()];
 | 
					
						
							|  |  |  |     //     for (let i = entities.length; i >= 0; --i) {
 | 
					
						
							|  |  |  |     //         const entity = this.entities[i];
 | 
					
						
							|  |  |  |     //         if (entity instanceof entityClass) {
 | 
					
						
							|  |  |  |     //             result.push(entity);
 | 
					
						
							|  |  |  |     //         }
 | 
					
						
							|  |  |  |     //     }
 | 
					
						
							|  |  |  |     //     return result;
 | 
					
						
							|  |  |  |     // }
 | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Unregisters all components of an entity from the component to entity mapping | 
					
						
							|  |  |  |      * @param {Entity} entity | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     unregisterEntityComponents(entity) { | 
					
						
							|  |  |  |         for (const componentId in entity.components) { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  |             const set = this.entitiesByComponent.get(componentId); | 
					
						
							|  |  |  |             if (set) set.delete(entity); | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Processes the entities to destroy and actually destroys them
 | 
					
						
							|  |  |  |     /* eslint-disable max-statements */ | 
					
						
							|  |  |  |     processDestroyList() { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  |         for (let i = this.destroyList.length - 1; i >= 0; --i) { | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  |             const entity = this.destroyList[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Remove from entities list
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  |             this.entities.delete(entity); | 
					
						
							| 
									
										
										
										
											2020-09-27 19:00:11 +00:00
										 |  |  |             this.entitiesByUid.delete(entity.uid); | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             // Remove from componentToEntity list
 | 
					
						
							|  |  |  |             this.unregisterEntityComponents(entity); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             entity.registered = false; | 
					
						
							| 
									
										
										
										
											2020-09-19 06:51:28 +00:00
										 |  |  |             entity.destroyed = true; | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             this.root.signals.entityDestroyed.dispatch(entity); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         this.destroyList = []; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Queues an entity for destruction | 
					
						
							|  |  |  |      * @param {Entity} entity | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     destroyEntity(entity) { | 
					
						
							|  |  |  |         if (entity.destroyed) { | 
					
						
							|  |  |  |             logger.error("Tried to destroy already destroyed entity:", entity.uid); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (entity.queuedForDestroy) { | 
					
						
							|  |  |  |             logger.error("Trying to destroy entity which is already queued for destroy!", entity.uid); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:55:45 +00:00
										 |  |  |         this.destroyList.push(entity); | 
					
						
							|  |  |  |         entity.queuedForDestroy = true; | 
					
						
							|  |  |  |         this.root.signals.entityQueuedForDestroy.dispatch(entity); | 
					
						
							| 
									
										
										
										
											2020-09-18 17:51:15 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } |