Songs window closes when clicking close button

I've spent a lot of time fixing bugs and cleaning up the tests, and now it's finally time to add some more functionality. The smallest step I can think of right now is to add a button to the view that closes it.

To get the full picture clear in my mind I have another look at the types I have in action.

First I would like the test to start the application and open the view, just like the other tests. Then I would like the test to (1) trigger a click on the close button, which is done by calling a method on the SongsViewFake. Since I don't want any logic inside the view, the view should just pass on the request by (2) raising an event in the ISongsView interface. Whoever listens to that notification, in this case Thingy, should (3) tell the view to close itself.

Now it's clear to me that the test should look like this.

[Test]
public void Songs_window_closes_when_clicking_close_button()
{
Application
.Start(view =>
{
view.ClickCloseButton();
Assert.That(view.HasBeenClosed, Is.True);
});
}

It's quite similar to the other tests. This is a bit cleaner in the sense that I can do the assertion inside the showAction and not have to use variables in the test to keep track of the state.

I extend the fake with the ClickCloseButton() method and the HasBeenClosed property.

public class SongsViewFake : ISongsView
{
public event Action OnCloseButtonClick;
public bool HasBeenClosed { get; private set; }
 
// ...
 
public SongsViewFake(Action<SongsViewFake> showAction)
{
ShowAction = showAction;
HasBeenClosed = false;
}
 
public void CloseView()
{
HasBeenClosed = true;
}
 
public void ClickCloseButton()
{
OnCloseButtonClick.Raise();
}
 
// ...
}

I also add the CloseView() method that I expect the production code to call to tell the view to close itself. The code compiles, so I run the tests. The result is something I recognize by now:

SongsFixture.Songs_window_closes_when_clicking_close_button : Failed
System.InvalidOperationException : No handler registered for the event.
at SongLibrary.ActionExtensions.Raise(Action me) in ActionExtensions.cs: line 9
at SongLibrary.Test.SongsViewFake.ClickCloseButton() in SongsViewFake.cs: line 30

There are no subscribers to the OnCloseButtonClick event.

So according to my initial thoughts, I continue to extend the ISongsView interface with the OnCloseButtonClick event and the CloseView() method. In order for the code to compile I also have to implement these in SongsView.

public interface ISongsView
{
event Action OnCloseButtonClick;
event Action OnViewClosed;
void ShowView();
void CloseView();
}
 
public partial class SongsView : Form, ISongsView
{
public event Action OnCloseButtonClick;
 
// ...
 
public void CloseView()
{
Close();
}
 
private void CloseButtonClick( object sender, EventArgs e)
{
OnCloseButtonClick.Raise();
}
}

The code compiles, but still shows the same failure message when running the tests. The smallest step I can take is to start listening to the event with an empty handler, so that's what I do.

public class Thingy : ApplicationContext
{
// ...
 
public void ShowView()
{
View.OnViewClosed += () => CloseAction( this );
View.OnCloseButtonClick += () => { };
View.ShowView();
}
}

And I get the failure I expect.

SongsFixture.Songs_window_closes_when_clicking_close_button : Failed
Expected: True
But was: False

And the solution is to change the event handler.

public class Thingy : ApplicationContext
{
// ...
 
public void ShowView()
{
View.OnViewClosed += () => CloseAction( this );
View.OnCloseButtonClick += View.CloseView;
View.ShowView();
}
}

One line of code, and I'm done. When the event is raised, call the CloseView() method on View. This solution works, the tests are all green.

This test was a bit different than the previous tests in that the interesting part of the use case is started with that an event is triggered inside a fake. The normal way of thinking about tests is that the test tells the code to perform something. It's a bit mind twisting when the test just fires an event, hoping that someone should take care of it. But it's something you get used to.

Running the application, clicking the close button works great.

I check in my code.

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>