Browse Source

Merge pull request #1135 from AvaloniaUI/lazy-initialize

Lazily initialize some things
pull/1126/merge
Nikita Tsukanov 9 years ago
committed by GitHub
parent
commit
c53f0e9786
  1. 15
      src/Avalonia.Controls/Application.cs
  2. 22
      src/Avalonia.Controls/Control.cs
  3. 14
      src/Avalonia.Controls/IControl.cs
  4. 6
      src/Avalonia.Controls/IGlobalDataTemplates.cs
  5. 24
      src/Avalonia.Controls/Templates/DataTemplateExtensions.cs
  6. 27
      src/Avalonia.Controls/Templates/IDataTemplateHost.cs
  7. 2
      src/Avalonia.Diagnostics/DevTools.xaml.cs
  8. 2
      src/Avalonia.Diagnostics/Views/ControlDetailsView.cs
  9. 20
      src/Avalonia.Interactivity/Interactive.cs
  10. 12
      src/Avalonia.Styling/Styling/IStyleHost.cs
  11. 11
      src/Avalonia.Styling/Styling/StyleExtensions.cs
  12. 5
      src/Avalonia.Styling/Styling/Styler.cs
  13. 2
      tests/Avalonia.Controls.UnitTests/ItemsControlTests.cs
  14. 8
      tests/Avalonia.Controls.UnitTests/ListBoxTests.cs
  15. 2
      tests/Avalonia.Controls.UnitTests/Presenters/ContentPresenterTests_Unrooted.cs
  16. 1
      tests/Avalonia.Controls.UnitTests/Primitives/PopupTests.cs
  17. 8
      tests/Avalonia.Controls.UnitTests/Primitives/TemplatedControlTests.cs
  18. 4
      tests/Avalonia.Controls.UnitTests/TabControlTests.cs
  19. 28
      tests/Avalonia.Controls.UnitTests/TreeViewTests.cs
  20. 2
      tests/Avalonia.Controls.UnitTests/UserControlTests.cs
  21. 1
      tests/Avalonia.Layout.UnitTests/FullLayoutTests.cs
  22. 2
      tests/Avalonia.LeakTests/ControlTests.cs
  23. 20
      tests/Avalonia.Styling.UnitTests/ResourceTests.cs

15
src/Avalonia.Controls/Application.cs

@ -39,6 +39,7 @@ namespace Avalonia
private readonly Lazy<IClipboard> _clipboard = private readonly Lazy<IClipboard> _clipboard =
new Lazy<IClipboard>(() => (IClipboard)AvaloniaLocator.Current.GetService(typeof(IClipboard))); new Lazy<IClipboard>(() => (IClipboard)AvaloniaLocator.Current.GetService(typeof(IClipboard)));
private readonly Styler _styler = new Styler(); private readonly Styler _styler = new Styler();
private Styles _styles;
/// <summary> /// <summary>
/// Initializes a new instance of the <see cref="Application"/> class. /// Initializes a new instance of the <see cref="Application"/> class.
@ -65,11 +66,7 @@ namespace Avalonia
/// <value> /// <value>
/// The application's global data templates. /// The application's global data templates.
/// </value> /// </value>
public DataTemplates DataTemplates public DataTemplates DataTemplates => _dataTemplates ?? (_dataTemplates = new DataTemplates());
{
get { return _dataTemplates ?? (_dataTemplates = new DataTemplates()); }
set { _dataTemplates = value; }
}
/// <summary> /// <summary>
/// Gets the application's focus manager. /// Gets the application's focus manager.
@ -109,13 +106,19 @@ namespace Avalonia
/// <remarks> /// <remarks>
/// Global styles apply to all windows in the application. /// Global styles apply to all windows in the application.
/// </remarks> /// </remarks>
public Styles Styles { get; } = new Styles(); public Styles Styles => _styles ?? (_styles = new Styles());
/// <inheritdoc/>
bool IDataTemplateHost.IsDataTemplatesInitialized => _dataTemplates != null;
/// <summary> /// <summary>
/// Gets the styling parent of the application, which is null. /// Gets the styling parent of the application, which is null.
/// </summary> /// </summary>
IStyleHost IStyleHost.StylingParent => null; IStyleHost IStyleHost.StylingParent => null;
/// <inheritdoc/>
bool IStyleHost.IsStylesInitialized => _styles != null;
/// <summary> /// <summary>
/// Initializes the application by loading XAML etc. /// Initializes the application by loading XAML etc.
/// </summary> /// </summary>

