From 011a2079544a93bc70720c2519d1c41a51281294 Mon Sep 17 00:00:00 2001 From: fenndragon Date: Wed, 11 Dec 2024 21:48:38 -0700 Subject: [PATCH 1/6] Create folder --- Content.Client/DeltaV/RainLizard/folder | 1 + 1 file changed, 1 insertion(+) create mode 100644 Content.Client/DeltaV/RainLizard/folder diff --git a/Content.Client/DeltaV/RainLizard/folder b/Content.Client/DeltaV/RainLizard/folder new file mode 100644 index 00000000000..8b137891791 --- /dev/null +++ b/Content.Client/DeltaV/RainLizard/folder @@ -0,0 +1 @@ + From 1c393df7846c36c8c468c1fc290e2fc098fcb479 Mon Sep 17 00:00:00 2001 From: fenndragon Date: Wed, 11 Dec 2024 21:49:04 -0700 Subject: [PATCH 2/6] Add files via upload --- .../DeltaV/RainLizard/SnakeOverlay.cs | 184 ++++++++++++++++++ .../DeltaV/RainLizard/SnakeOverlaySystem.cs | 32 +++ 2 files changed, 216 insertions(+) create mode 100644 Content.Client/DeltaV/RainLizard/SnakeOverlay.cs create mode 100644 Content.Client/DeltaV/RainLizard/SnakeOverlaySystem.cs diff --git a/Content.Client/DeltaV/RainLizard/SnakeOverlay.cs b/Content.Client/DeltaV/RainLizard/SnakeOverlay.cs new file mode 100644 index 00000000000..a8ed98c3369 --- /dev/null +++ b/Content.Client/DeltaV/RainLizard/SnakeOverlay.cs @@ -0,0 +1,184 @@ +/* +* This file is licensed under AGPLv3 +* Copyright (c) 2024 Rane +* See AGPLv3.txt for details. +*/ + +using Content.Shared.SegmentedEntity; +using Content.Shared.Humanoid; +using Content.Shared.Humanoid.Markings; +using Content.Client.Resources; +using Robust.Client.ResourceManagement; +using Robust.Client.Graphics; +using Robust.Shared.Enums; +using System.Numerics; +using System.Linq; + + +namespace Content.Client.DeltaV.Lamiae; + +/// +/// This draws lamia segments directly from polygons instead of sprites. This is a very novel approach as of the time this is being written (August 2024) but it wouldn't surprise me +/// if there's a better way to do this at some point. Currently we have a very heavy restriction on the tools we can make, forcing me to make several helpers that may be redundant later. +/// This will be overcommented because I know you haven't seen code like this before and you might want to copy it. +/// This is an expansion on some techniques I discovered in (https://github.com/Elijahrane/Delta-v/blob/49d76c437740eab79fc622ab50d628b926e6ddcb/Content.Client/DeltaV/Arcade/S3D/Renderer/S3DRenderer.cs) +/// +public sealed class SnakeOverlay : Overlay +{ + private readonly IResourceCache _resourceCache; + private readonly IEntityManager _entManager; + private readonly SharedTransformSystem _transform; + private readonly SharedHumanoidAppearanceSystem _humanoid = default!; + + // Look through these carefully. WorldSpace is useful for debugging. Note that this defaults to "screen space" which breaks when you try and get the world handle. + public override OverlaySpace Space => OverlaySpace.WorldSpaceEntities; + + // Overlays are strange and you need this pattern where you define readonly deps above, and then make a constructor with this pattern. Anything that creates this overlay will then + // have to provide all the deps. + public SnakeOverlay(IEntityManager entManager, IResourceCache resourceCache) + { + _resourceCache = resourceCache; + // we get ent manager from SnakeOverlaySystem turning this on and passing it + _entManager = entManager; + // with ent manager we can fetch our other entity systems + _transform = _entManager.EntitySysManager.GetEntitySystem(); + _humanoid = _entManager.EntitySysManager.GetEntitySystem(); + + // draw at drawdepth 3 + ZIndex = 3; + } + + // This step occurs each frame. For some overlays you may want to conisder limiting how often they update, but for player entities that move around fast we'll just do it every frame. + protected override void Draw(in OverlayDrawArgs args) + { + // load the handle, the "pen" we draw with + var handle = args.WorldHandle; + + // Get all lamiae the client knows of and their transform in a way we can enumerate over + var enumerator = _entManager.AllEntityQueryEnumerator(); + + // I go over the collection above, pulling out an EntityUid and the two components I need for each. + while (enumerator.MoveNext(out var uid, out var lamia, out var xform)) + { + // Skip ones that are off-map. "Map" in this context means interconnected stuff you can travel between by moving, rather than needing e.g. FTL to load a new map. + if (xform.MapID != args.MapId) + continue; + + // Skip ones where they are not loaded properly, uninitialized, or w/e + if (lamia.Segments.Count < lamia.NumberOfSegments) + { + _entManager.Dirty(uid, lamia); // pls give me an update... + continue; + } + + // By the way, there's a hack to mitigate overdraw somewhat. Check out whatever is going on with the variable called "bounds" in DoAfterOverlay. + // I won't do it here because (1) it's ugly and (2) theoretically these entities can be fucking huge and you'll see the tail end of them when they are way off screen. + // On a PVS level I think segmented entities should be all-or-nothing when it comes to PVS range, that is you either load all of their segments or none. + + // Color.White is drawing without modifying color. For clothed tails, we should use White. For skin, we should use the color of the marking. + // TODO: Better way to cache this + if (_entManager.TryGetComponent(uid, out var humanoid)) + { + if (humanoid.MarkingSet.TryGetCategory(MarkingCategories.Tail, out var tailMarkings)) + { + var col = tailMarkings.First().MarkingColors.First(); + DrawLamia(handle, lamia, col); + } + } + else + { + DrawLamia(handle, lamia, Color.White); + } + } + } + + // This is where we do the actual drawing. + private void DrawLamia(DrawingHandleWorld handle, SegmentedEntityComponent lamia, Color color) + { + // We're going to store all our verticies in here and then draw them + List verts = new List(); + + // Radius of the initial segment + float radius = lamia.InitialRadius; + + // We're storing the left and right verticies of the last segment so we can start drawing from there without gaps + Vector2? lastPtCW = null; + Vector2? lastPtCCW = null; + + var tex = _resourceCache.GetTexture(lamia.TexturePath); + + int i = 1; + // do each segment except the last one normally + while (i < lamia.Segments.Count - 1) + { + // get centerpoints of last segment and this one + var origin = _transform.GetWorldPosition(_entManager.GetEntity(lamia.Segments[i - 1])); + var destination = _transform.GetWorldPosition(_entManager.GetEntity(lamia.Segments[i])); + + // get direction between the two points and normalize it + var connectorVec = destination - origin; + connectorVec = connectorVec.Normalized(); + + //get one rotated 90 degrees clockwise + var offsetVecCW = new Vector2(connectorVec.Y, 0 - connectorVec.X); + + //and counterclockwise + var offsetVecCCW = new Vector2(0 - connectorVec.Y, connectorVec.X); + + /// tri 1: line across first segment and corner of second + if (lastPtCW == null) + { + verts.Add(new DrawVertexUV2D(origin + offsetVecCW * radius, Vector2.Zero)); + } + else + { + verts.Add(new DrawVertexUV2D((Vector2) lastPtCW, Vector2.Zero)); + } + + if (lastPtCCW == null) + { + verts.Add(new DrawVertexUV2D(origin + offsetVecCCW * radius, new Vector2(1, 0))); + } + else + { + verts.Add(new DrawVertexUV2D((Vector2) lastPtCCW, new Vector2(1, 0))); + } + + verts.Add(new DrawVertexUV2D(destination + offsetVecCW * radius, new Vector2(0, 1))); + + // tri 2: line across second segment and corner of first + if (lastPtCCW == null) + { + verts.Add(new DrawVertexUV2D(origin + offsetVecCCW * radius, new Vector2(1, 0))); + } + else + { + verts.Add(new DrawVertexUV2D((Vector2) lastPtCCW, new Vector2(1, 0))); + } + + lastPtCW = destination + offsetVecCW * radius; + verts.Add(new DrawVertexUV2D((Vector2) lastPtCW, new Vector2(0, 1))); + lastPtCCW = destination + offsetVecCCW * radius; + verts.Add(new DrawVertexUV2D((Vector2) lastPtCCW, new Vector2(1, 1))); + + // slim down a bit for next segment + radius *= lamia.SlimFactor; + + i++; + } + + // draw tail (1 tri) + if (lastPtCW != null && lastPtCCW != null) + { + verts.Add(new DrawVertexUV2D((Vector2) lastPtCW, new Vector2(0, 0))); + verts.Add(new DrawVertexUV2D((Vector2) lastPtCCW, new Vector2(1, 0))); + + var destination = _transform.GetWorldPosition(_entManager.GetEntity(lamia.Segments.Last())); + + verts.Add(new DrawVertexUV2D(destination, new Vector2(0.5f, 1f))); + } + + // Draw all of the triangles we just pit in at once + handle.DrawPrimitives(DrawPrimitiveTopology.TriangleList, texture: tex, verts.ToArray().AsSpan(), color); + } +} \ No newline at end of file diff --git a/Content.Client/DeltaV/RainLizard/SnakeOverlaySystem.cs b/Content.Client/DeltaV/RainLizard/SnakeOverlaySystem.cs new file mode 100644 index 00000000000..6be1a18dc14 --- /dev/null +++ b/Content.Client/DeltaV/RainLizard/SnakeOverlaySystem.cs @@ -0,0 +1,32 @@ +/* +* This file is licensed under AGPLv3 +* Copyright (c) 2024 Rane +* See AGPLv3.txt for details. +*/ + +using Robust.Client.Graphics; +using Robust.Client.ResourceManagement; + +namespace Content.Client.DeltaV.Lamiae; + +/// +/// This system turns on our always-on overlay. I have no opinion on this design pattern or the existence of this file. +/// It also fetches the deps it needs. +/// +public sealed class SnakeOverlaySystem : EntitySystem +{ + [Dependency] private readonly IOverlayManager _overlay = default!; + [Dependency] private readonly IResourceCache _resourceCache = default!; + + public override void Initialize() + { + base.Initialize(); + _overlay.AddOverlay(new SnakeOverlay(EntityManager, _resourceCache)); + } + + public override void Shutdown() + { + base.Shutdown(); + _overlay.RemoveOverlay(); + } +} \ No newline at end of file From 59bd1a183674dea92a97d2ec91551f99db8690a2 Mon Sep 17 00:00:00 2001 From: fenndragon Date: Sun, 15 Dec 2024 18:31:42 -0700 Subject: [PATCH 3/6] Delete Content.Client/DeltaV/RainLizard directory --- .../DeltaV/RainLizard/SnakeOverlay.cs | 184 ------------------ .../DeltaV/RainLizard/SnakeOverlaySystem.cs | 32 --- Content.Client/DeltaV/RainLizard/folder | 1 - 3 files changed, 217 deletions(-) delete mode 100644 Content.Client/DeltaV/RainLizard/SnakeOverlay.cs delete mode 100644 Content.Client/DeltaV/RainLizard/SnakeOverlaySystem.cs delete mode 100644 Content.Client/DeltaV/RainLizard/folder diff --git a/Content.Client/DeltaV/RainLizard/SnakeOverlay.cs b/Content.Client/DeltaV/RainLizard/SnakeOverlay.cs deleted file mode 100644 index a8ed98c3369..00000000000 --- a/Content.Client/DeltaV/RainLizard/SnakeOverlay.cs +++ /dev/null @@ -1,184 +0,0 @@ -/* -* This file is licensed under AGPLv3 -* Copyright (c) 2024 Rane -* See AGPLv3.txt for details. -*/ - -using Content.Shared.SegmentedEntity; -using Content.Shared.Humanoid; -using Content.Shared.Humanoid.Markings; -using Content.Client.Resources; -using Robust.Client.ResourceManagement; -using Robust.Client.Graphics; -using Robust.Shared.Enums; -using System.Numerics; -using System.Linq; - - -namespace Content.Client.DeltaV.Lamiae; - -/// -/// This draws lamia segments directly from polygons instead of sprites. This is a very novel approach as of the time this is being written (August 2024) but it wouldn't surprise me -/// if there's a better way to do this at some point. Currently we have a very heavy restriction on the tools we can make, forcing me to make several helpers that may be redundant later. -/// This will be overcommented because I know you haven't seen code like this before and you might want to copy it. -/// This is an expansion on some techniques I discovered in (https://github.com/Elijahrane/Delta-v/blob/49d76c437740eab79fc622ab50d628b926e6ddcb/Content.Client/DeltaV/Arcade/S3D/Renderer/S3DRenderer.cs) -/// -public sealed class SnakeOverlay : Overlay -{ - private readonly IResourceCache _resourceCache; - private readonly IEntityManager _entManager; - private readonly SharedTransformSystem _transform; - private readonly SharedHumanoidAppearanceSystem _humanoid = default!; - - // Look through these carefully. WorldSpace is useful for debugging. Note that this defaults to "screen space" which breaks when you try and get the world handle. - public override OverlaySpace Space => OverlaySpace.WorldSpaceEntities; - - // Overlays are strange and you need this pattern where you define readonly deps above, and then make a constructor with this pattern. Anything that creates this overlay will then - // have to provide all the deps. - public SnakeOverlay(IEntityManager entManager, IResourceCache resourceCache) - { - _resourceCache = resourceCache; - // we get ent manager from SnakeOverlaySystem turning this on and passing it - _entManager = entManager; - // with ent manager we can fetch our other entity systems - _transform = _entManager.EntitySysManager.GetEntitySystem(); - _humanoid = _entManager.EntitySysManager.GetEntitySystem(); - - // draw at drawdepth 3 - ZIndex = 3; - } - - // This step occurs each frame. For some overlays you may want to conisder limiting how often they update, but for player entities that move around fast we'll just do it every frame. - protected override void Draw(in OverlayDrawArgs args) - { - // load the handle, the "pen" we draw with - var handle = args.WorldHandle; - - // Get all lamiae the client knows of and their transform in a way we can enumerate over - var enumerator = _entManager.AllEntityQueryEnumerator(); - - // I go over the collection above, pulling out an EntityUid and the two components I need for each. - while (enumerator.MoveNext(out var uid, out var lamia, out var xform)) - { - // Skip ones that are off-map. "Map" in this context means interconnected stuff you can travel between by moving, rather than needing e.g. FTL to load a new map. - if (xform.MapID != args.MapId) - continue; - - // Skip ones where they are not loaded properly, uninitialized, or w/e - if (lamia.Segments.Count < lamia.NumberOfSegments) - { - _entManager.Dirty(uid, lamia); // pls give me an update... - continue; - } - - // By the way, there's a hack to mitigate overdraw somewhat. Check out whatever is going on with the variable called "bounds" in DoAfterOverlay. - // I won't do it here because (1) it's ugly and (2) theoretically these entities can be fucking huge and you'll see the tail end of them when they are way off screen. - // On a PVS level I think segmented entities should be all-or-nothing when it comes to PVS range, that is you either load all of their segments or none. - - // Color.White is drawing without modifying color. For clothed tails, we should use White. For skin, we should use the color of the marking. - // TODO: Better way to cache this - if (_entManager.TryGetComponent(uid, out var humanoid)) - { - if (humanoid.MarkingSet.TryGetCategory(MarkingCategories.Tail, out var tailMarkings)) - { - var col = tailMarkings.First().MarkingColors.First(); - DrawLamia(handle, lamia, col); - } - } - else - { - DrawLamia(handle, lamia, Color.White); - } - } - } - - // This is where we do the actual drawing. - private void DrawLamia(DrawingHandleWorld handle, SegmentedEntityComponent lamia, Color color) - { - // We're going to store all our verticies in here and then draw them - List verts = new List(); - - // Radius of the initial segment - float radius = lamia.InitialRadius; - - // We're storing the left and right verticies of the last segment so we can start drawing from there without gaps - Vector2? lastPtCW = null; - Vector2? lastPtCCW = null; - - var tex = _resourceCache.GetTexture(lamia.TexturePath); - - int i = 1; - // do each segment except the last one normally - while (i < lamia.Segments.Count - 1) - { - // get centerpoints of last segment and this one - var origin = _transform.GetWorldPosition(_entManager.GetEntity(lamia.Segments[i - 1])); - var destination = _transform.GetWorldPosition(_entManager.GetEntity(lamia.Segments[i])); - - // get direction between the two points and normalize it - var connectorVec = destination - origin; - connectorVec = connectorVec.Normalized(); - - //get one rotated 90 degrees clockwise - var offsetVecCW = new Vector2(connectorVec.Y, 0 - connectorVec.X); - - //and counterclockwise - var offsetVecCCW = new Vector2(0 - connectorVec.Y, connectorVec.X); - - /// tri 1: line across first segment and corner of second - if (lastPtCW == null) - { - verts.Add(new DrawVertexUV2D(origin + offsetVecCW * radius, Vector2.Zero)); - } - else - { - verts.Add(new DrawVertexUV2D((Vector2) lastPtCW, Vector2.Zero)); - } - - if (lastPtCCW == null) - { - verts.Add(new DrawVertexUV2D(origin + offsetVecCCW * radius, new Vector2(1, 0))); - } - else - { - verts.Add(new DrawVertexUV2D((Vector2) lastPtCCW, new Vector2(1, 0))); - } - - verts.Add(new DrawVertexUV2D(destination + offsetVecCW * radius, new Vector2(0, 1))); - - // tri 2: line across second segment and corner of first - if (lastPtCCW == null) - { - verts.Add(new DrawVertexUV2D(origin + offsetVecCCW * radius, new Vector2(1, 0))); - } - else - { - verts.Add(new DrawVertexUV2D((Vector2) lastPtCCW, new Vector2(1, 0))); - } - - lastPtCW = destination + offsetVecCW * radius; - verts.Add(new DrawVertexUV2D((Vector2) lastPtCW, new Vector2(0, 1))); - lastPtCCW = destination + offsetVecCCW * radius; - verts.Add(new DrawVertexUV2D((Vector2) lastPtCCW, new Vector2(1, 1))); - - // slim down a bit for next segment - radius *= lamia.SlimFactor; - - i++; - } - - // draw tail (1 tri) - if (lastPtCW != null && lastPtCCW != null) - { - verts.Add(new DrawVertexUV2D((Vector2) lastPtCW, new Vector2(0, 0))); - verts.Add(new DrawVertexUV2D((Vector2) lastPtCCW, new Vector2(1, 0))); - - var destination = _transform.GetWorldPosition(_entManager.GetEntity(lamia.Segments.Last())); - - verts.Add(new DrawVertexUV2D(destination, new Vector2(0.5f, 1f))); - } - - // Draw all of the triangles we just pit in at once - handle.DrawPrimitives(DrawPrimitiveTopology.TriangleList, texture: tex, verts.ToArray().AsSpan(), color); - } -} \ No newline at end of file diff --git a/Content.Client/DeltaV/RainLizard/SnakeOverlaySystem.cs b/Content.Client/DeltaV/RainLizard/SnakeOverlaySystem.cs deleted file mode 100644 index 6be1a18dc14..00000000000 --- a/Content.Client/DeltaV/RainLizard/SnakeOverlaySystem.cs +++ /dev/null @@ -1,32 +0,0 @@ -/* -* This file is licensed under AGPLv3 -* Copyright (c) 2024 Rane -* See AGPLv3.txt for details. -*/ - -using Robust.Client.Graphics; -using Robust.Client.ResourceManagement; - -namespace Content.Client.DeltaV.Lamiae; - -/// -/// This system turns on our always-on overlay. I have no opinion on this design pattern or the existence of this file. -/// It also fetches the deps it needs. -/// -public sealed class SnakeOverlaySystem : EntitySystem -{ - [Dependency] private readonly IOverlayManager _overlay = default!; - [Dependency] private readonly IResourceCache _resourceCache = default!; - - public override void Initialize() - { - base.Initialize(); - _overlay.AddOverlay(new SnakeOverlay(EntityManager, _resourceCache)); - } - - public override void Shutdown() - { - base.Shutdown(); - _overlay.RemoveOverlay(); - } -} \ No newline at end of file diff --git a/Content.Client/DeltaV/RainLizard/folder b/Content.Client/DeltaV/RainLizard/folder deleted file mode 100644 index 8b137891791..00000000000 --- a/Content.Client/DeltaV/RainLizard/folder +++ /dev/null @@ -1 +0,0 @@ - From 0ec437d26a0aedaaff8d8ebf3661701efaf5412f Mon Sep 17 00:00:00 2001 From: fenndragon Date: Sat, 21 Dec 2024 15:39:34 -0700 Subject: [PATCH 4/6] anchor --- .../PowerChargeBoundUserInterface.cs | 39 +++ .../Power/PowerCharge/PowerChargeComponent.cs | 10 + .../Power/PowerCharge/PowerChargeWindow.xaml | 33 ++ .../PowerCharge/PowerChargeWindow.xaml.cs | 72 +++++ .../Power/Components/PowerChargeComponent.cs | 66 ++++ .../Power/EntitySystems/PowerChargeSystem.cs | 283 ++++++++++++++++++ .../Components/StationAnchorComponent.cs | 11 + .../Shuttles/Systems/StationAnchorSystem.cs | 86 ++++++ Content.Shared/Power/SharedPowerCharge.cs | 77 +++++ .../Power/SharedPowerChargeComponent.cs | 14 + .../Shuttles/Systems/SharedShuttleSystem.cs | 3 +- .../components/station-anchor-component.ftl | 2 + .../components/power-charging-component.ftl | 22 ++ .../Circuitboards/Machine/production.yml | 16 + .../Entities/Structures/Machines/lathe.yml | 1 + .../Structures/Shuttles/station_anchor.yml | 118 ++++++++ .../Prototypes/Recipes/Lathes/electronics.yml | 11 + .../Machines/station_anchor.rsi/meta.json | 17 ++ .../station_anchor.rsi/station_anchor.png | Bin 0 -> 3751 bytes .../station_anchor_unlit.png | Bin 0 -> 3330 bytes 20 files changed, 880 insertions(+), 1 deletion(-) create mode 100644 Content.Client/Power/PowerCharge/PowerChargeBoundUserInterface.cs create mode 100644 Content.Client/Power/PowerCharge/PowerChargeComponent.cs create mode 100644 Content.Client/Power/PowerCharge/PowerChargeWindow.xaml create mode 100644 Content.Client/Power/PowerCharge/PowerChargeWindow.xaml.cs create mode 100644 Content.Server/Power/Components/PowerChargeComponent.cs create mode 100644 Content.Server/Power/EntitySystems/PowerChargeSystem.cs create mode 100644 Content.Server/Shuttles/Components/StationAnchorComponent.cs create mode 100644 Content.Server/Shuttles/Systems/StationAnchorSystem.cs create mode 100644 Content.Shared/Power/SharedPowerCharge.cs create mode 100644 Content.Shared/Power/SharedPowerChargeComponent.cs create mode 100644 Resources/Locale/en-US/components/station-anchor-component.ftl create mode 100644 Resources/Locale/en-US/power/components/power-charging-component.ftl create mode 100644 Resources/Prototypes/Entities/Structures/Shuttles/station_anchor.yml create mode 100644 Resources/Textures/Structures/Machines/station_anchor.rsi/meta.json create mode 100644 Resources/Textures/Structures/Machines/station_anchor.rsi/station_anchor.png create mode 100644 Resources/Textures/Structures/Machines/station_anchor.rsi/station_anchor_unlit.png diff --git a/Content.Client/Power/PowerCharge/PowerChargeBoundUserInterface.cs b/Content.Client/Power/PowerCharge/PowerChargeBoundUserInterface.cs new file mode 100644 index 00000000000..58ebdea7d27 --- /dev/null +++ b/Content.Client/Power/PowerCharge/PowerChargeBoundUserInterface.cs @@ -0,0 +1,39 @@ +using Content.Shared.Power; +using Robust.Client.UserInterface; + +namespace Content.Client.Power.PowerCharge; + +public sealed class PowerChargeBoundUserInterface : BoundUserInterface +{ + [ViewVariables] + private PowerChargeWindow? _window; + + public PowerChargeBoundUserInterface(EntityUid owner, Enum uiKey) : base(owner, uiKey) + { + } + + public void SetPowerSwitch(bool on) + { + SendMessage(new SwitchChargingMachineMessage(on)); + } + + protected override void Open() + { + base.Open(); + if (!EntMan.TryGetComponent(Owner, out PowerChargeComponent? component)) + return; + + _window = new PowerChargeWindow(); + _window.UpdateWindow(this, Loc.GetString(component.WindowTitle)); + + } + + protected override void UpdateState(BoundUserInterfaceState state) + { + base.UpdateState(state); + if (state is not PowerChargeState chargeState) + return; + + _window?.UpdateState(chargeState); + } +} diff --git a/Content.Client/Power/PowerCharge/PowerChargeComponent.cs b/Content.Client/Power/PowerCharge/PowerChargeComponent.cs new file mode 100644 index 00000000000..ab5baa4e2f5 --- /dev/null +++ b/Content.Client/Power/PowerCharge/PowerChargeComponent.cs @@ -0,0 +1,10 @@ +using Content.Shared.Power; + +namespace Content.Client.Power.PowerCharge; + +/// +[RegisterComponent] +public sealed partial class PowerChargeComponent : SharedPowerChargeComponent +{ + +} diff --git a/Content.Client/Power/PowerCharge/PowerChargeWindow.xaml b/Content.Client/Power/PowerCharge/PowerChargeWindow.xaml new file mode 100644 index 00000000000..4e61255326e --- /dev/null +++ b/Content.Client/Power/PowerCharge/PowerChargeWindow.xaml @@ -0,0 +1,33 @@ + + + + + +