Skip to content

Commit

Permalink
Revert "[X] do not apply Bindings if DataType doesnt match (#22056)" (#…
Browse files Browse the repository at this point in the history
…24049)

This reverts commit cb0a332.

# Conflicts:
#	src/Controls/src/Core/Binding.cs
#	src/Controls/src/Xaml/MarkupExtensions/BindingExtension.cs
#	src/Controls/src/Xaml/XamlServiceProvider.cs
  • Loading branch information
PureWeen authored Aug 6, 2024
1 parent 2ac35ae commit 8f5f161
Show file tree
Hide file tree
Showing 5 changed files with 8 additions and 91 deletions.
14 changes: 3 additions & 11 deletions src/Controls/src/Core/Binding.cs
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
using System.ComponentModel;
using System.Diagnostics.CodeAnalysis;
using System.Globalization;
using Microsoft.Maui.Controls.Xaml.Diagnostics;
using System.Reflection;

namespace Microsoft.Maui.Controls
{
Expand Down Expand Up @@ -106,8 +106,6 @@ public string UpdateSourceEventName
}
}

internal Type DataType { get; set; }

internal override void Apply(bool fromTarget)
{
base.Apply(fromTarget);
Expand All @@ -123,14 +121,7 @@ internal override void Apply(object context, BindableObject bindObj, BindablePro
object src = _source;
var isApplied = IsApplied;

var bindingContext = src ?? Context ?? context;
if (DataType != null && bindingContext != null && !DataType.IsAssignableFrom(bindingContext.GetType()))
{
BindingDiagnostics.SendBindingFailure(this, "Binding", "Mismatch between the specified x:DataType and the current binding context");
bindingContext = null;
}

base.Apply(bindingContext, bindObj, targetProperty, fromBindingContextChanged, specificity);
base.Apply(src ?? context, bindObj, targetProperty, fromBindingContextChanged, specificity);

if (src != null && isApplied && fromBindingContextChanged)
return;
Expand All @@ -141,6 +132,7 @@ internal override void Apply(object context, BindableObject bindObj, BindablePro
}
else
{
object bindingContext = src ?? Context ?? context;
if (_expression == null)
_expression = new BindingExpression(this, SelfPath);
_expression.Apply(bindingContext, bindObj, targetProperty, specificity);
Expand Down
5 changes: 0 additions & 5 deletions src/Controls/src/Core/IXamlDataTypeProvider.cs

This file was deleted.

10 changes: 1 addition & 9 deletions src/Controls/src/Xaml/MarkupExtensions/BindingExtension.cs
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@
namespace Microsoft.Maui.Controls.Xaml
{
[ContentProperty(nameof(Path))]
[RequireService([typeof(IXamlTypeResolver), typeof(IXamlDataTypeProvider)])]
[AcceptEmptyServiceProvider]
public sealed class BindingExtension : IMarkupExtension<BindingBase>
{
public string Path { get; set; } = Binding.SelfPath;
Expand Down Expand Up @@ -42,19 +42,11 @@ BindingBase IMarkupExtension<BindingBase>.ProvideValue(IServiceProvider serviceP
"In that case, we produce a warning that the binding could not be compiled.")]
BindingBase CreateBinding()
{
Type bindingXDataType = null;
if ((serviceProvider.GetService(typeof(IXamlTypeResolver)) is IXamlTypeResolver typeResolver)
&& (serviceProvider.GetService(typeof(IXamlDataTypeProvider)) is IXamlDataTypeProvider dataTypeProvider)
&& dataTypeProvider.BindingDataType != null)
{
typeResolver.TryResolve(dataTypeProvider.BindingDataType, out bindingXDataType);
}
return new Binding(Path, Mode, Converter, ConverterParameter, StringFormat, Source)
{
UpdateSourceEventName = UpdateSourceEventName,
FallbackValue = FallbackValue,
TargetNullValue = TargetNullValue,
DataType = bindingXDataType,
};
}
}
Expand Down
56 changes: 0 additions & 56 deletions src/Controls/src/Xaml/XamlServiceProvider.cs
Original file line number Diff line number Diff line change
Expand Up @@ -34,9 +34,6 @@ internal XamlServiceProvider(INode node, HydrationContext context)
IXmlLineInfoProvider = new XmlLineInfoProvider(xmlLineInfo);

IValueConverterProvider = defaultValueConverterProvider;

if (node is IElementNode elementNode)
Add(typeof(IXamlDataTypeProvider), new XamlDataTypeProvider(elementNode));
}

public XamlServiceProvider() => IValueConverterProvider = defaultValueConverterProvider;
Expand Down Expand Up @@ -292,57 +289,4 @@ public string LookupNamespace(string prefix)
public string LookupPrefix(string namespaceName) => throw new NotImplementedException();
public void Add(string prefix, string ns) => namespaces.Add(prefix, ns);
}

class XamlDataTypeProvider : IXamlDataTypeProvider
{
public XamlDataTypeProvider(IElementNode node)
{
static IElementNode GetParent(IElementNode node)
{
return node switch
{
{ Parent: ListNode { Parent: IElementNode parentNode } } => parentNode,
{ Parent: IElementNode parentNode } => parentNode,
_ => null,
};
}

static bool IsBindingContextBinding(IElementNode node)
{
if ( node.TryGetPropertyName(node.Parent, out XmlName name)
&& name.NamespaceURI == ""
&& name.LocalName == nameof(BindableObject.BindingContext))
return true;
return false;
}

INode dataTypeNode = null;
IElementNode n = node as IElementNode;

// Special handling for BindingContext={Binding ...}
// The order of checks is:
// - x:DataType on the binding itself
// - SKIP looking for x:DataType on the parent
// - continue looking for x:DataType on the parent's parent...
IElementNode skipNode = null;
if (IsBindingContextBinding(node))
{
skipNode = GetParent(node);
}

while (n != null)
{
if (n != skipNode && n.Properties.TryGetValue(XmlName.xDataType, out dataTypeNode))
{
break;
}

n = GetParent(n);
}
if (dataTypeNode is ValueNode valueNode)
BindingDataType = valueNode.Value as string;

}
public string BindingDataType { get; }
}
}
14 changes: 4 additions & 10 deletions src/Controls/tests/Xaml.UnitTests/BindingsCompiler.xaml.cs
Original file line number Diff line number Diff line change
Expand Up @@ -29,9 +29,10 @@ public class Tests
{
[SetUp] public void Setup() => DispatcherProvider.SetCurrent(new DispatcherProviderStub());
[TearDown] public void TearDown() => DispatcherProvider.SetCurrent(null);

[Test]
public void TestCompiledBindings([Values(false, true)]bool useCompiledXaml)

[TestCase(false)]
[TestCase(true)]
public void Test(bool useCompiledXaml)
{
if (useCompiledXaml)
MockCompiler.Compile(typeof(BindingsCompiler));
Expand Down Expand Up @@ -123,13 +124,6 @@ public void TestCompiledBindings([Values(false, true)]bool useCompiledXaml)
//testing source
Assert.That(layout.label12.Text, Is.EqualTo("Text for label12"));
}

[Test]
public void BindingsNotAppliedWithWrongContext([Values(false, true)]bool useCompiledXaml)
{
var page = new BindingsCompiler(useCompiledXaml) { BindingContext = new {Text="Foo"} };
Assert.AreEqual(null, page.label0.Text);
}
}
}

Expand Down

0 comments on commit 8f5f161

Please sign in to comment.