FromEventPattern.cs 5.4 KB

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