FromEventPattern.cs 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. // Licensed to the .NET Foundation under one or more agreements.
  2. // The .NET Foundation licenses this file to you under the MIT License.
  3. // See the LICENSE file in the project root for more information.
  4. using System.Reactive.Concurrency;
  5. using System.Reflection;
  6. using System.Threading;
  7. //
  8. // BREAKING CHANGE v2 > v1.x - FromEvent[Pattern] now has an implicit SubscribeOn and Publish operation.
  9. //
  10. // See FromEvent.cs for more information.
  11. //
  12. namespace System.Reactive.Linq.ObservableImpl
  13. {
  14. internal sealed class FromEventPattern
  15. {
  16. public sealed class Impl<TDelegate, TEventArgs> : ClassicEventProducer<TDelegate, EventPattern<TEventArgs>>
  17. {
  18. private readonly Func<EventHandler<TEventArgs>, TDelegate>? _conversion;
  19. public Impl(Action<TDelegate> addHandler, Action<TDelegate> removeHandler, IScheduler scheduler)
  20. : base(addHandler, removeHandler, scheduler)
  21. {
  22. }
  23. public Impl(Func<EventHandler<TEventArgs>, TDelegate> conversion, Action<TDelegate> addHandler, Action<TDelegate> removeHandler, IScheduler scheduler)
  24. : base(addHandler, removeHandler, scheduler)
  25. {
  26. _conversion = conversion;
  27. }
  28. protected override TDelegate GetHandler(Action<EventPattern<TEventArgs>> onNext)
  29. {
  30. TDelegate handler;
  31. if (_conversion == null)
  32. {
  33. Action<object, TEventArgs> h = (sender, eventArgs) => onNext(new EventPattern<TEventArgs>(sender, eventArgs));
  34. handler = ReflectionUtils.CreateDelegate<TDelegate>(h, typeof(Action<object, TEventArgs>).GetMethod(nameof(Action<object, TEventArgs>.Invoke)));
  35. }
  36. else
  37. {
  38. handler = _conversion((sender, eventArgs) => onNext(new EventPattern<TEventArgs>(sender, eventArgs)));
  39. }
  40. return handler;
  41. }
  42. }
  43. public sealed class Impl<TDelegate, TSender, TEventArgs> : ClassicEventProducer<TDelegate, EventPattern<TSender, TEventArgs>>
  44. {
  45. public Impl(Action<TDelegate> addHandler, Action<TDelegate> removeHandler, IScheduler scheduler)
  46. : base(addHandler, removeHandler, scheduler)
  47. {
  48. }
  49. protected override TDelegate GetHandler(Action<EventPattern<TSender, TEventArgs>> onNext)
  50. {
  51. Action<TSender, TEventArgs> h = (sender, eventArgs) => onNext(new EventPattern<TSender, TEventArgs>(sender, eventArgs));
  52. return ReflectionUtils.CreateDelegate<TDelegate>(h, typeof(Action<TSender, TEventArgs>).GetMethod(nameof(Action<TSender, TEventArgs>.Invoke)));
  53. }
  54. }
  55. public sealed class Handler<TSender, TEventArgs, TResult> : EventProducer<Delegate, TResult>
  56. {
  57. private readonly object _target;
  58. private readonly Type _delegateType;
  59. private readonly MethodInfo _addMethod;
  60. private readonly MethodInfo _removeMethod;
  61. private readonly Func<TSender, TEventArgs, TResult> _getResult;
  62. #if HAS_WINRT
  63. private readonly bool _isWinRT;
  64. #endif
  65. public Handler(object target, Type delegateType, MethodInfo addMethod, MethodInfo removeMethod, Func<TSender, TEventArgs, TResult> getResult, bool isWinRT, IScheduler scheduler)
  66. : base(scheduler)
  67. {
  68. #if HAS_WINRT
  69. _isWinRT = isWinRT;
  70. #else
  71. System.Diagnostics.Debug.Assert(!isWinRT);
  72. #endif
  73. _target = target;
  74. _delegateType = delegateType;
  75. _addMethod = addMethod;
  76. _removeMethod = removeMethod;
  77. _getResult = getResult;
  78. }
  79. protected override Delegate GetHandler(Action<TResult> onNext)
  80. {
  81. Action<TSender, TEventArgs> h = (sender, eventArgs) => onNext(_getResult(sender, eventArgs));
  82. return ReflectionUtils.CreateDelegate(_delegateType, h, typeof(Action<TSender, TEventArgs>).GetMethod(nameof(Action<TSender, TEventArgs>.Invoke)));
  83. }
  84. protected override IDisposable AddHandler(Delegate handler)
  85. {
  86. Action removeHandler;
  87. try
  88. {
  89. #if HAS_WINRT
  90. if (_isWinRT)
  91. {
  92. removeHandler = AddHandlerCoreWinRT(handler);
  93. }
  94. else
  95. #endif
  96. {
  97. removeHandler = AddHandlerCore(handler);
  98. }
  99. }
  100. catch (TargetInvocationException tie) when (tie.InnerException != null)
  101. {
  102. throw tie.InnerException;
  103. }
  104. return new RemoveHandlerDisposable(removeHandler);
  105. }
  106. private sealed class RemoveHandlerDisposable : IDisposable
  107. {
  108. private Action? _removeHandler;
  109. public RemoveHandlerDisposable(Action removeHandler)
  110. {
  111. Volatile.Write(ref _removeHandler, removeHandler);
  112. }
  113. public void Dispose()
  114. {
  115. try
  116. {
  117. Interlocked.Exchange(ref _removeHandler, null)?.Invoke();
  118. }
  119. catch (TargetInvocationException tie) when (tie.InnerException != null)
  120. {
  121. throw tie.InnerException;
  122. }
  123. }
  124. }
  125. private Action AddHandlerCore(Delegate handler)
  126. {
  127. _addMethod.Invoke(_target, new object[] { handler });
  128. return () => _removeMethod.Invoke(_target, new object[] { handler });
  129. }
  130. #if HAS_WINRT
  131. private Action AddHandlerCoreWinRT(Delegate handler)
  132. {
  133. var token = _addMethod.Invoke(_target, new object[] { handler });
  134. return () => _removeMethod.Invoke(_target, new[] { token });
  135. }
  136. #endif
  137. }
  138. }
  139. }