* DoAfters * Compact Clone() * Fix mice and cuffables * Try generalize attempt events * moves climbabledoafter event to shared, fixes issue with climbable target * Fix merge (cuffing) * Make all events netserializable * handful of doafter events moved * moves the rest of the events to their respective shared folders * Changes all mentions of server doafter to shared * stop stripping cancellation * fix merge errors * draw paused doafters * handle unpausing * missing netserializable ref * removes break on stun reference * removes cuffing state reference * Fix tools * Fix door prying. * Fix construction * Fix dumping * Fix wielding assert * fix rev * Fix test * more test fixes --------- Co-authored-by: keronshb <keronshb@live.com>
91 lines
3.1 KiB
C#
91 lines
3.1 KiB
C#
using Content.Server.Destructible;
|
|
using Content.Server.Gatherable.Components;
|
|
using Content.Shared.DoAfter;
|
|
using Content.Shared.EntityList;
|
|
using Content.Shared.Gatherable;
|
|
using Content.Shared.Interaction;
|
|
using Content.Shared.Tag;
|
|
using Robust.Shared.Prototypes;
|
|
using Robust.Shared.Random;
|
|
|
|
namespace Content.Server.Gatherable;
|
|
|
|
public sealed class GatherableSystem : EntitySystem
|
|
{
|
|
[Dependency] private readonly IPrototypeManager _prototypeManager = default!;
|
|
[Dependency] private readonly IRobustRandom _random = default!;
|
|
[Dependency] private readonly DestructibleSystem _destructible = default!;
|
|
[Dependency] private readonly SharedDoAfterSystem _doAfterSystem = default!;
|
|
[Dependency] private readonly SharedAudioSystem _audio = default!;
|
|
[Dependency] private readonly TagSystem _tagSystem = default!;
|
|
|
|
public override void Initialize()
|
|
{
|
|
base.Initialize();
|
|
|
|
SubscribeLocalEvent<GatherableComponent, InteractUsingEvent>(OnInteractUsing);
|
|
SubscribeLocalEvent<GatherableComponent, GatherableDoAfterEvent>(OnDoAfter);
|
|
}
|
|
|
|
private void OnInteractUsing(EntityUid uid, GatherableComponent component, InteractUsingEvent args)
|
|
{
|
|
if (!TryComp<GatheringToolComponent>(args.Used, out var tool) || component.ToolWhitelist?.IsValid(args.Used) == false)
|
|
return;
|
|
|
|
// Can't gather too many entities at once.
|
|
if (tool.MaxGatheringEntities < tool.GatheringEntities.Count + 1)
|
|
return;
|
|
|
|
var damageRequired = _destructible.DestroyedAt(uid);
|
|
var damageTime = (damageRequired / tool.Damage.Total).Float();
|
|
damageTime = Math.Max(1f, damageTime);
|
|
|
|
var doAfter = new DoAfterArgs(args.User, damageTime, new GatherableDoAfterEvent(), uid, target: uid, used: args.Used)
|
|
{
|
|
BreakOnDamage = true,
|
|
BreakOnTargetMove = true,
|
|
BreakOnUserMove = true,
|
|
MovementThreshold = 0.25f,
|
|
};
|
|
|
|
_doAfterSystem.TryStartDoAfter(doAfter);
|
|
}
|
|
|
|
private void OnDoAfter(EntityUid uid, GatherableComponent component, GatherableDoAfterEvent args)
|
|
{
|
|
if(!TryComp<GatheringToolComponent>(args.Args.Used, out var tool) || args.Args.Target == null)
|
|
return;
|
|
|
|
tool.GatheringEntities.Remove(args.Args.Target.Value);
|
|
if (args.Handled || args.Cancelled)
|
|
return;
|
|
|
|
// Complete the gathering process
|
|
_destructible.DestroyEntity(args.Args.Target.Value);
|
|
_audio.PlayPvs(tool.GatheringSound, args.Args.Target.Value);
|
|
|
|
// Spawn the loot!
|
|
if (component.MappedLoot == null)
|
|
return;
|
|
|
|
var playerPos = Transform(args.Args.User).MapPosition;
|
|
|
|
foreach (var (tag, table) in component.MappedLoot)
|
|
{
|
|
if (tag != "All")
|
|
{
|
|
if (!_tagSystem.HasTag(tool.Owner, tag))
|
|
continue;
|
|
}
|
|
var getLoot = _prototypeManager.Index<EntityLootTablePrototype>(table);
|
|
var spawnLoot = getLoot.GetSpawns();
|
|
var spawnPos = playerPos.Offset(_random.NextVector2(0.3f));
|
|
Spawn(spawnLoot[0], spawnPos);
|
|
}
|
|
args.Handled = true;
|
|
}
|
|
}
|
|
|
|
|
|
|