mirror of
https://gitea.wildfiregames.com/0ad/0ad
synced 2026-06-20 15:23:56 -07:00
Patch By: Freagarach Reviewed By: wraitii Differential Revision: https://code.wildfiregames.com/D2064 This was SVN commit r22864.
566 lines
15 KiB
JavaScript
566 lines
15 KiB
JavaScript
Engine.LoadHelperScript("DamageBonus.js");
|
|
Engine.LoadHelperScript("Attacking.js");
|
|
Engine.LoadHelperScript("Player.js");
|
|
Engine.LoadHelperScript("Sound.js");
|
|
Engine.LoadHelperScript("ValueModification.js");
|
|
Engine.LoadComponentScript("interfaces/Attack.js");
|
|
Engine.LoadComponentScript("interfaces/AttackDetection.js");
|
|
Engine.LoadComponentScript("interfaces/DelayedDamage.js");
|
|
Engine.LoadComponentScript("interfaces/Resistance.js");
|
|
Engine.LoadComponentScript("interfaces/Health.js");
|
|
Engine.LoadComponentScript("interfaces/Loot.js");
|
|
Engine.LoadComponentScript("interfaces/Player.js");
|
|
Engine.LoadComponentScript("interfaces/Promotion.js");
|
|
Engine.LoadComponentScript("interfaces/StatusEffectsReceiver.js");
|
|
Engine.LoadComponentScript("interfaces/ModifiersManager.js");
|
|
Engine.LoadComponentScript("interfaces/Timer.js");
|
|
Engine.LoadComponentScript("Attack.js");
|
|
Engine.LoadComponentScript("DelayedDamage.js");
|
|
Engine.LoadComponentScript("Timer.js");
|
|
|
|
function Test_Generic()
|
|
{
|
|
ResetState();
|
|
|
|
let cmpTimer = ConstructComponent(SYSTEM_ENTITY, "Timer");
|
|
cmpTimer.OnUpdate({ "turnLength": 1 });
|
|
let attacker = 11;
|
|
let atkPlayerEntity = 1;
|
|
let attackerOwner = 6;
|
|
let cmpAttack = ConstructComponent(attacker, "Attack",
|
|
{
|
|
"Ranged": {
|
|
"Damage": {
|
|
"Crush": 5,
|
|
},
|
|
"MaxRange": 50,
|
|
"MinRange": 0,
|
|
"Delay": 0,
|
|
"Projectile": {
|
|
"Speed": 75.0,
|
|
"Spread": 0.5,
|
|
"Gravity": 9.81,
|
|
"LaunchPoint": { "@y": 3 }
|
|
}
|
|
}
|
|
});
|
|
let damage = 5;
|
|
let target = 21;
|
|
let targetOwner = 7;
|
|
let targetPos = new Vector3D(3, 0, 3);
|
|
|
|
let type = "Melee";
|
|
let damageTaken = false;
|
|
|
|
cmpAttack.GetAttackStrengths = attackType => ({ "Hack": 0, "Pierce": 0, "Crush": damage });
|
|
|
|
let data = {
|
|
"type": "Melee",
|
|
"attackData": {
|
|
"Damage": { "Hack": 0, "Pierce": 0, "Crush": damage },
|
|
},
|
|
"target": target,
|
|
"attacker": attacker,
|
|
"attackerOwner": attackerOwner,
|
|
"position": targetPos,
|
|
"projectileId": 9,
|
|
"direction": new Vector3D(1, 0, 0)
|
|
};
|
|
|
|
AddMock(atkPlayerEntity, IID_Player, {
|
|
"GetEnemies": () => [targetOwner]
|
|
});
|
|
|
|
AddMock(SYSTEM_ENTITY, IID_PlayerManager, {
|
|
"GetPlayerByID": id => atkPlayerEntity,
|
|
"GetAllPlayers": () => [0, 1, 2, 3, 4]
|
|
});
|
|
|
|
AddMock(SYSTEM_ENTITY, IID_ProjectileManager, {
|
|
"RemoveProjectile": () => {},
|
|
"LaunchProjectileAtPoint": (ent, pos, speed, gravity) => {},
|
|
});
|
|
|
|
AddMock(target, IID_Position, {
|
|
"GetPosition": () => targetPos,
|
|
"GetPreviousPosition": () => targetPos,
|
|
"GetPosition2D": () => Vector2D.From(targetPos),
|
|
"IsInWorld": () => true,
|
|
});
|
|
|
|
AddMock(target, IID_Health, {
|
|
"TakeDamage": (effectData, __, ___, bonusMultiplier) => {
|
|
damageTaken = true;
|
|
return { "killed": false, "HPchange": -bonusMultiplier * effectData.Crush };
|
|
},
|
|
});
|
|
|
|
AddMock(SYSTEM_ENTITY, IID_DelayedDamage, {
|
|
"MissileHit": () => {
|
|
damageTaken = true;
|
|
},
|
|
});
|
|
|
|
Engine.PostMessage = function(ent, iid, message)
|
|
{
|
|
TS_ASSERT_UNEVAL_EQUALS({ "type": type, "target": target, "attacker": attacker, "attackerOwner": attackerOwner, "damage": damage, "capture": 0, "statusEffects": [] }, message);
|
|
};
|
|
|
|
AddMock(target, IID_Footprint, {
|
|
"GetShape": () => ({ "type": "circle", "radius": 20 }),
|
|
});
|
|
|
|
AddMock(attacker, IID_Ownership, {
|
|
"GetOwner": () => attackerOwner,
|
|
});
|
|
|
|
AddMock(attacker, IID_Position, {
|
|
"GetPosition": () => new Vector3D(2, 0, 3),
|
|
"GetRotation": () => new Vector3D(1, 2, 3),
|
|
"IsInWorld": () => true,
|
|
});
|
|
|
|
function TestDamage()
|
|
{
|
|
cmpTimer.OnUpdate({ "turnLength": 1 });
|
|
TS_ASSERT(damageTaken);
|
|
damageTaken = false;
|
|
}
|
|
|
|
Attacking.HandleAttackEffects(data.type, data.attackData, data.target, data.attacker, data.attackerOwner);
|
|
TestDamage();
|
|
|
|
data.type = "Ranged";
|
|
type = data.type;
|
|
Attacking.HandleAttackEffects(data.type, data.attackData, data.target, data.attacker, data.attackerOwner);
|
|
TestDamage();
|
|
|
|
// Check for damage still being dealt if the attacker dies
|
|
cmpAttack.PerformAttack("Ranged", target);
|
|
Engine.DestroyEntity(attacker);
|
|
TestDamage();
|
|
|
|
atkPlayerEntity = 1;
|
|
AddMock(atkPlayerEntity, IID_Player, {
|
|
"GetEnemies": () => [2, 3]
|
|
});
|
|
TS_ASSERT_UNEVAL_EQUALS(Attacking.GetPlayersToDamage(atkPlayerEntity, true), [0, 1, 2, 3, 4]);
|
|
TS_ASSERT_UNEVAL_EQUALS(Attacking.GetPlayersToDamage(atkPlayerEntity, false), [2, 3]);
|
|
}
|
|
|
|
Test_Generic();
|
|
|
|
function TestLinearSplashDamage()
|
|
{
|
|
ResetState();
|
|
Engine.PostMessage = (ent, iid, message) => {};
|
|
|
|
const attacker = 50;
|
|
const attackerOwner = 1;
|
|
|
|
const origin = new Vector2D(0, 0);
|
|
|
|
let data = {
|
|
"type": "Ranged",
|
|
"attackData": { "Damage": { "Hack": 100, "Pierce": 0, "Crush": 0 } },
|
|
"attacker": attacker,
|
|
"attackerOwner": attackerOwner,
|
|
"origin": origin,
|
|
"radius": 10,
|
|
"shape": "Linear",
|
|
"direction": new Vector3D(1, 747, 0),
|
|
"friendlyFire": false,
|
|
};
|
|
|
|
let fallOff = function(x, y)
|
|
{
|
|
return (1 - x * x / (data.radius * data.radius)) * (1 - 25 * y * y / (data.radius * data.radius));
|
|
};
|
|
|
|
let hitEnts = new Set();
|
|
|
|
AddMock(attackerOwner, IID_Player, {
|
|
"GetEnemies": () => [2]
|
|
});
|
|
|
|
AddMock(SYSTEM_ENTITY, IID_PlayerManager, {
|
|
"GetPlayerByID": id => attackerOwner,
|
|
"GetAllPlayers": () => [0, 1, 2]
|
|
});
|
|
|
|
AddMock(SYSTEM_ENTITY, IID_RangeManager, {
|
|
"ExecuteQueryAroundPos": () => [60, 61, 62],
|
|
});
|
|
|
|
AddMock(60, IID_Position, {
|
|
"GetPosition2D": () => new Vector2D(2.2, -0.4),
|
|
});
|
|
|
|
AddMock(61, IID_Position, {
|
|
"GetPosition2D": () => new Vector2D(0, 0),
|
|
});
|
|
|
|
AddMock(62, IID_Position, {
|
|
"GetPosition2D": () => new Vector2D(5, 2),
|
|
});
|
|
|
|
AddMock(60, IID_Health, {
|
|
"TakeDamage": (effectData, __, ___, mult) => {
|
|
hitEnts.add(60);
|
|
TS_ASSERT_EQUALS(mult * (effectData.Hack + effectData.Pierce + effectData.Crush), 100 * fallOff(2.2, -0.4));
|
|
return { "killed": false, "change": -mult * (effectData.Hack + effectData.Pierce + effectData.Crush) };
|
|
}
|
|
});
|
|
|
|
AddMock(61, IID_Health, {
|
|
"TakeDamage": (effectData, __, ___, mult) => {
|
|
hitEnts.add(61);
|
|
TS_ASSERT_EQUALS(mult * (effectData.Hack + effectData.Pierce + effectData.Crush), 100 * fallOff(0, 0));
|
|
return { "killed": false, "change": -mult * (effectData.Hack + effectData.Pierce + effectData.Crush) };
|
|
}
|
|
});
|
|
|
|
AddMock(62, IID_Health, {
|
|
"TakeDamage": (effectData, __, ___, mult) => {
|
|
hitEnts.add(62);
|
|
TS_ASSERT_EQUALS(mult * (effectData.Hack + effectData.Pierce + effectData.Crush), 0);
|
|
return { "killed": false, "change": -mult * (effectData.Hack + effectData.Pierce + effectData.Crush) };
|
|
}
|
|
});
|
|
|
|
Attacking.CauseDamageOverArea(data);
|
|
TS_ASSERT(hitEnts.has(60));
|
|
TS_ASSERT(hitEnts.has(61));
|
|
TS_ASSERT(hitEnts.has(62));
|
|
hitEnts.clear();
|
|
|
|
data.direction = new Vector3D(0.6, 747, 0.8);
|
|
|
|
AddMock(60, IID_Health, {
|
|
"TakeDamage": (effectData, __, ___, mult) => {
|
|
hitEnts.add(60);
|
|
TS_ASSERT_EQUALS(mult * (effectData.Hack + effectData.Pierce + effectData.Crush), 100 * fallOff(1, 2));
|
|
return { "killed": false, "change": -mult * (effectData.Hack + effectData.Pierce + effectData.Crush) };
|
|
}
|
|
});
|
|
|
|
Attacking.CauseDamageOverArea(data);
|
|
TS_ASSERT(hitEnts.has(60));
|
|
TS_ASSERT(hitEnts.has(61));
|
|
TS_ASSERT(hitEnts.has(62));
|
|
hitEnts.clear();
|
|
}
|
|
|
|
TestLinearSplashDamage();
|
|
|
|
function TestCircularSplashDamage()
|
|
{
|
|
ResetState();
|
|
Engine.PostMessage = (ent, iid, message) => {};
|
|
|
|
const radius = 10;
|
|
let attackerOwner = 1;
|
|
|
|
let fallOff = function(r)
|
|
{
|
|
return 1 - r * r / (radius * radius);
|
|
};
|
|
|
|
AddMock(attackerOwner, IID_Player, {
|
|
"GetEnemies": () => [2]
|
|
});
|
|
|
|
AddMock(SYSTEM_ENTITY, IID_PlayerManager, {
|
|
"GetPlayerByID": id => attackerOwner,
|
|
"GetAllPlayers": () => [0, 1, 2]
|
|
});
|
|
|
|
AddMock(SYSTEM_ENTITY, IID_RangeManager, {
|
|
"ExecuteQueryAroundPos": () => [60, 61, 62, 64],
|
|
});
|
|
|
|
AddMock(60, IID_Position, {
|
|
"GetPosition2D": () => new Vector2D(3, 4),
|
|
});
|
|
|
|
AddMock(61, IID_Position, {
|
|
"GetPosition2D": () => new Vector2D(0, 0),
|
|
});
|
|
|
|
AddMock(62, IID_Position, {
|
|
"GetPosition2D": () => new Vector2D(3.6, 3.2),
|
|
});
|
|
|
|
AddMock(63, IID_Position, {
|
|
"GetPosition2D": () => new Vector2D(10, -10),
|
|
});
|
|
|
|
// Target on the frontier of the shape
|
|
AddMock(64, IID_Position, {
|
|
"GetPosition2D": () => new Vector2D(9, -4),
|
|
});
|
|
|
|
AddMock(60, IID_Resistance, {
|
|
"TakeDamage": (effectData, __, ___, mult) => {
|
|
TS_ASSERT_EQUALS(mult * (effectData.Hack + effectData.Pierce + effectData.Crush), 100 * fallOff(0));
|
|
return { "killed": false, "change": -mult * (effectData.Hack + effectData.Pierce + effectData.Crush) };
|
|
}
|
|
});
|
|
|
|
AddMock(61, IID_Resistance, {
|
|
"TakeDamage": (effectData, __, ___, mult) => {
|
|
TS_ASSERT_EQUALS(mult * (effectData.Hack + effectData.Pierce + effectData.Crush), 100 * fallOff(5));
|
|
return { "killed": false, "change": -mult * (effectData.Hack + effectData.Pierce + effectData.Crush) };
|
|
}
|
|
});
|
|
|
|
AddMock(62, IID_Resistance, {
|
|
"TakeDamage": (effectData, __, ___, mult) => {
|
|
TS_ASSERT_EQUALS(mult * (effectData.Hack + effectData.Pierce + effectData.Crush), 100 * fallOff(1));
|
|
return { "killed": false, "change": -mult * (effectData.Hack + effectData.Pierce + effectData.Crush) };
|
|
}
|
|
});
|
|
|
|
AddMock(63, IID_Resistance, {
|
|
"TakeDamage": (effectData, __, ___, mult) => {
|
|
TS_ASSERT(false);
|
|
}
|
|
});
|
|
|
|
AddMock(64, IID_Resistance, {
|
|
"TakeDamage": (effectData, __, ___, mult) => {
|
|
TS_ASSERT_EQUALS(mult * (effectData.Hack + effectData.Pierce + effectData.Crush), 0);
|
|
return { "killed": false, "change": -mult * (effectData.Hack + effectData.Pierce + effectData.Crush) };
|
|
}
|
|
});
|
|
|
|
Attacking.CauseDamageOverArea({
|
|
"type": "Ranged",
|
|
"attackData": { "Damage": { "Hack": 100, "Pierce": 0, "Crush": 0 } },
|
|
"attacker": 50,
|
|
"attackerOwner": attackerOwner,
|
|
"origin": new Vector2D(3, 4),
|
|
"radius": radius,
|
|
"shape": "Circular",
|
|
"friendlyFire": false,
|
|
});
|
|
}
|
|
|
|
TestCircularSplashDamage();
|
|
|
|
function Test_MissileHit()
|
|
{
|
|
ResetState();
|
|
Engine.PostMessage = (ent, iid, message) => {};
|
|
|
|
let cmpDelayedDamage = ConstructComponent(SYSTEM_ENTITY, "DelayedDamage");
|
|
|
|
let target = 60;
|
|
let targetOwner = 1;
|
|
let targetPos = new Vector3D(3, 10, 0);
|
|
let hitEnts = new Set();
|
|
|
|
AddMock(SYSTEM_ENTITY, IID_Timer, {
|
|
"GetLatestTurnLength": () => 500
|
|
});
|
|
|
|
const radius = 10;
|
|
|
|
let data = {
|
|
"type": "Ranged",
|
|
"attackData": { "Damage": { "Hack": 0, "Pierce": 100, "Crush": 0 } },
|
|
"target": 60,
|
|
"attacker": 70,
|
|
"attackerOwner": 1,
|
|
"position": targetPos,
|
|
"direction": new Vector3D(1, 0, 0),
|
|
"projectileId": 9,
|
|
};
|
|
|
|
AddMock(SYSTEM_ENTITY, IID_PlayerManager, {
|
|
"GetPlayerByID": id => id == 1 ? 10 : 11,
|
|
"GetAllPlayers": () => [0, 1]
|
|
});
|
|
|
|
AddMock(SYSTEM_ENTITY, IID_ProjectileManager, {
|
|
"RemoveProjectile": () => {},
|
|
"LaunchProjectileAtPoint": (ent, pos, speed, gravity) => {},
|
|
});
|
|
|
|
AddMock(60, IID_Position, {
|
|
"GetPosition": () => targetPos,
|
|
"GetPreviousPosition": () => targetPos,
|
|
"GetPosition2D": () => Vector2D.From(targetPos),
|
|
"IsInWorld": () => true,
|
|
});
|
|
|
|
AddMock(60, IID_Health, {
|
|
"TakeDamage": (effectData, __, ___, mult) => {
|
|
hitEnts.add(60);
|
|
TS_ASSERT_EQUALS(mult * (effectData.Hack + effectData.Pierce + effectData.Crush), 100);
|
|
return { "killed": false, "change": -mult * (effectData.Hack + effectData.Pierce + effectData.Crush) };
|
|
}
|
|
});
|
|
|
|
AddMock(60, IID_Footprint, {
|
|
"GetShape": () => ({ "type": "circle", "radius": 20 }),
|
|
});
|
|
|
|
AddMock(70, IID_Ownership, {
|
|
"GetOwner": () => 1,
|
|
});
|
|
|
|
AddMock(70, IID_Position, {
|
|
"GetPosition": () => new Vector3D(0, 0, 0),
|
|
"GetRotation": () => new Vector3D(0, 0, 0),
|
|
"IsInWorld": () => true,
|
|
});
|
|
|
|
AddMock(10, IID_Player, {
|
|
"GetEnemies": () => [2]
|
|
});
|
|
|
|
cmpDelayedDamage.MissileHit(data, 0);
|
|
TS_ASSERT(hitEnts.has(60));
|
|
hitEnts.clear();
|
|
|
|
// The main target is not hit but another one is hit.
|
|
|
|
AddMock(60, IID_Position, {
|
|
"GetPosition": () => new Vector3D(900, 10, 0),
|
|
"GetPreviousPosition": () => new Vector3D(900, 10, 0),
|
|
"GetPosition2D": () => new Vector2D(900, 0),
|
|
"IsInWorld": () => true,
|
|
});
|
|
|
|
AddMock(60, IID_Health, {
|
|
"TakeDamage": (effectData, __, ___, mult) => {
|
|
TS_ASSERT_EQUALS(false);
|
|
return { "killed": false, "change": -mult * (effectData.Hack + effectData.Pierce + effectData.Crush) };
|
|
}
|
|
});
|
|
|
|
AddMock(SYSTEM_ENTITY, IID_RangeManager, {
|
|
"ExecuteQueryAroundPos": () => [61]
|
|
});
|
|
|
|
AddMock(61, IID_Position, {
|
|
"GetPosition": () => targetPos,
|
|
"GetPreviousPosition": () => targetPos,
|
|
"GetPosition2D": () => Vector2D.from3D(targetPos),
|
|
"IsInWorld": () => true,
|
|
});
|
|
|
|
AddMock(61, IID_Health, {
|
|
"TakeDamage": (effectData, __, ___, mult) => {
|
|
hitEnts.add(61);
|
|
TS_ASSERT_EQUALS(mult * (effectData.Hack + effectData.Pierce + effectData.Crush), 100);
|
|
return { "killed": false, "change": -mult * (effectData.Hack + effectData.Pierce + effectData.Crush) };
|
|
}
|
|
});
|
|
|
|
AddMock(61, IID_Footprint, {
|
|
"GetShape": () => ({ "type": "circle", "radius": 20 }),
|
|
});
|
|
|
|
cmpDelayedDamage.MissileHit(data, 0);
|
|
TS_ASSERT(hitEnts.has(61));
|
|
hitEnts.clear();
|
|
|
|
// Add a splash damage.
|
|
data.splash = {};
|
|
data.splash.friendlyFire = false;
|
|
data.splash.radius = 10;
|
|
data.splash.shape = "Circular";
|
|
data.splash.attackData = { "Damage": { "Hack": 0, "Pierce": 0, "Crush": 200 } };
|
|
|
|
AddMock(SYSTEM_ENTITY, IID_RangeManager, {
|
|
"ExecuteQueryAroundPos": () => [61, 62]
|
|
});
|
|
|
|
let dealtDamage = 0;
|
|
AddMock(61, IID_Health, {
|
|
"TakeDamage": (effectData, __, ___, mult) => {
|
|
hitEnts.add(61);
|
|
dealtDamage += mult * (effectData.Hack + effectData.Pierce + effectData.Crush);
|
|
return { "killed": false, "change": -mult * (effectData.Hack + effectData.Pierce + effectData.Crush) };
|
|
}
|
|
});
|
|
|
|
AddMock(62, IID_Position, {
|
|
"GetPosition": () => new Vector3D(8, 10, 0),
|
|
"GetPreviousPosition": () => new Vector3D(8, 10, 0),
|
|
"GetPosition2D": () => new Vector2D(8, 0),
|
|
"IsInWorld": () => true,
|
|
});
|
|
|
|
AddMock(62, IID_Health, {
|
|
"TakeDamage": (effectData, __, ___, mult) => {
|
|
hitEnts.add(62);
|
|
TS_ASSERT_EQUALS(mult * (effectData.Hack + effectData.Pierce + effectData.Crush), 200 * 0.75);
|
|
return { "killed": false, "change": -mult * (effectData.Hack + effectData.Pierce + effectData.Crush) };
|
|
}
|
|
});
|
|
|
|
AddMock(62, IID_Footprint, {
|
|
"GetShape": () => ({ "type": "circle", "radius": 20 }),
|
|
});
|
|
|
|
cmpDelayedDamage.MissileHit(data, 0);
|
|
TS_ASSERT(hitEnts.has(61));
|
|
TS_ASSERT_EQUALS(dealtDamage, 100 + 200);
|
|
dealtDamage = 0;
|
|
TS_ASSERT(hitEnts.has(62));
|
|
hitEnts.clear();
|
|
|
|
// Add some hard counters bonus.
|
|
|
|
Engine.DestroyEntity(62);
|
|
AddMock(SYSTEM_ENTITY, IID_RangeManager, {
|
|
"ExecuteQueryAroundPos": () => [61]
|
|
});
|
|
|
|
let bonus= { "BonusCav": { "Classes": "Cavalry", "Multiplier": 400 } };
|
|
let splashBonus = { "BonusCav": { "Classes": "Cavalry", "Multiplier": 10000 } };
|
|
|
|
AddMock(61, IID_Identity, {
|
|
"GetClassesList": () => ["Cavalry"]
|
|
});
|
|
|
|
data.attackData.Bonuses = bonus;
|
|
cmpDelayedDamage.MissileHit(data, 0);
|
|
TS_ASSERT(hitEnts.has(61));
|
|
TS_ASSERT_EQUALS(dealtDamage, 400 * 100 + 200);
|
|
dealtDamage = 0;
|
|
hitEnts.clear();
|
|
|
|
data.splash.attackData.Bonuses = splashBonus;
|
|
cmpDelayedDamage.MissileHit(data, 0);
|
|
TS_ASSERT(hitEnts.has(61));
|
|
TS_ASSERT_EQUALS(dealtDamage, 400 * 100 + 10000 * 200);
|
|
dealtDamage = 0;
|
|
hitEnts.clear();
|
|
|
|
data.attackData.Bonuses = undefined;
|
|
cmpDelayedDamage.MissileHit(data, 0);
|
|
TS_ASSERT(hitEnts.has(61));
|
|
TS_ASSERT_EQUALS(dealtDamage, 100 + 10000 * 200);
|
|
dealtDamage = 0;
|
|
hitEnts.clear();
|
|
|
|
data.attackData.Bonuses = null;
|
|
cmpDelayedDamage.MissileHit(data, 0);
|
|
TS_ASSERT(hitEnts.has(61));
|
|
TS_ASSERT_EQUALS(dealtDamage, 100 + 10000 * 200);
|
|
dealtDamage = 0;
|
|
hitEnts.clear();
|
|
|
|
data.attackData.Bonuses = {};
|
|
cmpDelayedDamage.MissileHit(data, 0);
|
|
TS_ASSERT(hitEnts.has(61));
|
|
TS_ASSERT_EQUALS(dealtDamage, 100 + 10000 * 200);
|
|
dealtDamage = 0;
|
|
hitEnts.clear();
|
|
}
|
|
|
|
Test_MissileHit();
|