2019-02-16 06:17:30 +02:00
|
|
|
if chunkUtilsG then
|
2020-04-28 05:41:18 +02:00
|
|
|
return chunkUtilsG
|
2019-02-16 06:17:30 +02:00
|
|
|
end
|
2016-08-04 00:22:20 +02:00
|
|
|
local chunkUtils = {}
|
2016-07-30 00:44:31 +02:00
|
|
|
|
2016-08-20 04:52:27 +02:00
|
|
|
-- imports
|
|
|
|
|
2018-01-15 09:41:55 +02:00
|
|
|
local baseUtils = require("BaseUtils")
|
2016-08-04 00:22:20 +02:00
|
|
|
local constants = require("Constants")
|
2017-11-21 09:27:03 +02:00
|
|
|
local mapUtils = require("MapUtils")
|
2018-01-15 09:41:55 +02:00
|
|
|
local chunkPropertyUtils = require("ChunkPropertyUtils")
|
2021-11-26 19:52:30 +02:00
|
|
|
local mathUtils = require("MathUtils")
|
2021-12-04 21:04:13 +02:00
|
|
|
local queryUtils = require("QueryUtils")
|
2016-08-05 06:47:51 +02:00
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
-- constants
|
2017-05-12 06:50:06 +02:00
|
|
|
|
2019-11-30 02:49:22 +02:00
|
|
|
local HIVE_BUILDINGS_TYPES = constants.HIVE_BUILDINGS_TYPES
|
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
local DEFINES_WIRE_TYPE_RED = defines.wire_type.red
|
|
|
|
local DEFINES_WIRE_TYPE_GREEN = defines.wire_type.green
|
2017-06-08 02:57:24 +02:00
|
|
|
|
2019-03-09 02:42:20 +02:00
|
|
|
local CHUNK_PASS_THRESHOLD = constants.CHUNK_PASS_THRESHOLD
|
|
|
|
|
2019-02-11 08:14:17 +02:00
|
|
|
local AI_STATE_ONSLAUGHT = constants.AI_STATE_ONSLAUGHT
|
|
|
|
|
2016-10-15 02:00:18 +02:00
|
|
|
local BASE_PHEROMONE = constants.BASE_PHEROMONE
|
2016-08-20 04:52:27 +02:00
|
|
|
local PLAYER_PHEROMONE = constants.PLAYER_PHEROMONE
|
2017-06-08 02:57:24 +02:00
|
|
|
local RESOURCE_PHEROMONE = constants.RESOURCE_PHEROMONE
|
2016-08-20 04:52:27 +02:00
|
|
|
local BUILDING_PHEROMONES = constants.BUILDING_PHEROMONES
|
|
|
|
|
2018-02-19 06:18:04 +02:00
|
|
|
local CHUNK_SIZE = constants.CHUNK_SIZE
|
|
|
|
local CHUNK_SIZE_DIVIDER = constants.CHUNK_SIZE_DIVIDER
|
|
|
|
|
2017-06-10 10:38:20 +02:00
|
|
|
local CHUNK_NORTH_SOUTH = constants.CHUNK_NORTH_SOUTH
|
|
|
|
local CHUNK_EAST_WEST = constants.CHUNK_EAST_WEST
|
2016-08-20 04:52:27 +02:00
|
|
|
|
2017-06-10 10:38:20 +02:00
|
|
|
local CHUNK_ALL_DIRECTIONS = constants.CHUNK_ALL_DIRECTIONS
|
|
|
|
local CHUNK_IMPASSABLE = constants.CHUNK_IMPASSABLE
|
2016-08-18 07:55:08 +02:00
|
|
|
|
2018-02-14 10:28:42 +02:00
|
|
|
local RESOURCE_NORMALIZER = constants.RESOURCE_NORMALIZER
|
|
|
|
|
2016-10-15 02:00:18 +02:00
|
|
|
local CHUNK_TICK = constants.CHUNK_TICK
|
|
|
|
|
2019-10-19 21:13:48 +02:00
|
|
|
local GENERATOR_PHEROMONE_LEVEL_1 = constants.GENERATOR_PHEROMONE_LEVEL_1
|
|
|
|
local GENERATOR_PHEROMONE_LEVEL_3 = constants.GENERATOR_PHEROMONE_LEVEL_3
|
|
|
|
local GENERATOR_PHEROMONE_LEVEL_5 = constants.GENERATOR_PHEROMONE_LEVEL_5
|
|
|
|
local GENERATOR_PHEROMONE_LEVEL_6 = constants.GENERATOR_PHEROMONE_LEVEL_6
|
|
|
|
|
2017-05-12 06:50:06 +02:00
|
|
|
-- imported functions
|
|
|
|
|
2021-12-04 21:04:13 +02:00
|
|
|
local setAreaInQuery = queryUtils.setAreaInQuery
|
|
|
|
|
2018-01-16 09:21:12 +02:00
|
|
|
local setPlayerBaseGenerator = chunkPropertyUtils.setPlayerBaseGenerator
|
|
|
|
local addPlayerBaseGenerator = chunkPropertyUtils.addPlayerBaseGenerator
|
|
|
|
local setResourceGenerator = chunkPropertyUtils.setResourceGenerator
|
|
|
|
local addResourceGenerator = chunkPropertyUtils.addResourceGenerator
|
2021-12-04 21:04:13 +02:00
|
|
|
|
|
|
|
local addNestCount = chunkPropertyUtils.addNestCount
|
|
|
|
local removeNestCount = chunkPropertyUtils.removeNestCount
|
|
|
|
local addHiveCount = chunkPropertyUtils.addHiveCount
|
|
|
|
local removeHiveCount = chunkPropertyUtils.removeHiveCount
|
|
|
|
local addTrapCount = chunkPropertyUtils.addTrapCount
|
|
|
|
local removeTrapCount = chunkPropertyUtils.removeTrapCount
|
|
|
|
local addTurretCount = chunkPropertyUtils.addTurretCount
|
|
|
|
local removeTurretCount = chunkPropertyUtils.removeTurretCount
|
|
|
|
local addUtilityCount = chunkPropertyUtils.addUtilityCount
|
|
|
|
local removeUtilityCount = chunkPropertyUtils.removeUtilityCount
|
|
|
|
|
2018-01-16 09:21:12 +02:00
|
|
|
local getPlayerBaseGenerator = chunkPropertyUtils.getPlayerBaseGenerator
|
|
|
|
local getNestCount = chunkPropertyUtils.getNestCount
|
2019-02-20 08:16:43 +02:00
|
|
|
local setRaidNestActiveness = chunkPropertyUtils.setRaidNestActiveness
|
|
|
|
local setNestActiveness = chunkPropertyUtils.setNestActiveness
|
2018-01-16 09:21:12 +02:00
|
|
|
|
2019-12-16 03:16:56 +02:00
|
|
|
local processNestActiveness = chunkPropertyUtils.processNestActiveness
|
|
|
|
|
2021-12-04 21:04:13 +02:00
|
|
|
local removeChunkBase = chunkPropertyUtils.removeChunkBase
|
2019-11-30 02:49:22 +02:00
|
|
|
local getEnemyStructureCount = chunkPropertyUtils.getEnemyStructureCount
|
|
|
|
|
2018-01-15 09:41:55 +02:00
|
|
|
local findNearbyBase = baseUtils.findNearbyBase
|
|
|
|
local createBase = baseUtils.createBase
|
2017-11-21 09:27:03 +02:00
|
|
|
|
2018-01-22 09:13:18 +02:00
|
|
|
local upgradeEntity = baseUtils.upgradeEntity
|
|
|
|
|
2021-11-26 19:52:30 +02:00
|
|
|
local euclideanDistancePoints = mathUtils.euclideanDistancePoints
|
|
|
|
|
2021-04-30 07:24:14 +02:00
|
|
|
local getChunkBase = chunkPropertyUtils.getChunkBase
|
2018-01-15 09:41:55 +02:00
|
|
|
local setChunkBase = chunkPropertyUtils.setChunkBase
|
2018-09-24 06:56:45 +02:00
|
|
|
local setPassable = chunkPropertyUtils.setPassable
|
|
|
|
local setPathRating = chunkPropertyUtils.setPathRating
|
2018-01-15 09:41:55 +02:00
|
|
|
|
2018-02-19 06:18:04 +02:00
|
|
|
local getChunkByXY = mapUtils.getChunkByXY
|
2018-01-14 07:48:21 +02:00
|
|
|
|
2020-05-25 05:25:21 +02:00
|
|
|
local mMin = math.min
|
|
|
|
local mMax = math.max
|
2017-11-21 09:27:03 +02:00
|
|
|
local mFloor = math.floor
|
2017-05-12 06:50:06 +02:00
|
|
|
|
2016-08-20 04:52:27 +02:00
|
|
|
-- module code
|
2017-05-06 11:03:28 +02:00
|
|
|
|
2018-01-14 07:48:21 +02:00
|
|
|
local function getEntityOverlapChunks(map, entity)
|
2021-11-26 08:49:28 +02:00
|
|
|
local boundingBox = entity.prototype.collision_box or entity.prototype.selection_box
|
2021-02-20 09:31:36 +02:00
|
|
|
local overlapArray = map.universe.chunkOverlapArray
|
2020-04-28 05:41:18 +02:00
|
|
|
|
2020-05-15 22:51:38 +02:00
|
|
|
overlapArray[1] = -1 --LeftTop
|
|
|
|
overlapArray[2] = -1 --RightTop
|
|
|
|
overlapArray[3] = -1 --LeftBottom
|
|
|
|
overlapArray[4] = -1 --RightBottom
|
2020-04-28 05:41:18 +02:00
|
|
|
|
|
|
|
if boundingBox then
|
|
|
|
local center = entity.position
|
|
|
|
|
2021-12-04 21:04:13 +02:00
|
|
|
local topXOffset = boundingBox.left_top.x
|
|
|
|
local topYOffset = boundingBox.left_top.y
|
2020-04-28 05:41:18 +02:00
|
|
|
|
2021-12-04 21:04:13 +02:00
|
|
|
local bottomXOffset = boundingBox.right_bottom.x
|
|
|
|
local bottomYOffset = boundingBox.right_bottom.y
|
2020-04-28 05:41:18 +02:00
|
|
|
|
|
|
|
local leftTopChunkX = mFloor((center.x + topXOffset) * CHUNK_SIZE_DIVIDER) * CHUNK_SIZE
|
|
|
|
local leftTopChunkY = mFloor((center.y + topYOffset) * CHUNK_SIZE_DIVIDER) * CHUNK_SIZE
|
|
|
|
|
|
|
|
local rightTopChunkX = mFloor((center.x + bottomXOffset) * CHUNK_SIZE_DIVIDER) * CHUNK_SIZE
|
|
|
|
local leftBottomChunkY = mFloor((center.y + bottomYOffset) * CHUNK_SIZE_DIVIDER) * CHUNK_SIZE
|
|
|
|
|
|
|
|
overlapArray[1] = getChunkByXY(map, leftTopChunkX, leftTopChunkY) -- LeftTop
|
|
|
|
if (leftTopChunkX ~= rightTopChunkX) then
|
|
|
|
overlapArray[2] = getChunkByXY(map, rightTopChunkX, leftTopChunkY) -- RightTop
|
|
|
|
end
|
|
|
|
if (leftTopChunkY ~= leftBottomChunkY) then
|
|
|
|
overlapArray[3] = getChunkByXY(map, leftTopChunkX, leftBottomChunkY) -- LeftBottom
|
|
|
|
end
|
|
|
|
if (leftTopChunkX ~= rightTopChunkX) and (leftTopChunkY ~= leftBottomChunkY) then
|
|
|
|
overlapArray[4] = getChunkByXY(map, rightTopChunkX, leftBottomChunkY) -- RightBottom
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return overlapArray
|
2017-11-21 09:27:03 +02:00
|
|
|
end
|
|
|
|
|
2021-02-20 07:41:30 +02:00
|
|
|
local function scanPaths(chunk, map)
|
|
|
|
local surface = map.surface
|
2020-04-28 05:41:18 +02:00
|
|
|
local pass = CHUNK_IMPASSABLE
|
|
|
|
|
|
|
|
local x = chunk.x
|
|
|
|
local y = chunk.y
|
|
|
|
|
2021-02-20 09:31:36 +02:00
|
|
|
local universe = map.universe
|
|
|
|
local filteredEntitiesCliffQuery = universe.filteredEntitiesCliffQuery
|
|
|
|
local filteredTilesPathQuery = universe.filteredTilesPathQuery
|
2020-04-28 05:41:18 +02:00
|
|
|
local count_entities_filtered = surface.count_entities_filtered
|
|
|
|
local count_tiles_filtered = surface.count_tiles_filtered
|
|
|
|
|
|
|
|
local passableNorthSouth = false
|
|
|
|
local passableEastWest = false
|
|
|
|
|
|
|
|
local topPosition = filteredEntitiesCliffQuery.area[1]
|
|
|
|
local bottomPosition = filteredEntitiesCliffQuery.area[2]
|
|
|
|
topPosition[2] = y
|
|
|
|
bottomPosition[2] = y + 32
|
|
|
|
|
|
|
|
for xi=x, x + 32 do
|
|
|
|
topPosition[1] = xi
|
|
|
|
bottomPosition[1] = xi + 1
|
|
|
|
if (count_entities_filtered(filteredEntitiesCliffQuery) == 0) and
|
|
|
|
(count_tiles_filtered(filteredTilesPathQuery) == 0)
|
|
|
|
then
|
|
|
|
passableNorthSouth = true
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
topPosition[1] = x
|
|
|
|
bottomPosition[1] = x + 32
|
|
|
|
|
|
|
|
for yi=y, y + 32 do
|
|
|
|
topPosition[2] = yi
|
|
|
|
bottomPosition[2] = yi + 1
|
|
|
|
if (count_entities_filtered(filteredEntitiesCliffQuery) == 0) and
|
|
|
|
(count_tiles_filtered(filteredTilesPathQuery) == 0)
|
|
|
|
then
|
|
|
|
passableEastWest = true
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
if passableEastWest and passableNorthSouth then
|
|
|
|
pass = CHUNK_ALL_DIRECTIONS
|
|
|
|
elseif passableEastWest then
|
|
|
|
pass = CHUNK_EAST_WEST
|
|
|
|
elseif passableNorthSouth then
|
|
|
|
pass = CHUNK_NORTH_SOUTH
|
|
|
|
end
|
|
|
|
return pass
|
2018-01-02 08:05:21 +02:00
|
|
|
end
|
2017-12-29 07:38:10 +02:00
|
|
|
|
2021-02-20 07:41:30 +02:00
|
|
|
local function scorePlayerBuildings(map)
|
|
|
|
local surface = map.surface
|
2021-02-20 09:31:36 +02:00
|
|
|
local universe = map.universe
|
|
|
|
if surface.count_entities_filtered(universe.hasPlayerStructuresQuery) > 0 then
|
|
|
|
return (surface.count_entities_filtered(universe.filteredEntitiesPlayerQueryLowest) * GENERATOR_PHEROMONE_LEVEL_1) +
|
|
|
|
(surface.count_entities_filtered(universe.filteredEntitiesPlayerQueryLow) * GENERATOR_PHEROMONE_LEVEL_3) +
|
|
|
|
(surface.count_entities_filtered(universe.filteredEntitiesPlayerQueryHigh) * GENERATOR_PHEROMONE_LEVEL_5) +
|
|
|
|
(surface.count_entities_filtered(universe.filteredEntitiesPlayerQueryHighest) * GENERATOR_PHEROMONE_LEVEL_6)
|
2020-05-22 08:57:03 +02:00
|
|
|
end
|
|
|
|
return 0
|
2018-01-02 08:05:21 +02:00
|
|
|
end
|
|
|
|
|
2021-02-20 07:41:30 +02:00
|
|
|
function chunkUtils.initialScan(chunk, map, tick)
|
|
|
|
local surface = map.surface
|
2021-02-20 09:31:36 +02:00
|
|
|
local universe = map.universe
|
|
|
|
local waterTiles = (1 - (surface.count_tiles_filtered(universe.filteredTilesQuery) * 0.0009765625)) * 0.80
|
|
|
|
local enemyBuildings = surface.find_entities_filtered(universe.filteredEntitiesEnemyStructureQuery)
|
2019-02-06 08:25:43 +02:00
|
|
|
|
2020-05-25 05:25:21 +02:00
|
|
|
if (waterTiles >= CHUNK_PASS_THRESHOLD) or (#enemyBuildings > 0) then
|
2021-02-14 06:49:54 +02:00
|
|
|
local neutralObjects = mMax(0,
|
2021-02-20 09:31:36 +02:00
|
|
|
mMin(1 - (surface.count_entities_filtered(universe.filteredEntitiesChunkNeutral) * 0.005),
|
2021-02-14 06:49:54 +02:00
|
|
|
1) * 0.20)
|
2021-02-20 07:41:30 +02:00
|
|
|
local pass = scanPaths(chunk, map)
|
2019-10-19 23:04:38 +02:00
|
|
|
|
2021-02-20 07:41:30 +02:00
|
|
|
local playerObjects = scorePlayerBuildings(map)
|
2017-12-29 07:38:10 +02:00
|
|
|
|
2020-04-28 05:41:18 +02:00
|
|
|
if ((playerObjects > 0) or (#enemyBuildings > 0)) and (pass == CHUNK_IMPASSABLE) then
|
|
|
|
pass = CHUNK_ALL_DIRECTIONS
|
|
|
|
end
|
2019-12-09 05:31:51 +02:00
|
|
|
|
2020-04-28 05:41:18 +02:00
|
|
|
if (pass ~= CHUNK_IMPASSABLE) then
|
2021-02-20 09:31:36 +02:00
|
|
|
local resources = surface.count_entities_filtered(universe.countResourcesQuery) * RESOURCE_NORMALIZER
|
2020-03-16 00:16:00 +02:00
|
|
|
|
2021-02-20 09:31:36 +02:00
|
|
|
local buildingHiveTypeLookup = universe.buildingHiveTypeLookup
|
2020-04-28 05:41:18 +02:00
|
|
|
local counts = map.chunkScanCounts
|
|
|
|
for i=1,#HIVE_BUILDINGS_TYPES do
|
|
|
|
counts[HIVE_BUILDINGS_TYPES[i]] = 0
|
2020-03-16 00:16:00 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
if (#enemyBuildings > 0) then
|
2021-02-21 01:31:48 +02:00
|
|
|
if universe.NEW_ENEMIES then
|
2020-04-28 05:41:18 +02:00
|
|
|
local base = findNearbyBase(map, chunk)
|
2021-12-04 21:04:13 +02:00
|
|
|
if not base then
|
2021-02-20 09:31:36 +02:00
|
|
|
base = createBase(map, chunk, tick)
|
2020-04-28 05:41:18 +02:00
|
|
|
end
|
|
|
|
|
2021-02-20 09:31:36 +02:00
|
|
|
local unitList = surface.find_entities_filtered(universe.filteredEntitiesUnitQuery)
|
2020-04-28 05:41:18 +02:00
|
|
|
for i=1,#unitList do
|
|
|
|
local unit = unitList[i]
|
|
|
|
if (unit.valid) then
|
|
|
|
unit.destroy()
|
2019-02-21 08:31:47 +02:00
|
|
|
end
|
2020-04-28 05:41:18 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
for i = 1, #enemyBuildings do
|
|
|
|
local enemyBuilding = enemyBuildings[i]
|
2021-12-04 21:04:13 +02:00
|
|
|
|
|
|
|
chunkUtils.registerEnemyBaseStructure(map, enemyBuilding, tick, base)
|
2021-02-20 22:44:52 +02:00
|
|
|
if not buildingHiveTypeLookup[enemyBuilding.name] then
|
2021-12-04 21:58:07 +02:00
|
|
|
upgradeEntity(enemyBuilding, base, map, nil, true)
|
2020-04-28 05:41:18 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
else
|
|
|
|
for i=1,#enemyBuildings do
|
|
|
|
local building = enemyBuildings[i]
|
2021-12-04 21:04:13 +02:00
|
|
|
chunkUtils.registerEnemyBaseStructure(map, building, tick)
|
2020-04-28 05:41:18 +02:00
|
|
|
end
|
|
|
|
end
|
2019-02-21 08:31:47 +02:00
|
|
|
end
|
2019-02-06 08:25:43 +02:00
|
|
|
|
2020-04-28 05:41:18 +02:00
|
|
|
setPlayerBaseGenerator(map, chunk, playerObjects)
|
|
|
|
setResourceGenerator(map, chunk, resources)
|
2019-02-06 08:25:43 +02:00
|
|
|
|
2020-04-28 05:41:18 +02:00
|
|
|
setPassable(map, chunk, pass)
|
2020-05-25 05:25:21 +02:00
|
|
|
setPathRating(map, chunk, waterTiles + neutralObjects)
|
2018-01-02 08:05:21 +02:00
|
|
|
|
2020-04-28 05:41:18 +02:00
|
|
|
return chunk
|
|
|
|
end
|
|
|
|
end
|
2017-12-29 07:38:10 +02:00
|
|
|
|
2020-05-15 22:51:38 +02:00
|
|
|
return -1
|
2018-01-02 08:05:21 +02:00
|
|
|
end
|
|
|
|
|
2021-02-20 07:41:30 +02:00
|
|
|
function chunkUtils.chunkPassScan(chunk, map)
|
|
|
|
local surface = map.surface
|
2021-02-20 09:31:36 +02:00
|
|
|
local universe = map.universe
|
|
|
|
local waterTiles = (1 - (surface.count_tiles_filtered(universe.filteredTilesQuery) * 0.0009765625)) * 0.80
|
2018-01-02 08:05:21 +02:00
|
|
|
|
2020-05-25 05:25:21 +02:00
|
|
|
if (waterTiles >= CHUNK_PASS_THRESHOLD) then
|
2021-02-14 06:49:54 +02:00
|
|
|
local neutralObjects = mMax(0,
|
2021-02-20 09:31:36 +02:00
|
|
|
mMin(1 - (surface.count_entities_filtered(universe.filteredEntitiesChunkNeutral) * 0.005),
|
2021-02-14 06:49:54 +02:00
|
|
|
1) * 0.20)
|
2021-02-20 07:41:30 +02:00
|
|
|
local pass = scanPaths(chunk, map)
|
2019-02-06 08:25:43 +02:00
|
|
|
|
2020-04-28 05:41:18 +02:00
|
|
|
local playerObjects = getPlayerBaseGenerator(map, chunk)
|
2018-01-02 08:05:21 +02:00
|
|
|
|
2020-04-28 05:41:18 +02:00
|
|
|
local nests = getNestCount(map, chunk)
|
2018-01-02 08:05:21 +02:00
|
|
|
|
2020-04-28 05:41:18 +02:00
|
|
|
if ((playerObjects > 0) or (nests > 0)) and (pass == CHUNK_IMPASSABLE) then
|
|
|
|
pass = CHUNK_ALL_DIRECTIONS
|
|
|
|
end
|
2018-01-02 08:05:21 +02:00
|
|
|
|
2020-04-28 05:41:18 +02:00
|
|
|
setPassable(map, chunk, pass)
|
2020-05-25 05:25:21 +02:00
|
|
|
setPathRating(map, chunk, waterTiles + neutralObjects)
|
2018-01-02 08:05:21 +02:00
|
|
|
|
2021-12-04 22:56:02 +02:00
|
|
|
if pass == CHUNK_IMPASSABLE then
|
|
|
|
return -1
|
|
|
|
end
|
|
|
|
|
2020-04-28 05:41:18 +02:00
|
|
|
return chunk
|
|
|
|
end
|
2018-01-02 08:05:21 +02:00
|
|
|
|
2020-05-15 22:51:38 +02:00
|
|
|
return -1
|
2018-01-02 08:05:21 +02:00
|
|
|
end
|
|
|
|
|
2021-02-20 07:41:30 +02:00
|
|
|
function chunkUtils.mapScanPlayerChunk(chunk, map)
|
|
|
|
local playerObjects = scorePlayerBuildings(map)
|
2020-05-22 08:57:03 +02:00
|
|
|
setPlayerBaseGenerator(map, chunk, playerObjects)
|
|
|
|
end
|
|
|
|
|
2021-02-20 07:41:30 +02:00
|
|
|
function chunkUtils.mapScanResourceChunk(chunk, map)
|
|
|
|
local surface = map.surface
|
2021-02-20 09:31:36 +02:00
|
|
|
local universe = map.universe
|
|
|
|
local resources = surface.count_entities_filtered(universe.countResourcesQuery) * RESOURCE_NORMALIZER
|
2020-04-28 05:41:18 +02:00
|
|
|
setResourceGenerator(map, chunk, resources)
|
2021-02-20 09:31:36 +02:00
|
|
|
local waterTiles = (1 - (surface.count_tiles_filtered(universe.filteredTilesQuery) * 0.0009765625)) * 0.80
|
2021-02-14 06:49:54 +02:00
|
|
|
local neutralObjects = mMax(0,
|
2021-02-20 09:31:36 +02:00
|
|
|
mMin(1 - (surface.count_entities_filtered(universe.filteredEntitiesChunkNeutral) * 0.005),
|
2021-02-14 06:49:54 +02:00
|
|
|
1) * 0.20)
|
2020-05-25 05:25:21 +02:00
|
|
|
setPathRating(map, chunk, waterTiles + neutralObjects)
|
2020-05-22 08:57:03 +02:00
|
|
|
end
|
|
|
|
|
2021-12-04 22:07:27 +02:00
|
|
|
function chunkUtils.mapScanEnemyChunk(chunk, map, tick)
|
2021-02-20 09:31:36 +02:00
|
|
|
local universe = map.universe
|
2021-12-04 21:04:13 +02:00
|
|
|
local query = universe.filteredEntitiesEnemyStructureQuery
|
|
|
|
setAreaInQuery(query, chunk, CHUNK_SIZE)
|
|
|
|
local buildings = map.surface.find_entities_filtered(query)
|
2020-04-28 05:41:18 +02:00
|
|
|
local counts = map.chunkScanCounts
|
|
|
|
for i=1,#HIVE_BUILDINGS_TYPES do
|
|
|
|
counts[HIVE_BUILDINGS_TYPES[i]] = 0
|
|
|
|
end
|
2021-12-04 23:05:47 +02:00
|
|
|
local base
|
|
|
|
if universe.NEW_ENEMIES then
|
|
|
|
base = findNearbyBase(map, chunk)
|
|
|
|
if not base then
|
|
|
|
base = createBase(map, chunk, tick)
|
|
|
|
end
|
2021-12-04 22:07:27 +02:00
|
|
|
end
|
2020-04-28 05:41:18 +02:00
|
|
|
for i=1,#buildings do
|
|
|
|
local building = buildings[i]
|
|
|
|
|
2021-12-04 22:07:27 +02:00
|
|
|
chunkUtils.registerEnemyBaseStructure(map, building, tick, base)
|
2021-12-04 21:04:13 +02:00
|
|
|
end
|
2016-07-30 00:44:31 +02:00
|
|
|
end
|
|
|
|
|
2019-03-10 21:28:43 +02:00
|
|
|
function chunkUtils.entityForPassScan(map, entity)
|
2020-04-28 05:41:18 +02:00
|
|
|
local overlapArray = getEntityOverlapChunks(map, entity)
|
|
|
|
|
|
|
|
for i=1,#overlapArray do
|
|
|
|
local chunk = overlapArray[i]
|
2020-05-15 22:51:38 +02:00
|
|
|
if (chunk ~= -1) then
|
2020-04-28 05:41:18 +02:00
|
|
|
map.chunkToPassScan[chunk] = true
|
|
|
|
end
|
|
|
|
end
|
2019-03-10 21:28:43 +02:00
|
|
|
end
|
|
|
|
|
2016-08-05 06:47:51 +02:00
|
|
|
function chunkUtils.createChunk(topX, topY)
|
2020-04-28 05:41:18 +02:00
|
|
|
local chunk = {
|
|
|
|
x = topX,
|
2021-11-26 19:52:30 +02:00
|
|
|
y = topY,
|
2021-11-26 20:31:15 +02:00
|
|
|
dOrigin = euclideanDistancePoints(topX, topY, 0, 0)
|
2020-04-28 05:41:18 +02:00
|
|
|
}
|
|
|
|
chunk[BASE_PHEROMONE] = 0
|
|
|
|
chunk[PLAYER_PHEROMONE] = 0
|
|
|
|
chunk[RESOURCE_PHEROMONE] = 0
|
|
|
|
chunk[CHUNK_TICK] = 0
|
|
|
|
|
|
|
|
return chunk
|
2016-07-30 00:44:31 +02:00
|
|
|
end
|
|
|
|
|
2021-12-04 21:04:13 +02:00
|
|
|
function chunkUtils.colorChunk(chunk, surface, color, ttl)
|
2020-05-24 05:47:14 +02:00
|
|
|
local lx = math.floor(chunk.x * CHUNK_SIZE_DIVIDER) * CHUNK_SIZE
|
|
|
|
local ly = math.floor(chunk.y * CHUNK_SIZE_DIVIDER) * CHUNK_SIZE
|
|
|
|
|
2021-02-14 06:49:54 +02:00
|
|
|
rendering.draw_rectangle({
|
2020-05-24 05:47:14 +02:00
|
|
|
color = color or {0.1, 0.3, 0.1, 0.6},
|
|
|
|
width = 32 * 32,
|
|
|
|
filled = true,
|
|
|
|
left_top = {lx, ly},
|
|
|
|
right_bottom = {lx+32, ly+32},
|
|
|
|
surface = surface,
|
2021-12-04 21:04:13 +02:00
|
|
|
time_to_live = ttl or 180,
|
2020-05-24 05:47:14 +02:00
|
|
|
draw_on_ground = true,
|
|
|
|
visible = true
|
|
|
|
})
|
|
|
|
end
|
|
|
|
|
|
|
|
function chunkUtils.colorXY(x, y, surface, color)
|
2020-04-28 05:41:18 +02:00
|
|
|
local lx = math.floor(x * CHUNK_SIZE_DIVIDER) * CHUNK_SIZE
|
|
|
|
local ly = math.floor(y * CHUNK_SIZE_DIVIDER) * CHUNK_SIZE
|
2020-05-24 05:47:14 +02:00
|
|
|
|
2021-02-14 06:49:54 +02:00
|
|
|
rendering.draw_rectangle({
|
2020-05-24 05:47:14 +02:00
|
|
|
color = color or {0.1, 0.3, 0.1, 0.6},
|
|
|
|
width = 32 * 32,
|
|
|
|
filled = true,
|
|
|
|
left_top = {lx, ly},
|
|
|
|
right_bottom = {lx+32, ly+32},
|
|
|
|
surface = surface,
|
|
|
|
time_to_live = 180,
|
|
|
|
draw_on_ground = true,
|
|
|
|
visible = true
|
|
|
|
})
|
2017-12-31 21:36:23 +02:00
|
|
|
end
|
|
|
|
|
2021-12-04 21:04:13 +02:00
|
|
|
function chunkUtils.registerEnemyBaseStructure(map, entity, tick, incomingBase)
|
2020-04-28 05:41:18 +02:00
|
|
|
local entityType = entity.type
|
|
|
|
|
2021-12-04 21:04:13 +02:00
|
|
|
local addFunc
|
|
|
|
local universe = map.universe
|
|
|
|
local hiveTypeLookup = universe.buildingHiveTypeLookup
|
|
|
|
local hiveType = hiveTypeLookup[entity.name]
|
|
|
|
if (hiveType == "spitter-spawner") or (hiveType == "biter-spawner") then
|
|
|
|
map.builtEnemyBuilding = map.builtEnemyBuilding + 1
|
|
|
|
addFunc = addNestCount
|
|
|
|
elseif (hiveType == "turret") then
|
|
|
|
map.builtEnemyBuilding = map.builtEnemyBuilding + 1
|
|
|
|
addFunc = addTurretCount
|
|
|
|
elseif (hiveType == "trap") then
|
|
|
|
addFunc = addTrapCount
|
|
|
|
elseif (hiveType == "utility") then
|
|
|
|
map.builtEnemyBuilding = map.builtEnemyBuilding + 1
|
|
|
|
addFunc = addUtilityCount
|
|
|
|
elseif (hiveType == "hive") then
|
|
|
|
map.builtEnemyBuilding = map.builtEnemyBuilding + 1
|
|
|
|
addFunc = addHiveCount
|
|
|
|
else
|
|
|
|
if (entityType == "turret") then
|
2021-02-20 09:31:36 +02:00
|
|
|
map.builtEnemyBuilding = map.builtEnemyBuilding + 1
|
2021-12-04 21:04:13 +02:00
|
|
|
addFunc = addTurretCount
|
2020-04-28 05:41:18 +02:00
|
|
|
else
|
2021-12-04 21:04:13 +02:00
|
|
|
map.builtEnemyBuilding = map.builtEnemyBuilding + 1
|
|
|
|
addFunc = addNestCount
|
2020-04-28 05:41:18 +02:00
|
|
|
end
|
2021-12-04 21:04:13 +02:00
|
|
|
end
|
2020-04-28 05:41:18 +02:00
|
|
|
|
2021-12-04 21:04:13 +02:00
|
|
|
local entityUnitNumber = entity.unit_number
|
|
|
|
local chunks = getEntityOverlapChunks(map, entity)
|
|
|
|
for i=1,#chunks do
|
|
|
|
local chunk = chunks[i]
|
|
|
|
if (chunk ~= -1) then
|
2021-12-04 23:05:47 +02:00
|
|
|
if universe.NEW_ENEMIES then
|
|
|
|
local base = incomingBase
|
2021-12-04 21:04:13 +02:00
|
|
|
if not base then
|
2021-12-04 23:05:47 +02:00
|
|
|
base = findNearbyBase(map, chunk)
|
|
|
|
if not base then
|
|
|
|
base = createBase(map, chunk, tick)
|
|
|
|
end
|
2021-12-04 21:04:13 +02:00
|
|
|
end
|
2021-12-04 23:05:47 +02:00
|
|
|
setChunkBase(map, chunk, base)
|
2021-12-04 21:04:13 +02:00
|
|
|
end
|
|
|
|
addFunc(map, chunk, entityUnitNumber)
|
|
|
|
if (hiveType == "spitter-spawner") or (hiveType == "biter-spawner") then
|
2021-02-20 07:41:30 +02:00
|
|
|
processNestActiveness(map, chunk)
|
2020-04-28 05:41:18 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2017-11-21 09:27:03 +02:00
|
|
|
end
|
|
|
|
|
2021-04-30 07:24:14 +02:00
|
|
|
function chunkUtils.unregisterEnemyBaseStructure(map, entity, damageType)
|
2020-04-28 05:41:18 +02:00
|
|
|
local entityType = entity.type
|
2021-12-04 21:04:13 +02:00
|
|
|
|
|
|
|
local removeFunc
|
|
|
|
local hiveTypeLookup = map.universe.buildingHiveTypeLookup
|
|
|
|
local hiveType = hiveTypeLookup[entity.name]
|
|
|
|
if (hiveType == "spitter-spawner") or (hiveType == "biter-spawner") then
|
|
|
|
map.lostEnemyBuilding = map.lostEnemyBuilding + 1
|
|
|
|
removeFunc = removeNestCount
|
|
|
|
elseif (hiveType == "turret") then
|
|
|
|
map.lostEnemyBuilding = map.lostEnemyBuilding + 1
|
|
|
|
removeFunc = removeTurretCount
|
|
|
|
elseif (hiveType == "trap") then
|
|
|
|
removeFunc = removeTrapCount
|
|
|
|
elseif (hiveType == "utility") then
|
|
|
|
map.lostEnemyBuilding = map.lostEnemyBuilding + 1
|
|
|
|
removeFunc = removeUtilityCount
|
|
|
|
elseif (hiveType == "hive") then
|
|
|
|
map.lostEnemyBuilding = map.lostEnemyBuilding + 1
|
|
|
|
removeFunc = removeHiveCount
|
|
|
|
else
|
|
|
|
if (entityType == "turret") then
|
2021-02-20 09:31:36 +02:00
|
|
|
map.lostEnemyBuilding = map.lostEnemyBuilding + 1
|
2021-12-04 21:04:13 +02:00
|
|
|
removeFunc = removeTurretCount
|
2020-04-28 05:41:18 +02:00
|
|
|
else
|
2021-12-04 21:04:13 +02:00
|
|
|
hiveType = "biter-spawner"
|
|
|
|
map.lostEnemyBuilding = map.lostEnemyBuilding + 1
|
|
|
|
removeFunc = removeNestCount
|
2020-04-28 05:41:18 +02:00
|
|
|
end
|
2021-12-04 21:04:13 +02:00
|
|
|
end
|
2020-04-28 05:41:18 +02:00
|
|
|
|
2021-12-04 21:04:13 +02:00
|
|
|
local entityUnitNumber = entity.unit_number
|
|
|
|
local usedBases = {}
|
|
|
|
local chunks = getEntityOverlapChunks(map, entity)
|
|
|
|
for i=1,#chunks do
|
|
|
|
local chunk = chunks[i]
|
|
|
|
if (chunk ~= -1) then
|
|
|
|
if (hiveType == "spitter-spawner") or (hiveType == "biter-spawner") then
|
|
|
|
setRaidNestActiveness(map, chunk, 0)
|
|
|
|
setNestActiveness(map, chunk, 0)
|
|
|
|
end
|
|
|
|
removeFunc(map, chunk, entityUnitNumber)
|
2021-12-04 23:05:47 +02:00
|
|
|
if map.universe.NEW_ENEMIES then
|
|
|
|
local base = getChunkBase(map, chunk)
|
|
|
|
if damageType and usedBases[base.id] then
|
|
|
|
usedBases[base.id] = true
|
|
|
|
local damageTypeName = damageType.name
|
|
|
|
base.damagedBy[damageTypeName] = (base.damagedBy[damageTypeName] or 0) + 3
|
|
|
|
base.deathEvents = base.deathEvents + 3
|
|
|
|
end
|
|
|
|
if (getEnemyStructureCount(map, chunk) <= 0) then
|
|
|
|
removeChunkBase(map, chunk, base)
|
|
|
|
end
|
2019-03-10 00:47:35 +02:00
|
|
|
end
|
2020-04-28 05:41:18 +02:00
|
|
|
end
|
|
|
|
end
|
2017-11-21 09:27:03 +02:00
|
|
|
end
|
|
|
|
|
2021-02-20 09:31:36 +02:00
|
|
|
function chunkUtils.accountPlayerEntity(entity, map, addObject, creditNatives)
|
2020-04-28 05:41:18 +02:00
|
|
|
if (BUILDING_PHEROMONES[entity.type] ~= nil) and (entity.force.name ~= "enemy") then
|
2021-04-30 15:15:29 +02:00
|
|
|
local universe = map.universe
|
2020-04-28 05:41:18 +02:00
|
|
|
local entityValue = BUILDING_PHEROMONES[entity.type]
|
|
|
|
local overlapArray = getEntityOverlapChunks(map, entity)
|
|
|
|
if not addObject then
|
|
|
|
if creditNatives then
|
2021-02-20 09:31:36 +02:00
|
|
|
map.destroyPlayerBuildings = map.destroyPlayerBuildings + 1
|
|
|
|
if (map.state == AI_STATE_ONSLAUGHT) then
|
|
|
|
map.points = map.points + entityValue
|
2021-04-30 15:15:29 +02:00
|
|
|
if universe.aiPointsPrintGainsToChat then
|
2021-05-01 19:26:05 +02:00
|
|
|
game.print(map.surface.name .. ": Points: +" .. math.floor(entityValue) .. ". [Structure Kill] Total: " .. string.format("%.2f", map.points))
|
2021-04-30 15:15:29 +02:00
|
|
|
end
|
2020-04-28 05:41:18 +02:00
|
|
|
else
|
2021-02-20 09:31:36 +02:00
|
|
|
map.points = map.points + (entityValue * 0.12)
|
2021-04-30 15:15:29 +02:00
|
|
|
if universe.aiPointsPrintGainsToChat then
|
2021-05-01 19:26:05 +02:00
|
|
|
game.print(map.surface.name .. ": Points: +" .. math.floor(entityValue * 0.12) .. ". [Structure Kill] Total: " .. string.format("%.2f", map.points))
|
2021-04-30 15:15:29 +02:00
|
|
|
end
|
2020-04-28 05:41:18 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
entityValue = -entityValue
|
|
|
|
end
|
|
|
|
|
|
|
|
for i=1,#overlapArray do
|
|
|
|
local chunk = overlapArray[i]
|
2020-05-15 22:51:38 +02:00
|
|
|
if (chunk ~= -1) then
|
2020-04-28 05:41:18 +02:00
|
|
|
addPlayerBaseGenerator(map, chunk, entityValue)
|
2019-03-10 00:47:35 +02:00
|
|
|
end
|
2020-04-28 05:41:18 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
return entity
|
2017-11-21 09:27:03 +02:00
|
|
|
end
|
|
|
|
|
2018-01-14 07:48:21 +02:00
|
|
|
function chunkUtils.unregisterResource(entity, map)
|
2020-04-28 05:41:18 +02:00
|
|
|
if entity.prototype.infinite_resource then
|
|
|
|
return
|
|
|
|
end
|
|
|
|
local overlapArray = getEntityOverlapChunks(map, entity)
|
|
|
|
|
|
|
|
for i=1,#overlapArray do
|
|
|
|
local chunk = overlapArray[i]
|
2020-05-15 22:51:38 +02:00
|
|
|
if (chunk ~= -1) then
|
2020-04-28 05:41:18 +02:00
|
|
|
addResourceGenerator(map, chunk, -RESOURCE_NORMALIZER)
|
|
|
|
end
|
|
|
|
end
|
2018-01-02 08:05:21 +02:00
|
|
|
end
|
|
|
|
|
2019-05-16 07:11:43 +02:00
|
|
|
function chunkUtils.registerResource(entity, map)
|
2020-04-28 05:41:18 +02:00
|
|
|
local overlapArray = getEntityOverlapChunks(map, entity)
|
|
|
|
|
|
|
|
for i=1,#overlapArray do
|
|
|
|
local chunk = overlapArray[i]
|
2020-05-15 22:51:38 +02:00
|
|
|
if (chunk ~= -1) then
|
2020-04-28 05:41:18 +02:00
|
|
|
addResourceGenerator(map, chunk, RESOURCE_NORMALIZER)
|
|
|
|
end
|
|
|
|
end
|
2019-05-16 07:11:43 +02:00
|
|
|
end
|
2019-03-10 00:47:35 +02:00
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
function chunkUtils.makeImmortalEntity(surface, entity)
|
2020-04-28 05:41:18 +02:00
|
|
|
local repairPosition = entity.position
|
|
|
|
local repairName = entity.name
|
|
|
|
local repairForce = entity.force
|
|
|
|
local repairDirection = entity.direction
|
|
|
|
|
|
|
|
local wires
|
|
|
|
if (entity.type == "electric-pole") then
|
|
|
|
wires = entity.neighbours
|
|
|
|
end
|
|
|
|
entity.destroy()
|
|
|
|
local newEntity = surface.create_entity({position=repairPosition,
|
|
|
|
name=repairName,
|
|
|
|
direction=repairDirection,
|
|
|
|
force=repairForce})
|
|
|
|
if wires then
|
|
|
|
for _,v in pairs(wires.copper) do
|
|
|
|
if (v.valid) then
|
|
|
|
newEntity.connect_neighbour(v);
|
|
|
|
end
|
|
|
|
end
|
|
|
|
for _,v in pairs(wires.red) do
|
|
|
|
if (v.valid) then
|
|
|
|
newEntity.connect_neighbour({wire = DEFINES_WIRE_TYPE_RED, target_entity = v});
|
|
|
|
end
|
|
|
|
end
|
|
|
|
for _,v in pairs(wires.green) do
|
|
|
|
if (v.valid) then
|
|
|
|
newEntity.connect_neighbour({wire = DEFINES_WIRE_TYPE_GREEN, target_entity = v});
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
newEntity.destructible = false
|
2017-11-21 09:27:03 +02:00
|
|
|
end
|
|
|
|
|
2019-02-16 06:17:30 +02:00
|
|
|
chunkUtilsG = chunkUtils
|
2016-08-30 06:08:22 +02:00
|
|
|
return chunkUtils
|