Общий Слабый Менеджер Событий Помощник - PullRequest
1 голос
/ 10 марта 2019

Если кто-то хочет использовать System.Windows.WeakEventManager (сборка WindowsBase) универсальным надежным способом, как:

  1. Избегать ненужной регистрации статических обработчиков?
  2. Убедитесь, что в источнике существует событие.

Поскольку отражение приходит к использованию, могут возникнуть некоторые исключения. Как с ними справиться?

WeakEventManager<TEventSource, TEventArgs>
  .AddHandler(TEventSource source, string eventName, EventHandler<TEventArgs> handler);
WeakEventManager<TEventSource, TEventArgs>
  .RemoveHandler(TEventSource source, string eventName, EventHandler<TEventArgs> handler);

1 Ответ

0 голосов
/ 10 марта 2019

Если производительность не является проблемой, универсальный помощник может быть выполнен как код ниже.

Поскольку статический обработчик (статической сборки) будет выгружен с доменом приложения, нет необходимости использовать для него слабые ссылки.

Может быть много исключений в реализации регистрации / отмены регистрации общих событий.

В частности, может быть лучше сделать реализацию слабого управления в случае необходимости.

public static class WeakEventManagerHelper
{

  public static void AddEventHandler<T, U>(string eventName, T eventSource, EventHandler<U> handler) where U : EventArgs
  {
    ThrowOnNullArg(eventName, eventSource, handler);

    EventInfo eventInfo = GetEventInfo(eventName, eventSource);
    ThrowIfEventNotExists(eventInfo, eventName);

    if (!eventInfo.AddMethod.IsPublic)
    {
      ThrowIfEventMethodIsNotPublic("add");
    }

    ThrowIfHandlerIsNotProper(eventInfo, handler);

    if (handler.Method.IsStatic && !IsCollectibleAssembly(handler))
    {
      AddStaticEventHandler(eventInfo, eventSource, handler);
    }
    else
    {
      AddInstanceEventHandler(eventName, eventSource, handler);
    }
  }       

  public static void RemoveEventHandler<T, U>(string eventName, T eventSource, EventHandler<U> handler) where U : EventArgs
  {
    ThrowOnNullArg(eventName, eventSource, handler);

    EventInfo eventInfo = GetEventInfo(eventName, eventSource);
    ThrowIfEventNotExists(eventInfo, eventName);

    if (!eventInfo.RemoveMethod.IsPublic)
    {
      ThrowIfEventMethodIsNotPublic("remove");
    }

    ThrowIfHandlerIsNotProper(eventInfo, handler);

    if (handler.Method.IsStatic && !IsCollectibleAssembly(handler))
    {
      RemoveStaticEventHandler(eventInfo, eventSource, handler);
    }
    else
    {
      RemoveInstanceEventHandler(eventName, eventSource, handler);
    }
  }

  private static void AddStaticEventHandler(EventInfo eventInfo, object eventSource, Delegate handler)
  {
    try
    {
      eventInfo.AddEventHandler(eventSource, handler);
    }
    catch
    {
      // Handling …
      throw;
    }      
  }

  private static void RemoveStaticEventHandler(EventInfo eventInfo, object eventSource, Delegate handler)
  {
    try
    {
      eventInfo.RemoveEventHandler(eventSource, handler);
    }
    catch
    {
      // Handling …
      throw;
    }      
  }

  private static void AddInstanceEventHandler<T, U>(string eventName, T eventSource, EventHandler<U> handler) where U : EventArgs
  {
    try
    {
      WeakEventManager<T, U>.AddHandler(
      eventSource,
      eventName,
      handler);
    }
    catch
    {
      // Handling …
      throw;
    }
  }

  private static void RemoveInstanceEventHandler<T, U>(string eventName, T eventSource, EventHandler<U> handler) where U : EventArgs
  {
    try
    {
      WeakEventManager<T, U>.RemoveHandler(
      eventSource,
      eventName,
      handler);
    }
    catch
    {
      // Handling …
      throw;
    }
  }

  private static EventInfo GetEventInfo<T>(string eventName, T eventSource)
  {
    return eventSource.GetType().GetEvent(eventName, BindingFlags.Public | BindingFlags.Instance | BindingFlags.Static);
  }

  private static void ThrowOnNullArg<T, U>(string eventName, T eventSource, EventHandler<U> handler) where U : EventArgs
  {
    if (eventName == null)
    {
      throw new ArgumentNullException(nameof(eventName));
    }
    else if (eventSource == null)
    {
      throw new ArgumentNullException(nameof(eventSource));
    }
    else if (handler == null)
    {
      throw new ArgumentNullException(nameof(handler));
    }
  }

  private static void ThrowIfEventNotExists(EventInfo eventInfo, string eventName)
  {
    if (eventInfo == null)
    {
      throw new ArgumentException($"Event source does not contain event named {eventName}!");
    }
  }

  private static void ThrowIfEventMethodIsNotPublic(string method)
  {
    throw new ArgumentException($"Event {method} method is not public!");
  }

  private static void ThrowIfHandlerIsNotProper(EventInfo eventInfo, Delegate handler)
  {
    if (eventInfo.EventHandlerType != handler.GetType())
    {
      throw new ArgumentException($"Improper handler type {handler}!");
    }
  }    

  private static bool IsCollectibleAssembly<U>(EventHandler<U> handler)
  {
    if (!handler.Method.DeclaringType.Assembly.IsDynamic)
    {
      return false;
    }

    try
    {
      Assembly handlerMethodAssembly = handler.Method.DeclaringType.Assembly;

      Type assemblyBuilderDataType = Assembly.GetAssembly(typeof(AssemblyBuilder))
          .GetType("System.Reflection.Emit.AssemblyBuilderData");

      object assemblyBuilderData = handlerMethodAssembly.GetType()
        .GetFields(BindingFlags.NonPublic | BindingFlags.Instance)
        .Single(fi => fi.FieldType == assemblyBuilderDataType)
        .GetValue(handlerMethodAssembly);

      object assemblyBuilderAccess = assemblyBuilderDataType
        .GetFields(BindingFlags.NonPublic | BindingFlags.Instance)
        .Single(fi => fi.FieldType == typeof(AssemblyBuilderAccess))
        .GetValue(assemblyBuilderData);

      return (AssemblyBuilderAccess)assemblyBuilderAccess == AssemblyBuilderAccess.RunAndCollect;
    }
    catch (Exception)
    {
      return false;
    }
  }
}
...