Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

ReactiveTeleportArmorRD #1802

Merged
merged 23 commits into from
Oct 4, 2024
Merged
Show file tree
Hide file tree
Changes from 8 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
21Melkuu marked this conversation as resolved.
Show resolved Hide resolved
21Melkuu marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace Content.Server.SS220.ReactiveTeleportArmor
{
/// <summary>
/// Intermediate component to work with TeleportOnDamageComponent
/// </summary>

[RegisterComponent]
public sealed partial class ReactiveTeleportArmorComponent : Component
{

}
}
21Melkuu marked this conversation as resolved.
Show resolved Hide resolved
DexlerXD marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,184 @@
// © SS220, An EULA/CLA with a hosting restriction, full text: https://raw.githubusercontent.com/SerbiaStrong-220/space-station-14/master/CLA.txt

using Robust.Shared.Random;
using Content.Shared.Damage;
using Content.Shared.Movement.Pulling.Systems;
using Content.Shared.Movement.Pulling.Components;
using Robust.Shared.Audio.Systems;
using Robust.Shared.Map;
using Robust.Shared.Physics;
using Robust.Shared.Physics.Components;
using Robust.Shared.Map.Components;
using System.Numerics;
using Content.Shared.Physics;
using Robust.Shared.Collections;
using Content.Shared.Clothing;
using Content.Shared.Clothing.EntitySystems;
using Content.Server.Explosion.EntitySystems;
using Content.Shared.Item;
using Content.Shared.Item.ItemToggle;
using Content.Shared.Item.ItemToggle.Components;
using Timer = Robust.Shared.Timing.Timer;