22
src/Avalonia.Controls/Control.cs

@ -97,8 +97,8 @@ namespace Avalonia.Controls
private bool _isAttachedToLogicalTree; private bool _isAttachedToLogicalTree;
private IAvaloniaList<ILogical> _logicalChildren; private IAvaloniaList<ILogical> _logicalChildren;
private INameScope _nameScope; private INameScope _nameScope;
private Styles _styles;
private bool _styled; private bool _styled;
private Styles _styles;
private Subject<IStyleable> _styleDetach = new Subject<IStyleable>(); private Subject<IStyleable> _styleDetach = new Subject<IStyleable>();
/// <summary> /// <summary>
@ -243,11 +243,7 @@ namespace Avalonia.Controls
/// Each control may define data templates which are applied to the control itself and its /// Each control may define data templates which are applied to the control itself and its
/// children. /// children.
/// </remarks> /// </remarks>
public DataTemplates DataTemplates public DataTemplates DataTemplates => _dataTemplates ?? (_dataTemplates = new DataTemplates());
{
get { return _dataTemplates ?? (_dataTemplates = new DataTemplates()); }
set { _dataTemplates = value; }
}
/// <summary> /// <summary>
/// Gets a value that indicates whether the element has finished initialization. /// Gets a value that indicates whether the element has finished initialization.
@ -259,18 +255,14 @@ namespace Avalonia.Controls
public bool IsInitialized { get; private set; } public bool IsInitialized { get; private set; }
/// <summary> /// <summary>
/// Gets or sets the styles for the control. /// Gets the styles for the control.
/// </summary> /// </summary>
/// <remarks> /// <remarks>
/// Styles for the entire application are added to the Application.Styles collection, but /// Styles for the entire application are added to the Application.Styles collection, but
/// each control may in addition define its own styles which are applied to the control /// each control may in addition define its own styles which are applied to the control
/// itself and its children. /// itself and its children.
/// </remarks> /// </remarks>
public Styles Styles public Styles Styles => _styles ?? (_styles = new Styles());
{
get { return _styles ?? (_styles = new Styles()); }
set { _styles = value; }
}
/// <summary> /// <summary>
/// Gets the control's logical parent. /// Gets the control's logical parent.
@ -304,6 +296,9 @@ namespace Avalonia.Controls
internal set { SetValue(TemplatedParentProperty, value); } internal set { SetValue(TemplatedParentProperty, value); }
} }
/// <inheritdoc/>
bool IDataTemplateHost.IsDataTemplatesInitialized => _dataTemplates != null;
/// <summary> /// <summary>
/// Gets a value indicating whether the element is attached to a rooted logical tree. /// Gets a value indicating whether the element is attached to a rooted logical tree.
/// </summary> /// </summary>
@ -336,6 +331,9 @@ namespace Avalonia.Controls
/// <inheritdoc/> /// <inheritdoc/>
IObservable<IStyleable> IStyleable.StyleDetach => _styleDetach; IObservable<IStyleable> IStyleable.StyleDetach => _styleDetach;
/// <inheritdoc/>
bool IStyleHost.IsStylesInitialized => _styles != null;
/// <inheritdoc/> /// <inheritdoc/>
IStyleHost IStyleHost.StylingParent => (IStyleHost)InheritanceParent; IStyleHost IStyleHost.StylingParent => (IStyleHost)InheritanceParent;

14
src/Avalonia.Controls/IControl.cs

@ -14,7 +14,14 @@ namespace Avalonia.Controls
/// <summary> /// <summary>
/// Interface for Avalonia controls. /// Interface for Avalonia controls.
/// </summary> /// </summary>
public interface IControl : IVisual, ILogical, ILayoutable, IInputElement, INamed, IStyleable, IStyleHost public interface IControl : IVisual,
IDataTemplateHost,
ILogical,
ILayoutable,
IInputElement,
INamed,
IStyleable,
IStyleHost
{ {
/// <summary> /// <summary>
/// Occurs when the control has finished initialization. /// Occurs when the control has finished initialization.
@ -31,11 +38,6 @@ namespace Avalonia.Controls
/// </summary> /// </summary>
object DataContext { get; set; } object DataContext { get; set; }
/// <summary>
/// Gets the data templates for the control.
/// </summary>
DataTemplates DataTemplates { get; }
/// <summary> /// <summary>
/// Gets a value that indicates whether the element has finished initialization. /// Gets a value that indicates whether the element has finished initialization.
/// </summary> /// </summary>

6
src/Avalonia.Controls/IGlobalDataTemplates.cs

@ -8,11 +8,7 @@ namespace Avalonia.Controls
/// <summary> /// <summary>
/// Defines the application-global data templates. /// Defines the application-global data templates.
/// </summary> /// </summary>
public interface IGlobalDataTemplates public interface IGlobalDataTemplates : IDataTemplateHost
{ {
/// <summary>
/// Gets the application-global data templates.
/// </summary>
DataTemplates DataTemplates { get; }
} }
} }

24
src/Avalonia.Controls/Templates/DataTemplateExtensions.cs

@ -17,8 +17,8 @@ namespace Avalonia.Controls.Templates
/// <param name="control">The control searching for the data template.</param> /// <param name="control">The control searching for the data template.</param>
/// <param name="data">The data.</param> /// <param name="data">The data.</param>
/// <param name="primary"> /// <param name="primary">
/// An optional primary template that can will be tried before the /// An optional primary template that can will be tried before the DataTemplates in the
/// <see cref="IControl.DataTemplates"/> in the tree are searched. /// tree are searched.
/// </param> /// </param>
/// <returns>The data template or null if no matching data template was found.</returns> /// <returns>The data template or null if no matching data template was found.</returns>
public static IDataTemplate FindDataTemplate( public static IDataTemplate FindDataTemplate(
@ -31,13 +31,16 @@ namespace Avalonia.Controls.Templates
return primary; return primary;
} }
foreach (var i in control.GetSelfAndLogicalAncestors().OfType<IControl>()) foreach (var i in control.GetSelfAndLogicalAncestors().OfType<IDataTemplateHost>())
{ {
foreach (IDataTemplate dt in i.DataTemplates) if (i.IsDataTemplatesInitialized)
{ {
if (dt.Match(data)) foreach (IDataTemplate dt in i.DataTemplates)
{ {
return dt; if (dt.Match(data))
{
return dt;
}
} }
} }
} }
@ -46,11 +49,14 @@ namespace Avalonia.Controls.Templates
if (global != null) if (global != null)
{ {
foreach (IDataTemplate dt in global.DataTemplates) if (global.IsDataTemplatesInitialized)
{ {
if (dt.Match(data)) foreach (IDataTemplate dt in global.DataTemplates)
{ {
return dt; if (dt.Match(data))
{
return dt;
}
} }
} }
} }

27
src/Avalonia.Controls/Templates/IDataTemplateHost.cs

@ -0,0 +1,27 @@
// Copyright (c) The Avalonia Project. All rights reserved.
// Licensed under the MIT license. See licence.md file in the project root for full license information.
using System;
namespace Avalonia.Controls.Templates
{
/// <summary>
/// Defines an element that has a <see cref="DataTemplates"/> collection.
/// </summary>
public interface IDataTemplateHost
{
/// <summary>
/// Gets the data templates for the element.
/// </summary>
DataTemplates DataTemplates { get; }
/// <summary>
/// Gets a value indicating whether <see cref="DataTemplates"/> is initialized.
/// </summary>
/// <remarks>
/// The <see cref="DataTemplates"/> property may be lazily initialized, if so this property
/// indicates whether it has been initialized.
/// </remarks>
bool IsDataTemplatesInitialized { get; }
}
}

2
src/Avalonia.Diagnostics/DevTools.xaml.cs

@ -71,7 +71,7 @@ namespace Avalonia.Diagnostics
Width = 1024, Width = 1024,
Height = 512, Height = 512,
Content = devTools, Content = devTools,
DataTemplates = new DataTemplates DataTemplates =
{ {
new ViewLocator<ViewModelBase>(), new ViewLocator<ViewModelBase>(),
} }

2
src/Avalonia.Diagnostics/Views/ControlDetailsView.cs

@ -42,7 +42,7 @@ namespace Avalonia.Diagnostics.Views
{ {
Content = _grid = new SimpleGrid Content = _grid = new SimpleGrid
{ {
Styles = new Styles Styles =
{ {
new Style(x => x.Is<Control>()) new Style(x => x.Is<Control>())
{ {

20
src/Avalonia.Interactivity/Interactive.cs

@ -16,14 +16,18 @@ namespace Avalonia.Interactivity
/// </summary> /// </summary>
public class Interactive : Layoutable, IInteractive public class Interactive : Layoutable, IInteractive
{ {
private readonly Dictionary<RoutedEvent, List<EventSubscription>> _eventHandlers = private Dictionary<RoutedEvent, List<EventSubscription>> _eventHandlers;
new Dictionary<RoutedEvent, List<EventSubscription>>();
/// <summary> /// <summary>
/// Gets the interactive parent of the object for bubbling and tunnelling events. /// Gets the interactive parent of the object for bubbling and tunnelling events.
/// </summary> /// </summary>
IInteractive IInteractive.InteractiveParent => ((IVisual)this).VisualParent as IInteractive; IInteractive IInteractive.InteractiveParent => ((IVisual)this).VisualParent as IInteractive;
private Dictionary<RoutedEvent, List<EventSubscription>> EventHandlers
{
get { return _eventHandlers ?? (_eventHandlers = new Dictionary<RoutedEvent, List<EventSubscription>>()); }
}
/// <summary> /// <summary>
/// Adds a handler for the specified routed event. /// Adds a handler for the specified routed event.
/// </summary> /// </summary>
@ -43,10 +47,10 @@ namespace Avalonia.Interactivity
List<EventSubscription> subscriptions; List<EventSubscription> subscriptions;
if (!_eventHandlers.TryGetValue(routedEvent, out subscriptions)) if (!EventHandlers.TryGetValue(routedEvent, out subscriptions))
{ {
subscriptions = new List<EventSubscription>(); subscriptions = new List<EventSubscription>();
_eventHandlers.Add(routedEvent, subscriptions); EventHandlers.Add(routedEvent, subscriptions);
} }
var sub = new EventSubscription var sub = new EventSubscription
@ -89,9 +93,9 @@ namespace Avalonia.Interactivity
Contract.Requires<ArgumentNullException>(routedEvent != null); Contract.Requires<ArgumentNullException>(routedEvent != null);
Contract.Requires<ArgumentNullException>(handler != null); Contract.Requires<ArgumentNullException>(handler != null);
List<EventSubscription> subscriptions; List<EventSubscription> subscriptions = null;
if (_eventHandlers.TryGetValue(routedEvent, out subscriptions)) if (_eventHandlers?.TryGetValue(routedEvent, out subscriptions) == true)
{ {
subscriptions.RemoveAll(x => x.Handler == handler); subscriptions.RemoveAll(x => x.Handler == handler);
} }
@ -181,9 +185,9 @@ namespace Avalonia.Interactivity
e.RoutedEvent.InvokeRaised(this, e); e.RoutedEvent.InvokeRaised(this, e);
List<EventSubscription> subscriptions; List<EventSubscription> subscriptions = null;
if (_eventHandlers.TryGetValue(e.RoutedEvent, out subscriptions)) if (_eventHandlers?.TryGetValue(e.RoutedEvent, out subscriptions) == true)
{ {
foreach (var sub in subscriptions.ToList()) foreach (var sub in subscriptions.ToList())
{ {

12
src/Avalonia.Styling/Styling/IStyleHost.cs

@ -1,6 +1,8 @@
// Copyright (c) The Avalonia Project. All rights reserved. // Copyright (c) The Avalonia Project. All rights reserved.
// Licensed under the MIT license. See licence.md file in the project root for full license information. // Licensed under the MIT license. See licence.md file in the project root for full license information.
using System;
namespace Avalonia.Styling namespace Avalonia.Styling
{ {
/// <summary> /// <summary>
@ -8,6 +10,15 @@ namespace Avalonia.Styling
/// </summary> /// </summary>
public interface IStyleHost public interface IStyleHost
{ {
/// <summary>
/// Gets a value indicating whether <see cref="Styles"/> is initialized.
/// </summary>
/// <remarks>
/// The <see cref="Styles"/> property may be lazily initialized, if so this property
/// indicates whether it has been initialized.
/// </remarks>
bool IsStylesInitialized { get; }
/// <summary> /// <summary>
/// Gets the styles for the element. /// Gets the styles for the element.
/// </summary> /// </summary>
@ -17,6 +28,5 @@ namespace Avalonia.Styling
/// Gets the parent style host element. /// Gets the parent style host element.
/// </summary> /// </summary>
IStyleHost StylingParent { get; } IStyleHost StylingParent { get; }
} }
} }

11
src/Avalonia.Styling/Styling/StyleExtensions.cs

@ -23,11 +23,14 @@ namespace Avalonia.Styling
while (control != null) while (control != null)
{ {
var result = control.Styles.FindResource(name); if (control.IsStylesInitialized)
if (result != AvaloniaProperty.UnsetValue)
{ {
return result; var result = control.Styles.FindResource(name);
if (result != AvaloniaProperty.UnsetValue)
{
return result;
}
} }
control = control.StylingParent; control = control.StylingParent;

5
src/Avalonia.Styling/Styling/Styler.cs

@ -29,7 +29,10 @@ namespace Avalonia.Styling
ApplyStyles(control, parentContainer); ApplyStyles(control, parentContainer);
} }
styleHost.Styles.Attach(control, styleHost); if (styleHost.IsStylesInitialized)
{
styleHost.Styles.Attach(control, styleHost);
}
} }
} }
} }

2
tests/Avalonia.Controls.UnitTests/ItemsControlTests.cs

@ -388,7 +388,7 @@ namespace Avalonia.Controls.UnitTests
{ {
Template = GetTemplate(), Template = GetTemplate(),
DataContext = "Base", DataContext = "Base",
DataTemplates = new DataTemplates DataTemplates =
{ {
new FuncDataTemplate<Item>(x => new Button { Content = x }) new FuncDataTemplate<Item>(x => new Button { Content = x })
}, },

8
tests/Avalonia.Controls.UnitTests/ListBoxTests.cs

@ -109,10 +109,10 @@ namespace Avalonia.Controls.UnitTests
{ {
Template = ListBoxTemplate(), Template = ListBoxTemplate(),
DataContext = "Base", DataContext = "Base",
DataTemplates = new DataTemplates DataTemplates =
{ {
new FuncDataTemplate<Item>(x => new Button { Content = x }) new FuncDataTemplate<Item>(x => new Button { Content = x })
}, },
Items = items, Items = items,
}; };

2
tests/Avalonia.Controls.UnitTests/Presenters/ContentPresenterTests_Unrooted.cs

@ -88,7 +88,7 @@ namespace Avalonia.Controls.UnitTests.Presenters
root.Child = null; root.Child = null;
root = new TestRoot root = new TestRoot
{ {
DataTemplates = new DataTemplates DataTemplates =
{ {
new FuncDataTemplate<string>(x => new Decorator()), new FuncDataTemplate<string>(x => new Decorator()),
}, },

1
tests/Avalonia.Controls.UnitTests/Primitives/PopupTests.cs

@ -265,6 +265,7 @@ namespace Avalonia.Controls.UnitTests.Primitives
}; };
var globalStyles = new Mock<IGlobalStyles>(); var globalStyles = new Mock<IGlobalStyles>();
globalStyles.Setup(x => x.IsStylesInitialized).Returns(true);
globalStyles.Setup(x => x.Styles).Returns(styles); globalStyles.Setup(x => x.Styles).Returns(styles);
var renderInterface = new Mock<IPlatformRenderInterface>(); var renderInterface = new Mock<IPlatformRenderInterface>();

8
tests/Avalonia.Controls.UnitTests/Primitives/TemplatedControlTests.cs

@ -399,7 +399,7 @@ namespace Avalonia.Controls.UnitTests.Primitives
TestTemplatedControl target; TestTemplatedControl target;
var root = new TestRoot var root = new TestRoot
{ {
Styles = new Styles Styles =
{ {
new Style(x => x.OfType<TestTemplatedControl>()) new Style(x => x.OfType<TestTemplatedControl>())
{ {
@ -435,7 +435,7 @@ namespace Avalonia.Controls.UnitTests.Primitives
TestTemplatedControl target; TestTemplatedControl target;
var root = new TestRoot var root = new TestRoot
{ {
Styles = new Styles Styles =
{ {
new Style(x => x.OfType<TestTemplatedControl>()) new Style(x => x.OfType<TestTemplatedControl>())
{ {
@ -474,7 +474,7 @@ namespace Avalonia.Controls.UnitTests.Primitives
var root = new TestRoot var root = new TestRoot
{ {
Styles = new Styles Styles =
{ {
new Style(x => x.OfType<TestTemplatedControl>()) new Style(x => x.OfType<TestTemplatedControl>())
{ {
@ -494,7 +494,7 @@ namespace Avalonia.Controls.UnitTests.Primitives
var root2 = new TestRoot var root2 = new TestRoot
{ {
Styles = new Styles Styles =
{ {
new Style(x => x.OfType<TestTemplatedControl>()) new Style(x => x.OfType<TestTemplatedControl>())
{ {

4
tests/Avalonia.Controls.UnitTests/TabControlTests.cs

@ -135,7 +135,7 @@ namespace Avalonia.Controls.UnitTests
{ {
var root = new TestRoot var root = new TestRoot
{ {
Styles = new Styles Styles =
{ {
new Style(x => x.OfType<TabItem>()) new Style(x => x.OfType<TabItem>())
{ {
@ -174,7 +174,7 @@ namespace Avalonia.Controls.UnitTests
{ {
Template = new FuncControlTemplate<TabControl>(CreateTabControlTemplate), Template = new FuncControlTemplate<TabControl>(CreateTabControlTemplate),
DataContext = "Base", DataContext = "Base",
DataTemplates = new DataTemplates DataTemplates =
{ {
new FuncDataTemplate<Item>(x => new Button { Content = x }) new FuncDataTemplate<Item>(x => new Button { Content = x })
}, },

28
tests/Avalonia.Controls.UnitTests/TreeViewTests.cs

@ -25,9 +25,9 @@ namespace Avalonia.Controls.UnitTests
{ {
Template = CreateTreeViewTemplate(), Template = CreateTreeViewTemplate(),
Items = CreateTestTreeData(), Items = CreateTestTreeData(),
DataTemplates = CreateNodeDataTemplate(),
}; };
CreateNodeDataTemplate(target);
ApplyTemplates(target); ApplyTemplates(target);
Assert.Equal(new[] { "Root" }, ExtractItemHeader(target, 0)); Assert.Equal(new[] { "Root" }, ExtractItemHeader(target, 0));
@ -69,9 +69,9 @@ namespace Avalonia.Controls.UnitTests
{ {
Template = CreateTreeViewTemplate(), Template = CreateTreeViewTemplate(),
Items = CreateTestTreeData(), Items = CreateTestTreeData(),
DataTemplates = CreateNodeDataTemplate(),
}; };
CreateNodeDataTemplate(target);
ApplyTemplates(target); ApplyTemplates(target);
var container = (TreeViewItem)target.ItemContainerGenerator.Containers.Single().ContainerControl; var container = (TreeViewItem)target.ItemContainerGenerator.Containers.Single().ContainerControl;
@ -87,7 +87,6 @@ namespace Avalonia.Controls.UnitTests
{ {
Template = CreateTreeViewTemplate(), Template = CreateTreeViewTemplate(),
Items = tree, Items = tree,
DataTemplates = CreateNodeDataTemplate(),
}; };
// For TreeViewItem to find its parent TreeView, OnAttachedToLogicalTree needs // For TreeViewItem to find its parent TreeView, OnAttachedToLogicalTree needs
@ -95,6 +94,7 @@ namespace Avalonia.Controls.UnitTests
var root = new TestRoot(); var root = new TestRoot();
root.Child = target; root.Child = target;
CreateNodeDataTemplate(target);
ApplyTemplates(target); ApplyTemplates(target);
var container = target.ItemContainerGenerator.Index.ContainerFromItem( var container = target.ItemContainerGenerator.Index.ContainerFromItem(
@ -116,11 +116,12 @@ namespace Avalonia.Controls.UnitTests
{ {
Template = CreateTreeViewTemplate(), Template = CreateTreeViewTemplate(),
Items = tree, Items = tree,
DataTemplates = CreateNodeDataTemplate(),
}; };
var visualRoot = new TestRoot(); var visualRoot = new TestRoot();
visualRoot.Child = target; visualRoot.Child = target;
CreateNodeDataTemplate(target);
ApplyTemplates(target); ApplyTemplates(target);
var item = tree[0].Children[1].Children[0]; var item = tree[0].Children[1].Children[0];
@ -146,11 +147,12 @@ namespace Avalonia.Controls.UnitTests
{ {
Template = CreateTreeViewTemplate(), Template = CreateTreeViewTemplate(),
Items = tree, Items = tree,
DataTemplates = CreateNodeDataTemplate(),
}; };
var visualRoot = new TestRoot(); var visualRoot = new TestRoot();
visualRoot.Child = target; visualRoot.Child = target;
CreateNodeDataTemplate(target);
ApplyTemplates(target); ApplyTemplates(target);
var item = tree[0].Children[1].Children[0]; var item = tree[0].Children[1].Children[0];
@ -191,12 +193,13 @@ namespace Avalonia.Controls.UnitTests
var target = new TreeView var target = new TreeView
{ {
Template = CreateTreeViewTemplate(), Template = CreateTreeViewTemplate(),
DataTemplates = CreateNodeDataTemplate(),
Items = tree, Items = tree,
}; };
var root = new TestRoot(); var root = new TestRoot();
root.Child = target; root.Child = target;
CreateNodeDataTemplate(target);
ApplyTemplates(target); ApplyTemplates(target);
Assert.Equal(5, target.ItemContainerGenerator.Index.Items.Count()); Assert.Equal(5, target.ItemContainerGenerator.Index.Items.Count());
@ -221,7 +224,7 @@ namespace Avalonia.Controls.UnitTests
{ {
Template = CreateTreeViewTemplate(), Template = CreateTreeViewTemplate(),
DataContext = "Base", DataContext = "Base",
DataTemplates = new DataTemplates DataTemplates =
{ {
new FuncDataTemplate<Node>(x => new Button { Content = x }) new FuncDataTemplate<Node>(x => new Button { Content = x })
}, },
@ -291,9 +294,9 @@ namespace Avalonia.Controls.UnitTests
{ {
Template = CreateTreeViewTemplate(), Template = CreateTreeViewTemplate(),
Items = data, Items = data,
DataTemplates = CreateNodeDataTemplate(),
}; };
CreateNodeDataTemplate(target);
ApplyTemplates(target); ApplyTemplates(target);
Assert.Equal(new[] { "Root" }, ExtractItemHeader(target, 0)); Assert.Equal(new[] { "Root" }, ExtractItemHeader(target, 0));
@ -328,7 +331,6 @@ namespace Avalonia.Controls.UnitTests
{ {
Template = CreateTreeViewTemplate(), Template = CreateTreeViewTemplate(),
Items = data, Items = data,
DataTemplates = CreateNodeDataTemplate(),
}; };
var button = new Button(); var button = new Button();
@ -341,6 +343,7 @@ namespace Avalonia.Controls.UnitTests
} }
}; };
CreateNodeDataTemplate(target);
ApplyTemplates(target); ApplyTemplates(target);
var item = data[0].Children[0]; var item = data[0].Children[0];
@ -411,12 +414,9 @@ namespace Avalonia.Controls.UnitTests
}; };
} }
private DataTemplates CreateNodeDataTemplate() private void CreateNodeDataTemplate(IControl control)
{ {
return new DataTemplates control.DataTemplates.Add(new TestTreeDataTemplate());
{
new TestTreeDataTemplate()
};
} }
private IControlTemplate CreateTreeViewTemplate() private IControlTemplate CreateTreeViewTemplate()

2
tests/Avalonia.Controls.UnitTests/UserControlTests.cs

@ -21,7 +21,7 @@ namespace Avalonia.Controls.UnitTests
var target = new UserControl(); var target = new UserControl();
var root = new TestRoot var root = new TestRoot
{ {
Styles = new Styles Styles =
{ {
new Style(x => x.OfType<ContentControl>()) new Style(x => x.OfType<ContentControl>())
{ {

1
tests/Avalonia.Layout.UnitTests/FullLayoutTests.cs

@ -193,6 +193,7 @@ namespace Avalonia.Layout.UnitTests
.Bind<IWindowingPlatform>().ToConstant(new Avalonia.Controls.UnitTests.WindowingPlatformMock(() => windowImpl.Object)); .Bind<IWindowingPlatform>().ToConstant(new Avalonia.Controls.UnitTests.WindowingPlatformMock(() => windowImpl.Object));
var theme = new DefaultTheme(); var theme = new DefaultTheme();
globalStyles.Setup(x => x.IsStylesInitialized).Returns(true);
globalStyles.Setup(x => x.Styles).Returns(theme); globalStyles.Setup(x => x.Styles).Returns(theme);
} }
} }

2
tests/Avalonia.LeakTests/ControlTests.cs

@ -276,7 +276,7 @@ namespace Avalonia.LeakTests
{ {
Content = target = new TreeView Content = target = new TreeView
{ {
DataTemplates = new DataTemplates DataTemplates =
{ {
new FuncTreeDataTemplate<Node>( new FuncTreeDataTemplate<Node>(
x => new TextBlock { Text = x.Name }, x => new TextBlock { Text = x.Name },

20
tests/Avalonia.Styling.UnitTests/ResourceTests.cs

@ -16,7 +16,7 @@ namespace Avalonia.Styling.UnitTests
var tree = new Decorator var tree = new Decorator
{ {
Styles = new Styles Styles =
{ {
new Style new Style
{ {
@ -29,7 +29,7 @@ namespace Avalonia.Styling.UnitTests
}, },
Child = target = new Border Child = target = new Border
{ {
Styles = new Styles Styles =
{ {
new Style new Style
{ {
@ -60,16 +60,16 @@ namespace Avalonia.Styling.UnitTests
var tree = target = new Border var tree = target = new Border
{ {
Styles = new Styles Styles =
{
new Style
{ {
new Style Resources = new StyleResources
{ {
Resources = new StyleResources { "Foo", "foo" },
{ }
{ "Foo", "foo" }, },
} }
},
}
}; };
Assert.Equal(AvaloniaProperty.UnsetValue, target.FindStyleResource("Baz")); Assert.Equal(AvaloniaProperty.UnsetValue, target.FindStyleResource("Baz"));

Loading…
Cancel
Save