csharpc-sharpdotnetxamlavaloniauicross-platformcross-platform-xamlavaloniaguimulti-platformuser-interfacedotnetcore
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
418 lines
14 KiB
418 lines
14 KiB
using System.Linq;
|
|
using Avalonia.Controls;
|
|
using Avalonia.Controls.Primitives;
|
|
using Avalonia.Controls.Templates;
|
|
using Avalonia.Media;
|
|
using Avalonia.Styling;
|
|
using Avalonia.UnitTests;
|
|
using Avalonia.VisualTree;
|
|
using Xunit;
|
|
|
|
#nullable enable
|
|
|
|
namespace Avalonia.Base.UnitTests.Styling;
|
|
|
|
public class StyledElementTests_Theming
|
|
{
|
|
public class InlineTheme
|
|
{
|
|
[Fact]
|
|
public void Theme_Is_Applied_When_Attached_To_Logical_Tree()
|
|
{
|
|
using var app = UnitTestApplication.Start(TestServices.RealStyler);
|
|
var target = CreateTarget();
|
|
|
|
Assert.Null(target.Template);
|
|
|
|
var root = CreateRoot(target);
|
|
Assert.NotNull(target.Template);
|
|
|
|
var border = Assert.IsType<Border>(target.VisualChild);
|
|
Assert.Equal(Brushes.Red, border.Background);
|
|
|
|
target.Classes.Add("foo");
|
|
Assert.Equal(Brushes.Green, border.Background);
|
|
}
|
|
|
|
[Fact]
|
|
public void Theme_Is_Applied_To_Derived_Class_When_Attached_To_Logical_Tree()
|
|
{
|
|
using var app = UnitTestApplication.Start(TestServices.RealStyler);
|
|
var target = new DerivedThemedControl
|
|
{
|
|
Theme = CreateTheme(),
|
|
};
|
|
|
|
Assert.Null(target.Template);
|
|
|
|
var root = CreateRoot(target);
|
|
Assert.NotNull(target.Template);
|
|
|
|
var border = Assert.IsType<Border>(target.VisualChild);
|
|
Assert.Equal(Brushes.Red, border.Background);
|
|
|
|
target.Classes.Add("foo");
|
|
Assert.Equal(Brushes.Green, border.Background);
|
|
}
|
|
|
|
[Fact]
|
|
public void Theme_Is_Detached_When_Theme_Property_Cleared()
|
|
{
|
|
using var app = UnitTestApplication.Start(TestServices.RealStyler);
|
|
var target = CreateTarget();
|
|
var root = CreateRoot(target);
|
|
|
|
Assert.NotNull(target.Template);
|
|
|
|
target.Theme = null;
|
|
Assert.Null(target.Template);
|
|
}
|
|
|
|
[Fact]
|
|
public void Theme_Is_Detached_From_Template_Controls_When_Theme_Property_Cleared()
|
|
{
|
|
using var app = UnitTestApplication.Start(TestServices.RealStyler);
|
|
|
|
var theme = new ControlTheme
|
|
{
|
|
TargetType = typeof(ThemedControl),
|
|
Children =
|
|
{
|
|
new Style(x => x.Nesting().Template().OfType<Canvas>())
|
|
{
|
|
Setters =
|
|
{
|
|
new Setter(Panel.BackgroundProperty, Brushes.Red),
|
|
}
|
|
},
|
|
}
|
|
};
|
|
|
|
var target = CreateTarget(theme);
|
|
target.Template = new FuncControlTemplate<ThemedControl>((o, n) => new Canvas());
|
|
|
|
var root = CreateRoot(target);
|
|
|
|
var canvas = Assert.IsType<Canvas>(target.VisualChild);
|
|
Assert.Equal(Brushes.Red, canvas.Background);
|
|
|
|
target.Theme = null;
|
|
|
|
Assert.IsType<Canvas>(target.VisualChild);
|
|
Assert.Null(canvas.Background);
|
|
}
|
|
|
|
[Fact]
|
|
public void Theme_Is_Applied_On_Layout_After_Theme_Property_Changes()
|
|
{
|
|
using var app = UnitTestApplication.Start(TestServices.RealStyler);
|
|
var target = new ThemedControl();
|
|
var root = CreateRoot(target);
|
|
|
|
Assert.Null(target.Template);
|
|
|
|
target.Theme = CreateTheme();
|
|
Assert.Null(target.Template);
|
|
|
|
root.LayoutManager.ExecuteLayoutPass();
|
|
|
|
var border = Assert.IsType<Border>(target.VisualChild);
|
|
Assert.NotNull(target.Template);
|
|
Assert.Equal(Brushes.Red, border.Background);
|
|
}
|
|
|
|
[Fact]
|
|
public void BasedOn_Theme_Is_Applied_When_Attached_To_Logical_Tree()
|
|
{
|
|
using var app = UnitTestApplication.Start(TestServices.RealStyler);
|
|
var target = CreateTarget(CreateDerivedTheme());
|
|
|
|
Assert.Null(target.Template);
|
|
|
|
var root = CreateRoot(target);
|
|
Assert.NotNull(target.Template);
|
|
Assert.Equal(Brushes.Blue, target.BorderBrush);
|
|
|
|
var border = Assert.IsType<Border>(target.VisualChild);
|
|
Assert.Equal(Brushes.Red, border.Background);
|
|
Assert.Equal(Brushes.Yellow, border.BorderBrush);
|
|
|
|
target.Classes.Add("foo");
|
|
Assert.Equal(Brushes.Green, border.Background);
|
|
Assert.Equal(Brushes.Cyan, border.BorderBrush);
|
|
}
|
|
|
|
private static ThemedControl CreateTarget(ControlTheme? theme = null)
|
|
{
|
|
return new ThemedControl
|
|
{
|
|
Theme = theme ?? CreateTheme(),
|
|
};
|
|
}
|
|
|
|
private static TestRoot CreateRoot(IControl child)
|
|
{
|
|
var result = new TestRoot(child);
|
|
result.LayoutManager.ExecuteInitialLayoutPass();
|
|
return result;
|
|
}
|
|
}
|
|
|
|
public class ImplicitTheme
|
|
{
|
|
[Fact]
|
|
public void Implicit_Theme_Is_Applied_When_Attached_To_Logical_Tree()
|
|
{
|
|
using var app = UnitTestApplication.Start(TestServices.RealStyler);
|
|
var target = CreateTarget();
|
|
var root = CreateRoot(target);
|
|
Assert.NotNull(target.Template);
|
|
|
|
var border = Assert.IsType<Border>(target.VisualChild);
|
|
Assert.Equal(Brushes.Red, border.Background);
|
|
|
|
target.Classes.Add("foo");
|
|
Assert.Equal(Brushes.Green, border.Background);
|
|
}
|
|
|
|
[Fact]
|
|
public void Implicit_Theme_Is_Cleared_When_Removed_From_Logical_Tree()
|
|
{
|
|
using var app = UnitTestApplication.Start(TestServices.RealStyler);
|
|
var target = CreateTarget();
|
|
var root = CreateRoot(target);
|
|
|
|
Assert.NotNull(((IStyleable)target).GetEffectiveTheme());
|
|
|
|
root.Child = null;
|
|
|
|
Assert.Null(((IStyleable)target).GetEffectiveTheme());
|
|
}
|
|
|
|
[Fact]
|
|
public void Nested_Style_Can_Override_Property_In_Inner_Templated_Control()
|
|
{
|
|
using var app = UnitTestApplication.Start(TestServices.RealStyler);
|
|
var target = new ThemedControl2
|
|
{
|
|
Theme = new ControlTheme(typeof(ThemedControl2))
|
|
{
|
|
Setters =
|
|
{
|
|
new Setter(
|
|
TemplatedControl.TemplateProperty,
|
|
new FuncControlTemplate<ThemedControl2>((o, n) => new ThemedControl())),
|
|
},
|
|
Children =
|
|
{
|
|
new Style(x => x.Nesting().Template().OfType<ThemedControl>())
|
|
{
|
|
Setters = { new Setter(TemplatedControl.CornerRadiusProperty, new CornerRadius(7)), }
|
|
},
|
|
}
|
|
},
|
|
};
|
|
|
|
var root = CreateRoot(target);
|
|
var inner = Assert.IsType<ThemedControl>(target.VisualChild);
|
|
|
|
Assert.Equal(new CornerRadius(7), inner.CornerRadius);
|
|
}
|
|
|
|
private static ThemedControl CreateTarget() => new ThemedControl();
|
|
|
|
private static TestRoot CreateRoot(IControl child)
|
|
{
|
|
var result = new TestRoot();
|
|
result.Resources.Add(typeof(ThemedControl), CreateTheme());
|
|
result.Child = child;
|
|
result.LayoutManager.ExecuteInitialLayoutPass();
|
|
return result;
|
|
}
|
|
}
|
|
|
|
public class ThemeFromStyle
|
|
{
|
|
[Fact]
|
|
public void Theme_Is_Applied_When_Attached_To_Logical_Tree()
|
|
{
|
|
using var app = UnitTestApplication.Start(TestServices.RealStyler);
|
|
var target = CreateTarget();
|
|
|
|
Assert.Null(target.Theme);
|
|
Assert.Null(target.Template);
|
|
|
|
var root = CreateRoot(target);
|
|
|
|
Assert.NotNull(target.Theme);
|
|
Assert.NotNull(target.Template);
|
|
|
|
var border = Assert.IsType<Border>(target.VisualChild);
|
|
Assert.Equal(border.Background, Brushes.Red);
|
|
|
|
target.Classes.Add("foo");
|
|
Assert.Equal(border.Background, Brushes.Green);
|
|
}
|
|
|
|
[Fact]
|
|
public void Theme_Can_Be_Changed_By_Style_Class()
|
|
{
|
|
using var app = UnitTestApplication.Start(TestServices.RealStyler);
|
|
var target = CreateTarget();
|
|
var theme1 = CreateTheme();
|
|
var theme2 = new ControlTheme(typeof(ThemedControl));
|
|
var root = new TestRoot()
|
|
{
|
|
Styles =
|
|
{
|
|
new Style(x => x.OfType<ThemedControl>())
|
|
{
|
|
Setters = { new Setter(StyledElement.ThemeProperty, theme1) }
|
|
},
|
|
new Style(x => x.OfType<ThemedControl>().Class("bar"))
|
|
{
|
|
Setters = { new Setter(StyledElement.ThemeProperty, theme2) }
|
|
},
|
|
}
|
|
};
|
|
|
|
root.Child = target;
|
|
root.LayoutManager.ExecuteInitialLayoutPass();
|
|
|
|
Assert.Same(theme1, target.Theme);
|
|
Assert.NotNull(target.Template);
|
|
|
|
target.Classes.Add("bar");
|
|
Assert.Same(theme2, target.Theme);
|
|
Assert.Null(target.Template);
|
|
}
|
|
|
|
[Fact]
|
|
public void Theme_Can_Be_Set_To_LocalValue_While_Updating_Due_To_Style_Class()
|
|
{
|
|
using var app = UnitTestApplication.Start(TestServices.RealStyler);
|
|
var target = CreateTarget();
|
|
var theme1 = CreateTheme();
|
|
var theme2 = new ControlTheme(typeof(ThemedControl));
|
|
var theme3 = new ControlTheme(typeof(ThemedControl));
|
|
var root = new TestRoot()
|
|
{
|
|
Styles =
|
|
{
|
|
new Style(x => x.OfType<ThemedControl>())
|
|
{
|
|
Setters = { new Setter(StyledElement.ThemeProperty, theme1) }
|
|
},
|
|
new Style(x => x.OfType<ThemedControl>().Class("bar"))
|
|
{
|
|
Setters = { new Setter(StyledElement.ThemeProperty, theme2) }
|
|
},
|
|
}
|
|
};
|
|
|
|
root.Child = target;
|
|
root.LayoutManager.ExecuteInitialLayoutPass();
|
|
|
|
Assert.Same(theme1, target.Theme);
|
|
Assert.NotNull(target.Template);
|
|
|
|
target.Classes.Add("bar");
|
|
|
|
// At this point, theme2 has been promoted to a local value internally in StyledElement;
|
|
// make sure that setting a new local value here doesn't cause it to be cleared when we
|
|
// do a layout pass because StyledElement thinks its clearing the promoted theme.
|
|
target.Theme = theme3;
|
|
|
|
root.LayoutManager.ExecuteLayoutPass();
|
|
|
|
Assert.Same(target.Theme, theme3);
|
|
}
|
|
|
|
private static ThemedControl CreateTarget()
|
|
{
|
|
return new ThemedControl();
|
|
}
|
|
|
|
private static TestRoot CreateRoot(IControl child)
|
|
{
|
|
var result = new TestRoot()
|
|
{
|
|
Styles =
|
|
{
|
|
new Style(x => x.OfType<ThemedControl>())
|
|
{
|
|
Setters = { new Setter(StyledElement.ThemeProperty, CreateTheme()) }
|
|
}
|
|
}
|
|
};
|
|
|
|
result.Child = child;
|
|
result.LayoutManager.ExecuteInitialLayoutPass();
|
|
return result;
|
|
}
|
|
}
|
|
|
|
private static ControlTheme CreateTheme()
|
|
{
|
|
var template = new FuncControlTemplate<ThemedControl>((o, n) => new Border());
|
|
|
|
return new ControlTheme
|
|
{
|
|
TargetType = typeof(ThemedControl),
|
|
Setters =
|
|
{
|
|
new Setter(TemplatedControl.TemplateProperty, template),
|
|
new Setter(TemplatedControl.CornerRadiusProperty, new CornerRadius(5)),
|
|
},
|
|
Children =
|
|
{
|
|
new Style(x => x.Nesting().Template().OfType<Border>())
|
|
{
|
|
Setters = { new Setter(Border.BackgroundProperty, Brushes.Red) }
|
|
},
|
|
new Style(x => x.Nesting().Class("foo").Template().OfType<Border>())
|
|
{
|
|
Setters = { new Setter(Border.BackgroundProperty, Brushes.Green) }
|
|
},
|
|
}
|
|
};
|
|
}
|
|
|
|
private static ControlTheme CreateDerivedTheme()
|
|
{
|
|
return new ControlTheme
|
|
{
|
|
TargetType = typeof(ThemedControl),
|
|
BasedOn = CreateTheme(),
|
|
Setters =
|
|
{
|
|
new Setter(Border.BorderBrushProperty, Brushes.Blue),
|
|
},
|
|
Children =
|
|
{
|
|
new Style(x => x.Nesting().Template().OfType<Border>())
|
|
{
|
|
Setters = { new Setter(Border.BorderBrushProperty, Brushes.Yellow) }
|
|
},
|
|
new Style(x => x.Nesting().Class("foo").Template().OfType<Border>())
|
|
{
|
|
Setters = { new Setter(Border.BorderBrushProperty, Brushes.Cyan) }
|
|
},
|
|
}
|
|
};
|
|
}
|
|
|
|
private class ThemedControl : TemplatedControl
|
|
{
|
|
public IVisual? VisualChild => VisualChildren?.SingleOrDefault();
|
|
}
|
|
|
|
private class ThemedControl2 : TemplatedControl
|
|
{
|
|
public IVisual? VisualChild => VisualChildren?.SingleOrDefault();
|
|
}
|
|
|
|
private class DerivedThemedControl : ThemedControl
|
|
{
|
|
}
|
|
}
|
|
|