Add chat.max_announcement_length cvar (#23571)
* Add announce message length to UI and make a cvar for it * Update comm console server-side trim to use the cvar * Rely on the new OnTextChanged event Because OnKeyBindUp only works for keys that have binds * Add a similar indicator to nukies' war declaration UI * Remove message length indicators for now cuz it requires the engine update * Rename cvar slightly * Refactor duplicated code to a helper method * Remove message trimming from *Window class as it's better to live in the BoundUserInterface where the other message handling happens * Rename to chat.max_announcement_length
This commit is contained in:
@@ -1,5 +1,7 @@
|
|||||||
using Content.Shared.Communications;
|
using Content.Shared.CCVar;
|
||||||
using Robust.Client.GameObjects;
|
using Content.Shared.Chat;
|
||||||
|
using Content.Shared.Communications;
|
||||||
|
using Robust.Shared.Configuration;
|
||||||
using Robust.Shared.Timing;
|
using Robust.Shared.Timing;
|
||||||
|
|
||||||
namespace Content.Client.Communications.UI
|
namespace Content.Client.Communications.UI
|
||||||
@@ -7,6 +9,7 @@ namespace Content.Client.Communications.UI
|
|||||||
public sealed class CommunicationsConsoleBoundUserInterface : BoundUserInterface
|
public sealed class CommunicationsConsoleBoundUserInterface : BoundUserInterface
|
||||||
{
|
{
|
||||||
[Dependency] private readonly IGameTiming _gameTiming = default!;
|
[Dependency] private readonly IGameTiming _gameTiming = default!;
|
||||||
|
[Dependency] private readonly IConfigurationManager _cfg = default!;
|
||||||
|
|
||||||
[ViewVariables]
|
[ViewVariables]
|
||||||
private CommunicationsConsoleMenu? _menu;
|
private CommunicationsConsoleMenu? _menu;
|
||||||
@@ -63,22 +66,9 @@ namespace Content.Client.Communications.UI
|
|||||||
|
|
||||||
public void AnnounceButtonPressed(string message)
|
public void AnnounceButtonPressed(string message)
|
||||||
{
|
{
|
||||||
var msg = (message.Length <= 256 ? message.Trim() : $"{message.Trim().Substring(0, 256)}...").ToCharArray();
|
var maxLength = _cfg.GetCVar(CCVars.ChatMaxAnnouncementLength);
|
||||||
|
var msg = SharedChatSystem.SanitizeAnnouncement(message, maxLength);
|
||||||
// No more than 2 newlines, other replaced to spaces
|
SendMessage(new CommunicationsConsoleAnnounceMessage(msg));
|
||||||
var newlines = 0;
|
|
||||||
for (var i = 0; i < msg.Length; i++)
|
|
||||||
{
|
|
||||||
if (msg[i] != '\n')
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if (newlines >= 2)
|
|
||||||
msg[i] = ' ';
|
|
||||||
|
|
||||||
newlines++;
|
|
||||||
}
|
|
||||||
|
|
||||||
SendMessage(new CommunicationsConsoleAnnounceMessage(new string(msg)));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void CallShuttle()
|
public void CallShuttle()
|
||||||
|
|||||||
@@ -23,7 +23,7 @@ namespace Content.Client.Communications.UI
|
|||||||
var loc = IoCManager.Resolve<ILocalizationManager>();
|
var loc = IoCManager.Resolve<ILocalizationManager>();
|
||||||
MessageInput.Placeholder = new Rope.Leaf(loc.GetString("comms-console-menu-announcement-placeholder"));
|
MessageInput.Placeholder = new Rope.Leaf(loc.GetString("comms-console-menu-announcement-placeholder"));
|
||||||
|
|
||||||
AnnounceButton.OnPressed += (_) => Owner.AnnounceButtonPressed(Rope.Collapse(MessageInput.TextRope).Trim());
|
AnnounceButton.OnPressed += (_) => Owner.AnnounceButtonPressed(Rope.Collapse(MessageInput.TextRope));
|
||||||
AnnounceButton.Disabled = !owner.CanAnnounce;
|
AnnounceButton.Disabled = !owner.CanAnnounce;
|
||||||
|
|
||||||
AlertLevelButton.OnItemSelected += args =>
|
AlertLevelButton.OnItemSelected += args =>
|
||||||
|
|||||||
@@ -1,13 +1,16 @@
|
|||||||
using Content.Shared.NukeOps;
|
using Content.Shared.CCVar;
|
||||||
|
using Content.Shared.Chat;
|
||||||
|
using Content.Shared.NukeOps;
|
||||||
using JetBrains.Annotations;
|
using JetBrains.Annotations;
|
||||||
using Robust.Client.GameObjects;
|
using Robust.Shared.Configuration;
|
||||||
using Robust.Shared.Timing;
|
|
||||||
|
|
||||||
namespace Content.Client.NukeOps;
|
namespace Content.Client.NukeOps;
|
||||||
|
|
||||||
[UsedImplicitly]
|
[UsedImplicitly]
|
||||||
public sealed class WarDeclaratorBoundUserInterface : BoundUserInterface
|
public sealed class WarDeclaratorBoundUserInterface : BoundUserInterface
|
||||||
{
|
{
|
||||||
|
[Dependency] private readonly IConfigurationManager _cfg = default!;
|
||||||
|
|
||||||
[ViewVariables]
|
[ViewVariables]
|
||||||
private WarDeclaratorWindow? _window;
|
private WarDeclaratorWindow? _window;
|
||||||
|
|
||||||
@@ -44,6 +47,8 @@ public sealed class WarDeclaratorBoundUserInterface : BoundUserInterface
|
|||||||
|
|
||||||
private void OnWarDeclaratorActivated(string message)
|
private void OnWarDeclaratorActivated(string message)
|
||||||
{
|
{
|
||||||
SendMessage(new WarDeclaratorActivateMessage(message));
|
var maxLength = _cfg.GetCVar(CCVars.ChatMaxAnnouncementLength);
|
||||||
|
var msg = SharedChatSystem.SanitizeAnnouncement(message, maxLength);
|
||||||
|
SendMessage(new WarDeclaratorActivateMessage(msg));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,7 +2,6 @@
|
|||||||
using Content.Shared.NukeOps;
|
using Content.Shared.NukeOps;
|
||||||
using Robust.Client.AutoGenerated;
|
using Robust.Client.AutoGenerated;
|
||||||
using Robust.Client.Graphics;
|
using Robust.Client.Graphics;
|
||||||
using Robust.Client.UserInterface.Controls;
|
|
||||||
using Robust.Client.UserInterface.CustomControls;
|
using Robust.Client.UserInterface.CustomControls;
|
||||||
using Robust.Client.UserInterface.XAML;
|
using Robust.Client.UserInterface.XAML;
|
||||||
using Robust.Shared.Timing;
|
using Robust.Shared.Timing;
|
||||||
@@ -27,7 +26,7 @@ public sealed partial class WarDeclaratorWindow : DefaultWindow
|
|||||||
|
|
||||||
_gameTiming = IoCManager.Resolve<IGameTiming>();
|
_gameTiming = IoCManager.Resolve<IGameTiming>();
|
||||||
|
|
||||||
WarButton.OnPressed += ActivateWarDeclarator;
|
WarButton.OnPressed += (_) => OnActivated?.Invoke(Rope.Collapse(MessageEdit.TextRope));
|
||||||
|
|
||||||
var loc = IoCManager.Resolve<ILocalizationManager>();
|
var loc = IoCManager.Resolve<ILocalizationManager>();
|
||||||
MessageEdit.Placeholder = new Rope.Leaf(loc.GetString("war-declarator-message-placeholder"));
|
MessageEdit.Placeholder = new Rope.Leaf(loc.GetString("war-declarator-message-placeholder"));
|
||||||
@@ -129,10 +128,4 @@ public sealed partial class WarDeclaratorWindow : DefaultWindow
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void ActivateWarDeclarator(BaseButton.ButtonEventArgs obj)
|
|
||||||
{
|
|
||||||
var message = Rope.Collapse(MessageEdit.TextRope);
|
|
||||||
OnActivated?.Invoke(message);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ using Content.Server.Station.Systems;
|
|||||||
using Content.Shared.Access.Components;
|
using Content.Shared.Access.Components;
|
||||||
using Content.Shared.Access.Systems;
|
using Content.Shared.Access.Systems;
|
||||||
using Content.Shared.CCVar;
|
using Content.Shared.CCVar;
|
||||||
|
using Content.Shared.Chat;
|
||||||
using Content.Shared.Communications;
|
using Content.Shared.Communications;
|
||||||
using Content.Shared.Database;
|
using Content.Shared.Database;
|
||||||
using Content.Shared.Emag.Components;
|
using Content.Shared.Emag.Components;
|
||||||
@@ -35,8 +36,6 @@ namespace Content.Server.Communications
|
|||||||
[Dependency] private readonly IConfigurationManager _cfg = default!;
|
[Dependency] private readonly IConfigurationManager _cfg = default!;
|
||||||
[Dependency] private readonly IAdminLogManager _adminLogger = default!;
|
[Dependency] private readonly IAdminLogManager _adminLogger = default!;
|
||||||
|
|
||||||
private const int MaxMessageLength = 256;
|
|
||||||
private const int MaxMessageNewlines = 2;
|
|
||||||
private const float UIUpdateInterval = 5.0f;
|
private const float UIUpdateInterval = 5.0f;
|
||||||
|
|
||||||
public override void Initialize()
|
public override void Initialize()
|
||||||
@@ -231,22 +230,8 @@ namespace Content.Server.Communications
|
|||||||
private void OnAnnounceMessage(EntityUid uid, CommunicationsConsoleComponent comp,
|
private void OnAnnounceMessage(EntityUid uid, CommunicationsConsoleComponent comp,
|
||||||
CommunicationsConsoleAnnounceMessage message)
|
CommunicationsConsoleAnnounceMessage message)
|
||||||
{
|
{
|
||||||
var msgWords = message.Message.Trim();
|
var maxLength = _cfg.GetCVar(CCVars.ChatMaxAnnouncementLength);
|
||||||
var msgChars = (msgWords.Length <= MaxMessageLength ? msgWords : $"{msgWords[0..MaxMessageLength]}...").ToCharArray();
|
var msg = SharedChatSystem.SanitizeAnnouncement(message.Message, maxLength);
|
||||||
|
|
||||||
var newlines = 0;
|
|
||||||
for (var i = 0; i < msgChars.Length; i++)
|
|
||||||
{
|
|
||||||
if (msgChars[i] != '\n')
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if (newlines >= MaxMessageNewlines)
|
|
||||||
msgChars[i] = ' ';
|
|
||||||
|
|
||||||
newlines++;
|
|
||||||
}
|
|
||||||
|
|
||||||
var msg = new string(msgChars);
|
|
||||||
var author = Loc.GetString("comms-console-announcement-unknown-sender");
|
var author = Loc.GetString("comms-console-announcement-unknown-sender");
|
||||||
if (message.Session.AttachedEntity is { Valid: true } mob)
|
if (message.Session.AttachedEntity is { Valid: true } mob)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -22,10 +22,6 @@ public sealed partial class WarDeclaratorComponent : Component
|
|||||||
[DataField]
|
[DataField]
|
||||||
public bool AllowEditingMessage = true;
|
public bool AllowEditingMessage = true;
|
||||||
|
|
||||||
[ViewVariables(VVAccess.ReadWrite)]
|
|
||||||
[DataField]
|
|
||||||
public int MaxMessageLength = 512;
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// War declarement text color
|
/// War declarement text color
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
|||||||
@@ -3,9 +3,12 @@ using Content.Server.GameTicking.Rules;
|
|||||||
using Content.Server.GameTicking.Rules.Components;
|
using Content.Server.GameTicking.Rules.Components;
|
||||||
using Content.Server.Popups;
|
using Content.Server.Popups;
|
||||||
using Content.Server.UserInterface;
|
using Content.Server.UserInterface;
|
||||||
|
using Content.Shared.CCVar;
|
||||||
|
using Content.Shared.Chat;
|
||||||
using Content.Shared.Database;
|
using Content.Shared.Database;
|
||||||
using Content.Shared.NukeOps;
|
using Content.Shared.NukeOps;
|
||||||
using Robust.Server.GameObjects;
|
using Robust.Server.GameObjects;
|
||||||
|
using Robust.Shared.Configuration;
|
||||||
|
|
||||||
namespace Content.Server.NukeOps;
|
namespace Content.Server.NukeOps;
|
||||||
|
|
||||||
@@ -18,6 +21,7 @@ public sealed class WarDeclaratorSystem : EntitySystem
|
|||||||
[Dependency] private readonly IAdminLogManager _adminLogger = default!;
|
[Dependency] private readonly IAdminLogManager _adminLogger = default!;
|
||||||
[Dependency] private readonly NukeopsRuleSystem _nukeopsRuleSystem = default!;
|
[Dependency] private readonly NukeopsRuleSystem _nukeopsRuleSystem = default!;
|
||||||
[Dependency] private readonly PopupSystem _popupSystem = default!;
|
[Dependency] private readonly PopupSystem _popupSystem = default!;
|
||||||
|
[Dependency] private readonly IConfigurationManager _cfg = default!;
|
||||||
|
|
||||||
public override void Initialize()
|
public override void Initialize()
|
||||||
{
|
{
|
||||||
@@ -52,22 +56,8 @@ public sealed class WarDeclaratorSystem : EntitySystem
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
var text = (args.Message.Length <= component.MaxMessageLength ? args.Message.Trim() : $"{args.Message.Trim().Substring(0, 256)}...").ToCharArray();
|
var maxLength = _cfg.GetCVar(CCVars.ChatMaxAnnouncementLength);
|
||||||
|
var message = SharedChatSystem.SanitizeAnnouncement(args.Message, maxLength);
|
||||||
// No more than 2 newlines, other replaced to spaces
|
|
||||||
var newlines = 0;
|
|
||||||
for (var i = 0; i < text.Length; i++)
|
|
||||||
{
|
|
||||||
if (text[i] != '\n')
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if (newlines >= 2)
|
|
||||||
text[i] = ' ';
|
|
||||||
|
|
||||||
newlines++;
|
|
||||||
}
|
|
||||||
|
|
||||||
string message = new string(text);
|
|
||||||
if (component.AllowEditingMessage && message != string.Empty)
|
if (component.AllowEditingMessage && message != string.Empty)
|
||||||
{
|
{
|
||||||
component.Message = message;
|
component.Message = message;
|
||||||
|
|||||||
@@ -1598,6 +1598,9 @@ namespace Content.Shared.CCVar
|
|||||||
public static readonly CVarDef<int> ChatMaxMessageLength =
|
public static readonly CVarDef<int> ChatMaxMessageLength =
|
||||||
CVarDef.Create("chat.max_message_length", 1000, CVar.SERVER | CVar.REPLICATED);
|
CVarDef.Create("chat.max_message_length", 1000, CVar.SERVER | CVar.REPLICATED);
|
||||||
|
|
||||||
|
public static readonly CVarDef<int> ChatMaxAnnouncementLength =
|
||||||
|
CVarDef.Create("chat.max_announcement_length", 256, CVar.SERVER | CVar.REPLICATED);
|
||||||
|
|
||||||
public static readonly CVarDef<bool> ChatSanitizerEnabled =
|
public static readonly CVarDef<bool> ChatSanitizerEnabled =
|
||||||
CVarDef.Create("chat.chat_sanitizer_enabled", true, CVar.SERVERONLY);
|
CVarDef.Create("chat.chat_sanitizer_enabled", true, CVar.SERVERONLY);
|
||||||
|
|
||||||
|
|||||||
@@ -184,4 +184,34 @@ public abstract class SharedChatSystem : EntitySystem
|
|||||||
|
|
||||||
return message;
|
return message;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static string SanitizeAnnouncement(string message, int maxLength = 0, int maxNewlines = 2)
|
||||||
|
{
|
||||||
|
var trimmed = message.Trim();
|
||||||
|
if (maxLength > 0 && trimmed.Length > maxLength)
|
||||||
|
{
|
||||||
|
trimmed = $"{message[..maxLength]}...";
|
||||||
|
}
|
||||||
|
|
||||||
|
// No more than max newlines, other replaced to spaces
|
||||||
|
if (maxNewlines > 0)
|
||||||
|
{
|
||||||
|
var chars = trimmed.ToCharArray();
|
||||||
|
var newlines = 0;
|
||||||
|
for (var i = 0; i < chars.Length; i++)
|
||||||
|
{
|
||||||
|
if (chars[i] != '\n')
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (newlines >= maxNewlines)
|
||||||
|
chars[i] = ' ';
|
||||||
|
|
||||||
|
newlines++;
|
||||||
|
}
|
||||||
|
|
||||||
|
return new string(chars);
|
||||||
|
}
|
||||||
|
|
||||||
|
return trimmed;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user