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.
144 lines
4.5 KiB
144 lines
4.5 KiB
using System;
|
|
using Avalonia.Collections;
|
|
using Avalonia.Controls.Presenters;
|
|
using Avalonia.Controls.Templates;
|
|
using Avalonia.Data;
|
|
using Avalonia.LogicalTree;
|
|
|
|
namespace Avalonia.Controls.Primitives
|
|
{
|
|
/// <summary>
|
|
/// Represents an <see cref="ItemsControl"/> with a related header.
|
|
/// </summary>
|
|
public class HeaderedItemsControl : ItemsControl, IContentPresenterHost
|
|
{
|
|
private IDisposable? _itemsBinding;
|
|
private bool _prepareItemContainerOnAttach;
|
|
|
|
/// <summary>
|
|
/// Defines the <see cref="Header"/> property.
|
|
/// </summary>
|
|
public static readonly StyledProperty<object?> HeaderProperty =
|
|
HeaderedContentControl.HeaderProperty.AddOwner<HeaderedItemsControl>();
|
|
|
|
/// <summary>
|
|
/// Defines the <see cref="HeaderTemplate"/> property.
|
|
/// </summary>
|
|
public static readonly StyledProperty<IDataTemplate?> HeaderTemplateProperty =
|
|
AvaloniaProperty.Register<HeaderedItemsControl, IDataTemplate?>(nameof(HeaderTemplate));
|
|
|
|
/// <summary>
|
|
/// Initializes static members of the <see cref="ContentControl"/> class.
|
|
/// </summary>
|
|
static HeaderedItemsControl()
|
|
{
|
|
HeaderProperty.Changed.AddClassHandler<HeaderedItemsControl>((x, e) => x.HeaderChanged(e));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets or sets the content of the control's header.
|
|
/// </summary>
|
|
public object? Header
|
|
{
|
|
get { return GetValue(HeaderProperty); }
|
|
set { SetValue(HeaderProperty, value); }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets or sets the data template used to display the header content of the control.
|
|
/// </summary>
|
|
public IDataTemplate? HeaderTemplate
|
|
{
|
|
get => GetValue(HeaderTemplateProperty);
|
|
set => SetValue(HeaderTemplateProperty, value);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the header presenter from the control's template.
|
|
/// </summary>
|
|
public IContentPresenter? HeaderPresenter
|
|
{
|
|
get;
|
|
private set;
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
IAvaloniaList<ILogical> IContentPresenterHost.LogicalChildren => LogicalChildren;
|
|
|
|
protected override void OnAttachedToLogicalTree(LogicalTreeAttachmentEventArgs e)
|
|
{
|
|
base.OnAttachedToLogicalTree(e);
|
|
|
|
if (_prepareItemContainerOnAttach)
|
|
{
|
|
PrepareItemContainer();
|
|
_prepareItemContainerOnAttach = false;
|
|
}
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
bool IContentPresenterHost.RegisterContentPresenter(IContentPresenter presenter)
|
|
{
|
|
return RegisterContentPresenter(presenter);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Called when an <see cref="IContentPresenter"/> is registered with the control.
|
|
/// </summary>
|
|
/// <param name="presenter">The presenter.</param>
|
|
protected virtual bool RegisterContentPresenter(IContentPresenter presenter)
|
|
{
|
|
if (presenter.Name == "PART_HeaderPresenter")
|
|
{
|
|
HeaderPresenter = presenter;
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
internal void PrepareItemContainer()
|
|
{
|
|
_itemsBinding?.Dispose();
|
|
_itemsBinding = null;
|
|
|
|
var item = Header;
|
|
|
|
if (item is null)
|
|
{
|
|
_prepareItemContainerOnAttach = false;
|
|
return;
|
|
}
|
|
|
|
var headerTemplate = HeaderTemplate;
|
|
|
|
if (headerTemplate is null)
|
|
{
|
|
if (((ILogical)this).IsAttachedToLogicalTree)
|
|
headerTemplate = this.FindDataTemplate(item);
|
|
else
|
|
_prepareItemContainerOnAttach = true;
|
|
}
|
|
|
|
if (headerTemplate is ITreeDataTemplate treeTemplate &&
|
|
treeTemplate.Match(item) &&
|
|
treeTemplate.ItemsSelector(item) is { } itemsBinding)
|
|
{
|
|
_itemsBinding = BindingOperations.Apply(this, ItemsProperty, itemsBinding, null);
|
|
}
|
|
}
|
|
|
|
private void HeaderChanged(AvaloniaPropertyChangedEventArgs e)
|
|
{
|
|
if (e.OldValue is ILogical oldChild)
|
|
{
|
|
LogicalChildren.Remove(oldChild);
|
|
}
|
|
|
|
if (e.NewValue is ILogical newChild)
|
|
{
|
|
LogicalChildren.Add(newChild);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|