Make the material ejection control completely generic (#23308)
* Finish decoupling material ejection from lathes * commented
This commit is contained in:
@@ -1,8 +1,6 @@
|
|||||||
using Content.Shared.Lathe;
|
using Content.Shared.Lathe;
|
||||||
using Content.Shared.Materials;
|
|
||||||
using Content.Shared.Research.Components;
|
using Content.Shared.Research.Components;
|
||||||
using JetBrains.Annotations;
|
using JetBrains.Annotations;
|
||||||
using Robust.Client.GameObjects;
|
|
||||||
|
|
||||||
namespace Content.Client.Lathe.UI
|
namespace Content.Client.Lathe.UI
|
||||||
{
|
{
|
||||||
@@ -33,11 +31,6 @@ namespace Content.Client.Lathe.UI
|
|||||||
SendMessage(new LatheQueueRecipeMessage(recipe, amount));
|
SendMessage(new LatheQueueRecipeMessage(recipe, amount));
|
||||||
};
|
};
|
||||||
|
|
||||||
_menu.OnEjectPressed += (material, sheetsToExtract) =>
|
|
||||||
{
|
|
||||||
SendMessage(new EjectMaterialMessage(material, sheetsToExtract));
|
|
||||||
};
|
|
||||||
|
|
||||||
_menu.OpenCentered();
|
_menu.OpenCentered();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -51,7 +44,6 @@ namespace Content.Client.Lathe.UI
|
|||||||
if (_menu != null)
|
if (_menu != null)
|
||||||
_menu.Recipes = msg.Recipes;
|
_menu.Recipes = msg.Recipes;
|
||||||
_menu?.PopulateRecipes(Owner);
|
_menu?.PopulateRecipes(Owner);
|
||||||
_menu?.PopulateMaterials(Owner);
|
|
||||||
_menu?.PopulateQueueList(msg.Queue);
|
_menu?.PopulateQueueList(msg.Queue);
|
||||||
_menu?.SetQueueInfo(msg.CurrentlyProducing);
|
_menu?.SetQueueInfo(msg.CurrentlyProducing);
|
||||||
break;
|
break;
|
||||||
@@ -64,7 +56,6 @@ namespace Content.Client.Lathe.UI
|
|||||||
if (!disposing)
|
if (!disposing)
|
||||||
return;
|
return;
|
||||||
_menu?.Dispose();
|
_menu?.Dispose();
|
||||||
//thom _materialsEjectionMenu?.Dispose();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,73 +0,0 @@
|
|||||||
using Content.Client.Stylesheets;
|
|
||||||
using Content.Shared.Materials;
|
|
||||||
using Robust.Client.AutoGenerated;
|
|
||||||
using Robust.Client.UserInterface;
|
|
||||||
using Robust.Client.UserInterface.Controls;
|
|
||||||
using Robust.Client.UserInterface.XAML;
|
|
||||||
using Robust.Shared.Prototypes;
|
|
||||||
|
|
||||||
namespace Content.Client.Lathe.UI;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// This widget is one row in the lathe eject menu.
|
|
||||||
/// </summary>
|
|
||||||
[GenerateTypedNameReferences]
|
|
||||||
public sealed partial class LatheMaterialEjector : PanelContainer
|
|
||||||
{
|
|
||||||
[Dependency] private readonly IPrototypeManager _prototypeManager = default!;
|
|
||||||
|
|
||||||
public string Material;
|
|
||||||
public Action<string, int>? OnEjectPressed;
|
|
||||||
public int VolumePerSheet;
|
|
||||||
|
|
||||||
public LatheMaterialEjector(string material, Action<string, int>? onEjectPressed, int volumePerSheet, int maxEjectableSheets)
|
|
||||||
{
|
|
||||||
RobustXamlLoader.Load(this);
|
|
||||||
IoCManager.InjectDependencies(this);
|
|
||||||
|
|
||||||
Material = material;
|
|
||||||
OnEjectPressed = onEjectPressed;
|
|
||||||
VolumePerSheet = volumePerSheet;
|
|
||||||
|
|
||||||
PopulateButtons(maxEjectableSheets);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void PopulateButtons(int maxEjectableSheets)
|
|
||||||
{
|
|
||||||
int[] sheetsToEjectArray = { 1, 5, 10 };
|
|
||||||
|
|
||||||
for (var i = 0; i < sheetsToEjectArray.Length; i++)
|
|
||||||
{
|
|
||||||
var sheetsToEject = sheetsToEjectArray[i];
|
|
||||||
|
|
||||||
var styleClass = StyleBase.ButtonOpenBoth;
|
|
||||||
if (i == 0)
|
|
||||||
styleClass = StyleBase.ButtonOpenRight;
|
|
||||||
else if (i == sheetsToEjectArray.Length - 1)
|
|
||||||
styleClass = StyleBase.ButtonOpenLeft;
|
|
||||||
|
|
||||||
var button = new Button
|
|
||||||
{
|
|
||||||
Name = $"{sheetsToEject}",
|
|
||||||
Access = AccessLevel.Public,
|
|
||||||
Text = Loc.GetString($"{sheetsToEject}"),
|
|
||||||
MinWidth = 45,
|
|
||||||
StyleClasses = { styleClass }
|
|
||||||
};
|
|
||||||
|
|
||||||
button.OnPressed += _ =>
|
|
||||||
{
|
|
||||||
OnEjectPressed?.Invoke(Material, sheetsToEject);
|
|
||||||
};
|
|
||||||
|
|
||||||
button.Disabled = maxEjectableSheets < sheetsToEject;
|
|
||||||
|
|
||||||
if (_prototypeManager.TryIndex<MaterialPrototype>(Material, out var proto))
|
|
||||||
{
|
|
||||||
button.ToolTip = Loc.GetString("lathe-menu-tooltip-display", ("amount", sheetsToEject), ("material", Loc.GetString(proto.Name)));
|
|
||||||
}
|
|
||||||
|
|
||||||
Content.AddChild(button);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
<DefaultWindow
|
<DefaultWindow
|
||||||
xmlns="https://spacestation14.io"
|
xmlns="https://spacestation14.io"
|
||||||
xmlns:gfx="clr-namespace:Robust.Client.Graphics;assembly=Robust.Client"
|
xmlns:gfx="clr-namespace:Robust.Client.Graphics;assembly=Robust.Client"
|
||||||
|
xmlns:ui="clr-namespace:Content.Client.Materials.UI"
|
||||||
Title="{Loc 'lathe-menu-title'}"
|
Title="{Loc 'lathe-menu-title'}"
|
||||||
MinSize="550 450"
|
MinSize="550 450"
|
||||||
SetSize="750 500">
|
SetSize="750 500">
|
||||||
@@ -135,14 +136,7 @@
|
|||||||
Orientation="Vertical"
|
Orientation="Vertical"
|
||||||
VerticalExpand="True"
|
VerticalExpand="True"
|
||||||
HorizontalExpand="True">
|
HorizontalExpand="True">
|
||||||
<BoxContainer
|
<ui:MaterialStorageControl Name="MaterialsList" SizeFlagsStretchRatio="8"/>
|
||||||
Name="MaterialsList"
|
|
||||||
Orientation="Vertical"
|
|
||||||
SizeFlagsStretchRatio="8"
|
|
||||||
HorizontalExpand="True"
|
|
||||||
VerticalExpand="True">
|
|
||||||
<!-- Materials populated in C# file -->
|
|
||||||
</BoxContainer>
|
|
||||||
</BoxContainer>
|
</BoxContainer>
|
||||||
</BoxContainer>
|
</BoxContainer>
|
||||||
</BoxContainer>
|
</BoxContainer>
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
|
using Content.Client.Materials;
|
||||||
using Content.Shared.Lathe;
|
using Content.Shared.Lathe;
|
||||||
using Content.Shared.Materials;
|
using Content.Shared.Materials;
|
||||||
using Content.Shared.Research.Prototypes;
|
using Content.Shared.Research.Prototypes;
|
||||||
@@ -19,16 +20,12 @@ public sealed partial class LatheMenu : DefaultWindow
|
|||||||
[Dependency] private readonly IPrototypeManager _prototypeManager = default!;
|
[Dependency] private readonly IPrototypeManager _prototypeManager = default!;
|
||||||
private readonly SpriteSystem _spriteSystem;
|
private readonly SpriteSystem _spriteSystem;
|
||||||
private readonly LatheSystem _lathe;
|
private readonly LatheSystem _lathe;
|
||||||
|
private readonly MaterialStorageSystem _materialStorage;
|
||||||
|
|
||||||
public event Action<BaseButton.ButtonEventArgs>? OnServerListButtonPressed;
|
public event Action<BaseButton.ButtonEventArgs>? OnServerListButtonPressed;
|
||||||
public event Action<string, int>? RecipeQueueAction;
|
public event Action<string, int>? RecipeQueueAction;
|
||||||
public event Action<string, int>? OnEjectPressed;
|
|
||||||
public List<ProtoId<LatheRecipePrototype>> Recipes = new();
|
|
||||||
|
|
||||||
/// <summary>
|
public List<ProtoId<LatheRecipePrototype>> Recipes = new();
|
||||||
/// Default volume for a sheet if the material's entity prototype has no material composition.
|
|
||||||
/// </summary>
|
|
||||||
private const int DEFAULT_SHEET_VOLUME = 100;
|
|
||||||
|
|
||||||
public LatheMenu(LatheBoundUserInterface owner)
|
public LatheMenu(LatheBoundUserInterface owner)
|
||||||
{
|
{
|
||||||
@@ -37,6 +34,7 @@ public sealed partial class LatheMenu : DefaultWindow
|
|||||||
|
|
||||||
_spriteSystem = _entityManager.System<SpriteSystem>();
|
_spriteSystem = _entityManager.System<SpriteSystem>();
|
||||||
_lathe = _entityManager.System<LatheSystem>();
|
_lathe = _entityManager.System<LatheSystem>();
|
||||||
|
_materialStorage = _entityManager.System<MaterialStorageSystem>();
|
||||||
|
|
||||||
Title = _entityManager.GetComponent<MetaDataComponent>(owner.Owner).EntityName;
|
Title = _entityManager.GetComponent<MetaDataComponent>(owner.Owner).EntityName;
|
||||||
|
|
||||||
@@ -58,50 +56,10 @@ public sealed partial class LatheMenu : DefaultWindow
|
|||||||
ServerListButton.Visible = false;
|
ServerListButton.Visible = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
MaterialsList.SetOwner(owner.Owner);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void PopulateMaterials(EntityUid lathe)
|
|
||||||
{
|
|
||||||
if (!_entityManager.TryGetComponent<MaterialStorageComponent>(lathe, out var materials))
|
|
||||||
return;
|
|
||||||
|
|
||||||
MaterialsList.DisposeAllChildren();
|
|
||||||
|
|
||||||
foreach (var (materialId, volume) in materials.Storage)
|
|
||||||
{
|
|
||||||
if (volume <= 0)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if (!_prototypeManager.TryIndex(materialId, out MaterialPrototype? material))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
var sheetVolume = SheetVolume(material);
|
|
||||||
var sheets = (float) volume / sheetVolume;
|
|
||||||
var maxEjectableSheets = (int) MathF.Floor(sheets);
|
|
||||||
|
|
||||||
var unit = Loc.GetString(material.Unit);
|
|
||||||
var amountText = Loc.GetString("lathe-menu-material-amount", ("amount", sheets), ("unit", unit));
|
|
||||||
var name = Loc.GetString(material.Name);
|
|
||||||
var mat = Loc.GetString("lathe-menu-material-display", ("material", name), ("amount", amountText));
|
|
||||||
|
|
||||||
var row = new LatheMaterialEjector(materialId, OnEjectPressed, sheetVolume, maxEjectableSheets)
|
|
||||||
{
|
|
||||||
Icon = { Texture = _spriteSystem.Frame0(material.Icon) },
|
|
||||||
ProductName = { Text = mat }
|
|
||||||
};
|
|
||||||
|
|
||||||
MaterialsList.AddChild(row);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (MaterialsList.ChildCount == 0)
|
|
||||||
{
|
|
||||||
var noMaterialsMsg = Loc.GetString("lathe-menu-no-materials-message");
|
|
||||||
var noItemRow = new Label();
|
|
||||||
noItemRow.Text = noMaterialsMsg;
|
|
||||||
noItemRow.Align = Label.AlignMode.Center;
|
|
||||||
MaterialsList.AddChild(noItemRow);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Populates the list of all the recipes
|
/// Populates the list of all the recipes
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -147,7 +105,7 @@ public sealed partial class LatheMenu : DefaultWindow
|
|||||||
sb.Append('\n');
|
sb.Append('\n');
|
||||||
|
|
||||||
var adjustedAmount = SharedLatheSystem.AdjustMaterial(amount, prototype.ApplyMaterialDiscount, component.MaterialUseMultiplier);
|
var adjustedAmount = SharedLatheSystem.AdjustMaterial(amount, prototype.ApplyMaterialDiscount, component.MaterialUseMultiplier);
|
||||||
var sheetVolume = SheetVolume(proto);
|
var sheetVolume = _materialStorage.GetSheetVolume(proto);
|
||||||
|
|
||||||
var unit = Loc.GetString(proto.Unit);
|
var unit = Loc.GetString(proto.Unit);
|
||||||
// rounded in locale not here
|
// rounded in locale not here
|
||||||
@@ -204,17 +162,4 @@ public sealed partial class LatheMenu : DefaultWindow
|
|||||||
: _spriteSystem.Frame0(recipe.Icon);
|
: _spriteSystem.Frame0(recipe.Icon);
|
||||||
NameLabel.Text = $"{recipe.Name}";
|
NameLabel.Text = $"{recipe.Name}";
|
||||||
}
|
}
|
||||||
|
|
||||||
private int SheetVolume(MaterialPrototype material)
|
|
||||||
{
|
|
||||||
if (material.StackEntity == null)
|
|
||||||
return DEFAULT_SHEET_VOLUME;
|
|
||||||
|
|
||||||
var proto = _prototypeManager.Index<EntityPrototype>(material.StackEntity);
|
|
||||||
|
|
||||||
if (!proto.TryGetComponent<PhysicalCompositionComponent>(out var composition))
|
|
||||||
return DEFAULT_SHEET_VOLUME;
|
|
||||||
|
|
||||||
return composition.MaterialComposition.FirstOrDefault(kvp => kvp.Key == material.ID).Value;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,8 +1,6 @@
|
|||||||
<PanelContainer xmlns="https://spacestation14.io"
|
<PanelContainer xmlns="https://spacestation14.io"
|
||||||
HorizontalExpand="True">
|
HorizontalExpand="True">
|
||||||
<BoxContainer Name="Content"
|
<BoxContainer Orientation="Horizontal">
|
||||||
Orientation="Horizontal"
|
|
||||||
HorizontalExpand="True">
|
|
||||||
<TextureRect Name="Icon"
|
<TextureRect Name="Icon"
|
||||||
Access="Public"
|
Access="Public"
|
||||||
MinSize="32 32"
|
MinSize="32 32"
|
||||||
@@ -16,6 +14,11 @@
|
|||||||
ClipText="True"
|
ClipText="True"
|
||||||
Margin="4 4 4 4"/>
|
Margin="4 4 4 4"/>
|
||||||
</BoxContainer>
|
</BoxContainer>
|
||||||
<!--Here go buttons which added in c#-->
|
<BoxContainer Name="Content"
|
||||||
|
Orientation="Horizontal"
|
||||||
|
HorizontalExpand="True"
|
||||||
|
HorizontalAlignment="Right">
|
||||||
|
<!--Here go buttons which added in c#-->
|
||||||
|
</BoxContainer>
|
||||||
</BoxContainer>
|
</BoxContainer>
|
||||||
</PanelContainer>
|
</PanelContainer>
|
||||||
110
Content.Client/Materials/UI/MaterialDisplay.xaml.cs
Normal file
110
Content.Client/Materials/UI/MaterialDisplay.xaml.cs
Normal file
@@ -0,0 +1,110 @@
|
|||||||
|
using Content.Client.Stylesheets;
|
||||||
|
using Content.Shared.Materials;
|
||||||
|
using Robust.Client.AutoGenerated;
|
||||||
|
using Robust.Client.GameObjects;
|
||||||
|
using Robust.Client.UserInterface;
|
||||||
|
using Robust.Client.UserInterface.Controls;
|
||||||
|
using Robust.Client.UserInterface.XAML;
|
||||||
|
using Robust.Shared.Prototypes;
|
||||||
|
|
||||||
|
namespace Content.Client.Materials.UI;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// This widget is one row in the material storage control.
|
||||||
|
/// </summary>
|
||||||
|
[GenerateTypedNameReferences]
|
||||||
|
public sealed partial class MaterialDisplay : PanelContainer
|
||||||
|
{
|
||||||
|
[Dependency] private readonly IPrototypeManager _prototypeManager = default!;
|
||||||
|
[Dependency] private readonly IEntityManager _entityManager = default!;
|
||||||
|
|
||||||
|
private readonly MaterialStorageSystem _materialStorage;
|
||||||
|
|
||||||
|
private readonly MaterialStorageUIController _materialUIController;
|
||||||
|
|
||||||
|
private int _volume;
|
||||||
|
private readonly EntityUid _ent;
|
||||||
|
public readonly string Material;
|
||||||
|
private readonly bool _canEject;
|
||||||
|
|
||||||
|
public MaterialDisplay(EntityUid ent, string material, int volume, bool canEject)
|
||||||
|
{
|
||||||
|
RobustXamlLoader.Load(this);
|
||||||
|
IoCManager.InjectDependencies(this);
|
||||||
|
|
||||||
|
_materialStorage = _entityManager.System<MaterialStorageSystem>();
|
||||||
|
_materialUIController = UserInterfaceManager.GetUIController<MaterialStorageUIController>();
|
||||||
|
|
||||||
|
var spriteSys = _entityManager.System<SpriteSystem>();
|
||||||
|
Icon.Texture = spriteSys.Frame0(_prototypeManager.Index<MaterialPrototype>(material).Icon);
|
||||||
|
|
||||||
|
_ent = ent;
|
||||||
|
Material = material;
|
||||||
|
_canEject = canEject;
|
||||||
|
UpdateVolume(volume);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UpdateVolume(int volume)
|
||||||
|
{
|
||||||
|
if (_volume == volume)
|
||||||
|
return;
|
||||||
|
|
||||||
|
_volume = volume;
|
||||||
|
var matProto = _prototypeManager.Index<MaterialPrototype>(Material);
|
||||||
|
|
||||||
|
var sheetVolume = _materialStorage.GetSheetVolume(matProto);
|
||||||
|
var sheets = (float) volume / sheetVolume;
|
||||||
|
var maxEjectableSheets = (int) MathF.Floor(sheets);
|
||||||
|
|
||||||
|
var unit = Loc.GetString(matProto.Unit);
|
||||||
|
var amountText = Loc.GetString("lathe-menu-material-amount", ("amount", sheets), ("unit", unit));
|
||||||
|
var name = Loc.GetString(matProto.Name);
|
||||||
|
var mat = Loc.GetString("lathe-menu-material-display", ("material", name), ("amount", amountText));
|
||||||
|
ProductName.Text = mat;
|
||||||
|
|
||||||
|
PopulateButtons(maxEjectableSheets);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void PopulateButtons(int maxEjectableSheets)
|
||||||
|
{
|
||||||
|
Content.Children.Clear();
|
||||||
|
if (!_canEject)
|
||||||
|
return;
|
||||||
|
|
||||||
|
int[] sheetsToEjectArray = { 1, 5, 10 };
|
||||||
|
|
||||||
|
for (var i = 0; i < sheetsToEjectArray.Length; i++)
|
||||||
|
{
|
||||||
|
var sheetsToEject = sheetsToEjectArray[i];
|
||||||
|
|
||||||
|
var styleClass = StyleBase.ButtonOpenBoth;
|
||||||
|
if (i == 0)
|
||||||
|
styleClass = StyleBase.ButtonOpenRight;
|
||||||
|
else if (i == sheetsToEjectArray.Length - 1)
|
||||||
|
styleClass = StyleBase.ButtonOpenLeft;
|
||||||
|
|
||||||
|
var button = new Button
|
||||||
|
{
|
||||||
|
Name = $"{sheetsToEject}",
|
||||||
|
Access = AccessLevel.Public,
|
||||||
|
Text = Loc.GetString($"{sheetsToEject}"),
|
||||||
|
MinWidth = 45,
|
||||||
|
StyleClasses = { styleClass }
|
||||||
|
};
|
||||||
|
|
||||||
|
button.OnPressed += _ =>
|
||||||
|
{
|
||||||
|
_materialUIController.SendLatheEjectMessage(_ent, Material, sheetsToEject);
|
||||||
|
};
|
||||||
|
|
||||||
|
button.Disabled = maxEjectableSheets < sheetsToEject;
|
||||||
|
|
||||||
|
if (_prototypeManager.TryIndex<MaterialPrototype>(Material, out var proto))
|
||||||
|
{
|
||||||
|
button.ToolTip = Loc.GetString("lathe-menu-tooltip-display", ("amount", sheetsToEject), ("material", Loc.GetString(proto.Name)));
|
||||||
|
}
|
||||||
|
|
||||||
|
Content.AddChild(button);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
7
Content.Client/Materials/UI/MaterialStorageControl.xaml
Normal file
7
Content.Client/Materials/UI/MaterialStorageControl.xaml
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
<BoxContainer xmlns="https://spacestation14.io"
|
||||||
|
Orientation="Vertical"
|
||||||
|
SizeFlagsStretchRatio="8"
|
||||||
|
HorizontalExpand="True"
|
||||||
|
VerticalExpand="True">
|
||||||
|
<Label Name="NoMatsLabel" Text="{Loc 'lathe-menu-no-materials-message'}" Align="Center"/>
|
||||||
|
</BoxContainer>
|
||||||
92
Content.Client/Materials/UI/MaterialStorageControl.xaml.cs
Normal file
92
Content.Client/Materials/UI/MaterialStorageControl.xaml.cs
Normal file
@@ -0,0 +1,92 @@
|
|||||||
|
using System.Linq;
|
||||||
|
using Content.Shared.Materials;
|
||||||
|
using Robust.Client.AutoGenerated;
|
||||||
|
using Robust.Client.UserInterface.Controls;
|
||||||
|
using Robust.Client.UserInterface.XAML;
|
||||||
|
using Robust.Shared.Timing;
|
||||||
|
|
||||||
|
namespace Content.Client.Materials.UI;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// This widget is one row in the lathe eject menu.
|
||||||
|
/// </summary>
|
||||||
|
[GenerateTypedNameReferences]
|
||||||
|
public sealed partial class MaterialStorageControl : BoxContainer
|
||||||
|
{
|
||||||
|
[Dependency] private readonly IEntityManager _entityManager = default!;
|
||||||
|
|
||||||
|
private EntityUid? _owner;
|
||||||
|
|
||||||
|
private Dictionary<string, int> _currentMaterials = new();
|
||||||
|
|
||||||
|
public MaterialStorageControl()
|
||||||
|
{
|
||||||
|
RobustXamlLoader.Load(this);
|
||||||
|
IoCManager.InjectDependencies(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetOwner(EntityUid owner)
|
||||||
|
{
|
||||||
|
_owner = owner;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void FrameUpdate(FrameEventArgs args)
|
||||||
|
{
|
||||||
|
base.FrameUpdate(args);
|
||||||
|
|
||||||
|
if (_owner == null)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (!_entityManager.TryGetComponent<MaterialStorageComponent>(_owner, out var materialStorage))
|
||||||
|
{
|
||||||
|
Dispose();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var canEject = materialStorage.CanEjectStoredMaterials;
|
||||||
|
var mats = materialStorage.Storage.Select(pair => (pair.Key.Id, pair.Value)).ToDictionary();
|
||||||
|
if (_currentMaterials.Equals(mats))
|
||||||
|
return;
|
||||||
|
|
||||||
|
var missing = new List<string>();
|
||||||
|
var extra = new List<string>();
|
||||||
|
foreach (var (mat, amount) in mats)
|
||||||
|
{
|
||||||
|
if (!_currentMaterials.ContainsKey(mat) ||
|
||||||
|
_currentMaterials[mat] == 0 && _currentMaterials[mat] != amount)
|
||||||
|
missing.Add(mat);
|
||||||
|
}
|
||||||
|
foreach (var (mat, amount) in _currentMaterials)
|
||||||
|
{
|
||||||
|
if (!mats.ContainsKey(mat) || amount == 0)
|
||||||
|
extra.Add(mat);
|
||||||
|
}
|
||||||
|
|
||||||
|
var children = new List<MaterialDisplay>();
|
||||||
|
children.AddRange(Children.OfType<MaterialDisplay>());
|
||||||
|
|
||||||
|
foreach (var display in children)
|
||||||
|
{
|
||||||
|
var mat = display.Material;
|
||||||
|
|
||||||
|
if (extra.Contains(mat))
|
||||||
|
{
|
||||||
|
RemoveChild(display);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!mats.TryGetValue(mat, out var newAmount))
|
||||||
|
continue;
|
||||||
|
display.UpdateVolume(newAmount);
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var mat in missing)
|
||||||
|
{
|
||||||
|
var volume = mats[mat];
|
||||||
|
AddChild(new MaterialDisplay(_owner.Value, mat, volume, canEject));
|
||||||
|
}
|
||||||
|
|
||||||
|
_currentMaterials = mats;
|
||||||
|
NoMatsLabel.Visible = ChildCount == 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
12
Content.Client/Materials/UI/MaterialStorageUIController.cs
Normal file
12
Content.Client/Materials/UI/MaterialStorageUIController.cs
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
using Content.Shared.Materials;
|
||||||
|
using Robust.Client.UserInterface.Controllers;
|
||||||
|
|
||||||
|
namespace Content.Client.Materials.UI;
|
||||||
|
|
||||||
|
public sealed class MaterialStorageUIController : UIController
|
||||||
|
{
|
||||||
|
public void SendLatheEjectMessage(EntityUid uid, string material, int sheetsToEject)
|
||||||
|
{
|
||||||
|
EntityManager.RaisePredictiveEvent(new EjectMaterialMessage(EntityManager.GetNetEntity(uid), material, sheetsToEject));
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -5,10 +5,10 @@ using Content.Shared.Popups;
|
|||||||
using Content.Shared.Stacks;
|
using Content.Shared.Stacks;
|
||||||
using Content.Server.Power.Components;
|
using Content.Server.Power.Components;
|
||||||
using Content.Server.Stack;
|
using Content.Server.Stack;
|
||||||
|
using Content.Shared.ActionBlocker;
|
||||||
using Content.Shared.Construction;
|
using Content.Shared.Construction;
|
||||||
using Content.Shared.Database;
|
using Content.Shared.Database;
|
||||||
using JetBrains.Annotations;
|
using JetBrains.Annotations;
|
||||||
using Robust.Shared.Audio;
|
|
||||||
using Robust.Shared.Audio.Systems;
|
using Robust.Shared.Audio.Systems;
|
||||||
using Robust.Shared.Map;
|
using Robust.Shared.Map;
|
||||||
using Robust.Shared.Prototypes;
|
using Robust.Shared.Prototypes;
|
||||||
@@ -22,6 +22,7 @@ public sealed class MaterialStorageSystem : SharedMaterialStorageSystem
|
|||||||
{
|
{
|
||||||
[Dependency] private readonly IAdminLogManager _adminLogger = default!;
|
[Dependency] private readonly IAdminLogManager _adminLogger = default!;
|
||||||
[Dependency] private readonly IPrototypeManager _prototypeManager = default!;
|
[Dependency] private readonly IPrototypeManager _prototypeManager = default!;
|
||||||
|
[Dependency] private readonly ActionBlockerSystem _actionBlocker = default!;
|
||||||
[Dependency] private readonly SharedAudioSystem _audio = default!;
|
[Dependency] private readonly SharedAudioSystem _audio = default!;
|
||||||
[Dependency] private readonly SharedPopupSystem _popup = default!;
|
[Dependency] private readonly SharedPopupSystem _popup = default!;
|
||||||
[Dependency] private readonly StackSystem _stackSystem = default!;
|
[Dependency] private readonly StackSystem _stackSystem = default!;
|
||||||
@@ -31,7 +32,7 @@ public sealed class MaterialStorageSystem : SharedMaterialStorageSystem
|
|||||||
base.Initialize();
|
base.Initialize();
|
||||||
SubscribeLocalEvent<MaterialStorageComponent, MachineDeconstructedEvent>(OnDeconstructed);
|
SubscribeLocalEvent<MaterialStorageComponent, MachineDeconstructedEvent>(OnDeconstructed);
|
||||||
|
|
||||||
SubscribeLocalEvent<MaterialStorageComponent, EjectMaterialMessage>(OnEjectMessage);
|
SubscribeAllEvent<EjectMaterialMessage>(OnEjectMessage);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnDeconstructed(EntityUid uid, MaterialStorageComponent component, MachineDeconstructedEvent args)
|
private void OnDeconstructed(EntityUid uid, MaterialStorageComponent component, MachineDeconstructedEvent args)
|
||||||
@@ -45,9 +46,23 @@ public sealed class MaterialStorageSystem : SharedMaterialStorageSystem
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnEjectMessage(EntityUid uid, MaterialStorageComponent component, EjectMaterialMessage message)
|
private void OnEjectMessage(EjectMaterialMessage msg, EntitySessionEventArgs args)
|
||||||
{
|
{
|
||||||
if (!component.CanEjectStoredMaterials || !_prototypeManager.TryIndex<MaterialPrototype>(message.Material, out var material))
|
if (args.SenderSession.AttachedEntity is not { } player)
|
||||||
|
return;
|
||||||
|
|
||||||
|
var uid = GetEntity(msg.Entity);
|
||||||
|
|
||||||
|
if (!TryComp<MaterialStorageComponent>(uid, out var component))
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (!Exists(uid))
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (!_actionBlocker.CanInteract(player, uid))
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (!component.CanEjectStoredMaterials || !_prototypeManager.TryIndex<MaterialPrototype>(msg.Material, out var material))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
var volume = 0;
|
var volume = 0;
|
||||||
@@ -57,13 +72,13 @@ public sealed class MaterialStorageSystem : SharedMaterialStorageSystem
|
|||||||
if (!_prototypeManager.Index<EntityPrototype>(material.StackEntity).TryGetComponent<PhysicalCompositionComponent>(out var composition))
|
if (!_prototypeManager.Index<EntityPrototype>(material.StackEntity).TryGetComponent<PhysicalCompositionComponent>(out var composition))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
var volumePerSheet = composition.MaterialComposition.FirstOrDefault(kvp => kvp.Key == message.Material).Value;
|
var volumePerSheet = composition.MaterialComposition.FirstOrDefault(kvp => kvp.Key == msg.Material).Value;
|
||||||
var sheetsToExtract = Math.Min(message.SheetsToExtract, _stackSystem.GetMaxCount(material.StackEntity));
|
var sheetsToExtract = Math.Min(msg.SheetsToExtract, _stackSystem.GetMaxCount(material.StackEntity));
|
||||||
|
|
||||||
volume = sheetsToExtract * volumePerSheet;
|
volume = sheetsToExtract * volumePerSheet;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (volume <= 0 || !TryChangeMaterialAmount(uid, message.Material, -volume))
|
if (volume <= 0 || !TryChangeMaterialAmount(uid, msg.Material, -volume))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
var mats = SpawnMultipleFromMaterial(volume, material, Transform(uid).Coordinates, out _);
|
var mats = SpawnMultipleFromMaterial(volume, material, Transform(uid).Coordinates, out _);
|
||||||
@@ -121,7 +136,7 @@ public sealed class MaterialStorageSystem : SharedMaterialStorageSystem
|
|||||||
overflowMaterial = 0;
|
overflowMaterial = 0;
|
||||||
if (!_prototypeManager.TryIndex<MaterialPrototype>(material, out var stackType))
|
if (!_prototypeManager.TryIndex<MaterialPrototype>(material, out var stackType))
|
||||||
{
|
{
|
||||||
Logger.Error("Failed to index material prototype " + material);
|
Log.Error("Failed to index material prototype " + material);
|
||||||
return new List<EntityUid>();
|
return new List<EntityUid>();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -105,13 +105,15 @@ public record struct GetMaterialWhitelistEvent(EntityUid Storage)
|
|||||||
/// Message sent to try and eject a material from a storage
|
/// Message sent to try and eject a material from a storage
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[Serializable, NetSerializable]
|
[Serializable, NetSerializable]
|
||||||
public sealed class EjectMaterialMessage : BoundUserInterfaceMessage
|
public sealed class EjectMaterialMessage : EntityEventArgs
|
||||||
{
|
{
|
||||||
|
public NetEntity Entity;
|
||||||
public string Material;
|
public string Material;
|
||||||
public int SheetsToExtract;
|
public int SheetsToExtract;
|
||||||
|
|
||||||
public EjectMaterialMessage(string material, int sheetsToExtract)
|
public EjectMaterialMessage(NetEntity entity, string material, int sheetsToExtract)
|
||||||
{
|
{
|
||||||
|
Entity = entity;
|
||||||
Material = material;
|
Material = material;
|
||||||
SheetsToExtract = sheetsToExtract;
|
SheetsToExtract = sheetsToExtract;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,6 +18,11 @@ public abstract class SharedMaterialStorageSystem : EntitySystem
|
|||||||
[Dependency] private readonly IGameTiming _timing = default!;
|
[Dependency] private readonly IGameTiming _timing = default!;
|
||||||
[Dependency] private readonly IPrototypeManager _prototype = default!;
|
[Dependency] private readonly IPrototypeManager _prototype = default!;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Default volume for a sheet if the material's entity prototype has no material composition.
|
||||||
|
/// </summary>
|
||||||
|
private const int DefaultSheetVolume = 100;
|
||||||
|
|
||||||
/// <inheritdoc/>
|
/// <inheritdoc/>
|
||||||
public override void Initialize()
|
public override void Initialize()
|
||||||
{
|
{
|
||||||
@@ -249,4 +254,17 @@ public abstract class SharedMaterialStorageSystem : EntitySystem
|
|||||||
return;
|
return;
|
||||||
args.Handled = TryInsertMaterialEntity(args.User, args.Used, uid, component);
|
args.Handled = TryInsertMaterialEntity(args.User, args.Used, uid, component);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public int GetSheetVolume(MaterialPrototype material)
|
||||||
|
{
|
||||||
|
if (material.StackEntity == null)
|
||||||
|
return DefaultSheetVolume;
|
||||||
|
|
||||||
|
var proto = _prototype.Index<EntityPrototype>(material.StackEntity);
|
||||||
|
|
||||||
|
if (!proto.TryGetComponent<PhysicalCompositionComponent>(out var composition))
|
||||||
|
return DefaultSheetVolume;
|
||||||
|
|
||||||
|
return composition.MaterialComposition.FirstOrDefault(kvp => kvp.Key == material.ID).Value;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user