зеркало из https://github.com/DeGsoft/maui-linux.git
489 строки
14 KiB
C#
489 строки
14 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Collections.ObjectModel;
|
|
using System.Collections.Specialized;
|
|
using System.ComponentModel;
|
|
using System.Linq;
|
|
using Xamarin.Forms.Internals;
|
|
|
|
namespace Xamarin.Forms
|
|
{
|
|
[ContentProperty("Children")]
|
|
public abstract class Layout<T> : Layout, IViewContainer<T> where T : View
|
|
{
|
|
readonly ElementCollection<T> _children;
|
|
|
|
protected Layout()
|
|
{
|
|
_children = new ElementCollection<T>(InternalChildren);
|
|
}
|
|
|
|
public new IList<T> Children
|
|
{
|
|
get { return _children; }
|
|
}
|
|
|
|
protected virtual void OnAdded(T view)
|
|
{
|
|
}
|
|
|
|
protected override void OnChildAdded(Element child)
|
|
{
|
|
base.OnChildAdded(child);
|
|
|
|
var typedChild = child as T;
|
|
if (typedChild != null)
|
|
OnAdded(typedChild);
|
|
}
|
|
|
|
protected override void OnChildRemoved(Element child)
|
|
{
|
|
base.OnChildRemoved(child);
|
|
|
|
var typedChild = child as T;
|
|
if (typedChild != null)
|
|
OnRemoved(typedChild);
|
|
}
|
|
|
|
protected virtual void OnRemoved(T view)
|
|
{
|
|
}
|
|
}
|
|
|
|
public abstract class Layout : View, ILayout, ILayoutController, IPaddingElement
|
|
{
|
|
public static readonly BindableProperty IsClippedToBoundsProperty = BindableProperty.Create("IsClippedToBounds", typeof(bool), typeof(Layout), false);
|
|
|
|
public static readonly BindableProperty CascadeInputTransparentProperty = BindableProperty.Create(
|
|
nameof(CascadeInputTransparent), typeof(bool), typeof(Layout), true);
|
|
|
|
public static readonly BindableProperty PaddingProperty = PaddingElement.PaddingProperty;
|
|
|
|
static IList<KeyValuePair<Layout, int>> s_resolutionList = new List<KeyValuePair<Layout, int>>();
|
|
static bool s_relayoutInProgress;
|
|
bool _allocatedFlag;
|
|
|
|
bool _hasDoneLayout;
|
|
Size _lastLayoutSize = new Size(-1, -1);
|
|
|
|
ReadOnlyCollection<Element> _logicalChildren;
|
|
|
|
protected Layout()
|
|
{
|
|
//if things were added in base ctor (through implicit styles), the items added aren't properly parented
|
|
if (InternalChildren.Count > 0)
|
|
InternalChildrenOnCollectionChanged(this, new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Add, InternalChildren));
|
|
|
|
InternalChildren.CollectionChanged += InternalChildrenOnCollectionChanged;
|
|
}
|
|
|
|
public bool IsClippedToBounds
|
|
{
|
|
get { return (bool)GetValue(IsClippedToBoundsProperty); }
|
|
set { SetValue(IsClippedToBoundsProperty, value); }
|
|
}
|
|
|
|
public Thickness Padding
|
|
{
|
|
get { return (Thickness)GetValue(PaddingElement.PaddingProperty); }
|
|
set { SetValue(PaddingElement.PaddingProperty, value); }
|
|
}
|
|
|
|
public bool CascadeInputTransparent
|
|
{
|
|
get { return (bool)GetValue(CascadeInputTransparentProperty); }
|
|
set { SetValue(CascadeInputTransparentProperty, value); }
|
|
}
|
|
|
|
Thickness IPaddingElement.PaddingDefaultValueCreator()
|
|
{
|
|
return default(Thickness);
|
|
}
|
|
|
|
void IPaddingElement.OnPaddingPropertyChanged(Thickness oldValue, Thickness newValue)
|
|
{
|
|
InvalidateLayout();
|
|
}
|
|
|
|
internal ObservableCollection<Element> InternalChildren { get; } = new ObservableCollection<Element>();
|
|
|
|
internal override ReadOnlyCollection<Element> LogicalChildrenInternal
|
|
{
|
|
get { return _logicalChildren ?? (_logicalChildren = new ReadOnlyCollection<Element>(InternalChildren)); }
|
|
}
|
|
|
|
public event EventHandler LayoutChanged;
|
|
|
|
[EditorBrowsable(EditorBrowsableState.Never)]
|
|
public IReadOnlyList<Element> Children
|
|
{
|
|
get { return InternalChildren; }
|
|
}
|
|
|
|
public void ForceLayout()
|
|
{
|
|
SizeAllocated(Width, Height);
|
|
}
|
|
|
|
[Obsolete("OnSizeRequest is obsolete as of version 2.2.0. Please use OnMeasure instead.")]
|
|
[EditorBrowsable(EditorBrowsableState.Never)]
|
|
public sealed override SizeRequest GetSizeRequest(double widthConstraint, double heightConstraint)
|
|
{
|
|
SizeRequest size = base.GetSizeRequest(widthConstraint - Padding.HorizontalThickness, heightConstraint - Padding.VerticalThickness);
|
|
return new SizeRequest(new Size(size.Request.Width + Padding.HorizontalThickness, size.Request.Height + Padding.VerticalThickness),
|
|
new Size(size.Minimum.Width + Padding.HorizontalThickness, size.Minimum.Height + Padding.VerticalThickness));
|
|
}
|
|
|
|
public static void LayoutChildIntoBoundingRegion(VisualElement child, Rectangle region)
|
|
{
|
|
var parent = child.Parent as IFlowDirectionController;
|
|
bool isRightToLeft = false;
|
|
if (parent != null && (isRightToLeft = parent.ApplyEffectiveFlowDirectionToChildContainer && parent.EffectiveFlowDirection.IsRightToLeft()))
|
|
region = new Rectangle(parent.Width - region.Right, region.Y, region.Width, region.Height);
|
|
|
|
var view = child as View;
|
|
if (view == null)
|
|
{
|
|
child.Layout(region);
|
|
return;
|
|
}
|
|
|
|
LayoutOptions horizontalOptions = view.HorizontalOptions;
|
|
if (horizontalOptions.Alignment != LayoutAlignment.Fill)
|
|
{
|
|
SizeRequest request = child.Measure(region.Width, region.Height, MeasureFlags.IncludeMargins);
|
|
double diff = Math.Max(0, region.Width - request.Request.Width);
|
|
double horizontalAlign = horizontalOptions.Alignment.ToDouble();
|
|
if (isRightToLeft)
|
|
horizontalAlign = 1 - horizontalAlign;
|
|
region.X += (int)(diff * horizontalAlign);
|
|
region.Width -= diff;
|
|
}
|
|
|
|
LayoutOptions verticalOptions = view.VerticalOptions;
|
|
if (verticalOptions.Alignment != LayoutAlignment.Fill)
|
|
{
|
|
SizeRequest request = child.Measure(region.Width, region.Height, MeasureFlags.IncludeMargins);
|
|
double diff = Math.Max(0, region.Height - request.Request.Height);
|
|
region.Y += (int)(diff * verticalOptions.Alignment.ToDouble());
|
|
region.Height -= diff;
|
|
}
|
|
|
|
Thickness margin = view.Margin;
|
|
region.X += margin.Left;
|
|
region.Width -= margin.HorizontalThickness;
|
|
region.Y += margin.Top;
|
|
region.Height -= margin.VerticalThickness;
|
|
|
|
child.Layout(region);
|
|
}
|
|
|
|
public void LowerChild(View view)
|
|
{
|
|
if (!InternalChildren.Contains(view) || InternalChildren.First() == view)
|
|
return;
|
|
|
|
InternalChildren.Move(InternalChildren.IndexOf(view), 0);
|
|
OnChildrenReordered();
|
|
}
|
|
|
|
public void RaiseChild(View view)
|
|
{
|
|
if (!InternalChildren.Contains(view) || InternalChildren.Last() == view)
|
|
return;
|
|
|
|
InternalChildren.Move(InternalChildren.IndexOf(view), InternalChildren.Count - 1);
|
|
OnChildrenReordered();
|
|
}
|
|
|
|
protected virtual void InvalidateLayout()
|
|
{
|
|
_hasDoneLayout = false;
|
|
InvalidateMeasureInternal(InvalidationTrigger.MeasureChanged);
|
|
if (!_hasDoneLayout)
|
|
ForceLayout();
|
|
}
|
|
|
|
protected abstract void LayoutChildren(double x, double y, double width, double height);
|
|
|
|
protected void OnChildMeasureInvalidated(object sender, EventArgs e)
|
|
{
|
|
InvalidationTrigger trigger = (e as InvalidationEventArgs)?.Trigger ?? InvalidationTrigger.Undefined;
|
|
OnChildMeasureInvalidated((VisualElement)sender, trigger);
|
|
OnChildMeasureInvalidated();
|
|
}
|
|
|
|
protected virtual void OnChildMeasureInvalidated()
|
|
{
|
|
}
|
|
|
|
protected override void OnSizeAllocated(double width, double height)
|
|
{
|
|
_allocatedFlag = true;
|
|
base.OnSizeAllocated(width, height);
|
|
UpdateChildrenLayout();
|
|
}
|
|
|
|
protected virtual bool ShouldInvalidateOnChildAdded(View child)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
protected virtual bool ShouldInvalidateOnChildRemoved(View child)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
protected void UpdateChildrenLayout()
|
|
{
|
|
_hasDoneLayout = true;
|
|
|
|
if (!ShouldLayoutChildren())
|
|
return;
|
|
|
|
var oldBounds = new Rectangle[LogicalChildrenInternal.Count];
|
|
for (var index = 0; index < oldBounds.Length; index++)
|
|
{
|
|
var c = (VisualElement)LogicalChildrenInternal[index];
|
|
oldBounds[index] = c.Bounds;
|
|
}
|
|
|
|
double width = Width;
|
|
double height = Height;
|
|
|
|
double x = Padding.Left;
|
|
double y = Padding.Top;
|
|
double w = Math.Max(0, width - Padding.HorizontalThickness);
|
|
double h = Math.Max(0, height - Padding.VerticalThickness);
|
|
|
|
var isHeadless = CompressedLayout.GetIsHeadless(this);
|
|
var headlessOffset = CompressedLayout.GetHeadlessOffset(this);
|
|
for (var i = 0; i < LogicalChildrenInternal.Count; i++)
|
|
CompressedLayout.SetHeadlessOffset((VisualElement)LogicalChildrenInternal[i], isHeadless ? new Point(headlessOffset.X + Bounds.X, headlessOffset.Y + Bounds.Y) : new Point());
|
|
|
|
_lastLayoutSize = new Size(width, height);
|
|
|
|
LayoutChildren(x, y, w, h);
|
|
|
|
for (var i = 0; i < oldBounds.Length; i++)
|
|
{
|
|
Rectangle oldBound = oldBounds[i];
|
|
Rectangle newBound = ((VisualElement)LogicalChildrenInternal[i]).Bounds;
|
|
if (oldBound != newBound)
|
|
{
|
|
LayoutChanged?.Invoke(this, EventArgs.Empty);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
internal static void LayoutChildIntoBoundingRegion(View child, Rectangle region, SizeRequest childSizeRequest)
|
|
{
|
|
var parent = child.Parent as IFlowDirectionController;
|
|
bool isRightToLeft = false;
|
|
if (parent != null && (isRightToLeft = parent.ApplyEffectiveFlowDirectionToChildContainer && parent.EffectiveFlowDirection.IsRightToLeft()))
|
|
region = new Rectangle(parent.Width - region.Right, region.Y, region.Width, region.Height);
|
|
|
|
if (region.Size != childSizeRequest.Request)
|
|
{
|
|
bool canUseAlreadyDoneRequest = region.Width >= childSizeRequest.Request.Width && region.Height >= childSizeRequest.Request.Height;
|
|
|
|
LayoutOptions horizontalOptions = child.HorizontalOptions;
|
|
if (horizontalOptions.Alignment != LayoutAlignment.Fill)
|
|
{
|
|
SizeRequest request = canUseAlreadyDoneRequest ? childSizeRequest : child.Measure(region.Width, region.Height, MeasureFlags.IncludeMargins);
|
|
double diff = Math.Max(0, region.Width - request.Request.Width);
|
|
double horizontalAlign = horizontalOptions.Alignment.ToDouble();
|
|
if (isRightToLeft)
|
|
horizontalAlign = 1 - horizontalAlign;
|
|
region.X += (int)(diff * horizontalAlign);
|
|
region.Width -= diff;
|
|
}
|
|
|
|
LayoutOptions verticalOptions = child.VerticalOptions;
|
|
if (verticalOptions.Alignment != LayoutAlignment.Fill)
|
|
{
|
|
SizeRequest request = canUseAlreadyDoneRequest ? childSizeRequest : child.Measure(region.Width, region.Height, MeasureFlags.IncludeMargins);
|
|
double diff = Math.Max(0, region.Height - request.Request.Height);
|
|
region.Y += (int)(diff * verticalOptions.Alignment.ToDouble());
|
|
region.Height -= diff;
|
|
}
|
|
}
|
|
|
|
Thickness margin = child.Margin;
|
|
region.X += margin.Left;
|
|
region.Width -= margin.HorizontalThickness;
|
|
region.Y += margin.Top;
|
|
region.Height -= margin.VerticalThickness;
|
|
|
|
child.Layout(region);
|
|
}
|
|
|
|
internal virtual void OnChildMeasureInvalidated(VisualElement child, InvalidationTrigger trigger)
|
|
{
|
|
ReadOnlyCollection<Element> children = LogicalChildrenInternal;
|
|
int count = children.Count;
|
|
for (var index = 0; index < count; index++)
|
|
{
|
|
var v = LogicalChildrenInternal[index] as VisualElement;
|
|
if (v != null && v.IsVisible && (!v.IsPlatformEnabled || !v.IsNativeStateConsistent))
|
|
return;
|
|
}
|
|
|
|
var view = child as View;
|
|
if (view != null)
|
|
{
|
|
// we can ignore the request if we are either fully constrained or when the size request changes and we were already fully constrainted
|
|
if ((trigger == InvalidationTrigger.MeasureChanged && view.Constraint == LayoutConstraint.Fixed) ||
|
|
(trigger == InvalidationTrigger.SizeRequestChanged && view.ComputedConstraint == LayoutConstraint.Fixed))
|
|
{
|
|
return;
|
|
}
|
|
if (trigger == InvalidationTrigger.HorizontalOptionsChanged || trigger == InvalidationTrigger.VerticalOptionsChanged)
|
|
{
|
|
ComputeConstraintForView(view);
|
|
}
|
|
}
|
|
|
|
_allocatedFlag = false;
|
|
if (trigger == InvalidationTrigger.RendererReady)
|
|
{
|
|
InvalidateMeasureInternal(InvalidationTrigger.RendererReady);
|
|
}
|
|
else
|
|
{
|
|
InvalidateMeasureInternal(InvalidationTrigger.MeasureChanged);
|
|
}
|
|
|
|
s_resolutionList.Add(new KeyValuePair<Layout, int>(this, GetElementDepth(this)));
|
|
if (!s_relayoutInProgress)
|
|
{
|
|
s_relayoutInProgress = true;
|
|
|
|
// Rather than recomputing the layout for each change as it happens, we accumulate them in
|
|
// s_resolutionList and schedule a single layout update operation to handle them all at once.
|
|
// This avoids a lot of unnecessary layout operations if something is triggering many property
|
|
// changes at once (e.g., a BindingContext change)
|
|
|
|
Device.BeginInvokeOnMainThread(() =>
|
|
{
|
|
// if thread safety mattered we would need to lock this and compareexchange above
|
|
IList<KeyValuePair<Layout, int>> copy = s_resolutionList;
|
|
s_resolutionList = new List<KeyValuePair<Layout, int>>();
|
|
s_relayoutInProgress = false;
|
|
|
|
foreach (KeyValuePair<Layout, int> kvp in copy.OrderBy(kvp => kvp.Value))
|
|
{
|
|
Layout layout = kvp.Key;
|
|
double width = layout.Width, height = layout.Height;
|
|
if (!layout._allocatedFlag && width >= 0 && height >= 0)
|
|
{
|
|
layout.SizeAllocated(width, height);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
internal override void OnIsVisibleChanged(bool oldValue, bool newValue)
|
|
{
|
|
base.OnIsVisibleChanged(oldValue, newValue);
|
|
if (newValue)
|
|
{
|
|
if (_lastLayoutSize != new Size(Width, Height))
|
|
{
|
|
UpdateChildrenLayout();
|
|
}
|
|
}
|
|
}
|
|
|
|
static int GetElementDepth(Element view)
|
|
{
|
|
var result = 0;
|
|
while (view.Parent != null)
|
|
{
|
|
result++;
|
|
view = view.Parent;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
void InternalChildrenOnCollectionChanged(object sender, NotifyCollectionChangedEventArgs e)
|
|
{
|
|
if (e.Action == NotifyCollectionChangedAction.Move)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (e.OldItems != null)
|
|
{
|
|
for (int i = 0; i < e.OldItems.Count; i++)
|
|
{
|
|
object item = e.OldItems[i];
|
|
var v = item as View;
|
|
if (v == null)
|
|
continue;
|
|
|
|
OnInternalRemoved(v);
|
|
}
|
|
}
|
|
|
|
if (e.NewItems != null)
|
|
{
|
|
for (int i = 0; i < e.NewItems.Count; i++)
|
|
{
|
|
object item = e.NewItems[i];
|
|
var v = item as View;
|
|
if (v == null)
|
|
continue;
|
|
|
|
if (item == this)
|
|
throw new InvalidOperationException("Can not add self to own child collection.");
|
|
|
|
OnInternalAdded(v);
|
|
}
|
|
}
|
|
}
|
|
|
|
void OnInternalAdded(View view)
|
|
{
|
|
var parent = view.Parent as Layout;
|
|
parent?.InternalChildren.Remove(view);
|
|
|
|
OnChildAdded(view);
|
|
if (ShouldInvalidateOnChildAdded(view))
|
|
InvalidateLayout();
|
|
|
|
view.MeasureInvalidated += OnChildMeasureInvalidated;
|
|
}
|
|
|
|
void OnInternalRemoved(View view)
|
|
{
|
|
view.MeasureInvalidated -= OnChildMeasureInvalidated;
|
|
|
|
OnChildRemoved(view);
|
|
if (ShouldInvalidateOnChildRemoved(view))
|
|
InvalidateLayout();
|
|
}
|
|
|
|
bool ShouldLayoutChildren()
|
|
{
|
|
if (Width <= 0 || Height <= 0 || !LogicalChildrenInternal.Any() || !IsVisible || !IsNativeStateConsistent || DisableLayout)
|
|
return false;
|
|
|
|
foreach (Element element in VisibleDescendants())
|
|
{
|
|
var visual = element as VisualElement;
|
|
if (visual == null || !visual.IsVisible)
|
|
continue;
|
|
|
|
if (!visual.IsPlatformEnabled || !visual.IsNativeStateConsistent)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
}
|