1
0
mirror of https://github.com/veden/Rampant.git synced 2025-01-24 03:16:25 +02:00
Rampant/libs/AIPlanning.lua

414 lines
15 KiB
Lua
Raw Normal View History

2019-02-15 20:17:30 -08:00
if aiPlanningG then
return aiPlanningG
end
2016-10-14 17:00:18 -07:00
local aiPlanning = {}
-- imports
local constants = require("Constants")
local mathUtils = require("MathUtils")
-- local aiPredicates = require("AIPredicates")
2017-05-31 18:46:53 -07:00
2016-10-14 17:00:18 -07:00
-- constants
2017-05-31 18:46:53 -07:00
local NO_RETREAT_BASE_PERCENT = constants.NO_RETREAT_BASE_PERCENT
local NO_RETREAT_EVOLUTION_BONUS_MAX = constants.NO_RETREAT_EVOLUTION_BONUS_MAX
2016-10-14 17:00:18 -07:00
local AI_STATE_PEACEFUL = constants.AI_STATE_PEACEFUL
local AI_STATE_AGGRESSIVE = constants.AI_STATE_AGGRESSIVE
local AI_STATE_RAIDING = constants.AI_STATE_RAIDING
local AI_STATE_MIGRATING = constants.AI_STATE_MIGRATING
local AI_STATE_ONSLAUGHT = constants.AI_STATE_ONSLAUGHT
local AI_STATE_SIEGE = constants.AI_STATE_SIEGE
2017-05-31 18:46:53 -07:00
local AI_UNIT_REFUND = constants.AI_UNIT_REFUND
2016-10-14 17:00:18 -07:00
local AI_MAX_POINTS = constants.AI_MAX_POINTS
local AI_POINT_GENERATOR_AMOUNT = constants.AI_POINT_GENERATOR_AMOUNT
local AI_MIN_STATE_DURATION = constants.AI_MIN_STATE_DURATION
local AI_MAX_STATE_DURATION = constants.AI_MAX_STATE_DURATION
2017-05-31 18:46:53 -07:00
local BASE_RALLY_CHANCE = constants.BASE_RALLY_CHANCE
local BONUS_RALLY_CHANCE = constants.BONUS_RALLY_CHANCE
2018-02-18 16:21:18 -08:00
local RETREAT_MOVEMENT_PHEROMONE_LEVEL_MIN = constants.RETREAT_MOVEMENT_PHEROMONE_LEVEL_MIN
local RETREAT_MOVEMENT_PHEROMONE_LEVEL_MAX = constants.RETREAT_MOVEMENT_PHEROMONE_LEVEL_MAX
2020-04-02 20:47:26 -07:00
local MINIMUM_AI_POINTS = constants.MINIMUM_AI_POINTS
2017-05-31 18:46:53 -07:00
2016-10-14 17:00:18 -07:00
-- imported functions
local randomTickEvent = mathUtils.randomTickEvent
local linearInterpolation = mathUtils.linearInterpolation
local mFloor = math.floor
local mRandom = math.random
2016-10-14 17:00:18 -07:00
local mMax = math.max
local mMin = math.min
2016-10-14 17:00:18 -07:00
-- module code
2021-02-19 23:31:36 -08:00
function aiPlanning.planning(map, evolution_factor, tick)
local universe = map.universe
map.evolutionLevel = evolution_factor
universe.evolutionLevel = evolution_factor
2019-12-15 17:16:56 -08:00
2020-04-02 20:47:26 -07:00
local maxPoints = mMax(AI_MAX_POINTS * evolution_factor, MINIMUM_AI_POINTS)
2021-04-29 22:24:14 -07:00
universe.maxPoints = maxPoints
2017-05-31 18:46:53 -07:00
2021-02-19 23:31:36 -08:00
if not universe.ranIncompatibleMessage and universe.newEnemies and
2021-02-13 20:49:54 -08:00
(game.active_mods["bobenemies"] or game.active_mods["Natural_Evolution_Enemies"]) then
2021-02-19 23:31:36 -08:00
universe.ranIncompatibleMessage = true
2020-04-27 20:41:18 -07:00
game.print({"description.rampant-bobs-nee-newEnemies"})
2019-12-06 21:57:20 -08:00
end
2019-10-19 12:13:48 -07:00
local maxOverflowPoints = maxPoints * 3
2017-05-31 18:46:53 -07:00
2021-02-19 23:31:36 -08:00
local attackWaveMaxSize = universe.attackWaveMaxSize
universe.retreatThreshold = linearInterpolation(evolution_factor,
RETREAT_MOVEMENT_PHEROMONE_LEVEL_MIN,
RETREAT_MOVEMENT_PHEROMONE_LEVEL_MAX)
2021-02-19 23:31:36 -08:00
universe.rallyThreshold = BASE_RALLY_CHANCE + (evolution_factor * BONUS_RALLY_CHANCE)
2021-04-16 13:47:43 -07:00
universe.formSquadThreshold = mMax((0.35 * evolution_factor), 0.1)
2021-02-19 23:31:36 -08:00
universe.attackWaveSize = attackWaveMaxSize * (evolution_factor ^ 1.4)
universe.attackWaveDeviation = (universe.attackWaveSize * 0.333)
universe.attackWaveUpperBound = universe.attackWaveSize + (universe.attackWaveSize * 0.35)
if (universe.attackWaveSize < 1) then
universe.attackWaveSize = 2
universe.attackWaveDeviation = 1
universe.attackWaveUpperBound = 3
2020-04-02 20:47:26 -07:00
end
2019-02-02 22:01:28 -08:00
2021-02-19 23:31:36 -08:00
universe.settlerWaveSize = linearInterpolation(evolution_factor ^ 1.66667,
universe.expansionMinSize,
universe.expansionMaxSize)
2021-02-19 23:31:36 -08:00
universe.settlerWaveDeviation = (universe.settlerWaveSize * 0.33)
2019-02-05 22:25:43 -08:00
2021-02-19 23:31:36 -08:00
universe.settlerCooldown = mFloor(linearInterpolation(evolution_factor ^ 1.66667,
universe.expansionMaxTime,
universe.expansionMinTime))
2019-02-02 22:01:28 -08:00
2021-02-19 23:31:36 -08:00
universe.unitRefundAmount = AI_UNIT_REFUND * evolution_factor
universe.kamikazeThreshold = NO_RETREAT_BASE_PERCENT + (evolution_factor * NO_RETREAT_EVOLUTION_BONUS_MAX)
2021-04-29 22:24:14 -07:00
local points = ((AI_POINT_GENERATOR_AMOUNT * mRandom()) + (map.activeNests * 0.003) +
(AI_POINT_GENERATOR_AMOUNT * mMax(evolution_factor ^ 2.5, 0.1)))
if (map.temperament < 0.05) or (map.temperament > 0.95) then
points = points + 0.3
elseif (map.temperament < 0.25) or (map.temperament > 0.75) then
points = points + 0.2
elseif (map.temperament < 0.40) or (map.temperament > 0.60) then
points = points + 0.1
end
2019-02-02 22:01:28 -08:00
2021-02-19 23:31:36 -08:00
if (map.state == AI_STATE_ONSLAUGHT) then
points = points * 2
end
2021-04-29 22:24:14 -07:00
points = points * universe.aiPointsScaler
2021-04-30 21:25:55 -07:00
map.baseIncrement = points * 3.5
2019-02-02 22:01:28 -08:00
2021-02-19 23:31:36 -08:00
local currentPoints = map.points
2019-12-15 17:16:56 -08:00
2019-10-19 12:13:48 -07:00
if (currentPoints < maxPoints) then
2021-02-19 23:31:36 -08:00
map.points = currentPoints + points
2019-10-19 12:13:48 -07:00
end
2019-10-19 12:13:48 -07:00
if (currentPoints > maxOverflowPoints) then
2021-02-19 23:31:36 -08:00
map.points = maxOverflowPoints
2016-10-14 17:00:18 -07:00
end
2019-02-02 22:01:28 -08:00
2021-02-19 23:31:36 -08:00
if (map.stateTick <= tick) then
local roll = mRandom()
2021-02-19 23:31:36 -08:00
if (map.temperament < 0.05) then -- 0 - 0.05
if universe.enabledMigration then
2021-03-28 22:00:49 -07:00
if (roll < 0.7) and universe.siegeAIToggle then
map.state = AI_STATE_SIEGE
else
map.state = AI_STATE_MIGRATING
end
else
2021-02-19 23:31:36 -08:00
if universe.raidAIToggle then
2019-12-15 17:16:56 -08:00
if (roll < 0.85) then
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_AGGRESSIVE
2019-12-15 17:16:56 -08:00
else
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_RAIDING
2019-12-15 17:16:56 -08:00
end
else
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_AGGRESSIVE
end
end
2021-02-19 23:31:36 -08:00
elseif (map.temperament < 0.20) then -- 0.05 - 0.2
if (universe.enabledMigration) then
2021-03-28 22:00:49 -07:00
if (roll < 0.4) and universe.siegeAIToggle then
map.state = AI_STATE_SIEGE
2019-12-15 17:16:56 -08:00
else
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_MIGRATING
2019-12-15 17:16:56 -08:00
end
else
2021-02-19 23:31:36 -08:00
if universe.raidAIToggle then
2019-12-15 17:16:56 -08:00
if (roll < 0.95) then
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_AGGRESSIVE
2019-12-15 17:16:56 -08:00
else
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_RAIDING
2019-12-15 17:16:56 -08:00
end
else
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_AGGRESSIVE
2019-12-15 17:16:56 -08:00
end
end
2021-02-19 23:31:36 -08:00
elseif (map.temperament < 0.4) then -- 0.2 - 0.4
if (universe.enabledMigration) then
2019-12-15 17:16:56 -08:00
if (roll < 0.2) then
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_AGGRESSIVE
2019-12-15 17:16:56 -08:00
elseif (roll < 0.8) then
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_MIGRATING
2021-04-16 13:47:43 -07:00
elseif universe.peacefulAIToggle then
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_PEACEFUL
2021-04-16 13:47:43 -07:00
else
map.state = AI_STATE_MIGRATING
2019-12-15 17:16:56 -08:00
end
else
2019-12-15 17:16:56 -08:00
if (roll < 0.6) then
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_AGGRESSIVE
2021-04-16 13:47:43 -07:00
elseif universe.peacefulAIToggle then
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_PEACEFUL
2021-04-16 13:47:43 -07:00
else
map.state = AI_STATE_AGGRESSIVE
2019-12-15 17:16:56 -08:00
end
end
2021-02-19 23:31:36 -08:00
elseif (map.temperament < 0.6) then -- 0.4 - 0.6
if (roll < 0.4) then
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_AGGRESSIVE
2021-04-16 13:47:43 -07:00
elseif universe.peacefulAIToggle then
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_PEACEFUL
2021-04-16 13:47:43 -07:00
else
if universe.enabledMigration then
map.state = AI_STATE_MIGRATING
else
map.state = AI_STATE_AGGRESSIVE
end
2019-12-15 17:16:56 -08:00
end
2021-02-19 23:31:36 -08:00
elseif (map.temperament < 0.8) then -- 0.6 - 0.8
2021-03-28 22:00:49 -07:00
if (roll < 0.4) then
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_AGGRESSIVE
2021-03-28 22:00:49 -07:00
elseif (roll < 0.6) then
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_ONSLAUGHT
2021-03-28 22:00:49 -07:00
elseif (roll < 0.8) then
map.state = AI_STATE_RAIDING
2021-04-16 13:47:43 -07:00
elseif universe.peacefulAIToggle then
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_PEACEFUL
2021-04-16 13:47:43 -07:00
else
map.state = AI_STATE_AGGRESSIVE
2019-12-15 17:16:56 -08:00
end
2021-03-28 22:00:49 -07:00
elseif (map.temperament < 0.95) then -- 0.8 - 0.95
2021-02-19 23:31:36 -08:00
if (universe.enabledMigration and universe.raidAIToggle) then
2021-03-28 22:00:49 -07:00
if (roll < 0.15) and universe.siegeAIToggle then
map.state = AI_STATE_SIEGE
elseif (roll < 0.4) then
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_RAIDING
2021-03-28 22:00:49 -07:00
elseif (roll < 0.8) then
map.state = AI_STATE_ONSLAUGHT
2019-12-15 17:16:56 -08:00
else
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_AGGRESSIVE
2019-12-15 17:16:56 -08:00
end
2021-02-19 23:31:36 -08:00
elseif (universe.enabledMigration) then
2021-03-28 22:00:49 -07:00
if (roll < 0.15) and universe.siegeAIToggle then
map.state = AI_STATE_SIEGE
2019-12-15 17:16:56 -08:00
elseif (roll < 0.7) then
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_ONSLAUGHT
2019-12-15 17:16:56 -08:00
else
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_AGGRESSIVE
2019-12-15 17:16:56 -08:00
end
2021-02-19 23:31:36 -08:00
elseif (universe.raidAIToggle) then
2019-12-15 17:16:56 -08:00
if (roll < 0.4) then
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_ONSLAUGHT
2019-12-15 17:16:56 -08:00
elseif (roll < 0.7) then
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_RAIDING
2019-12-15 17:16:56 -08:00
else
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_AGGRESSIVE
2019-12-15 17:16:56 -08:00
end
else
if (roll < 0.6) then
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_ONSLAUGHT
2019-12-15 17:16:56 -08:00
else
2021-02-19 23:31:36 -08:00
map.state = AI_STATE_AGGRESSIVE
2019-12-15 17:16:56 -08:00
end
2019-10-19 12:13:48 -07:00
end
2021-03-28 22:00:49 -07:00
else
if (universe.enabledMigration and universe.raidAIToggle) then
2021-04-29 22:24:14 -07:00
if (roll < 0.25) and universe.siegeAIToggle then
2021-03-28 22:00:49 -07:00
map.state = AI_STATE_SIEGE
elseif (roll < 0.6) then
map.state = AI_STATE_RAIDING
else
map.state = AI_STATE_ONSLAUGHT
end
elseif (universe.enabledMigration) then
2021-04-29 22:24:14 -07:00
if (roll < 0.25) and universe.siegeAIToggle then
2021-03-28 22:00:49 -07:00
map.state = AI_STATE_SIEGE
else
map.state = AI_STATE_ONSLAUGHT
end
elseif (universe.raidAIToggle) then
if (roll < 0.4) then
map.state = AI_STATE_ONSLAUGHT
else
map.state = AI_STATE_RAIDING
end
else
map.state = AI_STATE_ONSLAUGHT
end
2019-10-19 12:13:48 -07:00
end
2021-02-19 23:31:36 -08:00
map.destroyPlayerBuildings = 0
map.lostEnemyUnits = 0
map.lostEnemyBuilding = 0
map.rocketLaunched = 0
map.builtEnemyBuilding = 0
map.ionCannonBlasts = 0
map.artilleryBlasts = 0
2021-02-08 21:55:16 -08:00
2021-02-19 23:31:36 -08:00
map.stateTick = randomTickEvent(tick, AI_MIN_STATE_DURATION, AI_MAX_STATE_DURATION)
if universe.printAIStateChanges then
game.print(map.surface.name .. ": AI is now: " .. constants.stateEnglish[map.state] .. ", Next state change is in " .. string.format("%.2f", (map.stateTick - tick) / (60*60)) .. " minutes @ " .. getTimeStringFromTick(map.stateTick) .. " playtime")
end
end
2016-10-14 17:00:18 -07:00
end
function getTimeStringFromTick(tick)
local tickToSeconds = tick / 60
2021-05-01 10:26:05 -07:00
local days = math.floor(tickToSeconds / 86400)
local hours = math.floor((tickToSeconds % 86400) / 3600)
local minutes = math.floor((tickToSeconds % 3600) / 60)
local seconds = math.floor(tickToSeconds % 60)
return days .. "d " .. hours .. "h " .. minutes .. "m " .. seconds .. "s"
end
2021-02-19 23:31:36 -08:00
function aiPlanning.temperamentPlanner(map)
local destroyPlayerBuildings = map.destroyPlayerBuildings
local lostEnemyUnits = map.lostEnemyUnits
local lostEnemyBuilding = map.lostEnemyBuilding
local rocketLaunched = map.rocketLaunched
local builtEnemyBuilding = map.builtEnemyBuilding
local ionCannonBlasts = map.ionCannonBlasts
local artilleryBlasts = map.artilleryBlasts
local activeNests = map.activeNests
local activeRaidNests = map.activeRaidNests
2021-02-19 23:31:36 -08:00
local currentTemperament = map.temperamentScore
local delta = 0
if activeNests > 0 then
2021-03-28 22:00:49 -07:00
local val = (0.03 * activeNests)
delta = delta + val
else
delta = delta - 0.014463
2021-02-08 21:55:16 -08:00
end
2021-02-08 21:55:16 -08:00
if destroyPlayerBuildings > 0 then
if currentTemperament > 0 then
delta = delta - (0.014463 * destroyPlayerBuildings)
2021-02-08 21:55:16 -08:00
else
delta = delta + (0.014463 * destroyPlayerBuildings)
end
end
if activeRaidNests > 0 then
2021-04-29 22:24:14 -07:00
local val = (0.0006 * activeRaidNests)
delta = delta - val
else
2021-04-29 22:24:14 -07:00
delta = delta - 0.01
end
if lostEnemyUnits > 0 then
2021-02-08 21:55:16 -08:00
local multipler
2021-02-19 23:31:36 -08:00
if map.evolutionLevel < 0.3 then
multipler = 0.000434
2021-02-19 23:31:36 -08:00
elseif map.evolutionLevel < 0.5 then
multipler = 0.000217
2021-02-19 23:31:36 -08:00
elseif map.evolutionLevel < 0.7 then
multipler = 0.000108
2021-02-19 23:31:36 -08:00
elseif map.evolutionLevel < 0.9 then
multipler = 0.000054
2021-02-19 23:31:36 -08:00
elseif map.evolutionLevel < 0.9 then
multipler = 0.000027
2021-02-28 11:28:48 -08:00
else
multipler = 0.0000135
2021-02-08 21:55:16 -08:00
end
local val = (multipler * lostEnemyUnits)
if (currentTemperament > 0) then
delta = delta - val
else
delta = delta + val
end
end
if lostEnemyBuilding > 0 then
local val = (0.072317 * lostEnemyBuilding)
2021-02-08 21:55:16 -08:00
if (currentTemperament > 0) then
delta = delta - val
else
delta = delta + val
2021-02-13 20:49:54 -08:00
end
end
if (builtEnemyBuilding > 0) then
local val = (0.004339 * builtEnemyBuilding)
if (currentTemperament > 0) then
delta = delta - val
else
delta = delta + val
end
else
delta = delta - 0.007232
end
if (rocketLaunched > 0) then
local val = (0.289268 * rocketLaunched)
delta = delta + val
end
if (ionCannonBlasts > 0) then
local val = (0.144634 * ionCannonBlasts)
delta = delta + val
end
if (artilleryBlasts > 0) then
local val = (0.144634 * artilleryBlasts)
delta = delta + val
end
2021-04-16 13:47:43 -07:00
local universe = map.universe
delta = delta * universe.temperamentRateModifier
map.temperamentScore = mMin(10000, mMax(-10000, currentTemperament + delta))
map.temperament = ((map.temperamentScore + 10000) * 0.00005)
2019-12-15 17:16:56 -08:00
2021-04-16 13:47:43 -07:00
if universe.debugTemperament then
if game.tick % 240 == 0 then
game.print("Rampant Stats:")
game.print("aN:" .. map.activeNests .. ", aRN:" .. map.activeRaidNests .. ", dPB:" .. map.destroyPlayerBuildings ..
", lEU:" .. map.lostEnemyUnits .. ", lEB:" .. map.lostEnemyBuilding .. ", rL:" .. map.rocketLaunched .. ", bEB:" .. map.builtEnemyBuilding ..
", iCB:" .. map.ionCannonBlasts .. ", aB:" .. map.artilleryBlasts)
game.print("temp: " .. map.temperament .. ", tempScore:" .. map.temperamentScore .. ", points:" .. map.points .. ", state:" .. constants.stateEnglish[map.state] .. ", surface:" .. map.surface.index .. " [" .. map.surface.name .. "]")
2021-04-16 13:47:43 -07:00
game.print("aS:" .. universe.squadCount .. ", aB:" .. universe.builderCount .. ", atkSize:" .. universe.attackWaveSize .. ", stlSize:" .. universe.settlerWaveSize .. ", formGroup:" .. universe.formSquadThreshold)
end
end
end
2019-02-15 20:17:30 -08:00
aiPlanningG = aiPlanning
2016-10-14 17:00:18 -07:00
return aiPlanning