BindingExpressionTests.cs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Globalization;
  4. using System.Reactive.Linq;
  5. using System.Threading;
  6. using System.Threading.Tasks;
  7. using Avalonia.Data;
  8. using Avalonia.Data.Converters;
  9. using Avalonia.Data.Core;
  10. using Avalonia.Markup.Parsers;
  11. using Avalonia.UnitTests;
  12. using Moq;
  13. using Xunit;
  14. namespace Avalonia.Base.UnitTests.Data.Core
  15. {
  16. public class BindingExpressionTests : IClassFixture<InvariantCultureFixture>
  17. {
  18. [Fact]
  19. public async Task Should_Get_Simple_Property_Value()
  20. {
  21. var data = new Class1 { StringValue = "foo" };
  22. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.StringValue), typeof(string));
  23. var result = await target.Take(1);
  24. Assert.Equal("foo", result);
  25. GC.KeepAlive(data);
  26. }
  27. [Fact]
  28. public void Should_Set_Simple_Property_Value()
  29. {
  30. var data = new Class1 { StringValue = "foo" };
  31. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.StringValue), typeof(string));
  32. target.OnNext("bar");
  33. Assert.Equal("bar", data.StringValue);
  34. GC.KeepAlive(data);
  35. }
  36. [Fact]
  37. public void Should_Set_Indexed_Value()
  38. {
  39. var data = new { Foo = new[] { "foo" } };
  40. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.Foo[0]), typeof(string));
  41. target.OnNext("bar");
  42. Assert.Equal("bar", data.Foo[0]);
  43. GC.KeepAlive(data);
  44. }
  45. [Fact]
  46. public async Task Should_Convert_Get_String_To_Double()
  47. {
  48. var data = new Class1 { StringValue = $"{5.6}" };
  49. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.StringValue), typeof(double));
  50. var result = await target.Take(1);
  51. Assert.Equal(5.6, result);
  52. GC.KeepAlive(data);
  53. }
  54. [Fact]
  55. public async Task Getting_Invalid_Double_String_Should_Return_BindingError()
  56. {
  57. var data = new Class1 { StringValue = "foo" };
  58. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.StringValue), typeof(double));
  59. var result = await target.Take(1);
  60. Assert.IsType<BindingNotification>(result);
  61. GC.KeepAlive(data);
  62. }
  63. [Fact]
  64. public async Task Should_Coerce_Get_Null_Double_String_To_UnsetValue()
  65. {
  66. var data = new Class1 { StringValue = null };
  67. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.StringValue), typeof(double));
  68. var result = await target.Take(1);
  69. Assert.Equal(AvaloniaProperty.UnsetValue, result);
  70. GC.KeepAlive(data);
  71. }
  72. [Fact]
  73. public void Should_Convert_Set_String_To_Double()
  74. {
  75. var data = new Class1 { StringValue = $"{5.6}" };
  76. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.StringValue), typeof(double));
  77. target.OnNext(6.7);
  78. Assert.Equal($"{6.7}", data.StringValue);
  79. GC.KeepAlive(data);
  80. }
  81. [Fact]
  82. public async Task Should_Convert_Get_Double_To_String()
  83. {
  84. var data = new Class1 { DoubleValue = 5.6 };
  85. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.DoubleValue), typeof(string));
  86. var result = await target.Take(1);
  87. Assert.Equal($"{5.6}", result);
  88. GC.KeepAlive(data);
  89. }
  90. [Fact]
  91. public void Should_Convert_Set_Double_To_String()
  92. {
  93. var data = new Class1 { DoubleValue = 5.6 };
  94. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.DoubleValue), typeof(string));
  95. target.OnNext($"{6.7}");
  96. Assert.Equal(6.7, data.DoubleValue);
  97. GC.KeepAlive(data);
  98. }
  99. [Fact]
  100. public async Task Should_Return_BindingNotification_With_FallbackValue_For_NonConvertibe_Target_Value()
  101. {
  102. var data = new Class1 { StringValue = "foo" };
  103. var target = new BindingExpression(
  104. ExpressionObserver.Create(data, o => o.StringValue),
  105. typeof(int),
  106. 42,
  107. AvaloniaProperty.UnsetValue,
  108. DefaultValueConverter.Instance);
  109. var result = await target.Take(1);
  110. Assert.Equal(
  111. new BindingNotification(
  112. new InvalidCastException("'foo' is not a valid number."),
  113. BindingErrorType.Error,
  114. 42),
  115. result);
  116. GC.KeepAlive(data);
  117. }
  118. [Fact]
  119. public async Task Should_Return_BindingNotification_With_FallbackValue_For_NonConvertibe_Target_Value_With_Data_Validation()
  120. {
  121. var data = new Class1 { StringValue = "foo" };
  122. var target = new BindingExpression(
  123. ExpressionObserver.Create(data, o => o.StringValue, true),
  124. typeof(int),
  125. 42,
  126. AvaloniaProperty.UnsetValue,
  127. DefaultValueConverter.Instance);
  128. var result = await target.Take(1);
  129. Assert.Equal(
  130. new BindingNotification(
  131. new InvalidCastException("'foo' is not a valid number."),
  132. BindingErrorType.Error,
  133. 42),
  134. result);
  135. GC.KeepAlive(data);
  136. }
  137. [Fact]
  138. public async Task Should_Return_BindingNotification_For_Invalid_FallbackValue()
  139. {
  140. var data = new Class1 { StringValue = "foo" };
  141. var target = new BindingExpression(
  142. ExpressionObserver.Create(data, o => o.StringValue),
  143. typeof(int),
  144. "bar",
  145. AvaloniaProperty.UnsetValue,
  146. DefaultValueConverter.Instance);
  147. var result = await target.Take(1);
  148. Assert.Equal(
  149. new BindingNotification(
  150. new AggregateException(
  151. new InvalidCastException("'foo' is not a valid number."),
  152. new InvalidCastException("Could not convert FallbackValue 'bar' to 'System.Int32'")),
  153. BindingErrorType.Error),
  154. result);
  155. GC.KeepAlive(data);
  156. }
  157. [Fact]
  158. public async Task Should_Return_BindingNotification_For_Invalid_FallbackValue_With_Data_Validation()
  159. {
  160. var data = new Class1 { StringValue = "foo" };
  161. var target = new BindingExpression(
  162. ExpressionObserver.Create(data, o => o.StringValue, true),
  163. typeof(int),
  164. "bar",
  165. AvaloniaProperty.UnsetValue,
  166. DefaultValueConverter.Instance);
  167. var result = await target.Take(1);
  168. Assert.Equal(
  169. new BindingNotification(
  170. new AggregateException(
  171. new InvalidCastException("'foo' is not a valid number."),
  172. new InvalidCastException("Could not convert FallbackValue 'bar' to 'System.Int32'")),
  173. BindingErrorType.Error),
  174. result);
  175. GC.KeepAlive(data);
  176. }
  177. [Fact]
  178. public void Setting_Invalid_Double_String_Should_Not_Change_Target()
  179. {
  180. var data = new Class1 { DoubleValue = 5.6 };
  181. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.DoubleValue), typeof(string));
  182. target.OnNext("foo");
  183. Assert.Equal(5.6, data.DoubleValue);
  184. GC.KeepAlive(data);
  185. }
  186. [Fact]
  187. public void Setting_Invalid_Double_String_Should_Use_FallbackValue()
  188. {
  189. var data = new Class1 { DoubleValue = 5.6 };
  190. var target = new BindingExpression(
  191. ExpressionObserver.Create(data, o => o.DoubleValue),
  192. typeof(string),
  193. "9.8",
  194. AvaloniaProperty.UnsetValue,
  195. DefaultValueConverter.Instance);
  196. target.OnNext("foo");
  197. Assert.Equal(9.8, data.DoubleValue);
  198. GC.KeepAlive(data);
  199. }
  200. [Fact]
  201. public void Should_Coerce_Setting_Null_Double_To_Default_Value()
  202. {
  203. var data = new Class1 { DoubleValue = 5.6 };
  204. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.DoubleValue), typeof(string));
  205. target.OnNext(null);
  206. Assert.Equal(0, data.DoubleValue);
  207. GC.KeepAlive(data);
  208. }
  209. [Fact]
  210. public void Should_Coerce_Setting_UnsetValue_Double_To_Default_Value()
  211. {
  212. var data = new Class1 { DoubleValue = 5.6 };
  213. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.DoubleValue), typeof(string));
  214. target.OnNext(AvaloniaProperty.UnsetValue);
  215. Assert.Equal(0, data.DoubleValue);
  216. GC.KeepAlive(data);
  217. }
  218. [Fact]
  219. public void Should_Pass_ConverterParameter_To_Convert()
  220. {
  221. var data = new Class1 { DoubleValue = 5.6 };
  222. var converter = new Mock<IValueConverter>();
  223. var target = new BindingExpression(
  224. ExpressionObserver.Create(data, o => o.DoubleValue),
  225. typeof(string),
  226. converter.Object,
  227. converterParameter: "foo");
  228. target.Subscribe(_ => { });
  229. converter.Verify(x => x.Convert(5.6, typeof(string), "foo", CultureInfo.CurrentCulture));
  230. GC.KeepAlive(data);
  231. }
  232. [Fact]
  233. public void Should_Pass_ConverterParameter_To_ConvertBack()
  234. {
  235. var data = new Class1 { DoubleValue = 5.6 };
  236. var converter = new Mock<IValueConverter>();
  237. var target = new BindingExpression(
  238. ExpressionObserver.Create(data, o => o.DoubleValue),
  239. typeof(string),
  240. converter.Object,
  241. converterParameter: "foo");
  242. target.OnNext("bar");
  243. converter.Verify(x => x.ConvertBack("bar", typeof(double), "foo", CultureInfo.CurrentCulture));
  244. GC.KeepAlive(data);
  245. }
  246. [Fact]
  247. public void Should_Handle_DataValidation()
  248. {
  249. var data = new Class1 { DoubleValue = 5.6 };
  250. var converter = new Mock<IValueConverter>();
  251. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.DoubleValue, true), typeof(string));
  252. var result = new List<object>();
  253. target.Subscribe(x => result.Add(x));
  254. target.OnNext(1.2);
  255. target.OnNext($"{3.4}");
  256. target.OnNext("bar");
  257. Assert.Equal(
  258. new[]
  259. {
  260. new BindingNotification($"{5.6}"),
  261. new BindingNotification($"{1.2}"),
  262. new BindingNotification($"{3.4}"),
  263. new BindingNotification(
  264. new InvalidCastException("'bar' is not a valid number."),
  265. BindingErrorType.Error)
  266. },
  267. result);
  268. GC.KeepAlive(data);
  269. }
  270. [Fact]
  271. public void Second_Subscription_Should_Fire_Immediately()
  272. {
  273. var data = new Class1 { StringValue = "foo" };
  274. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.StringValue), typeof(string));
  275. object result = null;
  276. target.Subscribe();
  277. target.Subscribe(x => result = x);
  278. Assert.Equal("foo", result);
  279. GC.KeepAlive(data);
  280. }
  281. [Fact]
  282. public void Null_Value_Should_Use_TargetNullValue()
  283. {
  284. var data = new Class1 { StringValue = "foo" };
  285. var target = new BindingExpression(
  286. ExpressionObserver.Create(data, o => o.StringValue),
  287. typeof(string),
  288. AvaloniaProperty.UnsetValue,
  289. "bar",
  290. DefaultValueConverter.Instance);
  291. object result = null;
  292. target.Subscribe(x => result = x);
  293. Assert.Equal("foo", result);
  294. data.StringValue = null;
  295. Assert.Equal("bar", result);
  296. GC.KeepAlive(data);
  297. }
  298. private class Class1 : NotifyingBase
  299. {
  300. private string _stringValue;
  301. private double _doubleValue;
  302. public string StringValue
  303. {
  304. get { return _stringValue; }
  305. set { _stringValue = value; RaisePropertyChanged(); }
  306. }
  307. public double DoubleValue
  308. {
  309. get { return _doubleValue; }
  310. set { _doubleValue = value; RaisePropertyChanged(); }
  311. }
  312. }
  313. }
  314. }