mirror of
https://github.com/WWhiteDreamProject/wwdpublic.git
synced 2026-04-27 02:27:46 +03:00
<!-- This is a semi-strict format, you can add/remove sections as needed but the order/format should be kept the same Remove these comments before submitting --> # Description the adding AI is now up to y'all because i'm not touching loadout code for name datasets, but it shouldn't be too bad from here --------- Signed-off-by: sleepyyapril <123355664+sleepyyapril@users.noreply.github.com> Signed-off-by: SolStar <44028047+ewokswagger@users.noreply.github.com> Signed-off-by: deltanedas <39013340+deltanedas@users.noreply.github.com> Co-authored-by: themias <89101928+themias@users.noreply.github.com> Co-authored-by: Verm <32827189+Vermidia@users.noreply.github.com> Co-authored-by: DrSmugleaf <10968691+DrSmugleaf@users.noreply.github.com> Co-authored-by: Sphiral <145869023+SphiraI@users.noreply.github.com> Co-authored-by: Ed <96445749+TheShuEd@users.noreply.github.com> Co-authored-by: Mr. 27 <45323883+Dutch-VanDerLinde@users.noreply.github.com> Co-authored-by: metalgearsloth <comedian_vs_clown@hotmail.com> Co-authored-by: Alzore <140123969+Blackern5000@users.noreply.github.com> Co-authored-by: ravage <142820619+ravage123321@users.noreply.github.com> Co-authored-by: slarticodefast <161409025+slarticodefast@users.noreply.github.com> Co-authored-by: Intoxicating-Innocence <188202277+Intoxicating-Innocence@users.noreply.github.com> Co-authored-by: Saphire <lattice@saphi.re> Co-authored-by: metalgearsloth <31366439+metalgearsloth@users.noreply.github.com> Co-authored-by: Errant <35878406+Errant-4@users.noreply.github.com> Co-authored-by: Tayrtahn <tayrtahn@gmail.com> Co-authored-by: CaasGit <87243814+CaasGit@users.noreply.github.com> Co-authored-by: BramvanZijp <56019239+BramvanZijp@users.noreply.github.com> Co-authored-by: Boaz1111 <149967078+Boaz1111@users.noreply.github.com> Co-authored-by: NakataRin <45946146+NakataRin@users.noreply.github.com> Co-authored-by: Kara <lunarautomaton6@gmail.com> Co-authored-by: Plykiya <58439124+Plykiya@users.noreply.github.com> Co-authored-by: SlamBamActionman <slambamactionman@gmail.com> Co-authored-by: Doomsdrayk <robotdoughnut@comcast.net> Co-authored-by: Brandon Hu <103440971+Brandon-Huu@users.noreply.github.com> Co-authored-by: SlamBamActionman <83650252+SlamBamActionman@users.noreply.github.com> Co-authored-by: ElectroJr <leonsfriedrich@gmail.com> Co-authored-by: Pieter-Jan Briers <pieterjan.briers+git@gmail.com> Co-authored-by: DrSmugleaf <DrSmugleaf@users.noreply.github.com> Co-authored-by: Julian Giebel <juliangiebel@live.de> Co-authored-by: nikthechampiongr <32041239+nikthechampiongr@users.noreply.github.com> Co-authored-by: Repo <47093363+Titian3@users.noreply.github.com> Co-authored-by: Chief-Engineer <119664036+Chief-Engineer@users.noreply.github.com> Co-authored-by: icekot8 <93311212+icekot8@users.noreply.github.com> Co-authored-by: AJCM-git <60196617+AJCM-git@users.noreply.github.com> Co-authored-by: Leon Friedrich <60421075+ElectroJr@users.noreply.github.com> Co-authored-by: no <165581243+pissdemon@users.noreply.github.com> Co-authored-by: Tornado Tech <54727692+Tornado-Technology@users.noreply.github.com> Co-authored-by: osjarw <62134478+osjarw@users.noreply.github.com> Co-authored-by: Simon <63975668+Simyon264@users.noreply.github.com> Co-authored-by: TGRCDev <tgrc@tgrc.dev> Co-authored-by: Milon <milonpl.git@proton.me> Co-authored-by: deltanedas <39013340+deltanedas@users.noreply.github.com> Co-authored-by: ShadowCommander <10494922+ShadowCommander@users.noreply.github.com> Co-authored-by: Fildrance <fildrance@gmail.com> Co-authored-by: pa.pecherskij <pa.pecherskij@interfax.ru> Co-authored-by: chavonadelal <156101927+chavonadelal@users.noreply.github.com> Co-authored-by: SolStar <44028047+ewokswagger@users.noreply.github.com> Co-authored-by: K-Dynamic <20566341+K-Dynamic@users.noreply.github.com> Co-authored-by: lzk <124214523+lzk228@users.noreply.github.com> Co-authored-by: ArchRBX <5040911+ArchRBX@users.noreply.github.com> Co-authored-by: archrbx <punk.gear5260@fastmail.com> Co-authored-by: Radezolid <snappednexus@gmail.com> Co-authored-by: Nemanja <98561806+EmoGarbage404@users.noreply.github.com> Co-authored-by: EmoGarbage404 <retron404@gmail.com> Co-authored-by: MilenVolf <63782763+MilenVolf@users.noreply.github.com> Co-authored-by: Velcroboy <107660393+IamVelcroboy@users.noreply.github.com> Co-authored-by: Velcroboy <velcroboy333@hotmail.com> Co-authored-by: neuPanda <chriseparton@gmail.com> Co-authored-by: neuPanda <spainman0@yahoo.com> Co-authored-by: Dvir <39403717+dvir001@users.noreply.github.com> Co-authored-by: Whatstone <whatston3@gmail.com> Co-authored-by: VideoKompany <135313844+VlaDOS1408@users.noreply.github.com> (cherry picked from commit 93ed70acfeda357133a701f637d3faeec02749bb)
244 lines
8.7 KiB
C#
244 lines
8.7 KiB
C#
using System.Diagnostics.CodeAnalysis;
|
|
using System.Linq;
|
|
using Content.Client.Examine;
|
|
using Content.Client.Gameplay;
|
|
using Content.Client.Popups;
|
|
using Content.Shared.Examine;
|
|
using Content.Shared.Tag;
|
|
using Content.Shared.Verbs;
|
|
using JetBrains.Annotations;
|
|
using Robust.Client.GameObjects;
|
|
using Robust.Client.Graphics;
|
|
using Robust.Client.Player;
|
|
using Robust.Client.State;
|
|
using Robust.Shared.Map;
|
|
using Robust.Shared.Utility;
|
|
|
|
namespace Content.Client.Verbs
|
|
{
|
|
[UsedImplicitly]
|
|
public sealed class VerbSystem : SharedVerbSystem
|
|
{
|
|
[Dependency] private readonly PopupSystem _popupSystem = default!;
|
|
[Dependency] private readonly ExamineSystem _examine = default!;
|
|
[Dependency] private readonly TagSystem _tagSystem = default!;
|
|
[Dependency] private readonly SharedTransformSystem _transform = default!;
|
|
[Dependency] private readonly IStateManager _stateManager = default!;
|
|
[Dependency] private readonly EntityLookupSystem _entityLookup = default!;
|
|
[Dependency] private readonly IPlayerManager _playerManager = default!;
|
|
|
|
/// <summary>
|
|
/// When a user right clicks somewhere, how large is the box we use to get entities for the context menu?
|
|
/// </summary>
|
|
public const float EntityMenuLookupSize = 0.25f;
|
|
|
|
[Dependency] private readonly IEyeManager _eyeManager = default!;
|
|
|
|
/// <summary>
|
|
/// These flags determine what entities the user can see on the context menu.
|
|
/// </summary>
|
|
public MenuVisibility Visibility;
|
|
|
|
public Action<VerbsResponseEvent>? OnVerbsResponse;
|
|
|
|
public override void Initialize()
|
|
{
|
|
base.Initialize();
|
|
|
|
SubscribeNetworkEvent<VerbsResponseEvent>(HandleVerbResponse);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get all of the entities in an area for displaying on the context menu.
|
|
/// </summary>
|
|
public bool TryGetEntityMenuEntities(MapCoordinates targetPos, [NotNullWhen(true)] out List<EntityUid>? result)
|
|
{
|
|
result = null;
|
|
|
|
if (_stateManager.CurrentState is not GameplayStateBase gameScreenBase)
|
|
return false;
|
|
|
|
var player = _playerManager.LocalEntity;
|
|
if (player == null)
|
|
return false;
|
|
|
|
// If FOV drawing is disabled, we will modify the visibility option to ignore visiblity checks.
|
|
var visibility = _eyeManager.CurrentEye.DrawFov
|
|
? Visibility
|
|
: Visibility | MenuVisibility.NoFov;
|
|
|
|
|
|
// Get entities
|
|
List<EntityUid> entities;
|
|
|
|
// Do we have to do FoV checks?
|
|
if ((visibility & MenuVisibility.NoFov) == 0)
|
|
{
|
|
var entitiesUnderMouse = gameScreenBase.GetClickableEntities(targetPos).ToHashSet();
|
|
bool Predicate(EntityUid e) => e == player || entitiesUnderMouse.Contains(e);
|
|
|
|
// first check the general location.
|
|
if (!_examine.CanExamine(player.Value, targetPos, Predicate))
|
|
return false;
|
|
|
|
TryComp(player.Value, out ExaminerComponent? examiner);
|
|
|
|
// Then check every entity
|
|
entities = new();
|
|
foreach (var ent in _entityLookup.GetEntitiesInRange(targetPos, EntityMenuLookupSize))
|
|
{
|
|
if (_examine.CanExamine(player.Value, targetPos, Predicate, ent, examiner))
|
|
entities.Add(ent);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
entities = _entityLookup.GetEntitiesInRange(targetPos, EntityMenuLookupSize).ToList();
|
|
}
|
|
|
|
if (entities.Count == 0)
|
|
return false;
|
|
|
|
if (visibility == MenuVisibility.All)
|
|
{
|
|
result = entities;
|
|
return true;
|
|
}
|
|
|
|
// remove any entities in containers
|
|
if ((visibility & MenuVisibility.InContainer) == 0)
|
|
{
|
|
for (var i = entities.Count - 1; i >= 0; i--)
|
|
{
|
|
var entity = entities[i];
|
|
|
|
if (ContainerSystem.IsInSameOrTransparentContainer(player.Value, entity))
|
|
continue;
|
|
|
|
entities.RemoveSwap(i);
|
|
}
|
|
}
|
|
|
|
// remove any invisible entities
|
|
if ((visibility & MenuVisibility.Invisible) == 0)
|
|
{
|
|
var spriteQuery = GetEntityQuery<SpriteComponent>();
|
|
|
|
for (var i = entities.Count - 1; i >= 0; i--)
|
|
{
|
|
var entity = entities[i];
|
|
|
|
if (!spriteQuery.TryGetComponent(entity, out var spriteComponent) ||
|
|
!spriteComponent.Visible ||
|
|
_tagSystem.HasTag(entity, "HideContextMenu"))
|
|
{
|
|
entities.RemoveSwap(i);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Remove any entities that do not have LOS
|
|
if ((visibility & MenuVisibility.NoFov) == 0)
|
|
{
|
|
var xformQuery = GetEntityQuery<TransformComponent>();
|
|
var playerPos = _transform.GetMapCoordinates(player.Value, xform: xformQuery.GetComponent(player.Value));
|
|
|
|
for (var i = entities.Count - 1; i >= 0; i--)
|
|
{
|
|
var entity = entities[i];
|
|
|
|
if (!_examine.InRangeUnOccluded(
|
|
playerPos,
|
|
_transform.GetMapCoordinates(entity, xform: xformQuery.GetComponent(entity)),
|
|
ExamineSystemShared.ExamineRange,
|
|
null))
|
|
{
|
|
entities.RemoveSwap(i);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (entities.Count == 0)
|
|
return false;
|
|
|
|
result = entities;
|
|
return true;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Ask the server to send back a list of server-side verbs, and for now return an incomplete list of verbs
|
|
/// (only those defined locally).
|
|
/// </summary>
|
|
public SortedSet<Verb> GetVerbs(NetEntity target, EntityUid user, List<Type> verbTypes, out List<VerbCategory> extraCategories, bool force = false)
|
|
{
|
|
if (!target.IsClientSide())
|
|
RaiseNetworkEvent(new RequestServerVerbsEvent(target, verbTypes, adminRequest: force));
|
|
|
|
// Some admin menu interactions will try get verbs for entities that have not yet been sent to the player.
|
|
if (!TryGetEntity(target, out var local))
|
|
{
|
|
extraCategories = new();
|
|
return new();
|
|
}
|
|
|
|
return GetLocalVerbs(local.Value, user, verbTypes, out extraCategories, force);
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
/// Execute actions associated with the given verb.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// Unless this is a client-exclusive verb, this will also tell the server to run the same verb.
|
|
/// </remarks>
|
|
public void ExecuteVerb(EntityUid target, Verb verb)
|
|
{
|
|
ExecuteVerb(GetNetEntity(target), verb);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Execute actions associated with the given verb.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// Unless this is a client-exclusive verb, this will also tell the server to run the same verb.
|
|
/// </remarks>
|
|
public void ExecuteVerb(NetEntity target, Verb verb)
|
|
{
|
|
if ( _playerManager.LocalEntity is not {} user)
|
|
return;
|
|
|
|
// is this verb actually valid?
|
|
if (verb.Disabled)
|
|
{
|
|
// maybe send an informative pop-up message.
|
|
if (!string.IsNullOrWhiteSpace(verb.Message))
|
|
_popupSystem.PopupEntity(verb.Message, user);
|
|
|
|
return;
|
|
}
|
|
|
|
if (verb.ClientExclusive || target.IsClientSide())
|
|
// is this a client exclusive (gui) verb?
|
|
ExecuteVerb(verb, user, GetEntity(target));
|
|
else
|
|
EntityManager.RaisePredictiveEvent(new ExecuteVerbEvent(target, verb));
|
|
}
|
|
|
|
private void HandleVerbResponse(VerbsResponseEvent msg)
|
|
{
|
|
OnVerbsResponse?.Invoke(msg);
|
|
}
|
|
}
|
|
|
|
[Flags]
|
|
public enum MenuVisibility
|
|
{
|
|
// What entities can a user see on the entity menu?
|
|
Default = 0, // They can only see entities in FoV.
|
|
NoFov = 1 << 0, // They ignore FoV restrictions
|
|
InContainer = 1 << 1, // They can see through containers.
|
|
Invisible = 1 << 2, // They can see entities without sprites and the "HideContextMenu" tag is ignored.
|
|
All = NoFov | InContainer | Invisible
|
|
}
|
|
}
|