| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  | import { globalConfig } from "../../core/config"; | 
					
						
							|  |  |  | import { DrawParameters } from "../../core/draw_parameters"; | 
					
						
							| 
									
										
										
										
											2020-06-27 07:59:48 +00:00
										 |  |  | import { gMetaBuildingRegistry } from "../../core/global_registries"; | 
					
						
							| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  | import { Loader } from "../../core/loader"; | 
					
						
							| 
									
										
										
										
											2020-06-25 10:18:48 +00:00
										 |  |  | import { createLogger } from "../../core/logging"; | 
					
						
							| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  | import { AtlasSprite } from "../../core/sprites"; | 
					
						
							| 
									
										
										
										
											2020-06-27 07:59:48 +00:00
										 |  |  | import { fastArrayDeleteValue } from "../../core/utils"; | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  | import { enumDirection, enumDirectionToVector, enumInvertedDirections, Vector } from "../../core/vector"; | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  | import { BeltPath } from "../belt_path"; | 
					
						
							| 
									
										
										
										
											2020-06-27 07:59:48 +00:00
										 |  |  | import { arrayBeltVariantToRotation, MetaBeltBaseBuilding } from "../buildings/belt_base"; | 
					
						
							| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  | import { BeltComponent } from "../components/belt"; | 
					
						
							|  |  |  | import { Entity } from "../entity"; | 
					
						
							|  |  |  | import { GameSystemWithFilter } from "../game_system_with_filter"; | 
					
						
							|  |  |  | import { MapChunkView } from "../map_chunk_view"; | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  | import { defaultBuildingVariant } from "../meta_building"; | 
					
						
							| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-25 10:18:48 +00:00
										 |  |  | export const BELT_ANIM_COUNT = 28; | 
					
						
							| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-18 14:08:33 +00:00
										 |  |  | const logger = createLogger("belt"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Manages all belts | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  | export class BeltSystem extends GameSystemWithFilter { | 
					
						
							|  |  |  |     constructor(root) { | 
					
						
							|  |  |  |         super(root, [BeltComponent]); | 
					
						
							|  |  |  |         /** | 
					
						
							|  |  |  |          * @type {Object.<enumDirection, Array<AtlasSprite>>} | 
					
						
							|  |  |  |          */ | 
					
						
							|  |  |  |         this.beltSprites = { | 
					
						
							|  |  |  |             [enumDirection.top]: Loader.getSprite("sprites/belt/forward_0.png"), | 
					
						
							|  |  |  |             [enumDirection.left]: Loader.getSprite("sprites/belt/left_0.png"), | 
					
						
							|  |  |  |             [enumDirection.right]: Loader.getSprite("sprites/belt/right_0.png"), | 
					
						
							|  |  |  |         }; | 
					
						
							| 
									
										
										
										
											2020-06-25 10:18:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-28 17:34:10 +00:00
										 |  |  |         /** | 
					
						
							|  |  |  |          * @type {Object.<enumDirection, Array<AtlasSprite>>} | 
					
						
							|  |  |  |          */ | 
					
						
							|  |  |  |         this.wireSprites = { | 
					
						
							|  |  |  |             [enumDirection.top]: Loader.getSprite("sprites/buildings/wire_top.png"), | 
					
						
							|  |  |  |             [enumDirection.left]: Loader.getSprite("sprites/buildings/wire_left.png"), | 
					
						
							|  |  |  |             [enumDirection.right]: Loader.getSprite("sprites/buildings/wire_right.png"), | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-25 10:18:48 +00:00
										 |  |  |         /** | 
					
						
							| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  |          * @type {Object.<enumDirection, Array<AtlasSprite>>} | 
					
						
							|  |  |  |          */ | 
					
						
							|  |  |  |         this.beltAnimations = { | 
					
						
							| 
									
										
										
										
											2020-06-25 10:18:48 +00:00
										 |  |  |             [enumDirection.top]: [], | 
					
						
							|  |  |  |             [enumDirection.left]: [], | 
					
						
							|  |  |  |             [enumDirection.right]: [], | 
					
						
							| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  |         }; | 
					
						
							| 
									
										
										
										
											2020-05-10 15:00:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-25 10:18:48 +00:00
										 |  |  |         for (let i = 0; i < BELT_ANIM_COUNT; ++i) { | 
					
						
							|  |  |  |             this.beltAnimations[enumDirection.top].push( | 
					
						
							|  |  |  |                 Loader.getSprite("sprites/belt/forward_" + i + ".png") | 
					
						
							|  |  |  |             ); | 
					
						
							|  |  |  |             this.beltAnimations[enumDirection.left].push(Loader.getSprite("sprites/belt/left_" + i + ".png")); | 
					
						
							|  |  |  |             this.beltAnimations[enumDirection.right].push( | 
					
						
							|  |  |  |                 Loader.getSprite("sprites/belt/right_" + i + ".png") | 
					
						
							|  |  |  |             ); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |         this.root.signals.entityDestroyed.add(this.onEntityDestroyed, this); | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  |         this.root.signals.entityDestroyed.add(this.updateSurroundingBeltPlacement, this); | 
					
						
							| 
									
										
										
										
											2020-05-18 14:08:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  |         // Notice: These must come *after* the entity destroyed signals
 | 
					
						
							|  |  |  |         this.root.signals.entityAdded.add(this.onEntityAdded, this); | 
					
						
							|  |  |  |         this.root.signals.entityAdded.add(this.updateSurroundingBeltPlacement, this); | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         /** @type {Array<BeltPath>} */ | 
					
						
							|  |  |  |         this.beltPaths = []; | 
					
						
							| 
									
										
										
										
											2020-06-26 16:24:02 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-26 16:24:02 +00:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Serializes all belt paths | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  |      * @returns {Array<object>} | 
					
						
							| 
									
										
										
										
											2020-06-26 16:24:02 +00:00
										 |  |  |      */ | 
					
						
							|  |  |  |     serializePaths() { | 
					
						
							|  |  |  |         let data = []; | 
					
						
							|  |  |  |         for (let i = 0; i < this.beltPaths.length; ++i) { | 
					
						
							|  |  |  |             data.push(this.beltPaths[i].serialize()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return data; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Deserializes all belt paths | 
					
						
							|  |  |  |      * @param {Array<any>} data | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     deserializePaths(data) { | 
					
						
							|  |  |  |         if (!Array.isArray(data)) { | 
					
						
							|  |  |  |             return "Belt paths are not an array: " + typeof data; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (let i = 0; i < data.length; ++i) { | 
					
						
							|  |  |  |             const path = BeltPath.fromSerialized(this.root, data[i]); | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  |             // If path is a string, that means its an error
 | 
					
						
							| 
									
										
										
										
											2020-06-26 16:24:02 +00:00
										 |  |  |             if (!(path instanceof BeltPath)) { | 
					
						
							|  |  |  |                 return "Failed to create path from belt data: " + path; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             this.beltPaths.push(path); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (this.beltPaths.length === 0) { | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  |             // Old savegames might not have paths yet
 | 
					
						
							|  |  |  |             logger.warn("Recomputing belt paths (most likely the savegame is old or empty)"); | 
					
						
							| 
									
										
										
										
											2020-06-26 16:24:02 +00:00
										 |  |  |             this.recomputeAllBeltPaths(); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             logger.warn("Restored", this.beltPaths.length, "belt paths"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-27 08:38:11 +00:00
										 |  |  |         if (G_IS_DEV && globalConfig.debug.checkBeltPaths) { | 
					
						
							|  |  |  |             this.debug_verifyBeltPaths(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-05-10 15:00:02 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Updates the belt placement after an entity has been added / deleted | 
					
						
							|  |  |  |      * @param {Entity} entity | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     updateSurroundingBeltPlacement(entity) { | 
					
						
							| 
									
										
										
										
											2020-05-16 20:45:40 +00:00
										 |  |  |         if (!this.root.gameInitialized) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-10 15:00:02 +00:00
										 |  |  |         const staticComp = entity.components.StaticMapEntity; | 
					
						
							|  |  |  |         if (!staticComp) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-30 06:23:05 +00:00
										 |  |  |         const metaBelt = gMetaBuildingRegistry.findByClass(MetaBeltBaseBuilding); | 
					
						
							| 
									
										
										
										
											2020-05-10 15:00:02 +00:00
										 |  |  |         // Compute affected area
 | 
					
						
							|  |  |  |         const originalRect = staticComp.getTileSpaceBounds(); | 
					
						
							|  |  |  |         const affectedArea = originalRect.expandedInAllDirections(1); | 
					
						
							| 
									
										
										
										
											2020-06-24 18:36:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-10 13:02:14 +00:00
										 |  |  |         /** @type {Set<BeltPath>} */ | 
					
						
							|  |  |  |         const changedPaths = new Set(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-10 15:00:02 +00:00
										 |  |  |         for (let x = affectedArea.x; x < affectedArea.right(); ++x) { | 
					
						
							|  |  |  |             for (let y = affectedArea.y; y < affectedArea.bottom(); ++y) { | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  |                 if (originalRect.containsPoint(x, y)) { | 
					
						
							|  |  |  |                     // Make sure we don't update the original entity
 | 
					
						
							|  |  |  |                     continue; | 
					
						
							| 
									
										
										
										
											2020-05-10 15:00:02 +00:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2020-06-17 00:23:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-30 07:27:30 +00:00
										 |  |  |                 const targetEntities = this.root.map.getLayersContentsMultipleXY(x, y); | 
					
						
							| 
									
										
										
										
											2020-06-30 06:23:05 +00:00
										 |  |  |                 for (let i = 0; i < targetEntities.length; ++i) { | 
					
						
							|  |  |  |                     const targetEntity = targetEntities[i]; | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-30 06:23:05 +00:00
										 |  |  |                     const targetBeltComp = targetEntity.components.Belt; | 
					
						
							|  |  |  |                     const targetStaticComp = targetEntity.components.StaticMapEntity; | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-30 06:23:05 +00:00
										 |  |  |                     if (!targetBeltComp) { | 
					
						
							|  |  |  |                         // Not a belt
 | 
					
						
							|  |  |  |                         continue; | 
					
						
							|  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-30 06:23:05 +00:00
										 |  |  |                     const { | 
					
						
							|  |  |  |                         rotation, | 
					
						
							|  |  |  |                         rotationVariant, | 
					
						
							|  |  |  |                     } = metaBelt.computeOptimalDirectionAndRotationVariantAtTile({ | 
					
						
							|  |  |  |                         root: this.root, | 
					
						
							|  |  |  |                         tile: new Vector(x, y), | 
					
						
							|  |  |  |                         rotation: targetStaticComp.originalRotation, | 
					
						
							|  |  |  |                         variant: defaultBuildingVariant, | 
					
						
							|  |  |  |                         layer: targetEntity.layer, | 
					
						
							|  |  |  |                     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     // Compute delta to see if anything changed
 | 
					
						
							|  |  |  |                     const newDirection = arrayBeltVariantToRotation[rotationVariant]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     if (targetStaticComp.rotation !== rotation || newDirection !== targetBeltComp.direction) { | 
					
						
							|  |  |  |                         // Ok, first remove it from its current path
 | 
					
						
							|  |  |  |                         this.deleteEntityFromPath(targetBeltComp.assignedPath, targetEntity); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         // Change stuff
 | 
					
						
							|  |  |  |                         targetStaticComp.rotation = rotation; | 
					
						
							|  |  |  |                         metaBelt.updateVariants(targetEntity, rotationVariant, defaultBuildingVariant); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         // Now add it again
 | 
					
						
							|  |  |  |                         this.addEntityToPaths(targetEntity); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         // Sanity
 | 
					
						
							|  |  |  |                         if (G_IS_DEV && globalConfig.debug.checkBeltPaths) { | 
					
						
							|  |  |  |                             this.debug_verifyBeltPaths(); | 
					
						
							|  |  |  |                         } | 
					
						
							| 
									
										
										
										
											2020-07-06 16:36:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |                         // Make sure the chunks know about the update
 | 
					
						
							|  |  |  |                         this.root.signals.entityChanged.dispatch(targetEntity); | 
					
						
							| 
									
										
										
										
											2020-06-27 08:38:11 +00:00
										 |  |  |                     } | 
					
						
							| 
									
										
										
										
											2020-08-10 13:02:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |                     if (targetBeltComp.assignedPath) { | 
					
						
							|  |  |  |                         changedPaths.add(targetBeltComp.assignedPath); | 
					
						
							|  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2020-06-25 10:42:48 +00:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-06-17 00:23:11 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-10 13:02:14 +00:00
										 |  |  |         // notify all paths *afterwards* to avoid multi-updates
 | 
					
						
							|  |  |  |         changedPaths.forEach(path => path.onSurroundingsChanged()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-27 08:38:11 +00:00
										 |  |  |         if (G_IS_DEV && globalConfig.debug.checkBeltPaths) { | 
					
						
							|  |  |  |             this.debug_verifyBeltPaths(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Called when an entity got destroyed | 
					
						
							|  |  |  |      * @param {Entity} entity | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     onEntityDestroyed(entity) { | 
					
						
							|  |  |  |         if (!this.root.gameInitialized) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!entity.components.Belt) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const assignedPath = entity.components.Belt.assignedPath; | 
					
						
							|  |  |  |         assert(assignedPath, "Entity has no belt path assigned"); | 
					
						
							| 
									
										
										
										
											2020-06-26 15:02:52 +00:00
										 |  |  |         this.deleteEntityFromPath(assignedPath, entity); | 
					
						
							| 
									
										
										
										
											2020-06-27 08:38:11 +00:00
										 |  |  |         if (G_IS_DEV && globalConfig.debug.checkBeltPaths) { | 
					
						
							|  |  |  |             this.debug_verifyBeltPaths(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-26 15:02:52 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-26 15:02:52 +00:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Attempts to delete the belt from its current path | 
					
						
							|  |  |  |      * @param {BeltPath} path | 
					
						
							|  |  |  |      * @param {Entity} entity | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     deleteEntityFromPath(path, entity) { | 
					
						
							|  |  |  |         if (path.entityPath.length === 1) { | 
					
						
							|  |  |  |             // This is a single entity path, easy to do, simply erase whole path
 | 
					
						
							|  |  |  |             fastArrayDeleteValue(this.beltPaths, path); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-26 15:02:52 +00:00
										 |  |  |         // Notice: Since there might be circular references, it is important to check
 | 
					
						
							|  |  |  |         // which role the entity has
 | 
					
						
							|  |  |  |         if (path.isStartEntity(entity)) { | 
					
						
							|  |  |  |             // We tried to delete the start
 | 
					
						
							|  |  |  |             path.deleteEntityOnStart(entity); | 
					
						
							|  |  |  |         } else if (path.isEndEntity(entity)) { | 
					
						
							|  |  |  |             // We tried to delete the end
 | 
					
						
							|  |  |  |             path.deleteEntityOnEnd(entity); | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2020-06-26 15:02:52 +00:00
										 |  |  |             // We tried to delete something inbetween
 | 
					
						
							|  |  |  |             const newPath = path.deleteEntityOnPathSplitIntoTwo(entity); | 
					
						
							|  |  |  |             this.beltPaths.push(newPath); | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Sanity
 | 
					
						
							|  |  |  |         entity.components.Belt.assignedPath = null; | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  |      * Adds the given entity to the appropriate paths | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |      * @param {Entity} entity | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  |     addEntityToPaths(entity) { | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |         const fromEntity = this.findSupplyingEntity(entity); | 
					
						
							|  |  |  |         const toEntity = this.findFollowUpEntity(entity); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Check if we can add the entity to the previous path
 | 
					
						
							|  |  |  |         if (fromEntity) { | 
					
						
							|  |  |  |             const fromPath = fromEntity.components.Belt.assignedPath; | 
					
						
							|  |  |  |             fromPath.extendOnEnd(entity); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Check if we now can extend the current path by the next path
 | 
					
						
							|  |  |  |             if (toEntity) { | 
					
						
							|  |  |  |                 const toPath = toEntity.components.Belt.assignedPath; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-26 14:31:36 +00:00
										 |  |  |                 if (fromPath === toPath) { | 
					
						
							|  |  |  |                     // This is a circular dependency -> Ignore
 | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     fromPath.extendByPath(toPath); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     // Delete now obsolete path
 | 
					
						
							|  |  |  |                     fastArrayDeleteValue(this.beltPaths, toPath); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             if (toEntity) { | 
					
						
							|  |  |  |                 // Prepend it to the other path
 | 
					
						
							|  |  |  |                 const toPath = toEntity.components.Belt.assignedPath; | 
					
						
							|  |  |  |                 toPath.extendOnBeginning(entity); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 // This is an empty belt path
 | 
					
						
							|  |  |  |                 const path = new BeltPath(this.root, [entity]); | 
					
						
							|  |  |  |                 this.beltPaths.push(path); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Called when an entity got added | 
					
						
							|  |  |  |      * @param {Entity} entity | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     onEntityAdded(entity) { | 
					
						
							|  |  |  |         if (!this.root.gameInitialized) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!entity.components.Belt) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-26 14:31:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  |         this.addEntityToPaths(entity); | 
					
						
							| 
									
										
										
										
											2020-06-27 08:38:11 +00:00
										 |  |  |         if (G_IS_DEV && globalConfig.debug.checkBeltPaths) { | 
					
						
							|  |  |  |             this.debug_verifyBeltPaths(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Draws all belt paths | 
					
						
							|  |  |  |      * @param {DrawParameters} parameters | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-08-10 19:24:58 +00:00
										 |  |  |     drawBeltItems(parameters) { | 
					
						
							| 
									
										
										
										
											2020-06-26 15:28:19 +00:00
										 |  |  |         for (let i = 0; i < this.beltPaths.length; ++i) { | 
					
						
							| 
									
										
										
										
											2020-08-10 19:24:58 +00:00
										 |  |  |             this.beltPaths[i].draw(parameters); | 
					
						
							| 
									
										
										
										
											2020-06-26 15:28:19 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-26 14:31:36 +00:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Verifies all belt paths | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-06-27 08:38:11 +00:00
										 |  |  |     debug_verifyBeltPaths() { | 
					
						
							|  |  |  |         for (let i = 0; i < this.beltPaths.length; ++i) { | 
					
						
							|  |  |  |             this.beltPaths[i].debug_checkIntegrity("general-verify"); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-26 14:31:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-27 08:38:11 +00:00
										 |  |  |         const belts = this.root.entityMgr.getAllWithComponent(BeltComponent); | 
					
						
							|  |  |  |         for (let i = 0; i < belts.length; ++i) { | 
					
						
							|  |  |  |             const path = belts[i].components.Belt.assignedPath; | 
					
						
							|  |  |  |             if (!path) { | 
					
						
							|  |  |  |                 throw new Error("Belt has no path: " + belts[i].uid); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (this.beltPaths.indexOf(path) < 0) { | 
					
						
							|  |  |  |                 throw new Error("Path of entity not contained: " + belts[i].uid); | 
					
						
							| 
									
										
										
										
											2020-06-26 14:31:36 +00:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-18 10:53:01 +00:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2020-05-18 14:08:33 +00:00
										 |  |  |      * Finds the follow up entity for a given belt. Used for building the dependencies | 
					
						
							| 
									
										
										
										
											2020-05-18 10:53:01 +00:00
										 |  |  |      * @param {Entity} entity | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  |      * @returns {Entity|null} | 
					
						
							| 
									
										
										
										
											2020-05-18 10:53:01 +00:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-05-18 14:08:33 +00:00
										 |  |  |     findFollowUpEntity(entity) { | 
					
						
							|  |  |  |         const staticComp = entity.components.StaticMapEntity; | 
					
						
							|  |  |  |         const beltComp = entity.components.Belt; | 
					
						
							| 
									
										
										
										
											2020-05-18 10:53:01 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-18 14:08:33 +00:00
										 |  |  |         const followUpDirection = staticComp.localDirectionToWorld(beltComp.direction); | 
					
						
							|  |  |  |         const followUpVector = enumDirectionToVector[followUpDirection]; | 
					
						
							| 
									
										
										
										
											2020-05-18 10:53:01 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-18 14:08:33 +00:00
										 |  |  |         const followUpTile = staticComp.origin.add(followUpVector); | 
					
						
							| 
									
										
										
										
											2020-06-28 17:34:10 +00:00
										 |  |  |         const followUpEntity = this.root.map.getLayerContentXY(followUpTile.x, followUpTile.y, entity.layer); | 
					
						
							| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-18 17:23:37 +00:00
										 |  |  |         // Check if theres a belt at the tile we point to
 | 
					
						
							| 
									
										
										
										
											2020-05-18 14:08:33 +00:00
										 |  |  |         if (followUpEntity) { | 
					
						
							|  |  |  |             const followUpBeltComp = followUpEntity.components.Belt; | 
					
						
							|  |  |  |             if (followUpBeltComp) { | 
					
						
							| 
									
										
										
										
											2020-05-18 17:23:37 +00:00
										 |  |  |                 const followUpStatic = followUpEntity.components.StaticMapEntity; | 
					
						
							| 
									
										
										
										
											2020-06-28 17:34:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-10 13:02:14 +00:00
										 |  |  |                 const acceptedDirection = followUpStatic.localDirectionToWorld(enumDirection.top); | 
					
						
							|  |  |  |                 if (acceptedDirection === followUpDirection) { | 
					
						
							|  |  |  |                     return followUpEntity; | 
					
						
							| 
									
										
										
										
											2020-05-18 17:23:37 +00:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2020-05-18 14:08:33 +00:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-05-18 10:53:01 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-18 14:08:33 +00:00
										 |  |  |         return null; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Finds the supplying belt for a given belt. Used for building the dependencies | 
					
						
							|  |  |  |      * @param {Entity} entity | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  |      * @returns {Entity|null} | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |      */ | 
					
						
							|  |  |  |     findSupplyingEntity(entity) { | 
					
						
							|  |  |  |         const staticComp = entity.components.StaticMapEntity; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const supplyDirection = staticComp.localDirectionToWorld(enumDirection.bottom); | 
					
						
							|  |  |  |         const supplyVector = enumDirectionToVector[supplyDirection]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const supplyTile = staticComp.origin.add(supplyVector); | 
					
						
							| 
									
										
										
										
											2020-06-28 17:34:10 +00:00
										 |  |  |         const supplyEntity = this.root.map.getLayerContentXY(supplyTile.x, supplyTile.y, entity.layer); | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Check if theres a belt at the tile we point to
 | 
					
						
							|  |  |  |         if (supplyEntity) { | 
					
						
							|  |  |  |             const supplyBeltComp = supplyEntity.components.Belt; | 
					
						
							|  |  |  |             if (supplyBeltComp) { | 
					
						
							|  |  |  |                 const supplyStatic = supplyEntity.components.StaticMapEntity; | 
					
						
							| 
									
										
										
										
											2020-08-10 13:02:14 +00:00
										 |  |  |                 const otherDirection = supplyStatic.localDirectionToWorld( | 
					
						
							|  |  |  |                     enumInvertedDirections[supplyBeltComp.direction] | 
					
						
							|  |  |  |                 ); | 
					
						
							| 
									
										
										
										
											2020-06-28 17:34:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-10 13:02:14 +00:00
										 |  |  |                 if (otherDirection === supplyDirection) { | 
					
						
							|  |  |  |                     return supplyEntity; | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return null; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  |      * Recomputes the belt path network. Only required for old savegames | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-06-26 16:24:02 +00:00
										 |  |  |     recomputeAllBeltPaths() { | 
					
						
							|  |  |  |         logger.warn("Recomputing all belt paths"); | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |         const visitedUids = new Set(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const result = []; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (let i = 0; i < this.allEntities.length; ++i) { | 
					
						
							|  |  |  |             const entity = this.allEntities[i]; | 
					
						
							|  |  |  |             if (visitedUids.has(entity.uid)) { | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Mark entity as visited
 | 
					
						
							|  |  |  |             visitedUids.add(entity.uid); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Compute path, start with entity and find precedors / successors
 | 
					
						
							|  |  |  |             const path = [entity]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  |             // Prevent infinite loops
 | 
					
						
							|  |  |  |             let maxIter = 99999; | 
					
						
							| 
									
										
										
										
											2020-06-26 14:31:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |             // Find precedors
 | 
					
						
							|  |  |  |             let prevEntity = this.findSupplyingEntity(entity); | 
					
						
							| 
									
										
										
										
											2020-06-26 14:31:36 +00:00
										 |  |  |             while (prevEntity && --maxIter > 0) { | 
					
						
							| 
									
										
										
										
											2020-06-26 15:02:52 +00:00
										 |  |  |                 if (visitedUids.has(prevEntity.uid)) { | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 path.unshift(prevEntity); | 
					
						
							|  |  |  |                 visitedUids.add(prevEntity.uid); | 
					
						
							|  |  |  |                 prevEntity = this.findSupplyingEntity(prevEntity); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Find succedors
 | 
					
						
							|  |  |  |             let nextEntity = this.findFollowUpEntity(entity); | 
					
						
							| 
									
										
										
										
											2020-06-26 14:31:36 +00:00
										 |  |  |             while (nextEntity && --maxIter > 0) { | 
					
						
							| 
									
										
										
										
											2020-06-26 15:02:52 +00:00
										 |  |  |                 if (visitedUids.has(nextEntity.uid)) { | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 path.push(nextEntity); | 
					
						
							|  |  |  |                 visitedUids.add(nextEntity.uid); | 
					
						
							|  |  |  |                 nextEntity = this.findFollowUpEntity(nextEntity); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-26 15:02:52 +00:00
										 |  |  |             assert(maxIter > 1, "Ran out of iterations"); | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |             result.push(new BeltPath(this.root, path)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         logger.log("Found", this.beltPaths.length, "belt paths"); | 
					
						
							|  |  |  |         this.beltPaths = result; | 
					
						
							| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Updates all belts | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-05-18 10:53:01 +00:00
										 |  |  |     update() { | 
					
						
							| 
									
										
										
										
											2020-06-27 08:38:11 +00:00
										 |  |  |         if (G_IS_DEV && globalConfig.debug.checkBeltPaths) { | 
					
						
							|  |  |  |             this.debug_verifyBeltPaths(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-26 14:31:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |         for (let i = 0; i < this.beltPaths.length; ++i) { | 
					
						
							|  |  |  |             this.beltPaths[i].update(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-27 08:38:11 +00:00
										 |  |  |         if (G_IS_DEV && globalConfig.debug.checkBeltPaths) { | 
					
						
							|  |  |  |             this.debug_verifyBeltPaths(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-05-18 10:53:01 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  |      * Draws a given chunk | 
					
						
							| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  |      * @param {DrawParameters} parameters | 
					
						
							|  |  |  |      * @param {MapChunkView} chunk | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     drawChunk(parameters, chunk) { | 
					
						
							|  |  |  |         if (parameters.zoomLevel < globalConfig.mapChunkOverviewMinZoom) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-27 07:59:48 +00:00
										 |  |  |         // Limit speed to avoid belts going backwards
 | 
					
						
							| 
									
										
										
										
											2020-08-10 19:24:58 +00:00
										 |  |  |         const speedMultiplier = Math.min(this.root.hubGoals.getBeltBaseSpeed(), 10); | 
					
						
							| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-18 09:47:17 +00:00
										 |  |  |         // SYNC with systems/item_acceptor.js:drawEntityUnderlays!
 | 
					
						
							| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  |         // 126 / 42 is the exact animation speed of the png animation
 | 
					
						
							|  |  |  |         const animationIndex = Math.floor( | 
					
						
							| 
									
										
										
										
											2020-06-27 07:59:48 +00:00
										 |  |  |             ((this.root.time.realtimeNow() * speedMultiplier * BELT_ANIM_COUNT * 126) / 42) * | 
					
						
							| 
									
										
										
										
											2020-08-10 19:24:58 +00:00
										 |  |  |                 globalConfig.itemSpacingOnBelts | 
					
						
							| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  |         ); | 
					
						
							|  |  |  |         const contents = chunk.contents; | 
					
						
							|  |  |  |         for (let y = 0; y < globalConfig.mapChunkSize; ++y) { | 
					
						
							|  |  |  |             for (let x = 0; x < globalConfig.mapChunkSize; ++x) { | 
					
						
							|  |  |  |                 const entity = contents[x][y]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (entity && entity.components.Belt) { | 
					
						
							|  |  |  |                     const direction = entity.components.Belt.direction; | 
					
						
							|  |  |  |                     const sprite = this.beltAnimations[direction][animationIndex % BELT_ANIM_COUNT]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     entity.components.StaticMapEntity.drawSpriteOnFullEntityBounds( | 
					
						
							|  |  |  |                         parameters, | 
					
						
							|  |  |  |                         sprite, | 
					
						
							|  |  |  |                         0, | 
					
						
							|  |  |  |                         false | 
					
						
							|  |  |  |                     ); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2020-06-27 07:34:09 +00:00
										 |  |  |      * Draws the belt path debug overlays | 
					
						
							| 
									
										
										
										
											2020-06-26 11:57:07 +00:00
										 |  |  |      * @param {DrawParameters} parameters | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     drawBeltPathDebug(parameters) { | 
					
						
							|  |  |  |         for (let i = 0; i < this.beltPaths.length; ++i) { | 
					
						
							|  |  |  |             this.beltPaths[i].drawDebug(parameters); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-05-09 14:45:23 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } |