Adding identification to our smart-house application

As a part of our smart-house application, we want the application to recognize who we are. Doing so opens up the opportunity to get responses and actions from the application, tailored to you.

Create a new project for the smart-house application, based on the MVVM template we created earlier.

With the new project created, add the Microsoft.ProjectOxford.Face NuGet package.

As we will be building this application throughout this book, we will start small. In the MainView.xaml file, add a TabControl property containing two items. The two items should be two user controls, one called the AdministrationView.xaml file and the other called the HomeView.xaml file.

The administration control will be where we administer different parts of the application. The home control will be the starting point and the main control to use.

Add corresponding ViewModel instances to the Views. Make sure they are declared and created in MainViewModel.cs, as we have seen throughout this chapter. Make sure that the application compiles and runs before moving on.

Before we can go on to identify a person, we need to have something to identify them from. To identify a person, we need a PersonGroup property. This is a group that contains several Persons properties.

In the administration control, we will execute several operations in this regard. The UI should contain two textbox elements, two list box elements, and six buttons. The two textbox elements will allow us to input a name for the person group and a name for the person. One list box will list all person groups that we have available. The other will list all the persons in any given group.

We have buttons for each of the operations that we want to execute, which are as follows:

The View model should have two ObservableCollection properties: one of a PersonGroup type and the other of a Person type. We should also add three string properties. One will be for our person group name, the other for our person name. The last will hold some status text. We also want a PersonGroup property for the selected person group. Finally, we want a Person property holding the selected person.

In our View model, we want to add a private variable for the FaceServiceClient method, as shown in the following code:

This should be assigned in the constructor, which should accept a parameter of a FaceServiceClient type. It should also call an initialization function, which will initialize six ICommand properties. These maps to the buttons, created earlier. The initialization function should call the GetPersonGroups function to list all person groups available, as shown in the following code:

The ListPersonGroupsAsync function does not take any parameters, and returns a PersonGroup array if successfully executed, as shown in the following code:

We then check to see whether the array contains any elements. If it does, we clear out the existing PersonGroups list. Then we loop through each item of the PersonGroup array and add them to the PersonGroups list.

If no person groups exist, we can add a new one by filling in a name. The name you fill in here will also be used as a person group ID. This means that it can include numbers and English lowercase letters, the "-" character (hyphen), and the "_" character (underscore). The maximum length is 64 characters. When it is filled in, we can add a person group.

First, we call the DoesPersonGroupExistAsync function, specifying PersonGroupName as a parameter, as shown in the following code. If this is true, then the name we have given already exists, and as such, we are not allowed to add it. Note how we call the ToLower function on the name. This is so we are sure that the ID is in lowercase:

If the person group does not exist, we call the CreatePersonGroupAsync function, as shown in the following code. Again, we specify the PersonGroupName as lowercase in the first parameter. This represents the ID of the group. The second parameter indicates the name we want. We end the function by calling the GetPersonGroups function again, so we get the newly added group in our list:

The DoesPersonGroupExistAsync function makes one API call. It tries to call the GetPersonGroupAsync function, with the person group ID specified as a parameter. If the resultant PersonGroup list is anything but null, we return true.

To delete a person group, a group must be selected as follows:

The API call to the DeletePersonGroupAsync function requires a person group ID as a parameter. We get this from the selected person group. If no exception is caught, then the call has completed successfully, and we call the GetPersonGroups function to update our list.

When a person group is selected from the list, we make sure that we call the GetPersons function. This will update the list of persons, as follows:

We make sure the selected person group is not null. If it is not, we clear our persons list. The API call to the GetPersonsAsync function requires a person group ID as a parameter. A successful call will result in a Person array.

If the resultant array contains any elements, we loop through it. Each Person object is added to our persons list, as shown in the following code:

If no persons exist, we can add new ones. To add a new one, a person group must be selected, and a name of the person must be filled in. With this in place, we can click on the Add button:

The API call to the CreatePersonAsync function requires a person group ID as the first parameter. The next parameter is the name of the person. Optionally, we can add user data as a third parameter. In this case, it should be a string. When a new person has been created, we update the persons list by calling the GetPersons function again.

If we have selected a person group and a person, then we will be able to delete that person, as shown in the following code:

To delete a person, we make a call to the DeletePersonAsync function. This requires the person group ID of the person group the person lives in. It also requires the ID of the person we want to delete. If no exceptions are caught, then the call succeeded, and we call the GetPersons function to update our person list.

Our administration control now looks similar to the following screenshot:

Adding new persons

To identify a person, we are first going to upload an image. Open the HomeView.xaml file and add a ListBox element to the UI. This will contain the person groups to choose from when identifying a person. We will need to add a button element to find an image, upload it, and identify the person. A TextBox element is added to show the working response. For our own convenience, we also add an image element to show the image we are using.

In the View model, add an ObservableCollection property of a PersonGroup type. We need to add a property for the selected PersonGroup type. Also, add a BitmapImage property for our image, and a string property for the response. We will also need an ICommand property for our button.

Add a private variable for the FaceServiceClient type, as follows:

This will be assigned in our constructor, which should accept a parameter of a FaceServiceClient type. From the constructor, call on the Initialize function to initialize everything, as shown in the following code:

First, we call the GetPersonGroups function to retrieve all the person groups. This function makes a call to the ListPersonGroupsAsync API, which we saw earlier. The result is added to our PersonGroup list's ObservableCollection parameter.

Next, we create our ICommand object. The CanUploadOwnerImage function will return true if we have selected an item from the PersonGroup list. If we have not, it will return false, and we will not be able to identify anyone.

In the UploadOwnerImage function, we first browse to an image and then load it. With an image loaded and a file path available, we can start to identify the person in the image, as shown in the following code:

We open the image as a Stream type, as shown in the following code. Using this, we detect faces in the image. From the detected faces, we get all the face IDs in an array:

The array of face IDs will be sent as the second parameter to the IdentifyAsync API call. Remember that when we detect a face, it is stored for 24 hours. Proceeding to use the corresponding face ID will make sure that the service knows which face to use for identification.

The first parameter used is the ID of the person group we have selected. The last parameter in the call is the number of candidates returned. As we do not want to identify more than one person at a time, we specify one. Because of this, we should ensure that there is only one face in the image we upload.

A successful API call will result in an array of the IdentifyResult parameter, as shown in the following code. Each item in this array will contain candidates:

We loop through the array of results, as shown in the following code. If we do not have any candidates, we just break out of the loop. If, however, we do have candidates, we get the PersonId parameter of the first candidate (we asked for only one candidate earlier, so this is okay):

With the personId parameter, we get a single Person object, using the API to call the GetPersonAsync function. If the call is successful, we print a welcome message to the correct person (as shown in the following screenshot) and break out of the loop:

Identifying a person