125 lines
4.1 KiB
C#
125 lines
4.1 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Numerics;
|
|
|
|
namespace Content.Shared.Administration
|
|
{
|
|
/// <summary>
|
|
/// Contains various helper methods for working with admin flags.
|
|
/// </summary>
|
|
public static class AdminFlagsHelper
|
|
{
|
|
// As you can tell from the boatload of bitwise ops,
|
|
// writing this class was genuinely fun.
|
|
|
|
private static readonly Dictionary<string, AdminFlags> NameFlagsMap = new Dictionary<string, AdminFlags>();
|
|
private static readonly string[] FlagsNameMap = new string[32];
|
|
|
|
/// <summary>
|
|
/// Every admin flag in the game, at once!
|
|
/// </summary>
|
|
public static readonly AdminFlags Everything;
|
|
|
|
/// <summary>
|
|
/// A list of all individual admin flags.
|
|
/// </summary>
|
|
public static readonly IReadOnlyList<AdminFlags> AllFlags;
|
|
|
|
static AdminFlagsHelper()
|
|
{
|
|
var t = typeof(AdminFlags);
|
|
var flags = (AdminFlags[]) Enum.GetValues(t);
|
|
var allFlags = new List<AdminFlags>();
|
|
|
|
foreach (var value in flags)
|
|
{
|
|
var name = value.ToString().ToUpper();
|
|
|
|
// If, in the future, somebody adds a combined admin flag or something for convenience,
|
|
// ignore it.
|
|
if (BitOperations.PopCount((uint) value) != 1)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
allFlags.Add(value);
|
|
Everything |= value;
|
|
NameFlagsMap.Add(name, value);
|
|
FlagsNameMap[BitOperations.Log2((uint) value)] = name;
|
|
}
|
|
|
|
AllFlags = allFlags.ToArray();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Converts an enumerable of admin flag names to a bitfield.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// The flags must all be uppercase.
|
|
/// </remarks>
|
|
/// <exception cref="ArgumentException">
|
|
/// Thrown if a string that is not a valid admin flag is contained in <paramref name="names"/>.
|
|
/// </exception>
|
|
public static AdminFlags NamesToFlags(IEnumerable<string> names)
|
|
{
|
|
var flags = AdminFlags.None;
|
|
foreach (var name in names)
|
|
{
|
|
if (!NameFlagsMap.TryGetValue(name, out var value))
|
|
{
|
|
throw new ArgumentException($"Invalid admin flag name: {name}");
|
|
}
|
|
|
|
flags |= value;
|
|
}
|
|
|
|
return flags;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the flag bit for an admin flag name.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// The flag name must be all uppercase.
|
|
/// </remarks>
|
|
/// <exception cref="KeyNotFoundException">
|
|
/// Thrown if <paramref name="name"/> is not a valid admin flag name.
|
|
/// </exception>
|
|
public static AdminFlags NameToFlag(string name)
|
|
{
|
|
return NameFlagsMap[name];
|
|
}
|
|
|
|
/// <summary>
|
|
/// Converts a bitfield of admin flags to an array of all the flag names set.
|
|
/// </summary>
|
|
public static string[] FlagsToNames(AdminFlags flags)
|
|
{
|
|
var array = new string[BitOperations.PopCount((uint) flags)];
|
|
var highest = BitOperations.LeadingZeroCount((uint) flags);
|
|
|
|
var ai = 0;
|
|
for (var i = 0; i < 32 - highest; i++)
|
|
{
|
|
var flagValue = (AdminFlags) (1u << i);
|
|
if ((flags & flagValue) != 0)
|
|
{
|
|
array[ai++] = FlagsNameMap[i];
|
|
}
|
|
}
|
|
|
|
return array;
|
|
}
|
|
|
|
public static string PosNegFlagsText(AdminFlags posFlags, AdminFlags negFlags)
|
|
{
|
|
var posFlagNames = FlagsToNames(posFlags).Select(f => (flag: f, fText: $"+{f}"));
|
|
var negFlagNames = FlagsToNames(negFlags).Select(f => (flag: f, fText: $"-{f}"));
|
|
|
|
var flagsText = string.Join(' ', posFlagNames.Concat(negFlagNames).OrderBy(f => f.flag).Select(p => p.fText));
|
|
return flagsText;
|
|
}
|
|
}
|
|
}
|