Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion Content.Client/Pinpointer/UI/NavMapControl.cs
Original file line number Diff line number Diff line change
Expand Up @@ -81,7 +81,7 @@ public partial class NavMapControl : MapGridControl
// Components
private NavMapComponent? _navMap;
private MapGridComponent? _grid;
private TransformComponent? _xform;
protected TransformComponent? _xform;
private PhysicsComponent? _physics;
private FixturesComponent? _fixtures;

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
// LuaWorld - This file is licensed under AGPLv3
// Copyright (c) 2026 LuaCorp
// See AGPLv3.txt for details.

using Content.Shared._Lua.StationTeleporter;
using Robust.Client.UserInterface;

namespace Content.Client._Lua.StationTeleporter;

public sealed class StationTeleporterBoundUserInterface : BoundUserInterface
{
private StationTeleporterWindow? _window;

public StationTeleporterBoundUserInterface(EntityUid owner, Enum key) : base(owner, key)
{
}

protected override void Open()
{
base.Open();
_window = this.CreateWindow<StationTeleporterWindow>();
_window.SetOwner(Owner);
_window.OnTeleporterClick += targetUid =>
{ SendMessage(new StationTeleporterClickMessage(targetUid)); };
_window.OnRename += newName =>
{ SendMessage(new StationTeleporterRenameMessage(newName)); };
}

protected override void UpdateState(BoundUserInterfaceState state)
{
base.UpdateState(state);
if (state is StationTeleporterState s) _window?.UpdateState(s);
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
// LuaWorld - This file is licensed under AGPLv3
// Copyright (c) 2026 LuaCorp
// See AGPLv3.txt for details.

using Content.Client.Pinpointer.UI;
using Robust.Client.Graphics;
using Robust.Client.UserInterface.Controls;
using Robust.Shared.Map;
using System.Numerics;

namespace Content.Client._Lua.StationTeleporter;

public sealed class StationTeleporterNavMapControl : NavMapControl
{
[Dependency] private readonly IEntitySystemManager _entitySystemManager = default!;
private readonly SharedTransformSystem _transformSystem;
public List<(MapCoordinates A, MapCoordinates B)> LinkedTeleporterPairs = new();

public StationTeleporterNavMapControl() : base()
{
IoCManager.InjectDependencies(this);
_transformSystem = _entitySystemManager.GetEntitySystem<SharedTransformSystem>();
PostWallDrawingAction += DrawTeleporterLinks;
try
{
var topContainer = (BoxContainer) GetChild(0);
var topPanel = (PanelContainer) topContainer.GetChild(0);
var innerBox = (BoxContainer) topPanel.GetChild(0);
var beaconsCheckBox = (CheckBox) innerBox.GetChild(1);
beaconsCheckBox.Pressed = false;
}
catch
{
}
}

private void DrawTeleporterLinks(DrawingHandleScreen handle)
{
if (_xform == null) return;
var offset = GetOffset();
var invMatrix = _transformSystem.GetInvWorldMatrix(_xform);
foreach (var (a, b) in LinkedTeleporterPairs)
{
if (a.MapId != b.MapId || a.MapId == MapId.Nullspace) continue;
var posA = Vector2.Transform(a.Position, invMatrix) - offset;
posA = ScalePosition(new Vector2(posA.X, -posA.Y));
var posB = Vector2.Transform(b.Position, invMatrix) - offset;
posB = ScalePosition(new Vector2(posB.X, -posB.Y));
handle.DrawLine(posA, posB, Color.FromHex("#81ddeb"));
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,84 @@
// LuaWorld - This file is licensed under AGPLv3
// Copyright (c) 2026 LuaCorp
// See AGPLv3.txt for details.

using Content.Client.Shuttles.UI;
using Robust.Client.Graphics;
using Robust.Client.UserInterface;
using Robust.Shared.Map;
using System.Numerics;

namespace Content.Client._Lua.StationTeleporter;

public sealed class StationTeleporterSectorMapControl : Control
{
public readonly ShuttleMapControl InnerMap;
private readonly TeleporterLinkOverlay _overlay;

public List<(MapCoordinates A, MapCoordinates B)> LinkedTeleporterPairs
{
get => _overlay.LinkedTeleporterPairs;
set => _overlay.LinkedTeleporterPairs = value;
}

public StationTeleporterSectorMapControl()
{
InnerMap = new ShuttleMapControl();
_overlay = new TeleporterLinkOverlay(InnerMap);

AddChild(InnerMap);
AddChild(_overlay);
MouseFilter = MouseFilterMode.Pass;
_overlay.MouseFilter = MouseFilterMode.Ignore;
}

protected override Vector2 ArrangeOverride(Vector2 finalSize)
{
var side = Math.Min(finalSize.X, finalSize.Y);
var square = new Vector2(side, side);
var box = UIBox2.FromDimensions(Vector2.Zero, square);
InnerMap.Arrange(box);
_overlay.Arrange(box);
return square;
}

protected override Vector2 MeasureOverride(Vector2 availableSize)
{
var side = availableSize.X;
var square = new Vector2(side, side);
InnerMap.Measure(square);
_overlay.Measure(square);
return square;
}
private sealed class TeleporterLinkOverlay : Control
{
private readonly ShuttleMapControl _map;
public List<(MapCoordinates A, MapCoordinates B)> LinkedTeleporterPairs = new();
public TeleporterLinkOverlay(ShuttleMapControl map)
{ _map = map; }

protected override void Draw(DrawingHandleScreen handle)
{
if (LinkedTeleporterPairs.Count == 0) return;
var worldRange = _map.WorldRange;
if (worldRange <= 0f) return;
var px = Math.Min(_map.PixelWidth, _map.PixelHeight);
var midPoint = px / 2f;
var minimapScale = midPoint / worldRange;
var midVec = new Vector2(midPoint, midPoint);
var offset = _map.Offset;
var matty = Matrix3Helpers.CreateInverseTransform(offset, Angle.Zero);
foreach (var (a, b) in LinkedTeleporterPairs)
{
if (a.MapId != b.MapId || a.MapId == MapId.Nullspace) continue;
var posA = Vector2.Transform(a.Position, matty);
posA = posA with { Y = -posA.Y };
posA = posA * minimapScale + midVec;
var posB = Vector2.Transform(b.Position, matty);
posB = posB with { Y = -posB.Y };
posB = posB * minimapScale + midVec;
handle.DrawLine(posA, posB, Color.FromHex("#81ddeb"));
}
}
}
}
29 changes: 29 additions & 0 deletions Content.Client/_Lua/StationTeleporter/StationTeleporterSystem.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
// LuaWorld - This file is licensed under AGPLv3
// Copyright (c) 2026 LuaCorp
// See AGPLv3.txt for details.

using Content.Shared._Lua.StationTeleporter;
using Content.Shared._Lua.StationTeleporter.Components;
using Robust.Client.GameObjects;

namespace Content.Client._Lua.StationTeleporter;

public sealed class StationTeleporterSystem : SharedStationTeleporterSystem
{
[Dependency] private readonly SharedAppearanceSystem _appearance = default!;

public override void Initialize()
{
base.Initialize();
SubscribeLocalEvent<StationTeleporterComponent, AppearanceChangeEvent>(OnAppearanceChange);
}

private void OnAppearanceChange(Entity<StationTeleporterComponent> ent, ref AppearanceChangeEvent args)
{
if (ent.Comp.PortalLayerMap is null) return;
if (!_appearance.TryGetData<Color>(ent, TeleporterPortalVisuals.Color, out var newColor, args.Component)) return;
if (!TryComp<SpriteComponent>(ent, out var sprite)) return;
if (!sprite.LayerMapTryGet(ent.Comp.PortalLayerMap, out var index)) return;
sprite.LayerSetColor(index, newColor);
}
}
70 changes: 70 additions & 0 deletions Content.Client/_Lua/StationTeleporter/StationTeleporterWindow.xaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
<controls:FancyWindow xmlns="https://spacestation14.io"
xmlns:lua="clr-namespace:Content.Client._Lua.StationTeleporter"
xmlns:controls="clr-namespace:Content.Client.UserInterface.Controls"
Title="{Loc 'station-teleporter-console-title'}"
Resizable="False"
SetSize="1010 750"
MinSize="1010 750">
<BoxContainer Orientation="Vertical">

<BoxContainer Orientation="Horizontal" VerticalExpand="True" HorizontalExpand="True">

<!-- Левая панель: карта (переключаются через Visible, скрытый не занимает место) -->
<BoxContainer Orientation="Vertical" Margin="5 5">
<!-- Local/Interstellar: NavMap грида -->
<lua:StationTeleporterNavMapControl Name="NavMap"
Visible="False"/>
<!-- Sector: карта сектора -->
<lua:StationTeleporterSectorMapControl Name="SectorMap"
SetSize="650 650"
Visible="False"/>
</BoxContainer>

<!-- Правая панель -->
<BoxContainer Orientation="Vertical" SetWidth="340" Margin="0 0 10 10">

<controls:StripeBack>
<PanelContainer>
<Label Name="StationName"
Text=""
HorizontalAlignment="Center"
VerticalAlignment="Center"
Margin="0 5 0 3"/>
</PanelContainer>
</controls:StripeBack>

<BoxContainer Name="RenamePanel" Orientation="Vertical" Margin="0 6 0 6" Visible="False">
<Label Text="{Loc 'station-teleporter-rename-label'}"
HorizontalAlignment="Center"
Margin="0 0 0 2"/>
<BoxContainer Orientation="Horizontal" HorizontalExpand="True">
<LineEdit Name="RenameEdit"
HorizontalExpand="True"
PlaceHolder="{Loc 'station-teleporter-rename-placeholder'}"/>
<Button Name="RenameBtn"
Text="{Loc 'station-teleporter-btn-rename'}"
MinWidth="90"
Margin="4 0 0 0"/>
</BoxContainer>
</BoxContainer>

<ScrollContainer HorizontalExpand="True" VerticalExpand="True" Margin="0 4 0 0">
<BoxContainer Name="TeleporterList" Orientation="Vertical" HorizontalExpand="True"/>
</ScrollContainer>

</BoxContainer>
</BoxContainer>

<BoxContainer Orientation="Vertical">
<PanelContainer StyleClasses="LowDivider"/>
<BoxContainer Orientation="Horizontal" Margin="10 2 5 0" VerticalAlignment="Bottom">
<Label Text="{Loc 'station-teleporter-flavor-left'}" StyleClasses="WindowFooterText"/>
<Label Text="{Loc 'station-teleporter-flavor-right'}" StyleClasses="WindowFooterText"
HorizontalAlignment="Right" HorizontalExpand="True" Margin="0 0 5 0"/>
<TextureRect StyleClasses="NTLogoDark" Stretch="KeepAspectCentered"
VerticalAlignment="Center" HorizontalAlignment="Right" SetSize="19 19"/>
</BoxContainer>
</BoxContainer>

</BoxContainer>
</controls:FancyWindow>
Loading
Loading