namespace Content.Server.SS220.ReactiveTeleportArmor
{
internal class ReactiveTeleportArmorSystem : EntitySystem
{
[Dependency] private readonly IRobustRandom _random = default!;
[Dependency] private readonly SharedAudioSystem _audio = default!;
[Dependency] private readonly SharedTransformSystem _xform = default!;
[Dependency] private readonly PullingSystem _pullingSystem = default!;
[Dependency] private readonly EntityLookupSystem _lookupSystem = default!;
[Dependency] private readonly SharedMapSystem _mapSystem = default!;
[Dependency] private readonly ExplosionSystem _explosion = default!;
[Dependency] private readonly SharedItemSystem _item = default!;
[Dependency] private readonly ClothingSystem _clothing = default!;
[Dependency] private readonly ItemToggleSystem _toggle = default!;

private EntityQuery<PhysicsComponent> _physicsQuery;
private HashSet<Entity<MapGridComponent>> _targetGrids = [];
public override void Initialize()
{
_physicsQuery = GetEntityQuery<PhysicsComponent>();

base.Initialize();
SubscribeLocalEvent<TeleportOnDamageComponent, DamageChangedEvent>(TeleporWhenDamaged);
SubscribeLocalEvent<ReactiveTeleportArmorComponent, ClothingGotEquippedEvent>(OnEquip);
SubscribeLocalEvent<ReactiveTeleportArmorComponent, ClothingGotUnequippedEvent>(OnUnequip);
SubscribeLocalEvent<ReactiveTeleportArmorComponent, ItemToggledEvent>(ToggleDone);
}

private void OnEquip(Entity<ReactiveTeleportArmorComponent> ent, ref ClothingGotEquippedEvent args)
{
EnsureComp<TeleportOnDamageComponent>(args.Wearer, out var comp);

comp.SavedUid = ent;
}
private void OnUnequip(Entity<ReactiveTeleportArmorComponent> ent, ref ClothingGotUnequippedEvent args)
21Melkuu marked this conversation as resolved.
Show resolved Hide resolved
{
RemComp<TeleportOnDamageComponent>(args.Wearer);
}
private void ToggleDone(Entity<ReactiveTeleportArmorComponent> ent, ref ItemToggledEvent args)
{
var prefix = args.Activated ? "on" : null;
_item.SetHeldPrefix(ent, prefix);
_clothing.SetEquippedPrefix(ent, prefix);
}

private void TeleporWhenDamaged(Entity<TeleportOnDamageComponent> ent, ref DamageChangedEvent args)
{
if (!TryComp<TeleportOnDamageComponent>(ent, out var armor))
return;

var xform = Transform(ent.Owner);
var targetCoords = SelectRandomTileInRange(xform, armor.TeleportRadius);

if (!args.DamageIncreased || args.DamageDelta == null)
return;
///teleport entity if taken damage && coord = !null && armor is on && !cooldown
if (args.DamageDelta.GetTotal() >= ent.Comp.WakeThreshold && targetCoords != null && _toggle.IsActivated(ent.Comp.SavedUid) && !ent.Comp.OnCoolDown)
{
ent.Comp.OnCoolDown = true;

// We need stop the user from being pulled so they don't just get "attached" with whoever is pulling them.
// This can for example happen when the user is cuffed and being pulled.
if (TryComp<PullableComponent>(ent.Owner, out var pull) && _pullingSystem.IsPulled(ent.Owner, pull))
_pullingSystem.TryStopPull(ent.Owner, pull);

switch (_random.Prob(ent.Comp.TeleportChance))
21Melkuu marked this conversation as resolved.
Show resolved Hide resolved
{
case true:

_xform.SetCoordinates(ent.Owner, targetCoords.Value);
_audio.PlayPvs(armor.TeleportSound, ent.Owner);
SelectRandomTileInRange(xform, armor.TeleportRadius);
break;
case false:

_explosion.TriggerExplosive(ent.Comp.SavedUid);
break;
}
Timer.Spawn(ent.Comp.CoolDownTime, () => ent.Comp.OnCoolDown = false);
}
}

private EntityCoordinates? SelectRandomTileInRange(TransformComponent userXform, float radius)
{
var userCoords = userXform.Coordinates.ToMap(EntityManager, _xform);
_targetGrids.Clear();
_lookupSystem.GetEntitiesInRange(userCoords, radius, _targetGrids);
Entity<MapGridComponent>? targetGrid = null;

if (_targetGrids.Count == 0)
return null;

// Give preference to the grid the entity is currently on.
// This does not guarantee that if the probability fails that the owner's grid won't be picked.
// In reality the probability is higher and depends on the number of grids.
if (userXform.GridUid != null && TryComp<MapGridComponent>(userXform.GridUid, out var gridComp))
{
var userGrid = new Entity<MapGridComponent>(userXform.GridUid.Value, gridComp);
if (_random.Prob(0.5f))
{
_targetGrids.Remove(userGrid);
targetGrid = userGrid;
}
}

if (targetGrid == null)
targetGrid = _random.GetRandom().PickAndTake(_targetGrids);

EntityCoordinates? targetCoords = null;

do
{
var valid = false;

var range = (float)Math.Sqrt(radius);
var box = Box2.CenteredAround(userCoords.Position, new Vector2(range, range));
var tilesInRange = _mapSystem.GetTilesEnumerator(targetGrid.Value.Owner, targetGrid.Value.Comp, box, false);
var tileList = new ValueList<Vector2i>();

while (tilesInRange.MoveNext(out var tile))
{
tileList.Add(tile.GridIndices);
}

while (tileList.Count != 0)
{
var tile = tileList.RemoveSwap(_random.Next(tileList.Count));
valid = true;
foreach (var entity in _mapSystem.GetAnchoredEntities(targetGrid.Value.Owner, targetGrid.Value.Comp,
tile))
{
if (!_physicsQuery.TryGetComponent(entity, out var body))
continue;

if (body.BodyType != BodyType.Static ||
!body.Hard ||
(body.CollisionLayer & (int)CollisionGroup.MobMask) == 0)
continue;

valid = false;
break;
}

if (valid)
{
targetCoords = new EntityCoordinates(targetGrid.Value.Owner,
_mapSystem.TileCenterToVector(targetGrid.Value, tile));
break;
}
}

if (valid || _targetGrids.Count == 0) // if we don't do the check here then PickAndTake will blow up on an empty set.
break;

targetGrid = _random.GetRandom().PickAndTake(_targetGrids);
} while (true);
DexlerXD marked this conversation as resolved.
Show resolved Hide resolved

return targetCoords;

}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
using Robust.Shared.Audio;
using Content.Shared.FixedPoint;

namespace Content.Server.SS220.ReactiveTeleportArmor;

/// <summary>
/// Randomly teleports entity when damaged.
/// </summary>
[RegisterComponent]
public sealed partial class TeleportOnDamageComponent : Component
{
/// <summary>
/// Up to how far to teleport the user
/// </summary>
[DataField, ViewVariables(VVAccess.ReadWrite)]
public float TeleportRadius = 30f;

[DataField, ViewVariables(VVAccess.ReadWrite)]
public SoundSpecifier TeleportSound = new SoundPathSpecifier("/Audio/Effects/teleport_arrival.ogg");

/// <summary>
/// How much damage of any type it takes to wake this entity.
/// </summary>
[DataField]
public FixedPoint2 WakeThreshold = FixedPoint2.New(4);

[DataField, ViewVariables(VVAccess.ReadWrite)]
public float TeleportChance = .9f;

[ViewVariables, AutoNetworkedField]
21Melkuu marked this conversation as resolved.
Show resolved Hide resolved
public EntityUid SavedUid;
21Melkuu marked this conversation as resolved.
Show resolved Hide resolved

public bool OnCoolDown = false;

[DataField]
public TimeSpan CoolDownTime = TimeSpan.FromSeconds(5);
}


2 changes: 2 additions & 0 deletions Resources/Prototypes/Catalog/Fills/Lockers/heads.yml
21Melkuu marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
Expand Up @@ -252,6 +252,7 @@
- id: ClothingHeadsetAltScience
- id: EncryptionKeyBinary
- id: ClothingWristWatchGold # SS220 Wrist Watch
- id: ClothingOuterReactiveArmor # SS220

- type: entity
id: LockerResearchDirectorFilled
Expand All @@ -272,6 +273,7 @@
- id: ClothingHeadsetAltScience
- id: EncryptionKeyBinary
- id: ClothingWristWatchGold # SS220 Wrist Watch
- id: ClothingOuterReactiveArmor # SS220

- type: entity
id: LockerHeadOfSecurityFilledHardsuit
Expand Down
8 changes: 8 additions & 0 deletions Resources/Prototypes/Objectives/stealTargetGroups.yml
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,14 @@
sprite:
sprite: Objects/Devices/hand_teleporter.rsi
state: icon
# SS220 - ReactiveArmor-start
- type: stealTargetGroup
id: ClothingOuterHardsuitRd
name: экспериментальная реактивная броня
sprite:
sprite: Clothing/OuterClothing/Hardsuits/rd.rsi #да-да
state: icon
# SS220 - ReactiveArmor-end
21Melkuu marked this conversation as resolved.
Show resolved Hide resolved

- type: stealTargetGroup
id: BookSecretDocuments
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
- type: entity
parent: [ClothingOuterBaseLarge, AllowSuitStorageClothing, BaseToggleClothing]
id: ClothingOuterReactiveArmor
name: "name1" # да-да, сделаю
description: "Desc2." # да............ точно сделаю, кстати всем ревьюевирам дарова
DexlerXD marked this conversation as resolved.
Show resolved Hide resolved
components:
- type: Sprite
sprite: Clothing/OuterClothing/Armor/captain_carapace.rsi #спрайтеры будут ныть как бляди если это будут спрайты с 13
- type: Clothing
sprite: Clothing/OuterClothing/Armor/captain_carapace.rsi #инфа по борщу не БУДЕТ
- type: Armor
modifiers:
coefficients:
Blunt: 0.5
Slash: 0.5
Piercing: 0.6
Heat: 0.5
Caustic: 0.9
- type: HeldSpeedModifier
damageCoefficient: 0.65
- type: ToggleClothing
action: ActionToggleReactiveArmor
- type: GroupExamine
- type: ReactiveTeleportArmor
- type: Explosive
explosionType: Default
maxIntensity: 2
totalIntensity: 1.5
intensitySlope: 1
canCreateVacuum: false
deleteAfterExplosion: false
repeatable: true


- type: entity
id: ActionToggleReactiveArmor
name: Переключение реактивной брони
description: Включает или выключает реактивную броню.
components:
- type: InstantAction
itemIconStyle: BigItem
event: !type:ToggleActionEvent
Loading