Migrating the notes application to MVVM
In the previous tutorial, you created a note-taking app. Here, you’ll continue to develop the app and learn how to:
- Implement the model-view-viewmodel (MVVM) pattern.
- Use an additional style of query string for passing data during navigation.
The content of the tutorial is heavily based on the standard Microsoft tutorial which uses Visual Studio IDE as the development tool. This version has been modified to use Visual Studio Code (VSCode) but the result should be the same.
1. Understanding MVVM
The .NET MAUI developer experience typically involves creating a user interface in XAML, and then adding code-behind that operates on the user interface. Complex maintenance issues can arise as apps are modified and grow in size and scope. These issues include the tight coupling between the UI controls and the business logic, which increases the cost of making UI modifications, and the difficulty of unit testing such code.
The model-view-viewmodel (MVVM) pattern helps cleanly separate an app’s business and presentation logic from its user interface (UI). Maintaining a clean separation between app logic and the UI helps address numerous development issues and makes an app easier to test, maintain, and evolve. It can also significantly improve code reuse opportunities and allows developers and UI designers to collaborate more easily when developing their respective parts of an app.
The pattern
There are three core components in the MVVM pattern: the model, the view, and the viewmodel. Each serves a distinct purpose. The following diagram shows the relationships between the three components.
In addition to understanding the responsibilities of each component, it’s also important to understand how they interact. At a high level, the view “knows about” the viewmodel, and the viewmodel “knows about” the model, but the model is unaware of the viewmodel, and the viewmodel is unaware of the view. Therefore, the viewmodel isolates the view from the model, and allows the model to evolve independently of the view.
The key to using MVVM effectively lies in understanding how to factor app code into the correct classes and how the classes interact.
View
The view is responsible for defining the structure, layout, and appearance of what the user sees on screen. Ideally, each view is defined in XAML, with a limited code-behind that doesn’t contain business logic. However, in some cases, the code-behind might contain UI logic that implements visual behaviour that is difficult to express in XAML, such as animations.
Viewmodel
The viewmodel implements properties and commands to which the view can data bind to, and notifies the view of any state changes through change notification events. The properties and commands that the viewmodel provides define the functionality to be offered by the UI, but the view determines how that functionality is to be displayed.
The viewmodel is also responsible for coordinating the view’s interactions with any model classes that are required. There’s typically a one-to-many relationship between the viewmodel and the model classes.
Each viewmodel provides data from a model in a form that the view can easily consume. To accomplish this, the viewmodel sometimes performs data conversion. Placing this data conversion in the viewmodel is a good idea because it provides properties that the view can bind to. For example, the viewmodel might combine the values of two properties to make it easier to display by the view.
Model
Model classes are non-visual classes that encapsulate the app’s data. Therefore, the model can be thought of as representing the app’s domain model, which usually includes a data model along with business and validation logic.
2. Clean up the model
In the previous tutorial, the model types were acting both as the model (data) and as a viewmodel (data preparation), which was mapped directly to a view. The following table describes the model:
Code file | Description |
---|---|
Models/About.cs | The About model. Contains read-only fields that describe the app itself, such as the app title and version. |
Models/Note.cs | The Note model. Represents a note. |
Models/AllNotes.cs | The AllNotes model. Loads all of the notes on the device into a collection. |
Thinking about the app itself, there is only one piece of data that is used by the app, the Note
. Notes are loaded from the device, saved to the device, and edited through the app UI. There really isn’t a need for the About
and AllNotes
models. Remove these models from the project:
- Go to the Solution Explorer pane in VSCode.
- Right-click on the Models/About.cs file and select Delete. Click Yes to confirm.
- Right-click on the Models/AllNotes.cs file and select Delete. Click Yes to confirm.
The only model file remaining is the Models/Note.cs file.
3. Update the model
The Note
model contains:
- A unique identifier which is the file name of the note as stored on the device.
- The text of the note.
- A date to indicate when the note was created or last updated.
Currently, loading and saving the model was done through the views, and in some cases, by the other model types that you just removed. The code you have for the Note
type should be the following:
1
2
3
4
5
6
7
8
namespace Notes.Models;
internal class Note
{
public string? Filename { get; set; }
public string? Text { get; set; }
public DateTime? Date { get; set; }
}
The Note
model is going to be expanded to handle loading, saving, and deleting notes.
- In the Solution Explorer, click on Models\Note.cs.
- Remove the question marks from the datatypes in the property declarations. This was a temporary measure to allow the previous version of the code to run without error. From here on, we will not accept null values.
-
In the code editor, add the following two methods to the
Note
class. These methods are instance-based and handle saving or deleting the current note to or from the device, respectively:1 2 3 4 5
public void Save() => File.WriteAllText(System.IO.Path.Combine(FileSystem.AppDataDirectory, Filename), Text); public void Delete() => File.Delete(System.IO.Path.Combine(FileSystem.AppDataDirectory, Filename));
-
The app needs to load notes in two ways, loading an individual note from a file and loading all notes on the device. The code to handle loading can be static members, not requiring a class instance to run.
Add the following code to the class to load a note by file name:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
public static Note Load(string filename) { filename = System.IO.Path.Combine(FileSystem.AppDataDirectory, filename); if (!File.Exists(filename)) throw new FileNotFoundException("Unable to find file on local storage.", filename); return new() { Filename = Path.GetFileName(filename), Text = File.ReadAllText(filename), Date = File.GetLastWriteTime(filename) }; }
This code takes the file name as a parameter, builds the path to where the notes are stored on the device, and attempts to load the file if it exists.
-
The second way to load notes is to enumerate all notes on the device and load them into a collection.
Add the following code to the class:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
public static IEnumerable<Note> LoadAll() { // Get the folder where the notes are stored. string appDataPath = FileSystem.AppDataDirectory; // Use Linq extensions to load the *.notes.txt files. return Directory // Select the file names from the directory .EnumerateFiles(appDataPath, "*.notes.txt") // Each file name is used to load a note .Select(filename => Note.Load(Path.GetFileName(filename))) // With the final collection of notes, order them by date .OrderByDescending(note => note.Date); }
This code returns an enumerable collection of
Note
model types by retrieving the files on the device that match the notes file pattern: *.notes.txt. Each file name is passed to theLoad
method, loading an individual note. Finally, the collection of notes is ordered by the date of each note and returned to the caller. -
Lastly, add a constructor to the class which sets the default values for the properties, including a random file name:
1 2 3 4 5 6
public Note() { Filename = $"{Path.GetRandomFileName()}.notes.txt"; Date = DateTime.Now; Text = ""; }
The Note
class code should look like the following:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
namespace Notes.Models;
internal class Note
{
public string Filename { get; set; }
public string Text { get; set; }
public DateTime Date { get; set; }
public Note()
{
Filename = $"{Path.GetRandomFileName()}.notes.txt";
Date = DateTime.Now;
Text = "";
}
public void Save() =>
File.WriteAllText(System.IO.Path.Combine(FileSystem.AppDataDirectory, Filename), Text);
public void Delete() =>
File.Delete(System.IO.Path.Combine(FileSystem.AppDataDirectory, Filename));
public static Note Load(string filename)
{
filename = System.IO.Path.Combine(FileSystem.AppDataDirectory, filename);
if (!File.Exists(filename))
throw new FileNotFoundException("Unable to find file on local storage.", filename);
return
new()
{
Filename = Path.GetFileName(filename),
Text = File.ReadAllText(filename),
Date = File.GetLastWriteTime(filename)
};
}
public static IEnumerable<Note> LoadAll()
{
// Get the folder where the notes are stored.
string appDataPath = FileSystem.AppDataDirectory;
// Use Linq extensions to load the *.notes.txt files.
return Directory
// Select the file names from the directory
.EnumerateFiles(appDataPath, "*.notes.txt")
// Each file name is used to load a note
.Select(filename => Note.Load(Path.GetFileName(filename)))
// With the final collection of notes, order them by date
.OrderByDescending(note => note.Date);
}
}
Now that the Note
model is complete the viewmodels can be created.
4. Create the About viewmodel
Before adding viewmodels to the project, install the .NET Community Toolkit package which provides types and systems that help implement the MVVM pattern:
In the Solution Explorer, right-click on the Notes project and select Add NuGet Package…. When the next dialog appears, type in “CommunityToolkit” and press ENTER. Then select the CommunityToolkit.Mvvm package as shown below.
Decouple with viewmodels
The view-to-viewmodel relationship relies heavily on the binding system provided by .NET Multi-platform App UI (.NET MAUI). The app is already using binding in the views to display a list of notes and to present the text and date of a single note. The app logic is currently provided by the view’s code-behind and is directly tied to the view. For example, when a user is editing a note and presses the Save button, the Clicked
event for the button is raised. Then, the code-behind for the event handler saves the note text to a file and navigates to the previous screen.
Having app logic in the code-behind of a view can become an issue when the view changes. For example if the button is replaced with a different input control, or the name of a control is changed, event handlers may become invalid. Regardless of how the view is designed, the purpose of the view is to invoke some sort of app logic and to present information to the user. For this app, the Save button is saving the note and then navigating back to the previous screen.
The viewmodel gives the app a specific place to put the app logic regardless of how the UI is designed or how the data is being loaded or saved. The viewmodel is the glue that represents and interacts with the data model on behalf of the view.
The viewmodels are stored in a ViewModels folder.
- In the Solution Explorer right-click on the Notes project and select New Folder…. Name the folder ViewModels.
- Right-click on the ViewModels folder and select Add New File… to add a new class called AboutViewModel.cs.
- Repeat the previous step and create two more viewmodels:
- NoteViewModel.cs
- NotesViewModel.cs
Your project structure should look like the following image:
Note that VSCode shows that there is an error in AboutPage.xaml. That is because it contains a reference to the About
model which we have just deleted. We will resolve the invalid reference in the next section.
About viewmodel and About view
The About view displays some data on the screen and optionally navigates to a website with more information. Since this view doesn’t have any data to change, like with a text entry control or selecting items from a list, it’s a good candidate to demonstrate adding a viewmodel. For the About ViewModel, there isn’t a backing model.
Create the About ViewModel:
- In the Solution Explorer, click on ViewModels/AboutViewModel.cs.
-
Paste in the following code:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
using CommunityToolkit.Mvvm.Input; using System.Windows.Input; namespace Notes.ViewModels; internal class AboutViewModel { public string Title => AppInfo.Name; public string Version => AppInfo.VersionString; public string MoreInfoUrl => "https://aka.ms/maui"; public string Message => "This app is written in XAML and C# with .NET MAUI."; public ICommand ShowMoreInfoCommand { get; } public AboutViewModel() { ShowMoreInfoCommand = new AsyncRelayCommand(ShowMoreInfo); } async Task ShowMoreInfo() => await Launcher.Default.OpenAsync(MoreInfoUrl); }
The previous code snippet contains some properties that represent information about the app, such as the name and version. This snippet is exactly the same as the About model you deleted earlier. However, this viewmodel contains a new concept, the ShowMoreInfoCommand
command property.
Commands are bindable actions that invoke code, and are a great place to put app logic. In this example, the ShowMoreInfoCommand
points to the ShowMoreInfo
method, which opens the web browser to a specific page. You’ll learn more about the command system in the next section.
About view
The About view needs to be changed slightly to hook it up to the viewmodel that was created in the previous section. In the Views/AboutPage.xaml file, apply the following changes:
- Update the
xmlns:models
XML namespace toxmlns:viewModels
and target theNotes.ViewModels
.NET namespace. - Change the
ContentPage.BindingContext
property to a new instance of the About ViewModel. - Remove the button’s
Clicked
event handler and use theCommand
property.
Update the About view:
- In the Solution Explorer, click on Views/AboutPage.xaml.
-
Paste in the following code:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
<?xml version="1.0" encoding="utf-8" ?> <ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui" xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml" xmlns:viewModels="clr-namespace:Notes.ViewModels" x:Class="Notes.Views.AboutPage"> <ContentPage.BindingContext> <viewModels:AboutViewModel /> </ContentPage.BindingContext> <VerticalStackLayout Spacing="10" Margin="10"> <HorizontalStackLayout Spacing="10"> <Image Source="dotnet\_bot.png" SemanticProperties.Description="The dot net bot waving hello!" HeightRequest="64" /> <Label FontSize="22" FontAttributes="Bold" Text="{Binding Title}" VerticalOptions="End" /> <Label FontSize="22" Text="{Binding Version}" VerticalOptions="End" /> </HorizontalStackLayout> <Label Text="{Binding Message}" /> <Button Text="Learn more..." Command="{Binding ShowMoreInfoCommand}" /> </VerticalStackLayout> </ContentPage>
Notice that the button is using the Command
property. Many controls have a Command
property that is invoked when the user interacts with the control. When used with a button, the command is invoked when a user presses the button, similar to how the Clicked
event handler is invoked, except that you can bind Command
to a property in the viewmodel.
In this view, when the user presses the button, the Command
is invoked. The Command
is bound to the ShowMoreInfoCommand
property in the viewmodel, and when invoked, runs the code in the ShowMoreInfo
method, which opens the web browser to a specific page.
Clean up the About code-behind
The ShowMoreInfo
button isn’t using the event handler, so the LearnMore_Clicked
code should be removed from the Views/AboutPage.xaml.cs file. Delete that code, the class should only contain the constructor:
-
In the Solution Explorer, click on Views/AboutPage.xaml.cs.
Tip
You may need to expand Views/AboutPage.xaml to show the file.
-
Replace the code with the following snippet:
1 2 3 4 5 6 7 8 9
namespace Notes.Views; public partial class AboutPage : ContentPage { public AboutPage() { InitializeComponent(); } }
5. Create the Note viewmodel
The goal of updating the Note view is to move as much functionality as possible out of the XAML code-behind and put it in the Note viewmodel.
Note viewmodel
Based on what the Note view requires, the Note viewmodel needs to provide the following items:
- The text of the note.
- The date/time the note was created or last updated.
- A command that saves the note.
- A command that deletes the note.
Create the Note viewmodel:
- In the Solution Explorer, click on ViewModels/NoteViewModel.cs.
-
Replace the code in this file with the following snippet:
1 2 3 4 5 6 7 8 9 10 11
using CommunityToolkit.Mvvm.Input; using CommunityToolkit.Mvvm.ComponentModel; using System.Windows.Input; namespace Notes.ViewModels; internal class NoteViewModel : ObservableObject, IQueryAttributable { private Models.Note _note; }
This code is the blank
Note
viewmodel where you’ll add properties and commands to support the Note view. Notice that theCommunityToolkit.Mvvm.ComponentModel
namespace is being imported. This namespace provides theObservableObject
used as the base class. You’ll learn more aboutObservableObject
in the next step. TheCommunityToolkit.Mvvm.Input
namespace is also imported. This namespace provides some command-types that invoke methods asynchronously.The
Models.Note
model is being stored as a private field. The properties and methods of this class will use this field. -
Add the following properties to the class:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
public string Text { get => _note.Text; set { if (_note.Text != value) { _note.Text = value; OnPropertyChanged(); } } } public DateTime Date => _note.Date; public string Identifier => _note.Filename;
The
Date
andIdentifier
properties are simple properties that just retrieve the corresponding values from the model.Tip
For properties, the => syntax creates a get-only property where the statement to the right of => must evaluate to a value to return.
The
Text
property first checks if the value being set is a different value. If the value is different, that value is passed on to the model’s property, and theOnPropertyChanged
method is called.The
OnPropertyChanged
method is provided by theObservableObject
base class. This method uses the name of the calling code, in this case, the property name of Text, and raises theObservableObject.PropertyChanged
event. This event supplies the name of the property to any event subscribers. The binding system provided by .NET MAUI recognizes this event, and updates any related bindings in the UI. For the Note viewmodel, when theText
property changes, the event is raised, and any UI element that is bound to theText
property is notified that the property changed. -
Add the following command-properties to the class, which are the commands that the view can bind to:
1 2
public ICommand SaveCommand { get; private set; } public ICommand DeleteCommand { get; private set; }
-
Add the following constructors to the class:
1 2 3 4 5 6 7 8 9 10 11 12 13
public NoteViewModel() { _note = new Models.Note(); SaveCommand = new AsyncRelayCommand(Save); DeleteCommand = new AsyncRelayCommand(Delete); } public NoteViewModel(Models.Note note) { _note = note; SaveCommand = new AsyncRelayCommand(Save); DeleteCommand = new AsyncRelayCommand(Delete); }
These two constructors are used to either create the viewmodel with a new backing model, which is an empty note, or to create a viewmodel that uses the specified model instance.
The constructors also set up the commands for the viewmodel. Next, add the code for these commands.
-
Add the
Save
andDelete
methods:1 2 3 4 5 6 7 8 9 10 11 12
private async Task Save() { _note.Date = DateTime.Now; _note.Save(); await Shell.Current.GoToAsync($"..?saved={_note.Filename}"); } private async Task Delete() { _note.Delete(); await Shell.Current.GoToAsync($"..?deleted={_note.Filename}"); }
These methods are invoked by associated commands. They perform the related actions on the model and make the app navigate to the previous page. A query string parameter is added to the
..
navigation path, indicating which action was taken and the note’s unique identifier. -
Next, add the
ApplyQueryAttributes
method to the class, which satisfies the requirements of theIQueryAttributable
interface:1 2 3 4 5 6 7 8
void IQueryAttributable.ApplyQueryAttributes(IDictionary<string, object> query) { if (query.ContainsKey("load")) { _note = Models.Note.Load(query["load"].ToString()); RefreshProperties(); } }
When a page, or the binding context of a page, implements this interface, the query string parameters used in navigation are passed to the
ApplyQueryAttributes
method. This viewmodel is used as the binding context for the Note view. When the Note view is navigated to, the view’s binding context (this viewmodel) is passed the query string parameters used during navigation.This code checks if the
load
key was provided in thequery
dictionary. If this key is found, the value should be the identifier (the file name) of the note to load. That note is loaded and set as the underlying model object of this viewmodel instance. -
Finally, add these two helper methods to the class:
1 2 3 4 5 6 7 8 9 10 11
public void Reload() { _note = Models.Note.Load(_note.Filename); RefreshProperties(); } private void RefreshProperties() { OnPropertyChanged(nameof(Text)); OnPropertyChanged(nameof(Date)); }
The
Reload
method is a helper method that refreshes the backing model object, reloading it from device storage.The
RefreshProperties
method is another helper method to ensure that any subscribers bound to this object are notified that theText
andDate
properties have changed. Since the underlying model (the_note
field) is changed when the note is loaded during navigation, theText
andDate
properties aren’t actually set to new values. Since these properties aren’t directly set, any bindings attached to those properties wouldn’t be notified becauseOnPropertyChanged
isn’t called for each property.RefreshProperties
ensures bindings to these properties are refreshed.
The code for the class should look like the following snippet:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
using CommunityToolkit.Mvvm.Input;
using CommunityToolkit.Mvvm.ComponentModel;
using System.Windows.Input;
namespace Notes.ViewModels;
internal class NoteViewModel : ObservableObject, IQueryAttributable
{
private Models.Note _note;
public string Text
{
get => _note.Text;
set
{
if (_note.Text != value)
{
_note.Text = value;
OnPropertyChanged();
}
}
}
public DateTime Date => _note.Date;
public string Identifier => _note.Filename;
public ICommand SaveCommand { get; private set; }
public ICommand DeleteCommand { get; private set; }
public NoteViewModel()
{
_note = new Models.Note();
SaveCommand = new AsyncRelayCommand(Save);
DeleteCommand = new AsyncRelayCommand(Delete);
}
public NoteViewModel(Models.Note note)
{
_note = note;
SaveCommand = new AsyncRelayCommand(Save);
DeleteCommand = new AsyncRelayCommand(Delete);
}
private async Task Save()
{
_note.Date = DateTime.Now;
_note.Save();
await Shell.Current.GoToAsync($"..?saved={_note.Filename}");
}
private async Task Delete()
{
_note.Delete();
await Shell.Current.GoToAsync($"..?deleted={_note.Filename}");
}
void IQueryAttributable.ApplyQueryAttributes(IDictionary<string, object> query)
{
if (query.ContainsKey("load"))
{
_note = Models.Note.Load(query["load"].ToString());
RefreshProperties();
}
}
public void Reload()
{
\_note = Models.Note.Load(\_note.Filename);
RefreshProperties();
}
private void RefreshProperties()
{
OnPropertyChanged(nameof(Text));
OnPropertyChanged(nameof(Date));
}
}
Note view
Now that the viewmodel has been created, update the Note view. In the Views/NotePage.xaml file, apply the following changes:
- Add the
xmlns:viewModels
XML namespace that targets theNotes.ViewModels
.NET namespace. - Add a
BindingContext
to the page. - Remove the delete and save button
Clicked
event handlers and replace them with commands.
Update the Note view:
- In the Solution Explorer, click on Views/NotePage.xaml to open the XAML editor.
-
Paste in the following code:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
<?xml version="1.0" encoding="utf-8" ?> <ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui" xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml" xmlns:viewModels="clr-namespace:Notes.ViewModels" x:Class="Notes.Views.NotePage" Title="Note"> <ContentPage.BindingContext> <viewModels:NoteViewModel /> </ContentPage.BindingContext> <VerticalStackLayout Spacing="10" Margin="5"> <Editor x:Name="TextEditor" Placeholder="Enter your note" Text="{Binding Text}" HeightRequest="100" /> <Grid ColumnDefinitions="*,*" ColumnSpacing="4"> <Button Text="Save" Command="{Binding SaveCommand}"/> <Button Grid.Column="1" Text="Delete" Command="{Binding DeleteCommand}"/> </Grid> </VerticalStackLayout> </ContentPage>
Previously, this view didn’t declare a binding context, as it was supplied by the code-behind of the page itself. Setting the binding context directly in the XAML provides two things:
- At run-time, when the page is navigated to, it displays a blank note. This is because the parameterless constructor for the binding context, the viewmodel, is invoked.
-
The intellisense in the XAML editor shows the available properties as soon as you start typing
{Binding
syntax. The syntax is also validated and alerts you of an invalid value. Try changing the binding syntax for theSaveCommand
toSave123Command
. If you hover the mouse cursor over the text, you’ll notice that a tooltip is displayed informing you thatSave123Command
isn’t found. This notification isn’t considered an error because bindings are dynamic, it’s really a small warning that may help you notice when you typed the wrong property.If you changed the
SaveCommand
to a different value, restore it now.
Clean up the Note code-behind
Now that the interaction with the view has changed from event handlers to commands, open the Views\NotePage.xaml.cs file and replace all the code with a class that only contains the constructor:
- In the Solution Explorer, click on Views/NotePage.xaml.cs.
-
Replace the code with the following snippet:
1 2 3 4 5 6 7 8 9
namespace Notes.Views; public partial class NotePage : ContentPage { public NotePage() { InitializeComponent(); } }
6. Create the Notes ViewModel
The final viewmodel-view pair is the Notes viewmodel and AllNotes view. Currently though, the view is binding directly to the model, which was deleted at the start of this tutorial. The goal in updating the AllNotes view is to move as much functionality as possible out of the XAML code-behind and put it in the viewmodel. Again, the benefit being that the view can change its design with little effect to your code.
Notes viewmodel
Based on what the AllNotes view is going to display and what interactions the user will do, the Notes viewmodel must provide the following items:
- A collection of notes.
- A command to handle navigating to a note.
- A command to create a new note.
- Update the list of notes when one is created, deleted, or changed.
Create the Notes ViewModel:
- In the Solution Explorer, click on ViewModels/NotesViewModel.cs.
-
Replace the code in this file with the following code:
1 2 3 4 5 6 7 8 9 10
using CommunityToolkit.Mvvm.Input; using System.Collections.ObjectModel; using System.Windows.Input; using Notes.Models; namespace Notes.ViewModels; internal class NotesViewModel: IQueryAttributable { }
This code is the blank
NotesViewModel
where you’ll add properties and commands to support theAllNotes
view. -
In the
NotesViewModel
class code, add the following properties:1 2 3
public ObservableCollection<ViewModels.NoteViewModel> AllNotes { get; } public ICommand NewCommand { get; } public ICommand SelectNoteCommand { get; }
The
AllNotes
property is anObservableCollection
that stores all of the notes loaded from the device. The two commands will be used by the view to trigger the actions of creating a note or selecting an existing note. -
Add a parameterless constructor to the class, which initializes the commands and loads the notes from the model:
1 2 3 4 5 6
public NotesViewModel() { AllNotes = new ObservableCollection<ViewModels.NoteViewModel>(Models.Note.LoadAll().Select(n => new NoteViewModel(n))); NewCommand = new AsyncRelayCommand(NewNoteAsync); SelectNoteCommand = new AsyncRelayCommand<ViewModels.NoteViewModel>(SelectNoteAsync); }
Notice that the
AllNotes
collection uses theModels.Note.LoadAll
method to fill the observable collection with notes. TheLoadAll
method returns the notes as theModels.Note
type, but the observable collection is a collection ofViewModels.NoteViewModel
types. The code uses theSelect
Linq extension to create viewmodel instances from the note models returned fromLoadAll
. -
Create the methods targeted by the commands:
1 2 3 4 5 6 7 8 9 10
private async Task NewNoteAsync() { await Shell.Current.GoToAsync(nameof(Views.NotePage)); } private async Task SelectNoteAsync(ViewModels.NoteViewModel note) { if (note != null) await Shell.Current.GoToAsync($"{nameof(Views.NotePage)}?load={note.Identifier}"); }
Notice that the
NewNoteAsync
method doesn’t take a parameter while theSelectNoteAsync
does. Commands can optionally have a single parameter that is provided when the command is invoked. For theSelectNoteAsync
method, the parameter represents the note that’s being selected. -
Finally, implement the
IQueryAttributable.ApplyQueryAttributes
method:1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
void IQueryAttributable.ApplyQueryAttributes(IDictionary<string, object> query) { if (query.ContainsKey("deleted")) { string noteId = query["deleted"].ToString(); NoteViewModel matchedNote = AllNotes.Where((n) => n.Identifier == noteId).FirstOrDefault(); // If note exists, delete it if (matchedNote != null) AllNotes.Remove(matchedNote); } else if (query.ContainsKey("saved")) { string noteId = query["saved"].ToString(); NoteViewModel matchedNote = AllNotes.Where((n) => n.Identifier == noteId).FirstOrDefault(); // If note is found, update it if (matchedNote != null) matchedNote.Reload(); // If note isn't found, it's new; add it. else AllNotes.Add(new NoteViewModel(Note.Load(noteId))); } }
The Note viewmodel created in the previous tutorial step, used navigation when the note was saved or deleted. The viewmodel navigated back to the AllNotes view, which this viewmodel is associated with. This code detects if the query string contains either the
deleted
orsaved
key. The value of the key is the unique identifier of the note.If the note was deleted, that note is matched in the
AllNotes
collection by the provided identifier, and removed.There are two possible reasons a note is saved. The note was either just created or an existing note was changed. If the note is already in the
AllNotes
collection, it’s a note that was updated. In this case, the note instance in the collection just needs to be refreshed. If the note is missing from the collection, it’s a new note and must be added to the collection.
The code for the class should look like the following snippet:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
using CommunityToolkit.Mvvm.Input;
using Notes.Models;
using System.Collections.ObjectModel;
using System.Windows.Input;
namespace Notes.ViewModels;
internal class NotesViewModel : IQueryAttributable
{
public ObservableCollection<ViewModels.NoteViewModel> AllNotes { get; }
public ICommand NewCommand { get; }
public ICommand SelectNoteCommand { get; }
public NotesViewModel()
{
AllNotes = new ObservableCollection<ViewModels.NoteViewModel>(Models.Note.LoadAll().Select(n => new NoteViewModel(n)));
NewCommand = new AsyncRelayCommand(NewNoteAsync);
SelectNoteCommand = new AsyncRelayCommand<ViewModels.NoteViewModel>(SelectNoteAsync);
}
private async Task NewNoteAsync()
{
await Shell.Current.GoToAsync(nameof(Views.NotePage));
}
private async Task SelectNoteAsync(ViewModels.NoteViewModel note)
{
if (note != null)
await Shell.Current.GoToAsync($"{nameof(Views.NotePage)}?load={note.Identifier}");
}
void IQueryAttributable.ApplyQueryAttributes(IDictionary query)
{
if (query.ContainsKey("deleted"))
{
string noteId = query["deleted"].ToString();
NoteViewModel matchedNote = AllNotes.Where((n) => n.Identifier == noteId).FirstOrDefault();
// If note exists, delete it
if (matchedNote != null)
AllNotes.Remove(matchedNote);
}
else if (query.ContainsKey("saved"))
{
string noteId = query["saved"].ToString();
NoteViewModel matchedNote = AllNotes.Where((n) => n.Identifier == noteId).FirstOrDefault();
// If note is found, update it
if (matchedNote != null)
matchedNote.Reload();
// If note isn't found, it's new; add it.
else
AllNotes.Add(new NoteViewModel(Note.Load(noteId)));
}
}
}
AllNotes view
Now that the viewmodel has been created, update the AllNotes view to point to the viewmodel properties. In the Views/AllNotesPage.xaml file, apply the following changes:
- Add the
xmlns:viewModels
XML namespace that targets theNotes.ViewModels
.NET namespace. - Add a
BindingContext
to the page. - Remove the toolbar button’s
Clicked
event and use theCommand
property. - Change the
CollectionView
to bind itsItemSource
toAllNotes
. - Change the
CollectionView
to use commanding to react to when the selected item changes.
Update the AllNotes view:
- In the Solution Explorer, click on Views/AllNotesPage.xaml.
-
Paste in the following code:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui" xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml" xmlns:viewModels="clr-namespace:Notes.ViewModels" x:Class="Notes.Views.AllNotesPage" Title="Your Notes"> <ContentPage.BindingContext> <viewModels:NotesViewModel /> </ContentPage.BindingContext> <!-- Add an item to the toolbar --> <ContentPage.ToolbarItems> <ToolbarItem Text="Add" Command="{Binding NewCommand}" IconImageSource="{FontImage Glyph='+', Color=Black, Size=22}" /> </ContentPage.ToolbarItems> <!-- Display notes in a list --> <CollectionView x:Name="notesCollection" ItemsSource="{Binding AllNotes}" Margin="20" SelectionMode="Single" SelectionChangedCommand="{Binding SelectNoteCommand}" SelectionChangedCommandParameter="{Binding Source={RelativeSource Self}, Path=SelectedItem}"> <!-- Designate how the collection of items are laid out --> <CollectionView.ItemsLayout> <LinearItemsLayout Orientation="Vertical" ItemSpacing="10" /> </CollectionView.ItemsLayout> <!-- Define the appearance of each item in the list --> <CollectionView.ItemTemplate> <DataTemplate> <StackLayout> <Label Text="{Binding Text}" FontSize="22"/> <Label Text="{Binding Date}" FontSize="14" TextColor="Silver"/> </StackLayout> </DataTemplate> </CollectionView.ItemTemplate> </CollectionView> </ContentPage>
The toolbar no longer uses the Clicked
event and instead uses a command.
The CollectionView
supports commanding with the SelectionChangedCommand
and SelectionChangedCommandParameter
properties. In the updated XAML, the SelectionChangedCommand
property is bound to the viewmodel’s SelectNoteCommand
, which means the command is invoked when the selected item changes. When the command is invoked, the SelectionChangedCommandParameter
property value is passed to the command.
Look at the binding used for the CollectionView
:
1
2
3
4
5
6
<CollectionView x:Name="notesCollection"
ItemsSource="{Binding AllNotes}"
Margin="20"
SelectionMode="Single"
SelectionChangedCommand="{Binding SelectNoteCommand}"
SelectionChangedCommandParameter="{Binding Source={RelativeSource Self}, Path=SelectedItem}">
The SelectionChangedCommandParameter
property uses Source={RelativeSource Self}
binding. The Self
references the current object, which is the CollectionView
. Notice that the binding path is the SelectedItem
property. When the command is invoked by changing the selected item, the SelectNoteCommand
command is invoked and the selected item is passed to the command as a parameter.
Clean up the AllNotes code-behind
Now that the interaction with the view has changed from event handlers to commands, open the Views\AllNotesPage.xaml.cs file and replace all the code with a class that only contains the constructor:
- In the Solution Explorer, click on Views/AllNotesPage.xaml.cs.
-
Replace the code with the following snippet:
1 2 3 4 5 6 7 8 9
namespace Notes.Views; public partial class AllNotesPage : ContentPage { public AllNotesPage() { InitializeComponent(); } }
Run the app
You can now run the app, and everything is working. However, there are two problems with how the app behaves:
- If you select a note, which opens the editor, press Save, and then try to select the same note, it doesn’t work.
- Whenever a note is changed or added, the list of notes isn’t reordered to show the latest notes at the top.
These two problems are fixed in the next tutorial step.
7. Fix the app behaviour
Now that the app code can compile and run, you’ll likely have noticed that there are two flaws with how the app behaves. The app doesn’t let you reselect a note that’s already selected, and the list of notes isn’t reordered after a note is created or changed.
Get notes to the top of the list
First, fix the reordering problem with the notes list. In the ViewModels/NotesViewModel.cs file, the AllNotes
collection contains all of the notes to be presented to the user. Unfortunately, the downside to using an ObservableCollection
is that it must be manually sorted. To get the new or updated items to the top of the list, perform the following steps:
- In the Solution Explorer, click on ViewModels/NotesViewModel.cs.
- In the
ApplyQueryAttributes
method, look at the logic for the saved query string key. -
When the
matchedNote
isn’t null, the note is being updated. Use theAllNotes.Move
method to move thematchedNote
to index 0, which is the top of the list.1 2 3 4 5 6 7 8 9
string noteId = query["saved"].ToString(); NoteViewModel matchedNote = AllNotes.Where((n) => n.Identifier == noteId).FirstOrDefault(); // If note is found, update it if (matchedNote != null) { matchedNote.Reload(); AllNotes.Move(AllNotes.IndexOf(matchedNote), 0); }
The
AllNotes.Move
method takes two parameters to move an object’s position in the collection. The first parameter is the index of the object to move, and the second parameters is the index of where to move the object. TheAllNotes.IndexOf
method retrieves the index of the note. -
When the
matchedNote
is null, the note is new and is being added to the list. Instead of adding it, which appends the note to the end of the list, insert the note at index 0, which is the top of the list. Change theAllNotes.Add
method toAllNotes.Insert
.1 2 3 4 5 6 7 8 9 10 11 12
string noteId = query["saved"].ToString(); NoteViewModel matchedNote = AllNotes.Where((n) => n.Identifier == noteId).FirstOrDefault(); // If note is found, update it if (matchedNote != null) { matchedNote.Reload(); AllNotes.Move(AllNotes.IndexOf(matchedNote), 0); } // If note isn't found, it's new; add it. else AllNotes.Insert(0, new NoteViewModel(Models.Note.Load(noteId)));
The ApplyQueryAttributes method should look like the following code snippet:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
void IQueryAttributable.ApplyQueryAttributes(IDictionary<string, object> query)
{
if (query.ContainsKey("deleted"))
{
string noteId = query["deleted"].ToString();
NoteViewModel matchedNote = AllNotes.Where((n) => n.Identifier == noteId).FirstOrDefault();
// If note exists, delete it
if (matchedNote != null)
AllNotes.Remove(matchedNote);
}
else if (query.ContainsKey("saved"))
{
string noteId = query["saved"].ToString();
NoteViewModel matchedNote = AllNotes.Where((n) => n.Identifier == noteId).FirstOrDefault();
// If note is found, update it
if (matchedNote != null)
{
matchedNote.Reload();
AllNotes.Move(AllNotes.IndexOf(matchedNote), 0);
}
// If note isn't found, it's new; add it.
else
AllNotes.Insert(0, new NoteViewModel(Models.Note.Load(noteId)));
}
}
Allow selecting a note twice
In the AllNotes view, the CollectionView
lists all of the notes, but doesn’t allow you to select the same note twice. There are two ways the item remains selected: when the user changes an existing note, and when the user forcibly navigates backwards. The case where the user saves a note is fixed with the code change in previous section that uses AllNotes.Move
, so you don’t have to worry about that case.
The problem you have to solve now is related to navigation. No matter how the Allnotes view is navigated to, the NavigatedTo
event is raised for the page. This event is a perfect place to forcibly unselect the selected item in the CollectionView
.
However, with the MVVM pattern being applied here, the viewmodel can’t trigger something directly on the view, such as clearing the selected item after the note is saved. So how do you get that to happen? A good implementation of the MVVM pattern minimizes the code-behind in the view. There are a few different ways to solve this problem to support the MVVM separation pattern. However, it’s also OK to put code in the code-behind of the view, especially when it’s directly tied to the view. MVVM has many great designs and concepts that help you compartmentalize your app, improving maintainability and making it easier for you to add new features. However, in some cases, you may find that MVVM encourages overengineering.
Don’t overengineer a solution for this problem, and just use the NavigatedTo
event to clear the selected item from the CollectionView
.
- In the Solution Explorer, click on Views/AllNotesPage.xaml.
-
In the XAML for the , add the
viewModels
namespace reference,NavigatedTo
event and theBindingContext
definition (NB. Don’t overwrite the whole file):1 2 3 4 5 6 7 8 9
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui" xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml" xmlns:viewModels="clr-namespace:Notes.ViewModels" x:Class="Notes.Views.AllNotesPage" Title="Your Notes" NavigatedTo="ContentPage\_NavigatedTo"> <ContentPage.BindingContext> <viewModels:NotesViewModel /> </ContentPage.BindingContext>
- Open the Views/AllNotesPage.xaml.cs in the code editor.
-
Add the
ContentPage_NavigatedTo
handler function to the class:1 2 3 4
private void ContentPage_NavigatedTo(object sender, NavigatedToEventArgs e) { notesCollection.SelectedItem = null; }
In the XAML, the
CollectionView
was given the name ofnotesCollection
. This code uses that name to access theCollectionView
, and setSelectedItem
to null. The selected item is cleared every time the page is navigated to.
Now, run your app. Try to navigate to a note, press the back button, and select the same note a second time. The app behaviour is fixed!