| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529 |
- using System.Linq;
- using System.Net;
- using System.Net.Sockets;
- using Content.Client.Administration.UI.CustomControls;
- using Content.Shared.Administration;
- using Content.Shared.CCVar;
- using Content.Shared.Database;
- using Content.Shared.Roles;
- using Robust.Client.AutoGenerated;
- using Robust.Client.Graphics;
- using Robust.Client.UserInterface;
- using Robust.Client.UserInterface.Controls;
- using Robust.Client.UserInterface.CustomControls;
- using Robust.Client.UserInterface.XAML;
- using Robust.Shared.Configuration;
- using Robust.Shared.Prototypes;
- using Robust.Shared.Timing;
- using Robust.Shared.Utility;
- namespace Content.Client.Administration.UI.BanPanel;
- [GenerateTypedNameReferences]
- public sealed partial class BanPanel : DefaultWindow
- {
- public event Action<string?, (IPAddress, int)?, bool, ImmutableTypedHwid?, bool, uint, string, NoteSeverity, string[]?, bool>? BanSubmitted;
- public event Action<string>? PlayerChanged;
- private string? PlayerUsername { get; set; }
- private (IPAddress, int)? IpAddress { get; set; }
- private ImmutableTypedHwid? Hwid { get; set; }
- private double TimeEntered { get; set; }
- private uint Multiplier { get; set; }
- private bool HasBanFlag { get; set; }
- private TimeSpan? ButtonResetOn { get; set; }
- // This is less efficient than just holding a reference to the root control and enumerating children, but you
- // have to know how the controls are nested, which makes the code more complicated.
- private readonly List<CheckBox> _roleCheckboxes = new();
- private readonly ISawmill _banpanelSawmill;
- [Dependency] private readonly IGameTiming _gameTiming = default!;
- [Dependency] private readonly IConfigurationManager _cfg = default!;
- [Dependency] private readonly ILogManager _logManager = default!;
- private enum TabNumbers
- {
- BasicInfo,
- //Text,
- Players,
- Roles
- }
- private enum Multipliers
- {
- Minutes,
- Hours,
- Days,
- Weeks,
- Months,
- Years,
- Permanent
- }
- private enum Types
- {
- None,
- Server,
- Role
- }
- public BanPanel()
- {
- RobustXamlLoader.Load(this);
- IoCManager.InjectDependencies(this);
- _banpanelSawmill = _logManager.GetSawmill("admin.banpanel");
- PlayerList.OnSelectionChanged += OnPlayerSelectionChanged;
- PlayerNameLine.OnFocusExit += _ => OnPlayerNameChanged();
- PlayerCheckbox.OnPressed += _ =>
- {
- PlayerNameLine.Editable = PlayerCheckbox.Pressed;
- PlayerNameLine.ModulateSelfOverride = null;
- };
- TimeLine.OnTextChanged += OnMinutesChanged;
- MultiplierOption.OnItemSelected += args =>
- {
- MultiplierOption.SelectId(args.Id);
- OnMultiplierChanged();
- };
- IpLine.OnFocusExit += _ => OnIpChanged();
- IpCheckbox.OnPressed += _ =>
- {
- IpLine.Editable = IpCheckbox.Pressed;
- OnIpChanged();
- };
- HwidLine.OnFocusExit += _ => OnHwidChanged();
- HwidCheckbox.OnPressed += _ =>
- {
- HwidLine.Editable = HwidCheckbox.Pressed;
- OnHwidChanged();
- };
- TypeOption.OnItemSelected += args =>
- {
- TypeOption.SelectId(args.Id);
- OnTypeChanged();
- };
- LastConnCheckbox.OnPressed += args =>
- {
- IpLine.ModulateSelfOverride = null;
- HwidLine.ModulateSelfOverride = null;
- OnIpChanged();
- OnHwidChanged();
- };
- SubmitButton.OnPressed += SubmitButtonOnOnPressed;
- IpCheckbox.Pressed = _cfg.GetCVar(CCVars.ServerBanIpBanDefault);
- HwidCheckbox.Pressed = _cfg.GetCVar(CCVars.ServerBanHwidBanDefault);
- LastConnCheckbox.Pressed = _cfg.GetCVar(CCVars.ServerBanUseLastDetails);
- EraseCheckbox.Pressed = _cfg.GetCVar(CCVars.ServerBanErasePlayer);
- SeverityOption.AddItem(Loc.GetString("admin-note-editor-severity-none"), (int) NoteSeverity.None);
- SeverityOption.AddItem(Loc.GetString("admin-note-editor-severity-low"), (int) NoteSeverity.Minor);
- SeverityOption.AddItem(Loc.GetString("admin-note-editor-severity-medium"), (int) NoteSeverity.Medium);
- SeverityOption.AddItem(Loc.GetString("admin-note-editor-severity-high"), (int) NoteSeverity.High);
- SeverityOption.SelectId((int) NoteSeverity.Medium);
- SeverityOption.OnItemSelected += args => SeverityOption.SelectId(args.Id);
- MultiplierOption.AddItem(Loc.GetString("ban-panel-minutes"), (int) Multipliers.Minutes);
- MultiplierOption.AddItem(Loc.GetString("ban-panel-hours"), (int) Multipliers.Hours);
- MultiplierOption.AddItem(Loc.GetString("ban-panel-days"), (int) Multipliers.Days);
- MultiplierOption.AddItem(Loc.GetString("ban-panel-weeks"), (int) Multipliers.Weeks);
- MultiplierOption.AddItem(Loc.GetString("ban-panel-months"), (int) Multipliers.Months);
- MultiplierOption.AddItem(Loc.GetString("ban-panel-years"), (int) Multipliers.Years);
- MultiplierOption.AddItem(Loc.GetString("ban-panel-permanent"), (int) Multipliers.Permanent);
- MultiplierOption.SelectId((int) Multipliers.Minutes);
- OnMultiplierChanged();
- Tabs.SetTabTitle((int) TabNumbers.BasicInfo, Loc.GetString("ban-panel-tabs-basic"));
- //Tabs.SetTabTitle((int) TabNumbers.Text, Loc.GetString("ban-panel-tabs-reason"));
- Tabs.SetTabTitle((int) TabNumbers.Players, Loc.GetString("ban-panel-tabs-players"));
- Tabs.SetTabTitle((int) TabNumbers.Roles, Loc.GetString("ban-panel-tabs-role"));
- Tabs.SetTabVisible((int) TabNumbers.Roles, false);
- TypeOption.AddItem(Loc.GetString("ban-panel-select"), (int) Types.None);
- TypeOption.AddItem(Loc.GetString("ban-panel-server"), (int) Types.Server);
- TypeOption.AddItem(Loc.GetString("ban-panel-role"), (int) Types.Role);
- ReasonTextEdit.Placeholder = new Rope.Leaf(Loc.GetString("ban-panel-reason"));
- var prototypeManager = IoCManager.Resolve<IPrototypeManager>();
- foreach (var proto in prototypeManager.EnumeratePrototypes<DepartmentPrototype>())
- {
- CreateRoleGroup(proto.ID, proto.Roles.Select(p => p.Id), proto.Color);
- }
- CreateRoleGroup("Antagonist", prototypeManager.EnumeratePrototypes<AntagPrototype>().Select(p => p.ID), Color.Red);
- }
- private void CreateRoleGroup(string roleName, IEnumerable<string> roleList, Color color)
- {
- var outerContainer = new BoxContainer
- {
- Name = $"{roleName}GroupOuterBox",
- HorizontalExpand = true,
- VerticalExpand = true,
- Orientation = BoxContainer.LayoutOrientation.Vertical,
- Margin = new Thickness(4)
- };
- var departmentCheckbox = new CheckBox
- {
- Name = $"{roleName}GroupCheckbox",
- Text = roleName,
- Modulate = color,
- HorizontalAlignment = HAlignment.Left
- };
- outerContainer.AddChild(departmentCheckbox);
- var innerContainer = new BoxContainer
- {
- Name = $"{roleName}GroupInnerBox",
- HorizontalExpand = true,
- Orientation = BoxContainer.LayoutOrientation.Horizontal
- };
- departmentCheckbox.OnToggled += args =>
- {
- foreach (var child in innerContainer.Children)
- {
- if (child is CheckBox c)
- {
- c.Pressed = args.Pressed;
- }
- }
- if (args.Pressed)
- {
- if (!Enum.TryParse(_cfg.GetCVar(CCVars.DepartmentBanDefaultSeverity), true, out NoteSeverity newSeverity))
- {
- _banpanelSawmill
- .Warning("Departmental role ban severity could not be parsed from config!");
- return;
- }
- SeverityOption.SelectId((int) newSeverity);
- }
- else
- {
- foreach (var childContainer in RolesContainer.Children)
- {
- if (childContainer is Container)
- {
- foreach (var child in childContainer.Children)
- {
- if (child is CheckBox { Pressed: true })
- return;
- }
- }
- }
- if (!Enum.TryParse(_cfg.GetCVar(CCVars.RoleBanDefaultSeverity), true, out NoteSeverity newSeverity))
- {
- _banpanelSawmill
- .Warning("Role ban severity could not be parsed from config!");
- return;
- }
- SeverityOption.SelectId((int) newSeverity);
- }
- };
- outerContainer.AddChild(innerContainer);
- foreach (var role in roleList)
- {
- AddRoleCheckbox(role, innerContainer, departmentCheckbox);
- }
- RolesContainer.AddChild(new PanelContainer
- {
- PanelOverride = new StyleBoxFlat
- {
- BackgroundColor = color
- }
- });
- RolesContainer.AddChild(outerContainer);
- RolesContainer.AddChild(new HSeparator());
- }
- private void AddRoleCheckbox(string role, Control container, CheckBox header)
- {
- var roleCheckbox = new CheckBox
- {
- Name = $"{role}RoleCheckbox",
- Text = role
- };
- roleCheckbox.OnToggled += args =>
- {
- if (args is { Pressed: true, Button.Parent: { } } && args.Button.Parent.Children.Where(e => e is CheckBox).All(e => ((CheckBox) e).Pressed))
- header.Pressed = args.Pressed;
- else
- header.Pressed = false;
- };
- container.AddChild(roleCheckbox);
- _roleCheckboxes.Add(roleCheckbox);
- }
- public void UpdateBanFlag(bool newFlag)
- {
- HasBanFlag = newFlag;
- SubmitButton.Visible = HasBanFlag;
- ModulateSelfOverride = HasBanFlag ? Color.Red : null;
- }
- public void UpdatePlayerData(string playerName)
- {
- if (string.IsNullOrEmpty(playerName))
- {
- PlayerNameLine.ModulateSelfOverride = Color.Red;
- ErrorLevel |= ErrorLevelEnum.PlayerName;
- UpdateSubmitEnabled();
- return;
- }
- PlayerNameLine.ModulateSelfOverride = null;
- ErrorLevel &= ~ErrorLevelEnum.PlayerName;
- UpdateSubmitEnabled();
- PlayerUsername = playerName;
- PlayerNameLine.Text = playerName;
- }
- [Flags]
- private enum ErrorLevelEnum : byte
- {
- None = 0,
- Minutes = 1 << 0,
- PlayerName = 1 << 1,
- IpAddress = 1 << 2,
- Hwid = 1 << 3,
- }
- private ErrorLevelEnum ErrorLevel { get; set; }
- private void OnMinutesChanged(LineEdit.LineEditEventArgs args)
- {
- TimeLine.Text = args.Text;
- if (!double.TryParse(args.Text, out var result))
- {
- ExpiresLabel.Text = "err";
- ErrorLevel |= ErrorLevelEnum.Minutes;
- TimeLine.ModulateSelfOverride = Color.Red;
- UpdateSubmitEnabled();
- return;
- }
- ErrorLevel &= ~ErrorLevelEnum.Minutes;
- TimeLine.ModulateSelfOverride = null;
- TimeEntered = result;
- UpdateSubmitEnabled();
- UpdateExpiresLabel();
- }
- private void OnMultiplierChanged()
- {
- TimeLine.Editable = MultiplierOption.SelectedId != (int) Multipliers.Permanent;
- Multiplier = MultiplierOption.SelectedId switch
- {
- (int) Multipliers.Minutes => 1,
- (int) Multipliers.Hours => 60,
- (int) Multipliers.Days => 60 * 24,
- (int) Multipliers.Weeks => 60 * 24 * 7,
- (int) Multipliers.Months => 60 * 24 * 30,
- (int) Multipliers.Years => 60 * 24 * 365,
- (int) Multipliers.Permanent => 0,
- _ => throw new ArgumentOutOfRangeException(nameof(MultiplierOption.SelectedId), "Multiplier out of range")
- };
- UpdateExpiresLabel();
- }
- private void UpdateExpiresLabel()
- {
- var minutes = (uint) (TimeEntered * Multiplier);
- ExpiresLabel.Text = minutes == 0
- ? $"{Loc.GetString("admin-note-editor-expiry-label")} {Loc.GetString("server-ban-string-never")}"
- : $"{Loc.GetString("admin-note-editor-expiry-label")} {DateTime.Now + TimeSpan.FromMinutes(minutes):yyyy/MM/dd HH:mm:ss}";
- }
- private void OnIpChanged()
- {
- if (LastConnCheckbox.Pressed && IpAddress is null || !IpCheckbox.Pressed)
- {
- IpAddress = null;
- ErrorLevel &= ~ErrorLevelEnum.IpAddress;
- IpLine.ModulateSelfOverride = null;
- UpdateSubmitEnabled();
- return;
- }
- var ip = IpLine.Text;
- var hid = "0";
- if (ip.Contains('/'))
- {
- var split = ip.Split('/');
- ip = split[0];
- hid = split[1];
- }
- if (!IPAddress.TryParse(ip, out var parsedIp) || !byte.TryParse(hid, out var hidInt) || hidInt > 128 || hidInt > 32 && parsedIp.AddressFamily == AddressFamily.InterNetwork)
- {
- ErrorLevel |= ErrorLevelEnum.IpAddress;
- IpLine.ModulateSelfOverride = Color.Red;
- UpdateSubmitEnabled();
- return;
- }
- if (hidInt == 0)
- hidInt = (byte) (parsedIp.AddressFamily == AddressFamily.InterNetworkV6 ? 128 : 32);
- IpAddress = (parsedIp, hidInt);
- ErrorLevel &= ~ErrorLevelEnum.IpAddress;
- IpLine.ModulateSelfOverride = null;
- UpdateSubmitEnabled();
- }
- private void OnHwidChanged()
- {
- var hwidString = HwidLine.Text;
- ImmutableTypedHwid? hwid = null;
- if (HwidCheckbox.Pressed && !(string.IsNullOrEmpty(hwidString) && LastConnCheckbox.Pressed) && !ImmutableTypedHwid.TryParse(hwidString, out hwid))
- {
- ErrorLevel |= ErrorLevelEnum.Hwid;
- HwidLine.ModulateSelfOverride = Color.Red;
- UpdateSubmitEnabled();
- return;
- }
- ErrorLevel &= ~ErrorLevelEnum.Hwid;
- HwidLine.ModulateSelfOverride = null;
- UpdateSubmitEnabled();
- if (LastConnCheckbox.Pressed || !HwidCheckbox.Pressed)
- {
- Hwid = null;
- return;
- }
- Hwid = hwid;
- }
- private void OnTypeChanged()
- {
- TypeOption.ModulateSelfOverride = null;
- Tabs.SetTabVisible((int) TabNumbers.Roles, TypeOption.SelectedId == (int) Types.Role);
- NoteSeverity? newSeverity = null;
- switch (TypeOption.SelectedId)
- {
- case (int)Types.Server:
- if (Enum.TryParse(_cfg.GetCVar(CCVars.ServerBanDefaultSeverity), true, out NoteSeverity serverSeverity))
- newSeverity = serverSeverity;
- else
- {
- _banpanelSawmill
- .Warning("Server ban severity could not be parsed from config!");
- }
- break;
- case (int) Types.Role:
- if (Enum.TryParse(_cfg.GetCVar(CCVars.RoleBanDefaultSeverity), true, out NoteSeverity roleSeverity))
- {
- newSeverity = roleSeverity;
- }
- else
- {
- _banpanelSawmill
- .Warning("Role ban severity could not be parsed from config!");
- }
- break;
- }
- if (newSeverity != null)
- SeverityOption.SelectId((int) newSeverity.Value);
- }
- private void UpdateSubmitEnabled()
- {
- SubmitButton.Disabled = ErrorLevel != ErrorLevelEnum.None;
- }
- private void OnPlayerNameChanged()
- {
- if (PlayerUsername == PlayerNameLine.Text)
- return;
- PlayerUsername = PlayerNameLine.Text;
- if (!PlayerCheckbox.Pressed)
- return;
- if (string.IsNullOrWhiteSpace(PlayerUsername))
- ErrorLevel |= ErrorLevelEnum.PlayerName;
- else
- ErrorLevel &= ~ErrorLevelEnum.PlayerName;
- UpdateSubmitEnabled();
- PlayerChanged?.Invoke(PlayerUsername);
- }
- public void OnPlayerSelectionChanged(PlayerInfo? player)
- {
- PlayerNameLine.Text = player?.Username ?? string.Empty;
- OnPlayerNameChanged();
- }
- private void ResetTextEditor(GUIBoundKeyEventArgs _)
- {
- ReasonTextEdit.ModulateSelfOverride = null;
- ReasonTextEdit.OnKeyBindDown -= ResetTextEditor;
- }
- private void SubmitButtonOnOnPressed(BaseButton.ButtonEventArgs obj)
- {
- string[]? roles = null;
- if (TypeOption.SelectedId == (int) Types.Role)
- {
- var rolesList = new List<string>();
- if (_roleCheckboxes.Count == 0)
- throw new DebugAssertException("RoleCheckboxes was empty");
- rolesList.AddRange(_roleCheckboxes.Where(c => c is { Pressed: true, Text: { } }).Select(c => c.Text!));
- if (rolesList.Count == 0)
- {
- Tabs.CurrentTab = (int) TabNumbers.Roles;
- return;
- }
- roles = rolesList.ToArray();
- }
- if (TypeOption.SelectedId == (int) Types.None)
- {
- TypeOption.ModulateSelfOverride = Color.Red;
- Tabs.CurrentTab = (int) TabNumbers.BasicInfo;
- return;
- }
- var reason = Rope.Collapse(ReasonTextEdit.TextRope);
- if (string.IsNullOrWhiteSpace(reason))
- {
- //Tabs.CurrentTab = (int) TabNumbers.Text;
- Tabs.CurrentTab = (int) TabNumbers.BasicInfo;
- ReasonTextEdit.GrabKeyboardFocus();
- ReasonTextEdit.ModulateSelfOverride = Color.Red;
- ReasonTextEdit.OnKeyBindDown += ResetTextEditor;
- return;
- }
- if (ButtonResetOn is null)
- {
- ButtonResetOn = _gameTiming.CurTime.Add(TimeSpan.FromSeconds(3));
- SubmitButton.ModulateSelfOverride = Color.Red;
- SubmitButton.Text = Loc.GetString("ban-panel-confirm");
- return;
- }
- var player = PlayerCheckbox.Pressed ? PlayerUsername : null;
- var useLastIp = IpCheckbox.Pressed && LastConnCheckbox.Pressed && IpAddress is null;
- var useLastHwid = HwidCheckbox.Pressed && LastConnCheckbox.Pressed && Hwid is null;
- var severity = (NoteSeverity) SeverityOption.SelectedId;
- var erase = EraseCheckbox.Pressed;
- BanSubmitted?.Invoke(player, IpAddress, useLastIp, Hwid, useLastHwid, (uint) (TimeEntered * Multiplier), reason, severity, roles, erase);
- }
- protected override void FrameUpdate(FrameEventArgs args)
- {
- base.FrameUpdate(args);
- // This checks for null for free, do not invert it as null always produces a false value
- if (_gameTiming.CurTime > ButtonResetOn)
- {
- ButtonResetOn = null;
- SubmitButton.ModulateSelfOverride = null;
- SubmitButton.Text = Loc.GetString("ban-panel-submit");
- }
- }
- }
|