-
Notifications
You must be signed in to change notification settings - Fork 321
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
add Lua burnblow fallback "over range" and "leash" customparams (#3918)
Preventing projectiles from exceeding their intended max range. Adds custom params to weapon defs to control this behaviour.
- Loading branch information
1 parent
17acfeb
commit e57f99e
Showing
1 changed file
with
223 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,223 @@ | ||
function gadget:GetInfo() | ||
return { | ||
name = "Projectile Over-Range and Leashing", | ||
desc = "Destroys projectiles if they exceed defined ranges", | ||
author = "SethDGamre", | ||
layer = 1, | ||
enabled = true | ||
} | ||
end | ||
|
||
if not gadgetHandler:IsSyncedCode() then return end | ||
|
||
---- unit customParams ---- | ||
-- use weaponDef customparams overrange_distance to destroy projectiles exceeding its limit. | ||
-- use weaponDef customparams leash_distance to destroy projectiles when they exceed range/overrange_distance and leash_distance relative to unit position. | ||
|
||
---- optional customParams ---- | ||
-- use weaponDef customparams projectile_destruction_method = "string" to change how projectiles are destroyed. | ||
-- "explode" (default if undefined) detonates the projectile. | ||
-- "descend" moves the projectile downward until it is destroyed by collision event. | ||
|
||
--static values | ||
local lateralMultiplier = 0.85 | ||
local compoundingMultiplier = 1.1 --compounding multiplier that influences the arc at which projectiles are forced to descend | ||
local descentSpeedStartingMultiplier = 0.15 | ||
|
||
local descentModulo = math.floor(Game.gameSpeed / 4) | ||
local leashModulo = math.ceil(Game.gameSpeed / 3) | ||
|
||
--functions | ||
local spGetUnitPosition = Spring.GetUnitPosition | ||
local mathRandom = math.random | ||
local mathCeil = math.ceil | ||
local mathSqrt = math.sqrt | ||
local spGetProjectilePosition = Spring.GetProjectilePosition | ||
local spSetProjectileCollision = Spring.SetProjectileCollision | ||
local spGetProjectileVelocity = Spring.GetProjectileVelocity | ||
local spSetProjectileVelocity = Spring.SetProjectileVelocity | ||
|
||
--tables | ||
local defWatchTable = {} | ||
local proMetaData = {} | ||
local flightTimeWatch = {} | ||
local descentTable = {} | ||
local killQueue = {} | ||
local leashWatch = {} | ||
|
||
--variables | ||
local gameFrame = 0 | ||
|
||
|
||
for weaponDefID, weaponDef in pairs(WeaponDefs) do | ||
local customParams = weaponDef.customParams | ||
if customParams.leash_distance or customParams.overrange_distance then | ||
local watchParams = {} | ||
if customParams.leash_distance then | ||
watchParams.leashRangeSq = tonumber(customParams.leash_distance) ^ 2 | ||
end | ||
|
||
local overRange = tonumber(customParams.overrange_distance) or weaponDef.range | ||
watchParams.overRange = overRange | ||
|
||
local ascentFrames = 0 | ||
if weaponDef.type == "StarburstLauncher" then | ||
ascentFrames = weaponDef.uptime * Game.gameSpeed | ||
end | ||
watchParams.flightTimeFrames = | ||
math.max(math.floor(((overRange / weaponDef.projectilespeed) + ascentFrames)), 1) | ||
watchParams.weaponDefID = weaponDefID | ||
|
||
local destructionMethod = customParams.projectile_destruction_method or "explode" | ||
if destructionMethod == "descend" then | ||
watchParams.descentMethod = true | ||
end | ||
|
||
defWatchTable[weaponDefID] = watchParams | ||
Script.SetWatchWeapon(weaponDefID, true) | ||
end | ||
end | ||
|
||
|
||
local function leashCheck(maxRangeSq, proOwnerID, proX, proZ) | ||
local ownerX, _, ownerZ = spGetUnitPosition(proOwnerID) | ||
if ownerX then | ||
local dx = ownerX - proX | ||
local dz = ownerZ - proZ | ||
if (dx * dx + dz * dz) > maxRangeSq then | ||
return true | ||
end | ||
else | ||
return true | ||
end | ||
end | ||
|
||
|
||
local function recalculateFlightTime(proID, maxRange, x1, z1, x2, z2) | ||
local dx = x2 - x1 | ||
local dz = z2 - z1 | ||
local distance = mathSqrt(dx * dx + dz * dz) | ||
|
||
-- Check if the projectile is within the max range | ||
if distance < maxRange then | ||
local vx, _, vz = spGetProjectileVelocity(proID) | ||
if not vx then return false end | ||
|
||
local remainingDistance = maxRange - distance | ||
|
||
local proSpeed = mathSqrt(vx * vx + vz * vz) | ||
|
||
local frames = mathCeil(remainingDistance / proSpeed) | ||
|
||
return frames | ||
else | ||
return false | ||
end | ||
end | ||
|
||
local function setDestructionFrame(proID, newFlightTime) | ||
local triggerFrame = gameFrame + newFlightTime --mathCeil(gameFrame + mathRandom(projectileWatchModulus)) | ||
|
||
killQueue[triggerFrame] = killQueue[triggerFrame] or {} | ||
killQueue[triggerFrame][#killQueue[triggerFrame] + 1] = proID | ||
end | ||
|
||
|
||
local function setFlightTimeFrame(proID, newFlightTime) | ||
local triggerFrame = gameFrame + newFlightTime | ||
|
||
flightTimeWatch[triggerFrame] = flightTimeWatch[triggerFrame] or {} | ||
flightTimeWatch[triggerFrame][#flightTimeWatch[triggerFrame] + 1] = proID | ||
end | ||
|
||
|
||
function gadget:ProjectileCreated(proID, proOwnerID, weaponDefID) | ||
local defData = defWatchTable[weaponDefID] | ||
if not defData then return end | ||
|
||
setFlightTimeFrame(proID, defData.flightTimeFrames) | ||
|
||
local metaData = { weaponDefID = weaponDefID, proOwnerID = proOwnerID } | ||
local originX, _, originZ = spGetUnitPosition(proOwnerID) | ||
metaData.originX = originX | ||
metaData.originZ = originZ | ||
|
||
proMetaData[proID] = metaData | ||
end | ||
|
||
function gadget:ProjectileDestroyed(proID) | ||
proMetaData[proID] = nil | ||
end | ||
|
||
function gadget:GameFrame(frame) | ||
gameFrame = frame | ||
|
||
if flightTimeWatch[frame] then | ||
for _, proID in ipairs(flightTimeWatch[frame]) do | ||
local projectileX, _, projectileZ = spGetProjectilePosition(proID) | ||
if projectileX then | ||
local proData = proMetaData[proID] | ||
local defData = defWatchTable[proData.weaponDefID] | ||
local newFlightTime = recalculateFlightTime(proID, defData.overRange, proData.originX, proData.originZ, projectileX, projectileZ) | ||
if newFlightTime then | ||
setFlightTimeFrame(proID, newFlightTime) | ||
else | ||
if defData.leashRangeSq then | ||
leashWatch[proID] = defData.leashRangeSq | ||
else | ||
setDestructionFrame(proID, 1) --destroy next frame | ||
end | ||
end | ||
else | ||
proMetaData[proID] = nil | ||
end | ||
end | ||
flightTimeWatch[frame] = nil | ||
end | ||
|
||
if frame % leashModulo == 3 then | ||
for proID, leashRangeSq in pairs(leashWatch) do | ||
local projectileX, _, projectileZ = spGetProjectilePosition(proID) | ||
if projectileX then | ||
local proData = proMetaData[proID] | ||
if leashCheck(leashRangeSq, proData.proOwnerID, projectileX, projectileZ) then | ||
setDestructionFrame(proID, mathRandom(leashModulo)) --destroy randomly between now and next frame check to reduce simultaneous projectile destructions | ||
leashWatch[proID] = nil | ||
end | ||
else | ||
leashWatch[proID] = nil | ||
proMetaData[proID] = nil | ||
end | ||
end | ||
end | ||
|
||
if killQueue[frame] then | ||
local descentMultiplier = descentSpeedStartingMultiplier | ||
for _, proID in ipairs(killQueue[frame]) do | ||
local proData = proMetaData[proID] | ||
if proData then | ||
local defData = defWatchTable[proData.weaponDefID] | ||
if defData.descentMethod then | ||
descentTable[proID] = descentMultiplier | ||
else | ||
spSetProjectileCollision(proID) | ||
end | ||
end | ||
proMetaData[proID] = nil | ||
end | ||
killQueue[frame] = nil | ||
end | ||
|
||
if frame % descentModulo == 3 then | ||
for proID, descentMultiplier in pairs(descentTable) do | ||
local velocityX, velocityY, velocityZ, velocityOverall = spGetProjectileVelocity(proID) | ||
if velocityY then | ||
local newVelocityY = velocityY - velocityOverall * descentMultiplier | ||
spSetProjectileVelocity(proID, velocityX * lateralMultiplier, newVelocityY, velocityZ * lateralMultiplier) | ||
descentTable[proID] = descentMultiplier * compoundingMultiplier | ||
else | ||
descentTable[proID] = nil | ||
end | ||
end | ||
end | ||
end |