BindingExpressionTests.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374
  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 void Should_Convert_Set_String_To_Double()
  65. {
  66. var data = new Class1 { StringValue = $"{5.6}" };
  67. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.StringValue), typeof(double));
  68. target.OnNext(6.7);
  69. Assert.Equal($"{6.7}", data.StringValue);
  70. GC.KeepAlive(data);
  71. }
  72. [Fact]
  73. public async Task Should_Convert_Get_Double_To_String()
  74. {
  75. var data = new Class1 { DoubleValue = 5.6 };
  76. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.DoubleValue), typeof(string));
  77. var result = await target.Take(1);
  78. Assert.Equal($"{5.6}", result);
  79. GC.KeepAlive(data);
  80. }
  81. [Fact]
  82. public void Should_Convert_Set_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. target.OnNext($"{6.7}");
  87. Assert.Equal(6.7, data.DoubleValue);
  88. GC.KeepAlive(data);
  89. }
  90. [Fact]
  91. public async Task Should_Return_BindingNotification_With_FallbackValue_For_NonConvertibe_Target_Value()
  92. {
  93. var data = new Class1 { StringValue = "foo" };
  94. var target = new BindingExpression(
  95. ExpressionObserver.Create(data, o => o.StringValue),
  96. typeof(int),
  97. 42,
  98. AvaloniaProperty.UnsetValue,
  99. DefaultValueConverter.Instance);
  100. var result = await target.Take(1);
  101. Assert.Equal(
  102. new BindingNotification(
  103. new InvalidCastException("'foo' is not a valid number."),
  104. BindingErrorType.Error,
  105. 42),
  106. result);
  107. GC.KeepAlive(data);
  108. }
  109. [Fact]
  110. public async Task Should_Return_BindingNotification_With_FallbackValue_For_NonConvertibe_Target_Value_With_Data_Validation()
  111. {
  112. var data = new Class1 { StringValue = "foo" };
  113. var target = new BindingExpression(
  114. ExpressionObserver.Create(data, o => o.StringValue, true),
  115. typeof(int),
  116. 42,
  117. AvaloniaProperty.UnsetValue,
  118. DefaultValueConverter.Instance);
  119. var result = await target.Take(1);
  120. Assert.Equal(
  121. new BindingNotification(
  122. new InvalidCastException("'foo' is not a valid number."),
  123. BindingErrorType.Error,
  124. 42),
  125. result);
  126. GC.KeepAlive(data);
  127. }
  128. [Fact]
  129. public async Task Should_Return_BindingNotification_For_Invalid_FallbackValue()
  130. {
  131. var data = new Class1 { StringValue = "foo" };
  132. var target = new BindingExpression(
  133. ExpressionObserver.Create(data, o => o.StringValue),
  134. typeof(int),
  135. "bar",
  136. AvaloniaProperty.UnsetValue,
  137. DefaultValueConverter.Instance);
  138. var result = await target.Take(1);
  139. Assert.Equal(
  140. new BindingNotification(
  141. new AggregateException(
  142. new InvalidCastException("'foo' is not a valid number."),
  143. new InvalidCastException("Could not convert FallbackValue 'bar' to 'System.Int32'")),
  144. BindingErrorType.Error),
  145. result);
  146. GC.KeepAlive(data);
  147. }
  148. [Fact]
  149. public async Task Should_Return_BindingNotification_For_Invalid_FallbackValue_With_Data_Validation()
  150. {
  151. var data = new Class1 { StringValue = "foo" };
  152. var target = new BindingExpression(
  153. ExpressionObserver.Create(data, o => o.StringValue, true),
  154. typeof(int),
  155. "bar",
  156. AvaloniaProperty.UnsetValue,
  157. DefaultValueConverter.Instance);
  158. var result = await target.Take(1);
  159. Assert.Equal(
  160. new BindingNotification(
  161. new AggregateException(
  162. new InvalidCastException("'foo' is not a valid number."),
  163. new InvalidCastException("Could not convert FallbackValue 'bar' to 'System.Int32'")),
  164. BindingErrorType.Error),
  165. result);
  166. GC.KeepAlive(data);
  167. }
  168. [Fact]
  169. public void Setting_Invalid_Double_String_Should_Not_Change_Target()
  170. {
  171. var data = new Class1 { DoubleValue = 5.6 };
  172. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.DoubleValue), typeof(string));
  173. target.OnNext("foo");
  174. Assert.Equal(5.6, data.DoubleValue);
  175. GC.KeepAlive(data);
  176. }
  177. [Fact]
  178. public void Setting_Invalid_Double_String_Should_Use_FallbackValue()
  179. {
  180. var data = new Class1 { DoubleValue = 5.6 };
  181. var target = new BindingExpression(
  182. ExpressionObserver.Create(data, o => o.DoubleValue),
  183. typeof(string),
  184. "9.8",
  185. AvaloniaProperty.UnsetValue,
  186. DefaultValueConverter.Instance);
  187. target.OnNext("foo");
  188. Assert.Equal(9.8, data.DoubleValue);
  189. GC.KeepAlive(data);
  190. }
  191. [Fact]
  192. public void Should_Coerce_Setting_UnsetValue_Double_To_Default_Value()
  193. {
  194. var data = new Class1 { DoubleValue = 5.6 };
  195. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.DoubleValue), typeof(string));
  196. target.OnNext(AvaloniaProperty.UnsetValue);
  197. Assert.Equal(0, data.DoubleValue);
  198. GC.KeepAlive(data);
  199. }
  200. [Fact]
  201. public void Should_Pass_ConverterParameter_To_Convert()
  202. {
  203. var data = new Class1 { DoubleValue = 5.6 };
  204. var converter = new Mock<IValueConverter>();
  205. var target = new BindingExpression(
  206. ExpressionObserver.Create(data, o => o.DoubleValue),
  207. typeof(string),
  208. converter.Object,
  209. converterParameter: "foo");
  210. target.Subscribe(_ => { });
  211. converter.Verify(x => x.Convert(5.6, typeof(string), "foo", CultureInfo.CurrentCulture));
  212. GC.KeepAlive(data);
  213. }
  214. [Fact]
  215. public void Should_Pass_ConverterParameter_To_ConvertBack()
  216. {
  217. var data = new Class1 { DoubleValue = 5.6 };
  218. var converter = new Mock<IValueConverter>();
  219. var target = new BindingExpression(
  220. ExpressionObserver.Create(data, o => o.DoubleValue),
  221. typeof(string),
  222. converter.Object,
  223. converterParameter: "foo");
  224. target.OnNext("bar");
  225. converter.Verify(x => x.ConvertBack("bar", typeof(double), "foo", CultureInfo.CurrentCulture));
  226. GC.KeepAlive(data);
  227. }
  228. [Fact]
  229. public void Should_Handle_DataValidation()
  230. {
  231. var data = new Class1 { DoubleValue = 5.6 };
  232. var converter = new Mock<IValueConverter>();
  233. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.DoubleValue, true), typeof(string));
  234. var result = new List<object>();
  235. target.Subscribe(x => result.Add(x));
  236. target.OnNext(1.2);
  237. target.OnNext($"{3.4}");
  238. target.OnNext("bar");
  239. Assert.Equal(
  240. new[]
  241. {
  242. new BindingNotification($"{5.6}"),
  243. new BindingNotification($"{1.2}"),
  244. new BindingNotification($"{3.4}"),
  245. new BindingNotification(
  246. new InvalidCastException("'bar' is not a valid number."),
  247. BindingErrorType.Error)
  248. },
  249. result);
  250. GC.KeepAlive(data);
  251. }
  252. [Fact]
  253. public void Second_Subscription_Should_Fire_Immediately()
  254. {
  255. var data = new Class1 { StringValue = "foo" };
  256. var target = new BindingExpression(ExpressionObserver.Create(data, o => o.StringValue), typeof(string));
  257. object result = null;
  258. target.Subscribe();
  259. target.Subscribe(x => result = x);
  260. Assert.Equal("foo", result);
  261. GC.KeepAlive(data);
  262. }
  263. [Fact]
  264. public void Null_Value_Should_Use_TargetNullValue()
  265. {
  266. var data = new Class1 { StringValue = "foo" };
  267. var target = new BindingExpression(
  268. ExpressionObserver.Create(data, o => o.StringValue),
  269. typeof(string),
  270. AvaloniaProperty.UnsetValue,
  271. "bar",
  272. DefaultValueConverter.Instance);
  273. object result = null;
  274. target.Subscribe(x => result = x);
  275. Assert.Equal("foo", result);
  276. data.StringValue = null;
  277. Assert.Equal("bar", result);
  278. GC.KeepAlive(data);
  279. }
  280. private class Class1 : NotifyingBase
  281. {
  282. private string _stringValue;
  283. private double _doubleValue;
  284. public string StringValue
  285. {
  286. get { return _stringValue; }
  287. set { _stringValue = value; RaisePropertyChanged(); }
  288. }
  289. public double DoubleValue
  290. {
  291. get { return _doubleValue; }
  292. set { _doubleValue = value; RaisePropertyChanged(); }
  293. }
  294. }
  295. }
  296. }