forked from wylab/wylab-station-14
25c5e59248
Signed-off-by: Prole <172158352+Prole0@users.noreply.github.com> Co-authored-by: PJBot <pieterjan.briers+bot@gmail.com> Co-authored-by: ScarKy0 <106310278+ScarKy0@users.noreply.github.com> Co-authored-by: Samuka-C <47865393+Samuka-C@users.noreply.github.com> Co-authored-by: ArtisticRoomba <145879011+ArtisticRoomba@users.noreply.github.com> Co-authored-by: Partmedia <kevinz5000@gmail.com> Co-authored-by: slarticodefast <161409025+slarticodefast@users.noreply.github.com> Co-authored-by: themias <89101928+themias@users.noreply.github.com> Co-authored-by: Victor Shen <71985089+Vexerot@users.noreply.github.com> Co-authored-by: Ed <96445749+TheShuEd@users.noreply.github.com> Co-authored-by: Milon <milonpl.git@proton.me> Co-authored-by: Kirus59 <145689588+Kirus59@users.noreply.github.com> Co-authored-by: Nemanja <98561806+EmoGarbage404@users.noreply.github.com> Co-authored-by: Stomf <5dorkydorks@gmail.com> Co-authored-by: drakewill-CRL <46307022+drakewill-CRL@users.noreply.github.com> Co-authored-by: PraxisMapper <praxismapper@gmail.com> Co-authored-by: EmoGarbage404 <retron404@gmail.com> Co-authored-by: lzk <124214523+lzk228@users.noreply.github.com> Co-authored-by: Princess Cheeseballs <66055347+Pronana@users.noreply.github.com> Co-authored-by: IProduceWidgets <107586145+IProduceWidgets@users.noreply.github.com> Co-authored-by: TytosB <54259736+TytosB@users.noreply.github.com> Co-authored-by: abadaba695 <spacestation13thingy@gmail.com> Co-authored-by: kosticia <kosticia46@gmail.com> Co-authored-by: Thinbug <101073555+Thinbug0@users.noreply.github.com> Co-authored-by: pathetic meowmeow <uhhadd@gmail.com> Co-authored-by: SlamBamActionman <83650252+SlamBamActionman@users.noreply.github.com> Co-authored-by: Boaz1111 <149967078+Boaz1111@users.noreply.github.com> Co-authored-by: ActiveMammmoth <140334666+ActiveMammmoth@users.noreply.github.com> Co-authored-by: Myra <vasilis@pikachu.systems> Co-authored-by: Whatstone <166147148+whatston3@users.noreply.github.com> Co-authored-by: Pieter-Jan Briers <pieterjan.briers+git@gmail.com> Co-authored-by: K-Dynamic <20566341+K-Dynamic@users.noreply.github.com> Co-authored-by: Gentleman-Bird <dcgreen406@gmail.com> Co-authored-by: metalgearsloth <31366439+metalgearsloth@users.noreply.github.com> Co-authored-by: BIGZi0348 <svalker0348@gmail.com> Co-authored-by: LaCumbiaDelCoronavirus <90893484+LaCumbiaDelCoronavirus@users.noreply.github.com> Co-authored-by: imatsoup <93290208+imatsoup@users.noreply.github.com> Co-authored-by: Matthew Herber <32679887+happyrobot33@users.noreply.github.com> Co-authored-by: Ertanic <36124833+Ertanic@users.noreply.github.com> Co-authored-by: MissKay1994 <15877268+MissKay1994@users.noreply.github.com> Co-authored-by: Errant <35878406+Errant-4@users.noreply.github.com> Co-authored-by: eoineoineoin <helloworld@eoinrul.es> Co-authored-by: Tiniest Shark <head.rebel@yahoo.com> Co-authored-by: nikitosych <boriszyn@gmail.com> Co-authored-by: Tayrtahn <tayrtahn@gmail.com> Co-authored-by: Perry Fraser <perryprog@users.noreply.github.com> Co-authored-by: YoungThug <ramialanbagy@gmail.com> Co-authored-by: beck-thompson <107373427+beck-thompson@users.noreply.github.com> Co-authored-by: metalgearsloth <comedian_vs_clown@hotmail.com> Co-authored-by: Southbridge <7013162+southbridge-fur@users.noreply.github.com> Co-authored-by: Vladislav Suchkov <20380250+murolem@users.noreply.github.com> Co-authored-by: Prole <172158352+Prole0@users.noreply.github.com> Co-authored-by: Unkn0wn_Gh0st <shadowstalkermll@gmail.com> Co-authored-by: 3nderall <101940324+3nderall@users.noreply.github.com> Co-authored-by: Radezolid <snappednexus@gmail.com> Co-authored-by: J <billsmith116@gmail.com> Co-authored-by: Ghagliiarghii <68826635+Ghagliiarghii@users.noreply.github.com> Co-authored-by: chromiumboy <50505512+chromiumboy@users.noreply.github.com> Co-authored-by: youtissoum <51883137+youtissoum@users.noreply.github.com> Co-authored-by: Minemoder5000 <minemoder50000@gmail.com> Co-authored-by: Spanky <scott@wearejacob.com> Co-authored-by: Spessmann <156740760+Spessmann@users.noreply.github.com> Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com> Co-authored-by: brainfood1183 <113240905+brainfood1183@users.noreply.github.com> Co-authored-by: Deerstop <edainturner@gmail.com> Co-authored-by: B_Kirill <153602297+B-Kirill@users.noreply.github.com> Co-authored-by: archee1 <archee3@hotmail.co.uk> Co-authored-by: Cojoke <83733158+Cojoke-dot@users.noreply.github.com> Co-authored-by: Quantum-cross <7065792+Quantum-cross@users.noreply.github.com> Co-authored-by: poklj <compgeek223@gmail.com> Co-authored-by: Krunklehorn <42424291+Krunklehorn@users.noreply.github.com> Co-authored-by: OnyxTheBrave <131422822+OnyxTheBrave@users.noreply.github.com> Co-authored-by: UpAndLeaves <92269094+Alpha-Two@users.noreply.github.com> Co-authored-by: Flareguy <78941145+Flareguy@users.noreply.github.com> Co-authored-by: Zalycon <84675130+Zalycon@users.noreply.github.com> Co-authored-by: deltanedas <39013340+deltanedas@users.noreply.github.com> Co-authored-by: Verm <32827189+Vermidia@users.noreply.github.com> Co-authored-by: nikthechampiongr <32041239+nikthechampiongr@users.noreply.github.com> Co-authored-by: ScarKy0 <scarky0@onet.eu> Co-authored-by: Dmitry <57028746+dimm00n@users.noreply.github.com>
313 lines
11 KiB
C#
313 lines
11 KiB
C#
using System.Linq;
|
|
using System.Text;
|
|
using Content.Client.Administration.Managers;
|
|
using Content.Client.Administration.UI.CustomControls;
|
|
using Content.Client.UserInterface.Systems.Bwoink;
|
|
using Content.Shared.Administration;
|
|
using Content.Shared.CCVar;
|
|
using Robust.Client.AutoGenerated;
|
|
using Robust.Client.Console;
|
|
using Robust.Client.UserInterface;
|
|
using Robust.Client.UserInterface.Controls;
|
|
using Robust.Client.UserInterface.XAML;
|
|
using Robust.Shared.Network;
|
|
using Robust.Shared.Configuration;
|
|
using Robust.Shared.Utility;
|
|
|
|
namespace Content.Client.Administration.UI.Bwoink
|
|
{
|
|
/// <summary>
|
|
/// This window connects to a BwoinkSystem channel. BwoinkSystem manages the rest.
|
|
/// </summary>
|
|
[GenerateTypedNameReferences]
|
|
public sealed partial class BwoinkControl : Control
|
|
{
|
|
[Dependency] private readonly IClientAdminManager _adminManager = default!;
|
|
[Dependency] private readonly IClientConsoleHost _console = default!;
|
|
[Dependency] private readonly IUserInterfaceManager _ui = default!;
|
|
[Dependency] private readonly IConfigurationManager _cfg = default!;
|
|
public AdminAHelpUIHandler AHelpHelper = default!;
|
|
|
|
private PlayerInfo? _currentPlayer;
|
|
|
|
public BwoinkControl()
|
|
{
|
|
RobustXamlLoader.Load(this);
|
|
IoCManager.InjectDependencies(this);
|
|
|
|
var newPlayerThreshold = 0;
|
|
_cfg.OnValueChanged(CCVars.NewPlayerThreshold, (val) => { newPlayerThreshold = val; }, true);
|
|
|
|
var uiController = _ui.GetUIController<AHelpUIController>();
|
|
if (uiController.UIHelper is not AdminAHelpUIHandler helper)
|
|
return;
|
|
|
|
AHelpHelper = helper;
|
|
|
|
_adminManager.AdminStatusUpdated += UpdateButtons;
|
|
UpdateButtons();
|
|
|
|
AdminOnly.OnToggled += args => PlaySound.Disabled = args.Pressed;
|
|
|
|
ChannelSelector.OnSelectionChanged += sel =>
|
|
{
|
|
_currentPlayer = sel;
|
|
SwitchToChannel(sel?.SessionId);
|
|
ChannelSelector.PlayerListContainer.DirtyList();
|
|
};
|
|
|
|
ChannelSelector.OverrideText += (info, text) =>
|
|
{
|
|
var sb = new StringBuilder();
|
|
|
|
if (info.Connected)
|
|
sb.Append(info.ActiveThisRound ? '⚫' : '◐');
|
|
else
|
|
sb.Append(info.ActiveThisRound ? '⭘' : '·');
|
|
|
|
sb.Append(' ');
|
|
if (AHelpHelper.TryGetChannel(info.SessionId, out var panel) && panel.Unread > 0)
|
|
{
|
|
if (panel.Unread < 11)
|
|
sb.Append(new Rune('➀' + (panel.Unread-1)));
|
|
else
|
|
sb.Append(new Rune(0x2639)); // ☹
|
|
sb.Append(' ');
|
|
}
|
|
|
|
// Mark antagonists with symbol
|
|
if (info.Antag && info.ActiveThisRound)
|
|
sb.Append(new Rune(0x1F5E1)); // 🗡
|
|
|
|
// Mark new players with symbol
|
|
if (IsNewPlayer(info))
|
|
sb.Append(new Rune(0x23F2)); // ⏲
|
|
|
|
sb.AppendFormat("\"{0}\"", text);
|
|
|
|
return sb.ToString();
|
|
};
|
|
|
|
// <summary>
|
|
// Returns true if the player's overall playtime is under the set threshold
|
|
// </summary>
|
|
bool IsNewPlayer(PlayerInfo info)
|
|
{
|
|
// Don't show every disconnected player as new, don't show 0-minute players as new if threshold is
|
|
if (newPlayerThreshold <= 0 || info.OverallPlaytime is null && !info.Connected)
|
|
return false;
|
|
|
|
return (info.OverallPlaytime is null
|
|
|| info.OverallPlaytime < TimeSpan.FromMinutes(newPlayerThreshold));
|
|
}
|
|
|
|
ChannelSelector.Comparison = (a, b) =>
|
|
{
|
|
var ach = AHelpHelper.EnsurePanel(a.SessionId);
|
|
var bch = AHelpHelper.EnsurePanel(b.SessionId);
|
|
|
|
// Pinned players first
|
|
if (a.IsPinned != b.IsPinned)
|
|
return a.IsPinned ? -1 : 1;
|
|
|
|
// Then, any chat with unread messages.
|
|
var aUnread = ach.Unread > 0;
|
|
var bUnread = bch.Unread > 0;
|
|
if (aUnread != bUnread)
|
|
return aUnread ? -1 : 1;
|
|
|
|
// Then, any chat with recent messages from the current round
|
|
var aRecent = a.ActiveThisRound && ach.LastMessage != DateTime.MinValue;
|
|
var bRecent = b.ActiveThisRound && bch.LastMessage != DateTime.MinValue;
|
|
if (aRecent != bRecent)
|
|
return aRecent ? -1 : 1;
|
|
|
|
// Sort by connection status. Disconnected players will be last.
|
|
if (a.Connected != b.Connected)
|
|
return a.Connected ? -1 : 1;
|
|
|
|
// Sort connected players by whether they have joined the round, then by New Player status, then by Antag status
|
|
if (a.Connected && b.Connected)
|
|
{
|
|
var aNewPlayer = IsNewPlayer(a);
|
|
var bNewPlayer = IsNewPlayer(b);
|
|
|
|
// Players who have joined the round will be listed before players in the lobby
|
|
if (a.ActiveThisRound != b.ActiveThisRound)
|
|
return a.ActiveThisRound ? -1 : 1;
|
|
|
|
// Within both the joined group and lobby group, new players will be grouped and listed first
|
|
if (aNewPlayer != bNewPlayer)
|
|
return aNewPlayer ? -1 : 1;
|
|
|
|
// Within all four previous groups, antagonists will be listed first.
|
|
if (a.Antag != b.Antag)
|
|
return a.Antag ? -1 : 1;
|
|
}
|
|
|
|
// Sort disconnected players by participation in the round
|
|
if (!a.Connected && !b.Connected)
|
|
{
|
|
if (a.ActiveThisRound != b.ActiveThisRound)
|
|
return a.ActiveThisRound ? -1 : 1;
|
|
}
|
|
|
|
// Finally, sort by the most recent message.
|
|
return bch.LastMessage.CompareTo(ach.LastMessage);
|
|
};
|
|
|
|
|
|
Bans.OnPressed += _ =>
|
|
{
|
|
if (_currentPlayer is not null)
|
|
_console.ExecuteCommand($"banlist \"{_currentPlayer.SessionId}\"");
|
|
};
|
|
|
|
Notes.OnPressed += _ =>
|
|
{
|
|
if (_currentPlayer is not null)
|
|
_console.ExecuteCommand($"adminnotes \"{_currentPlayer.SessionId}\"");
|
|
};
|
|
|
|
Ban.OnPressed += _ =>
|
|
{
|
|
if (_currentPlayer is not null)
|
|
_console.ExecuteCommand($"banpanel \"{_currentPlayer.SessionId}\"");
|
|
};
|
|
|
|
Kick.OnPressed += _ =>
|
|
{
|
|
// TODO: Reason field
|
|
if (_currentPlayer is not null)
|
|
_console.ExecuteCommand($"kick \"{_currentPlayer.Username}\"");
|
|
};
|
|
|
|
Follow.OnPressed += _ =>
|
|
{
|
|
if (_currentPlayer is not null)
|
|
_console.ExecuteCommand($"follow \"{_currentPlayer.NetEntity}\"");
|
|
};
|
|
|
|
Respawn.OnPressed += _ =>
|
|
{
|
|
if (_currentPlayer is not null)
|
|
_console.ExecuteCommand($"respawn \"{_currentPlayer.Username}\"");
|
|
};
|
|
|
|
PopOut.OnPressed += _ =>
|
|
{
|
|
uiController.PopOut();
|
|
};
|
|
}
|
|
|
|
public void OnBwoink(NetUserId channel)
|
|
{
|
|
ChannelSelector.PopulateList();
|
|
}
|
|
|
|
|
|
public void SelectChannel(NetUserId channel)
|
|
{
|
|
if (!ChannelSelector.PlayerInfo.TryFirstOrDefault(
|
|
i => i.SessionId == channel, out var info))
|
|
return;
|
|
|
|
// clear filter if we're trying to select a channel for a player that isn't currently filtered
|
|
// i.e. through the message verb.
|
|
var data = new PlayerListData(info);
|
|
if (!ChannelSelector.PlayerListContainer.Data.Contains(data))
|
|
{
|
|
ChannelSelector.StopFiltering();
|
|
}
|
|
|
|
ChannelSelector.PopulateList();
|
|
ChannelSelector.PlayerListContainer.Select(data);
|
|
}
|
|
|
|
public void UpdateButtons()
|
|
{
|
|
var disabled = _currentPlayer == null;
|
|
|
|
Bans.Visible = _adminManager.HasFlag(AdminFlags.Ban);
|
|
Bans.Disabled = !Bans.Visible || disabled;
|
|
|
|
Notes.Visible = _adminManager.HasFlag(AdminFlags.ViewNotes);
|
|
Notes.Disabled = !Notes.Visible || disabled;
|
|
|
|
Ban.Visible = _adminManager.HasFlag(AdminFlags.Ban);
|
|
Ban.Disabled = !Ban.Visible || disabled;
|
|
|
|
Kick.Visible = _adminManager.CanCommand("kick");
|
|
Kick.Disabled = !Kick.Visible || disabled;
|
|
|
|
Respawn.Visible = _adminManager.CanCommand("respawn");
|
|
Respawn.Disabled = !Respawn.Visible || disabled;
|
|
|
|
Follow.Visible = _adminManager.CanCommand("follow");
|
|
Follow.Disabled = !Follow.Visible || disabled;
|
|
}
|
|
|
|
private string FormatTabTitle(ItemList.Item li, PlayerInfo? pl = default)
|
|
{
|
|
pl ??= (PlayerInfo) li.Metadata!;
|
|
var sb = new StringBuilder();
|
|
sb.Append(pl.Connected ? '●' : '○');
|
|
sb.Append(' ');
|
|
if (AHelpHelper.TryGetChannel(pl.SessionId, out var panel) && panel.Unread > 0)
|
|
{
|
|
if (panel.Unread < 11)
|
|
sb.Append(new Rune('➀' + (panel.Unread-1)));
|
|
else
|
|
sb.Append(new Rune(0x2639)); // ☹
|
|
sb.Append(' ');
|
|
}
|
|
|
|
if (pl.Antag)
|
|
sb.Append(new Rune(0x1F5E1)); // 🗡
|
|
|
|
if (pl.OverallPlaytime <= TimeSpan.FromMinutes(_cfg.GetCVar(CCVars.NewPlayerThreshold)))
|
|
sb.Append(new Rune(0x23F2)); // ⏲
|
|
|
|
sb.AppendFormat("\"{0}\"", pl.CharacterName);
|
|
|
|
if (pl.IdentityName != pl.CharacterName && pl.IdentityName != string.Empty)
|
|
sb.Append(' ').AppendFormat("[{0}]", pl.IdentityName);
|
|
|
|
sb.Append(' ').Append(pl.Username);
|
|
|
|
return sb.ToString();
|
|
}
|
|
|
|
private void SwitchToChannel(NetUserId? ch)
|
|
{
|
|
UpdateButtons();
|
|
|
|
AHelpHelper.HideAllPanels();
|
|
if (ch != null)
|
|
{
|
|
var panel = AHelpHelper.EnsurePanel(ch.Value);
|
|
panel.Visible = true;
|
|
}
|
|
}
|
|
|
|
public void PopulateList()
|
|
{
|
|
// Maintain existing pin statuses
|
|
var pinnedPlayers = ChannelSelector.PlayerInfo.Where(p => p.IsPinned).ToDictionary(p => p.SessionId);
|
|
|
|
ChannelSelector.PopulateList();
|
|
|
|
// Restore pin statuses
|
|
foreach (var player in ChannelSelector.PlayerInfo)
|
|
{
|
|
if (pinnedPlayers.TryGetValue(player.SessionId, out var pinnedPlayer))
|
|
{
|
|
player.IsPinned = pinnedPlayer.IsPinned;
|
|
}
|
|
}
|
|
|
|
UpdateButtons();
|
|
}
|
|
}
|
|
}
|