ExpressionObserverTests_Property.cs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681
  1. // Copyright (c) The Avalonia Project. All rights reserved.
  2. // Licensed under the MIT license. See licence.md file in the project root for full license information.
  3. using System;
  4. using System.Collections.Generic;
  5. using System.Reactive;
  6. using System.Reactive.Linq;
  7. using System.Reactive.Subjects;
  8. using Microsoft.Reactive.Testing;
  9. using Avalonia.Data;
  10. using Avalonia.Data.Core;
  11. using Avalonia.UnitTests;
  12. using Xunit;
  13. using System.Threading.Tasks;
  14. namespace Avalonia.Base.UnitTests.Data.Core
  15. {
  16. public class ExpressionObserverTests_Property
  17. {
  18. [Fact]
  19. public async Task Should_Get_Simple_Property_Value()
  20. {
  21. var data = new { Foo = "foo" };
  22. var target = new ExpressionObserver(data, "Foo");
  23. var result = await target.Take(1);
  24. Assert.Equal("foo", result);
  25. GC.KeepAlive(data);
  26. }
  27. [Fact]
  28. public void Should_Get_Simple_Property_Value_Type()
  29. {
  30. var data = new { Foo = "foo" };
  31. var target = new ExpressionObserver(data, "Foo");
  32. target.Subscribe(_ => { });
  33. Assert.Equal(typeof(string), target.ResultType);
  34. GC.KeepAlive(data);
  35. }
  36. [Fact]
  37. public async Task Should_Get_Simple_Property_Value_Null()
  38. {
  39. var data = new { Foo = (string)null };
  40. var target = new ExpressionObserver(data, "Foo");
  41. var result = await target.Take(1);
  42. Assert.Null(result);
  43. GC.KeepAlive(data);
  44. }
  45. [Fact]
  46. public async Task Should_Get_Simple_Property_From_Base_Class()
  47. {
  48. var data = new Class3 { Foo = "foo" };
  49. var target = new ExpressionObserver(data, "Foo");
  50. var result = await target.Take(1);
  51. Assert.Equal("foo", result);
  52. GC.KeepAlive(data);
  53. }
  54. [Fact]
  55. public async Task Should_Return_BindingNotification_Error_For_Root_Null()
  56. {
  57. var data = new Class3 { Foo = "foo" };
  58. var target = new ExpressionObserver(default(object), "Foo");
  59. var result = await target.Take(1);
  60. Assert.Equal(
  61. new BindingNotification(
  62. new MarkupBindingChainException("Null value", "Foo", string.Empty),
  63. BindingErrorType.Error,
  64. AvaloniaProperty.UnsetValue),
  65. result);
  66. GC.KeepAlive(data);
  67. }
  68. [Fact]
  69. public async Task Should_Return_BindingNotification_Error_For_Root_UnsetValue()
  70. {
  71. var data = new Class3 { Foo = "foo" };
  72. var target = new ExpressionObserver(AvaloniaProperty.UnsetValue, "Foo");
  73. var result = await target.Take(1);
  74. Assert.Equal(
  75. new BindingNotification(
  76. new MarkupBindingChainException("Null value", "Foo", string.Empty),
  77. BindingErrorType.Error,
  78. AvaloniaProperty.UnsetValue),
  79. result);
  80. GC.KeepAlive(data);
  81. }
  82. [Fact]
  83. public async Task Should_Return_BindingNotification_Error_For_Observable_Root_Null()
  84. {
  85. var data = new Class3 { Foo = "foo" };
  86. var target = new ExpressionObserver(Observable.Return(default(object)), "Foo");
  87. var result = await target.Take(1);
  88. Assert.Equal(
  89. new BindingNotification(
  90. new MarkupBindingChainException("Null value", "Foo", string.Empty),
  91. BindingErrorType.Error,
  92. AvaloniaProperty.UnsetValue),
  93. result);
  94. GC.KeepAlive(data);
  95. }
  96. [Fact]
  97. public async void Should_Return_BindingNotification_Error_For_Observable_Root_UnsetValue()
  98. {
  99. var data = new Class3 { Foo = "foo" };
  100. var target = new ExpressionObserver(Observable.Return(AvaloniaProperty.UnsetValue), "Foo");
  101. var result = await target.Take(1);
  102. Assert.Equal(
  103. new BindingNotification(
  104. new MarkupBindingChainException("Null value", "Foo", string.Empty),
  105. BindingErrorType.Error,
  106. AvaloniaProperty.UnsetValue),
  107. result);
  108. GC.KeepAlive(data);
  109. }
  110. [Fact]
  111. public async Task Should_Get_Simple_Property_Chain()
  112. {
  113. var data = new { Foo = new { Bar = new { Baz = "baz" } } };
  114. var target = new ExpressionObserver(data, "Foo.Bar.Baz");
  115. var result = await target.Take(1);
  116. Assert.Equal("baz", result);
  117. GC.KeepAlive(data);
  118. }
  119. [Fact]
  120. public void Should_Get_Simple_Property_Chain_Type()
  121. {
  122. var data = new { Foo = new { Bar = new { Baz = "baz" } } };
  123. var target = new ExpressionObserver(data, "Foo.Bar.Baz");
  124. target.Subscribe(_ => { });
  125. Assert.Equal(typeof(string), target.ResultType);
  126. GC.KeepAlive(data);
  127. }
  128. [Fact]
  129. public async Task Should_Return_BindingNotification_Error_For_Broken_Chain()
  130. {
  131. var data = new { Foo = new { Bar = 1 } };
  132. var target = new ExpressionObserver(data, "Foo.Bar.Baz");
  133. var result = await target.Take(1);
  134. Assert.IsType<BindingNotification>(result);
  135. Assert.Equal(
  136. new BindingNotification(
  137. new MissingMemberException("Could not find CLR property 'Baz' on '1'"), BindingErrorType.Error),
  138. result);
  139. GC.KeepAlive(data);
  140. }
  141. [Fact]
  142. public void Should_Return_BindingNotification_Error_For_Chain_With_Null_Value()
  143. {
  144. var data = new { Foo = default(object) };
  145. var target = new ExpressionObserver(data, "Foo.Bar.Baz");
  146. var result = new List<object>();
  147. target.Subscribe(x => result.Add(x));
  148. Assert.Equal(
  149. new[]
  150. {
  151. new BindingNotification(
  152. new MarkupBindingChainException("Null value", "Foo.Bar.Baz", "Foo"),
  153. BindingErrorType.Error,
  154. AvaloniaProperty.UnsetValue),
  155. },
  156. result);
  157. GC.KeepAlive(data);
  158. }
  159. [Fact]
  160. public void Should_Have_Null_ResultType_For_Broken_Chain()
  161. {
  162. var data = new { Foo = new { Bar = 1 } };
  163. var target = new ExpressionObserver(data, "Foo.Bar.Baz");
  164. Assert.Null(target.ResultType);
  165. GC.KeepAlive(data);
  166. }
  167. [Fact]
  168. public void Should_Track_Simple_Property_Value()
  169. {
  170. var data = new Class1 { Foo = "foo" };
  171. var target = new ExpressionObserver(data, "Foo");
  172. var result = new List<object>();
  173. var sub = target.Subscribe(x => result.Add(x));
  174. data.Foo = "bar";
  175. Assert.Equal(new[] { "foo", "bar" }, result);
  176. sub.Dispose();
  177. Assert.Equal(0, data.PropertyChangedSubscriptionCount);
  178. GC.KeepAlive(data);
  179. }
  180. [Fact]
  181. public void Should_Trigger_PropertyChanged_On_Null_Or_Empty_String()
  182. {
  183. var data = new Class1 { Bar = "foo" };
  184. var target = new ExpressionObserver(data, "Bar");
  185. var result = new List<object>();
  186. var sub = target.Subscribe(x => result.Add(x));
  187. Assert.Equal(new[] { "foo" }, result);
  188. data.Bar = "bar";
  189. Assert.Equal(new[] { "foo" }, result);
  190. data.RaisePropertyChanged(string.Empty);
  191. Assert.Equal(new[] { "foo", "bar" }, result);
  192. data.RaisePropertyChanged(null);
  193. Assert.Equal(new[] { "foo", "bar", "bar" }, result);
  194. sub.Dispose();
  195. Assert.Equal(0, data.PropertyChangedSubscriptionCount);
  196. GC.KeepAlive(data);
  197. }
  198. [Fact]
  199. public void Should_Track_End_Of_Property_Chain_Changing()
  200. {
  201. var data = new Class1 { Next = new Class2 { Bar = "bar" } };
  202. var target = new ExpressionObserver(data, "Next.Bar");
  203. var result = new List<object>();
  204. var sub = target.Subscribe(x => result.Add(x));
  205. ((Class2)data.Next).Bar = "baz";
  206. ((Class2)data.Next).Bar = null;
  207. Assert.Equal(new[] { "bar", "baz", null }, result);
  208. sub.Dispose();
  209. Assert.Equal(0, data.PropertyChangedSubscriptionCount);
  210. Assert.Equal(0, data.Next.PropertyChangedSubscriptionCount);
  211. GC.KeepAlive(data);
  212. }
  213. [Fact]
  214. public void Should_Track_Property_Chain_Changing()
  215. {
  216. var data = new Class1 { Next = new Class2 { Bar = "bar" } };
  217. var target = new ExpressionObserver(data, "Next.Bar");
  218. var result = new List<object>();
  219. var sub = target.Subscribe(x => result.Add(x));
  220. var old = data.Next;
  221. data.Next = new Class2 { Bar = "baz" };
  222. data.Next = new Class2 { Bar = null };
  223. Assert.Equal(new[] { "bar", "baz", null }, result);
  224. sub.Dispose();
  225. Assert.Equal(0, data.PropertyChangedSubscriptionCount);
  226. Assert.Equal(0, data.Next.PropertyChangedSubscriptionCount);
  227. Assert.Equal(0, old.PropertyChangedSubscriptionCount);
  228. GC.KeepAlive(data);
  229. }
  230. [Fact]
  231. public void Should_Track_Property_Chain_Breaking_With_Null_Then_Mending()
  232. {
  233. var data = new Class1
  234. {
  235. Next = new Class2
  236. {
  237. Next = new Class2
  238. {
  239. Bar = "bar"
  240. }
  241. }
  242. };
  243. var target = new ExpressionObserver(data, "Next.Next.Bar");
  244. var result = new List<object>();
  245. var sub = target.Subscribe(x => result.Add(x));
  246. var old = data.Next;
  247. data.Next = new Class2 { Bar = "baz" };
  248. data.Next = old;
  249. Assert.Equal(
  250. new object[]
  251. {
  252. "bar",
  253. new BindingNotification(
  254. new MarkupBindingChainException("Null value", "Next.Next.Bar", "Next.Next"),
  255. BindingErrorType.Error,
  256. AvaloniaProperty.UnsetValue),
  257. "bar"
  258. },
  259. result);
  260. sub.Dispose();
  261. Assert.Equal(0, data.PropertyChangedSubscriptionCount);
  262. Assert.Equal(0, data.Next.PropertyChangedSubscriptionCount);
  263. Assert.Equal(0, old.PropertyChangedSubscriptionCount);
  264. GC.KeepAlive(data);
  265. }
  266. [Fact]
  267. public void Should_Track_Property_Chain_Breaking_With_Missing_Member_Then_Mending()
  268. {
  269. var data = new Class1 { Next = new Class2 { Bar = "bar" } };
  270. var target = new ExpressionObserver(data, "Next.Bar");
  271. var result = new List<object>();
  272. var sub = target.Subscribe(x => result.Add(x));
  273. var old = data.Next;
  274. var breaking = new WithoutBar();
  275. data.Next = breaking;
  276. data.Next = new Class2 { Bar = "baz" };
  277. Assert.Equal(
  278. new object[]
  279. {
  280. "bar",
  281. new BindingNotification(
  282. new MissingMemberException("Could not find CLR property 'Bar' on 'Avalonia.Base.UnitTests.Data.Core.ExpressionObserverTests_Property+WithoutBar'"),
  283. BindingErrorType.Error),
  284. "baz",
  285. },
  286. result);
  287. sub.Dispose();
  288. Assert.Equal(0, data.PropertyChangedSubscriptionCount);
  289. Assert.Equal(0, data.Next.PropertyChangedSubscriptionCount);
  290. Assert.Equal(0, breaking.PropertyChangedSubscriptionCount);
  291. Assert.Equal(0, old.PropertyChangedSubscriptionCount);
  292. GC.KeepAlive(data);
  293. }
  294. [Fact]
  295. public void Empty_Expression_Should_Track_Root()
  296. {
  297. var data = new Class1 { Foo = "foo" };
  298. var update = new Subject<Unit>();
  299. var target = new ExpressionObserver(() => data.Foo, "", update);
  300. var result = new List<object>();
  301. target.Subscribe(x => result.Add(x));
  302. data.Foo = "bar";
  303. update.OnNext(Unit.Default);
  304. Assert.Equal(new[] { "foo", "bar" }, result);
  305. GC.KeepAlive(data);
  306. }
  307. [Fact]
  308. public void Should_Track_Property_Value_From_Observable_Root()
  309. {
  310. var scheduler = new TestScheduler();
  311. var source = scheduler.CreateColdObservable(
  312. OnNext(1, new Class1 { Foo = "foo" }),
  313. OnNext(2, new Class1 { Foo = "bar" }));
  314. var target = new ExpressionObserver(source, "Foo");
  315. var result = new List<object>();
  316. using (target.Subscribe(x => result.Add(x)))
  317. {
  318. scheduler.Start();
  319. }
  320. Assert.Equal(new[] { "foo", "bar" }, result);
  321. Assert.All(source.Subscriptions, x => Assert.NotEqual(Subscription.Infinite, x.Unsubscribe));
  322. }
  323. [Fact]
  324. public void Subscribing_Multiple_Times_Should_Return_Values_To_All()
  325. {
  326. var data = new Class1 { Foo = "foo" };
  327. var target = new ExpressionObserver(data, "Foo");
  328. var result1 = new List<object>();
  329. var result2 = new List<object>();
  330. var result3 = new List<object>();
  331. target.Subscribe(x => result1.Add(x));
  332. target.Subscribe(x => result2.Add(x));
  333. data.Foo = "bar";
  334. target.Subscribe(x => result3.Add(x));
  335. Assert.Equal(new[] { "foo", "bar" }, result1);
  336. Assert.Equal(new[] { "foo", "bar" }, result2);
  337. Assert.Equal(new[] { "bar" }, result3);
  338. GC.KeepAlive(data);
  339. }
  340. [Fact]
  341. public void Subscribing_Multiple_Times_Should_Only_Add_PropertyChanged_Handlers_Once()
  342. {
  343. var data = new Class1 { Foo = "foo" };
  344. var target = new ExpressionObserver(data, "Foo");
  345. var sub1 = target.Subscribe(x => { });
  346. var sub2 = target.Subscribe(x => { });
  347. Assert.Equal(1, data.PropertyChangedSubscriptionCount);
  348. sub1.Dispose();
  349. sub2.Dispose();
  350. Assert.Equal(0, data.PropertyChangedSubscriptionCount);
  351. GC.KeepAlive(data);
  352. }
  353. [Fact]
  354. public void SetValue_Should_Set_Simple_Property_Value()
  355. {
  356. var data = new Class1 { Foo = "foo" };
  357. var target = new ExpressionObserver(data, "Foo");
  358. using (target.Subscribe(_ => { }))
  359. {
  360. Assert.True(target.SetValue("bar"));
  361. }
  362. Assert.Equal("bar", data.Foo);
  363. GC.KeepAlive(data);
  364. }
  365. [Fact]
  366. public void SetValue_Should_Set_Property_At_The_End_Of_Chain()
  367. {
  368. var data = new Class1 { Next = new Class2 { Bar = "bar" } };
  369. var target = new ExpressionObserver(data, "Next.Bar");
  370. using (target.Subscribe(_ => { }))
  371. {
  372. Assert.True(target.SetValue("baz"));
  373. }
  374. Assert.Equal("baz", ((Class2)data.Next).Bar);
  375. GC.KeepAlive(data);
  376. }
  377. [Fact]
  378. public void SetValue_Should_Return_False_For_Missing_Property()
  379. {
  380. var data = new Class1 { Next = new WithoutBar() };
  381. var target = new ExpressionObserver(data, "Next.Bar");
  382. using (target.Subscribe(_ => { }))
  383. {
  384. Assert.False(target.SetValue("baz"));
  385. }
  386. GC.KeepAlive(data);
  387. }
  388. [Fact]
  389. public void SetValue_Should_Notify_New_Value_With_Inpc()
  390. {
  391. var data = new Class1();
  392. var target = new ExpressionObserver(data, "Foo");
  393. var result = new List<object>();
  394. target.Subscribe(x => result.Add(x));
  395. target.SetValue("bar");
  396. Assert.Equal(new[] { null, "bar" }, result);
  397. GC.KeepAlive(data);
  398. }
  399. [Fact]
  400. public void SetValue_Should_Notify_New_Value_Without_Inpc()
  401. {
  402. var data = new Class1();
  403. var target = new ExpressionObserver(data, "Bar");
  404. var result = new List<object>();
  405. target.Subscribe(x => result.Add(x));
  406. target.SetValue("bar");
  407. Assert.Equal(new[] { null, "bar" }, result);
  408. GC.KeepAlive(data);
  409. }
  410. [Fact]
  411. public void SetValue_Should_Return_False_For_Missing_Object()
  412. {
  413. var data = new Class1();
  414. var target = new ExpressionObserver(data, "Next.Bar");
  415. using (target.Subscribe(_ => { }))
  416. {
  417. Assert.False(target.SetValue("baz"));
  418. }
  419. GC.KeepAlive(data);
  420. }
  421. [Fact]
  422. public void Can_Replace_Root()
  423. {
  424. var first = new Class1 { Foo = "foo" };
  425. var second = new Class1 { Foo = "bar" };
  426. var root = first;
  427. var update = new Subject<Unit>();
  428. var target = new ExpressionObserver(() => root, "Foo", update);
  429. var result = new List<object>();
  430. var sub = target.Subscribe(x => result.Add(x));
  431. root = second;
  432. update.OnNext(Unit.Default);
  433. root = null;
  434. update.OnNext(Unit.Default);
  435. Assert.Equal(
  436. new object[]
  437. {
  438. "foo",
  439. "bar",
  440. new BindingNotification(
  441. new MarkupBindingChainException("Null value", "Foo", string.Empty),
  442. BindingErrorType.Error,
  443. AvaloniaProperty.UnsetValue)
  444. },
  445. result);
  446. Assert.Equal(0, first.PropertyChangedSubscriptionCount);
  447. Assert.Equal(0, second.PropertyChangedSubscriptionCount);
  448. GC.KeepAlive(first);
  449. GC.KeepAlive(second);
  450. }
  451. [Fact]
  452. public void Should_Not_Keep_Source_Alive()
  453. {
  454. Func<Tuple<ExpressionObserver, WeakReference>> run = () =>
  455. {
  456. var source = new Class1 { Foo = "foo" };
  457. var target = new ExpressionObserver(source, "Foo");
  458. return Tuple.Create(target, new WeakReference(source));
  459. };
  460. var result = run();
  461. result.Item1.Subscribe(x => { });
  462. GC.Collect();
  463. Assert.Null(result.Item2.Target);
  464. }
  465. private interface INext
  466. {
  467. int PropertyChangedSubscriptionCount { get; }
  468. }
  469. private class Class1 : NotifyingBase
  470. {
  471. private string _foo;
  472. private INext _next;
  473. public string Foo
  474. {
  475. get { return _foo; }
  476. set
  477. {
  478. _foo = value;
  479. RaisePropertyChanged(nameof(Foo));
  480. }
  481. }
  482. private string _bar;
  483. public string Bar
  484. {
  485. get { return _bar; }
  486. set { _bar = value; }
  487. }
  488. public INext Next
  489. {
  490. get { return _next; }
  491. set
  492. {
  493. _next = value;
  494. RaisePropertyChanged(nameof(Next));
  495. }
  496. }
  497. }
  498. private class Class2 : NotifyingBase, INext
  499. {
  500. private string _bar;
  501. private INext _next;
  502. public string Bar
  503. {
  504. get { return _bar; }
  505. set
  506. {
  507. _bar = value;
  508. RaisePropertyChanged(nameof(Bar));
  509. }
  510. }
  511. public INext Next
  512. {
  513. get { return _next; }
  514. set
  515. {
  516. _next = value;
  517. RaisePropertyChanged(nameof(Next));
  518. }
  519. }
  520. }
  521. private class Class3 : Class1
  522. {
  523. }
  524. private class WithoutBar : NotifyingBase, INext
  525. {
  526. }
  527. private Recorded<Notification<object>> OnNext(long time, object value)
  528. {
  529. return new Recorded<Notification<object>>(time, Notification.CreateOnNext<object>(value));
  530. }
  531. }
  532. }