diff --git a/Content.Client/Administration/UI/Tabs/AdminTab/AdminTab.xaml b/Content.Client/Administration/UI/Tabs/AdminTab/AdminTab.xaml
index 8b68487547..b558d6114e 100644
--- a/Content.Client/Administration/UI/Tabs/AdminTab/AdminTab.xaml
+++ b/Content.Client/Administration/UI/Tabs/AdminTab/AdminTab.xaml
@@ -3,6 +3,7 @@
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:cc="clr-namespace:Content.Client.Administration.UI.CustomControls"
xmlns:at="clr-namespace:Content.Client.Administration.UI.Tabs.AdminTab"
+ xmlns:cdAdmin="clr-namespace:Content.Client._CD.Admin.UI"
Margin="4"
MinSize="50 50">
@@ -16,6 +17,8 @@
+
+
diff --git a/Content.Client/Humanoid/HumanoidAppearanceSystem.cs b/Content.Client/Humanoid/HumanoidAppearanceSystem.cs
index 0416f4b813..d735e264ce 100644
--- a/Content.Client/Humanoid/HumanoidAppearanceSystem.cs
+++ b/Content.Client/Humanoid/HumanoidAppearanceSystem.cs
@@ -5,6 +5,8 @@ using Content.Shared.Preferences;
using Robust.Client.GameObjects;
using Robust.Shared.Prototypes;
using Robust.Shared.Utility;
+using System.Numerics; // CD - Character Records
+using Robust.Client.Console; // CD - Character Records
namespace Content.Client.Humanoid;
@@ -30,6 +32,13 @@ public sealed class HumanoidAppearanceSystem : SharedHumanoidAppearanceSystem
UpdateLayers(component, sprite);
ApplyMarkingSet(component, sprite);
+ // Begin CD - Character Records
+ var speciesPrototype = _prototypeManager.Index(component.Species);
+ var height = Math.Clamp(MathF.Round(component.Height, 2), speciesPrototype.MinHeight, speciesPrototype.MaxHeight); // should NOT be locked, at all
+
+ sprite.Scale = speciesPrototype.BaseScale * new Vector2(speciesPrototype.ScaleHeight ? height : 1f, height); // DV - CD Character Records shouldn't nuke species heights
+ // End CD - Character Records
+
sprite[sprite.LayerMapReserveBlank(HumanoidVisualLayers.Eyes)].Color = component.EyeColor;
}
@@ -199,6 +208,7 @@ public sealed class HumanoidAppearanceSystem : SharedHumanoidAppearanceSystem
humanoid.Species = profile.Species;
humanoid.SkinColor = profile.Appearance.SkinColor;
humanoid.EyeColor = profile.Appearance.EyeColor;
+ humanoid.Height = profile.Height; // CD - Character Records
UpdateSprite(humanoid, Comp(uid));
}
diff --git a/Content.Client/Lobby/UI/HumanoidProfileEditor.xaml b/Content.Client/Lobby/UI/HumanoidProfileEditor.xaml
index 703b64bce3..54c6356d68 100644
--- a/Content.Client/Lobby/UI/HumanoidProfileEditor.xaml
+++ b/Content.Client/Lobby/UI/HumanoidProfileEditor.xaml
@@ -67,6 +67,15 @@
+
+
+
+
+
+
+
+
+
diff --git a/Content.Client/Lobby/UI/HumanoidProfileEditor.xaml.cs b/Content.Client/Lobby/UI/HumanoidProfileEditor.xaml.cs
index e37733ec34..ac1fcdd8b0 100644
--- a/Content.Client/Lobby/UI/HumanoidProfileEditor.xaml.cs
+++ b/Content.Client/Lobby/UI/HumanoidProfileEditor.xaml.cs
@@ -33,6 +33,11 @@ using Robust.Shared.Enums;
using Robust.Shared.Prototypes;
using Robust.Shared.Utility;
using Direction = Robust.Shared.Maths.Direction;
+// Begin CD - Character Records
+using System.Globalization;
+using Content.Client._CD.Records.UI;
+using Content.Shared._CD.Records;
+// End CD - Character Records
namespace Content.Client.Lobby.UI
{
@@ -96,6 +101,12 @@ namespace Content.Client.Lobby.UI
private bool _isDirty;
+ // Begin CD - Station Records
+ private float _defaultHeight = 1f;
+
+ private readonly RecordEditorGui _recordsTab;
+ // End CD - Station Records
+
[ValidatePrototypeId]
private const string DefaultSpeciesGuidebook = "Species";
@@ -220,6 +231,43 @@ namespace Content.Client.Lobby.UI
OnSkinColorOnValueChanged();
};
+ // Begin CD - Character Records
+ #region CDHeight
+
+ CDHeight.OnTextChanged += args =>
+ {
+ if (Profile is null || !float.TryParse(args.Text, out var newHeight))
+ return;
+
+ var prototype = _prototypeManager.Index(Profile.Species);
+ newHeight = MathF.Round(Math.Clamp(newHeight, prototype.MinHeight, prototype.MaxHeight), 2);
+
+ // The percentage between the start and end numbers, aka "inverse lerp"
+ var sliderPercent = (newHeight - prototype.MinHeight) /
+ (prototype.MaxHeight - prototype.MinHeight);
+ CDHeightSlider.Value = sliderPercent;
+
+ SetProfileHeight(newHeight);
+ };
+
+ CDHeightReset.OnPressed += _ =>
+ {
+ CDHeight.SetText(_defaultHeight.ToString(CultureInfo.InvariantCulture), true);
+ };
+
+ CDHeightSlider.OnValueChanged += _ =>
+ {
+ if (Profile is null)
+ return;
+ var prototype = _prototypeManager.Index(Profile.Species);
+ var newHeight = MathF.Round(MathHelper.Lerp(prototype.MinHeight, prototype.MaxHeight, CDHeightSlider.Value), 2);
+ CDHeight.Text = newHeight.ToString(CultureInfo.InvariantCulture);
+ SetProfileHeight(newHeight);
+ };
+
+ #endregion CDHeight
+ // End CD - Character Records
+
#region Skin
Skin.OnValueChanged += _ =>
@@ -413,6 +461,16 @@ namespace Content.Client.Lobby.UI
#endregion Markings
+ // Begin CD - Character Records
+ #region CosmaticRecords
+
+ _recordsTab = new RecordEditorGui(UpdateProfileRecords);
+ TabContainer.AddChild(_recordsTab);
+ TabContainer.SetTabTitle(TabContainer.ChildCount - 1, Loc.GetString("humanoid-profile-editor-cd-records-tab"));
+
+ #endregion CosmaticRecords
+ // End CD - Character Records
+
RefreshFlavorText();
#region Dummy
@@ -754,6 +812,11 @@ namespace Content.Client.Lobby.UI
UpdateCMarkingsHair();
UpdateCMarkingsFacialHair();
+ // Begin CD - Character Records
+ UpdateHeightControls();
+ _recordsTab.Update(profile);
+ // End CD - Character Records
+
RefreshAntags();
RefreshJobs();
RefreshLoadouts();
@@ -1050,6 +1113,16 @@ namespace Content.Client.Lobby.UI
UpdateJobPriorities();
}
+ // Start CD - Character Records
+ private void UpdateProfileRecords(PlayerProvidedCharacterRecords records)
+ {
+ if (Profile is null)
+ return;
+ Profile = Profile.WithCDCharacterRecords(records);
+ IsDirty = true;
+ }
+ // End CD - Character Records
+
private void OnFlavorTextChange(string content)
{
if (Profile is null)
@@ -1231,6 +1304,15 @@ namespace Content.Client.Lobby.UI
_entManager.System().SetEntityName(PreviewDummy, newName);
}
+ // Begin CD - Character Records
+ private void SetProfileHeight(float height)
+ {
+ Profile = Profile?.WithHeight(height);
+ SetDirty();
+ ReloadProfilePreview();
+ }
+ // End CD - Character Records
+
private void SetSpawnPriority(SpawnPriorityPreference newSpawnPriority)
{
Profile = Profile?.WithSpawnPriorityPreference(newSpawnPriority);
@@ -1416,6 +1498,26 @@ namespace Content.Client.Lobby.UI
PronounsButton.SelectId((int) Profile.Gender);
}
+ // Begin CD - Character Records
+ private void UpdateHeightControls()
+ {
+ if (Profile == null)
+ {
+ return;
+ }
+
+ var species = _species.Find(x => x.ID == Profile.Species);
+ if (species != null)
+ _defaultHeight = species.DefaultHeight;
+
+ var prototype = _prototypeManager.Index(Profile.Species);
+ var sliderPercent = (Profile.Height - prototype.MinHeight) /
+ (prototype.MaxHeight - prototype.MinHeight);
+ CDHeightSlider.Value = sliderPercent;
+ CDHeight.Text = Profile.Height.ToString(CultureInfo.InvariantCulture);
+ }
+ // End CD - Character Records
+
private void UpdateSpawnPriorityControls()
{
if (Profile == null)
@@ -1558,6 +1660,8 @@ namespace Content.Client.Lobby.UI
var name = HumanoidCharacterProfile.GetName(Profile.Species, Profile.Gender);
SetName(name);
UpdateNameEdit();
+
+ _recordsTab.Update(Profile); // CD - Character Records
}
private async void ExportImage()
diff --git a/Content.Client/_CD/Admin/UI/ModifyCharacterRecords.xaml b/Content.Client/_CD/Admin/UI/ModifyCharacterRecords.xaml
new file mode 100644
index 0000000000..9cb3184970
--- /dev/null
+++ b/Content.Client/_CD/Admin/UI/ModifyCharacterRecords.xaml
@@ -0,0 +1,14 @@
+
+
+
+
+
+
+
+
+
+
+
diff --git a/Content.Client/_CD/Admin/UI/ModifyCharacterRecords.xaml.cs b/Content.Client/_CD/Admin/UI/ModifyCharacterRecords.xaml.cs
new file mode 100644
index 0000000000..9a86865099
--- /dev/null
+++ b/Content.Client/_CD/Admin/UI/ModifyCharacterRecords.xaml.cs
@@ -0,0 +1,48 @@
+using Content.Shared._CD.Records;
+using Robust.Client.AutoGenerated;
+using Robust.Client.UserInterface.CustomControls;
+using Robust.Client.UserInterface.XAML;
+
+namespace Content.Client._CD.Admin.UI;
+
+[GenerateTypedNameReferences]
+public sealed partial class ModifyCharacterRecords : DefaultWindow
+{
+ public ModifyCharacterRecords()
+ {
+ RobustXamlLoader.Load(this);
+ IoCManager.InjectDependencies(this);
+
+ foreach (var v in Enum.GetValues())
+ {
+ EntityEntryType.AddItem(v.ToString());
+ }
+
+ EntityEntryType.OnItemSelected += args =>
+ {
+ EntityEntryType.SelectId(args.Id);
+ UpdateCommands();
+ };
+
+ EntityEdit.OnTextChanged += _ => UpdateCommands();
+ EntityEntryIndex.OnTextChanged += _ => UpdateCommands();
+ }
+
+ private void UpdateCommands()
+ {
+ if (!int.TryParse(EntityEdit.Text, out var uid))
+ {
+ return;
+ }
+
+ if (!int.TryParse(EntityEntryIndex.Text, out var idx))
+ {
+ return;
+ }
+
+ var ty = (CharacterRecordType)EntityEntryType.SelectedId;
+
+ PurgeCommand.Command = $"purgecharacterrecords {uid}";
+ DelCommand.Command = $"delrecordentry {uid} {ty.ToString()} {idx}";
+ }
+}
diff --git a/Content.Client/_CD/Records/UI/CharacterRecordConsoleBoundUserInterface.cs b/Content.Client/_CD/Records/UI/CharacterRecordConsoleBoundUserInterface.cs
new file mode 100644
index 0000000000..4a1e9a392b
--- /dev/null
+++ b/Content.Client/_CD/Records/UI/CharacterRecordConsoleBoundUserInterface.cs
@@ -0,0 +1,75 @@
+using Content.Shared.CriminalRecords.Components;
+using Content.Shared.CriminalRecords;
+using Content.Shared.StationRecords;
+using Content.Shared._CD.Records;
+using JetBrains.Annotations;
+
+namespace Content.Client._CD.Records.UI;
+
+[UsedImplicitly]
+public sealed class CharacterRecordConsoleBoundUserInterface(EntityUid owner, Enum key) : BoundUserInterface(owner, key)
+{
+ [ViewVariables] private CharacterRecordViewer? _window;
+
+ protected override void UpdateState(BoundUserInterfaceState baseState)
+ {
+ base.UpdateState(baseState);
+ if (baseState is not CharacterRecordConsoleState state)
+ return;
+
+ if (_window?.IsSecurity() ?? false)
+ {
+ var comp = EntMan.GetComponent(Owner);
+ _window!.SecurityWantedStatusMaxLength = comp.MaxStringLength;
+ }
+
+ _window?.UpdateState(state);
+ }
+
+ protected override void Open()
+ {
+ base.Open();
+
+ _window = new();
+ _window.OnClose += Close;
+ _window.OnListingItemSelected += meta =>
+ {
+ SendMessage(new CharacterRecordConsoleSelectMsg(meta?.CharacterRecordKey));
+
+ // If we are a security records console, we also need to inform the criminal records
+ // system of our state.
+ if (_window.IsSecurity() && meta?.StationRecordKey != null)
+ {
+ SendMessage(new SelectStationRecord(meta.Value.StationRecordKey.Value));
+ _window.SetSecurityStatusEnabled(true);
+ }
+ else
+ {
+ // If the user does not have criminal records for some reason, we should not be able
+ // to set their wanted status
+ _window.SetSecurityStatusEnabled(false);
+ }
+ };
+
+ _window.OnFiltersChanged += (ty, txt) =>
+ {
+ SendMessage(txt == null
+ ? new CharacterRecordsConsoleFilterMsg(null)
+ : new CharacterRecordsConsoleFilterMsg(new StationRecordsFilter(ty, txt)));
+ };
+
+ _window.OnSetSecurityStatus += (status, reason) =>
+ {
+ SendMessage(new CriminalRecordChangeStatus(status, reason));
+ };
+
+ _window.OpenCentered();
+ }
+
+ protected override void Dispose(bool disposing)
+ {
+ base.Dispose(disposing);
+
+ _window?.Close();
+ }
+}
diff --git a/Content.Client/_CD/Records/UI/CharacterRecordViewer.xaml b/Content.Client/_CD/Records/UI/CharacterRecordViewer.xaml
new file mode 100644
index 0000000000..a952accd81
--- /dev/null
+++ b/Content.Client/_CD/Records/UI/CharacterRecordViewer.xaml
@@ -0,0 +1,117 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/Content.Client/_CD/Records/UI/CharacterRecordViewer.xaml.cs b/Content.Client/_CD/Records/UI/CharacterRecordViewer.xaml.cs
new file mode 100644
index 0000000000..76cc56b2ed
--- /dev/null
+++ b/Content.Client/_CD/Records/UI/CharacterRecordViewer.xaml.cs
@@ -0,0 +1,450 @@
+using Content.Client.UserInterface.Controls;
+using Content.Shared.Administration;
+using Content.Shared.Security;
+using Content.Shared.StationRecords;
+using Content.Shared._CD.Records;
+using Robust.Client.AutoGenerated;
+using Robust.Client.UserInterface.XAML;
+using System.Linq;
+
+namespace Content.Client._CD.Records.UI;
+
+[GenerateTypedNameReferences]
+public sealed partial class CharacterRecordViewer : FancyWindow
+{
+ public struct CharacterListMetadata
+ {
+ public uint CharacterRecordKey;
+ public uint? StationRecordKey;
+ }
+
+ public static readonly Color BackgroundColor = Color.FromHex("#25252a"); // Dark grey
+ public static readonly Color ContentPanelColor = Color.FromHex("#1a1a1a"); // Darker grey for content areas
+ public static readonly Color BorderColor = Color.FromHex("#404040"); // Light grey border
+ public static readonly Color ErrorColor = Color.FromHex("#ff0000"); // Red for validation errors
+
+ public event Action? OnListingItemSelected;
+ public event Action? OnFiltersChanged;
+
+ private bool _isPopulating;
+ private StationRecordFilterType _filterType;
+
+ private RecordConsoleType? _type;
+
+ private readonly RecordEntryViewPopup _entryView = new();
+ private List? _entries;
+
+ private DialogWindow? _wantedReasonDialog;
+
+ ///
+ /// The key to the record of the currently selected item in the listing.
+ ///
+ private uint? _selectedListingKey;
+
+ ///
+ /// The key to the record that is currently visible.
+ ///
+ ///
+ /// This may differ from because this contents has not been updated yet to reflect the new selection.
+ ///
+ private uint? _openRecordKey;
+ public event Action? OnSetSecurityStatus;
+
+ public uint? SecurityWantedStatusMaxLength;
+
+ public CharacterRecordViewer()
+ {
+ RobustXamlLoader.Load(this);
+
+ // There is no reason why we can't just steal the StationRecordFilter class.
+ // If wizden adds a new kind of filtering we want to replicate it here.
+ foreach (var item in Enum.GetValues())
+ {
+ RecordFilterType.AddItem(GetTypeFilterLocals(item), (int)item);
+ }
+
+ // Again, if wizden changes something about Criminal Records, we want to replicate the
+ // functionality here.
+ foreach (var status in Enum.GetValues())
+ {
+ var name = Loc.GetString($"criminal-records-status-{status.ToString().ToLower()}");
+ StatusOptionButton.AddItem(name, (int)status);
+ }
+
+ CharacterListing.OnItemSelected += _ =>
+ {
+ if (!CharacterListing.GetSelected().Any())
+ return;
+ var selected = CharacterListing.GetSelected().First();
+ var meta = (CharacterListMetadata)selected.Metadata!;
+ _selectedListingKey = meta.CharacterRecordKey;
+ if (!_isPopulating)
+ OnListingItemSelected?.Invoke(meta);
+ };
+
+ CharacterListing.OnItemDeselected += _ =>
+ {
+ // When we populate the records, we clear the contents of the listing.
+ // This could cause a deselection but we don't want to really deselect because it would
+ // interrupt what the player is doing.
+ if (!_isPopulating)
+ OnListingItemSelected?.Invoke(null);
+ _selectedListingKey = null;
+ };
+
+ RecordFilters.OnPressed += _ =>
+ {
+ OnFiltersChanged?.Invoke(_filterType, RecordFiltersValue.Text);
+ };
+
+ RecordFiltersReset.OnPressed += _ =>
+ {
+ OnFiltersChanged?.Invoke(StationRecordFilterType.Name, null);
+ RecordFiltersValue.Clear();
+ };
+
+ RecordFiltersValue.OnTextEntered += text =>
+ {
+ OnFiltersChanged?.Invoke(_filterType, text.Text);
+ };
+
+ RecordFilterType.OnItemSelected += eventArgs =>
+ {
+ var type = (StationRecordFilterType)eventArgs.Id;
+ _filterType = type;
+ RecordFilterType.SelectId(eventArgs.Id);
+ };
+
+ RecordEntryViewButton.OnPressed += _ =>
+ {
+ if (_entries == null || !RecordEntryList.GetSelected().Any())
+ return;
+ var idx = RecordEntryList.IndexOf(RecordEntryList.GetSelected().First());
+ _entryView.SetContents(_entries[idx]);
+ _entryView.Open();
+ };
+
+ StatusOptionButton.OnItemSelected += args =>
+ {
+ var status = (SecurityStatus)args.Id;
+ // This should reflect SetStatus in CriminalRecordsConsoleWindow.xaml.cs
+ if (status == SecurityStatus.Wanted || status == SecurityStatus.Suspected)
+ SetStatusWithReason(status);
+ else
+ OnSetSecurityStatus?.Invoke(status, null);
+ };
+
+ OnClose += () => _entryView.Close();
+
+ // Admin console entry type selector
+ RecordEntryViewType.AddItem(Loc.GetString("department-Security"));
+ RecordEntryViewType.AddItem(Loc.GetString("department-Medical"));
+ RecordEntryViewType.AddItem(Loc.GetString("humanoid-profile-editor-cd-records-employment"));
+ RecordEntryViewType.OnItemSelected += args =>
+ {
+ if (args.Id == RecordEntryViewType.SelectedId)
+ return;
+ RecordEntryViewType.SelectId(args.Id);
+ // This is a hack to get the server to send us another packet with the new entries
+ OnFiltersChanged?.Invoke(_filterType, RecordFiltersValue.Text);
+ };
+ }
+
+ // If we are using wizden's class we might as well use their localization.
+ private string GetTypeFilterLocals(StationRecordFilterType type)
+ {
+ return Loc.GetString($"general-station-record-{type.ToString().ToLower()}-filter");
+ }
+
+ ///
+ /// Select the record in the listing for the given key.
+ ///
+ /// The index of the record in the dictionary
+ private void SelectRecordKey(uint? key)
+ {
+ if (_selectedListingKey == key)
+ return;
+ _selectedListingKey = key;
+
+ _isPopulating = true;
+
+ CharacterListing.ClearSelected();
+
+ // I wish there was a better way of doing this
+ if (key != null)
+ {
+ foreach (var item in CharacterListing)
+ {
+ if (((CharacterListMetadata) item.Metadata!).CharacterRecordKey == key)
+ {
+ item.Selected = true;
+ break;
+ }
+ }
+ }
+
+ _isPopulating = false;
+ }
+
+ private bool CharacterListNeedsRepopulating(IReadOnlyDictionary newKeys)
+ {
+ var newCount = newKeys.Count;
+ if (newCount != CharacterListing.Count)
+ return true;
+
+ // Given that there is the same number of keys in the dictionary as in items in the listing, they are not equal
+ // if and only if there exists a key in the listing that is not in the dictionary
+ foreach (var item in CharacterListing)
+ {
+ var key = ((CharacterListMetadata)item.Metadata!).CharacterRecordKey;
+ if (!newKeys.ContainsKey(key))
+ {
+ return true;
+ }
+ }
+
+ return false;
+ }
+
+ public void UpdateState(CharacterRecordConsoleState state)
+ {
+ #region Visibility
+
+ RecordEntryViewType.Visible = false;
+ _type = state.ConsoleType;
+
+ // Disable listing if we don't have one selected
+ if (state.CharacterList == null)
+ {
+ CharacterListingStatus.Visible = true;
+ CharacterListing.Visible = false;
+ CharacterListingStatus.Text = Loc.GetString("cd-record-viewer-empty-state");
+ RecordContainer.Visible = false;
+ RecordContainerStatus.Visible = false;
+ return;
+ }
+
+ CharacterListingStatus.Visible = false;
+ CharacterListing.Visible = true;
+
+ // Enable extended filtering only for admin and security consoles
+ switch (_type)
+ {
+ case RecordConsoleType.Employment:
+ RecordFilterType.Visible = false;
+ RecordFilterType.SelectId((int)StationRecordFilterType.Name);
+
+ Title = Loc.GetString("cd-character-records-viewer-title-employ");
+ break;
+ case RecordConsoleType.Medical:
+ RecordFilterType.Visible = false;
+ RecordFilterType.SelectId((int)StationRecordFilterType.Name);
+
+ Title = Loc.GetString("cd-character-records-viewer-title-med");
+ break;
+ case RecordConsoleType.Security:
+ RecordFilterType.Visible = true;
+
+ Title = Loc.GetString("cd-character-records-viewer-title-sec");
+ break;
+ case RecordConsoleType.Admin:
+ RecordFilterType.Visible = true;
+ Title = "Admin records console";
+ RecordEntryViewType.Visible = true;
+
+ break;
+ }
+
+ #endregion
+
+ #region PopulateListing
+
+ if (state.Filter != null)
+ {
+ RecordFiltersValue.SetText(state.Filter.Value);
+ RecordFilterType.SelectId((int) state.Filter.Type);
+ }
+
+ if (CharacterListNeedsRepopulating(state.CharacterList))
+ {
+ _isPopulating = true;
+
+ CharacterListing.Clear();
+
+ // Add the records to the listing in a sorted order. There is probably are faster way of doing this, but
+ // this is not really a hot code path.
+ state.CharacterList
+ // The items in this tuple are as follows: (name of character, CharacterListMetadata)
+ .Select(r
+ => (CharacterName: r.Value.CharacterDisplayName, new CharacterListMetadata() { CharacterRecordKey = r.Key, StationRecordKey = r.Value.StationRecordKey}))
+ .OrderBy(r => r.Item1)
+ .ToList()
+ .ForEach(r => CharacterListing.AddItem(r.Item1, metadata: r.Item2));
+
+ _isPopulating = false;
+ }
+
+ SelectRecordKey(state.SelectedIndex);
+
+ #endregion
+
+ #region FillRecordContainer
+
+ // Enable container if we have a record selected
+ if (state.SelectedRecord == null)
+ {
+ RecordContainerStatus.Visible = true;
+ RecordContainer.Visible = false;
+ return;
+ }
+
+ RecordContainerStatus.Visible = false;
+ RecordContainer.Visible = true;
+
+ // Do not needlessly reload the record if not needed. This is mainly done to prevent a bug in the admin record viewer.
+ if (state.SelectedIndex == _openRecordKey)
+ return;
+ _openRecordKey = state.SelectedIndex;
+
+ var record = state.SelectedRecord!;
+ var cr = record.PRecords;
+
+ // Basic info
+ RecordContainerName.Text = record.Name;
+ RecordContainerAge.Text = record.Age.ToString();
+ RecordContainerJob.Text = record.JobTitle; /* At some point in the future we might want to display the icon */
+ RecordContainerGender.Text = record.Gender.ToString();
+ RecordContainerSpecies.Text = record.Species;
+ RecordContainerHeight.Text = cr.Height + " " + UnitConversion.GetImperialDisplayLength(cr.Height);
+ RecordContainerWeight.Text = cr.Weight + " " + UnitConversion.GetImperialDisplayMass(cr.Weight);
+ RecordContainerContactName.SetValue(cr.EmergencyContactName);
+
+ RecordContainerEmployment.Visible = false;
+ RecordContainerMedical.Visible = false;
+ RecordContainerSecurity.Visible = false;
+
+ switch (_type)
+ {
+ case RecordConsoleType.Employment:
+ SetEntries(cr.EmploymentEntries);
+ UpdateRecordBoxEmployment(record);
+ break;
+ case RecordConsoleType.Medical:
+ SetEntries(cr.MedicalEntries);
+ UpdateRecordBoxMedical(record);
+ break;
+ case RecordConsoleType.Security:
+ SetEntries(cr.SecurityEntries);
+ UpdateRecordBoxSecurity(record, state.SelectedSecurityStatus);
+ break;
+ case RecordConsoleType.Admin:
+ UpdateRecordBoxEmployment(record);
+ UpdateRecordBoxMedical(record);
+ UpdateRecordBoxSecurity(record, state.SelectedSecurityStatus);
+ switch ((RecordConsoleType) RecordEntryViewType.SelectedId)
+ {
+ case RecordConsoleType.Employment:
+ SetEntries(cr.EmploymentEntries, true);
+ break;
+ case RecordConsoleType.Medical:
+ SetEntries(cr.MedicalEntries, true);
+ break;
+ case RecordConsoleType.Security:
+ SetEntries(cr.SecurityEntries, true);
+ break;
+ }
+ break;
+ }
+
+ #endregion
+
+ }
+
+ private void SetEntries(List entries, bool addIndex = false)
+ {
+ _entries = entries;
+ RecordEntryList.Clear();
+ var i = 0;
+ foreach (var entry in entries)
+ {
+ RecordEntryList.AddItem(addIndex ? $"({i.ToString()}) " + entry.Title : entry.Title);
+ ++i;
+ }
+ }
+
+ private void UpdateRecordBoxEmployment(FullCharacterRecords record)
+ {
+ RecordContainerEmployment.Visible = true;
+ RecordContainerWorkAuth.Text = record.PRecords.HasWorkAuthorization ? "yes" : "no";
+ }
+
+ private void UpdateRecordBoxMedical(FullCharacterRecords record)
+ {
+ RecordContainerMedical.Visible = true;
+ var cr = record.PRecords;
+ RecordContainerMedical.Visible = true;
+ RecordContainerAllergies.SetValue(cr.Allergies);
+ RecordContainerDrugAllergies.SetValue(cr.DrugAllergies);
+ RecordContainerPostmortem.SetValue(cr.PostmortemInstructions);
+ RecordContainerSex.Text = record.Sex.ToString();
+ }
+
+ private void UpdateRecordBoxSecurity(FullCharacterRecords record, (SecurityStatus, string?)? criminal)
+ {
+ RecordContainerSecurity.Visible = true;
+ RecordContainerIdentFeatures.SetValue(record.PRecords.IdentifyingFeatures);
+ RecordContainerFingerprint.Text = record.Fingerprint ?? Loc.GetString("cd-character-records-viewer-unknown");
+ RecordContainerDNA.Text = record.DNA ?? Loc.GetString("cd-character-records-viewer-unknown");
+
+ RecordContainerWantedReason.Visible = false;
+ if (criminal != null)
+ {
+ var (stat, reason) = criminal.Value;
+ StatusOptionButton.Select((int)stat);
+ RecordContainerWantedReason.Text = reason;
+ RecordContainerWantedReason.Visible = reason != null;
+ }
+ }
+
+ // This is copied almost verbatim from CriminalRecordsConsoleWindow.xaml.cs
+ private void SetStatusWithReason(SecurityStatus status)
+ {
+ if (_wantedReasonDialog != null)
+ {
+ _wantedReasonDialog.MoveToFront();
+ return;
+ }
+
+ const string field = "reason";
+ var title = Loc.GetString("criminal-records-status-" + status.ToString().ToLower());
+ var placeholder = Loc.GetString("cd-character-records-viewer-setwanted-placeholder");
+ var prompt = Loc.GetString("criminal-records-console-reason");
+ var entry = new QuickDialogEntry(field, QuickDialogEntryType.LongText, prompt, placeholder);
+ var entries = new List() { entry };
+ _wantedReasonDialog = new DialogWindow(title, entries);
+
+ _wantedReasonDialog.OnConfirmed += responses =>
+ {
+ var reason = responses[field];
+ if (reason.Length < 1 || reason.Length > SecurityWantedStatusMaxLength)
+ return;
+
+ OnSetSecurityStatus?.Invoke(status, reason);
+ };
+
+ _wantedReasonDialog.OnClose += () => { _wantedReasonDialog = null; };
+ }
+ public bool IsSecurity()
+ {
+ return _type == RecordConsoleType.Security || _type == RecordConsoleType.Admin;
+ }
+
+ public void SetSecurityStatusEnabled(bool setting)
+ {
+ for (var i = 0; i < StatusOptionButton.ItemCount; ++i)
+ {
+ StatusOptionButton.SetItemDisabled(i, !setting);
+ }
+ }
+}
+
diff --git a/Content.Client/_CD/Records/UI/RecordEditorEntrySelector.xaml b/Content.Client/_CD/Records/UI/RecordEditorEntrySelector.xaml
new file mode 100644
index 0000000000..328e36d0b8
--- /dev/null
+++ b/Content.Client/_CD/Records/UI/RecordEditorEntrySelector.xaml
@@ -0,0 +1,14 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/Content.Client/_CD/Records/UI/RecordEditorEntrySelector.xaml.cs b/Content.Client/_CD/Records/UI/RecordEditorEntrySelector.xaml.cs
new file mode 100644
index 0000000000..5bc045c5bc
--- /dev/null
+++ b/Content.Client/_CD/Records/UI/RecordEditorEntrySelector.xaml.cs
@@ -0,0 +1,135 @@
+using Content.Shared._CD.Records;
+using Robust.Client.AutoGenerated;
+using Robust.Client.UserInterface.XAML;
+using Robust.Client.UserInterface;
+using System.Linq;
+
+namespace Content.Client._CD.Records.UI;
+
+///
+/// The box that contains the list of entities in the record editor. We create one for each record type
+///
+[GenerateTypedNameReferences]
+public sealed partial class RecordEditorEntrySelector : Control
+{
+ private List _entries = new();
+
+ public event Action? OnUpdateEntries;
+
+ private readonly RecordEntryEditPopup _editPopup = new();
+ private readonly RecordEntryViewPopup _entryViewPopup = new();
+ private int _editIdx;
+
+ public RecordEditorEntrySelector()
+ {
+ RobustXamlLoader.Load(this);
+
+ AddButton.OnPressed += _ =>
+ {
+ _editIdx = _entries.Count;
+ _editPopup.SetContents(new PlayerProvidedCharacterRecords.RecordEntry("", "", ""));
+ _editPopup.Open();
+ };
+
+ EditButton.OnPressed += _ =>
+ {
+ if (!EntrySelector.GetSelected().Any())
+ return;
+ _editIdx = EntrySelector.IndexOf(EntrySelector.GetSelected().First());
+ _editPopup.SetContents(_entries[_editIdx]);
+ _editPopup.Open();
+ };
+
+ ViewButton.OnPressed += _ =>
+ {
+ if (!EntrySelector.GetSelected().Any())
+ return;
+ var idx = EntrySelector.IndexOf(EntrySelector.GetSelected().First());
+ _entryViewPopup.SetContents(_entries[idx]);
+ _entryViewPopup.Open();
+ };
+
+ RemoveButton.OnPressed += _ =>
+ {
+ if (!EntrySelector.GetSelected().Any())
+ return;
+ // Remove the entry, being careful to set the index correctly
+ var idx = EntrySelector.IndexOf(EntrySelector.GetSelected().First());
+ EntrySelector.RemoveAt(idx);
+ _entries.RemoveAt(idx);
+ if (idx == _editIdx)
+ _editPopup.Close();
+ _editIdx--;
+ OnUpdateEntries?.Invoke(new RecordEditorEntryUpdateArgs(_entries));
+ };
+
+ UpButton.OnPressed += _ =>
+ {
+ if (!EntrySelector.GetSelected().Any())
+ return;
+ var idx = EntrySelector.IndexOf(EntrySelector.GetSelected().First());
+ if (idx < 1)
+ return;
+ (_entries[idx], _entries[idx - 1]) = (_entries[idx - 1], _entries[idx]);
+ (EntrySelector[idx], EntrySelector[idx - 1]) = (EntrySelector[idx - 1], EntrySelector[idx]);
+ OnUpdateEntries?.Invoke(new RecordEditorEntryUpdateArgs(_entries));
+ };
+
+ DownButton.OnPressed += _ =>
+ {
+ if (!EntrySelector.GetSelected().Any())
+ return;
+ var idx = EntrySelector.IndexOf(EntrySelector.GetSelected().First());
+ if (idx >= EntrySelector.Count - 1)
+ return;
+ (_entries[idx], _entries[idx + 1]) = (_entries[idx + 1], _entries[idx]);
+ (EntrySelector[idx], EntrySelector[idx + 1]) = (EntrySelector[idx + 1], EntrySelector[idx]);
+ OnUpdateEntries?.Invoke(new RecordEditorEntryUpdateArgs(_entries));
+ };
+
+ _editPopup.SaveButton.OnPressed += _ =>
+ {
+ if (_editIdx >= _entries.Count)
+ {
+ var rec = _editPopup.GetContents();
+ _entries.Add(rec);
+ EntrySelector.AddItem(rec.Title);
+ }
+ else
+ {
+ _entries[_editIdx] = _editPopup.GetContents();
+ EntrySelector[_editIdx].Text = _entries[_editIdx].Title;
+ }
+ OnUpdateEntries?.Invoke(new RecordEditorEntryUpdateArgs(_entries));
+ };
+ OnVisibilityChanged += _ =>
+ {
+ _editPopup.Close();
+ };
+ }
+
+ public void UpdateContents(List entries)
+ {
+ _entries = entries;
+ RefreshSelector();
+ }
+
+ private void RefreshSelector()
+ {
+ EntrySelector.Clear();
+ foreach (var entry in _entries)
+ {
+ EntrySelector.AddItem(entry.Title);
+ }
+ }
+
+ public sealed class RecordEditorEntryUpdateArgs
+ {
+ public List Entries { get; private set; }
+
+ public RecordEditorEntryUpdateArgs(List entries)
+ {
+ Entries = entries;
+ }
+ }
+}
diff --git a/Content.Client/_CD/Records/UI/RecordEditorGui.xaml b/Content.Client/_CD/Records/UI/RecordEditorGui.xaml
new file mode 100644
index 0000000000..ef1b6e2135
--- /dev/null
+++ b/Content.Client/_CD/Records/UI/RecordEditorGui.xaml
@@ -0,0 +1,68 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/Content.Client/_CD/Records/UI/RecordEditorGui.xaml.cs b/Content.Client/_CD/Records/UI/RecordEditorGui.xaml.cs
new file mode 100644
index 0000000000..f360a20623
--- /dev/null
+++ b/Content.Client/_CD/Records/UI/RecordEditorGui.xaml.cs
@@ -0,0 +1,155 @@
+using Content.Shared.Preferences;
+using Content.Shared._CD.Records;
+using Robust.Client.AutoGenerated;
+using Robust.Client.UserInterface.XAML;
+using Robust.Client.UserInterface;
+
+namespace Content.Client._CD.Records.UI;
+
+///
+/// The record editor tab that gets "injected" into the character editor.
+///
+[GenerateTypedNameReferences]
+public sealed partial class RecordEditorGui : Control
+{
+ ///
+ /// Delegate that tells the editor to save records when the save button is pressed
+ ///
+ private readonly Action _updateProfileRecords;
+ private PlayerProvidedCharacterRecords _records = default!;
+
+ public RecordEditorGui(Action updateProfileRecords)
+ {
+ RobustXamlLoader.Load(this);
+ _updateProfileRecords = updateProfileRecords;
+
+ #region General
+
+ HeightEdit.OnTextChanged += args =>
+ {
+ if (!int.TryParse(args.Text, out var newHeight))
+ return;
+ UpdateImperialHeight(newHeight);
+ UpdateRecords(_records.WithHeight(newHeight));
+ };
+
+ WeightEdit.OnTextChanged += args =>
+ {
+ if (!int.TryParse(args.Text, out var newWeight))
+ return;
+ UpdateImperialWeight(newWeight);
+ UpdateRecords(_records.WithWeight(newWeight));
+ };
+
+ ContactNameEdit.OnTextChanged += args =>
+ {
+ UpdateRecords(_records.WithContactName(args.Text));
+ };
+
+ #endregion
+
+ #region Employment
+
+ WorkAuthCheckBox.OnToggled += args =>
+ {
+ UpdateRecords(_records.WithWorkAuth(args.Pressed));
+ };
+
+ #endregion
+
+ #region Security
+
+ IdentifyingFeaturesEdit.OnTextChanged += args =>
+ {
+ UpdateRecords(_records.WithIdentifyingFeatures(args.Text));
+ };
+
+ #endregion
+
+ #region Medical
+
+ AllergiesEdit.OnTextChanged += args =>
+ {
+ UpdateRecords(_records.WithAllergies(args.Text));
+ };
+
+ DrugAllergiesEdit.OnTextChanged += args =>
+ {
+ UpdateRecords(_records.WithDrugAllergies(args.Text));
+ };
+
+ PostmortemEdit.OnTextChanged += args =>
+ {
+ UpdateRecords(_records.WithPostmortemInstructions(args.Text));
+ };
+
+ #endregion
+
+ #region Entries
+
+ EntryEditorTabs.SetTabTitle(0, Loc.GetString("humanoid-profile-editor-cd-records-employment"));
+ EntryEditorTabs.SetTabTitle(1, Loc.GetString("department-Medical"));
+ EntryEditorTabs.SetTabTitle(2, Loc.GetString("department-Security"));
+
+ EmploymentEntrySelector.OnUpdateEntries += args =>
+ {
+ UpdateRecords(_records.WithEmploymentEntries(args.Entries));
+ };
+
+ MedicalEntrySelector.OnUpdateEntries += args =>
+ {
+ UpdateRecords(_records.WithMedicalEntries(args.Entries));
+ };
+
+ SecurityEntrySelector.OnUpdateEntries += args =>
+ {
+ UpdateRecords(_records.WithSecurityEntries(args.Entries));
+ };
+
+ #endregion
+ }
+
+ public void Update(HumanoidCharacterProfile? profile)
+ {
+ _records = profile?.CDCharacterRecords ?? PlayerProvidedCharacterRecords.DefaultRecords();
+ EmploymentEntrySelector.UpdateContents(_records.EmploymentEntries);
+ MedicalEntrySelector.UpdateContents(_records.MedicalEntries);
+ SecurityEntrySelector.UpdateContents(_records.SecurityEntries);
+ UpdateWidgets();
+ }
+
+ private void UpdateRecords(PlayerProvidedCharacterRecords records)
+ {
+ records.EnsureValid();
+ _records = records;
+ _updateProfileRecords(_records);
+ UpdateWidgets();
+ }
+
+ private void UpdateWidgets()
+ {
+ HeightEdit.SetText(_records.Height.ToString());
+ UpdateImperialHeight(_records.Height);
+ WeightEdit.SetText(_records.Weight.ToString());
+ UpdateImperialWeight(_records.Weight);
+ ContactNameEdit.SetText(_records.EmergencyContactName);
+
+ WorkAuthCheckBox.Pressed = _records.HasWorkAuthorization;
+
+ IdentifyingFeaturesEdit.SetText(_records.IdentifyingFeatures);
+
+ AllergiesEdit.SetText(_records.Allergies);
+ DrugAllergiesEdit.SetText(_records.DrugAllergies);
+ PostmortemEdit.SetText(_records.PostmortemInstructions);
+ }
+
+ private void UpdateImperialHeight(int newHeight)
+ {
+ HeightImperialLabel.Text = UnitConversion.GetImperialDisplayLength(newHeight);
+ }
+
+ private void UpdateImperialWeight(int newWeight)
+ {
+ WeightImperialLabel.Text = UnitConversion.GetImperialDisplayMass(newWeight);
+ }
+}
diff --git a/Content.Client/_CD/Records/UI/RecordEntryEditPopup.xaml b/Content.Client/_CD/Records/UI/RecordEntryEditPopup.xaml
new file mode 100644
index 0000000000..c76e57f724
--- /dev/null
+++ b/Content.Client/_CD/Records/UI/RecordEntryEditPopup.xaml
@@ -0,0 +1,36 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/Content.Client/_CD/Records/UI/RecordEntryEditPopup.xaml.cs b/Content.Client/_CD/Records/UI/RecordEntryEditPopup.xaml.cs
new file mode 100644
index 0000000000..df9e0951f5
--- /dev/null
+++ b/Content.Client/_CD/Records/UI/RecordEntryEditPopup.xaml.cs
@@ -0,0 +1,145 @@
+using Content.Client.UserInterface.Controls;
+using Content.Shared._CD.Records;
+using Robust.Client.AutoGenerated;
+using Robust.Client.Graphics;
+using Robust.Client.UserInterface.XAML;
+using Robust.Shared.Utility;
+using System.Linq;
+
+namespace Content.Client._CD.Records.UI;
+
+[GenerateTypedNameReferences]
+public sealed partial class RecordEntryEditPopup : FancyWindow
+{
+ private bool _isValid;
+
+ private record struct ValidationRule(
+ Func IsInvalid,
+ string LocalizationKey,
+ (string, object)[]? Parameters = null);
+
+ public RecordEntryEditPopup()
+ {
+ RobustXamlLoader.Load(this);
+ IoCManager.InjectDependencies(this);
+
+ ValidationLabel.FontColorOverride = CharacterRecordViewer.ErrorColor;
+ ValidationLabel.Visible = false;
+
+ // Style the content panel
+ var styleBox = new StyleBoxFlat
+ {
+ BackgroundColor = CharacterRecordViewer.ContentPanelColor,
+ BorderColor = CharacterRecordViewer.BorderColor,
+ BorderThickness = new Thickness(1),
+ };
+
+ ContentPanel.PanelOverride = styleBox;
+
+ SetupEventHandlers();
+ }
+
+ private void SetupEventHandlers()
+ {
+ TitleEdit.OnTextChanged += args =>
+ {
+ if (args.Text.Length > PlayerProvidedCharacterRecords.TextMedLen)
+ {
+ TitleEdit.Text = args.Text[..PlayerProvidedCharacterRecords.TextMedLen];
+ }
+ ValidateFields();
+ };
+
+ InvolvedEdit.OnTextChanged += args =>
+ {
+ if (args.Text.Length > PlayerProvidedCharacterRecords.TextMedLen)
+ {
+ InvolvedEdit.Text = args.Text[..PlayerProvidedCharacterRecords.TextMedLen];
+ }
+ ValidateFields();
+ };
+
+ DescriptionEdit.OnTextChanged += _ => ValidateFields();
+
+ SaveButton.OnPressed += _ =>
+ {
+ if (!_isValid)
+ return;
+ Close();
+ };
+
+ DescriptionEdit.Placeholder =
+ new Rope.Leaf(Loc.GetString("cd-records-entry-edit-popup-description-placeholder"));
+
+ TitleEdit.PlaceHolder = Loc.GetString("cd-records-entry-edit-popup-title-placeholder");
+ InvolvedEdit.PlaceHolder = Loc.GetString("cd-records-entry-edit-popup-involved-placeholder");
+ }
+
+ private void ValidateFields()
+ {
+ var descriptionText = Rope.Collapse(DescriptionEdit.TextRope);
+ var descriptionLength = descriptionText.Length;
+
+ // Validation rules in priority order
+ // Overcomplicated into oblivion just because I didn't like how the else if statements looked
+ var rules = new[]
+ {
+ new ValidationRule(
+ () => string.IsNullOrWhiteSpace(TitleEdit.Text),
+ "cd-records-entry-edit-popup-title-required"),
+
+ new ValidationRule(
+ () => string.IsNullOrWhiteSpace(InvolvedEdit.Text),
+ "cd-records-entry-edit-popup-involved-required"),
+
+ new ValidationRule(
+ () => string.IsNullOrWhiteSpace(descriptionText),
+ "cd-records-entry-edit-popup-description-required"),
+
+ new ValidationRule(
+ () => descriptionLength > PlayerProvidedCharacterRecords.TextVeryLargeLen,
+ "cd-records-entry-edit-popup-description-too-long",
+ new[]
+ {
+ ("current", (object)descriptionLength),
+ ("max", PlayerProvidedCharacterRecords.TextVeryLargeLen)
+ }),
+ };
+
+ // Find first failing validation rule
+ var failedRule = rules.FirstOrDefault(rule => rule.IsInvalid());
+
+ if (failedRule.IsInvalid != null)
+ {
+ ValidationLabel.Text = failedRule.Parameters != null
+ ? Loc.GetString(failedRule.LocalizationKey, failedRule.Parameters)
+ : Loc.GetString(failedRule.LocalizationKey);
+ ValidationLabel.Visible = true;
+ _isValid = false;
+ }
+ else
+ {
+ ValidationLabel.Visible = false;
+ _isValid = true;
+ }
+
+ SaveButton.Disabled = !_isValid;
+ }
+
+ public PlayerProvidedCharacterRecords.RecordEntry GetContents()
+ {
+ var desc = Rope.Collapse(DescriptionEdit.TextRope).Trim();
+ return new PlayerProvidedCharacterRecords.RecordEntry(
+ TitleEdit.Text.Trim(),
+ InvolvedEdit.Text.Trim(),
+ desc);
+ }
+
+ public void SetContents(PlayerProvidedCharacterRecords.RecordEntry entry)
+ {
+ TitleEdit.Text = entry.Title;
+ InvolvedEdit.Text = entry.Involved;
+ DescriptionEdit.TextRope = new Rope.Leaf(entry.Description);
+ ValidateFields();
+ }
+}
diff --git a/Content.Client/_CD/Records/UI/RecordEntryViewPopup.xaml b/Content.Client/_CD/Records/UI/RecordEntryViewPopup.xaml
new file mode 100644
index 0000000000..b7f1ce16be
--- /dev/null
+++ b/Content.Client/_CD/Records/UI/RecordEntryViewPopup.xaml
@@ -0,0 +1,38 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/Content.Client/_CD/Records/UI/RecordEntryViewPopup.xaml.cs b/Content.Client/_CD/Records/UI/RecordEntryViewPopup.xaml.cs
new file mode 100644
index 0000000000..d347f9d524
--- /dev/null
+++ b/Content.Client/_CD/Records/UI/RecordEntryViewPopup.xaml.cs
@@ -0,0 +1,46 @@
+using Content.Client.UserInterface.Controls;
+using Content.Shared._CD.Records;
+using Robust.Client.AutoGenerated;
+using Robust.Client.Graphics;
+using Robust.Client.UserInterface.RichText;
+using Robust.Client.UserInterface.XAML;
+using Robust.Shared.Utility;
+
+namespace Content.Client._CD.Records.UI;
+
+[GenerateTypedNameReferences]
+public sealed partial class RecordEntryViewPopup : FancyWindow
+{
+ // Font tags bad
+ private static readonly Type[] AllowedTags =
+ [
+ typeof(BoldItalicTag),
+ typeof(BoldTag),
+ typeof(BulletTag),
+ typeof(ColorTag),
+ typeof(HeadingTag),
+ typeof(ItalicTag),
+ ];
+
+ public RecordEntryViewPopup()
+ {
+ RobustXamlLoader.Load(this);
+
+ // Create a styled box for the content panel
+ var styleBox = new StyleBoxFlat
+ {
+ BackgroundColor = CharacterRecordViewer.ContentPanelColor,
+ BorderColor = CharacterRecordViewer.BorderColor,
+ BorderThickness = new Thickness(1),
+ };
+
+ ContentPanel.PanelOverride = styleBox;
+ }
+
+ public void SetContents(PlayerProvidedCharacterRecords.RecordEntry entry)
+ {
+ EntryTitle.SetMessage(entry.Title);
+ EntryInvolved.SetMessage(entry.Involved);
+ EntryDesc.SetMessage(FormattedMessage.FromMarkupPermissive(entry.Description.Trim()), AllowedTags);
+ }
+}
diff --git a/Content.Client/_CD/Records/UI/RecordLongItemDisplay.cs b/Content.Client/_CD/Records/UI/RecordLongItemDisplay.cs
new file mode 100644
index 0000000000..7eca7114f7
--- /dev/null
+++ b/Content.Client/_CD/Records/UI/RecordLongItemDisplay.cs
@@ -0,0 +1,65 @@
+using Robust.Client.UserInterface;
+using Robust.Client.UserInterface.Controls;
+
+namespace Content.Client._CD.Records.UI;
+
+///
+/// Widget that displays the record on one line if it is short enough, and on two lines if it is
+/// too long. This should only be used if you know the length may be long enough to break things when
+/// using a normal Label.
+///
+public sealed class RecordLongItemDisplay : BoxContainer
+{
+ private const int MaxShortLength = 32;
+
+ public string? Title
+ {
+ get => _titleLabel.Text;
+ set => _titleLabel.Text = value;
+ }
+
+ // Row containing the title and short value
+ private readonly BoxContainer _firstRow = new()
+ {
+ Orientation = LayoutOrientation.Horizontal,
+ HorizontalExpand = true
+ };
+ // Row containing the long value
+ private readonly BoxContainer _secondRow = new()
+ {
+ Orientation = LayoutOrientation.Horizontal,
+ HorizontalExpand = true,
+ Visible = false,
+ };
+ private readonly Label _titleLabel = new();
+ private readonly Label _shortContents = new() { Visible = true, Align = Label.AlignMode.Right };
+ private readonly RichTextLabel _longContents = new() { HorizontalExpand = true };
+
+ public RecordLongItemDisplay()
+ {
+ Orientation = LayoutOrientation.Vertical;
+ _firstRow.AddChild(_titleLabel);
+ _firstRow.AddChild(new Control() { HorizontalExpand = true });
+ _firstRow.AddChild(_shortContents);
+ AddChild(_firstRow);
+ _secondRow.AddChild(new Control() { HorizontalExpand = true, SizeFlagsStretchRatio = 0.15f});
+ _secondRow.AddChild(_longContents);
+ AddChild(_secondRow);
+ }
+
+ public void SetValue(string s)
+ {
+ if (s.Length > MaxShortLength)
+ {
+ _longContents.SetMessage(s);
+ _secondRow.Visible = true;
+ _shortContents.Visible = false;
+ }
+ else
+ {
+ _shortContents.Text = s;
+ _shortContents.Visible = true;
+ _secondRow.Visible = false;
+ }
+ }
+}
diff --git a/Content.Client/_CD/Records/UI/UnitConversion.cs b/Content.Client/_CD/Records/UI/UnitConversion.cs
new file mode 100644
index 0000000000..239ec6937e
--- /dev/null
+++ b/Content.Client/_CD/Records/UI/UnitConversion.cs
@@ -0,0 +1,16 @@
+namespace Content.Client._CD.Records.UI;
+
+public static class UnitConversion
+{
+ public static string GetImperialDisplayLength(int lengthCm)
+ {
+ var heightIn = (int) Math.Round(lengthCm * 0.3937007874 /* cm to in*/);
+ return $"({heightIn / 12}'{heightIn % 12}'')";
+ }
+
+ public static string GetImperialDisplayMass(int massKg)
+ {
+ var weightLbs = (int) Math.Round(massKg * 2.2046226218 /* kg to lbs */);
+ return $"({weightLbs} lbs)";
+ }
+}
diff --git a/Content.Server.Database/CDModel.cs b/Content.Server.Database/CDModel.cs
new file mode 100644
index 0000000000..f32de41a52
--- /dev/null
+++ b/Content.Server.Database/CDModel.cs
@@ -0,0 +1,56 @@
+// File to store as much CD related database things outside of Model.cs
+
+using System;
+using System.Collections.Generic;
+using System.ComponentModel.DataAnnotations.Schema;
+using System.Text.Json;
+using Microsoft.EntityFrameworkCore;
+
+namespace Content.Server.Database;
+
+public static class CDModel
+{
+ ///
+ /// Stores CD Character data separately from the main Profile. This is done to work around a bug
+ /// in EFCore migrations.
+ ///
+ /// There is no way of forcing a dependent table to exist in EFCore (according to MS).
+ /// You must always account for the possibility of this table not existing.
+ ///
+ public class CDProfile
+ {
+ public int Id { get; set; }
+
+ public int ProfileId { get; set; }
+ public Profile Profile { get; set; } = null!;
+
+ public float Height { get; set; } = 1f;
+
+ [Column("character_records", TypeName = "jsonb")]
+ public JsonDocument? CharacterRecords { get; set; }
+
+ public List CharacterRecordEntries { get; set; } = new();
+
+ }
+ public enum DbRecordEntryType : byte
+ {
+ Medical = 0, Security = 1, Employment = 2
+ }
+
+ [Table("cd_character_record_entries"), Index(nameof(Id))]
+ public sealed class CharacterRecordEntry
+ {
+ public int Id { get; set; }
+
+ public string Title { get; set; } = null!;
+
+ public string Involved { get; set; } = null!;
+
+ public string Description { get; set; } = null!;
+
+ public DbRecordEntryType Type { get; set; }
+
+ public int CDProfileId { get; set; }
+ public CDProfile CDProfile { get; set; } = null!;
+ }
+}
diff --git a/Content.Server.Database/Migrations/Postgres/20250303043202_CosmaticDriftCharacterRecords.Designer.cs b/Content.Server.Database/Migrations/Postgres/20250303043202_CosmaticDriftCharacterRecords.Designer.cs
new file mode 100644
index 0000000000..4f6dd27fda
--- /dev/null
+++ b/Content.Server.Database/Migrations/Postgres/20250303043202_CosmaticDriftCharacterRecords.Designer.cs
@@ -0,0 +1,2216 @@
+//
+using System;
+using System.Net;
+using System.Text.Json;
+using Content.Server.Database;
+using Microsoft.EntityFrameworkCore;
+using Microsoft.EntityFrameworkCore.Infrastructure;
+using Microsoft.EntityFrameworkCore.Migrations;
+using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
+using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
+using NpgsqlTypes;
+
+#nullable disable
+
+namespace Content.Server.Database.Migrations.Postgres
+{
+ [DbContext(typeof(PostgresServerDbContext))]
+ [Migration("20250303043202_CosmaticDriftCharacterRecords")]
+ partial class CosmaticDriftCharacterRecords
+ {
+ ///
+ protected override void BuildTargetModel(ModelBuilder modelBuilder)
+ {
+#pragma warning disable 612, 618
+ modelBuilder
+ .HasAnnotation("ProductVersion", "8.0.0")
+ .HasAnnotation("Relational:MaxIdentifierLength", 63);
+
+ NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder);
+
+ modelBuilder.Entity("Content.Server.Database.Admin", b =>
+ {
+ b.Property("UserId")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("uuid")
+ .HasColumnName("user_id");
+
+ b.Property("AdminRankId")
+ .HasColumnType("integer")
+ .HasColumnName("admin_rank_id");
+
+ b.Property("Deadminned")
+ .HasColumnType("boolean")
+ .HasColumnName("deadminned");
+
+ b.Property("Suspended")
+ .HasColumnType("boolean")
+ .HasColumnName("suspended");
+
+ b.Property("Title")
+ .HasColumnType("text")
+ .HasColumnName("title");
+
+ b.HasKey("UserId")
+ .HasName("PK_admin");
+
+ b.HasIndex("AdminRankId")
+ .HasDatabaseName("IX_admin_admin_rank_id");
+
+ b.ToTable("admin", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.AdminFlag", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("admin_flag_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("AdminId")
+ .HasColumnType("uuid")
+ .HasColumnName("admin_id");
+
+ b.Property("Flag")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("flag");
+
+ b.Property("Negative")
+ .HasColumnType("boolean")
+ .HasColumnName("negative");
+
+ b.HasKey("Id")
+ .HasName("PK_admin_flag");
+
+ b.HasIndex("AdminId")
+ .HasDatabaseName("IX_admin_flag_admin_id");
+
+ b.HasIndex("Flag", "AdminId")
+ .IsUnique();
+
+ b.ToTable("admin_flag", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.AdminLog", b =>
+ {
+ b.Property("RoundId")
+ .HasColumnType("integer")
+ .HasColumnName("round_id");
+
+ b.Property("Id")
+ .HasColumnType("integer")
+ .HasColumnName("admin_log_id");
+
+ b.Property("Date")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("date");
+
+ b.Property("Impact")
+ .HasColumnType("smallint")
+ .HasColumnName("impact");
+
+ b.Property("Json")
+ .IsRequired()
+ .HasColumnType("jsonb")
+ .HasColumnName("json");
+
+ b.Property("Message")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("message");
+
+ b.Property("Type")
+ .HasColumnType("integer")
+ .HasColumnName("type");
+
+ b.HasKey("RoundId", "Id")
+ .HasName("PK_admin_log");
+
+ b.HasIndex("Date");
+
+ b.HasIndex("Message")
+ .HasAnnotation("Npgsql:TsVectorConfig", "english");
+
+ NpgsqlIndexBuilderExtensions.HasMethod(b.HasIndex("Message"), "GIN");
+
+ b.HasIndex("Type")
+ .HasDatabaseName("IX_admin_log_type");
+
+ b.ToTable("admin_log", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.AdminLogPlayer", b =>
+ {
+ b.Property("RoundId")
+ .HasColumnType("integer")
+ .HasColumnName("round_id");
+
+ b.Property("LogId")
+ .HasColumnType("integer")
+ .HasColumnName("log_id");
+
+ b.Property("PlayerUserId")
+ .HasColumnType("uuid")
+ .HasColumnName("player_user_id");
+
+ b.HasKey("RoundId", "LogId", "PlayerUserId")
+ .HasName("PK_admin_log_player");
+
+ b.HasIndex("PlayerUserId")
+ .HasDatabaseName("IX_admin_log_player_player_user_id");
+
+ b.ToTable("admin_log_player", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.AdminMessage", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("admin_messages_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("CreatedAt")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("created_at");
+
+ b.Property("CreatedById")
+ .HasColumnType("uuid")
+ .HasColumnName("created_by_id");
+
+ b.Property("Deleted")
+ .HasColumnType("boolean")
+ .HasColumnName("deleted");
+
+ b.Property("DeletedAt")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("deleted_at");
+
+ b.Property("DeletedById")
+ .HasColumnType("uuid")
+ .HasColumnName("deleted_by_id");
+
+ b.Property("Dismissed")
+ .HasColumnType("boolean")
+ .HasColumnName("dismissed");
+
+ b.Property("ExpirationTime")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("expiration_time");
+
+ b.Property("LastEditedAt")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("last_edited_at");
+
+ b.Property("LastEditedById")
+ .HasColumnType("uuid")
+ .HasColumnName("last_edited_by_id");
+
+ b.Property("Message")
+ .IsRequired()
+ .HasMaxLength(4096)
+ .HasColumnType("character varying(4096)")
+ .HasColumnName("message");
+
+ b.Property("PlayerUserId")
+ .HasColumnType("uuid")
+ .HasColumnName("player_user_id");
+
+ b.Property("PlaytimeAtNote")
+ .HasColumnType("interval")
+ .HasColumnName("playtime_at_note");
+
+ b.Property("RoundId")
+ .HasColumnType("integer")
+ .HasColumnName("round_id");
+
+ b.Property("Seen")
+ .HasColumnType("boolean")
+ .HasColumnName("seen");
+
+ b.HasKey("Id")
+ .HasName("PK_admin_messages");
+
+ b.HasIndex("CreatedById");
+
+ b.HasIndex("DeletedById");
+
+ b.HasIndex("LastEditedById");
+
+ b.HasIndex("PlayerUserId")
+ .HasDatabaseName("IX_admin_messages_player_user_id");
+
+ b.HasIndex("RoundId")
+ .HasDatabaseName("IX_admin_messages_round_id");
+
+ b.ToTable("admin_messages", null, t =>
+ {
+ t.HasCheckConstraint("NotDismissedAndSeen", "NOT dismissed OR seen");
+ });
+ });
+
+ modelBuilder.Entity("Content.Server.Database.AdminNote", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("admin_notes_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("CreatedAt")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("created_at");
+
+ b.Property("CreatedById")
+ .HasColumnType("uuid")
+ .HasColumnName("created_by_id");
+
+ b.Property("Deleted")
+ .HasColumnType("boolean")
+ .HasColumnName("deleted");
+
+ b.Property("DeletedAt")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("deleted_at");
+
+ b.Property("DeletedById")
+ .HasColumnType("uuid")
+ .HasColumnName("deleted_by_id");
+
+ b.Property("ExpirationTime")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("expiration_time");
+
+ b.Property("LastEditedAt")
+ .IsRequired()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("last_edited_at");
+
+ b.Property("LastEditedById")
+ .HasColumnType("uuid")
+ .HasColumnName("last_edited_by_id");
+
+ b.Property("Message")
+ .IsRequired()
+ .HasMaxLength(4096)
+ .HasColumnType("character varying(4096)")
+ .HasColumnName("message");
+
+ b.Property("PlayerUserId")
+ .HasColumnType("uuid")
+ .HasColumnName("player_user_id");
+
+ b.Property("PlaytimeAtNote")
+ .HasColumnType("interval")
+ .HasColumnName("playtime_at_note");
+
+ b.Property("RoundId")
+ .HasColumnType("integer")
+ .HasColumnName("round_id");
+
+ b.Property("Secret")
+ .HasColumnType("boolean")
+ .HasColumnName("secret");
+
+ b.Property("Severity")
+ .HasColumnType("integer")
+ .HasColumnName("severity");
+
+ b.HasKey("Id")
+ .HasName("PK_admin_notes");
+
+ b.HasIndex("CreatedById");
+
+ b.HasIndex("DeletedById");
+
+ b.HasIndex("LastEditedById");
+
+ b.HasIndex("PlayerUserId")
+ .HasDatabaseName("IX_admin_notes_player_user_id");
+
+ b.HasIndex("RoundId")
+ .HasDatabaseName("IX_admin_notes_round_id");
+
+ b.ToTable("admin_notes", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.AdminRank", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("admin_rank_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("Name")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("name");
+
+ b.HasKey("Id")
+ .HasName("PK_admin_rank");
+
+ b.ToTable("admin_rank", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.AdminRankFlag", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("admin_rank_flag_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("AdminRankId")
+ .HasColumnType("integer")
+ .HasColumnName("admin_rank_id");
+
+ b.Property("Flag")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("flag");
+
+ b.HasKey("Id")
+ .HasName("PK_admin_rank_flag");
+
+ b.HasIndex("AdminRankId");
+
+ b.HasIndex("Flag", "AdminRankId")
+ .IsUnique();
+
+ b.ToTable("admin_rank_flag", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.AdminWatchlist", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("admin_watchlists_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("CreatedAt")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("created_at");
+
+ b.Property("CreatedById")
+ .HasColumnType("uuid")
+ .HasColumnName("created_by_id");
+
+ b.Property("Deleted")
+ .HasColumnType("boolean")
+ .HasColumnName("deleted");
+
+ b.Property("DeletedAt")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("deleted_at");
+
+ b.Property("DeletedById")
+ .HasColumnType("uuid")
+ .HasColumnName("deleted_by_id");
+
+ b.Property("ExpirationTime")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("expiration_time");
+
+ b.Property("LastEditedAt")
+ .IsRequired()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("last_edited_at");
+
+ b.Property("LastEditedById")
+ .HasColumnType("uuid")
+ .HasColumnName("last_edited_by_id");
+
+ b.Property("Message")
+ .IsRequired()
+ .HasMaxLength(4096)
+ .HasColumnType("character varying(4096)")
+ .HasColumnName("message");
+
+ b.Property("PlayerUserId")
+ .HasColumnType("uuid")
+ .HasColumnName("player_user_id");
+
+ b.Property("PlaytimeAtNote")
+ .HasColumnType("interval")
+ .HasColumnName("playtime_at_note");
+
+ b.Property("RoundId")
+ .HasColumnType("integer")
+ .HasColumnName("round_id");
+
+ b.HasKey("Id")
+ .HasName("PK_admin_watchlists");
+
+ b.HasIndex("CreatedById");
+
+ b.HasIndex("DeletedById");
+
+ b.HasIndex("LastEditedById");
+
+ b.HasIndex("PlayerUserId")
+ .HasDatabaseName("IX_admin_watchlists_player_user_id");
+
+ b.HasIndex("RoundId")
+ .HasDatabaseName("IX_admin_watchlists_round_id");
+
+ b.ToTable("admin_watchlists", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.Antag", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("antag_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("AntagName")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("antag_name");
+
+ b.Property("ProfileId")
+ .HasColumnType("integer")
+ .HasColumnName("profile_id");
+
+ b.HasKey("Id")
+ .HasName("PK_antag");
+
+ b.HasIndex("ProfileId", "AntagName")
+ .IsUnique();
+
+ b.ToTable("antag", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.AssignedUserId", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("assigned_user_id_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("UserId")
+ .HasColumnType("uuid")
+ .HasColumnName("user_id");
+
+ b.Property("UserName")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("user_name");
+
+ b.HasKey("Id")
+ .HasName("PK_assigned_user_id");
+
+ b.HasIndex("UserId")
+ .IsUnique();
+
+ b.HasIndex("UserName")
+ .IsUnique();
+
+ b.ToTable("assigned_user_id", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.BanTemplate", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("ban_template_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("AutoDelete")
+ .HasColumnType("boolean")
+ .HasColumnName("auto_delete");
+
+ b.Property("ExemptFlags")
+ .HasColumnType("integer")
+ .HasColumnName("exempt_flags");
+
+ b.Property("Hidden")
+ .HasColumnType("boolean")
+ .HasColumnName("hidden");
+
+ b.Property("Length")
+ .HasColumnType("interval")
+ .HasColumnName("length");
+
+ b.Property("Reason")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("reason");
+
+ b.Property("Severity")
+ .HasColumnType("integer")
+ .HasColumnName("severity");
+
+ b.Property("Title")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("title");
+
+ b.HasKey("Id")
+ .HasName("PK_ban_template");
+
+ b.ToTable("ban_template", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.Blacklist", b =>
+ {
+ b.Property("UserId")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("uuid")
+ .HasColumnName("user_id");
+
+ b.HasKey("UserId")
+ .HasName("PK_blacklist");
+
+ b.ToTable("blacklist", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.CDModel+CDProfile", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("cdprofile_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("CharacterRecords")
+ .HasColumnType("jsonb")
+ .HasColumnName("character_records");
+
+ b.Property("Height")
+ .HasColumnType("real")
+ .HasColumnName("height");
+
+ b.Property("ProfileId")
+ .HasColumnType("integer")
+ .HasColumnName("profile_id");
+
+ b.HasKey("Id")
+ .HasName("PK_cdprofile");
+
+ b.HasIndex("ProfileId")
+ .IsUnique();
+
+ b.ToTable("cdprofile", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.CDModel+CharacterRecordEntry", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("cd_character_record_entries_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("CDProfileId")
+ .HasColumnType("integer")
+ .HasColumnName("cdprofile_id");
+
+ b.Property("Description")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("description");
+
+ b.Property("Involved")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("involved");
+
+ b.Property("Title")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("title");
+
+ b.Property("Type")
+ .HasColumnType("smallint")
+ .HasColumnName("type");
+
+ b.HasKey("Id")
+ .HasName("PK_cd_character_record_entries");
+
+ b.HasIndex("CDProfileId");
+
+ b.HasIndex("Id")
+ .HasDatabaseName("IX_cd_character_record_entries_cd_character_record_entries_id");
+
+ b.ToTable("cd_character_record_entries", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.ConnectionLog", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("connection_log_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("Address")
+ .IsRequired()
+ .HasColumnType("inet")
+ .HasColumnName("address");
+
+ b.Property("Denied")
+ .HasColumnType("smallint")
+ .HasColumnName("denied");
+
+ b.Property("ServerId")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasDefaultValue(0)
+ .HasColumnName("server_id");
+
+ b.Property("Time")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("time");
+
+ b.Property("Trust")
+ .HasColumnType("real")
+ .HasColumnName("trust");
+
+ b.Property("UserId")
+ .HasColumnType("uuid")
+ .HasColumnName("user_id");
+
+ b.Property("UserName")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("user_name");
+
+ b.HasKey("Id")
+ .HasName("PK_connection_log");
+
+ b.HasIndex("ServerId")
+ .HasDatabaseName("IX_connection_log_server_id");
+
+ b.HasIndex("Time");
+
+ b.HasIndex("UserId");
+
+ b.ToTable("connection_log", null, t =>
+ {
+ t.HasCheckConstraint("AddressNotIPv6MappedIPv4", "NOT inet '::ffff:0.0.0.0/96' >>= address");
+ });
+ });
+
+ modelBuilder.Entity("Content.Server.Database.IPIntelCache", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("ipintel_cache_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("Address")
+ .IsRequired()
+ .HasColumnType("inet")
+ .HasColumnName("address");
+
+ b.Property("Score")
+ .HasColumnType("real")
+ .HasColumnName("score");
+
+ b.Property("Time")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("time");
+
+ b.HasKey("Id")
+ .HasName("PK_ipintel_cache");
+
+ b.ToTable("ipintel_cache", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.Job", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("job_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("JobName")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("job_name");
+
+ b.Property("Priority")
+ .HasColumnType("integer")
+ .HasColumnName("priority");
+
+ b.Property("ProfileId")
+ .HasColumnType("integer")
+ .HasColumnName("profile_id");
+
+ b.HasKey("Id")
+ .HasName("PK_job");
+
+ b.HasIndex("ProfileId");
+
+ b.HasIndex("ProfileId", "JobName")
+ .IsUnique();
+
+ b.HasIndex(new[] { "ProfileId" }, "IX_job_one_high_priority")
+ .IsUnique()
+ .HasFilter("priority = 3");
+
+ b.ToTable("job", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.PlayTime", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("play_time_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("PlayerId")
+ .HasColumnType("uuid")
+ .HasColumnName("player_id");
+
+ b.Property("TimeSpent")
+ .HasColumnType("interval")
+ .HasColumnName("time_spent");
+
+ b.Property("Tracker")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("tracker");
+
+ b.HasKey("Id")
+ .HasName("PK_play_time");
+
+ b.HasIndex("PlayerId", "Tracker")
+ .IsUnique();
+
+ b.ToTable("play_time", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.Player", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("player_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("FirstSeenTime")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("first_seen_time");
+
+ b.Property("LastReadRules")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("last_read_rules");
+
+ b.Property("LastSeenAddress")
+ .IsRequired()
+ .HasColumnType("inet")
+ .HasColumnName("last_seen_address");
+
+ b.Property("LastSeenTime")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("last_seen_time");
+
+ b.Property("LastSeenUserName")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("last_seen_user_name");
+
+ b.Property("UserId")
+ .HasColumnType("uuid")
+ .HasColumnName("user_id");
+
+ b.HasKey("Id")
+ .HasName("PK_player");
+
+ b.HasAlternateKey("UserId")
+ .HasName("ak_player_user_id");
+
+ b.HasIndex("LastSeenUserName");
+
+ b.HasIndex("UserId")
+ .IsUnique();
+
+ b.ToTable("player", null, t =>
+ {
+ t.HasCheckConstraint("LastSeenAddressNotIPv6MappedIPv4", "NOT inet '::ffff:0.0.0.0/96' >>= last_seen_address");
+ });
+ });
+
+ modelBuilder.Entity("Content.Server.Database.Preference", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("preference_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("AdminOOCColor")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("admin_ooc_color");
+
+ b.Property("SelectedCharacterSlot")
+ .HasColumnType("integer")
+ .HasColumnName("selected_character_slot");
+
+ b.Property("UserId")
+ .HasColumnType("uuid")
+ .HasColumnName("user_id");
+
+ b.HasKey("Id")
+ .HasName("PK_preference");
+
+ b.HasIndex("UserId")
+ .IsUnique();
+
+ b.ToTable("preference", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.Profile", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("profile_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("Age")
+ .HasColumnType("integer")
+ .HasColumnName("age");
+
+ b.Property("CharacterName")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("char_name");
+
+ b.Property("EyeColor")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("eye_color");
+
+ b.Property("FacialHairColor")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("facial_hair_color");
+
+ b.Property("FacialHairName")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("facial_hair_name");
+
+ b.Property("FlavorText")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("flavor_text");
+
+ b.Property("Gender")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("gender");
+
+ b.Property("HairColor")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("hair_color");
+
+ b.Property("HairName")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("hair_name");
+
+ b.Property("Markings")
+ .HasColumnType("jsonb")
+ .HasColumnName("markings");
+
+ b.Property("PreferenceId")
+ .HasColumnType("integer")
+ .HasColumnName("preference_id");
+
+ b.Property("PreferenceUnavailable")
+ .HasColumnType("integer")
+ .HasColumnName("pref_unavailable");
+
+ b.Property("Sex")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("sex");
+
+ b.Property("SkinColor")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("skin_color");
+
+ b.Property("Slot")
+ .HasColumnType("integer")
+ .HasColumnName("slot");
+
+ b.Property("SpawnPriority")
+ .HasColumnType("integer")
+ .HasColumnName("spawn_priority");
+
+ b.Property("Species")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("species");
+
+ b.HasKey("Id")
+ .HasName("PK_profile");
+
+ b.HasIndex("PreferenceId")
+ .HasDatabaseName("IX_profile_preference_id");
+
+ b.HasIndex("Slot", "PreferenceId")
+ .IsUnique();
+
+ b.ToTable("profile", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.ProfileLoadout", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("profile_loadout_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("LoadoutName")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("loadout_name");
+
+ b.Property("ProfileLoadoutGroupId")
+ .HasColumnType("integer")
+ .HasColumnName("profile_loadout_group_id");
+
+ b.HasKey("Id")
+ .HasName("PK_profile_loadout");
+
+ b.HasIndex("ProfileLoadoutGroupId");
+
+ b.ToTable("profile_loadout", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.ProfileLoadoutGroup", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("profile_loadout_group_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("GroupName")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("group_name");
+
+ b.Property("ProfileRoleLoadoutId")
+ .HasColumnType("integer")
+ .HasColumnName("profile_role_loadout_id");
+
+ b.HasKey("Id")
+ .HasName("PK_profile_loadout_group");
+
+ b.HasIndex("ProfileRoleLoadoutId");
+
+ b.ToTable("profile_loadout_group", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.ProfileRoleLoadout", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("profile_role_loadout_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("ProfileId")
+ .HasColumnType("integer")
+ .HasColumnName("profile_id");
+
+ b.Property("RoleName")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("role_name");
+
+ b.HasKey("Id")
+ .HasName("PK_profile_role_loadout");
+
+ b.HasIndex("ProfileId");
+
+ b.ToTable("profile_role_loadout", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.RoleWhitelist", b =>
+ {
+ b.Property("PlayerUserId")
+ .HasColumnType("uuid")
+ .HasColumnName("player_user_id");
+
+ b.Property("RoleId")
+ .HasColumnType("text")
+ .HasColumnName("role_id");
+
+ b.HasKey("PlayerUserId", "RoleId")
+ .HasName("PK_role_whitelists");
+
+ b.ToTable("role_whitelists", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.Round", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("round_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("ServerId")
+ .HasColumnType("integer")
+ .HasColumnName("server_id");
+
+ b.Property("StartDate")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("start_date");
+
+ b.HasKey("Id")
+ .HasName("PK_round");
+
+ b.HasIndex("ServerId")
+ .HasDatabaseName("IX_round_server_id");
+
+ b.HasIndex("StartDate");
+
+ b.ToTable("round", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.Server", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("server_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("Name")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("name");
+
+ b.HasKey("Id")
+ .HasName("PK_server");
+
+ b.ToTable("server", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.ServerBan", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("server_ban_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("Address")
+ .HasColumnType("inet")
+ .HasColumnName("address");
+
+ b.Property("AutoDelete")
+ .HasColumnType("boolean")
+ .HasColumnName("auto_delete");
+
+ b.Property("BanTime")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("ban_time");
+
+ b.Property("BanningAdmin")
+ .HasColumnType("uuid")
+ .HasColumnName("banning_admin");
+
+ b.Property("ExemptFlags")
+ .HasColumnType("integer")
+ .HasColumnName("exempt_flags");
+
+ b.Property("ExpirationTime")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("expiration_time");
+
+ b.Property("Hidden")
+ .HasColumnType("boolean")
+ .HasColumnName("hidden");
+
+ b.Property("LastEditedAt")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("last_edited_at");
+
+ b.Property("LastEditedById")
+ .HasColumnType("uuid")
+ .HasColumnName("last_edited_by_id");
+
+ b.Property("PlayerUserId")
+ .HasColumnType("uuid")
+ .HasColumnName("player_user_id");
+
+ b.Property("PlaytimeAtNote")
+ .HasColumnType("interval")
+ .HasColumnName("playtime_at_note");
+
+ b.Property("Reason")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("reason");
+
+ b.Property("RoundId")
+ .HasColumnType("integer")
+ .HasColumnName("round_id");
+
+ b.Property("Severity")
+ .HasColumnType("integer")
+ .HasColumnName("severity");
+
+ b.HasKey("Id")
+ .HasName("PK_server_ban");
+
+ b.HasIndex("Address");
+
+ b.HasIndex("BanningAdmin");
+
+ b.HasIndex("LastEditedById");
+
+ b.HasIndex("PlayerUserId")
+ .HasDatabaseName("IX_server_ban_player_user_id");
+
+ b.HasIndex("RoundId")
+ .HasDatabaseName("IX_server_ban_round_id");
+
+ b.ToTable("server_ban", null, t =>
+ {
+ t.HasCheckConstraint("AddressNotIPv6MappedIPv4", "NOT inet '::ffff:0.0.0.0/96' >>= address");
+
+ t.HasCheckConstraint("HaveEitherAddressOrUserIdOrHWId", "address IS NOT NULL OR player_user_id IS NOT NULL OR hwid IS NOT NULL");
+ });
+ });
+
+ modelBuilder.Entity("Content.Server.Database.ServerBanExemption", b =>
+ {
+ b.Property("UserId")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("uuid")
+ .HasColumnName("user_id");
+
+ b.Property("Flags")
+ .HasColumnType("integer")
+ .HasColumnName("flags");
+
+ b.HasKey("UserId")
+ .HasName("PK_server_ban_exemption");
+
+ b.ToTable("server_ban_exemption", null, t =>
+ {
+ t.HasCheckConstraint("FlagsNotZero", "flags != 0");
+ });
+ });
+
+ modelBuilder.Entity("Content.Server.Database.ServerBanHit", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("server_ban_hit_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("BanId")
+ .HasColumnType("integer")
+ .HasColumnName("ban_id");
+
+ b.Property("ConnectionId")
+ .HasColumnType("integer")
+ .HasColumnName("connection_id");
+
+ b.HasKey("Id")
+ .HasName("PK_server_ban_hit");
+
+ b.HasIndex("BanId")
+ .HasDatabaseName("IX_server_ban_hit_ban_id");
+
+ b.HasIndex("ConnectionId")
+ .HasDatabaseName("IX_server_ban_hit_connection_id");
+
+ b.ToTable("server_ban_hit", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.ServerRoleBan", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("server_role_ban_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("Address")
+ .HasColumnType("inet")
+ .HasColumnName("address");
+
+ b.Property("BanTime")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("ban_time");
+
+ b.Property("BanningAdmin")
+ .HasColumnType("uuid")
+ .HasColumnName("banning_admin");
+
+ b.Property("ExpirationTime")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("expiration_time");
+
+ b.Property("Hidden")
+ .HasColumnType("boolean")
+ .HasColumnName("hidden");
+
+ b.Property("LastEditedAt")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("last_edited_at");
+
+ b.Property("LastEditedById")
+ .HasColumnType("uuid")
+ .HasColumnName("last_edited_by_id");
+
+ b.Property("PlayerUserId")
+ .HasColumnType("uuid")
+ .HasColumnName("player_user_id");
+
+ b.Property("PlaytimeAtNote")
+ .HasColumnType("interval")
+ .HasColumnName("playtime_at_note");
+
+ b.Property("Reason")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("reason");
+
+ b.Property("RoleId")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("role_id");
+
+ b.Property("RoundId")
+ .HasColumnType("integer")
+ .HasColumnName("round_id");
+
+ b.Property("Severity")
+ .HasColumnType("integer")
+ .HasColumnName("severity");
+
+ b.HasKey("Id")
+ .HasName("PK_server_role_ban");
+
+ b.HasIndex("Address");
+
+ b.HasIndex("BanningAdmin");
+
+ b.HasIndex("LastEditedById");
+
+ b.HasIndex("PlayerUserId")
+ .HasDatabaseName("IX_server_role_ban_player_user_id");
+
+ b.HasIndex("RoundId")
+ .HasDatabaseName("IX_server_role_ban_round_id");
+
+ b.ToTable("server_role_ban", null, t =>
+ {
+ t.HasCheckConstraint("AddressNotIPv6MappedIPv4", "NOT inet '::ffff:0.0.0.0/96' >>= address");
+
+ t.HasCheckConstraint("HaveEitherAddressOrUserIdOrHWId", "address IS NOT NULL OR player_user_id IS NOT NULL OR hwid IS NOT NULL");
+ });
+ });
+
+ modelBuilder.Entity("Content.Server.Database.ServerRoleUnban", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("role_unban_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("BanId")
+ .HasColumnType("integer")
+ .HasColumnName("ban_id");
+
+ b.Property("UnbanTime")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("unban_time");
+
+ b.Property("UnbanningAdmin")
+ .HasColumnType("uuid")
+ .HasColumnName("unbanning_admin");
+
+ b.HasKey("Id")
+ .HasName("PK_server_role_unban");
+
+ b.HasIndex("BanId")
+ .IsUnique();
+
+ b.ToTable("server_role_unban", (string)null);
+ });
+
+ modelBuilder.Entity("Content.Server.Database.ServerUnban", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("unban_id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("BanId")
+ .HasColumnType("integer")
+ .HasColumnName("ban_id");
+
+ b.Property("UnbanTime")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("unban_time");
+
+ b.Property