mirror of
https://github.com/WWhiteDreamProject/wwdpublic.git
synced 2026-04-17 13:37:47 +03:00
Ports Blob from https://github.com/Goob-Station/Goob-Station/pull/975 that was ported from https://github.com/Rxup/space-station-14. Credit to VigersRay for original code, Roudenn and Rxup for maintaining and jorgun for the Goob port. --- - [X] Port https://github.com/Goob-Station/Goob-Station/pull/975; - [X] Port https://github.com/Goob-Station/Goob-Station/pull/1209; - [X] Port Blob related code from https://github.com/Goob-Station/Goob-Station/pull/1262; - [X] Port Blob related code from https://github.com/Goob-Station/Goob-Station/pull/1340; - [X] Port https://github.com/Goob-Station/Goob-Station/pull/1408; - [X] Port https://github.com/Goob-Station/Goob-Station/pull/1419; - [X] Port https://github.com/Goob-Station/Goob-Station/pull/1440; - [X] Port https://github.com/Goob-Station/Goob-Station/pull/1817; - [X] Port https://github.com/Goob-Station/Goob-Station/pull/2077; - [ ] ~Port https://github.com/Goob-Station/Goob-Station/pull/1916~; - [ ] ~Port https://github.com/Goob-Station/Goob-Station/pull/1917~; - [X] Port https://github.com/Goob-Station/Goob-Station/pull/2077; - [X] Port https://github.com/Goob-Station/Goob-Station/pull/2092; - [X] Port https://github.com/Goob-Station/Goob-Station/pull/2546; - [X] Port https://github.com/Rxup/space-station-14/pull/963; - [X] Port https://github.com/Rxup/space-station-14/pull/998; - [ ] ~Port https://github.com/Goob-Station/Goob-Station/pull/2563~. - [X] Enable Blob and Blob gamemode; - [X] Add `StationGlobConfig` to all stations; - [X] Use `AnnouncerSystem` in `BlobRuleSystem.cs`; - [X] Blob language and Hivemind (from https://github.com/Rxup/space-station-14/pull/176); - [x] Change CVars location; - [X] Add media. --- <details><summary><h1>Media</h1></summary> <p> https://youtu.be/-WtMQwRcmrU?si=su3An6RtiCTZg-DV </p> </details> --- 🆑 VigersRay, Roudenn, Rxup, vladospupuos, fishbait and Kyoth25f - add: Added a new antagonist: Blob --------- Co-authored-by: fishbait <gnesse@gmail.com> Co-authored-by: Fishbait <Fishbait@git.ml> Co-authored-by: Aiden <aiden@djkraz.com> Co-authored-by: beck-thompson <107373427+beck-thompson@users.noreply.github.com> Co-authored-by: lanse12 <cloudability.ez@gmail.com> Co-authored-by: BombasterDS <deniskaporoshok@gmail.com> Co-authored-by: Aviu00 <93730715+Aviu00@users.noreply.github.com> Co-authored-by: Piras314 <p1r4s@proton.me> Co-authored-by: shibe <95730644+shibechef@users.noreply.github.com> Co-authored-by: Ilya246 <57039557+Ilya246@users.noreply.github.com> Co-authored-by: JohnOakman <sremy2012@hotmail.fr> Co-authored-by: Fat Engineer Gaming <159075414+Fat-Engineer-Gaming@users.noreply.github.com> Co-authored-by: coderabbitai[bot] <136622811+coderabbitai[bot]@users.noreply.github.com> Co-authored-by: Rouden <149893554+Roudenn@users.noreply.github.com>
182 lines
7.9 KiB
C#
182 lines
7.9 KiB
C#
using Content.Server.Atmos.Components;
|
|
using Content.Server.Destructible;
|
|
using Content.Shared.Atmos;
|
|
using Content.Shared.Damage;
|
|
using Content.Shared.Explosion;
|
|
using Content.Shared.Explosion.EntitySystems;
|
|
using Content.Shared.FixedPoint;
|
|
using Robust.Shared.Map.Components;
|
|
namespace Content.Server.Explosion.EntitySystems;
|
|
|
|
public sealed partial class ExplosionSystem : SharedExplosionSystem
|
|
{
|
|
[Dependency] private readonly DestructibleSystem _destructibleSystem = default!;
|
|
|
|
private readonly Dictionary<string, int> _explosionTypes = new();
|
|
|
|
private void InitAirtightMap()
|
|
{
|
|
// Currently explosion prototype hot-reload isn't supported, as it would involve completely re-computing the
|
|
// airtight map. Could be done, just not yet implemented.
|
|
|
|
// for storing airtight entity damage thresholds for all anchored airtight entities, we will use integers in
|
|
// place of id-strings. This initializes the string <--> id association.
|
|
// This allows us to replace a Dictionary<string, float> with just a float[].
|
|
int index = 0;
|
|
foreach (var prototype in _prototypeManager.EnumeratePrototypes<ExplosionPrototype>())
|
|
{
|
|
_explosionTypes.Add(prototype.ID, index);
|
|
index++;
|
|
}
|
|
}
|
|
|
|
// The explosion intensity required to break an entity depends on the explosion type. So it is stored in a
|
|
// Dictionary<string, float>
|
|
//
|
|
// Hence, each tile has a tuple (Dictionary<string, float>, AtmosDirection). This specifies what directions are
|
|
// blocked, and how intense a given explosion type needs to be in order to destroy ALL airtight entities on that
|
|
// tile. This is the TileData struct.
|
|
//
|
|
// We then need this data for every tile on a grid. So this mess of a variable maps the Grid ID and Vector2i grid
|
|
// indices to this tile-data struct.
|
|
private Dictionary<EntityUid, Dictionary<Vector2i, TileData>> _airtightMap = new();
|
|
|
|
public void UpdateAirtightMap(EntityUid gridId, Vector2i tile, MapGridComponent? grid = null, EntityQuery<AirtightComponent>? query = null)
|
|
{
|
|
if (Resolve(gridId, ref grid, false))
|
|
UpdateAirtightMap(gridId, grid, tile, query);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Update the map of explosion blockers.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// Gets a list of all airtight entities on a tile. Assembles a <see cref="AtmosDirection"/> that specifies
|
|
/// what directions are blocked, along with the largest explosion tolerance. Note that as we only keep track
|
|
/// of the largest tolerance, this means that the explosion map will actually be inaccurate if you have
|
|
/// something like a normal and a reinforced windoor on the same tile. But given that this is a pretty rare
|
|
/// occurrence, I am fine with this.
|
|
/// </remarks>
|
|
public void UpdateAirtightMap(EntityUid gridId, MapGridComponent grid, Vector2i tile, EntityQuery<AirtightComponent>? query = null)
|
|
{
|
|
var tolerance = new float[_explosionTypes.Count];
|
|
var blockedDirections = AtmosDirection.Invalid;
|
|
|
|
if (!_airtightMap.ContainsKey(gridId))
|
|
_airtightMap[gridId] = new();
|
|
|
|
query ??= EntityManager.GetEntityQuery<AirtightComponent>();
|
|
var damageQuery = EntityManager.GetEntityQuery<DamageableComponent>();
|
|
var destructibleQuery = EntityManager.GetEntityQuery<DestructibleComponent>();
|
|
var anchoredEnumerator = grid.GetAnchoredEntitiesEnumerator(tile);
|
|
|
|
while (anchoredEnumerator.MoveNext(out var uid))
|
|
{
|
|
if (!query.Value.TryGetComponent(uid, out var airtight) || !airtight.AirBlocked)
|
|
continue;
|
|
|
|
blockedDirections |= airtight.AirBlockedDirection;
|
|
var entityTolerances = GetExplosionTolerance(uid.Value, damageQuery, destructibleQuery);
|
|
for (var i = 0; i < tolerance.Length; i++)
|
|
{
|
|
tolerance[i] = Math.Max(tolerance[i], entityTolerances[i]);
|
|
}
|
|
}
|
|
|
|
if (blockedDirections != AtmosDirection.Invalid)
|
|
_airtightMap[gridId][tile] = new(tolerance, blockedDirections);
|
|
else
|
|
_airtightMap[gridId].Remove(tile);
|
|
}
|
|
|
|
/// <summary>
|
|
/// On receiving damage, re-evaluate how much explosion damage is needed to destroy an airtight entity.
|
|
/// </summary>
|
|
private void OnAirtightDamaged(EntityUid uid, AirtightComponent airtight, DamageChangedEvent args)
|
|
{
|
|
// do we need to update our explosion blocking map?
|
|
if (!airtight.AirBlocked)
|
|
return;
|
|
|
|
if (!EntityManager.TryGetComponent(uid, out TransformComponent? transform) || !transform.Anchored)
|
|
return;
|
|
|
|
if (!TryComp<MapGridComponent>(transform.GridUid, out var grid))
|
|
return;
|
|
|
|
UpdateAirtightMap(transform.GridUid.Value, grid, grid.CoordinatesToTile(transform.Coordinates));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Return a dictionary that specifies how intense a given explosion type needs to be in order to destroy an entity.
|
|
/// </summary>
|
|
public float[] GetExplosionTolerance(
|
|
EntityUid uid,
|
|
EntityQuery<DamageableComponent> damageQuery,
|
|
EntityQuery<DestructibleComponent> destructibleQuery)
|
|
{
|
|
// How much total damage is needed to destroy this entity? This also includes "break" behaviors. This ASSUMES
|
|
// that this will result in a non-airtight entity.Entities that ONLY break via construction graph node changes
|
|
// are currently effectively "invincible" as far as this is concerned. This really should be done more rigorously.
|
|
var totalDamageTarget = FixedPoint2.MaxValue;
|
|
if (destructibleQuery.TryGetComponent(uid, out var destructible))
|
|
{
|
|
totalDamageTarget = _destructibleSystem.DestroyedAt(uid, destructible);
|
|
}
|
|
|
|
var explosionTolerance = new float[_explosionTypes.Count];
|
|
if (totalDamageTarget == FixedPoint2.MaxValue || !damageQuery.TryGetComponent(uid, out var damageable))
|
|
{
|
|
for (var i = 0; i < explosionTolerance.Length; i++)
|
|
{
|
|
explosionTolerance[i] = float.MaxValue;
|
|
}
|
|
return explosionTolerance;
|
|
}
|
|
|
|
// What multiple of each explosion type damage set will result in the damage exceeding the required amount? This
|
|
// does not support entities dynamically changing explosive resistances (e.g. via clothing). But these probably
|
|
// shouldn't be airtight structures anyways....
|
|
|
|
foreach (var (id, index) in _explosionTypes)
|
|
{
|
|
if (!_prototypeManager.TryIndex<ExplosionPrototype>(id, out var explosionType))
|
|
continue;
|
|
|
|
// evaluate the damage that this damage type would do to this entity
|
|
var damagePerIntensity = FixedPoint2.Zero;
|
|
foreach (var (type, value) in explosionType.DamagePerIntensity.DamageDict)
|
|
{
|
|
if (!damageable.Damage.DamageDict.ContainsKey(type))
|
|
continue;
|
|
|
|
var ev = new GetExplosionResistanceEvent(explosionType.ID);
|
|
RaiseLocalEvent(uid, ref ev);
|
|
|
|
damagePerIntensity += value * Math.Max(0, ev.DamageCoefficient);
|
|
}
|
|
|
|
explosionTolerance[index] = damagePerIntensity > 0
|
|
? (float) ((totalDamageTarget - damageable.TotalDamage) / damagePerIntensity)
|
|
: float.MaxValue;
|
|
}
|
|
|
|
return explosionTolerance;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Data struct that describes the explosion-blocking airtight entities on a tile.
|
|
/// </summary>
|
|
public struct TileData
|
|
{
|
|
public TileData(float[] explosionTolerance, AtmosDirection blockedDirections)
|
|
{
|
|
ExplosionTolerance = explosionTolerance;
|
|
BlockedDirections = blockedDirections;
|
|
}
|
|
|
|
public float[] ExplosionTolerance;
|
|
public AtmosDirection BlockedDirections = AtmosDirection.Invalid;
|
|
}
|
|
}
|