2019-02-16 06:17:30 +02:00
|
|
|
if movementUtilsG then
|
|
|
|
return movementUtilsG
|
|
|
|
end
|
2017-06-13 05:16:43 +02:00
|
|
|
local movementUtils = {}
|
|
|
|
|
|
|
|
-- imports
|
|
|
|
|
|
|
|
local constants = require("Constants")
|
2017-11-21 09:27:03 +02:00
|
|
|
local mapUtils = require("MapUtils")
|
2017-06-13 05:16:43 +02:00
|
|
|
local mathUtils = require("MathUtils")
|
|
|
|
|
|
|
|
-- constants
|
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
local MAGIC_MAXIMUM_NUMBER = constants.MAGIC_MAXIMUM_NUMBER
|
|
|
|
|
2021-12-06 06:27:03 +02:00
|
|
|
local SQUAD_SETTLING = constants.SQUAD_SETTLING
|
|
|
|
|
2017-06-13 05:16:43 +02:00
|
|
|
-- imported functions
|
|
|
|
|
2021-12-06 06:27:03 +02:00
|
|
|
local gaussianRandomRangeRG = mathUtils.gaussianRandomRangeRG
|
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
local canMoveChunkDirection = mapUtils.canMoveChunkDirection
|
2019-10-14 07:49:52 +02:00
|
|
|
local getNeighborChunks = mapUtils.getNeighborChunks
|
2017-11-21 09:27:03 +02:00
|
|
|
|
2017-06-13 05:16:43 +02:00
|
|
|
local tableRemove = table.remove
|
|
|
|
local tableInsert = table.insert
|
|
|
|
|
|
|
|
local distortPosition = mathUtils.distortPosition
|
|
|
|
|
|
|
|
-- module code
|
|
|
|
|
2019-10-19 21:13:48 +02:00
|
|
|
function movementUtils.findMovementPosition(surface, position)
|
2017-06-13 05:16:43 +02:00
|
|
|
local pos = position
|
2020-05-23 06:22:40 +02:00
|
|
|
pos = surface.find_non_colliding_position("chunk-scanner-squad-movement-rampant", pos, 10, 2, false)
|
2019-10-19 21:13:48 +02:00
|
|
|
return pos
|
2017-06-13 05:16:43 +02:00
|
|
|
end
|
|
|
|
|
2019-10-21 02:53:16 +02:00
|
|
|
function movementUtils.findMovementPositionEntity(entityName, surface, position)
|
|
|
|
local pos = position
|
2020-05-23 06:22:40 +02:00
|
|
|
pos = surface.find_non_colliding_position(entityName, pos, 5, 4, true)
|
2019-10-21 02:53:16 +02:00
|
|
|
return pos
|
|
|
|
end
|
|
|
|
|
2019-10-19 21:13:48 +02:00
|
|
|
function movementUtils.findMovementPositionDistort(surface, position)
|
2019-10-14 07:49:52 +02:00
|
|
|
local pos = position
|
2020-05-23 06:22:40 +02:00
|
|
|
pos = surface.find_non_colliding_position("chunk-scanner-squad-movement-rampant", pos, 10, 2, false)
|
2019-10-21 02:53:16 +02:00
|
|
|
return distortPosition(pos, 8)
|
2019-10-14 07:49:52 +02:00
|
|
|
end
|
|
|
|
|
2021-02-14 06:49:54 +02:00
|
|
|
function movementUtils.addMovementPenalty(squad, chunk)
|
2020-05-17 07:06:55 +02:00
|
|
|
if (chunk == -1) then
|
2020-05-15 22:51:38 +02:00
|
|
|
return
|
|
|
|
end
|
2020-05-24 06:17:18 +02:00
|
|
|
local penalties = squad.penalties
|
|
|
|
local penaltyCount = #penalties
|
|
|
|
for i=1,penaltyCount do
|
|
|
|
local penalty = penalties[i]
|
2021-12-05 20:19:04 +02:00
|
|
|
if (penalty.c.id == chunk.id) then
|
2021-04-05 06:46:43 +02:00
|
|
|
penalty.v = penalty.v + 1
|
2021-12-06 06:27:03 +02:00
|
|
|
if (penalty.v > 2) and (penalty.v < 15) then
|
2020-05-24 06:17:18 +02:00
|
|
|
squad.kamikaze = true
|
2021-12-06 06:27:03 +02:00
|
|
|
elseif penalty.v >= 15 then
|
|
|
|
local universe = squad.map.universe
|
|
|
|
if universe.enabledMigration and
|
|
|
|
(universe.builderCount < universe.AI_MAX_BUILDER_COUNT) then
|
|
|
|
squad.settler = true
|
|
|
|
squad.originPosition.x = squad.group.position.x
|
|
|
|
squad.originPosition.y = squad.group.position.y
|
|
|
|
squad.maxDistance = gaussianRandomRangeRG(universe.expansionMaxDistance * 0.5,
|
|
|
|
universe.expansionMaxDistanceDerivation,
|
|
|
|
10,
|
|
|
|
universe.expansionMaxDistance,
|
|
|
|
universe.random)
|
|
|
|
|
|
|
|
squad.status = SQUAD_SETTLING
|
|
|
|
else
|
|
|
|
squad.group.destroy()
|
|
|
|
end
|
2020-05-24 06:17:18 +02:00
|
|
|
end
|
|
|
|
return
|
|
|
|
end
|
|
|
|
end
|
2021-04-05 06:46:43 +02:00
|
|
|
if (penaltyCount == 10) then
|
|
|
|
tableRemove(penalties, 10)
|
2020-05-24 06:17:18 +02:00
|
|
|
end
|
|
|
|
tableInsert(penalties,
|
|
|
|
1,
|
|
|
|
{ v = 1,
|
|
|
|
c = chunk })
|
2017-06-13 05:16:43 +02:00
|
|
|
end
|
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
--[[
|
|
|
|
Expects all neighbors adjacent to a chunk
|
|
|
|
--]]
|
2021-02-14 06:49:54 +02:00
|
|
|
function movementUtils.scoreNeighborsForAttack(map, chunk, neighborDirectionChunks, scoreFunction)
|
2020-05-15 22:51:38 +02:00
|
|
|
local highestChunk = -1
|
2017-11-21 09:27:03 +02:00
|
|
|
local highestScore = -MAGIC_MAXIMUM_NUMBER
|
2019-02-06 08:25:43 +02:00
|
|
|
local highestDirection
|
2019-03-09 08:23:00 +02:00
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
for x=1,8 do
|
|
|
|
local neighborChunk = neighborDirectionChunks[x]
|
2020-05-15 22:51:38 +02:00
|
|
|
if (neighborChunk ~= -1) then
|
2021-12-05 20:19:04 +02:00
|
|
|
if (chunk == -1) or canMoveChunkDirection(map, x, chunk, neighborChunk) then
|
2021-02-14 06:49:54 +02:00
|
|
|
local score = scoreFunction(map, neighborChunk)
|
2020-05-15 22:51:38 +02:00
|
|
|
if (score > highestScore) then
|
|
|
|
highestScore = score
|
|
|
|
highestChunk = neighborChunk
|
|
|
|
highestDirection = x
|
|
|
|
end
|
2017-11-21 09:27:03 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2019-02-06 08:25:43 +02:00
|
|
|
|
2020-05-23 06:22:40 +02:00
|
|
|
local nextHighestChunk = -1
|
|
|
|
local nextHighestScore = highestScore
|
|
|
|
local nextHighestDirection
|
|
|
|
|
2020-05-15 22:51:38 +02:00
|
|
|
if (highestChunk ~= -1) then
|
2019-10-14 07:49:52 +02:00
|
|
|
neighborDirectionChunks = getNeighborChunks(map, highestChunk.x, highestChunk.y)
|
|
|
|
for x=1,8 do
|
|
|
|
local neighborChunk = neighborDirectionChunks[x]
|
2021-12-05 20:19:04 +02:00
|
|
|
if ((neighborChunk ~= -1) and ((chunk == -1) or (neighborChunk.id ~= chunk.id)) and
|
2019-10-19 21:13:48 +02:00
|
|
|
canMoveChunkDirection(map, x, highestChunk, neighborChunk)) then
|
2021-02-14 06:49:54 +02:00
|
|
|
local score = scoreFunction(map, neighborChunk)
|
2019-10-14 07:49:52 +02:00
|
|
|
if (score > nextHighestScore) then
|
|
|
|
nextHighestScore = score
|
|
|
|
nextHighestChunk = neighborChunk
|
|
|
|
nextHighestDirection = x
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
return highestChunk, highestDirection, nextHighestChunk, nextHighestDirection
|
2017-11-21 09:27:03 +02:00
|
|
|
end
|
|
|
|
|
2018-09-24 06:56:45 +02:00
|
|
|
|
|
|
|
--[[
|
|
|
|
Expects all neighbors adjacent to a chunk
|
|
|
|
--]]
|
2021-02-14 06:49:54 +02:00
|
|
|
function movementUtils.scoreNeighborsForSettling(map, chunk, neighborDirectionChunks, scoreFunction)
|
2020-05-15 22:51:38 +02:00
|
|
|
local highestChunk = -1
|
2018-09-24 06:56:45 +02:00
|
|
|
local highestScore = -MAGIC_MAXIMUM_NUMBER
|
2019-12-09 05:31:51 +02:00
|
|
|
local highestDirection = 0
|
2019-10-20 22:45:43 +02:00
|
|
|
|
2018-09-24 06:56:45 +02:00
|
|
|
for x=1,8 do
|
|
|
|
local neighborChunk = neighborDirectionChunks[x]
|
2020-05-15 22:51:38 +02:00
|
|
|
if (neighborChunk ~= -1) then
|
2021-12-05 20:19:04 +02:00
|
|
|
if (chunk == -1) or canMoveChunkDirection(map, x, chunk, neighborChunk) then
|
2021-02-14 06:49:54 +02:00
|
|
|
local score = scoreFunction(map, neighborChunk)
|
2020-05-15 22:51:38 +02:00
|
|
|
if (score > highestScore) then
|
|
|
|
highestScore = score
|
|
|
|
highestChunk = neighborChunk
|
|
|
|
highestDirection = x
|
|
|
|
end
|
2018-09-24 06:56:45 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-02-14 06:49:54 +02:00
|
|
|
if (chunk ~= -1) and (scoreFunction(map, chunk) > highestScore) then
|
2020-05-15 22:51:38 +02:00
|
|
|
return chunk, 0, -1, 0
|
2018-09-24 06:56:45 +02:00
|
|
|
end
|
2019-02-06 08:25:43 +02:00
|
|
|
|
2020-05-15 22:51:38 +02:00
|
|
|
local nextHighestChunk = -1
|
2020-05-23 06:22:40 +02:00
|
|
|
local nextHighestScore = highestScore
|
2019-12-09 05:31:51 +02:00
|
|
|
local nextHighestDirection = 0
|
2019-10-20 22:45:43 +02:00
|
|
|
|
2020-05-15 22:51:38 +02:00
|
|
|
if (highestChunk ~= -1) then
|
2019-10-19 21:13:48 +02:00
|
|
|
neighborDirectionChunks = getNeighborChunks(map, highestChunk.x, highestChunk.y)
|
|
|
|
for x=1,8 do
|
|
|
|
local neighborChunk = neighborDirectionChunks[x]
|
2021-12-05 20:19:04 +02:00
|
|
|
if ((neighborChunk ~= -1) and ((chunk == -1) or (neighborChunk.id ~= chunk.id)) and
|
2019-10-19 21:13:48 +02:00
|
|
|
canMoveChunkDirection(map, x, highestChunk, neighborChunk)) then
|
2021-02-14 06:49:54 +02:00
|
|
|
local score = scoreFunction(map, neighborChunk)
|
2019-10-19 21:13:48 +02:00
|
|
|
if (score > nextHighestScore) then
|
|
|
|
nextHighestScore = score
|
|
|
|
nextHighestChunk = neighborChunk
|
|
|
|
nextHighestDirection = x
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
return highestChunk, highestDirection, nextHighestChunk, nextHighestDirection
|
2018-09-24 06:56:45 +02:00
|
|
|
end
|
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
--[[
|
|
|
|
Expects all neighbors adjacent to a chunk
|
|
|
|
--]]
|
2019-02-06 08:25:43 +02:00
|
|
|
function movementUtils.scoreNeighborsForResource(chunk, neighborDirectionChunks, validFunction, scoreFunction, map)
|
2020-05-15 22:51:38 +02:00
|
|
|
local highestChunk = -1
|
2018-02-14 10:28:42 +02:00
|
|
|
local highestScore = -MAGIC_MAXIMUM_NUMBER
|
2019-02-06 08:25:43 +02:00
|
|
|
local highestDirection
|
2018-02-14 10:28:42 +02:00
|
|
|
for x=1,8 do
|
|
|
|
local neighborChunk = neighborDirectionChunks[x]
|
2021-02-14 06:49:54 +02:00
|
|
|
if (neighborChunk ~= -1) and
|
|
|
|
canMoveChunkDirection(map, x, chunk, neighborChunk) and
|
|
|
|
validFunction(map, chunk, neighborChunk)
|
|
|
|
then
|
2020-05-24 05:47:14 +02:00
|
|
|
local score = scoreFunction(map, neighborChunk)
|
2018-02-14 10:28:42 +02:00
|
|
|
if (score > highestScore) then
|
|
|
|
highestScore = score
|
|
|
|
highestChunk = neighborChunk
|
|
|
|
highestDirection = x
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-05-26 02:05:01 +02:00
|
|
|
if (chunk ~= -1) and (scoreFunction(map, chunk) > highestScore) then
|
2020-05-15 22:51:38 +02:00
|
|
|
return -1, -1
|
2018-02-14 10:28:42 +02:00
|
|
|
end
|
2019-02-06 08:25:43 +02:00
|
|
|
|
2018-02-14 10:28:42 +02:00
|
|
|
return highestChunk, highestDirection
|
|
|
|
end
|
2017-11-21 09:27:03 +02:00
|
|
|
|
|
|
|
--[[
|
|
|
|
Expects all neighbors adjacent to a chunk
|
|
|
|
--]]
|
2019-02-06 08:25:43 +02:00
|
|
|
function movementUtils.scoreNeighborsForRetreat(chunk, neighborDirectionChunks, scoreFunction, map)
|
2020-05-15 22:51:38 +02:00
|
|
|
local highestChunk = -1
|
2017-11-21 09:27:03 +02:00
|
|
|
local highestScore = -MAGIC_MAXIMUM_NUMBER
|
2019-02-06 08:25:43 +02:00
|
|
|
local highestDirection
|
2019-10-20 22:45:43 +02:00
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
for x=1,8 do
|
|
|
|
local neighborChunk = neighborDirectionChunks[x]
|
2020-05-15 22:51:38 +02:00
|
|
|
if (neighborChunk ~= -1) then
|
2021-12-05 20:19:04 +02:00
|
|
|
if (chunk == -1) or canMoveChunkDirection(map, x, chunk, neighborChunk) then
|
2020-05-15 22:51:38 +02:00
|
|
|
local score = scoreFunction(map, neighborChunk)
|
|
|
|
if (score > highestScore) then
|
|
|
|
highestScore = score
|
|
|
|
highestChunk = neighborChunk
|
|
|
|
highestDirection = x
|
|
|
|
end
|
2017-11-21 09:27:03 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2019-02-06 08:25:43 +02:00
|
|
|
|
2020-05-23 06:22:40 +02:00
|
|
|
local nextHighestChunk = -1
|
|
|
|
local nextHighestScore = highestScore
|
|
|
|
local nextHighestDirection
|
|
|
|
|
2020-05-15 22:51:38 +02:00
|
|
|
if (highestChunk ~= -1) then
|
2019-10-20 22:45:43 +02:00
|
|
|
neighborDirectionChunks = getNeighborChunks(map, highestChunk.x, highestChunk.y)
|
|
|
|
for x=1,8 do
|
|
|
|
local neighborChunk = neighborDirectionChunks[x]
|
|
|
|
|
2021-12-05 20:19:04 +02:00
|
|
|
if ((neighborChunk ~= -1) and ((chunk == -1) or (neighborChunk.id ~= chunk.id)) and
|
2019-10-20 22:45:43 +02:00
|
|
|
canMoveChunkDirection(map, x, highestChunk, neighborChunk)) then
|
|
|
|
local score = scoreFunction(map, neighborChunk)
|
|
|
|
if (score > nextHighestScore) then
|
|
|
|
nextHighestScore = score
|
|
|
|
nextHighestChunk = neighborChunk
|
|
|
|
nextHighestDirection = x
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
if (nextHighestChunk == nil) then
|
2020-05-15 22:51:38 +02:00
|
|
|
nextHighestChunk = -1
|
2019-10-20 22:45:43 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
return highestChunk, highestDirection, nextHighestChunk, nextHighestDirection
|
2017-11-21 09:27:03 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
--[[
|
|
|
|
Expects all neighbors adjacent to a chunk
|
|
|
|
--]]
|
2019-02-06 08:25:43 +02:00
|
|
|
function movementUtils.scoreNeighborsForFormation(neighborChunks, validFunction, scoreFunction, map)
|
2020-05-15 22:51:38 +02:00
|
|
|
local highestChunk = -1
|
2017-11-21 09:27:03 +02:00
|
|
|
local highestScore = -MAGIC_MAXIMUM_NUMBER
|
|
|
|
local highestDirection
|
|
|
|
for x=1,8 do
|
|
|
|
local neighborChunk = neighborChunks[x]
|
2020-05-15 22:51:38 +02:00
|
|
|
if (neighborChunk ~= -1) and validFunction(map, neighborChunk) then
|
2020-05-24 05:47:14 +02:00
|
|
|
local score = scoreFunction(map, neighborChunk)
|
2017-11-21 09:27:03 +02:00
|
|
|
if (score > highestScore) then
|
|
|
|
highestScore = score
|
|
|
|
highestChunk = neighborChunk
|
2019-10-14 07:49:52 +02:00
|
|
|
highestDirection = x
|
2017-11-21 09:27:03 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
return highestChunk, highestDirection
|
|
|
|
end
|
|
|
|
|
2019-02-16 06:17:30 +02:00
|
|
|
movementUtilsG = movementUtils
|
2017-06-13 05:16:43 +02:00
|
|
|
return movementUtils
|