Raise event duplication

Implementing the last test left me with some duplication. Here is the copy found in the production code that I want to extract into a reusable method.

public partial class SongsView : Form, ISongsView
{
public event Action OnViewClosed;
 
// ...
 
private void SongsViewFormClosed( object sender, FormClosedEventArgs e)
{
var handler = OnViewClosed;
if (handler == null ) throw new InvalidOperationException("No handler registered for the event .");
handler();
}
}

This is a refactoring that probably all of today's IDE:s have built-in: extract method. I just extract the method from the last two lines into a method called Raise(). The method does not access the instance, so I also make it static.

private void SongsViewFormClosed( object sender, FormClosedEventArgs e)
{
var handler = OnViewClosed;
Raise(handler);
}
 
private static void Raise(Action handler)
{
if (handler == null ) throw new InvalidOperationException("No handler registered for the event .");
handler();
}

Now I would like to use the same method from the test code. This suggests that the method does not belong to SongsView, but rather in some shared location.

Since it is a static method, I could convert it into a non-static method that belongs to the type of the parameter, Action. However, Action is a delegate which can't have methods on it, and it is anyway not something that I can modify since it is part of the .NET framework. So I will instead convert the method into an extension method on Action.

Extension methods are a really powerful feature that is not found in java. They enable you to add methods to an existing type without modifying that type. That is, to the caller it looks like it is a normal instance method while it is actually defined elsewhere.

This allows you to add functionality to types that you don't own, and also add functionality to types that can't have implementations, such as delegates and interfaces.

While extension methods look the same to the caller, they are different from normal instance methods in the sense that they don't have access to non-public members.

First, I create an empty static class ActionExtensions:

public static class ActionExtensions
{
}

Then I use my refactoring tool's "Move To Another Type" on the method and specify that I want to move it to the ActionExtensions class. Apart from moving the method, all references to the method are updated to point to the new location. At the same time, I make the method public so that it is accessible from the caller.

public partial class SongsView : Form, ISongsView
{
// ...
 
private void SongsViewFormClosed( object sender, FormClosedEventArgs e)
{
var handler = OnViewClosed;
ActionExtensions.Raise(handler);
}
}
 
public static class ActionExtensions
{
public static void Raise(Action handler)
{
if (handler == null ) throw new InvalidOperationException("No handler registered for the event .");
handler();
}
}

Finally, I convert the Raise() method into an extension method by adding a this modifier on the first parameter, and also modify the caller to use it as such:

public partial class SongsView : Form, ISongsView
{
// ...
 
private void SongsViewFormClosed( object sender, FormClosedEventArgs e)
{
OnViewClosed.Raise();
}
}
 
public static class ActionExtensions
{
public static void Raise( this Action handler)
{
// ...
}
}

By moving the method to an extension method I have accomplished two things. One is that I have found a nice home for a shared method that could otherwise be hard to find a good home for. The other is that the code raising the event now looks really slick and easy to read.

Finally, I change the other copy of the event raising code to use the same extension method:

public class SongsViewFake : ISongsView
{
// ...
 
private void ClickWindowCloseButton()
{
OnViewClosed.Raise();
}
}

Really nice.

I do one more thing with the extension method, and that is that I rename the parameter from handler to me:

public static class ActionExtensions
{
public static void Raise( this Action me) { ... }
}

The reason is that this parameter is a bit special. It corresponds to this in a normal instance method, and I use the convention of always naming it me to remind me of that.

A reflection I make is that now that the event raising logic is in a separate method, it is also much easier to test that InvalidOperaionException is thrown when there are no event subscribers. I could easily write a unit test that only executes the ActionExtensions.Raise() method.

For now, I choose not to do that though. When the check was part of the SongsView I felt a little bit nervous about it. The check felt misplaced and was surrounded by other, unrelated things. Now that it is extracted into its own class and method I feel much safer. The chances that someone would get confused about the code and accidently break it are small. So though improved design, I feel I have also reduced the need for full test coverage.

The tests are all green, and running the application works fine, so I check in my changes.

Leave a Reply

Your email address will not be published. Required fields are marked *

*

You may use these HTML tags and attributes: <a href="" title=""> <abbr title=""> <acronym title=""> <b> <blockquote cite=""> <cite> <code> <del datetime=""> <em> <i> <q cite=""> <strike> <strong>