mirror of
https://github.com/veden/Rampant.git
synced 2025-01-22 03:08:51 +02:00
177 lines
5.4 KiB
Lua
Executable File
177 lines
5.4 KiB
Lua
Executable File
local movementUtils = {}
|
|
|
|
-- imports
|
|
|
|
local constants = require("Constants")
|
|
local unitGroupUtils = require("UnitGroupUtils")
|
|
local mapUtils = require("MapUtils")
|
|
local mathUtils = require("MathUtils")
|
|
|
|
-- constants
|
|
|
|
local MOVEMENT_PHEROMONE_GENERATOR_AMOUNT = constants.MOVEMENT_PHEROMONE_GENERATOR_AMOUNT
|
|
local MAX_PENALTY_BEFORE_PURGE = constants.MAX_PENALTY_BEFORE_PURGE
|
|
|
|
local MAGIC_MAXIMUM_NUMBER = constants.MAGIC_MAXIMUM_NUMBER
|
|
|
|
local RESOURCE_PHEROMONE = constants.RESOURCE_PHEROMONE
|
|
|
|
local SENTINEL_IMPASSABLE_CHUNK = constants.SENTINEL_IMPASSABLE_CHUNK
|
|
|
|
-- imported functions
|
|
|
|
local canMoveChunkDirection = mapUtils.canMoveChunkDirection
|
|
|
|
local recycleBiters = unitGroupUtils.recycleBiters
|
|
|
|
local tableRemove = table.remove
|
|
local tableInsert = table.insert
|
|
|
|
local distortPosition = mathUtils.distortPosition
|
|
|
|
-- module code
|
|
|
|
function movementUtils.findMovementPosition(surface, position, distort)
|
|
local pos = position
|
|
if not surface.can_place_entity({name="behemoth-biter", position=position}) then
|
|
pos = surface.find_non_colliding_position("behemoth-biter", position, 5, 2)
|
|
end
|
|
return (distort and distortPosition(pos)) or pos
|
|
end
|
|
|
|
function movementUtils.addMovementPenalty(natives, units, chunk)
|
|
local penalties = units.penalties
|
|
for i=1,#penalties do
|
|
local penalty = penalties[i]
|
|
if (penalty.c == chunk) then
|
|
penalty.v = penalty.v + MOVEMENT_PHEROMONE_GENERATOR_AMOUNT
|
|
if (penalty.v > MAX_PENALTY_BEFORE_PURGE) then
|
|
local group = units.group
|
|
if group then
|
|
recycleBiters(natives, group.members)
|
|
group.destroy()
|
|
end
|
|
end
|
|
return
|
|
end
|
|
end
|
|
if (#penalties == 7) then
|
|
tableRemove(penalties, 7)
|
|
end
|
|
tableInsert(penalties,
|
|
1,
|
|
{ v = MOVEMENT_PHEROMONE_GENERATOR_AMOUNT,
|
|
c = chunk })
|
|
end
|
|
|
|
function movementUtils.lookupMovementPenalty(squad, x, y)
|
|
local penalties = squad.penalties
|
|
for i=1,#penalties do
|
|
local penalty = penalties[i]
|
|
if (penalty.x == x) and (penalty.y == y) then
|
|
return penalty.v
|
|
end
|
|
end
|
|
return 0
|
|
end
|
|
|
|
|
|
--[[
|
|
Expects all neighbors adjacent to a chunk
|
|
--]]
|
|
function movementUtils.scoreNeighborsForAttack(chunk, neighborDirectionChunks, scoreFunction, squad)
|
|
local highestChunk = SENTINEL_IMPASSABLE_CHUNK
|
|
local highestScore = -MAGIC_MAXIMUM_NUMBER
|
|
local highestDirection
|
|
for x=1,8 do
|
|
local neighborChunk = neighborDirectionChunks[x]
|
|
if (neighborChunk ~= SENTINEL_IMPASSABLE_CHUNK) and canMoveChunkDirection(x, chunk, neighborChunk) then
|
|
local score = scoreFunction(squad, neighborChunk)
|
|
if (score > highestScore) then
|
|
highestScore = score
|
|
highestChunk = neighborChunk
|
|
highestDirection = x
|
|
end
|
|
end
|
|
end
|
|
|
|
if (chunk ~= SENTINEL_IMPASSABLE_CHUNK) and (scoreFunction(squad, chunk) > highestScore) then
|
|
return SENTINEL_IMPASSABLE_CHUNK, -1
|
|
end
|
|
|
|
return highestChunk, highestDirection
|
|
end
|
|
|
|
--[[
|
|
Expects all neighbors adjacent to a chunk
|
|
--]]
|
|
function movementUtils.scoreNeighborsForResource(chunk, neighborDirectionChunks, scoreFunction, squad, threshold)
|
|
local highestChunk = SENTINEL_IMPASSABLE_CHUNK
|
|
local highestScore = -MAGIC_MAXIMUM_NUMBER
|
|
local highestDirection
|
|
for x=1,8 do
|
|
local neighborChunk = neighborDirectionChunks[x]
|
|
if (neighborChunk ~= SENTINEL_IMPASSABLE_CHUNK) and canMoveChunkDirection(x, chunk, neighborChunk) and (neighborChunk[RESOURCE_PHEROMONE] > (threshold or 1)) then
|
|
local score = scoreFunction(squad, neighborChunk)
|
|
if (score > highestScore) then
|
|
highestScore = score
|
|
highestChunk = neighborChunk
|
|
highestDirection = x
|
|
end
|
|
end
|
|
end
|
|
|
|
if scoreFunction(squad, chunk) > highestScore then
|
|
return SENTINEL_IMPASSABLE_CHUNK, -1
|
|
end
|
|
|
|
return highestChunk, highestDirection
|
|
end
|
|
|
|
--[[
|
|
Expects all neighbors adjacent to a chunk
|
|
--]]
|
|
function movementUtils.scoreNeighborsForRetreat(chunk, neighborDirectionChunks, scoreFunction, map)
|
|
local highestChunk = SENTINEL_IMPASSABLE_CHUNK
|
|
local highestScore = -MAGIC_MAXIMUM_NUMBER
|
|
local highestDirection
|
|
for x=1,8 do
|
|
local neighborChunk = neighborDirectionChunks[x]
|
|
if (neighborChunk ~= SENTINEL_IMPASSABLE_CHUNK) and canMoveChunkDirection(x, chunk, neighborChunk) then
|
|
local score = scoreFunction(map, neighborChunk)
|
|
if (score > highestScore) then
|
|
highestScore = score
|
|
highestChunk = neighborChunk
|
|
highestDirection = x
|
|
end
|
|
end
|
|
end
|
|
|
|
return highestChunk, highestDirection
|
|
end
|
|
|
|
|
|
--[[
|
|
Expects all neighbors adjacent to a chunk
|
|
--]]
|
|
function movementUtils.scoreNeighborsForFormation(neighborChunks, validFunction, scoreFunction, map)
|
|
local highestChunk = SENTINEL_IMPASSABLE_CHUNK
|
|
local highestScore = -MAGIC_MAXIMUM_NUMBER
|
|
local highestDirection
|
|
for x=1,8 do
|
|
local neighborChunk = neighborChunks[x]
|
|
if (neighborChunk ~= SENTINEL_IMPASSABLE_CHUNK) and validFunction(map, neighborChunk) then
|
|
local score = scoreFunction(neighborChunk)
|
|
if (score > highestScore) then
|
|
highestScore = score
|
|
highestChunk = neighborChunk
|
|
highestDirection = x
|
|
end
|
|
end
|
|
end
|
|
|
|
return highestChunk, highestDirection
|
|
end
|
|
|
|
return movementUtils
|