Penumbra/Penumbra/UI/ModsTab/Groups/ModGroupDrawer.cs
Ottermandias cabcaadde3
Some checks failed
.NET Build / build (push) Has been cancelled
Current state, all done except for file system selector.
2025-12-28 00:06:32 +01:00

248 lines
8.8 KiB
C#

using ImSharp;
using Luna;
using Penumbra.Collections;
using Penumbra.Collections.Manager;
using Penumbra.Mods;
using Penumbra.Mods.Groups;
using Penumbra.Mods.Settings;
using Penumbra.Mods.SubMods;
namespace Penumbra.UI.ModsTab.Groups;
public sealed class ModGroupDrawer(Configuration config, CollectionManager collectionManager, SingleGroupCombo combo)
: IUiService
{
private readonly List<(IModGroup, int)> _blockGroupCache = [];
private bool _temporary;
private bool _locked;
private TemporaryModSettings? _tempSettings;
private ModSettings? _settings;
public void Draw(Mod mod, ModSettings settings, TemporaryModSettings? tempSettings)
{
if (mod.Groups.Count <= 0)
return;
_blockGroupCache.Clear();
_settings = settings;
_tempSettings = tempSettings;
_temporary = tempSettings != null;
_locked = (tempSettings?.Lock ?? 0) > 0;
var useDummy = true;
foreach (var (idx, group) in mod.Groups.Index())
{
if (!group.IsOption)
continue;
switch (group.Behaviour)
{
case GroupDrawBehaviour.SingleSelection when group.Options.Count <= config.SingleGroupRadioMax:
case GroupDrawBehaviour.MultiSelection:
_blockGroupCache.Add((group, idx));
break;
case GroupDrawBehaviour.SingleSelection:
if (useDummy)
{
Im.Dummy(UiHelpers.DefaultSpace);
useDummy = false;
}
DrawSingleGroupCombo(group, idx, settings.IsEmpty ? group.DefaultSettings : settings.Settings[idx]);
break;
}
}
useDummy = true;
foreach (var (group, idx) in _blockGroupCache)
{
if (useDummy)
{
Im.Dummy(UiHelpers.DefaultSpace);
useDummy = false;
}
var option = settings.IsEmpty ? group.DefaultSettings : settings.Settings[idx];
if (group.Behaviour is GroupDrawBehaviour.MultiSelection)
DrawMultiGroup(group, idx, option);
else
DrawSingleGroupRadio(group, idx, option);
}
}
/// <summary>
/// Draw a single group selector as a combo box.
/// If a description is provided, add a help marker besides it.
/// </summary>
private void DrawSingleGroupCombo(IModGroup group, int groupIdx, Setting setting)
{
using var id = Im.Id.Push(groupIdx);
using var disabled = Im.Disabled(_locked);
combo.Draw(this, (SingleModGroup)group, groupIdx, setting);
if (group.Description.Length > 0)
{
LunaStyle.DrawHelpMarkerLabel(group.Name, group.Description);
}
else
{
Im.Line.SameInner();
Im.Text(group.Name);
}
}
/// <summary>
/// Draw a single group selector as a set of radio buttons.
/// If a description is provided, add a help marker besides it.
/// </summary>
private void DrawSingleGroupRadio(IModGroup group, int groupIdx, Setting setting)
{
using var id = Im.Id.Push(groupIdx);
var options = group.Options;
var selectedOption = setting.AsIndex;
using var g = ImEx.FramedGroup(group.Name, LunaStyle.HelpMarker, group.Description);
DrawCollapseHandling(options, g.MinimumWidth, DrawOptions);
return;
void DrawOptions()
{
using var disabled = Im.Disabled(_locked);
for (var idx = 0; idx < group.Options.Count; ++idx)
{
using var i = Im.Id.Push(idx);
var option = options[idx];
if (Im.RadioButton(option.Name, selectedOption == idx))
SetModSetting(group, groupIdx, Setting.Single(idx));
if (option.Description.Length is 0)
continue;
Im.Line.SameInner();
LunaStyle.DrawAlignedHelpMarker(option.Description, treatAsHovered: Im.Item.Hovered());
}
}
}
/// <summary>
/// Draw a multi group selector as a bordered set of checkboxes.
/// If a description is provided, add a help marker in the title.
/// </summary>
private void DrawMultiGroup(IModGroup group, int groupIdx, Setting setting)
{
using var id = Im.Id.Push(groupIdx);
var options = group.Options;
using (var g = ImEx.FramedGroup(group.Name, LunaStyle.HelpMarker, group.Description))
{
DrawCollapseHandling(options, g.MinimumWidth, DrawOptions);
}
var label = new StringU8($"##multi{groupIdx}");
if (Im.Item.RightClicked())
Im.Popup.Open(label);
DrawMultiPopup(group, groupIdx, label);
return;
void DrawOptions()
{
using var disabled = Im.Disabled(_locked);
for (var idx = 0; idx < options.Count; ++idx)
{
using var i = Im.Id.Push(idx);
var option = options[idx];
var enabled = setting.HasFlag(idx);
if (Im.Checkbox(option.Name, ref enabled))
SetModSetting(group, groupIdx, setting.SetBit(idx, enabled));
if (option.Description.Length > 0)
{
Im.Line.SameInner();
LunaStyle.DrawAlignedHelpMarker(option.Description, treatAsHovered: Im.Item.Hovered());
}
}
}
}
private void DrawMultiPopup(IModGroup group, int groupIdx, StringU8 label)
{
using var style = ImStyleSingle.PopupBorderThickness.Push(Im.Style.GlobalScale);
using var popup = Im.Popup.Begin(label);
if (!popup)
return;
Im.Text(group.Name);
using var disabled = Im.Disabled(_locked);
Im.Separator();
if (Im.Selectable("Enable All"u8))
SetModSetting(group, groupIdx, Setting.AllBits(group.Options.Count));
if (Im.Selectable("Disable All"u8))
SetModSetting(group, groupIdx, Setting.Zero);
}
private void DrawCollapseHandling(IReadOnlyList<IModOption> options, float minWidth, Action draw)
{
if (options.Count <= config.OptionGroupCollapsibleMin)
{
draw();
}
else
{
var collapseId = Im.Id.Get("Collapse"u8);
var shown = Im.State.Storage.GetBool(collapseId, true);
var buttonTextShow = new StringU8($"Show {options.Count} Options");
var buttonTextHide = new StringU8($"Hide {options.Count} Options");
var buttonWidth = Math.Max(Im.Font.CalculateSize(buttonTextShow).X, Im.Font.CalculateSize(buttonTextHide).X)
+ 2 * Im.Style.FramePadding.X;
minWidth = Math.Max(buttonWidth, minWidth);
if (shown)
{
var pos = Im.Cursor.Position;
Im.FrameDummy();
using (Im.Group())
{
draw();
}
var width = Math.Max(Im.Item.Size.X, minWidth);
var endPos = Im.Cursor.Position;
Im.Cursor.Position = pos;
if (Im.Button(buttonTextHide, new Vector2(width, 0)))
Im.State.Storage.SetBool(collapseId, !shown);
Im.Cursor.Position = endPos;
}
else
{
var optionWidth = options.Max(o => Im.Font.CalculateSize(o.Name).X)
+ Im.Style.ItemInnerSpacing.X
+ Im.Style.FrameHeight
+ Im.Style.FramePadding.X;
var width = Math.Max(optionWidth, minWidth);
if (Im.Button(buttonTextShow, new Vector2(width, 0)))
Im.State.Storage.SetBool(collapseId, !shown);
}
}
}
private ModCollection Current
=> collectionManager.Active.Current;
[MethodImpl(MethodImplOptions.AggressiveInlining | MethodImplOptions.AggressiveOptimization)]
internal void SetModSetting(IModGroup group, int groupIdx, Setting setting)
{
if (_temporary || config.DefaultTemporaryMode)
{
_tempSettings ??= new TemporaryModSettings(group.Mod, _settings);
_tempSettings!.ForceInherit = false;
_tempSettings!.Settings[groupIdx] = setting;
collectionManager.Editor.SetTemporarySettings(Current, group.Mod, _tempSettings);
}
else
{
collectionManager.Editor.SetModSetting(Current, group.Mod, groupIdx, setting);
}
}
}