using System.Linq; using Content.Client.Stylesheets; using Content.Shared.Chemistry; using Content.Shared.Chemistry.Reagent; using Robust.Client.AutoGenerated; using Robust.Client.UserInterface; using Robust.Client.UserInterface.Controls; using Robust.Client.UserInterface.CustomControls; using Robust.Client.UserInterface.XAML; using Robust.Client.Utility; using Robust.Shared.Prototypes; using Robust.Shared.Utility; using static Robust.Client.UserInterface.Controls.BoxContainer; namespace Content.Client.Chemistry.UI { /// /// Client-side UI used to control a /// [GenerateTypedNameReferences] public sealed partial class ChemMasterWindow : DefaultWindow { [Dependency] private readonly IPrototypeManager _prototypeManager = default!; public event Action? OnReagentButtonPressed; public readonly Button[] PillTypeButtons; private const string PillsRsiPath = "/Textures/Objects/Specific/Chemistry/pills.rsi"; /// /// Create and initialize the chem master UI client-side. Creates the basic layout, /// actual data isn't filled in until the server sends data about the chem master. /// public ChemMasterWindow() { RobustXamlLoader.Load(this); IoCManager.InjectDependencies(this); // Pill type selection buttons, in total there are 20 pills. // Pill rsi file should have states named as pill1, pill2, and so on. var resourcePath = new ResourcePath(PillsRsiPath); var pillTypeGroup = new ButtonGroup(); PillTypeButtons = new Button[20]; for (uint i = 0; i < PillTypeButtons.Length; i++) { // For every button decide which stylebase to have // Every row has 10 buttons String styleBase = StyleBase.ButtonOpenBoth; uint modulo = i % 10; if (i > 0 && modulo == 0) styleBase = StyleBase.ButtonOpenRight; else if (i > 0 && modulo == 9) styleBase = StyleBase.ButtonOpenLeft; else if (i == 0) styleBase = StyleBase.ButtonOpenRight; // Generate buttons PillTypeButtons[i] = new Button { Access = AccessLevel.Public, StyleClasses = { styleBase }, MaxSize = (42, 28), Group = pillTypeGroup }; // Generate buttons textures var specifier = new SpriteSpecifier.Rsi(resourcePath, "pill" + (i + 1)); TextureRect pillTypeTexture = new TextureRect { Texture = specifier.Frame0(), TextureScale = (1.75f, 1.75f), Stretch = TextureRect.StretchMode.KeepCentered, }; PillTypeButtons[i].AddChild(pillTypeTexture); Grid.AddChild(PillTypeButtons[i]); } PillAmount.InitDefaultButtons(); BottleAmount.InitDefaultButtons(); } private ReagentButton MakeReagentButton(string text, ChemMasterReagentAmount amount, string id, bool isBuffer, string styleClass) { var button = new ReagentButton(text, amount, id, isBuffer, styleClass); button.OnPressed += args => OnReagentButtonPressed?.Invoke(args, button); return button; } /// /// Update the UI state when new state data is received from the server. /// /// State data sent by the server. public void UpdateState(BoundUserInterfaceState state) { var castState = (ChemMasterBoundUserInterfaceState) state; Title = castState.DispenserName; if (castState.UpdateLabel) LabelLine = GenerateLabel(castState); UpdatePanelInfo(castState); if (Contents.Children != null) { EjectButton.Disabled = !castState.HasContainer(); } PillTypeButtons[castState.SelectedPillType].Pressed = true; PillAmount.IsValid = x => x > 0 && x <= castState.PillProductionLimit; BottleAmount.IsValid = x => x > 0 && x <= castState.BottleProductionLimit; } /// /// Generate a product label based on reagents in the buffer. /// /// State data sent by the server. private string GenerateLabel(ChemMasterBoundUserInterfaceState state) { if (state.BufferCurrentVolume == 0) return ""; else { var reagent = state.BufferReagents.OrderBy(r => r.Quantity).First(); _prototypeManager.TryIndex(reagent.ReagentId, out ReagentPrototype? proto); return proto?.LocalizedName ?? ""; } } /// /// Update the container, buffer, and packaging panels. /// /// State data for the dispenser. private void UpdatePanelInfo(ChemMasterBoundUserInterfaceState state) { BufferTransferButton.Pressed = state.Mode == Shared.Chemistry.ChemMasterMode.Transfer; BufferDiscardButton.Pressed = state.Mode == Shared.Chemistry.ChemMasterMode.Discard; ContainerInfo.Children.Clear(); if (!state.HasContainer()) { ContainerInfo.Children.Add(new Label {Text = Loc.GetString("chem-master-window-no-container-loaded-text") }); } else { ContainerInfo.Children.Add(new BoxContainer // Name of the container and its fill status (Ex: 44/100u) { Orientation = LayoutOrientation.Horizontal, Children = { new Label {Text = $"{state.ContainerName}: "}, new Label { Text = $"{state.ContainerCurrentVolume}/{state.ContainerMaxVolume}", StyleClasses = {StyleNano.StyleClassLabelSecondaryColor} } } }); foreach (var reagent in state.ContainerReagents!.OrderBy( r => {_prototypeManager.TryIndex(r.ReagentId, out ReagentPrototype? p); return p?.LocalizedName;})) { // Try to get the prototype for the given reagent. This gives us its name. _prototypeManager.TryIndex(reagent.ReagentId, out ReagentPrototype? proto); var name = proto?.LocalizedName ?? Loc.GetString("chem-master-window-unknown-reagent-text"); if (proto != null) { ContainerInfo.Children.Add(new BoxContainer { Orientation = LayoutOrientation.Horizontal, Children = { new Label {Text = $"{name}: "}, new Label { Text = $"{reagent.Quantity}u", StyleClasses = {StyleNano.StyleClassLabelSecondaryColor} }, // Padding new Control {HorizontalExpand = true}, MakeReagentButton("1", ChemMasterReagentAmount.U1, reagent.ReagentId, false, StyleBase.ButtonOpenRight), MakeReagentButton("5", ChemMasterReagentAmount.U5, reagent.ReagentId, false, StyleBase.ButtonOpenBoth), MakeReagentButton("10", ChemMasterReagentAmount.U10, reagent.ReagentId, false, StyleBase.ButtonOpenBoth), MakeReagentButton("25", ChemMasterReagentAmount.U25, reagent.ReagentId, false, StyleBase.ButtonOpenBoth), MakeReagentButton(Loc.GetString("chem-master-window-buffer-all-amount"), ChemMasterReagentAmount.All, reagent.ReagentId, false, StyleBase.ButtonOpenLeft), } }); } } } BufferInfo.Children.Clear(); if (!state.BufferReagents.Any()) { BufferInfo.Children.Add(new Label { Text = Loc.GetString("chem-master-window-buffer-empty-text") }); return; } var bufferHBox = new BoxContainer { Orientation = LayoutOrientation.Horizontal }; BufferInfo.AddChild(bufferHBox); var bufferLabel = new Label { Text = $"{Loc.GetString("chem-master-window-buffer-label")} " }; bufferHBox.AddChild(bufferLabel); var bufferVol = new Label { Text = $"{state.BufferCurrentVolume}", StyleClasses = {StyleNano.StyleClassLabelSecondaryColor} }; bufferHBox.AddChild(bufferVol); foreach (var reagent in state.BufferReagents) { // Try to get the prototype for the given reagent. This gives us its name. _prototypeManager.TryIndex(reagent.ReagentId, out ReagentPrototype? proto); var name = proto?.LocalizedName ?? Loc.GetString("chem-master-window-unknown-reagent-text"); if (proto != null) { BufferInfo.Children.Add(new BoxContainer { Orientation = LayoutOrientation.Horizontal, Children = { new Label {Text = $"{name}: "}, new Label { Text = $"{reagent.Quantity}u", StyleClasses = {StyleNano.StyleClassLabelSecondaryColor} }, // Padding new Control {HorizontalExpand = true}, MakeReagentButton("1", ChemMasterReagentAmount.U1, reagent.ReagentId, true, StyleBase.ButtonOpenRight), MakeReagentButton("5", ChemMasterReagentAmount.U5, reagent.ReagentId, true, StyleBase.ButtonOpenBoth), MakeReagentButton("10", ChemMasterReagentAmount.U10, reagent.ReagentId, true, StyleBase.ButtonOpenBoth), MakeReagentButton("25", ChemMasterReagentAmount.U25, reagent.ReagentId, true, StyleBase.ButtonOpenBoth), MakeReagentButton(Loc.GetString("chem-master-window-buffer-all-amount"), ChemMasterReagentAmount.All, reagent.ReagentId, true, StyleBase.ButtonOpenLeft), } }); } } } public String LabelLine { get { return LabelLineEdit.Text; } set { LabelLineEdit.Text = value; } } } public sealed class ReagentButton : Button { public ChemMasterReagentAmount Amount { get; set; } public bool IsBuffer = true; public string Id { get; set; } public ReagentButton(string text, ChemMasterReagentAmount amount, string id, bool isBuffer, string styleClass) { AddStyleClass(styleClass); Text = text; Amount = amount; Id = id; IsBuffer = isBuffer; } } }