A cross-platform UI framework for .NET
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.
 
 
 

187 lines
6.6 KiB

// -----------------------------------------------------------------------
// <copyright file="Panel.cs" company="Steven Kirk">
// Copyright 2014 MIT Licence. See licence.md for more information.
// </copyright>
// -----------------------------------------------------------------------
namespace Perspex.Controls
{
using System;
using System.Collections.Generic;
using System.Collections.Specialized;
using System.Linq;
using Perspex.Collections;
/// <summary>
/// Base class for controls that can contain multiple children.
/// </summary>
/// <remarks>
/// Controls can be added to a <see cref="Panel"/> by adding them to its <see cref="Children"/>
/// collection. All children are layed out to fill the panel.
/// </remarks>
public class Panel : Control, IReparentingControl
{
private Controls children;
private ILogical childLogicalParent;
/// <summary>
/// Initializes a new instance of the <see cref="Panel"/> class.
/// </summary>
public Panel()
{
this.childLogicalParent = this;
}
/// <summary>
/// Gets or sets the children of the <see cref="Panel"/>.
/// </summary>
/// <remarks>
/// Even though this property can be set, the setter is only intended for use in object
/// initializers. Assigning to this property does not change the underlying collection,
/// it simply clears the existing collection and addds the contents of the assigned
/// collection.
/// </remarks>
public Controls Children
{
get
{
if (this.children == null)
{
this.children = new Controls();
this.children.CollectionChanged += this.ChildrenChanged;
}
return this.children;
}
set
{
Contract.Requires<ArgumentNullException>(value != null);
if (this.children != value)
{
if (this.children != null)
{
this.ClearLogicalParent(this.children);
this.children.CollectionChanged -= this.ChildrenChanged;
}
this.children = value;
this.ClearVisualChildren();
if (this.children != null)
{
this.children.CollectionChanged += this.ChildrenChanged;
this.AddVisualChildren(value);
this.SetLogicalParent(value);
this.InvalidateMeasure();
}
}
}
}
/// <summary>
/// Requests that the visual children of the panel use another control as their logical
/// parent.
/// </summary>
/// <param name="logicalParent">
/// The logical parent for the visual children of the panel.
/// </param>
/// <param name="children">
/// The <see cref="ILogical.LogicalChildren"/> collection to modify.
/// </param>
void IReparentingControl.ReparentLogicalChildren(ILogical logicalParent, IPerspexList<ILogical> children)
{
Contract.Requires<ArgumentNullException>(logicalParent != null);
Contract.Requires<ArgumentNullException>(children != null);
this.childLogicalParent = logicalParent;
this.RedirectLogicalChildren(children);
foreach (var control in this.Children)
{
((ISetLogicalParent)control).SetParent(null);
((ISetLogicalParent)control).SetParent((IControl)logicalParent);
children.Add(control);
}
}
/// <inheritdoc/>
protected virtual void OnChildrenAdded(IEnumerable<Control> child)
{
}
/// <inheritdoc/>
protected virtual void OnChildrenRemoved(IEnumerable<Control> child)
{
}
/// <summary>
/// Clears <see cref="IControl.Parent"/> for the specified controls.
/// </summary>
/// <param name="controls">The controls.</param>
private void ClearLogicalParent(IEnumerable<IControl> controls)
{
foreach (var control in controls)
{
((ISetLogicalParent)control).SetParent(null);
}
}
/// <summary>
/// Sets <see cref="IControl.Parent"/> for the specified controls.
/// </summary>
/// <param name="controls">The controls.</param>
private void SetLogicalParent(IEnumerable<IControl> controls)
{
var parent = this.childLogicalParent as Control;
foreach (var control in controls)
{
((ISetLogicalParent)control).SetParent(parent);
}
}
/// <summary>
/// Called when the <see cref="Children"/> collection changes.
/// </summary>
/// <param name="sender">The event sender.</param>
/// <param name="e">The event args.</param>
private void ChildrenChanged(object sender, NotifyCollectionChangedEventArgs e)
{
List<Control> controls;
// TODO: Handle Replace.
switch (e.Action)
{
case NotifyCollectionChangedAction.Add:
controls = e.NewItems.OfType<Control>().ToList();
this.SetLogicalParent(controls);
this.AddVisualChildren(e.NewItems.OfType<Visual>());
this.LogicalChildren.InsertRange(e.NewStartingIndex, controls);
this.OnChildrenAdded(controls);
break;
case NotifyCollectionChangedAction.Remove:
controls = e.OldItems.OfType<Control>().ToList();
this.ClearLogicalParent(e.OldItems.OfType<Control>());
this.LogicalChildren.RemoveAll(controls);
this.RemoveVisualChildren(e.OldItems.OfType<Visual>());
this.OnChildrenRemoved(controls);
break;
case NotifyCollectionChangedAction.Reset:
controls = e.OldItems.OfType<Control>().ToList();
this.ClearLogicalParent(controls);
this.LogicalChildren.Clear();
this.ClearVisualChildren();
this.AddVisualChildren(this.children);
this.OnChildrenAdded(controls);
break;
}
this.InvalidateMeasure();
}
}
}