From fb5b405250333b088d5806d956191bc93fd59ff0 Mon Sep 17 00:00:00 2001 From: michael-hawker <24302614+michael-hawker@users.noreply.github.com> Date: Tue, 7 Sep 2021 17:27:21 -0700 Subject: [PATCH] Fix #4214 - Update AttachedDropShadow visual when element layout changes and visibility changes --- .../Shadows/AttachedDropShadow.cs | 72 +++++++++++++++++-- .../Shadows/AttachedShadowElementContext.cs | 2 + 2 files changed, 70 insertions(+), 4 deletions(-) diff --git a/Microsoft.Toolkit.Uwp.UI/Shadows/AttachedDropShadow.cs b/Microsoft.Toolkit.Uwp.UI/Shadows/AttachedDropShadow.cs index 070603cb7d0..f6fe707a7cf 100644 --- a/Microsoft.Toolkit.Uwp.UI/Shadows/AttachedDropShadow.cs +++ b/Microsoft.Toolkit.Uwp.UI/Shadows/AttachedDropShadow.cs @@ -2,6 +2,7 @@ // The .NET Foundation licenses this file to you under the MIT license. // See the LICENSE file in the project root for more information. +using System; using System.Linq; using System.Numerics; using Windows.Foundation; @@ -166,6 +167,16 @@ protected internal override void OnElementContextUninitialized(AttachedShadowEle _container.Children.Remove(context.SpriteVisual); } + context.SpriteVisual?.StopAnimation("Size"); + + context.Element.LayoutUpdated -= Element_LayoutUpdated; + + if (context.VisibilityToken != null) + { + context.Element.UnregisterPropertyChangedCallback(UIElement.VisibilityProperty, context.VisibilityToken.Value); + context.VisibilityToken = null; + } + base.OnElementContextUninitialized(context); } @@ -176,6 +187,50 @@ protected override void SetElementChildVisual(AttachedShadowElementContext conte { _container.Children.InsertAtTop(context.SpriteVisual); } + + // Handles size changing and other elements around it updating. + context.Element.LayoutUpdated -= Element_LayoutUpdated; + context.Element.LayoutUpdated += Element_LayoutUpdated; + + if (context.VisibilityToken != null) + { + context.Element.UnregisterPropertyChangedCallback(UIElement.VisibilityProperty, context.VisibilityToken.Value); + context.VisibilityToken = null; + } + + context.VisibilityToken = context.Element.RegisterPropertyChangedCallback(UIElement.VisibilityProperty, Element_VisibilityChanged); + } + + private void Element_LayoutUpdated(object sender, object e) + { + // Update other shadows to account for layout changes + CastToElement_SizeChanged(null, null); + } + + private void Element_VisibilityChanged(DependencyObject sender, DependencyProperty dp) + { + if (sender is FrameworkElement element) + { + var context = GetElementContext(element); + + if (element.Visibility == Visibility.Collapsed) + { + if (_container != null && _container.Children.Contains(context.SpriteVisual)) + { + _container.Children.Remove(context.SpriteVisual); + } + } + else + { + if (_container != null && !_container.Children.Contains(context.SpriteVisual)) + { + _container.Children.InsertAtTop(context.SpriteVisual); + } + } + } + + // Update other shadows to account for layout changes + CastToElement_SizeChanged(null, null); } /// @@ -255,12 +310,24 @@ protected override CompositionBrush GetShadowMask(AttachedShadowElementContext c } // Position our shadow in the correct spot to match the corresponding element. - context.SpriteVisual.Size = context.Element.RenderSize.ToVector2(); context.SpriteVisual.Offset = context.Element.CoordinatesFrom(CastTo).ToVector3(); + BindSizeAndScale(context.SpriteVisual, context.Element); + return mask; } + private static void BindSizeAndScale(CompositionObject source, UIElement target) + { + var visual = ElementCompositionPreview.GetElementVisual(target); + var bindSizeAnimation = source.Compositor.CreateExpressionAnimation($"{nameof(visual)}.Size * {nameof(visual)}.Scale.XY"); + + bindSizeAnimation.SetReferenceParameter(nameof(visual), visual); + + // Start the animation + source.StartAnimation("Size", bindSizeAnimation); + } + private void CustomMaskedElement_Loaded(object sender, RoutedEventArgs e) { var context = GetElementContext(sender as FrameworkElement); @@ -274,9 +341,6 @@ private void CustomMaskedElement_Loaded(object sender, RoutedEventArgs e) /// protected internal override void OnSizeChanged(AttachedShadowElementContext context, Size newSize, Size previousSize) { - var sizeAsVec2 = newSize.ToVector2(); - - context.SpriteVisual.Size = sizeAsVec2; context.SpriteVisual.Offset = context.Element.CoordinatesFrom(CastTo).ToVector3(); UpdateShadowClip(context); diff --git a/Microsoft.Toolkit.Uwp.UI/Shadows/AttachedShadowElementContext.cs b/Microsoft.Toolkit.Uwp.UI/Shadows/AttachedShadowElementContext.cs index 1d8803d23af..63afa4581e3 100644 --- a/Microsoft.Toolkit.Uwp.UI/Shadows/AttachedShadowElementContext.cs +++ b/Microsoft.Toolkit.Uwp.UI/Shadows/AttachedShadowElementContext.cs @@ -20,6 +20,8 @@ public sealed class AttachedShadowElementContext private Dictionary _resources; + internal long? VisibilityToken { get; set; } + /// /// Gets a value indicating whether or not this has been initialized. ///