* Apply patch 1777eea9a4..6b32bb2b14
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* make red squiggly line go away
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* Add todo list
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* Add palette to `TextureButton`
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* Rename `PalettedButtonSheetlet` to `NTButtonSheetlet` and move useful methods to `ButtonSheetlet`
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* migrate `ContextMenu` styles
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* Update todo
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* tweak NT colors
* New stylesheet: `InterfaceStylesheet` & `InterfaceTooltipSheetlet`
* Move inheritance of `IButtonConfig` to `NanotransenStylesheet.Buttons`
* move `MenuButtonSheetlet` & actually implement `InterfaceStylesheet` correctly
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* tweak color & update todo
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* chat is this real (update chat palette)
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* Update todo
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* `SmallButton` and remove some obsolete things from `StyleNano`
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* rename `StyleClasses` to `StyleClass` so `Stylesheets.Redux.StyleClasses` syntax is dead
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* replace `ButtonColorGreen` with `Positive`
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* `Placeholder`
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* Examine popup buttons
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* move over more things & cleanup `StyleNano` more (under 1000 lines!!!!)
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* Remove some more redundant stuff
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* Undo style change for chat window
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* paper editing works now
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* `OptionButton` styles
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* `ListContainer`, move `DefaultWindow` styles (for now) & more cleanup
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* fix `ActionButton` not having highlighting
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* remove imports of `Robust.Client.UserInterface.StylesheetHelpers` & format
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* `ButtonBig` and more cleanup
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* Move items inheriting from `ISheetletConfig` into their own directory
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* Cleanup & move `Label` styles
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* Action search box styles
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* Moved, stuff is
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* rename `LabelSubtext` to `LabelSubText` & move more stuff (were almost there!!)
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* yap & move over MORE stuff (just like one thing left!!!)
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* Change status classes to appropriate existing classes
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* remove remaining references to `StyleNano`
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* Fix some hardcoding & broken code, `GetFromControl`
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* Scrollbars!
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* chores
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* clean up `StyleClass.cs`
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* `ItemListSheetlet` refactor
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* more chores!
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* Consistency w/ directory structure
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* Move `MainMenuSheetlet`
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* `ColorPalette`
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* whoopsie
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* Remove most sheet-specific sheetlets
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* fix warnings, cleanup, & fix scrollbar (this is why we fix warnings boys)
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* yap
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* MASSIVE resharper skill issue
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* actually use `ISheetletConfig`
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* have specific sheetlet be specific
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* `GetResourceOr`
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* cleanup & move / remove `IPalette`s
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* actually do specific stylesheets correctly & fix tooltips
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* cleanup & logging
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* Move `FontKind` and `FontKindExtensions` to their own files
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* rename `InterfaceStylesheet` to `SystemStylesheet`
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* change `ButtonHovered` etc to `PseudoHovered` etc
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* give the palettes fun names
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* `StyleSpace` is no more
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* It should compile now! I am now going to bed (fr) if it fails it fails
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* make squiggly red line go away
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
* add additional type restrictions to sheetlets
* `CommonStylesheet`
* minor cleanup
* Make `GetSheetletRules` not horrible
* wait this was duplicating style rules. oops!
* move some sheetlets to their associated xamls
* oh wait apparently that was important
* review pass 1
* review pass 2 (font & color stuff)
* review pass 3: remove unused stuff / filename fix
* fix warnings & "replace cast with explicit variable type"
* move `Palette` stuff to its own directory
* tweak colors (they're different now that I actually fixed the OKlab thing)
* review pass 4: little things
* make window close button grey before hovering
* refactor `HLine` to make it less terrible and allow it to be styled
* fix `NanoHeading` (it's been broken for a while whoops) and cleanup hardcoding
* band-aid missing references in `StyleNano`
* move `StyleBox` generating functions out of `IButtonSheetlet` into `StyleBoxHelper`
* remove dictionary field from `IStylesheetManager`
* Add check for unloaded sheetlets
* style tweaks to satisfy OCD
* I somehow missed this: `Caution` styleclass replaced with `negative`, refactor `PowerChargeWindow`
* tweak palettes for like the fourth time
* construct `StyleNano` / `StyleSpace` in `StylesheetManager` and mark them as obsolete
* rename `BackgroundPanel` classes for consistency
* tweak window / `ListContainer`
* oh right you use `///` not `/**`
* font system is bad, make it temporary
* acknowledge Divider funkyness
* remove use of class `Disabled`
* `ColorPalette` allow overriding colors with brace initialization
* review pass again
* tweak disabled button colors
* `StatusPalette` tweaks
* typo
* Make squiggly red line go away
* Delete `Redux`
* Remove all references to `Redux`
* make red less radioactive
* Store stylesheet name inside stylesheet class
* fix merge errors
* use RT's Oklab support instead
* shuffle around `StylesheetManager` fields
* apply stylesheets based off `StylesheetComponent`
* simplify `ColorPalette` construction
* add todo for `SheetletConfigType`
* `OptionButton` has a background color now
* fix disabled buttons
* sigh (red color palette fixed)
* make `ItemList` use primary palette
* Revert "apply stylesheets based off `StylesheetComponent`"
This reverts commit c05b147da845f6e04ff33d1cbd91a18a92c676d7.
* dead code removal
* buttons are green when pressed (we need togglebuttons)
---------
Signed-off-by: Brandon Li <sirbrandonthenerd@gmail.com>
Co-authored-by: Janet Blackquill <uhhadd@gmail.com>
289 lines
9.6 KiB
C#
289 lines
9.6 KiB
C#
using System.Linq;
|
|
using System.Numerics;
|
|
using Content.Client.Computer;
|
|
using Content.Client.Stylesheets;
|
|
using Content.Client.UserInterface.Controls;
|
|
using Content.Shared.SensorMonitoring;
|
|
using JetBrains.Annotations;
|
|
using Robust.Client.AutoGenerated;
|
|
using Robust.Client.Graphics;
|
|
using Robust.Client.UserInterface;
|
|
using Robust.Client.UserInterface.Controls;
|
|
using Robust.Client.UserInterface.XAML;
|
|
using Robust.Shared.Timing;
|
|
using ConsoleUIState = Content.Shared.SensorMonitoring.SensorMonitoringConsoleBoundInterfaceState;
|
|
using IncrementalUIState = Content.Shared.SensorMonitoring.SensorMonitoringIncrementalUpdate;
|
|
|
|
namespace Content.Client.SensorMonitoring;
|
|
|
|
[GenerateTypedNameReferences]
|
|
public sealed partial class SensorMonitoringWindow : FancyWindow, IComputerWindow<ConsoleUIState>
|
|
{
|
|
[Dependency] private readonly IGameTiming _gameTiming = default!;
|
|
[Dependency] private readonly ILocalizationManager _loc = default!;
|
|
|
|
private TimeSpan _retentionTime;
|
|
private readonly Dictionary<int, SensorData> _sensorData = new();
|
|
|
|
/// <summary>
|
|
/// <para>A shared array used to store vertices for drawing graphs in <see cref="GraphView"/>.
|
|
/// Prevents excessive allocations by reusing the same array across multiple graph views.</para>
|
|
/// <para>This effectively makes it so that each <see cref="SensorMonitoringWindow"/> has its own pooled array.</para>
|
|
/// </summary>
|
|
private Vector2[] _sharedVertices = [];
|
|
|
|
/// <summary>
|
|
/// Retrieves a shared array of vertices, ensuring that it has at least the requested size.
|
|
/// Assigns a new array of the requested size if the current shared array is smaller than the requested size.
|
|
/// </summary>
|
|
/// <param name="requestedSize">The minimum number of vertices required in the shared array.</param>
|
|
/// <returns>An array of <see cref="System.Numerics.Vector2"/> representing the shared vertices.</returns>
|
|
/// <remarks>This does not prevent other threads from accessing the same shared pool.</remarks>
|
|
public Vector2[] GetSharedVertices(int requestedSize)
|
|
{
|
|
if (_sharedVertices.Length < requestedSize)
|
|
{
|
|
_sharedVertices = new Vector2[requestedSize];
|
|
}
|
|
return _sharedVertices;
|
|
}
|
|
|
|
public SensorMonitoringWindow()
|
|
{
|
|
RobustXamlLoader.Load(this);
|
|
IoCManager.InjectDependencies(this);
|
|
}
|
|
|
|
public void UpdateState(ConsoleUIState state)
|
|
{
|
|
_retentionTime = state.RetentionTime;
|
|
|
|
_sensorData.Clear();
|
|
|
|
foreach (var netSensor in state.Sensors)
|
|
{
|
|
var sensor = new SensorData
|
|
{
|
|
Name = netSensor.Name,
|
|
Address = netSensor.Address,
|
|
DeviceType = netSensor.DeviceType
|
|
};
|
|
|
|
_sensorData.Add(netSensor.NetId, sensor);
|
|
|
|
foreach (var netStream in netSensor.Streams)
|
|
{
|
|
var stream = new SensorStream
|
|
{
|
|
Name = netStream.Name,
|
|
Unit = netStream.Unit
|
|
};
|
|
|
|
sensor.Streams.Add(netStream.NetId, stream);
|
|
|
|
foreach (var sample in netStream.Samples)
|
|
{
|
|
stream.Samples.Enqueue(sample);
|
|
}
|
|
}
|
|
}
|
|
|
|
Update();
|
|
}
|
|
|
|
public void ReceiveMessage(BoundUserInterfaceMessage message)
|
|
{
|
|
if (message is not IncrementalUIState incremental)
|
|
return;
|
|
|
|
foreach (var removed in incremental.RemovedSensors)
|
|
{
|
|
_sensorData.Remove(removed);
|
|
}
|
|
|
|
foreach (var netSensor in incremental.Sensors)
|
|
{
|
|
// TODO: Fuck this doesn't work if a sensor is added while the UI is open.
|
|
if (!_sensorData.TryGetValue(netSensor.NetId, out var sensor))
|
|
continue;
|
|
|
|
foreach (var netStream in netSensor.Streams)
|
|
{
|
|
// TODO: Fuck this doesn't work if a stream is added while the UI is open.
|
|
if (!sensor.Streams.TryGetValue(netStream.NetId, out var stream))
|
|
continue;
|
|
|
|
foreach (var (time, value) in netStream.Samples)
|
|
{
|
|
stream.Samples.Enqueue(new SensorSample(time + incremental.RelTime, value));
|
|
}
|
|
}
|
|
}
|
|
|
|
CullOldSamples();
|
|
Update();
|
|
}
|
|
|
|
private void Update()
|
|
{
|
|
Asdf.RemoveAllChildren();
|
|
|
|
var curTime = _gameTiming.CurTime;
|
|
var startTime = curTime - _retentionTime;
|
|
|
|
foreach (var sensor in _sensorData.Values)
|
|
{
|
|
var labelName = new Label { Text = sensor.Name, StyleClasses = { StyleClass.LabelHeading } };
|
|
var labelAddress = new Label
|
|
{
|
|
Text = sensor.Address,
|
|
Margin = new Thickness(4, 0),
|
|
VerticalAlignment = VAlignment.Bottom,
|
|
StyleClasses = { StyleClass.LabelWeak }
|
|
};
|
|
|
|
Asdf.AddChild(new BoxContainer
|
|
{
|
|
Orientation = BoxContainer.LayoutOrientation.Horizontal, Children =
|
|
{
|
|
labelName,
|
|
labelAddress
|
|
}
|
|
});
|
|
|
|
foreach (var stream in sensor.Streams.Values)
|
|
{
|
|
var maxValue = stream.Samples.Max(x => x.Value);
|
|
|
|
// TODO: Better way to do this?
|
|
var lastSample = stream.Samples.Last();
|
|
|
|
Asdf.AddChild(new BoxContainer
|
|
{
|
|
Orientation = BoxContainer.LayoutOrientation.Horizontal,
|
|
Children =
|
|
{
|
|
new Label { Text = stream.Name, StyleClasses = { "Monospace" }, HorizontalExpand = true },
|
|
new Label { Text = FormatValue(stream.Unit, lastSample.Value) }
|
|
}
|
|
});
|
|
|
|
Asdf.AddChild(new GraphView(stream.Samples, startTime, curTime, maxValue * 1.1f, this) { MinHeight = 150 });
|
|
Asdf.AddChild(new PanelContainer { StyleClasses = { StyleClass.LowDivider } });
|
|
}
|
|
}
|
|
}
|
|
|
|
private string FormatValue(SensorUnit unit, float value)
|
|
{
|
|
return _loc.GetString(
|
|
"sensor-monitoring-value-display",
|
|
("unit", unit.ToString()),
|
|
("value", value));
|
|
}
|
|
|
|
private void CullOldSamples()
|
|
{
|
|
var startTime = _gameTiming.CurTime - _retentionTime;
|
|
|
|
foreach (var sensor in _sensorData.Values)
|
|
{
|
|
foreach (var stream in sensor.Streams.Values)
|
|
{
|
|
while (stream.Samples.TryPeek(out var sample) && sample.Time < startTime)
|
|
{
|
|
stream.Samples.Dequeue();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private sealed class SensorData
|
|
{
|
|
public string Name = "";
|
|
public string Address = "";
|
|
public SensorDeviceType DeviceType;
|
|
|
|
public readonly Dictionary<int, SensorStream> Streams = new();
|
|
}
|
|
|
|
private sealed class SensorStream
|
|
{
|
|
public string Name = "";
|
|
public SensorUnit Unit;
|
|
public readonly Queue<SensorSample> Samples = new();
|
|
}
|
|
|
|
private sealed class GraphView : Control
|
|
{
|
|
private readonly Queue<SensorSample> _samples;
|
|
private readonly TimeSpan _startTime;
|
|
private readonly TimeSpan _curTime;
|
|
private readonly float _maxY;
|
|
private readonly SensorMonitoringWindow _parentWindow;
|
|
|
|
public GraphView(Queue<SensorSample> samples,
|
|
TimeSpan startTime,
|
|
TimeSpan curTime,
|
|
float maxY,
|
|
SensorMonitoringWindow parentWindow)
|
|
{
|
|
_samples = samples;
|
|
_startTime = startTime;
|
|
_curTime = curTime;
|
|
_maxY = maxY;
|
|
RectClipContent = true;
|
|
_parentWindow = parentWindow;
|
|
}
|
|
|
|
protected override void Draw(DrawingHandleScreen handle)
|
|
{
|
|
base.Draw(handle);
|
|
|
|
var window = (float)(_curTime - _startTime).TotalSeconds;
|
|
var countVtx = 0;
|
|
|
|
var lastPoint = new Vector2(float.NaN, float.NaN);
|
|
var requiredVertices = 6 * (_samples.Count - 1);
|
|
|
|
var vertices = _parentWindow.GetSharedVertices(requiredVertices);
|
|
|
|
foreach (var (time, sample) in _samples)
|
|
{
|
|
var relTime = (float)(time - _startTime).TotalSeconds;
|
|
|
|
var posY = PixelHeight - (sample / _maxY) * PixelHeight;
|
|
var posX = (relTime / window) * PixelWidth;
|
|
|
|
var newPoint = new Vector2(posX, posY);
|
|
|
|
if (float.IsFinite(lastPoint.X))
|
|
{
|
|
handle.DrawLine(lastPoint, newPoint, Color.White);
|
|
|
|
vertices[countVtx++] = lastPoint;
|
|
vertices[countVtx++] = lastPoint with { Y = PixelHeight };
|
|
vertices[countVtx++] = newPoint;
|
|
vertices[countVtx++] = newPoint;
|
|
vertices[countVtx++] = lastPoint with { Y = PixelHeight };
|
|
vertices[countVtx++] = newPoint with { Y = PixelHeight };
|
|
}
|
|
|
|
lastPoint = newPoint;
|
|
}
|
|
handle.DrawPrimitives(DrawPrimitiveTopology.TriangleList,
|
|
vertices.AsSpan(0, countVtx),
|
|
Color.White.WithAlpha(0.1f));
|
|
}
|
|
}
|
|
}
|
|
|
|
[UsedImplicitly]
|
|
public sealed class
|
|
SensorMonitoringConsoleBoundUserInterface : ComputerBoundUserInterface<SensorMonitoringWindow, ConsoleUIState>
|
|
{
|
|
public SensorMonitoringConsoleBoundUserInterface(EntityUid owner, Enum uiKey) : base(owner, uiKey)
|
|
{
|
|
}
|
|
}
|