1
0
mirror of https://github.com/veden/Rampant.git synced 2025-01-05 22:53:24 +02:00
Rampant/prototypes/SwarmUtils.lua

368 lines
9.5 KiB
Lua
Raw Normal View History

2018-01-14 02:12:09 +02:00
local swarmUtils = {}
2018-01-13 04:55:20 +02:00
-- imports
local biterUtils = require("utils/BiterUtils")
package.path = "../?.lua;" .. package.path
2018-01-13 04:55:20 +02:00
local mathUtils = require("libs/MathUtils")
-- imported functions
2018-01-14 02:12:09 +02:00
local gaussianRandomRangeRG = mathUtils.gaussianRandomRangeRG
2018-01-13 04:55:20 +02:00
local mMax = math.max
2018-01-14 02:12:09 +02:00
local mMin = math.min
local mFloor = math.floor
2018-01-13 04:55:20 +02:00
local deepcopy = util.table.deepcopy
2018-01-14 02:12:09 +02:00
local xorRandom = mathUtils.xorRandom(settings.startup["rampant-enemySeed"].value)
local makeBiterCorpse = biterUtils.makeBiterCorpse
local makeUnitSpawnerCorpse = biterUtils.makeUnitSpawnerCorpse
local makeWormCorpse = biterUtils.makeWormCorpse
local makeSpitterCorpse = biterUtils.makeSpitterCorpse
2018-01-13 04:55:20 +02:00
local makeBiter = biterUtils.makeBiter
local makeSpitter = biterUtils.makeSpitter
local makeWorm = biterUtils.makeWorm
local makeUnitSpawner = biterUtils.makeUnitSpawner
-- module code
local function unitSetToProbabilityTable(points, upgradeTable, unitSet)
local dividers = {}
for i=1,#unitSet do
dividers[i] = 1
end
2018-01-14 02:12:09 +02:00
if upgradeTable then
while (points > 0) do
2018-01-18 09:15:10 +02:00
local index = mFloor(xorRandom() * #upgradeTable)+1
local upgrade = upgradeTable[index]
2018-01-13 04:55:20 +02:00
2018-01-18 09:15:10 +02:00
dividers[index] = dividers[index] + upgrade
2018-01-14 02:12:09 +02:00
2018-01-18 09:15:10 +02:00
points = points - 1
2018-01-14 02:12:09 +02:00
end
2018-01-13 04:55:20 +02:00
end
2018-01-14 02:12:09 +02:00
2018-01-13 04:55:20 +02:00
local total = 0
for i=1,#dividers do
total = total + dividers[i]
end
local runningTotal = 0
for i=1,#dividers do
runningTotal = runningTotal + (dividers[i] / total)
dividers[i] = runningTotal
end
2018-01-14 02:12:09 +02:00
local stepUnit = 1 / (#unitSet[1] + 1)
2018-01-13 04:55:20 +02:00
local probabilityTable = {}
for i=1,#unitSet do
local result
if (i == 1) then
result = {
{
0,
stepUnit
},
{
dividers[i],
0
}
}
elseif (i == #unitSet) then
result = {
{
dividers[i-2],
0
},
{
1,
stepUnit
}
}
else
result = {
{
((i - 2) > 0 and dividers[i-2]) or 0,
0
},
{
dividers[i-1],
stepUnit
},
{
dividers[i],
0
}
}
end
probabilityTable[i] = result
end
local result = {}
for i=1, #probabilityTable do
local probability = probabilityTable[i]
for x=1, #unitSet[i] do
result[#result+1] = {unitSet[i][x], probability}
end
end
return result
end
local function processUpgradeTable(upgradeTable)
for i=1,#upgradeTable do
local bonuses = upgradeTable[i]
for ii=1,#bonuses do
local bonus = bonuses[ii]
if (bonus.type == "attribute") or (bonus.type == "attack") then
if not bonus.mapping then
for x = 1, #bonus do
bonus[x] = bonus[x] * 0.10
end
end
elseif (bonus.type == "resistance") then
if bonus.decrease then
for x = 1, #bonus.decrease do
bonus.decrease[x] = bonus.decrease[x] * 0.10
end
end
if bonus.percent then
for x = 1, #bonus.percent do
bonus.percent[x] = bonus.percent[x] * 0.10
end
end
end
end
end
return upgradeTable
end
2018-01-18 09:15:10 +02:00
local function upgradeEntity(points, entity, upgradeTable, tier)
2018-01-14 02:12:09 +02:00
local remainingPoints = points
if upgradeTable then
while (remainingPoints > 0) do
local upgrade = upgradeTable[mFloor(xorRandom() * #upgradeTable)+1]
2018-01-13 04:55:20 +02:00
2018-01-18 09:15:10 +02:00
for i=1, #upgrade do
local bonus = upgrade[i]
2018-01-14 02:12:09 +02:00
if (bonus.type == "attribute") then
if bonus.mapping then
entity.attributes[bonus.name] = bonus.mapping[entity.attributes[bonus.name] or "default"]
else
2018-01-18 09:15:10 +02:00
local adj = bonus[tier]
adj = gaussianRandomRangeRG(adj, adj * 0.2, adj * 0.875, adj * 1.25, xorRandom)
entity.attributes[bonus.name] = (entity.attributes[bonus.name] or 0) + adj
2018-01-14 02:12:09 +02:00
end
2018-01-13 04:55:20 +02:00
end
2018-01-14 02:12:09 +02:00
if (bonus.type == "resistance") then
local field = bonus.name
2018-01-14 02:12:09 +02:00
if not entity.resistances[field] then
entity.resistances[field] = {}
2018-01-13 04:55:20 +02:00
end
2018-01-18 09:15:10 +02:00
local adj
if bonus.decrease then
2018-01-18 09:15:10 +02:00
adj = bonus.decrease[tier]
entity.resistances[field].decrease = (entity.resistances[field].decrease or 0) + adj
2018-01-13 04:55:20 +02:00
end
if bonus.percent then
2018-01-18 09:15:10 +02:00
adj = bonus.percent[tier]
entity.resistances[field].percent = mMin((entity.resistances[field].percent or 0) + adj, 100)
2018-01-13 04:55:20 +02:00
end
end
2018-01-14 02:12:09 +02:00
if (bonus.type == "attack") then
if bonus.mapping then
entity.attack[bonus.name] = bonus.mapping[entity.attack[bonus.name] or "default"]
2018-01-14 02:12:09 +02:00
else
2018-01-18 09:15:10 +02:00
local adj = bonus[tier]
-- if adj < 0 then
-- adj = -adj
-- adj = -gaussianRandomRangeRG(adj, adj * 0.2, adj * 0.75, adj * 1.25, xorRandom)
-- else
adj = gaussianRandomRangeRG(adj, adj * 0.2, adj * 0.875, adj * 1.25, xorRandom)
--end
entity.attack[bonus.name] = (entity.attack[bonus.name] or 0) + adj
2018-01-14 02:12:09 +02:00
end
2018-01-13 04:55:20 +02:00
end
end
2018-01-14 02:12:09 +02:00
2018-01-18 09:15:10 +02:00
remainingPoints = remainingPoints - 1
2018-01-14 02:12:09 +02:00
end
end
end
local function calculateRGBa(tint, tier)
local r = gaussianRandomRangeRG(tint.r, tint.r * 0.10 + (0.005 * tier), mMax(tint.r * 0.85 - (0.005 * tier), 0), mMin(tint.r * 1.15, 1), xorRandom)
local g = gaussianRandomRangeRG(tint.g, tint.g * 0.10 + (0.005 * tier), mMax(tint.g * 0.85 - (0.005 * tier), 0), mMin(tint.g * 1.15, 1), xorRandom)
local b = gaussianRandomRangeRG(tint.b, tint.b * 0.10 + (0.005 * tier), mMax(tint.b * 0.85 - (0.005 * tier), 0), mMin(tint.b * 1.15, 1), xorRandom)
local a = gaussianRandomRangeRG(tint.a, tint.a * 0.10 + (0.005 * tier), mMax(tint.a * 0.85 - (0.005 * tier), 0), mMin(tint.a * 1.15, 1), xorRandom)
return { r=r, g=g, b=b, a=a }
end
2018-01-14 02:12:09 +02:00
local function generateApperance(unit, tier)
local scaleValue = unit.scales[tier]
local scale = gaussianRandomRangeRG(scaleValue, scaleValue * 0.12, scaleValue * 0.60, scaleValue * 1.40, xorRandom) + (0.05 * tier)
2018-01-13 04:55:20 +02:00
unit.scale = scale
unit.attributes.scale = scale
if unit.tint then
local tint = calculateRGBa(unit.tint, tier)
unit.attributes.tint = tint
if unit.attack then
unit.attack.scale = scale
unit.attack.tint = tint
end
else
local tint1 = calculateRGBa(unit.tint1, tier)
local tint2 = calculateRGBa(unit.tint2, tier)
2018-01-14 02:12:09 +02:00
unit.attributes.tint1 = tint1
unit.attributes.tint2 = tint2
unit.attack.scale = scale
unit.attack.tint1 = tint1
unit.attack.tint2 = tint2
if unit.pTint then
unit.attack.pTint = calculateRGBa(unit.pTint, tier)
end
if unit.sTint then
unit.attack.sTint = calculateRGBa(unit.sTint, tier)
end
if unit.smTint then
unit.attack.smTint = calculateRGBa(unit.smTint, tier)
end
2018-01-14 02:12:09 +02:00
end
end
2018-01-13 04:55:20 +02:00
local function buildUnits(startingPoints, template, attackGenerator, upgradeTable, variations, tiers)
2018-01-14 02:12:09 +02:00
local unitSet = {}
for t=1, tiers do
local result = {}
for i=1,variations do
local unit = deepcopy(template)
unit.name = unit.name .. "-v" .. i .. "-t" .. t
2018-01-14 02:12:09 +02:00
generateApperance(unit, t)
2018-01-18 09:15:10 +02:00
upgradeEntity(startingPoints, unit, upgradeTable, t)
2018-01-14 02:12:09 +02:00
if unit.attackName then
unit.attack.name = unit.attackName .. "-v" .. i .. "-t" .. t
end
2018-01-13 04:55:20 +02:00
local entity
if (unit.type == "spitter") then
unit.attributes.corpse = makeSpitterCorpse(unit)
2018-01-13 04:55:20 +02:00
entity = makeSpitter(unit.name,
unit.attributes,
attackGenerator(unit.attack),
unit.resistances)
elseif (unit.type == "biter") then
unit.attributes.corpse = makeBiterCorpse(unit)
2018-01-13 04:55:20 +02:00
entity = makeBiter(unit.name,
unit.attributes,
attackGenerator(unit.attack),
unit.resistances)
end
result[i] = entity.name
data:extend({entity})
end
unitSet[t] = result
end
return unitSet
end
2018-01-18 09:15:10 +02:00
function swarmUtils.buildUnitSpawner(templates, upgradeTable, attackGenerator, variations, tiers)
local unitPoints = #upgradeTable.unit * 10
2018-01-18 09:15:10 +02:00
local unitSpawnerPoints = #upgradeTable.unitSpawner * 10
local probabilityPoints = #upgradeTable.probabilityTable * 10
processUpgradeTable(upgradeTable.unit)
processUpgradeTable(upgradeTable.unitSpawner)
local unitSet = buildUnits(unitPoints,
2018-01-15 01:10:56 +02:00
templates.unit,
attackGenerator,
upgradeTable.unit,
variations.unit,
tiers.unit)
for t=1, tiers.unitSpawner do
2018-01-13 04:55:20 +02:00
2018-01-15 01:10:56 +02:00
for i=1,variations.unitSpawner do
local unitSpawner = deepcopy(templates.unitSpawner)
unitSpawner.name = unitSpawner.name .. "-v" .. i .. "-t" .. t
2018-01-18 09:15:10 +02:00
local unitTable = unitSetToProbabilityTable(probabilityPoints,
2018-01-15 01:10:56 +02:00
upgradeTable.probabilityTable,
unitSet)
2018-01-14 02:12:09 +02:00
generateApperance(unitSpawner, t)
2018-01-18 09:15:10 +02:00
upgradeEntity(unitSpawnerPoints, unitSpawner, upgradeTable.unitSpawner, t)
2018-01-13 04:55:20 +02:00
if unitSpawner.autoplace then
unitSpawner.attributes["autoplace"] = unitSpawner.autoplace[t]
end
unitSpawner.attributes.corpse = makeUnitSpawnerCorpse(unitSpawner)
2018-01-13 04:55:20 +02:00
data:extend({
makeUnitSpawner(unitSpawner.name,
unitSpawner.attributes,
unitSpawner.resistances,
unitTable)
})
end
end
end
2018-01-18 09:15:10 +02:00
function swarmUtils.buildWorm(template, upgradeTable, attackGenerator, variations, tiers)
local wormPoints = #upgradeTable * 10
processUpgradeTable(upgradeTable)
2018-01-13 04:55:20 +02:00
for t=1, tiers do
for i=1,variations do
local worm = deepcopy(template)
worm.name = worm.name .. "-v" .. i .. "-t" .. t
2018-01-14 02:12:09 +02:00
generateApperance(worm, t)
2018-01-18 09:15:10 +02:00
upgradeEntity(wormPoints, worm, upgradeTable, t)
if worm.attackName then
worm.attack.name = worm.attackName .. "-v" .. i .. "-t" .. t
end
if worm.autoplace then
worm.attributes["autoplace"] = worm.autoplace[t]
end
worm.attributes.corpse = makeWormCorpse(worm)
2018-01-13 04:55:20 +02:00
data:extend({
makeWorm(worm.name,
worm.attributes,
attackGenerator(worm.attack),
worm.resistances)
})
end
end
end
2018-01-14 02:12:09 +02:00
return swarmUtils