From 5ea000a76d03a5328fe18761fbdf8b05e26a170b Mon Sep 17 00:00:00 2001 From: Stephane Delcroix Date: Fri, 7 Apr 2017 09:48:17 +0200 Subject: [PATCH] [Xaml] Set the TargetProperty on ServiceProvider (#847) --- .../BaseTestFixture.cs | 4 +- .../Issues/Bz53275.xaml | 8 +++ .../Issues/Bz53275.xaml.cs | 57 ++++++++++++++++ .../TypeExtension.xaml.cs | 13 ++++ .../Xamarin.Forms.Xaml.UnitTests.csproj | 6 ++ Xamarin.Forms.Xaml/ApplyPropertiesVisitor.cs | 67 ++++++++++++++----- Xamarin.Forms.Xaml/XamlServiceProvider.cs | 2 +- 7 files changed, 140 insertions(+), 17 deletions(-) create mode 100644 Xamarin.Forms.Xaml.UnitTests/Issues/Bz53275.xaml create mode 100644 Xamarin.Forms.Xaml.UnitTests/Issues/Bz53275.xaml.cs diff --git a/Xamarin.Forms.Core.UnitTests/BaseTestFixture.cs b/Xamarin.Forms.Core.UnitTests/BaseTestFixture.cs index 69c3e32aa..23b7c2db1 100644 --- a/Xamarin.Forms.Core.UnitTests/BaseTestFixture.cs +++ b/Xamarin.Forms.Core.UnitTests/BaseTestFixture.cs @@ -11,6 +11,8 @@ namespace Xamarin.Forms.Core.UnitTests [SetUp] public virtual void Setup () { + Device.PlatformServices = new MockPlatformServices(); + #if !WINDOWS_PHONE var culture = Environment.GetEnvironmentVariable ("UNIT_TEST_CULTURE"); @@ -24,7 +26,7 @@ namespace Xamarin.Forms.Core.UnitTests [TearDown] public virtual void TearDown () { - + Device.PlatformServices = null; } } } diff --git a/Xamarin.Forms.Xaml.UnitTests/Issues/Bz53275.xaml b/Xamarin.Forms.Xaml.UnitTests/Issues/Bz53275.xaml new file mode 100644 index 000000000..cab52a021 --- /dev/null +++ b/Xamarin.Forms.Xaml.UnitTests/Issues/Bz53275.xaml @@ -0,0 +1,8 @@ + + + \ No newline at end of file diff --git a/Xamarin.Forms.Xaml.UnitTests/Issues/Bz53275.xaml.cs b/Xamarin.Forms.Xaml.UnitTests/Issues/Bz53275.xaml.cs new file mode 100644 index 000000000..ccef92dda --- /dev/null +++ b/Xamarin.Forms.Xaml.UnitTests/Issues/Bz53275.xaml.cs @@ -0,0 +1,57 @@ +using System; +using System.Reflection; +using NUnit.Framework; +using Xamarin.Forms.Core.UnitTests; + +namespace Xamarin.Forms.Xaml.UnitTests +{ + public class TargetPropertyExtension : IMarkupExtension + { + public object ProvideValue(IServiceProvider serviceProvider) + { + var targetProperty = (serviceProvider?.GetService(typeof(IProvideValueTarget)) as IProvideValueTarget)?.TargetProperty; + return (targetProperty as BindableProperty)?.PropertyName ?? (targetProperty as PropertyInfo)?.Name; + } + } + + public partial class Bz53275 : ContentPage + { + public Bz53275() + { + InitializeComponent(); + } + + public Bz53275(bool useCompiledXaml) + { + //this stub will be replaced at compile time + } + + public string ANonBindableProperty { get; set; } + + [TestFixture] + class Tests + { + [SetUp] + public void Setup() + { + Device.PlatformServices = new MockPlatformServices(); + } + + [TearDown] + public void TearDown() + { + Device.PlatformServices = null; + } + + [TestCase(true)] + [TestCase(false)] + public void TargetPropertyIsSetOnMarkups(bool useCompiledXaml) + { + var page = new Bz53275(useCompiledXaml); + Assert.AreEqual("ANonBindableProperty", page.ANonBindableProperty); + var l0 = page.label; + Assert.AreEqual("Text", l0.Text); + } + } + } +} \ No newline at end of file diff --git a/Xamarin.Forms.Xaml.UnitTests/TypeExtension.xaml.cs b/Xamarin.Forms.Xaml.UnitTests/TypeExtension.xaml.cs index 6f9ef80a6..f1e0cd9a6 100644 --- a/Xamarin.Forms.Xaml.UnitTests/TypeExtension.xaml.cs +++ b/Xamarin.Forms.Xaml.UnitTests/TypeExtension.xaml.cs @@ -3,6 +3,7 @@ using System.Collections.Generic; using System.Windows.Input; using Xamarin.Forms; using NUnit.Framework; +using Xamarin.Forms.Core.UnitTests; namespace Xamarin.Forms.Xaml.UnitTests { @@ -46,6 +47,18 @@ namespace Xamarin.Forms.Xaml.UnitTests [TestFixture] public class Tests { + [SetUp] + public void Setup() + { + Device.PlatformServices = new MockPlatformServices(); + } + + [TearDown] + public void TearDown() + { + Device.PlatformServices = null; + } + [TestCase(false)] [TestCase(true)] public void NestedMarkupExtensionInsideDataTemplate(bool useCompiledXaml) diff --git a/Xamarin.Forms.Xaml.UnitTests/Xamarin.Forms.Xaml.UnitTests.csproj b/Xamarin.Forms.Xaml.UnitTests/Xamarin.Forms.Xaml.UnitTests.csproj index 24d2bbde2..a2c9d4960 100644 --- a/Xamarin.Forms.Xaml.UnitTests/Xamarin.Forms.Xaml.UnitTests.csproj +++ b/Xamarin.Forms.Xaml.UnitTests/Xamarin.Forms.Xaml.UnitTests.csproj @@ -470,6 +470,9 @@ Bz41048.xaml + + Bz53275.xaml + @@ -860,6 +863,9 @@ MSBuild:UpdateDesignTimeXaml + + MSBuild:UpdateDesignTimeXaml + diff --git a/Xamarin.Forms.Xaml/ApplyPropertiesVisitor.cs b/Xamarin.Forms.Xaml/ApplyPropertiesVisitor.cs index 337b2ace4..af7acafed 100644 --- a/Xamarin.Forms.Xaml/ApplyPropertiesVisitor.cs +++ b/Xamarin.Forms.Xaml/ApplyPropertiesVisitor.cs @@ -77,7 +77,8 @@ namespace Xamarin.Forms.Xaml public void Visit(ElementNode node, INode parentNode) { var propertyName = XmlName.Empty; - if (TryGetPropertyName(node, parentNode, out propertyName) && propertyName == XmlName._CreateContent){ + + if (TryGetPropertyName(node, parentNode, out propertyName) && propertyName == XmlName._CreateContent) { var s0 = Values[parentNode]; if (s0 is ElementTemplate) { SetTemplate(s0 as ElementTemplate, node); @@ -85,21 +86,7 @@ namespace Xamarin.Forms.Xaml } } - var value = Values [node]; var parentElement = parentNode as IElementNode; - var markupExtension = value as IMarkupExtension; - var valueProvider = value as IValueProvider; - - if (markupExtension != null) { - var serviceProvider = value.GetType().GetTypeInfo().GetCustomAttribute() == null ? new XamlServiceProvider(node, Context) : null; - value = markupExtension.ProvideValue(serviceProvider); - } - - if (valueProvider != null) { - var serviceProvider = value.GetType().GetTypeInfo().GetCustomAttribute() == null ? new XamlServiceProvider(node, Context) : null; - value = valueProvider.ProvideValue(serviceProvider); - } - propertyName = XmlName.Empty; //Simplify ListNodes with single elements @@ -110,6 +97,15 @@ namespace Xamarin.Forms.Xaml parentElement = parentNode as IElementNode; } + var value = Values[node]; + + var markupExtension = value as IMarkupExtension; + var valueProvider = value as IValueProvider; + XamlServiceProvider serviceProvider = null; + if (markupExtension != null || valueProvider != null) + serviceProvider = value.GetType().GetTypeInfo().GetCustomAttribute() == null ? new XamlServiceProvider(node, Context) : null; + + if (propertyName != XmlName.Empty || TryGetPropertyName(node, parentNode, out propertyName)) { if (Skips.Contains(propertyName)) return; @@ -117,8 +113,21 @@ namespace Xamarin.Forms.Xaml return; var source = Values [parentNode]; + if (serviceProvider != null) + ((XamlValueTargetProvider)serviceProvider.IProvideValueTarget).TargetProperty = GetTargetProperty(source, propertyName, Context, node); + + if (markupExtension != null) + value = markupExtension.ProvideValue(serviceProvider); + else if (valueProvider != null) + value = valueProvider.ProvideValue(serviceProvider); + SetPropertyValue(source, propertyName, value, Context.RootElement, node, Context, node); } else if (IsCollectionItem(node, parentNode) && parentNode is IElementNode) { + if (markupExtension != null) + value = markupExtension.ProvideValue(serviceProvider); + else if (valueProvider != null) + value = valueProvider.ProvideValue(serviceProvider); + // Collection element, implicit content, or implicit collection element. string contentProperty; if (typeof(IEnumerable).GetTypeInfo().IsAssignableFrom(Context.Types [parentElement].GetTypeInfo()) && Context.Types[parentElement].GetRuntimeMethods().Any(mi => mi.Name == "Add" && mi.GetParameters().Length == 1)) { @@ -140,6 +149,11 @@ namespace Xamarin.Forms.Xaml } else throw new XamlParseException($"Can not set the content of {((IElementNode)parentNode).XmlType.Name} as it doesn't have a ContentPropertyAttribute", node); } else if (IsCollectionItem(node, parentNode) && parentNode is ListNode) { + if (markupExtension != null) + value = markupExtension.ProvideValue(serviceProvider); + else if (valueProvider != null) + value = valueProvider.ProvideValue(serviceProvider); + var parentList = (ListNode)parentNode; var source = Values [parentNode.Parent]; @@ -269,6 +283,22 @@ namespace Xamarin.Forms.Xaml return null; } + static object GetTargetProperty(object xamlelement, XmlName propertyName, HydratationContext context, IXmlLineInfo lineInfo) + { + var localName = propertyName.LocalName; + //If it's an attached BP, update elementType and propertyName + var bpOwnerType = xamlelement.GetType(); + GetRealNameAndType(ref bpOwnerType, propertyName.NamespaceURI, ref localName, context, lineInfo); + var property = GetBindableProperty(bpOwnerType, localName, lineInfo, false); + + if (property != null) + return property; + + var elementType = xamlelement.GetType(); + var propertyInfo = elementType.GetRuntimeProperties().FirstOrDefault(p => p.Name == localName); + return propertyInfo; + } + public static void SetPropertyValue(object xamlelement, XmlName propertyName, object value, object rootElement, INode node, HydratationContext context, IXmlLineInfo lineInfo) { var localName = propertyName.LocalName; @@ -397,6 +427,9 @@ namespace Xamarin.Forms.Xaml if (property == null) return false; + if (serviceProvider != null && serviceProvider.IProvideValueTarget != null) + ((XamlValueTargetProvider)serviceProvider.IProvideValueTarget).TargetProperty = property; + Func minforetriever; if (attached) minforetriever = () => property.DeclaringType.GetRuntimeMethod("Get" + property.PropertyName, new [] { typeof(BindableObject) }); @@ -436,6 +469,9 @@ namespace Xamarin.Forms.Xaml if (!IsVisibleFrom(setter, context.RootElement)) return false; + if (serviceProvider != null && serviceProvider.IProvideValueTarget != null) + ((XamlValueTargetProvider)serviceProvider.IProvideValueTarget).TargetProperty = propertyInfo; + object convertedValue = value.ConvertTo(propertyInfo.PropertyType, () => propertyInfo, serviceProvider); if (convertedValue != null && !propertyInfo.PropertyType.IsInstanceOfType(convertedValue)) return false; @@ -475,6 +511,7 @@ namespace Xamarin.Forms.Xaml if (addMethod == null) return false; + ((XamlValueTargetProvider)serviceProvider.IProvideValueTarget).TargetProperty = propertyInfo; addMethod.Invoke(collection, new [] { value.ConvertTo(addMethod.GetParameters() [0].ParameterType, (Func)null, serviceProvider) }); return true; } diff --git a/Xamarin.Forms.Xaml/XamlServiceProvider.cs b/Xamarin.Forms.Xaml/XamlServiceProvider.cs index 5d99f952c..674795ff3 100644 --- a/Xamarin.Forms.Xaml/XamlServiceProvider.cs +++ b/Xamarin.Forms.Xaml/XamlServiceProvider.cs @@ -103,7 +103,7 @@ namespace Xamarin.Forms.Xaml.Internals HydratationContext Context { get; } public object TargetObject { get; } - public object TargetProperty { get; } = null; + public object TargetProperty { get; internal set; } = null; IEnumerable IProvideParentValues.ParentObjects {