Anybody come across a clever way of closing a view in a viewmodel using MVVM?
Maybe there is a way of using binding to signal the view (window) to close?
I w
Not sure what MVVM framework you are using, but most contain some sort of messaging / notification solution that is easy have things register for messages which are sent. There is no reason that I can imagine that your view could not register for a message such as "CloseWindowsBoundTo" and the viewModel as the sender. Then in your view, you can just register for that message, and compare your current datacontext to the sender. If they match, close the window.
Simple, and keeps your view abstracted from your viewmodel.
Here would be my approach using MVVM-light toolkit:
In the ViewModel:
public void notifyWindowToClose()
{
Messenger.Default.Send<NotificationMessage>(
new NotificationMessage(this, "CloseWindowsBoundToMe")
);
}
And in the View:
Messenger.Default.Register<NotificationMessage>(this, (nm) =>
{
if (nm.Notification == "CloseWindowsBoundToMe")
{
if (nm.Sender == this.DataContext)
this.Close();
}
});
Generally you would use some kind of controller/presenter/service to drive the screen activation/deactivation. MVVM is not meant to be the One Pattern to Rule Them All. You will need to combine it with other patterns in any non-trivial application.
That said, in some situations in makes sense to have a view model that manages the life cycle of child view models. For example, you might have an EditorViewModel
that manages a collection of child view models - one for each document being edited. In that case, simply adding/removing to/from this collection can result in the view activating/deactivating. But this does not sound like it fits your use case.
To close the view from viewmodel i used the Galasoft MVVM Light Toolkit which you can download here: http://www.mvvmlight.net/
create a class like this: public class ClosingRequested : MessageBase { }
add this to your view constructor: Messenger.Default.Register(this, vm, msg=> Close());
call this to close your window: Messenger.Default.Send(new ClosingRequested(), this);
Edit: See my blog post for a more detailed explanation.
When I need to achieve that, I use a IRequestCloseViewModel interface that I created.
This interface contains only one event: RequestClose. This event is raised by the ViewModel (which inherits from a ViewModelBase class AND implement IRequestCloseViewModel) when it wants to close its associated view.
In my app, all Window inherit from an abstract class ApplicationWindow. This abstract class is notified each time the DataContext changed and in the handler checks if the DataContext support the IRequestCloseViewModel. If this is the case, an event handler is set up to close the Window when the event is fired.
Alternatively, like Kent said, you can use screen controller that handle this mecanism in an external class.
Just close in an EventHandler in code behind and handle everything else in the view model where you can use a command binding.