maui-linux/Xamarin.Forms.Xaml.UnitTests/MarkupExpressionParserTests.cs

383 строки
12 KiB
C#
Исходник Обычный вид История

2016-03-22 23:02:25 +03:00
using System;
using System.Collections.Generic;
using System.Reflection;
Implement OnPlatform/OnIdiom markup extensions (#2615) * Added OnPlatform markup extension supporting iOS/Android/UWP for #2608 * Add Default and Other properties to OnPlatformExtension This allows setting a default value for unknown platforms, as well as specify values for arbitrary platforms by using a named parameter like syntax: `{OnPlatform Android=15, iOS=10, UWP=12, Default=11, Other=Tizen:20}` The `Other` supports multiple semi-colon separated values. By using this format, we can make the string more concise than if we used `=` which would have to be escaped in quotes. For example: `{OnPlatform Default=10, Other=Tizen:22;Xbox:20;Switch=25;PlayStation=22}` Added unit tests that verify all the supported combinations. * Add OnIdiomExtension The extension allows the following syntax: `{OnIdiom Phone=23, Tablet=25, Desktop=26, TV=30, Watch=10, Default=20}` At least one value or `Default` must be specified. `Default` is returned whenever the specific idiom was not specified. * Add missing known platforms and return Default if missing Add all strings that are provided in `Device`. Convert individual asserts into test cases for better reporting. Also, whenever a matching platform value isn't specified, return Default instead of null. * Turn OnIdiom asserts into test cases This makes for easier to read, document and report tests. * Add missing platforms to null check * Make Default the content property * Perform type conversion as expected by XamlC Leverage the conversion that is used elsewhere, to return a properly typed object that can be assigned directly to the property. Update tests with typed values since now we get integers, rather than strings out of the parser. * Add Converter/ConverterParameter support * Message should state that the value must be non-null * Remove Unsupported idiom since it's not useful to set You can use Default instead. * Use new GetService<T> extension method for conciseness * Don't fail if service provider is null * Remove Other from OnPlatformExtension As suggested, this might come back in the future in some other form.
2018-05-14 16:56:20 +03:00
using System.Xml;
using NUnit.Framework;
using Xamarin.Forms.Core.UnitTests;
2016-03-22 23:02:25 +03:00
namespace Xamarin.Forms.Xaml.UnitTests
{
[TestFixture]
public class MarkupExpressionParserTests : BaseTestFixture
{
IXamlTypeResolver typeResolver;
class MockElementNode : IElementNode, IValueNode, IXmlLineInfo
{
public bool HasLineInfo () { return false; }
public int LineNumber {
get { return -1; }
}
public int LinePosition {
get { return -1; }
}
public IXmlNamespaceResolver NamespaceResolver {
get {
throw new NotImplementedException ();
}
}
public object Value {get;set;}
public Dictionary<XmlName, INode> Properties { get; set; }
public List<XmlName> SkipProperties { get; set; }
public Forms.Internals.INameScope Namescope {
get {
throw new NotImplementedException ();
}
}
public XmlType XmlType {
get;
set;
}
public string NamespaceURI {
get {
throw new NotImplementedException ();
}
}
public INode Parent {
get {
throw new NotImplementedException ();
}
set { throw new NotImplementedException (); }
}
public List<INode> CollectionItems { get; set; }
public void Accept (IXamlNodeVisitor visitor, INode parentNode)
{
throw new NotImplementedException ();
}
2016-03-22 23:02:25 +03:00
public List<string> IgnorablePrefixes { get; set; }
public INode Clone()
{
throw new NotImplementedException();
}
2016-03-22 23:02:25 +03:00
}
[SetUp]
public override void Setup ()
{
base.Setup ();
var nsManager = new XmlNamespaceManager (new NameTable ());
nsManager.AddNamespace ("local", "clr-namespace:Xamarin.Forms.Xaml.UnitTests;assembly=Xamarin.Forms.Xaml.UnitTests");
nsManager.AddNamespace ("x", "http://schemas.microsoft.com/winfx/2009/xaml");
typeResolver = new Internals.XamlTypeResolver (nsManager, XamlParser.GetElementType, Assembly.GetCallingAssembly ());
}
[Test]
public void BindingOnSelf ()
{
var bindingString = "{Binding}";
var binding = (new MarkupExtensionParser ()).ParseExpression (ref bindingString, new Internals.XamlServiceProvider (null, null) {
IXamlTypeResolver = typeResolver,
});
Assert.That (binding, Is.InstanceOf<Binding> ());
Assert.AreEqual (Binding.SelfPath, ((Binding)binding).Path);
}
[Test]
public void BindingWithImplicitPath ()
{
var bindingString = "{Binding Foo}";
var binding = (new MarkupExtensionParser ()).ParseExpression (ref bindingString, new Internals.XamlServiceProvider (null, null) {
IXamlTypeResolver = typeResolver,
});
Assert.That (binding, Is.InstanceOf<Binding> ());
Assert.AreEqual ("Foo", ((Binding)binding).Path);
}
[Test]
public void BindingWithPath ()
{
var bindingString = "{Binding Path=Foo}";
var binding = (new MarkupExtensionParser ()).ParseExpression (ref bindingString, new Internals.XamlServiceProvider (null, null) {
IXamlTypeResolver = typeResolver,
});
Assert.That (binding, Is.InstanceOf<Binding> ());
Assert.AreEqual ("Foo", ((Binding)binding).Path);
}
[Test]
public void BindingWithComposedPath ()
{
var bindingString = "{Binding Path=Foo.Bar}";
var binding = (new MarkupExtensionParser ()).ParseExpression (ref bindingString, new Internals.XamlServiceProvider (null, null) {
IXamlTypeResolver = typeResolver,
});
Assert.That (binding, Is.InstanceOf<Binding> ());
Assert.AreEqual ("Foo.Bar", ((Binding)binding).Path);
}
[Test]
public void BindingWithImplicitComposedPath ()
{
var bindingString = "{Binding Path=Foo.Bar}";
var binding = (new MarkupExtensionParser ()).ParseExpression (ref bindingString, new Internals.XamlServiceProvider (null, null) {
IXamlTypeResolver = typeResolver,
});
Assert.That (binding, Is.InstanceOf<Binding> ());
Assert.AreEqual ("Foo.Bar", ((Binding)binding).Path);
}
class MockValueProvider : IProvideParentValues, IProvideValueTarget
{
public MockValueProvider (string key, object resource)
{
var rd = new ResourceDictionary {
{key, resource}
};
ve = new VisualElement {
Resources = rd,
};
}
VisualElement ve;
public IEnumerable<object> ParentObjects {
get {
yield return ve;
}
}
public object TargetObject => null;
2016-03-22 23:02:25 +03:00
Implement OnPlatform/OnIdiom markup extensions (#2615) * Added OnPlatform markup extension supporting iOS/Android/UWP for #2608 * Add Default and Other properties to OnPlatformExtension This allows setting a default value for unknown platforms, as well as specify values for arbitrary platforms by using a named parameter like syntax: `{OnPlatform Android=15, iOS=10, UWP=12, Default=11, Other=Tizen:20}` The `Other` supports multiple semi-colon separated values. By using this format, we can make the string more concise than if we used `=` which would have to be escaped in quotes. For example: `{OnPlatform Default=10, Other=Tizen:22;Xbox:20;Switch=25;PlayStation=22}` Added unit tests that verify all the supported combinations. * Add OnIdiomExtension The extension allows the following syntax: `{OnIdiom Phone=23, Tablet=25, Desktop=26, TV=30, Watch=10, Default=20}` At least one value or `Default` must be specified. `Default` is returned whenever the specific idiom was not specified. * Add missing known platforms and return Default if missing Add all strings that are provided in `Device`. Convert individual asserts into test cases for better reporting. Also, whenever a matching platform value isn't specified, return Default instead of null. * Turn OnIdiom asserts into test cases This makes for easier to read, document and report tests. * Add missing platforms to null check * Make Default the content property * Perform type conversion as expected by XamlC Leverage the conversion that is used elsewhere, to return a properly typed object that can be assigned directly to the property. Update tests with typed values since now we get integers, rather than strings out of the parser. * Add Converter/ConverterParameter support * Message should state that the value must be non-null * Remove Unsupported idiom since it's not useful to set You can use Default instead. * Use new GetService<T> extension method for conciseness * Don't fail if service provider is null * Remove Other from OnPlatformExtension As suggested, this might come back in the future in some other form.
2018-05-14 16:56:20 +03:00
public object TargetProperty { get; set; } = null;
2016-03-22 23:02:25 +03:00
}
[Test]
public void BindingWithImplicitPathAndConverter ()
{
var bindingString = "{Binding Foo, Converter={StaticResource Bar}}";
var binding = (new MarkupExtensionParser ()).ParseExpression (ref bindingString, new Internals.XamlServiceProvider (null, null) {
IXamlTypeResolver = typeResolver,
IProvideValueTarget = new MockValueProvider ("Bar", new ReverseConverter()),
});
Assert.That (binding, Is.InstanceOf<Binding> ());
Assert.AreEqual ("Foo", ((Binding)binding).Path);
Assert.NotNull (((Binding)binding).Converter);
Assert.That (((Binding)binding).Converter, Is.InstanceOf<ReverseConverter> ());
}
[Test]
public void BindingWithPathAndConverter ()
{
var bindingString = "{Binding Path=Foo, Converter={StaticResource Bar}}";
var binding = (new MarkupExtensionParser ()).ParseExpression (ref bindingString, new Internals.XamlServiceProvider (null, null) {
IXamlTypeResolver = typeResolver,
IProvideValueTarget = new MockValueProvider ("Bar", new ReverseConverter()),
});
Assert.That (binding, Is.InstanceOf<Binding> ());
Assert.AreEqual ("Foo", ((Binding)binding).Path);
Assert.NotNull (((Binding)binding).Converter);
Assert.That (((Binding)binding).Converter, Is.InstanceOf<ReverseConverter> ());
}
[Test]
public void TestBindingMode ()
{
var bindingString = "{Binding Foo, Mode=TwoWay}";
var binding = (new MarkupExtensionParser ()).ParseExpression (ref bindingString, new Internals.XamlServiceProvider (null, null) {
IXamlTypeResolver = typeResolver,
});
Assert.That (binding, Is.InstanceOf<Binding> ());
Assert.AreEqual ("Foo", ((Binding)binding).Path);
Assert.AreEqual (BindingMode.TwoWay, ((Binding)binding).Mode);
}
[Test]
public void BindingStringFormat ()
{
var bindingString = "{Binding Foo, StringFormat=Bar}";
var binding = (new MarkupExtensionParser ()).ParseExpression (ref bindingString, new Internals.XamlServiceProvider (null, null) {
IXamlTypeResolver = typeResolver,
});
Assert.That (binding, Is.InstanceOf<Binding> ());
Assert.AreEqual ("Foo", ((Binding)binding).Path);
Assert.AreEqual ("Bar", ((Binding)binding).StringFormat);
}
[Test]
public void BindingStringFormatWithEscapes ()
{
var bindingString = "{Binding Foo, StringFormat='{}Hello {0}'}";
var binding = (new MarkupExtensionParser ()).ParseExpression (ref bindingString, new Internals.XamlServiceProvider (null, null) {
IXamlTypeResolver = typeResolver,
});
Assert.That (binding, Is.InstanceOf<Binding> ());
Assert.AreEqual ("Foo", ((Binding)binding).Path);
Assert.AreEqual ("Hello {0}", ((Binding)binding).StringFormat);
}
[Test]
public void BindingStringFormatWithoutEscaping ()
2016-03-22 23:02:25 +03:00
{
var bindingString = "{Binding Foo, StringFormat='{0,20}'}";
2016-03-22 23:02:25 +03:00
var binding = (new MarkupExtensionParser ()).ParseExpression (ref bindingString, new Internals.XamlServiceProvider (null, null) {
2016-03-22 23:02:25 +03:00
IXamlTypeResolver = typeResolver,
});
Assert.That (binding, Is.InstanceOf<Binding> ());
Assert.AreEqual ("Foo", ((Binding)binding).Path);
Assert.AreEqual ("{0,20}", ((Binding)binding).StringFormat);
2016-03-22 23:02:25 +03:00
}
[Test]
public void BindingStringFormatNumeric ()
{
var bindingString = "{Binding Foo, StringFormat=P2}";
var binding = (new MarkupExtensionParser ()).ParseExpression (ref bindingString, new Internals.XamlServiceProvider (null, null) {
IXamlTypeResolver = typeResolver,
});
Assert.That (binding, Is.InstanceOf<Binding> ());
Assert.AreEqual ("Foo", ((Binding)binding).Path);
Assert.AreEqual ("P2", ((Binding)binding).StringFormat);
}
[Test]
public void BindingConverterParameter ()
{
var bindingString = "{Binding Foo, ConverterParameter='Bar'}";
var binding = (new MarkupExtensionParser ()).ParseExpression (ref bindingString, new Internals.XamlServiceProvider (null, null) {
IXamlTypeResolver = typeResolver,
});
Assert.That (binding, Is.InstanceOf<Binding> ());
Assert.AreEqual ("Foo", ((Binding)binding).Path);
Assert.AreEqual ("Bar", ((Binding)binding).ConverterParameter);
}
[Test]
public void BindingsCompleteString ()
{
var bindingString = "{Binding Path=Foo.Bar, StringFormat='{}Qux, {0}', Converter={StaticResource Baz}, Mode=OneWayToSource}";
var binding = (new MarkupExtensionParser ()).ParseExpression (ref bindingString, new Internals.XamlServiceProvider (null, null) {
IXamlTypeResolver = typeResolver,
IProvideValueTarget = new MockValueProvider ("Baz", new ReverseConverter()),
});
Assert.That (binding, Is.InstanceOf<Binding> ());
Assert.AreEqual ("Foo.Bar", ((Binding)binding).Path);
Assert.NotNull (((Binding)binding).Converter);
Assert.That (((Binding)binding).Converter, Is.InstanceOf<ReverseConverter> ());
Assert.AreEqual (BindingMode.OneWayToSource, ((Binding)binding).Mode);
Assert.AreEqual ("Qux, {0}", ((Binding)binding).StringFormat);
}
[Test]
public void BindingWithStaticConverter ()
{
var bindingString = "{Binding Converter={x:Static local:ReverseConverter.Instance}}";
var binding = (new MarkupExtensionParser ()).ParseExpression (ref bindingString, new Internals.XamlServiceProvider (null, null) {
IXamlTypeResolver = typeResolver,
}) as Binding;
Assert.NotNull (binding);
Assert.AreEqual(".", binding.Path);
Assert.That (binding.Converter, Is.TypeOf<ReverseConverter> ());
}
Implement OnPlatform/OnIdiom markup extensions (#2615) * Added OnPlatform markup extension supporting iOS/Android/UWP for #2608 * Add Default and Other properties to OnPlatformExtension This allows setting a default value for unknown platforms, as well as specify values for arbitrary platforms by using a named parameter like syntax: `{OnPlatform Android=15, iOS=10, UWP=12, Default=11, Other=Tizen:20}` The `Other` supports multiple semi-colon separated values. By using this format, we can make the string more concise than if we used `=` which would have to be escaped in quotes. For example: `{OnPlatform Default=10, Other=Tizen:22;Xbox:20;Switch=25;PlayStation=22}` Added unit tests that verify all the supported combinations. * Add OnIdiomExtension The extension allows the following syntax: `{OnIdiom Phone=23, Tablet=25, Desktop=26, TV=30, Watch=10, Default=20}` At least one value or `Default` must be specified. `Default` is returned whenever the specific idiom was not specified. * Add missing known platforms and return Default if missing Add all strings that are provided in `Device`. Convert individual asserts into test cases for better reporting. Also, whenever a matching platform value isn't specified, return Default instead of null. * Turn OnIdiom asserts into test cases This makes for easier to read, document and report tests. * Add missing platforms to null check * Make Default the content property * Perform type conversion as expected by XamlC Leverage the conversion that is used elsewhere, to return a properly typed object that can be assigned directly to the property. Update tests with typed values since now we get integers, rather than strings out of the parser. * Add Converter/ConverterParameter support * Message should state that the value must be non-null * Remove Unsupported idiom since it's not useful to set You can use Default instead. * Use new GetService<T> extension method for conciseness * Don't fail if service provider is null * Remove Other from OnPlatformExtension As suggested, this might come back in the future in some other form.
2018-05-14 16:56:20 +03:00
public int FontSize { get; set; }
[TestCase("{OnPlatform 20, Android=23}", Device.Android, 23)]
[TestCase("{OnPlatform Android=20, iOS=25}", Device.iOS, 25)]
[TestCase("{OnPlatform Android=20, GTK=25}", Device.GTK, 25)]
[TestCase("{OnPlatform Android=20, macOS=25}", Device.macOS, 25)]
[TestCase("{OnPlatform Android=20, Tizen=25}", Device.Tizen, 25)]
[TestCase("{OnPlatform Android=20, UWP=25}", Device.UWP, 25)]
[TestCase("{OnPlatform Android=20, WPF=25}", Device.WPF, 25)]
[TestCase("{OnPlatform 20}", Device.iOS, 20)]
[TestCase("{OnPlatform 20}", Device.GTK, 20)]
[TestCase("{OnPlatform 20}", Device.macOS, 20)]
[TestCase("{OnPlatform 20}", Device.Tizen, 20)]
[TestCase("{OnPlatform 20}", Device.UWP, 20)]
[TestCase("{OnPlatform 20}", Device.WPF, 20)]
[TestCase("{OnPlatform 20}", "Foo", 20)]
[TestCase("{OnPlatform Android=23, Default=20}", "Foo", 20)]
public void OnPlatformExtension(string markup, string platform, int expected)
{
var services = new MockPlatformServices
{
RuntimePlatform = platform
};
Device.PlatformServices = services;
var actual = (new MarkupExtensionParser()).ParseExpression(ref markup, new Internals.XamlServiceProvider(null, null)
{
IXamlTypeResolver = typeResolver,
IProvideValueTarget = new MockValueProvider("foo", new object())
{
TargetProperty = GetType().GetProperty(nameof(FontSize))
}
});
Assert.AreEqual(expected, actual);
}
[TestCase("{OnIdiom Phone=23, Tablet=25, Default=20}", TargetIdiom.Phone, 23)]
[TestCase("{OnIdiom Phone=23, Tablet=25, Default=20}", TargetIdiom.Tablet, 25)]
[TestCase("{OnIdiom 20, Phone=23, Tablet=25}", TargetIdiom.Desktop, 20)]
[TestCase("{OnIdiom Phone=23, Tablet=25, Desktop=26, TV=30, Watch=10}", TargetIdiom.Desktop, 26)]
[TestCase("{OnIdiom Phone=23, Tablet=25, Desktop=26, TV=30, Watch=10}", TargetIdiom.TV, 30)]
[TestCase("{OnIdiom Phone=23, Tablet=25, Desktop=26, TV=30, Watch=10}", TargetIdiom.Watch, 10)]
[TestCase("{OnIdiom Phone=23}", TargetIdiom.Desktop, default(int))]
public void OnIdiomExtension(string markup, TargetIdiom idiom, int expected)
{
Device.SetIdiom(idiom);
var actual = (new MarkupExtensionParser()).ParseExpression(ref markup, new Internals.XamlServiceProvider(null, null)
{
IXamlTypeResolver = typeResolver,
IProvideValueTarget = new MockValueProvider("foo", new object())
{
TargetProperty = GetType().GetProperty(nameof(FontSize))
}
});
Assert.AreEqual(expected, actual);
}
2016-03-22 23:02:25 +03:00
}
}