зеркало из https://github.com/AvaloniaUI/Avalonia.git
418 строки
13 KiB
C#
418 строки
13 KiB
C#
// -----------------------------------------------------------------------
|
|
// <copyright file="Layoutable.cs" company="Steven Kirk">
|
|
// Copyright 2014 MIT Licence. See licence.md for more information.
|
|
// </copyright>
|
|
// -----------------------------------------------------------------------
|
|
|
|
namespace Perspex.Layout
|
|
{
|
|
using System;
|
|
using System.Linq;
|
|
using Splat;
|
|
|
|
public enum HorizontalAlignment
|
|
{
|
|
Stretch,
|
|
Left,
|
|
Center,
|
|
Right,
|
|
}
|
|
|
|
public enum VerticalAlignment
|
|
{
|
|
Stretch,
|
|
Top,
|
|
Center,
|
|
Bottom,
|
|
}
|
|
|
|
public class Layoutable : Visual, ILayoutable, IEnableLogger
|
|
{
|
|
public static readonly PerspexProperty<Size> ActualSizeProperty =
|
|
PerspexProperty.Register<Layoutable, Size>("ActualSize");
|
|
|
|
public static readonly PerspexProperty<double> WidthProperty =
|
|
PerspexProperty.Register<Layoutable, double>("Width", double.NaN);
|
|
|
|
public static readonly PerspexProperty<double> HeightProperty =
|
|
PerspexProperty.Register<Layoutable, double>("Height", double.NaN);
|
|
|
|
public static readonly PerspexProperty<double> MinWidthProperty =
|
|
PerspexProperty.Register<Layoutable, double>("MinWidth");
|
|
|
|
public static readonly PerspexProperty<double> MaxWidthProperty =
|
|
PerspexProperty.Register<Layoutable, double>("MaxWidth", double.PositiveInfinity);
|
|
|
|
public static readonly PerspexProperty<double> MinHeightProperty =
|
|
PerspexProperty.Register<Layoutable, double>("MinHeight");
|
|
|
|
public static readonly PerspexProperty<double> MaxHeightProperty =
|
|
PerspexProperty.Register<Layoutable, double>("MaxHeight", double.PositiveInfinity);
|
|
|
|
public static readonly PerspexProperty<Thickness> MarginProperty =
|
|
PerspexProperty.Register<Layoutable, Thickness>("Margin");
|
|
|
|
public static readonly PerspexProperty<HorizontalAlignment> HorizontalAlignmentProperty =
|
|
PerspexProperty.Register<Layoutable, HorizontalAlignment>("HorizontalAlignment");
|
|
|
|
public static readonly PerspexProperty<VerticalAlignment> VerticalAlignmentProperty =
|
|
PerspexProperty.Register<Layoutable, VerticalAlignment>("VerticalAlignment");
|
|
|
|
private Size? previousMeasure;
|
|
|
|
private Rect? previousArrange;
|
|
|
|
static Layoutable()
|
|
{
|
|
Layoutable.AffectsMeasure(Visual.IsVisibleProperty);
|
|
Layoutable.AffectsMeasure(Layoutable.WidthProperty);
|
|
Layoutable.AffectsMeasure(Layoutable.HeightProperty);
|
|
Layoutable.AffectsMeasure(Layoutable.MinWidthProperty);
|
|
Layoutable.AffectsMeasure(Layoutable.MaxWidthProperty);
|
|
Layoutable.AffectsMeasure(Layoutable.MinHeightProperty);
|
|
Layoutable.AffectsMeasure(Layoutable.MaxHeightProperty);
|
|
Layoutable.AffectsMeasure(Layoutable.MarginProperty);
|
|
Layoutable.AffectsMeasure(Layoutable.HorizontalAlignmentProperty);
|
|
Layoutable.AffectsMeasure(Layoutable.VerticalAlignmentProperty);
|
|
}
|
|
|
|
public double Width
|
|
{
|
|
get { return this.GetValue(WidthProperty); }
|
|
set { this.SetValue(WidthProperty, value); }
|
|
}
|
|
|
|
public double Height
|
|
{
|
|
get { return this.GetValue(HeightProperty); }
|
|
set { this.SetValue(HeightProperty, value); }
|
|
}
|
|
|
|
public double MinWidth
|
|
{
|
|
get { return this.GetValue(MinWidthProperty); }
|
|
set { this.SetValue(MinWidthProperty, value); }
|
|
}
|
|
|
|
public double MaxWidth
|
|
{
|
|
get { return this.GetValue(MaxWidthProperty); }
|
|
set { this.SetValue(MaxWidthProperty, value); }
|
|
}
|
|
|
|
public double MinHeight
|
|
{
|
|
get { return this.GetValue(MinHeightProperty); }
|
|
set { this.SetValue(MinHeightProperty, value); }
|
|
}
|
|
|
|
public double MaxHeight
|
|
{
|
|
get { return this.GetValue(MaxHeightProperty); }
|
|
set { this.SetValue(MaxHeightProperty, value); }
|
|
}
|
|
|
|
public Thickness Margin
|
|
{
|
|
get { return this.GetValue(MarginProperty); }
|
|
set { this.SetValue(MarginProperty, value); }
|
|
}
|
|
|
|
public HorizontalAlignment HorizontalAlignment
|
|
{
|
|
get { return this.GetValue(HorizontalAlignmentProperty); }
|
|
set { this.SetValue(HorizontalAlignmentProperty, value); }
|
|
}
|
|
|
|
public VerticalAlignment VerticalAlignment
|
|
{
|
|
get { return this.GetValue(VerticalAlignmentProperty); }
|
|
set { this.SetValue(VerticalAlignmentProperty, value); }
|
|
}
|
|
|
|
public Size ActualSize
|
|
{
|
|
get { return this.GetValue(ActualSizeProperty); }
|
|
}
|
|
|
|
public Size? DesiredSize
|
|
{
|
|
get;
|
|
set;
|
|
}
|
|
|
|
public bool IsMeasureValid
|
|
{
|
|
get;
|
|
private set;
|
|
}
|
|
|
|
public bool IsArrangeValid
|
|
{
|
|
get;
|
|
private set;
|
|
}
|
|
|
|
Size? ILayoutable.PreviousMeasure
|
|
{
|
|
get { return this.previousMeasure; }
|
|
}
|
|
|
|
Rect? ILayoutable.PreviousArrange
|
|
{
|
|
get { return this.previousArrange; }
|
|
}
|
|
|
|
public virtual void ApplyTemplate()
|
|
{
|
|
}
|
|
|
|
public void Measure(Size availableSize, bool force = false)
|
|
{
|
|
if (double.IsNaN(availableSize.Width) || double.IsNaN(availableSize.Height))
|
|
{
|
|
throw new InvalidOperationException("Cannot call Measure using a size with NaN values.");
|
|
}
|
|
|
|
if (force || !this.IsMeasureValid || this.previousMeasure != availableSize)
|
|
{
|
|
this.IsMeasureValid = true;
|
|
this.DesiredSize = this.MeasureCore(availableSize).Constrain(availableSize);
|
|
this.previousMeasure = availableSize;
|
|
|
|
this.Log().Debug(
|
|
"Measure of {0} (#{1:x8}) requested {2} ",
|
|
this.GetType().Name,
|
|
this.GetHashCode(),
|
|
this.DesiredSize);
|
|
}
|
|
}
|
|
|
|
public void Arrange(Rect rect, bool force = false)
|
|
{
|
|
if (rect.Width < 0 || rect.Height < 0 ||
|
|
double.IsInfinity(rect.Width) || double.IsInfinity(rect.Height) ||
|
|
double.IsNaN(rect.Width) || double.IsNaN(rect.Height))
|
|
{
|
|
throw new InvalidOperationException("Invalid Arrange rectangle.");
|
|
}
|
|
|
|
if (!this.DesiredSize.HasValue)
|
|
{
|
|
throw new InvalidOperationException("Arrange called before Measure.");
|
|
}
|
|
|
|
if (force || !this.IsArrangeValid || this.previousArrange != rect)
|
|
{
|
|
this.Log().Debug(
|
|
"Arrange of {0} (#{1:x8}) gave {2} ",
|
|
this.GetType().Name,
|
|
this.GetHashCode(),
|
|
rect);
|
|
|
|
this.IsArrangeValid = true;
|
|
this.ArrangeCore(rect);
|
|
this.previousArrange = rect;
|
|
}
|
|
}
|
|
|
|
public void InvalidateMeasure()
|
|
{
|
|
var parent = this.GetVisualParent<ILayoutable>();
|
|
|
|
this.IsMeasureValid = false;
|
|
this.IsArrangeValid = false;
|
|
this.previousMeasure = null;
|
|
this.previousArrange = null;
|
|
|
|
if (parent != null && IsResizable(parent))
|
|
{
|
|
parent.InvalidateMeasure();
|
|
}
|
|
else
|
|
{
|
|
var root = this.GetLayoutRoot();
|
|
|
|
if (root != null && root.Item1.LayoutManager != null)
|
|
{
|
|
root.Item1.LayoutManager.InvalidateMeasure(this, root.Item2);
|
|
}
|
|
}
|
|
}
|
|
|
|
public void InvalidateArrange()
|
|
{
|
|
var root = this.GetLayoutRoot();
|
|
|
|
this.IsArrangeValid = false;
|
|
this.previousArrange = null;
|
|
|
|
if (root != null && root.Item1.LayoutManager != null)
|
|
{
|
|
root.Item1.LayoutManager.InvalidateArrange(this, root.Item2);
|
|
}
|
|
}
|
|
|
|
protected static void AffectsArrange(PerspexProperty property)
|
|
{
|
|
property.Changed.Subscribe(AffectsArrangeInvalidate);
|
|
}
|
|
|
|
protected static void AffectsMeasure(PerspexProperty property)
|
|
{
|
|
property.Changed.Subscribe(AffectsMeasureInvalidate);
|
|
}
|
|
|
|
protected virtual void ArrangeCore(Rect finalRect)
|
|
{
|
|
if (this.IsVisible)
|
|
{
|
|
double originX = finalRect.X + this.Margin.Left;
|
|
double originY = finalRect.Y + this.Margin.Top;
|
|
var size = new Size(
|
|
Math.Max(0, finalRect.Width - this.Margin.Left - this.Margin.Right),
|
|
Math.Max(0, finalRect.Height - this.Margin.Top - this.Margin.Bottom));
|
|
|
|
if (this.HorizontalAlignment != HorizontalAlignment.Stretch)
|
|
{
|
|
size = size.WithWidth(Math.Min(size.Width, this.DesiredSize.Value.Width));
|
|
}
|
|
|
|
if (this.VerticalAlignment != VerticalAlignment.Stretch)
|
|
{
|
|
size = size.WithHeight(Math.Min(size.Height, this.DesiredSize.Value.Height));
|
|
}
|
|
|
|
size = LayoutHelper.ApplyLayoutConstraints(this, size);
|
|
size = this.ArrangeOverride(size).Constrain(size);
|
|
|
|
switch (this.HorizontalAlignment)
|
|
{
|
|
case HorizontalAlignment.Center:
|
|
originX += (finalRect.Width - size.Width) / 2;
|
|
break;
|
|
case HorizontalAlignment.Right:
|
|
originX += finalRect.Width - size.Width;
|
|
break;
|
|
}
|
|
|
|
switch (this.VerticalAlignment)
|
|
{
|
|
case VerticalAlignment.Center:
|
|
originY += (finalRect.Height - size.Height) / 2;
|
|
break;
|
|
case VerticalAlignment.Bottom:
|
|
originY += finalRect.Height - size.Height;
|
|
break;
|
|
}
|
|
|
|
var bounds = new Rect(originX, originY, size.Width, size.Height);
|
|
this.SetVisualBounds(bounds);
|
|
this.SetValue(ActualSizeProperty, bounds.Size);
|
|
}
|
|
}
|
|
|
|
protected virtual Size ArrangeOverride(Size finalSize)
|
|
{
|
|
foreach (ILayoutable child in this.GetVisualChildren().OfType<ILayoutable>())
|
|
{
|
|
child.Arrange(new Rect(finalSize));
|
|
}
|
|
|
|
return finalSize;
|
|
}
|
|
|
|
protected virtual Size MeasureCore(Size availableSize)
|
|
{
|
|
if (this.IsVisible)
|
|
{
|
|
this.ApplyTemplate();
|
|
|
|
var constrained = LayoutHelper
|
|
.ApplyLayoutConstraints(this, availableSize)
|
|
.Deflate(this.Margin);
|
|
|
|
var measured = this.MeasureOverride(constrained);
|
|
var width = measured.Width;
|
|
var height = measured.Height;
|
|
|
|
if (!double.IsNaN(this.Width))
|
|
{
|
|
width = this.Width;
|
|
}
|
|
|
|
width = Math.Min(width, this.MaxWidth);
|
|
width = Math.Max(width, this.MinWidth);
|
|
|
|
if (!double.IsNaN(this.Height))
|
|
{
|
|
height = this.Height;
|
|
}
|
|
|
|
height = Math.Min(height, this.MaxHeight);
|
|
height = Math.Max(height, this.MinHeight);
|
|
|
|
return new Size(width, height).Inflate(this.Margin);
|
|
}
|
|
else
|
|
{
|
|
return new Size();
|
|
}
|
|
}
|
|
|
|
protected virtual Size MeasureOverride(Size availableSize)
|
|
{
|
|
double width = 0;
|
|
double height = 0;
|
|
|
|
foreach (ILayoutable child in this.GetVisualChildren().OfType<ILayoutable>())
|
|
{
|
|
child.Measure(availableSize);
|
|
width = Math.Max(width, child.DesiredSize.Value.Width);
|
|
height = Math.Max(height, child.DesiredSize.Value.Height);
|
|
}
|
|
|
|
return new Size(width, height);
|
|
}
|
|
|
|
private static void AffectsArrangeInvalidate(PerspexPropertyChangedEventArgs e)
|
|
{
|
|
ILayoutable control = e.Sender as ILayoutable;
|
|
|
|
if (control != null)
|
|
{
|
|
control.InvalidateArrange();
|
|
}
|
|
}
|
|
|
|
private static void AffectsMeasureInvalidate(PerspexPropertyChangedEventArgs e)
|
|
{
|
|
ILayoutable control = e.Sender as ILayoutable;
|
|
|
|
if (control != null)
|
|
{
|
|
control.InvalidateMeasure();
|
|
}
|
|
}
|
|
|
|
private static bool IsResizable(ILayoutable control)
|
|
{
|
|
return double.IsNaN(control.Width) || double.IsNaN(control.Height);
|
|
}
|
|
|
|
private Tuple<ILayoutRoot, int> GetLayoutRoot()
|
|
{
|
|
var control = (IVisual)this;
|
|
var distance = 0;
|
|
|
|
while (control != null && !(control is ILayoutRoot))
|
|
{
|
|
control = control.GetVisualParent();
|
|
++distance;
|
|
}
|
|
|
|
return control != null ? Tuple.Create((ILayoutRoot)control, distance) : null;
|
|
}
|
|
}
|
|
}
|