mirror of
https://github.com/Facepunch/sbox-public.git
synced 2025-12-23 22:48:07 -05:00
Material palette groups (#3596)
https://files.facepunch.com/louie/1b1511b1/sbox-dev_yO6MOKHQIG.mp4
This commit is contained in:
@@ -1,4 +1,6 @@
|
||||
namespace Editor.MeshEditor;
|
||||
using Sandbox.UI;
|
||||
|
||||
namespace Editor.MeshEditor;
|
||||
|
||||
class ActiveMaterialWidget : ControlWidget
|
||||
{
|
||||
@@ -10,29 +12,26 @@ class ActiveMaterialWidget : ControlWidget
|
||||
public ActiveMaterialWidget( SerializedProperty property ) : base( property )
|
||||
{
|
||||
FixedHeight = 220;
|
||||
|
||||
Layout = Layout.Row();
|
||||
Layout.Margin = 8;
|
||||
|
||||
ToolTip = "";
|
||||
|
||||
_materialWidget = Layout.Add( new MaterialWidget() );
|
||||
_materialWidget.ToolTip = "Active Material";
|
||||
_materialWidget.FixedSize = FixedHeight - 26;
|
||||
_materialWidget.Cursor = CursorShape.Finger;
|
||||
|
||||
Layout.AddStretchCell( 1 );
|
||||
|
||||
_paletteStrip = Layout.Add( new MaterialPaletteWidget() );
|
||||
_paletteStrip.MaterialClicked += OnPaletteMaterialClicked;
|
||||
_paletteStrip.FixedHeight = FixedHeight - 26;
|
||||
_paletteStrip.FixedHeight = FixedHeight - 8;
|
||||
_paletteStrip.FixedWidth = 64;
|
||||
_paletteStrip.GetActiveMaterial = () => _materialWidget.Material;
|
||||
|
||||
Layout.AddStretchCell( 1 );
|
||||
Layout.AddSpacingCell( 1 );
|
||||
|
||||
_materialWidget = Layout.Add( new MaterialWidget() );
|
||||
_materialWidget.ToolTip = "Active Material";
|
||||
_materialWidget.FixedSize = FixedHeight - 22;
|
||||
_materialWidget.Cursor = CursorShape.Finger;
|
||||
|
||||
|
||||
Frame();
|
||||
}
|
||||
|
||||
protected override void OnPaint()
|
||||
{
|
||||
// nothing
|
||||
@@ -99,7 +98,7 @@ class ActiveMaterialWidget : ControlWidget
|
||||
base.OnMouseClick( e );
|
||||
|
||||
// If we are selecting the Material Widget continue. (Probably better way of doing this)
|
||||
if ( !_materialWidget.LocalRect.IsInside( e.LocalPosition ) )
|
||||
if ( _materialWidget.ContentRect.IsInside( e.LocalPosition ) )
|
||||
return;
|
||||
|
||||
if ( ReadOnly ) return;
|
||||
|
||||
@@ -33,13 +33,13 @@ public class MaterialWidget : Widget
|
||||
|
||||
if ( icon is not null )
|
||||
{
|
||||
Paint.Draw( LocalRect.Shrink( 2 ), icon );
|
||||
Paint.Draw( LocalRect, icon );
|
||||
}
|
||||
|
||||
if ( ShowFilename && asset is not null )
|
||||
{
|
||||
Paint.SetDefaultFont( 7 );
|
||||
Theme.DrawFilename( LocalRect.Shrink( 4 ), asset.RelativePath, TextFlag.LeftBottom, Color.White );
|
||||
Theme.DrawFilename( LocalRect, asset.RelativePath, TextFlag.LeftBottom, Color.White );
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
@@ -2,42 +2,281 @@
|
||||
|
||||
public class MaterialPaletteWidget : Widget
|
||||
{
|
||||
const int MaxRecentMaterials = 12;
|
||||
const int RecentColumns = 6;
|
||||
const int MaxCells = 12;
|
||||
const int PaletteColumns = 6;
|
||||
|
||||
readonly List<Material> _recentMaterials = new();
|
||||
readonly RecentMaterialSlotWidget[] _slots;
|
||||
readonly PaletteMaterialSlotWidget[] _slots;
|
||||
|
||||
readonly List<string> _paletteNames = new();
|
||||
string _paletteId = "Default";
|
||||
|
||||
public event Action<Material> MaterialClicked;
|
||||
public Func<Material> GetActiveMaterial { get; set; }
|
||||
|
||||
public string PaletteId
|
||||
{
|
||||
get => _paletteId;
|
||||
set
|
||||
{
|
||||
if ( string.IsNullOrEmpty( value ) || _paletteId == value )
|
||||
return;
|
||||
|
||||
_paletteId = value;
|
||||
|
||||
SaveActivePalette();
|
||||
|
||||
LoadPaletteFromCookie();
|
||||
Update();
|
||||
}
|
||||
}
|
||||
|
||||
public MaterialPaletteWidget()
|
||||
{
|
||||
Layout = Layout.Column();
|
||||
Layout.Margin = 0;
|
||||
Layout.Alignment = TextFlag.Center;
|
||||
|
||||
var grid = Layout.Grid();
|
||||
grid.Spacing = 2;
|
||||
Layout.Add( grid );
|
||||
|
||||
_slots = new RecentMaterialSlotWidget[MaxRecentMaterials];
|
||||
_slots = new PaletteMaterialSlotWidget[MaxCells];
|
||||
|
||||
for ( int i = 0; i < MaxRecentMaterials; i++ )
|
||||
for ( int i = 0; i < MaxCells; i++ )
|
||||
{
|
||||
var col = i / RecentColumns;
|
||||
var row = i % RecentColumns;
|
||||
var col = i / PaletteColumns;
|
||||
var row = i % PaletteColumns;
|
||||
|
||||
var slot = new RecentMaterialSlotWidget( this )
|
||||
var slot = new PaletteMaterialSlotWidget( this )
|
||||
{
|
||||
ShowFilename = false,
|
||||
FixedSize = 32
|
||||
};
|
||||
|
||||
_slots[i] = slot;
|
||||
|
||||
grid.AddCell( col, row, slot );
|
||||
}
|
||||
|
||||
LoadPalettes();
|
||||
LoadPaletteFromCookie();
|
||||
}
|
||||
|
||||
void LoadPalettes()
|
||||
{
|
||||
_paletteNames.Clear();
|
||||
|
||||
string rawNames;
|
||||
try { rawNames = ProjectCookie.Get( "MeshEditor.MaterialPalettes.Names", string.Empty ); }
|
||||
catch { rawNames = string.Empty; }
|
||||
|
||||
if ( string.IsNullOrWhiteSpace( rawNames ) )
|
||||
{
|
||||
_paletteNames.Add( "Default" );
|
||||
}
|
||||
else
|
||||
{
|
||||
foreach ( var name in rawNames.Split( ';', StringSplitOptions.RemoveEmptyEntries | StringSplitOptions.TrimEntries ) )
|
||||
{
|
||||
if ( !_paletteNames.Contains( name ) )
|
||||
_paletteNames.Add( name );
|
||||
}
|
||||
|
||||
if ( _paletteNames.Count == 0 )
|
||||
_paletteNames.Add( "Default" );
|
||||
}
|
||||
|
||||
try { _paletteId = ProjectCookie.Get( "MeshEditor.MaterialPalettes.Active", _paletteNames[0] ); }
|
||||
catch { _paletteId = _paletteNames[0]; }
|
||||
|
||||
if ( !_paletteNames.Contains( _paletteId ) )
|
||||
_paletteId = _paletteNames[0];
|
||||
}
|
||||
|
||||
void SavePalettes()
|
||||
{
|
||||
ProjectCookie.Set( "MeshEditor.MaterialPalettes.Names", string.Join( ";", _paletteNames ) );
|
||||
SaveActivePalette();
|
||||
}
|
||||
|
||||
void SaveActivePalette()
|
||||
{
|
||||
ProjectCookie.Set( "MeshEditor.MaterialPalettes.Active", _paletteId ?? string.Empty );
|
||||
}
|
||||
|
||||
protected override void OnContextMenu( ContextMenuEvent e )
|
||||
{
|
||||
base.OnContextMenu( e );
|
||||
|
||||
var m = new ContextMenu();
|
||||
|
||||
AddPaletteMenu( m );
|
||||
|
||||
m.OpenAtCursor( false );
|
||||
e.Accepted = true;
|
||||
}
|
||||
|
||||
internal void AddPaletteMenu( ContextMenu m )
|
||||
{
|
||||
LoadPalettes();
|
||||
|
||||
var p = m.AddMenu( "Palettes", "palette" );
|
||||
|
||||
foreach ( var name in _paletteNames )
|
||||
{
|
||||
var localName = name;
|
||||
var icon = (localName == _paletteId) ? "check" : "palette";
|
||||
p.AddOption( localName, icon, () => PaletteId = localName );
|
||||
}
|
||||
|
||||
p.AddSeparator();
|
||||
|
||||
p.AddOption( "New Palette…", "add", ShowCreatePalettePopup );
|
||||
p.AddOption( "Rename Palette…", "edit", () => ShowRenamePalettePopup( _paletteId ) ).Enabled = _paletteNames.Count > 0;
|
||||
p.AddOption( "Duplicate Palette", "content_copy", () => DuplicatePalette( _paletteId ) ).Enabled = _paletteNames.Count > 0;
|
||||
|
||||
var del = p.AddOption( "Delete Palette", "delete", () => DeletePalette( _paletteId ) );
|
||||
del.Enabled = _paletteNames.Count > 1;
|
||||
}
|
||||
|
||||
void ShowCreatePalettePopup()
|
||||
{
|
||||
var popup = new PopupWidget( this );
|
||||
popup.FixedWidth = 220;
|
||||
popup.Layout = Layout.Column();
|
||||
popup.Layout.Margin = 8;
|
||||
popup.Layout.Spacing = 4;
|
||||
|
||||
_ = popup.Layout.Add( new Label.Small( "New palette" ) );
|
||||
var entry = popup.Layout.Add( new LineEdit( popup ) );
|
||||
entry.FixedHeight = Theme.RowHeight;
|
||||
entry.PlaceholderText = "Palette name…";
|
||||
|
||||
void Commit()
|
||||
{
|
||||
var name = entry.Value?.Trim();
|
||||
if ( string.IsNullOrEmpty( name ) ) { popup.Destroy(); return; }
|
||||
if ( _paletteNames.Contains( name ) ) { popup.Destroy(); return; }
|
||||
|
||||
_paletteNames.Add( name );
|
||||
_paletteId = name;
|
||||
|
||||
SavePalettes();
|
||||
LoadPaletteFromCookie();
|
||||
|
||||
popup.Destroy();
|
||||
}
|
||||
|
||||
entry.ReturnPressed += Commit;
|
||||
|
||||
popup.OpenAtCursor();
|
||||
entry.Focus();
|
||||
}
|
||||
|
||||
void ShowRenamePalettePopup( string oldName )
|
||||
{
|
||||
if ( string.IsNullOrEmpty( oldName ) )
|
||||
return;
|
||||
|
||||
var popup = new PopupWidget( this );
|
||||
popup.FixedWidth = 220;
|
||||
popup.Layout = Layout.Column();
|
||||
popup.Layout.Margin = 8;
|
||||
popup.Layout.Spacing = 4;
|
||||
|
||||
_ = popup.Layout.Add( new Label.Small( "Rename palette" ) );
|
||||
var entry = popup.Layout.Add( new LineEdit( popup ) );
|
||||
entry.FixedHeight = Theme.RowHeight;
|
||||
entry.Value = oldName;
|
||||
|
||||
void Commit()
|
||||
{
|
||||
var newName = entry.Value?.Trim();
|
||||
if ( string.IsNullOrEmpty( newName ) || newName == oldName ) { popup.Destroy(); return; }
|
||||
if ( _paletteNames.Contains( newName ) ) { popup.Destroy(); return; }
|
||||
|
||||
RenamePalette( oldName, newName );
|
||||
popup.Destroy();
|
||||
}
|
||||
|
||||
entry.ReturnPressed += Commit;
|
||||
|
||||
popup.OpenAtCursor();
|
||||
entry.Focus();
|
||||
}
|
||||
|
||||
void RenamePalette( string oldName, string newName )
|
||||
{
|
||||
var idx = _paletteNames.IndexOf( oldName );
|
||||
if ( idx < 0 ) return;
|
||||
|
||||
_paletteNames[idx] = newName;
|
||||
|
||||
var oldKey = $"MeshEditor.MaterialPalette.{oldName}";
|
||||
var newKey = $"MeshEditor.MaterialPalette.{newName}";
|
||||
|
||||
try
|
||||
{
|
||||
var data = ProjectCookie.Get( oldKey, string.Empty );
|
||||
ProjectCookie.Set( newKey, data );
|
||||
ProjectCookie.Set( oldKey, string.Empty );
|
||||
}
|
||||
catch { }
|
||||
|
||||
if ( _paletteId == oldName )
|
||||
_paletteId = newName;
|
||||
|
||||
SavePalettes();
|
||||
LoadPaletteFromCookie();
|
||||
}
|
||||
|
||||
void DuplicatePalette( string sourceName )
|
||||
{
|
||||
if ( string.IsNullOrEmpty( sourceName ) )
|
||||
return;
|
||||
|
||||
var baseName = $"{sourceName} Copy";
|
||||
var newName = baseName;
|
||||
int counter = 2;
|
||||
|
||||
while ( _paletteNames.Contains( newName ) )
|
||||
newName = $"{baseName} {counter++}";
|
||||
|
||||
_paletteNames.Add( newName );
|
||||
|
||||
var srcKey = $"MeshEditor.MaterialPalette.{sourceName}";
|
||||
var dstKey = $"MeshEditor.MaterialPalette.{newName}";
|
||||
|
||||
try
|
||||
{
|
||||
var data = ProjectCookie.Get( srcKey, string.Empty );
|
||||
ProjectCookie.Set( dstKey, data );
|
||||
}
|
||||
catch { }
|
||||
|
||||
_paletteId = newName;
|
||||
|
||||
SavePalettes();
|
||||
LoadPaletteFromCookie();
|
||||
}
|
||||
|
||||
void DeletePalette( string name )
|
||||
{
|
||||
if ( _paletteNames.Count <= 1 )
|
||||
return;
|
||||
|
||||
var idx = _paletteNames.IndexOf( name );
|
||||
if ( idx < 0 )
|
||||
return;
|
||||
|
||||
_paletteNames.RemoveAt( idx );
|
||||
|
||||
var key = $"MeshEditor.MaterialPalette.{name}";
|
||||
try { ProjectCookie.Set( key, string.Empty ); }
|
||||
catch { }
|
||||
|
||||
_paletteId = _paletteNames[Math.Clamp( idx - 1, 0, _paletteNames.Count - 1 )];
|
||||
|
||||
SavePalettes();
|
||||
LoadPaletteFromCookie();
|
||||
}
|
||||
|
||||
@@ -48,20 +287,14 @@ public class MaterialPaletteWidget : Widget
|
||||
var path = material.ResourcePath;
|
||||
|
||||
if ( !string.IsNullOrEmpty( path ) )
|
||||
{
|
||||
_recentMaterials.RemoveAll( m => m is not null && m.ResourcePath == path );
|
||||
}
|
||||
else
|
||||
{
|
||||
_recentMaterials.RemoveAll( m => m == material );
|
||||
}
|
||||
|
||||
_recentMaterials.Insert( 0, material );
|
||||
|
||||
if ( _recentMaterials.Count > MaxRecentMaterials )
|
||||
{
|
||||
if ( _recentMaterials.Count > _slots.Length )
|
||||
_recentMaterials.RemoveAt( _recentMaterials.Count - 1 );
|
||||
}
|
||||
|
||||
UpdateSlots();
|
||||
SavePaletteToCookie();
|
||||
@@ -70,16 +303,7 @@ public class MaterialPaletteWidget : Widget
|
||||
void UpdateSlots()
|
||||
{
|
||||
for ( int i = 0; i < _slots.Length; i++ )
|
||||
{
|
||||
if ( i < _recentMaterials.Count )
|
||||
{
|
||||
_slots[i].Material = _recentMaterials[i];
|
||||
}
|
||||
else
|
||||
{
|
||||
_slots[i].Material = null;
|
||||
}
|
||||
}
|
||||
_slots[i].Material = i < _recentMaterials.Count ? _recentMaterials[i] : null;
|
||||
}
|
||||
|
||||
internal void SlotClickedApply( Material material )
|
||||
@@ -88,7 +312,7 @@ public class MaterialPaletteWidget : Widget
|
||||
MaterialClicked?.Invoke( material );
|
||||
}
|
||||
|
||||
private void SlotSetMaterial( RecentMaterialSlotWidget slot, Material mat )
|
||||
private void SlotSetMaterial( PaletteMaterialSlotWidget slot, Material mat )
|
||||
{
|
||||
if ( slot is null ) return;
|
||||
|
||||
@@ -106,21 +330,18 @@ public class MaterialPaletteWidget : Widget
|
||||
SavePaletteToCookie();
|
||||
}
|
||||
|
||||
private void SlotAssignFromActive( RecentMaterialSlotWidget slot )
|
||||
private void SlotAssignFromActive( PaletteMaterialSlotWidget slot )
|
||||
{
|
||||
if ( GetActiveMaterial is null )
|
||||
return;
|
||||
if ( GetActiveMaterial is null ) return;
|
||||
|
||||
var mat = GetActiveMaterial();
|
||||
if ( mat is null )
|
||||
return;
|
||||
if ( mat is null ) return;
|
||||
|
||||
SlotSetMaterial( slot, mat );
|
||||
}
|
||||
|
||||
private void SlotAssignMaterial( RecentMaterialSlotWidget slot )
|
||||
private void SlotAssignMaterial( PaletteMaterialSlotWidget slot )
|
||||
{
|
||||
// Open a picker just for materials and assign the result to this slot.
|
||||
var picker = AssetPicker.Create( null, AssetType.Material, new AssetPicker.PickerOptions()
|
||||
{
|
||||
EnableMultiselect = false
|
||||
@@ -142,10 +363,7 @@ public class MaterialPaletteWidget : Widget
|
||||
picker.Show();
|
||||
}
|
||||
|
||||
private void SlotClear( RecentMaterialSlotWidget slot )
|
||||
{
|
||||
SlotSetMaterial( slot, null );
|
||||
}
|
||||
private void SlotClear( PaletteMaterialSlotWidget slot ) => SlotSetMaterial( slot, null );
|
||||
|
||||
void SavePaletteToCookie()
|
||||
{
|
||||
@@ -159,24 +377,14 @@ public class MaterialPaletteWidget : Widget
|
||||
.Take( _slots.Length )
|
||||
.Select( m => m is not null ? m.ResourcePath ?? string.Empty : string.Empty );
|
||||
|
||||
var data = string.Join( ";", parts );
|
||||
|
||||
// Maybe this should be scene specific?
|
||||
ProjectCookie.Set( "MeshEditor.MaterialPalette", data );
|
||||
ProjectCookie.Set( $"MeshEditor.MaterialPalette.{_paletteId}", string.Join( ";", parts ) );
|
||||
}
|
||||
|
||||
void LoadPaletteFromCookie()
|
||||
{
|
||||
string data;
|
||||
|
||||
try
|
||||
{
|
||||
data = ProjectCookie.Get( "MeshEditor.MaterialPalette", string.Empty );
|
||||
}
|
||||
catch
|
||||
{
|
||||
data = string.Empty;
|
||||
}
|
||||
try { data = ProjectCookie.Get( $"MeshEditor.MaterialPalette.{_paletteId}", string.Empty ); }
|
||||
catch { data = string.Empty; }
|
||||
|
||||
_recentMaterials.Clear();
|
||||
|
||||
@@ -188,7 +396,7 @@ public class MaterialPaletteWidget : Widget
|
||||
|
||||
var parts = data.Split( ';' );
|
||||
|
||||
for ( int i = 0; i < MaxRecentMaterials; i++ )
|
||||
for ( int i = 0; i < _slots.Length; i++ )
|
||||
{
|
||||
if ( i >= parts.Length || string.IsNullOrWhiteSpace( parts[i] ) )
|
||||
{
|
||||
@@ -204,23 +412,22 @@ public class MaterialPaletteWidget : Widget
|
||||
continue;
|
||||
}
|
||||
|
||||
var mat = asset.LoadResource( typeof( Material ) ) as Material;
|
||||
_recentMaterials.Add( mat );
|
||||
_recentMaterials.Add( asset.LoadResource( typeof( Material ) ) as Material );
|
||||
}
|
||||
|
||||
UpdateSlots();
|
||||
}
|
||||
|
||||
class RecentMaterialSlotWidget : MaterialWidget
|
||||
class PaletteMaterialSlotWidget : MaterialWidget
|
||||
{
|
||||
readonly MaterialPaletteWidget _strip;
|
||||
bool _isDownloading;
|
||||
bool _isValidDropHover;
|
||||
public RecentMaterialSlotWidget( MaterialPaletteWidget strip )
|
||||
|
||||
public PaletteMaterialSlotWidget( MaterialPaletteWidget strip )
|
||||
{
|
||||
_strip = strip;
|
||||
ToolTip = "";
|
||||
|
||||
AcceptDrops = true;
|
||||
Cursor = CursorShape.Finger;
|
||||
}
|
||||
@@ -229,26 +436,17 @@ public class MaterialPaletteWidget : Widget
|
||||
{
|
||||
base.OnMouseClick( e );
|
||||
|
||||
if ( Material is not null )
|
||||
{
|
||||
_strip.SlotClickedApply( Material );
|
||||
}
|
||||
else
|
||||
{
|
||||
_strip.SlotAssignFromActive( this );
|
||||
}
|
||||
if ( Material.IsValid() ) _strip.SlotClickedApply( Material );
|
||||
else _strip.SlotAssignFromActive( this );
|
||||
}
|
||||
|
||||
protected override void OnContextMenu( ContextMenuEvent e )
|
||||
{
|
||||
var m = new ContextMenu();
|
||||
bool hasMaterial = Material is not null;
|
||||
bool hasMaterial = Material.IsValid();
|
||||
|
||||
var text = hasMaterial ? "Change Material" : "Set Material";
|
||||
m.AddOption( text, "format_color_fill", () =>
|
||||
{
|
||||
_strip.SlotAssignMaterial( this );
|
||||
} );
|
||||
m.AddOption( text, "format_color_fill", () => _strip.SlotAssignMaterial( this ) );
|
||||
|
||||
m.AddSeparator();
|
||||
|
||||
@@ -263,10 +461,10 @@ public class MaterialPaletteWidget : Widget
|
||||
}
|
||||
}
|
||||
|
||||
m.AddOption( "Clear", "backspace", () =>
|
||||
{
|
||||
_strip.SlotClear( this );
|
||||
} ).Enabled = hasMaterial;
|
||||
_strip.AddPaletteMenu( m );
|
||||
m.AddSeparator();
|
||||
|
||||
m.AddOption( "Clear", "backspace", () => _strip.SlotClear( this ) ).Enabled = hasMaterial;
|
||||
|
||||
m.OpenAtCursor( false );
|
||||
e.Accepted = true;
|
||||
@@ -300,18 +498,25 @@ public class MaterialPaletteWidget : Widget
|
||||
Paint.SetBrushAndPen( Color.Transparent, Color.White );
|
||||
Paint.DrawRect( controlRect, 0 );
|
||||
}
|
||||
|
||||
}
|
||||
else
|
||||
{
|
||||
var baseFill = Theme.Text.WithAlpha( 0.01f );
|
||||
var baseLine = Theme.Text.WithAlpha( 0.1f );
|
||||
var iconColor = Theme.Text.WithAlpha( 0.1f );
|
||||
var baseFill = Theme.ControlBackground;
|
||||
var baseLine = Color.Transparent;
|
||||
var iconColor = Theme.TextLight;
|
||||
|
||||
if ( Paint.HasMouseOver )
|
||||
{
|
||||
baseFill = Theme.Text.WithAlpha( 0.04f );
|
||||
baseLine = Theme.Text.WithAlpha( 0.2f );
|
||||
iconColor = Theme.Text.WithAlpha( 0.2f );
|
||||
baseFill = Theme.ControlBackground;
|
||||
baseLine = Color.Transparent;
|
||||
iconColor = Theme.TextLight.Lighten( 0.8f );
|
||||
}
|
||||
else
|
||||
{
|
||||
baseFill = Theme.ControlBackground;
|
||||
baseLine = Color.Transparent;
|
||||
iconColor = Theme.TextLight;
|
||||
}
|
||||
|
||||
if ( _isValidDropHover )
|
||||
@@ -435,6 +640,9 @@ public class MaterialPaletteWidget : Widget
|
||||
|
||||
_strip.SlotSetMaterial( this, droppedMaterial );
|
||||
ev.Action = DropAction.Link;
|
||||
|
||||
_isValidDropHover = false;
|
||||
Update();
|
||||
}
|
||||
|
||||
async Task AssignFromUrlAsync( string identUrl )
|
||||
@@ -452,20 +660,7 @@ public class MaterialPaletteWidget : Widget
|
||||
if ( mat is null )
|
||||
return;
|
||||
|
||||
Material = mat;
|
||||
|
||||
var index = Array.IndexOf( _strip._slots, this );
|
||||
if ( index >= 0 )
|
||||
{
|
||||
if ( index >= _strip._recentMaterials.Count )
|
||||
{
|
||||
while ( _strip._recentMaterials.Count <= index )
|
||||
_strip._recentMaterials.Add( null );
|
||||
}
|
||||
|
||||
_strip._recentMaterials[index] = mat;
|
||||
_strip.SavePaletteToCookie();
|
||||
}
|
||||
_strip.SlotSetMaterial( this, mat );
|
||||
}
|
||||
finally
|
||||
{
|
||||
@@ -491,7 +686,6 @@ public class MaterialPaletteWidget : Widget
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
file class TextureTooltip : Widget
|
||||
{
|
||||
Widget target;
|
||||
|
||||
Reference in New Issue
Block a user