forked from space-syndicate/space-station-14
Signed-off-by: Prole <172158352+Prole0@users.noreply.github.com> Co-authored-by: PJBot <pieterjan.briers+bot@gmail.com> Co-authored-by: ScarKy0 <106310278+ScarKy0@users.noreply.github.com> Co-authored-by: Samuka-C <47865393+Samuka-C@users.noreply.github.com> Co-authored-by: ArtisticRoomba <145879011+ArtisticRoomba@users.noreply.github.com> Co-authored-by: Partmedia <kevinz5000@gmail.com> Co-authored-by: slarticodefast <161409025+slarticodefast@users.noreply.github.com> Co-authored-by: themias <89101928+themias@users.noreply.github.com> Co-authored-by: Victor Shen <71985089+Vexerot@users.noreply.github.com> Co-authored-by: Ed <96445749+TheShuEd@users.noreply.github.com> Co-authored-by: Milon <milonpl.git@proton.me> Co-authored-by: Kirus59 <145689588+Kirus59@users.noreply.github.com> Co-authored-by: Nemanja <98561806+EmoGarbage404@users.noreply.github.com> Co-authored-by: Stomf <5dorkydorks@gmail.com> Co-authored-by: drakewill-CRL <46307022+drakewill-CRL@users.noreply.github.com> Co-authored-by: PraxisMapper <praxismapper@gmail.com> Co-authored-by: EmoGarbage404 <retron404@gmail.com> Co-authored-by: lzk <124214523+lzk228@users.noreply.github.com> Co-authored-by: Princess Cheeseballs <66055347+Pronana@users.noreply.github.com> Co-authored-by: IProduceWidgets <107586145+IProduceWidgets@users.noreply.github.com> Co-authored-by: TytosB <54259736+TytosB@users.noreply.github.com> Co-authored-by: abadaba695 <spacestation13thingy@gmail.com> Co-authored-by: kosticia <kosticia46@gmail.com> Co-authored-by: Thinbug <101073555+Thinbug0@users.noreply.github.com> Co-authored-by: pathetic meowmeow <uhhadd@gmail.com> Co-authored-by: SlamBamActionman <83650252+SlamBamActionman@users.noreply.github.com> Co-authored-by: Boaz1111 <149967078+Boaz1111@users.noreply.github.com> Co-authored-by: ActiveMammmoth <140334666+ActiveMammmoth@users.noreply.github.com> Co-authored-by: Myra <vasilis@pikachu.systems> Co-authored-by: Whatstone <166147148+whatston3@users.noreply.github.com> Co-authored-by: Pieter-Jan Briers <pieterjan.briers+git@gmail.com> Co-authored-by: K-Dynamic <20566341+K-Dynamic@users.noreply.github.com> Co-authored-by: Gentleman-Bird <dcgreen406@gmail.com> Co-authored-by: metalgearsloth <31366439+metalgearsloth@users.noreply.github.com> Co-authored-by: BIGZi0348 <svalker0348@gmail.com> Co-authored-by: LaCumbiaDelCoronavirus <90893484+LaCumbiaDelCoronavirus@users.noreply.github.com> Co-authored-by: imatsoup <93290208+imatsoup@users.noreply.github.com> Co-authored-by: Matthew Herber <32679887+happyrobot33@users.noreply.github.com> Co-authored-by: Ertanic <36124833+Ertanic@users.noreply.github.com> Co-authored-by: MissKay1994 <15877268+MissKay1994@users.noreply.github.com> Co-authored-by: Errant <35878406+Errant-4@users.noreply.github.com> Co-authored-by: eoineoineoin <helloworld@eoinrul.es> Co-authored-by: Tiniest Shark <head.rebel@yahoo.com> Co-authored-by: nikitosych <boriszyn@gmail.com> Co-authored-by: Tayrtahn <tayrtahn@gmail.com> Co-authored-by: Perry Fraser <perryprog@users.noreply.github.com> Co-authored-by: YoungThug <ramialanbagy@gmail.com> Co-authored-by: beck-thompson <107373427+beck-thompson@users.noreply.github.com> Co-authored-by: metalgearsloth <comedian_vs_clown@hotmail.com> Co-authored-by: Southbridge <7013162+southbridge-fur@users.noreply.github.com> Co-authored-by: Vladislav Suchkov <20380250+murolem@users.noreply.github.com> Co-authored-by: Prole <172158352+Prole0@users.noreply.github.com> Co-authored-by: Unkn0wn_Gh0st <shadowstalkermll@gmail.com> Co-authored-by: 3nderall <101940324+3nderall@users.noreply.github.com> Co-authored-by: Radezolid <snappednexus@gmail.com> Co-authored-by: J <billsmith116@gmail.com> Co-authored-by: Ghagliiarghii <68826635+Ghagliiarghii@users.noreply.github.com> Co-authored-by: chromiumboy <50505512+chromiumboy@users.noreply.github.com> Co-authored-by: youtissoum <51883137+youtissoum@users.noreply.github.com> Co-authored-by: Minemoder5000 <minemoder50000@gmail.com> Co-authored-by: Spanky <scott@wearejacob.com> Co-authored-by: Spessmann <156740760+Spessmann@users.noreply.github.com> Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com> Co-authored-by: brainfood1183 <113240905+brainfood1183@users.noreply.github.com> Co-authored-by: Deerstop <edainturner@gmail.com> Co-authored-by: B_Kirill <153602297+B-Kirill@users.noreply.github.com> Co-authored-by: archee1 <archee3@hotmail.co.uk> Co-authored-by: Cojoke <83733158+Cojoke-dot@users.noreply.github.com> Co-authored-by: Quantum-cross <7065792+Quantum-cross@users.noreply.github.com> Co-authored-by: poklj <compgeek223@gmail.com> Co-authored-by: Krunklehorn <42424291+Krunklehorn@users.noreply.github.com> Co-authored-by: OnyxTheBrave <131422822+OnyxTheBrave@users.noreply.github.com> Co-authored-by: UpAndLeaves <92269094+Alpha-Two@users.noreply.github.com> Co-authored-by: Flareguy <78941145+Flareguy@users.noreply.github.com> Co-authored-by: Zalycon <84675130+Zalycon@users.noreply.github.com> Co-authored-by: deltanedas <39013340+deltanedas@users.noreply.github.com> Co-authored-by: Verm <32827189+Vermidia@users.noreply.github.com> Co-authored-by: nikthechampiongr <32041239+nikthechampiongr@users.noreply.github.com> Co-authored-by: ScarKy0 <scarky0@onet.eu> Co-authored-by: Dmitry <57028746+dimm00n@users.noreply.github.com>
449 lines
16 KiB
C#
449 lines
16 KiB
C#
using System.Diagnostics.CodeAnalysis;
|
|
using System.Linq;
|
|
using Content.Client.DisplacementMap;
|
|
using Content.Client.Examine;
|
|
using Content.Client.Strip;
|
|
using Content.Client.Verbs.UI;
|
|
using Content.Shared.Hands;
|
|
using Content.Shared.Hands.Components;
|
|
using Content.Shared.Hands.EntitySystems;
|
|
using Content.Shared.Inventory.VirtualItem;
|
|
using Content.Shared.Item;
|
|
using JetBrains.Annotations;
|
|
using Robust.Client.GameObjects;
|
|
using Robust.Client.Player;
|
|
using Robust.Client.UserInterface;
|
|
using Robust.Shared.Containers;
|
|
using Robust.Shared.GameStates;
|
|
using Robust.Shared.Player;
|
|
using Robust.Shared.Timing;
|
|
|
|
namespace Content.Client.Hands.Systems
|
|
{
|
|
[UsedImplicitly]
|
|
public sealed class HandsSystem : SharedHandsSystem
|
|
{
|
|
[Dependency] private readonly IPlayerManager _playerManager = default!;
|
|
[Dependency] private readonly IUserInterfaceManager _ui = default!;
|
|
|
|
[Dependency] private readonly SharedContainerSystem _containerSystem = default!;
|
|
[Dependency] private readonly StrippableSystem _stripSys = default!;
|
|
[Dependency] private readonly ExamineSystem _examine = default!;
|
|
[Dependency] private readonly DisplacementMapSystem _displacement = default!;
|
|
|
|
public event Action<string, HandLocation>? OnPlayerAddHand;
|
|
public event Action<string>? OnPlayerRemoveHand;
|
|
public event Action<string?>? OnPlayerSetActiveHand;
|
|
public event Action<HandsComponent>? OnPlayerHandsAdded;
|
|
public event Action? OnPlayerHandsRemoved;
|
|
public event Action<string, EntityUid>? OnPlayerItemAdded;
|
|
public event Action<string, EntityUid>? OnPlayerItemRemoved;
|
|
public event Action<string>? OnPlayerHandBlocked;
|
|
public event Action<string>? OnPlayerHandUnblocked;
|
|
|
|
public override void Initialize()
|
|
{
|
|
base.Initialize();
|
|
|
|
SubscribeLocalEvent<HandsComponent, LocalPlayerAttachedEvent>(HandlePlayerAttached);
|
|
SubscribeLocalEvent<HandsComponent, LocalPlayerDetachedEvent>(HandlePlayerDetached);
|
|
SubscribeLocalEvent<HandsComponent, ComponentStartup>(OnHandsStartup);
|
|
SubscribeLocalEvent<HandsComponent, ComponentShutdown>(OnHandsShutdown);
|
|
SubscribeLocalEvent<HandsComponent, ComponentHandleState>(HandleComponentState);
|
|
SubscribeLocalEvent<HandsComponent, VisualsChangedEvent>(OnVisualsChanged);
|
|
|
|
OnHandSetActive += OnHandActivated;
|
|
}
|
|
|
|
#region StateHandling
|
|
private void HandleComponentState(EntityUid uid, HandsComponent component, ref ComponentHandleState args)
|
|
{
|
|
if (args.Current is not HandsComponentState state)
|
|
return;
|
|
|
|
var handsModified = component.Hands.Count != state.Hands.Count;
|
|
// we need to check that, even if we have the same amount, that the individual hands didn't change.
|
|
if (!handsModified)
|
|
{
|
|
foreach (var hand in component.Hands.Values)
|
|
{
|
|
if (state.Hands.Contains(hand))
|
|
continue;
|
|
handsModified = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
var manager = EnsureComp<ContainerManagerComponent>(uid);
|
|
|
|
if (handsModified)
|
|
{
|
|
List<Hand> addedHands = new();
|
|
foreach (var hand in state.Hands)
|
|
{
|
|
if (component.Hands.ContainsKey(hand.Name))
|
|
continue;
|
|
|
|
var container = _containerSystem.EnsureContainer<ContainerSlot>(uid, hand.Name, manager);
|
|
var newHand = new Hand(hand.Name, hand.Location, container);
|
|
component.Hands.Add(hand.Name, newHand);
|
|
addedHands.Add(newHand);
|
|
}
|
|
|
|
foreach (var name in component.Hands.Keys)
|
|
{
|
|
if (!state.HandNames.Contains(name))
|
|
{
|
|
RemoveHand(uid, name, component);
|
|
}
|
|
}
|
|
|
|
component.SortedHands.Clear();
|
|
component.SortedHands.AddRange(state.HandNames);
|
|
var sorted = addedHands.OrderBy(hand => component.SortedHands.IndexOf(hand.Name));
|
|
|
|
foreach (var hand in sorted)
|
|
{
|
|
AddHand(uid, hand, component);
|
|
}
|
|
}
|
|
|
|
_stripSys.UpdateUi(uid);
|
|
|
|
if (component.ActiveHand == null && state.ActiveHand == null)
|
|
return; //edge case
|
|
|
|
if (component.ActiveHand != null && state.ActiveHand != component.ActiveHand.Name)
|
|
{
|
|
SetActiveHand(uid, component.Hands[state.ActiveHand!], component);
|
|
}
|
|
}
|
|
#endregion
|
|
|
|
public void ReloadHandButtons()
|
|
{
|
|
if (!TryGetPlayerHands(out var hands))
|
|
{
|
|
return;
|
|
}
|
|
|
|
OnPlayerHandsAdded?.Invoke(hands);
|
|
}
|
|
|
|
public override void DoDrop(EntityUid uid, Hand hand, bool doDropInteraction = true, HandsComponent? hands = null, bool log = true)
|
|
{
|
|
base.DoDrop(uid, hand, doDropInteraction, hands, log);
|
|
|
|
if (TryComp(hand.HeldEntity, out SpriteComponent? sprite))
|
|
sprite.RenderOrder = EntityManager.CurrentTick.Value;
|
|
}
|
|
|
|
public EntityUid? GetActiveHandEntity()
|
|
{
|
|
return TryGetPlayerHands(out var hands) ? hands.ActiveHandEntity : null;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get the hands component of the local player
|
|
/// </summary>
|
|
public bool TryGetPlayerHands([NotNullWhen(true)] out HandsComponent? hands)
|
|
{
|
|
var player = _playerManager.LocalEntity;
|
|
hands = null;
|
|
return player != null && TryComp(player.Value, out hands);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Called when a user clicked on their hands GUI
|
|
/// </summary>
|
|
public void UIHandClick(HandsComponent hands, string handName)
|
|
{
|
|
if (!hands.Hands.TryGetValue(handName, out var pressedHand))
|
|
return;
|
|
|
|
if (hands.ActiveHand == null)
|
|
return;
|
|
|
|
var pressedEntity = pressedHand.HeldEntity;
|
|
var activeEntity = hands.ActiveHand.HeldEntity;
|
|
|
|
if (pressedHand == hands.ActiveHand && activeEntity != null)
|
|
{
|
|
// use item in hand
|
|
// it will always be attack_self() in my heart.
|
|
EntityManager.RaisePredictiveEvent(new RequestUseInHandEvent());
|
|
return;
|
|
}
|
|
|
|
if (pressedHand != hands.ActiveHand && pressedEntity == null)
|
|
{
|
|
// change active hand
|
|
EntityManager.RaisePredictiveEvent(new RequestSetHandEvent(handName));
|
|
return;
|
|
}
|
|
|
|
if (pressedHand != hands.ActiveHand && pressedEntity != null && activeEntity != null)
|
|
{
|
|
// use active item on held item
|
|
EntityManager.RaisePredictiveEvent(new RequestHandInteractUsingEvent(pressedHand.Name));
|
|
return;
|
|
}
|
|
|
|
if (pressedHand != hands.ActiveHand && pressedEntity != null && activeEntity == null)
|
|
{
|
|
// move the item to the active hand
|
|
EntityManager.RaisePredictiveEvent(new RequestMoveHandItemEvent(pressedHand.Name));
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Called when a user clicks on the little "activation" icon in the hands GUI. This is currently only used
|
|
/// by storage (backpacks, etc).
|
|
/// </summary>
|
|
public void UIHandActivate(string handName)
|
|
{
|
|
EntityManager.RaisePredictiveEvent(new RequestActivateInHandEvent(handName));
|
|
}
|
|
|
|
public void UIInventoryExamine(string handName)
|
|
{
|
|
if (!TryGetPlayerHands(out var hands) ||
|
|
!hands.Hands.TryGetValue(handName, out var hand) ||
|
|
hand.HeldEntity is not { Valid: true } entity)
|
|
{
|
|
return;
|
|
}
|
|
|
|
_examine.DoExamine(entity);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Called when a user clicks on the little "activation" icon in the hands GUI. This is currently only used
|
|
/// by storage (backpacks, etc).
|
|
/// </summary>
|
|
public void UIHandOpenContextMenu(string handName)
|
|
{
|
|
if (!TryGetPlayerHands(out var hands) ||
|
|
!hands.Hands.TryGetValue(handName, out var hand) ||
|
|
hand.HeldEntity is not { Valid: true } entity)
|
|
{
|
|
return;
|
|
}
|
|
|
|
_ui.GetUIController<VerbMenuUIController>().OpenVerbMenu(entity);
|
|
}
|
|
|
|
public void UIHandAltActivateItem(string handName)
|
|
{
|
|
RaisePredictiveEvent(new RequestHandAltInteractEvent(handName));
|
|
}
|
|
|
|
#region visuals
|
|
|
|
protected override void HandleEntityInserted(EntityUid uid, HandsComponent hands, EntInsertedIntoContainerMessage args)
|
|
{
|
|
base.HandleEntityInserted(uid, hands, args);
|
|
|
|
if (!hands.Hands.TryGetValue(args.Container.ID, out var hand))
|
|
return;
|
|
UpdateHandVisuals(uid, args.Entity, hand);
|
|
_stripSys.UpdateUi(uid);
|
|
|
|
if (uid != _playerManager.LocalEntity)
|
|
return;
|
|
|
|
OnPlayerItemAdded?.Invoke(hand.Name, args.Entity);
|
|
|
|
if (HasComp<VirtualItemComponent>(args.Entity))
|
|
OnPlayerHandBlocked?.Invoke(hand.Name);
|
|
}
|
|
|
|
protected override void HandleEntityRemoved(EntityUid uid, HandsComponent hands, EntRemovedFromContainerMessage args)
|
|
{
|
|
base.HandleEntityRemoved(uid, hands, args);
|
|
|
|
if (!hands.Hands.TryGetValue(args.Container.ID, out var hand))
|
|
return;
|
|
UpdateHandVisuals(uid, args.Entity, hand);
|
|
_stripSys.UpdateUi(uid);
|
|
|
|
if (uid != _playerManager.LocalEntity)
|
|
return;
|
|
|
|
OnPlayerItemRemoved?.Invoke(hand.Name, args.Entity);
|
|
|
|
if (HasComp<VirtualItemComponent>(args.Entity))
|
|
OnPlayerHandUnblocked?.Invoke(hand.Name);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Update the players sprite with new in-hand visuals.
|
|
/// </summary>
|
|
private void UpdateHandVisuals(EntityUid uid, EntityUid held, Hand hand, HandsComponent? handComp = null, SpriteComponent? sprite = null)
|
|
{
|
|
if (!Resolve(uid, ref handComp, ref sprite, false))
|
|
return;
|
|
|
|
// visual update might involve changes to the entity's effective sprite -> need to update hands GUI.
|
|
if (uid == _playerManager.LocalEntity)
|
|
OnPlayerItemAdded?.Invoke(hand.Name, held);
|
|
|
|
if (!handComp.ShowInHands)
|
|
return;
|
|
|
|
// Remove old layers. We could also just set them to invisible, but as items may add arbitrary layers, this
|
|
// may eventually bloat the player with lots of layers.
|
|
if (handComp.RevealedLayers.TryGetValue(hand.Location, out var revealedLayers))
|
|
{
|
|
foreach (var key in revealedLayers)
|
|
{
|
|
sprite.RemoveLayer(key);
|
|
}
|
|
|
|
revealedLayers.Clear();
|
|
}
|
|
else
|
|
{
|
|
revealedLayers = new();
|
|
handComp.RevealedLayers[hand.Location] = revealedLayers;
|
|
}
|
|
|
|
if (hand.HeldEntity == null)
|
|
{
|
|
// the held item was removed.
|
|
RaiseLocalEvent(held, new HeldVisualsUpdatedEvent(uid, revealedLayers), true);
|
|
return;
|
|
}
|
|
|
|
var ev = new GetInhandVisualsEvent(uid, hand.Location);
|
|
RaiseLocalEvent(held, ev);
|
|
|
|
if (ev.Layers.Count == 0)
|
|
{
|
|
RaiseLocalEvent(held, new HeldVisualsUpdatedEvent(uid, revealedLayers), true);
|
|
return;
|
|
}
|
|
|
|
// add the new layers
|
|
foreach (var (key, layerData) in ev.Layers)
|
|
{
|
|
if (!revealedLayers.Add(key))
|
|
{
|
|
Log.Warning($"Duplicate key for in-hand visuals: {key}. Are multiple components attempting to modify the same layer? Entity: {ToPrettyString(held)}");
|
|
continue;
|
|
}
|
|
|
|
var index = sprite.LayerMapReserveBlank(key);
|
|
|
|
// In case no RSI is given, use the item's base RSI as a default. This cuts down on a lot of unnecessary yaml entries.
|
|
if (layerData.RsiPath == null
|
|
&& layerData.TexturePath == null
|
|
&& sprite[index].Rsi == null)
|
|
{
|
|
if (TryComp<ItemComponent>(held, out var itemComponent) && itemComponent.RsiPath != null)
|
|
sprite.LayerSetRSI(index, itemComponent.RsiPath);
|
|
else if (TryComp(held, out SpriteComponent? clothingSprite))
|
|
sprite.LayerSetRSI(index, clothingSprite.BaseRSI);
|
|
}
|
|
|
|
sprite.LayerSetData(index, layerData);
|
|
|
|
// Add displacement maps
|
|
var displacement = hand.Location switch
|
|
{
|
|
HandLocation.Left => handComp.LeftHandDisplacement,
|
|
HandLocation.Right => handComp.RightHandDisplacement,
|
|
_ => handComp.HandDisplacement
|
|
};
|
|
|
|
if (displacement is not null && _displacement.TryAddDisplacement(displacement, sprite, index, key, out var displacementKey))
|
|
revealedLayers.Add(displacementKey);
|
|
}
|
|
|
|
RaiseLocalEvent(held, new HeldVisualsUpdatedEvent(uid, revealedLayers), true);
|
|
}
|
|
|
|
private void OnVisualsChanged(EntityUid uid, HandsComponent component, VisualsChangedEvent args)
|
|
{
|
|
// update hands visuals if this item is in a hand (rather then inventory or other container).
|
|
if (component.Hands.TryGetValue(args.ContainerId, out var hand))
|
|
{
|
|
UpdateHandVisuals(uid, GetEntity(args.Item), hand, component);
|
|
}
|
|
}
|
|
#endregion
|
|
|
|
#region Gui
|
|
|
|
private void HandlePlayerAttached(EntityUid uid, HandsComponent component, LocalPlayerAttachedEvent args)
|
|
{
|
|
OnPlayerHandsAdded?.Invoke(component);
|
|
}
|
|
|
|
private void HandlePlayerDetached(EntityUid uid, HandsComponent component, LocalPlayerDetachedEvent args)
|
|
{
|
|
OnPlayerHandsRemoved?.Invoke();
|
|
}
|
|
|
|
private void OnHandsStartup(EntityUid uid, HandsComponent component, ComponentStartup args)
|
|
{
|
|
if (_playerManager.LocalEntity == uid)
|
|
OnPlayerHandsAdded?.Invoke(component);
|
|
}
|
|
|
|
private void OnHandsShutdown(EntityUid uid, HandsComponent component, ComponentShutdown args)
|
|
{
|
|
if (_playerManager.LocalEntity == uid)
|
|
OnPlayerHandsRemoved?.Invoke();
|
|
}
|
|
#endregion
|
|
|
|
private void AddHand(EntityUid uid, Hand newHand, HandsComponent? handsComp = null)
|
|
{
|
|
AddHand(uid, newHand.Name, newHand.Location, handsComp);
|
|
}
|
|
|
|
public override void AddHand(EntityUid uid, string handName, HandLocation handLocation, HandsComponent? handsComp = null)
|
|
{
|
|
base.AddHand(uid, handName, handLocation, handsComp);
|
|
|
|
if (uid == _playerManager.LocalEntity)
|
|
OnPlayerAddHand?.Invoke(handName, handLocation);
|
|
|
|
if (handsComp == null)
|
|
return;
|
|
|
|
if (handsComp.ActiveHand == null)
|
|
SetActiveHand(uid, handsComp.Hands[handName], handsComp);
|
|
}
|
|
public override void RemoveHand(EntityUid uid, string handName, HandsComponent? handsComp = null)
|
|
{
|
|
if (uid == _playerManager.LocalEntity && handsComp != null &&
|
|
handsComp.Hands.ContainsKey(handName) && uid ==
|
|
_playerManager.LocalEntity)
|
|
{
|
|
OnPlayerRemoveHand?.Invoke(handName);
|
|
}
|
|
|
|
base.RemoveHand(uid, handName, handsComp);
|
|
}
|
|
|
|
private void OnHandActivated(Entity<HandsComponent>? ent)
|
|
{
|
|
if (ent is not { } hand)
|
|
return;
|
|
|
|
if (_playerManager.LocalEntity != hand.Owner)
|
|
return;
|
|
|
|
if (hand.Comp.ActiveHand == null)
|
|
{
|
|
OnPlayerSetActiveHand?.Invoke(null);
|
|
return;
|
|
}
|
|
|
|
OnPlayerSetActiveHand?.Invoke(hand.Comp.ActiveHand.Name);
|
|
}
|
|
}
|
|
}
|