QueryLanguage.Joins.cs 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. // Copyright (c) Microsoft Open Technologies, Inc. All rights reserved. See License.txt in the project root for license information.
  2. using System.Collections.Generic;
  3. using System.Reactive.Disposables;
  4. using System.Reactive.Joins;
  5. namespace System.Reactive.Linq
  6. {
  7. internal partial class QueryLanguage
  8. {
  9. #region And
  10. public virtual Pattern<TLeft, TRight> And<TLeft, TRight>(IObservable<TLeft> left, IObservable<TRight> right)
  11. {
  12. return new Pattern<TLeft, TRight>(left, right);
  13. }
  14. #endregion
  15. #region Then
  16. public virtual Plan<TResult> Then<TSource, TResult>(IObservable<TSource> source, Func<TSource, TResult> selector)
  17. {
  18. return new Pattern<TSource>(source).Then(selector);
  19. }
  20. #endregion
  21. #region When
  22. public virtual IObservable<TResult> When<TResult>(params Plan<TResult>[] plans)
  23. {
  24. return When((IEnumerable<Plan<TResult>>)plans);
  25. }
  26. public virtual IObservable<TResult> When<TResult>(IEnumerable<Plan<TResult>> plans)
  27. {
  28. return new AnonymousObservable<TResult>(observer =>
  29. {
  30. var externalSubscriptions = new Dictionary<object, IJoinObserver>();
  31. var gate = new object();
  32. var activePlans = new List<ActivePlan>();
  33. var outObserver = Observer.Create<TResult>(observer.OnNext,
  34. exception =>
  35. {
  36. foreach (var po in externalSubscriptions.Values)
  37. {
  38. po.Dispose();
  39. }
  40. observer.OnError(exception);
  41. },
  42. observer.OnCompleted);
  43. try
  44. {
  45. foreach (var plan in plans)
  46. activePlans.Add(plan.Activate(externalSubscriptions, outObserver,
  47. activePlan =>
  48. {
  49. activePlans.Remove(activePlan);
  50. if (activePlans.Count == 0)
  51. outObserver.OnCompleted();
  52. }));
  53. }
  54. catch (Exception e)
  55. {
  56. //
  57. // [OK] Use of unsafe Subscribe: we're calling into a known producer implementation.
  58. //
  59. return Throw<TResult>(e).Subscribe/*Unsafe*/(observer);
  60. }
  61. var group = new CompositeDisposable(externalSubscriptions.Values.Count);
  62. foreach (var joinObserver in externalSubscriptions.Values)
  63. {
  64. joinObserver.Subscribe(gate);
  65. group.Add(joinObserver);
  66. }
  67. return group;
  68. });
  69. }
  70. #endregion
  71. }
  72. }