Files
sbox-public/engine/Sandbox.Tools/ControlWidget/EnumControlWidget.cs
s&box team 71f266059a Open source release
This commit imports the C# engine code and game files, excluding C++ source code.

[Source-Commit: ceb3d758046e50faa6258bc3b658a30c97743268]
2025-11-24 09:05:18 +00:00

256 lines
5.8 KiB
C#

using System;
namespace Editor;
[CustomEditor( typeof( Enum ) )]
public class EnumControlWidget : ControlWidget
{
/// <summary>
/// If true, then this control is operating in flags mode (FlagsAttribute)
/// </summary>
public bool IsFlagsMode { get; init; }
EnumDescription enumDesc;
public override bool IsControlActive => base.IsControlActive || _menu.IsValid();
public override bool IsControlButton => true;
public override bool IsControlHovered => base.IsControlHovered || _menu.IsValid();
public override bool SupportsMultiEdit => true;
protected virtual float? MenuWidthOverride => null;
public EnumControlWidget( SerializedProperty property ) : base( property )
{
var propertyType = property.NullableType ?? property.PropertyType;
var typeDesc = EditorTypeLibrary.GetType( propertyType );
if ( typeDesc is null )
{
Log.Warning( $"Couldn't create an enum editor for {propertyType} - it's not in EditorTypeLibrary" );
return;
}
Cursor = CursorShape.Finger;
IsFlagsMode = property.HasAttribute<FlagsAttribute>() || typeDesc.HasAttribute<FlagsAttribute>();
Layout = Layout.Row();
Layout.Spacing = 2;
enumDesc = EditorTypeLibrary.GetEnumDescription( propertyType );
}
protected override void PaintControl()
{
if ( enumDesc is null )
return;
var value = SerializedProperty.GetValue<long>( 0 );
var color = IsControlHovered ? Theme.Blue : Theme.TextControl;
if ( IsControlDisabled ) color = color.WithAlpha( 0.5f );
var rect = LocalRect;
rect = rect.Shrink( 8, 0 );
if ( SerializedProperty.IsMultipleDifferentValues )
{
Paint.SetPen( Theme.MultipleValues );
Paint.DrawText( rect, $"Multiple Values", TextFlag.LeftCenter );
}
else if ( IsFlagsMode )
{
var e = enumDesc.GetEntries( value );
var str = string.Join( ", ", e.Select( x => $"{x.Name}" ) );
Paint.SetPen( color );
Paint.DrawText( rect, str, TextFlag.LeftCenter );
}
else
{
var e = enumDesc.GetEntry( value );
if ( !string.IsNullOrEmpty( e.Icon ) )
{
Paint.SetPen( color.WithAlpha( 0.5f ) );
var i = Paint.DrawIcon( rect, e.Icon, 16, TextFlag.LeftCenter );
rect.Left += i.Width + 8;
}
Paint.SetPen( color );
Paint.DrawText( rect, e.Title ?? "Unset", TextFlag.LeftCenter );
}
Paint.SetPen( color );
Paint.DrawIcon( rect, "Arrow_Drop_Down", 17, TextFlag.RightCenter );
}
PopupWidget _menu;
public override void StartEditing()
{
if ( IsControlDisabled ) return;
if ( !_menu.IsValid )
{
OpenMenu();
}
}
protected override void OnMouseClick( MouseEvent e )
{
if ( IsControlDisabled ) return;
if ( e.LeftMouseButton && !_menu.IsValid() )
{
OpenMenu();
}
}
protected override void OnDoubleClick( MouseEvent e )
{
// nothing
}
void ToggleValue( EnumDescription.Entry e )
{
var value = SerializedProperty.GetValue<long>( 0 );
if ( IsFlagsMode )
{
if ( (value & e.IntegerValue) != 0 )
{
value &= ~e.IntegerValue;
}
else
{
value |= e.IntegerValue;
}
}
else
{
value = e.IntegerValue;
}
SerializedProperty.SetValue( value );
}
void OpenMenu()
{
PropertyStartEdit();
if ( enumDesc is null )
return;
_menu = new PopupWidget( null );
_menu.Layout = Layout.Column();
var menuWidth = MenuWidthOverride ?? ScreenRect.Width;
_menu.MinimumWidth = menuWidth;
_menu.MaximumWidth = menuWidth;
_menu.OnLostFocus += PropertyFinishEdit;
var scroller = _menu.Layout.Add( new ScrollArea( this ), 1 );
scroller.Canvas = new Widget( scroller )
{
Layout = Layout.Column(),
VerticalSizeMode = SizeMode.CanGrow | SizeMode.Expand,
MaximumWidth = menuWidth
};
var entries = IsFlagsMode ? enumDesc.AsEnumerable() : enumDesc;
foreach ( var o in entries )
{
if ( !o.Browsable )
continue;
var b = scroller.Canvas.Layout.Add( new MenuOption( o, SerializedProperty, IsFlagsMode ) );
b.MouseLeftPress = () =>
{
ToggleValue( o );
_menu.Update();
if ( !IsFlagsMode )
{
_menu.Close();
}
};
}
_menu.Position = ScreenRect.BottomLeft;
_menu.Visible = true;
_menu.AdjustSize();
_menu.ConstrainToScreen();
_menu.OnPaintOverride = PaintMenuBackground;
if ( scroller.VerticalScrollbar.Minimum != scroller.VerticalScrollbar.Maximum )
{
scroller.Canvas.MaximumWidth -= 8; // leave some space for the scrollbar
}
}
bool PaintMenuBackground()
{
Paint.SetBrushAndPen( Theme.ControlBackground );
Paint.DrawRect( Paint.LocalRect, 0 );
return true;
}
}
file class MenuOption : Widget
{
EnumDescription.Entry info;
SerializedProperty property;
bool flagMode;
public MenuOption( EnumDescription.Entry e, SerializedProperty p, bool flags ) : base( null )
{
info = e;
property = p;
flagMode = flags;
Layout = Layout.Row();
Layout.Margin = 8;
VerticalSizeMode = SizeMode.CanGrow;
if ( !string.IsNullOrWhiteSpace( e.Icon ) )
{
Layout.Add( new IconButton( e.Icon ) { Background = Color.Transparent, TransparentForMouseEvents = true, IconSize = 18 } );
}
Layout.AddSpacingCell( 8 );
var c = Layout.AddColumn();
var title = c.Add( new Label( e.Title ) );
title.SetStyles( $"font-size: 12px; font-weight: bold; font-family: {Theme.DefaultFont}; color: white;" );
if ( !string.IsNullOrWhiteSpace( e.Description ) )
{
var desc = c.Add( new Label( e.Description.Trim( '\n', '\r', '\t', ' ' ) ) );
desc.WordWrap = true;
desc.MinimumHeight = 1;
desc.VerticalSizeMode = SizeMode.CanGrow;
}
}
bool HasValue()
{
var value = property.GetValue<long>( 0 );
if ( flagMode ) return (value & info.IntegerValue) == info.IntegerValue;
return value == info.IntegerValue;
}
protected override void OnPaint()
{
if ( Paint.HasMouseOver || HasValue() )
{
Paint.SetBrushAndPen( Theme.Blue.WithAlpha( HasValue() ? 0.3f : 0.1f ) );
Paint.DrawRect( LocalRect.Shrink( 2 ), 2 );
}
}
}