It is highly recommended that you walk through the explanations by looking at the source code as it will help you gain a more solid understanding. You can also run the application on your computer. The goal of this project is to show you how to build a flexible and extensible application by using a very simple approach.
This is a WPF Master Detail application using the MVVM pattern at the UI tier. For now, let's just see what the application looks like.
There are a list of customer
s and each customer
can have multiple orders
. When you click on a customer
, it shows you the list of orders
belonging to that customer
:
data:image/s3,"s3://crabby-images/7bc68/7bc68d643a5310c8d9659688ceb6626637d2d74f" alt=""
You can add, edit, or delete customers
by clicking on the buttons. The Add and Edit Customer window uses the same View so they look the same. Deleting the customer
will delete all the orders
belonging to that customer
:
data:image/s3,"s3://crabby-images/5fba4/5fba47b473151b02b6ef6cc8753818cb13141bbc" alt=""
Similarly, you can add, edit, or delete an order
for a customer
. The Add and Edit Order window also uses the same View. There are validations to make sure the order information entered is correct. The button is enabled only after it passes the validations.
data:image/s3,"s3://crabby-images/3595f/3595f0437ca2bcf56894fed2547eb84d21b43d1c" alt=""
The UI tier is built using WPF MVVM pattern. It shows you how to:
- Design the
View
and the ViewModel
- Handle the user interactions
- Perform validation
- Open new windows using the MVVM pattern with an example of using Dependency Injection
The View and the ViewModel
Below is the diagram that shows you the main classes of the View
and the ViewModel
:
data:image/s3,"s3://crabby-images/fd204/fd20439f032296967d0cdbcb855374ea32c21015" alt=""
In the View
, we have:
MainWindow
-- the main window when the application starts CustomerListView
-- lists all the customers
, and can open the CustomerView
when clicked on the Add Customer or Edit Customer
CustomerView
-- the view to Add Customer or Edit Customer
OrderListView
-- lists all the orders belonging to the selected customer
, and can open the OrderView
when clicked on the Add Order or Edit Order
OrderView
-- the view to Add Order or Edit Order
The ViewModel
classes are designed for binding to the views. These classes are not meant to replicate the domain model which is already done in the Business tier.
The MainWindow
initializes the DataContext
for the application by assigning it to the CustomerListViewModel
singleton:
data:image/s3,"s3://crabby-images/a1204/a1204b4f936811e989bad0257f007f19a7640253" alt=""
In the XAML of the MainWindow
, it specifies the Views to use (CustomerListView
and OrderListView
) by using DataTemplates
:
data:image/s3,"s3://crabby-images/842dc/842dc66e77f9f3f595cc9b2d115a33af5c3629d9" alt=""
Notice that the CustomerListView
binds to the CustomerListViewModel
, and the OrderListView
binds to the CustomerListViewModel
's SelectedCustomer
property, which is a CustomerViewModel
that has a reference to the orders belonging to the customer. Below is the UI of the CustomerListView
:
data:image/s3,"s3://crabby-images/1e4a6/1e4a6aefc6cbcb349569c62a44a633a449b95107" alt=""
The CustomerListView
binds to the CustomerListViewModel
, and we specify the properties of the ViewModel
that we want each UI control to bind:
data:image/s3,"s3://crabby-images/1eb5e/1eb5eec0d4892182e3c93f684e628ab76a852b3c" alt=""
Notice that the CellTemplate
of the GridViewColumn
defines the Edit and the Delete buttons. But more importantly, the SelectedItem
of the ListView
is bound to the SelectedCustomer
of the ViewModel
, which shows the list of orders belonging to the customer
when the user clicks on the customer
. This is because OrderListView
is also bound to the SelectedCustomer
of the ViewModel
:
data:image/s3,"s3://crabby-images/f5634/f56345495ebcc5727920ebd94856200620af2257" alt=""
Below is the XAML of the OrderListView
:
data:image/s3,"s3://crabby-images/81e53/81e53eb12f08a036e18db91772942e49bfba751a" alt=""
The button checks to see if the DataContext
is null
(when no customer
is selected), if it is null
then the Add Order button is disabled. The ListView
binds to the Orders
property of the CustomerViewModel
to show the list of orders.
The CustomerView
window is opened when you click on the Add Customer or the Edit Customer button. Below is the UI of the CustomerView
which can be an Add Customer or an Edit Customer window:
data:image/s3,"s3://crabby-images/f2f2b/f2f2be3851942ed4ede30034e4279b925f5300f1" alt=""
The window determines if it is an Add or the Edit window is by looking at the Mode
property of the CustomerViewModel
. The Mode
can be either an Add or an Edit:
data:image/s3,"s3://crabby-images/9810d/9810d5989ef368a1a4f20a20b1cb675b1690d334" alt=""
The CustomerView
displays different texts by looking at the Mode
property:
data:image/s3,"s3://crabby-images/effb0/effb094b934f6d84368ba80a586199aed527a030" alt=""
Clicking on the Add or Save button on the window will call the Update()
method of the CustomerViewModel
, which determines if it should add or edit a customer
based on the Mode
property:
data:image/s3,"s3://crabby-images/217de/217de3352f166d0f2e7bf21aebeb660320b3248a" alt=""
The same concept is applied to the OrderView
.
Execute the Commands
To execute the commands, we write a CommandBase
base class that we can use in the ViewModel
. The base class merely passes the delegate that will be run when the command is executed:
data:image/s3,"s3://crabby-images/58888/58888db604fdd904a88dd74c81253ec6bd1edc42" alt=""
Then in the ViewModel
, we declare the commands that are available for the View to bind. For example, in the CustomerViewModel
, we define the command to delete the customer
:
data:image/s3,"s3://crabby-images/132fb/132fbbff3754b65e88e15136201daa2d95f11838" alt=""
Which will run the Delete()
method when called. Then in the CustomerListView,
we specify the command to execute when the delete button is clicked:
data:image/s3,"s3://crabby-images/415c3/415c35452b48f3c22776a2b6d2c5d0ee4d6be32b" alt=""
All of the commands are set up this way to facilitate the binding of the commands to the ViewModel
.
Setup Automatic UI Update
Next, we need to make sure the entire UI is updated when a property of the ViewModel
is changed. This is done by including a ViewModelBase
class in which all the ViewModel
s inherit:
data:image/s3,"s3://crabby-images/1e4a6/1e4a6aefc6cbcb349569c62a44a633a449b95107" alt=""
For example, the CustomerViewModel
inherits from the ViewModelBase
class:
data:image/s3,"s3://crabby-images/8a5b6/8a5b6f7ba55017de88bada0912a19ce188f91a95" alt=""
When the FirstName
property of the CustomerViewModel
changes (such as when editing a customer
), the entire UI should reflect such change. In the ViewModel
's property, we just call the OnPropertyChanged
method with the property name and WPF will take care of the rest:
data:image/s3,"s3://crabby-images/90eca/90eca8855befd6623ad5f5ea6dce688a9b0db3ca" alt=""
Undo After an UI Update
When a user cancels a change, you will encounter an issue with data binding. For example, the user may change the first name of a customer
in the edit customer window which will update the entire UI...
data:image/s3,"s3://crabby-images/bc2de/bc2de4be1a53e8574d6d154fd2cc65efbf75c7c9" alt=""
...then later decide to cancel by pressing the Cancel button. With the default two-way data binding, the UI will retain the changed value even though the user canceled the change.
The first thought may be to use one-way binding and commit the change only if the user clicks on the Save button. This will work except you will encounter a few more issues:
- You will need to pass multiple values to the save command, which clutters up the XAML and you will also need to cast the multi-value being passed from the command from a list of type
object.
- You will not be able to use the validation provided by WPF which requires data binding.
The way to solve this is to let WPF proceed with the binding and do a rollback when the cancel button is clicked. We store the original value using MemberwiseClone()
in the constructor and the Update()
method of the ViewModel
(although you can write your own deep copy methods if you have reference type fields):
data:image/s3,"s3://crabby-images/23aa7/23aa7b517c55cf3c58d60d86efd30119c00f9890" alt=""
Then the CancelCommand
in the ViewModel
will just call Undo()
and rollback to the original value and the entire UI will be updated:
data:image/s3,"s3://crabby-images/d239f/d239fdb0811fb244842e3ce6dba2ce8cd07e5118" alt=""
The same method is applied to the Edit Order screen.
Validation
Below is the OrderView
where the Save button is disabled with the validation message:
data:image/s3,"s3://crabby-images/bc4e3/bc4e3293653357f03c008f5ee5ffb1ebaf93876c" alt=""
The button is disabled based on the Validation.HasError
property of the textbox
es in the OrderView.xaml:
data:image/s3,"s3://crabby-images/d55b6/d55b62d8c65fae41326a24bf0f818f75923c6513" alt=""
To perform the validation on the textboxes, we set the ValidatesOnDataErrors
, NotifyOnValidationError
, and UpdateSourceTrigger
properties:
data:image/s3,"s3://crabby-images/a477c/a477c7e99b25e244b20468d6a0b6dac0160afd7a" alt=""
The PropertyChanged
value of the UpdateSourceTrigger
means that validation is performed on every keystroke in the textbox
. We define the UI to show when an error happens in ValidationStyle.xaml:
data:image/s3,"s3://crabby-images/543d9/543d9a2d0930bb1532adc727d9f40045cb7ad159" alt=""
The AdornedElementPlaceholder
shows the control being validated, and the ErrorContent
is the validation message string.
The validation check is done in the ViewModel
by implementing the IDataErrorInfo interface
:
data:image/s3,"s3://crabby-images/7fa91/7fa91842b7cf5917baeff300244ed97b85e2eefa" alt=""
In the implementation of the IDataErrorInfo interface
, we specify the message to display based on the condition:
data:image/s3,"s3://crabby-images/2fd95/2fd95b34b2fdff3cac528607f03867064f4e1c28" alt=""
Notice that we set the Quantity
property of the OrderViewModel
to string
instead of int
, this is so that we can display a more user friendly message when a non-integer is entered instead of the default message on casting which is not very intuitive to a regular user.
Open New Window
There are 2 ways to open a new window:
- Use the Code-Behind
- Use the ViewModel via Dependency Injection
Using the code-behind creates less code, while using the ViewModel
lets you abstract out the details on opening of a window by using an established framework.
Open Window Using the Code Behind
We open the Add/Edit Order window using the code-behind by assigning the DataContext
to the new window and open it:
data:image/s3,"s3://crabby-images/51bbd/51bbd5e5bcda2207c06bd338dc8224c4914e9bcb" alt=""
Open Window Using the ViewModel via Dependency Injection
Dependency Injection means you pre-register a type to an interface
, and when the interface
is requested, the pre-registered type is automatically created for the interface
. This is shown in the opening of the Add/Edit Customer
window. In the CustomerViewModel,
we define the command to open the Edit Customer window, which opens an IModalDialog
:
data:image/s3,"s3://crabby-images/a088f/a088fb5fcaba7306e748198e8af260fb5b12abb2" alt=""
Notice that the ViewModel
does not know what is the View being opened. It only needs to get the reference to the pre-registered IModalDialog
interface
and call its methods to open the window.
The pre-registration of the View to the IModalDialog
is done in the BootStrapper
when the application starts. In this case, we use the CustomerViewDialog
as the IModalDialog
when the interface
is requested:
data:image/s3,"s3://crabby-images/51bbd/51bbd5e5bcda2207c06bd338dc8224c4914e9bcb" alt=""
Also notice that we use the UnityServiceLocator
for managing the registrations, underneath the cover it simply uses the Unity Framework from the Microsoft Enterprise Library (you can use other frameworks if you like):
data:image/s3,"s3://crabby-images/d8bb2/d8bb2e7465265fab8ac98cc43ec45e926d93df53" alt=""
We hope you find this project useful. Although there are a lot more things you can do, this should give you a good start on how to build a flexible master details application.