Maui Docs
Maui Docs
.NET MAUI
What is .NET Multi-platform App UI?
Supported platforms
Get started
Installation
Build your first app
Migrate from Xamarin.Forms
Android
Emulator
How to enable hardware acceleration
Manage and create virtual devices
Edit virtual devices
Debug on a virtual device
Troubleshoot common problems
Devices
How to set up a physical device
Windows
How to set up Windows for debugging
XAML
Overview
Fundamentals
Get started
Essential syntax
Markup extensions
Data binding basics
Data binding and MVVM
Compilation
Field modifiers
Generics
Markup extensions
Consume markup extensions
Create markup extensions
Namespaces
Overview
Custom namespace schemas
Custom namespace prefixes
Pass arguments
Runtime loading
Fundamentals
Accessibility
App lifecycle
App startup
Behaviors
Data binding
Overview
Basic bindings
Binding mode
String formatting
Binding path
Binding value converters
Relative bindings
Binding fallbacks
Multi-bindings
Commanding
Compiled bindings
Gestures
Drag and drop
Pan
Pinch
Swipe
Tap
Properties
Bindable properties
Attached properties
Publish and subscribe to messages
Resource dictionaries
Single project
Templates
Control templates
Data templates
Triggers
User interface
Animation
Basic animation
Easing functions
Custom animation
Brushes
Overview
Solid colors
Gradients
Overview
Linear gradients
Radial gradients
Controls
Overview
Customize controls
Layouts
Overview
AbsoluteLayout
BindableLayout
FlexLayout
Grid
HorizontalStackLayout
RelativeLayout
StackLayout
VerticalStackLayout
Pages
ContentPage
FlyoutPage
NavigationPage
TabbedPage
Views
Present data
BlazorWebView
Border
BoxView
Frame
GraphicsView
Image
Label
ScrollView
Shapes
WebView
Initiate commands
Button
ImageButton
RadioButton
RefreshView
SearchBar
SwipeView
Set values
CheckBox
DatePicker
Slider
Stepper
Switch
TimePicker
Edit text
Editor
Entry
Indicate activity
ActivityIndicator
ProgressBar
Display collections
CarouselView
CollectionView
IndicatorView
ListView
Picker
TableView
ContentView
Display pop-ups
Graphics
Overview
Blend modes
Colors
Draw graphical objects
Images
Paint graphical objects
Transforms
Winding modes
Images
App icons
Splash screen
Shadows
Styles
Style apps using XAML
Style apps using CSS
Theming
Theme an app
Respond to system theme changes
Visual states
Platform integration
Configure multi-targeting
Invoke platform code
What is .NET MAUI?
3/18/2022 • 5 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) is a cross-platform framework for creating native mobile and desktop
apps with C# and XAML.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Using .NET MAUI, you can develop apps that can run on Android, iOS, macOS, and Windows from a single
shared code-base.
.NET MAUI is open-source and is the evolution of Xamarin.Forms, extended from mobile to desktop scenarios,
with UI controls rebuilt from the ground up for performance and extensibility. If you've previously used
Xamarin.Forms to build cross-platform user interfaces, you'll notice many similarities with .NET MAUI. However,
there are also some differences. Using .NET MAUI, you can create multi-platform apps using a single project, but
you can add platform-specific source code and resources if necessary. One of the key aims of .NET MAUI is to
enable you to implement as much of your app logic and UI layout as possible in a single code-base.
In a .NET MAUI app, you write code that primarily interacts with the .NET MAUI API (1). .NET MAUI then directly
consumes the native platform APIs (3). In addition, app code may directly exercise platform APIs (2), if required.
.NET MAUI apps can be written on PC or Mac, and compile into native app packages:
Android apps built using .NET MAUI compile from C# into intermediate language (IL) which is then just-in-
time (JIT) compiled to a native assembly when the app launches.
iOS apps built using .NET MAUI are fully ahead-of-time (AOT) compiled from C# into native ARM assembly
code.
macOS apps built using .NET MAUI use Mac Catalyst, a solution from Apple that brings your iOS app built
with UIKit to the desktop, and augments it with additional AppKit and platform APIs as required.
Windows apps built using .NET MAUI use Windows UI 3 (WinUI 3) library to create native apps that target
the Windows desktop. For more information about WinUI 3, see Windows UI Library.
NOTE
Building apps for iOS and macOS requires a Mac.
.NET Multi-platform App UI (.NET MAUI) apps can be written for the following platforms:
Android 5.0 (API 21) or higher.
iOS 10 or higher.
macOS 10.13 or higher (using Mac Catalyst).
Windows desktop and the Universal Windows Platform (UWP), using Windows UI Library (WinUI) 3.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
.NET MAUI apps for Android, iOS, and Windows can be built in Visual Studio. However, a networked Mac is
required for iOS development. From .NET MAUI Preview 6, the minimum required version of Xcode is 13.0 Beta
1.
.NET MAUI apps for Android, iOS, and macOS can be built in Visual Studio for Mac.
To create .NET Multi-platform App UI (.NET MAUI) apps, you need the latest preview versions of Visual Studio
2022 and .NET 6.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Visual Studio
To create .NET MAUI apps, you'll need the latest Visual Studio 2022 17.2 Preview. Either install Visual Studio, or
modify your installation, and install the following workload:
Mobile development with .NET
Next steps
To learn how to create and run your first .NET MAUI app, continue to build your first app.
Build your first app
Build your first app
3/18/2022 • 3 minutes to read • Edit Online
In this tutorial, you'll learn how to create and run your first .NET Multi-platform App UI (.NET MAUI) app.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Prerequisites
The latest preview of Visual Studio 2022 17.2 with the required workload. For more information, see
Installation.
Hardware acceleration must be enabled to maximize Android emulator performance. For more information,
see How to enable hardware acceleration with Android emulators (Hyper-V & HAXM).
2. In the Create a new project window, select MAUI in the Project type drop-down, select the .NET
MAUI App (Preview) template, and click the Next button:
3. In the Configure your new project window, name your project, choose a suitable location for it, and
click the Create button:
5. In the Android SDK License Acceptance window, press the Accept button:
Wait for Visual Studio to download, unzip, and create an Android emulator.
11. Close the Android Device Manager window:
12. In the Visual Studio toolbar, press the Pixel 5 - API 30 (Android 11.0 - API 30) button to build and
run the app:
Visual Studio will start the Android emulator, build the app, and deploy the app to the emulator.
WARNING
Hardware acceleration must be enabled to maximize Android emulator performance. Failure to do this will result in
the emulator running very slowly. For more information, see How to enable hardware acceleration with Android
emulators (Hyper-V & HAXM).
13. In the running app in the Android emulator, press the CLICK ME button several times and observe that
the count of the number of button clicks is incremented.
7. In the Visual Studio toolbar, use the Debug Target drop down to selectFramework and then the
net6.0-windows entry.
8. Press the Windows Machine button to deploy the app.
If you've not enabled Developer Mode, the Settings app should open to the appropriate page. Turn on
Developer Mode and accept the disclaimer.
9. In the running app, press the Click me button several times and observe that the count of the number of
button clicks is incremented.
Migrate your app from Xamarin.Forms
3/18/2022 • 2 minutes to read • Edit Online
You don't need to rewrite your Xamarin.Forms apps to move them to .NET Multi-platform App UI (.NET MAUI).
However, you need to make a small amount of code changes to each app. Similarly, you can use single-project
features without merging all of your Xamarin.Forms projects into one project.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The process for migrating a Xamarin.Forms app to .NET MAUI is expected to be:
1. Use the .NET Upgrade Assistant for .NET MAUI to migrate your Xamarin.Forms projects to .NET MAUI single
project, and perform well-known code namespace changes.
2. Update any dependencies to .NET 6 and .NET MAUI compatible versions.
3. Register any compatibility services or renderers.
4. Build and fix any issues.
5. Run the converted app and verify that it functions correctly.
WARNING
Do not currently migrate your production apps to .NET MAUI.
cd ButtonDemos
dotnet restore
NOTE
If you are unable to restore project dependencies, ensure that you have the latest .NET MAUI preview installed.
3. Copy the code files (except App.xaml) into the newly created project.
4. In the newly created project, replace the following namespaces:
O L D N A M ESPA C E N EW N A M ESPA C E
xmlns="http://xamarin.com/schemas/2014/forms" xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
6. Examine any run-time errors, some of which will be due to incomplete handler availability.
How to enable hardware acceleration with Android
emulators (Hyper-V & HAXM)
3/18/2022 • 4 minutes to read • Edit Online
This article explains how to use your computer's hardware acceleration features to maximize Android emulator
performance.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
With Visual Studio, you can easily test and debug your .NET MAUI app for Android in situations where an
Android device isn't available. However, if hardware acceleration isn't available or enabled, the emulator will run
too slow. You can drastically improve the performance of the emulator by enabling hardware acceleration and
using x86-64 or x86 virtual device images.
SC EN A RIO H A XM WH PX H Y P ERVISO R. F RA M EW O RK
IMPORTANT
You can't run a VM-accelerated emulator inside another VM, such as a VM hosted by VirtualBox, VMware, or Docker
(unless using WSL2). You must run the Android emulator directly on your system hardware.
For information about launching and debugging with the Android emulator, see Debugging on the Android
Emulator.
systeminfo
If all listed Hyper-V requirements have a value of Yes , then your computer can support Hyper-V. For example:
If the Hyper-V result indicates that a hypervisor is currently running, Hyper-V is already enabled.
Enabling Hyper-V acceleration in Windows and the emulator
If your computer meets the above criteria, use the following steps to accelerate the Android emulator with
Hyper-V:
1. Enter windows features in the Windows search box and select Turn Windows features on or off in
the search results. In the Windows Features dialog, enable both Hyper-V and Windows Hyper visor
Platform :
After making these changes, reboot your computer.
IMPORTANT
On Windows 10 October 2018 Update (RS5) and higher, you only need to enable Hyper-V, as it will use Windows
Hypervisor Platform (WHPX) automatically.
1. Make sure that the virtual device you created in the Android Device Manager is an x86-64 or x86 -based
system image. If you use an Arm-based system image, the virtual device won't be accelerated and will run
slowly.
After Hyper-V is enabled, you'll be able to run your accelerated Android emulator.
If your computer doesn't support Hyper-V, you may use HAXM to accelerate the Android emulator. To use HAXM,
disable Device Guard.
Verifying HAXM support
To determine if your hardware supports HAXM, follow the steps in Does My Processor Support Intel
Virtualization Technology?. If your hardware supports HAXM, you can check to see if HAXM is already installed
by using the following steps:
1. Open a command prompt window and enter the following command:
sc query intelhaxm
2. Examine the output to see if the HAXM process is running. If it is, you should see output listing the
intelhaxm state as RUNNING . For example:
If STATE isn't set to RUNNING , then HAXM isn't installed.
If your computer can support HAXM but HAXM isn't installed, use the steps in the next section to install HAXM.
Installing HAXM
HAXM install packages for Windows are available from the Intel Hardware Accelerated Execution Manager
GitHub releases page. Use the following steps to download and install HAXM:
1. From the Intel website, download the latest HAXM virtualization engine installer for Windows. The
advantage of downloading the HAXM installer directly from the Intel website is that you can be assured
of using the latest version.
2. Run intelhaxm-android.exe to start the HAXM installer. Accept the default values in the installer
dialogs.
When you create a virtual device, be sure to select an x86_64 or x86 -based system image. If you use an Arm-
based system image, the virtual device will not be accelerated and will run slowly.
Troubleshooting
For help with troubleshooting hardware acceleration issues, see the Android emulator Troubleshooting guide.
Related Links
Run Apps on the Android Emulator
Managing virtual devices with the Android Device
Manager
3/18/2022 • 6 minutes to read • Edit Online
This article explains how to use the Android Device Manager to create and configure Android Virtual Devices
(AVDs) that emulate physical Android devices. You can use these virtual devices to run and test your app without
having to rely on a physical device.
IMPORTANT
Enable hardware acceleration for the Android devices. For more information, see Hardware Acceleration for Emulator
Performance.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Requirements
To use the Android Device Manager, you'll need the following items:
Visual Studio 2022: Community, Professional, and Enterprise editions are supported.
The Android SDK API Level 30 or later. Be sure to install the Android SDK at its default location if it
isn't already installed: C:\Program Files (x86)\Android\android-sdk.
The following packages must be installed:
Android SDK Tools 5.0 or later
Android SDK Platform-Tools 31.0.3 or later
Android SDK Build-Tools 30.0.2 or later
Android Emulator 30.8.4 or later
These packages should be displayed with Installed status as seen in the following screenshot:
When you install the Mobile development with .NET workload in Visual Studio, everything is installed for
you. For more information on setting up .NET MAUI with Visual Studio, see Installation.
Main screen
When you run the Android Device Manager, it presents a screen that displays all currently-configured virtual
devices. For each virtual device, the Name , OS (Android Version), Processor , Memor y size, and screen
Resolution are displayed:
When you select a device in the list, the Star t button appears on the right. Press the Star t button to launch the
emulator with this virtual device. If the emulator is running with the selected virtual device, the Star t button
changes to a Stop button that you can use to halt the emulator.
Create a new device
To create a new device, press the New button:
The New Device window is displayed. To configure the device, follow these steps:
1. Give the device a new name. In the following example, the new device is named Pixel 3a - API 31.
2. Select a physical device to emulate by selecting a device in the Base Device box.
3. Select a processor type for this virtual device with the Processor box.
It's recommended that you choose x86_64 and enable hardware acceleration.
4. Select the Android version (API level) with the OS box.
If you select an Android API level that has not yet been installed, the Device Manager will display A new
device will be downloaded message at the bottom of the screen – it will download and install the
necessary files as it creates the new virtual device.
5. If you want to include Google Play Services APIs in your virtual device, select the Google APIs option. To
include the Google Play Store app on the virtual device, select the Google Play Store option
NOTE
Google Play Store images are available only for some base device types such as Pixel, Pixel 2, Pixel 3, and Nexus 5.
This is indicated by the text (+ Store) in the image name.
6. Use the property list to change some of the most commonly-modified properties. To make changes to
properties, see Editing Android Virtual Device Properties.
7. Add any additional properties that you need to explicitly set with the Add Proper ty box at the bottom of
the window:
You might get a License Acceptance screen when you create the device. Select Accept if you agree to
the license terms.
9. The Android Device Manager adds the new device to the list of installed virtual devices while displaying a
Creating progress indicator during device creation:
10. When the creation process is complete, the new device is shown in the list of installed virtual devices with
a Star t button, ready to launch
Edit device
To edit an existing virtual device, select the device and then press the Edit button:
Pressing Edit displays the Device Editor window for the selected virtual device.
The Device Editor window lists the properties of the virtual device under the Proper ty column, with the
corresponding values of each property in the Value column. When you select a property, a detailed description
of that property is displayed on the right.
To change a property, edit its value in the Value column. For example, in the following screenshot the
hw.lcd.density property is being changed to 240 :
After you've made the necessary configuration changes, press the Save button. For more information about
changing virtual device properties, see Editing Android Virtual Device Properties.
Additional options
Additional options for working with devices are available from the Additional Options (…) pull-down menu:
Troubleshooting
The following sections explain how to diagnose and work around problems that may occur when using the
Android Device Manager to configure virtual devices.
Wrong version of Android SDK Tools
If you have thew wrong Android SDK tools installed, installed, you may see this error dialog on launch:
If you see that error dialog, press Open SDK Manager to open the Android SDK Manager. In the Android SDK
Manager, go to the Tools tab and install the following packages:
Android SDK Command-line Tools 5.0 or later
Android SDK Platform-Tools 31.0.3 or later
Android SDK Build-Tools 30.0.3 or later
For more information, see Installation.
Snapshot disables Wi=Fi on Android Oreo
If you've an AVD configured for Android Oreo with simulated Wi-Fi access, restarting the AVD after a snapshot
may cause Wi-Fi access to become disabled.
To work around this problem,
1. Open the Android Device Manager .
2. Select the AVD in the Android Device Manager.
3. From the Additional Options (…) menu, select Reveal in Explorer .
4. Navigate to the snapshots > default_boot folder.
5. Delete the snapshot.pb file:
This article explains how to use the Android Device Manager (AVD) to edit the profile properties of an Android
virtual device.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
When you select a property, a detailed description of that property is displayed on the right. You can modify
hardware profile properties and AVD properties. Hardware profile properties (such as hw.ramSize and
hw.accelerometer ) describe the physical characteristics of the emulated device. These characteristics include
screen size, the amount of available RAM, whether or not an accelerometer is present. AVD properties specify
the operation of the AVD when it runs. For example, AVD properties can be configured to specify how the AVD
uses your development computer's graphics card for rendering.
You can change properties by using the following guidelines:
To change a boolean property, click the check mark to the right of the boolean property:
To change an enum (enumerated) property, click the down-arrow to the right of the property and choose
a new value.
To change a string or integer property, double-click the current string or integer setting in the value
column and enter a new value.
The following table provides a detailed explanation of the properties listed in the New Device and Device
Editor screens:
abi.type ABI type – Specifies the ABI x86, x86_64, armeabi-v7a, arm64-v8a
(application binary interface) type of
the emulated device. The x86 option is
for the instruction set commonly
referred to as "x86" or "IA-32." The
x86_64 option is for the 64-bit x86
instruction set. The armeabi-v7a
option is for the ARM instruction set
with v7-a ARM extensions. The
arm64-v8a option is for the ARM
instruction set that supports AArch64.
hw.gpu.mode GPU emulation mode – Determines auto, host, mesa, angle, swiftshader, off
how GPU emulation is implemented by
the emulator. If you select auto, the
emulator will choose hardware and
software acceleration based on your
development computer setup. If you
select host, the emulator will use your
development computer's graphics
processor to perform GPU emulation
for faster rendering. If your GPU is not
compatible with the emulator and you
are on Windows, you can try angle
instead of host. The angle mode uses
DirectX to provide performance similar
to host. If you select mesa, the
emulator will use the Mesa 3D
software library to render graphics.
Select mesa if you have problems
rendering via your development
computer's graphics processor. The
swiftshader mode can be used to
render graphics in software with
slightly less performance than using
your computer's GPU. The off option
(disable graphics hardware emulation)
is a deprecated option that can cause
improper rendering for some items
and is therefore not recommended.
hw.lcd.density LCD density – The density of the 120, 160, 240, 213, 320
emulated LCD display, measured in
density-independent pixels, or dp (dp
is a virtual pixel unit). When the setting
is 160 dp, each dp corresponds to one
physical pixel. At runtime, Android uses
this value to select and scale the
appropriate resources/assets for
correct display rendering.
For more information about these properties, see Hardware Profile Properties.
Debug on the Android Emulator
3/18/2022 • 2 minutes to read • Edit Online
The Android Emulator, installed as part of the Mobile development with .NET workload, can be run in
various configurations to simulate different Android devices. Each one of these configurations is created as a
virtual device. In this article, you'll learn how to launch the emulator from Visual Studio and run your app in a
virtual device. For more information about how to create and configure a virtual device, see Managing virtual
devices with the Android Device Manager.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
After the emulator starts, Visual Studio deploys the app to the virtual device. An example screenshot of the
Android Emulator is displayed below. In this example, the emulator is running the .NET MAUI template app.
When you're finished debugging and running your app, you can leave the emulator running. The first time a
.NET MAUI app is run in the emulator, the .NET MAUI shared runtime for the targeted API level is installed,
followed by the app. The runtime installation may take a few moments to install. By leaving the emulator
running, later debugging sessions start faster as the runtime is already present on the device. If the device is
restarted, the runtime will be redeployed to the device.
Fast boot
The Android Emulator includes a feature named Fast Boot which is enabled by default. This feature is configured
by each device's emulator settings. With this feature enabled, a snapshot of the virtual device is saved when the
emulator is closed. The snapshot is quickly restored the next time the device is started.
The first time a virtual device is started, a cold boot of the virtual device takes place without a speed
improvement because a snapshot hasn't yet been created:
When you exit out of the emulator, Fast Boot saves the state of the emulator in a snapshot:
The next time the virtual device starts, it loads much faster because the emulator simply restores the state at
which you closed the emulator.
Troubleshooting
For tips and workarounds for common emulator problems, see Android Emulator Troubleshooting.
For more information about using the Android Emulator, see the following Android Developer articles:
Navigating on the Screen
Performing Basic Tasks in the Emulator
Working with Extended Controls, Settings, and Help
Run the emulator with Quick Boot
Android emulator troubleshooting
3/18/2022 • 10 minutes to read • Edit Online
This article describes the most common warning messages and issues that occur while configuring and running
the Android Emulator. Also, it describes solutions for resolving these errors and various troubleshooting tips to
help you diagnose emulator problems.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
adb devices
4. If the emulator is accessible from the Android SDK, the emulator should appear in the list of attached
devices. For example:
5. If the emulator doesn't appear in this list, start the Android SDK Manager , apply all updates, then try
launching the emulator again.
MMIO access error
If the message An MMIO access error has occurred is displayed, restart the emulator.
NOTE
Google Play Store images are available only for some base device types such as Pixel, Pixel 2, Nexus 5, and Nexus 5X.
Performance issues
Performance issues are typically caused by one of the following problems:
The emulator is running without hardware acceleration.
The virtual device running in the emulator using an Arm-based image.
The following sections cover these scenarios in more detail.
Hardware acceleration isn't enabled
When you start a virtual device, and you don't have hardware acceleration enabled, the Device Manager
displays an error dialog similar to the following image:
To fix this error, follow the troubleshooting steps in the Hardware acceleration issues section.
Hardware acceleration issues
When using hardware acceleration, you may run into configuration problems or conflicts with other software on
your computer. The first step in troubleshooting is verifying that hardware acceleration is enabled. You can use
the Android's SDK to check this setting. Open a command prompt and entering the following command:
This command assumes that the Android SDK is installed at the default location of C:\Program Files
(x86)\Android\android-sdk. If the Android SDK is installed elsewhere, modify the preceding command to the
correct location.
Hardware acceleration not available
If Hyper-V is available, a message like the following example will be returned from the emulator-check .exe
accel command:
If hardware acceleration isn't available, a message like the following example will be displayed (the emulator
looks for HAXM if it's unable to find Hyper-V):
If hardware acceleration isn't available, see Enabling Hyper-V acceleration to learn how to enable hardware
acceleration on your computer.
Incorrect BIOS settings
If the BIOS hasn't been configured properly to support hardware acceleration, a message similar to the
following example will be displayed when you run the emulator-check .exe accel command:
To correct this problem, reboot into your computer's BIOS and enable the following options:
Virtualization Technology (may have a different label depending on motherboard manufacturer).
Hardware Enforced Data Execution Prevention.
If problems still occur because of issues related to Hyper-V and HAXM, see the following section.
Hyper-V issues
In some cases, enabling both Hyper-V and Windows Hyper visor Platform in the Turn Windows features
on or off dialog may not properly enable Hyper-V. To verify that Hyper-V is enabled, use the following steps:
1. Enter PowerShell in the Windows search box.
2. Right-click Windows PowerShell in the search results and select Run as administrator .
3. In the PowerShell console, enter the following command:
Get-WindowsOptionalFeature -FeatureName Microsoft-Hyper-V-All -Online
If Hyper-V isn't enabled, a message similar to the following example will be displayed to indicate that the
state of Hyper-V is Disabled :
FeatureName : Microsoft-Hyper-V-All
DisplayName : Hyper-V
Description : Provides services and management tools for creating and running virtual machines
and their resources.
RestartRequired : Possible
State : Disabled
CustomProperties :
If the Hypervisor isn't enabled, a message similar to the following example will be displayed to indicate
that the state of HypervisorPlatform is Disabled :
FeatureName : HypervisorPlatform
DisplayName : Windows Hypervisor Platform
Description : Enables virtualization software to run on the Windows hypervisor
RestartRequired : Possible
State : Disabled
CustomProperties :
If Hyper-V or HypervisorPlatform aren't enabled, use the following PowerShell commands to enable them:
sc query intelhaxm
If the HAXM process is running, you should see output similar to the following result:
SERVICE_NAME: intelhaxm
TYPE : 1 KERNEL_DRIVER
STATE : 4 RUNNING
(STOPPABLE, NOT_PAUSABLE, IGNORES_SHUTDOWN)
WIN32_EXIT_CODE : 0 (0x0)
SERVICE_EXIT_CODE : 0 (0x0)
CHECKPOINT : 0x0
WAIT_HINT : 0x0
If STATE isn't set to RUNNING , see How to Use the Intel Hardware Accelerated Execution Manager to resolve the
problem.
HAXM virtualization conflicts
HAXM can conflict with other technologies that use virtualization, such as Hyper-V, Windows Device Guard, and
some antivirus software:
Hyper-V —If you're using a version of Windows before the Windows 10 April 2018 update (build
1803) and Hyper-V is enabled, follow the steps in Disabling Hyper-V so that HAXM can be enabled.
Device Guard —Device Guard and Credential Guard can prevent Hyper-V from being disabled on
Windows machines. To disable Device Guard and Credential Guard, see Disabling Device Guard.
Antivirus Software —If you're running antivirus software that uses hardware-assisted virtualization
(such as Avast), disable or uninstall this software, reboot, and retry the Android emulator.
Incorrect BIOS settings for HAXM
On Windows, HAXM won't work unless virtualization technology (Intel VT-x) is enabled in the BIOS. If VT-x is
disabled, you'll get an error similar to the following when you attempt to start the Android Emulator:
This computer meets the requirements for HAXM, but Intel Virtualization Technology (VT-x) isn't turned on.
To correct this error, boot the computer into the BIOS, enable both VT-x and SLAT (Second-Level Address
Translation) and restart the computer.
Disabling Hyper-V
If you're using a version of Windows before the Windows 10 April 2018 Update (build 1803) and Hyper-V
is enabled, you must disable Hyper-V and reboot your computer to install and use HAXM. If you're using
Windows 10 April 2018 Update (build 1803) or later, Android Emulator version 27.2.7 or later can use
Hyper-V (instead of HAXM) for hardware acceleration, so it isn't necessary to disable Hyper-V.
You can disable Hyper-V from the Control Panel by following these steps:
1. Enter windows features in the Windows search box and select Turn Windows features on or off in
the search results.
2. Uncheck Hyper-V :
3. Restart the computer.
Alternately, you can use the following PowerShell command to disable the Hyper-V Hypervisor:
Intel HAXM and Microsoft Hyper-V can't both be active at the same time. Unfortunately, there's no way to switch
between Hyper-V and HAXM without restarting your computer.
It's possible that the preceding steps won't succeed in disabling Hyper-V if Device Guard and Credential Guard
are enabled. If you're unable to disable Hyper-V, or it seems to be disabled but HAXM installation still fails, use
the steps in the next section to disable Device Guard and Credential Guard.
Disabling Device Guard
Device Guard and Credential Guard can prevent Hyper-V from being disabled on Windows machines. This
situation is often a problem for domain-joined machines that are configured and controlled by an owning
organization. On Windows 10, use the following steps to see if Device Guard is running:
1. Enter System info in the Windows search box and select System Information in the search results.
2. In the System Summar y , look to see if Device Guard Vir tualization based security is present and
is in the Running state:
4. Change Turn On Vir tualization Based Security to Disabled (as shown above) and exit the Local
Group Policy Editor .
5. In the Windows Search Box, enter cmd . When Command Prompt appears in the search results, right-
click Command Prompt and select Run as Administrator .
6. Copy and paste the following commands into the command prompt window (if drive Z: is in use, pick an
unused drive letter to use instead):
mountvol Z: /s
copy %WINDIR%\System32\SecConfig.efi Z:\EFI\Microsoft\Boot\SecConfig.efi /Y
bcdedit /create {0cb3b571-2f2e-4343-a879-d86a476d7215} /d "DebugTool" /application osloader
bcdedit /set {0cb3b571-2f2e-4343-a879-d86a476d7215} path "\EFI\Microsoft\Boot\SecConfig.efi"
bcdedit /set {bootmgr} bootsequence {0cb3b571-2f2e-4343-a879-d86a476d7215}
bcdedit /set {0cb3b571-2f2e-4343-a879-d86a476d7215} loadoptions DISABLE-LSA-ISO,DISABLE-VBS
bcdedit /set {0cb3b571-2f2e-4343-a879-d86a476d7215} device partition=Z:
mountvol Z: /d
7. Restart your computer. On the boot screen, you should see a prompt similar to the following message:
This command assumes that the Android SDK is installed at the default location of C:\Program Files
(x86)\Android\android-sdk. If the Android SDK is installed elsewhere, modify the preceding command to the
correct location.
When you run this command, it produces many lines of output while the emulator starts up. Specifically, lines
such as the following example are printed if hardware acceleration is enabled and working properly. In this
example, HAXM is used for hardware acceleration:
Component Intel x86 Emulator Accelerator (HAXM installer) r6.2.1 [Extra: (Intel Corporation)] not present on
the system
Set up Android device for debugging
3/18/2022 • 3 minutes to read • Edit Online
While the Android emulator is a great way to rapidly develop and test your app, you'll want to test your apps on
a real Android device. To run on a device, you'll need to enable developer mode on the device and connect it to
your computer.
IMPORTANT
The steps in this article are written generically, to work on as many devices as possible. If you can't find these settings on
your device, consult your device manufacturer's documentation.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
If your computer isn't recognizing the device when it's plugged in, try installing a driver for the device. Consult
your device manufacturer's support documentation. You can also try installing the Google USB Driver through
the Android SDK Manager:
Enable WiFi debugging
It's possible to debug an android device over WiFi, without keeping the device physically connected to the
computer. This technique requires more effort, but could be useful when the device is too far from the computer
to remain constantly plugged-in via a cable.
Connecting over WiFi
By default, the Android Debug Bridge (adb) is configured to communicate with an Android device via USB. It's
possible to reconfigure it to use TCP/IP instead of USB. To do this, both the device and the computer must be on
the same WiFi network.
First, enable Wireless debugging on your Android device:
1. Follow the steps in the Enable developer mode on the device section.
2. Follow the steps in the Enable USB debugging section.
3. Go to the Settings screen.
4. Select Developer options .
5. Turn on the Wireless debugging option.
Depending on the UI your device is running, the Wireless debugging option may be in a different location.
Consult your device documentation if you can't find Wireless debugging .
Next, use adb to connect to your device, first through a USB connection:
1. Determine the IP address of your Android device. One way to find out the IP address is to look under
Settings > Network & internet > Wi-Fi , then tap on the WiFi network that the device is connected to,
and then tap on Advanced . This will open a dropdown showing information about the network
connection, similar to what is seen in the screenshot below:
On some versions of Android the IP address won't be listed there but can be found instead under
Settings > About phone > Status .
2. In Visual Studio, open the adb command prompt by selecting the menu option: Tools > Android >
Android Adb Command Prompt....
3. In the command prompt, use the adb tcpip command to tell the device to listen to TCP/IP connections
on port 5555.
When this command finishes, the Android device is connected to the computer via WiFi.
When you're finished debugging via WiFi, you can reset ADB back to USB mode with the following
command:
adb usb
To see the devices connected to the computer, use the adb devices command:
adb devices
Deploy and debug your .NET MAUI app on
Windows
3/18/2022 • 2 minutes to read • Edit Online
You can use your local Windows development computer to deploy and debug a .NET Multi-platform App UI
(.NET MAUI) app. This article describes how to configure Windows to debug a .NET MAUI app.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Configure Windows
You must enable Developer Mode in Windows. Both Windows 10 and Windows 11 are supported.
Windows 11
Developer Mode is enabled in Settings app, under Privacy & security > For developers . To enable
Developer Mode in Windows 11:
1. Open the Start Menu.
2. Type Developer settings in the search box and select it.
3. Turn on Developer Mode .
4. If you receive a warning message about Developer Mode, read it, and select Yes if you understand the
warning.
Windows 10
Developer Mode is enabled in Settings app, under Update & Security > For developers . To enable
Developer Mode in Windows 10:
1. Open the Start Menu.
2. Search for Developer settings , select it.
3. Turn on Developer Mode .
4. If you receive a warning message about Developer Mode, read it, and select Yes if you understand the
warning.
Target Windows
In Visual Studio, set the Debug Target to Framework (...) > net6.0-windows . There is a version number in
the item entry, which may or may not match the following screenshot:
XAML
3/18/2022 • 2 minutes to read • Edit Online
The eXtensible Application Markup Language (XAML) is an XML-based language that's an alternative to
programming code for instantiating and initializing objects, and organizing those objects in parent-child
hierarchies.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
XAML allows developers to define user interfaces in .NET Multi-platform App UI (.NET MAUI) apps using markup
rather than code. XAML is not required in a .NET MAUI app, but it is the recommended approach to developing
your UI because it's often more succinct, more visually coherent, and has tooling support. XAML is also well
suited for use with the Model-View-ViewModel (MVVM) pattern, where XAML defines the view that is linked to
viewmodel code through XAML-based data bindings.
Within a XAML file, you can define user interfaces using all the .NET MAUI views, layouts, and pages, as well as
custom classes. The XAML file can be either compiled or embedded in the app package. Either way, the XAML is
parsed at build time to locate named objects, and at runtime the objects represented by the XAML are
instantiated and initialized.
XAML has several advantages over equivalent code:
XAML is often more succinct and readable than equivalent code.
The parent-child hierarchy inherent in XML allows XAML to mimic with greater visual clarity the parent-child
hierarchy of user-interface objects.
There are also disadvantages, mostly related to limitations that are intrinsic to markup languages:
XAML cannot contain code. All event handlers must be defined in a code file.
XAML cannot contain loops for repetitive processing.
XAML cannot contain conditional processing. However, a data-binding can reference a code-based binding
converter that effectively allows some conditional processing.
XAML generally cannot instantiate classes that do not define a parameterless constructor, although this
restriction can sometimes be overcome.
XAML generally cannot call methods, although this restriction can sometimes be overcome.
There is no visual designer for producing XAML in .NET MAUI apps. All XAML must be hand-written, but you can
use XAML hot reload to view your UI as you edit it.
XAML is basically XML, but XAML has some unique syntax features. The most important are:
Property elements
Attached properties
Markup extensions
These features are not XML extensions. XAML is entirely legal XML. But these XAML syntax features use XML in
unique ways.
Get started with XAML
3/18/2022 • 8 minutes to read • Edit Online
In a .NET Multi-platform App UI (.NET MAUI) app, XAML is mostly used to define the visual contents of a page
and works together with a C# code-behind file. The code-behind file provides code support for the markup.
Together, these two files contribute to a new class definition that includes child views and property initialization.
Within the XAML file, classes and properties are referenced with XML elements and attributes, and links between
the markup and code are established.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The first file pairing is App.xaml , a XAML file, and App.xaml.cs , a C# code-behind file associated with the XAML
file. Both App.xaml and App.xaml.cs contribute to a class named App that derives from Application . Most
other classes with XAML files contribute to a class that derives from ContentPage , and define the UI of a page.
This is true of the MainPage.xaml and MainPage.xaml.cs files.
The MainPage.xaml file has the following structure:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="MyMauiApp.MainPage">
...
</ContentPage>
The two XML namespace ( xmlns ) declarations refer to URIs on microsoft.com. However, there's no content at
these URIs, and they basically function as version identifiers.
The first XML namespace declaration means that tags defined within the XAML file with no prefix refer to classes
in .NET MAUI, for example ContentPage . The second namespace declaration defines a prefix of x . This is used
for several elements and attributes that are intrinsic to XAML itself and which are supported by other
implementations of XAML. However, these elements and attributes are slightly different depending on the year
embedded in the URI. .NET MAUI supports the 2009 XAML specification.
At the end of the first tag, the x prefix is used for an attribute named Class . Because the use of this x prefix is
virtually universal for the XAML namespace, XAML attributes such as Class are almost always referred to as
x:Class . The x:Class attribute specifies a fully qualified .NET class name: the MainPage class in the MyMauiApp
namespace. This means that this XAML file defines a new class named MainPage in the MyMauiApp namespace
that derives from ContentPage (the tag in which the x:Class attribute appears).
The x:Class attribute can only appear in the root element of a XAML file to define a derived C# class. This is the
only new class defined in the XAML file. Everything else that appears in a XAML file is instead simply instantiated
from existing classes and initialized.
The MainPage.xaml.cs file looks similar to this:
namespace MyMauiApp;
The MainPage class derives from ContentPage , and is a partial class definition.
When Visual Studio builds the project, a source generator generates new C# source that contains the definition
of the InitializeComponent method that's called from the MainPage constructor and adds it to the compilation
object.
At runtime, code in the MauiProgram class bootstraps the app and executes the App class constructor, which
instantiates MainPage . The MainPage constructor calls InitializeComponent , which initializes all the objects
defined in the XAML file, connects them all together in parent-child relationships, attaches event handlers
defined in code to events set in the XAML file, and sets the resultant tree of objects as the content of the page.
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="XamlSamples.HelloXamlPage"
Title="Hello XAML Page">
<Label Text="Hello, XAML!"
VerticalOptions="Center"
HorizontalTextAlignment="Center"
Rotation="-15"
FontSize="Large"
FontAttributes="Bold"
TextColor="Blue" />
</ContentPage>
From the example above the relationship between classes, properties, and XML should be evident. A .NET MAUI
class (such as ContentPage or Label ) appears in the XAML file as an XML element. Properties of that class—
including Title on ContentPage and seven properties of Label usually appear as XML attributes.
Many shortcuts exist to set the values of these properties. Some properties are basic data types. For example,
the Title and Text properties are of type string , and Rotation is of type double . The
HorizontalTextAlignment property is of type TextAlignment , which is an enumeration. For a property of any
enumeration type, all you need to supply is a member name.
For properties of more complex types, however, converters are used for parsing the XAML. These are classes in
.NET MAUI that derive from TypeConverter . For the example above, several .NET MAUI converters are
automatically applied to convert string values to their correct type:
LayoutOptionsConverterfor the VerticalOptions property. This converter converts the names of public static
fields of the LayoutOptions structure to values of type LayoutOptions .
FontSizeConverter for the FontSize property. The converter converts a NamedSize member or a numeric
font size.
ColorTypeConverter for the TextColor property. This converter converts the names of public static fields of
the Colors structure or hexadecimal RGB values, with or without an alpha channel.
Page navigation
When you run a .NET MAUI app, the MainPage is typically displayed. To see a different page you can either set
that as the new startup page in the App.xaml.cs file, or navigate to the new page from MainPage .
To implement navigation, first change code in the App.xaml.cs constructor so that a NavigationPage object is
created:
public App()
{
InitializeComponent();
MainPage = new NavigationPage(new MainPage());
}
The NavigationPage constructor takes a Page argument that specifies the initial page that will be displayed.
In the MainPage.xaml.cs constructor, you can create a simple Button and use the event handler to navigate to
HelloXamlPage :
public MainPage()
{
InitializeComponent();
Content = button;
}
When you compile and deploy the new version of this app, a button appears on the screen. Pressing it navigates
to HelloXamlPage :
You can navigate back to MainPage using the navigation bar that appears on each platform.
NOTE
An alternative to this navigation model is to use .NET MAUI Shell.
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="XamlSamples.XamlPlusCodePage"
Title="XAML + Code Page">
<StackLayout>
<Slider VerticalOptions="Center" />
<Label Text="A simple Label"
Font="Large"
HorizontalOptions="Center"
VerticalOptions="Center" />
<Button Text="Click Me!"
HorizontalOptions="Center"
VerticalOptions="Center" />
</StackLayout>
</ContentPage>
This XAML file is syntactically complete, and produces the following UI:
However, while you can interact with the Slider and Button , the UI isn't updated. The Slider should cause the
Label to display the current value, and the Button should do something.
Displaying a Slider value using a Label can be achieved entirely in XAML with a data binding. However, it's
useful to see the code solution first. Even so, handling the Button click definitely requires code. This means that
the code-behind file for XamlPlusCodePage must contain handlers for the ValueChanged event of the Slider and
the Clicked event of the Button :
namespace XamlSamples
{
public partial class XamlPlusCodePage
{
public XamlPlusCodePage()
{
InitializeComponent();
}
Back in the XAML file, the Slider and Button tags need to include attributes for the ValueChanged and
Clicked events that reference these handlers:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="XamlSamples.XamlPlusCodePage"
Title="XAML + Code Page">
<StackLayout>
<Slider VerticalOptions="Center"
ValueChanged="OnSliderValueChanged" />
<Label x:Name="valueLabel"
Text="A simple Label"
Font="Large"
HorizontalOptions="Center"
VerticalOptions="Center" />
<Button Text="Click Me!"
HorizontalOptions="Center"
VerticalOptions="Center"
Clicked="OnButtonClicked" />
</StackLayout>
</ContentPage>
Notice that assigning a handler to an event has the same syntax as assigning a value to a property. In addition,
for the ValueChanged event handler of the Slider to use the Label to display the current value, the handler
needs to reference that object from code. Therefore, the Label needs a name, which is specified with the
x:Name attribute. The x prefix of the x:Name attribute indicates that this attribute is intrinsic to XAML. The
name you assign to the x:Name attribute has the same rules as C# variable names. For example, it must begin
with a letter or underscore and contain no embedded spaces.
The ValueChanged event handler can now set the Label to display the new Slider value, which is available
from the event arguments:
The result is that any manipulation of the Slider causes its value to be displayed in the Label :
In the example above the Button simulates a response to a Clicked event by displaying an alert with the Text
of the button. Therefore, the event handler can cast the sender argument to a Button and then access its
properties:
The OnButtonClicked method is defined as async because the DisplayAlert method is asynchronous and
should be prefaced with the await operator, which returns when the method completes. Because this method
obtains the Button firing the event from the sender argument, the same handler could be used for multiple
buttons.
Next steps
XAML is mostly designed for instantiating and initializing objects. But often, properties must be set to complex
objects that cannot easily be represented as XML strings, and sometimes properties defined by one class must
be set on a child class. These two needs require the essential XAML syntax features of property elements and
attached properties.
Essential XAML syntax
Essential XAML syntax
3/18/2022 • 5 minutes to read • Edit Online
XAML is mostly designed for instantiating and initializing objects. But often, properties must be set to complex
objects that cannot easily be represented as XML strings, and sometimes properties defined by one class must
be set on a child class. These two needs require the essential XAML syntax features of property elements and
attached properties.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Property elements
In .NET Multi-platform App UI (.NET MAUI) XAML, properties of classes are normally set as XML attributes:
These two examples that specify the TextColor property are functionally equivalent, and enable the
introduction of some basic terminology:
Label is an object element. It is a .NET MAUI object expressed as an XML element.
Text , VerticalOptions , FontAttributes and FontSize are property attributes. They are .NET MAUI
properties expressed as XML attributes.
In the second example, TextColor has become a property element. It is a .NET MAUI property expressed as
an XML element.
NOTE
In a property element, the value of the property is always defined as the content between the property-element start
and end tags.
Property-element syntax can also be used on more than one property of an object:
<Label Text="Hello, XAML!"
VerticalOptions="Center">
<Label.FontAttributes>
Bold
</Label.FontAttributes>
<Label.FontSize>
Large
</Label.FontSize>
<Label.TextColor>
Aqua
</Label.TextColor>
</Label>
While property-element syntax might seem unnecessary, it's essential when the value of a property is too
complex to be expressed as a simple string. Within the property-element tags you can instantiate another object
and set its properties. For example, the Grid layout has properties named RowDefinitions and
ColumnDefinitions , which are of type RowDefinitionCollection and ColumnDefinitionCollection respectively.
These types are collections of RowDefinition and ColumnDefinition objects, and you typically use property
element syntax to set them:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="XamlSamples.GridDemoPage"
Title="Grid Demo Page">
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="Auto" />
<RowDefinition Height="*" />
<RowDefinition Height="100" />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="Auto" />
<ColumnDefinition Width="*" />
<ColumnDefinition Width="100" />
</Grid.ColumnDefinitions>
...
</Grid>
</ContentPage>
Attached properties
In the previous example you saw that the Grid requires property elements for the RowDefinitions and
ColumnDefinitions collections to define the rows and columns. This suggests that there must be a technique for
indicating the row and column where each child of the Grid resides.
Within the tag for each child of the Grid you specify the row and column of that child using the Grid.Row and
Grid.Column attributes, which have default values of 0. You can also indicate if a child spans more than one row
or column with the Grid.RowSpan and Grid.ColumnSpan attributes, which have default values of 1.
The following example demonstrates placing children within a Grid :
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="XamlSamples.GridDemoPage"
Title="Grid Demo Page">
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="Auto" />
<RowDefinition Height="*" />
<RowDefinition Height="100" />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="Auto" />
<ColumnDefinition Width="*" />
<ColumnDefinition Width="100" />
</Grid.ColumnDefinitions>
</Grid>
</ContentPage>
NOTE
When you wish to use these attached properties in code, the Grid class provides static methods named GetRow ,
SetRow , GetColumn , SetColumn , GetRowSpan , SetRowSpan , GetColumnSpan , and SetColumnSpan .
Attached properties are recognizable in XAML as attributes containing both a class and a property name
separated by a period. They are called attached properties because they are defined by one class (in this case,
Grid ) but attached to other objects (in this case, children of the Grid ). During layout, the Grid can interrogate
the values of these attached properties to know where to place each child.
Content properties
In the previous example, the Grid object was set to the Content property of the ContentPage . However, the
Content property wasn't referenced in the XAML but can be:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="XamlSamples.XamlPlusCodePage"
Title="XAML + Code Page">
<ContentPage.Content>
<Grid>
...
</Grid>
</ContentPage.Content>
</ContentPage>
The Content property isn't required in XAML because elements defined for use in .NET MAUI XAML are allowed
to have one property specified as the ContentProperty attribute on the class:
[ContentProperty("Content")]
public class ContentPage : TemplatedPage
{
...
}
Any property specified as the ContentProperty of a class means that the property-element tags for the property
are not required. Therefore, the example above specifies that any XAML content that appears between the start
and end ContentPage tags is assigned to the Content property.
Many classes also have ContentProperty attribute definitions. For example, the content property of Label is
Text .
Platform differences
.NET MAUI apps can customize UI appearance on a per-platform basis. This can be achieved in XAML using the
OnPlatform and On classes:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="...">
<ContentPage.Padding>
<OnPlatform x:TypeArguments="Thickness">
<On Platform="iOS" Value="0, 20, 0, 0" />
<On Platform="Android" Value="10, 20, 20, 10" />
</OnPlatform>
</ContentPage.Padding>
...
</ContentPage>
OnPlatform is a generic class and so you need to specify the generic type argument, in this case, Thickness ,
which is the type of Padding property. This is achieved with the x:TypeArguments XAML attribute. The
OnPlatform class has a property named Platforms , that is an IList of On objects. Each On object can set the
Platform and Value property to define the Thickness value for a specific platform.
In addition, the Platform property of On is of type IList<string> , so you can include multiple platforms if the
values are the same:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="...">
<ContentPage.Padding>
<OnPlatform x:TypeArguments="Thickness">
<On Platform="iOS, Android" Value="10, 20, 20, 10" />
</OnPlatform>
</ContentPage.Padding>
...
</ContentPage>
NOTE
If the Value property of an On object can't be represented by a single string, you can define property elements for it.
Next steps
.NET MAUI XAML markup extensions enable properties to be set to objects or values that are referenced
indirectly from other sources. XAML markup extensions are particularly important for sharing objects, and
referencing constants used throughout an app.
XAML markup extensions
XAML markup extensions
3/18/2022 • 8 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) XAML markup extensions enable properties to be set to objects or
values that are referenced indirectly from other sources. XAML markup extensions are particularly important for
sharing objects, and referencing constants used throughout an app, but they find their greatest utility in data
bindings.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Typically, you use XAML to set properties of an object to explicit values, such as a string, a number, an
enumeration member, or a string that is converted to a value behind the scenes. Sometimes, however,
properties must instead reference values defined somewhere else, or which might require a little processing by
code at runtime. For these purposes, XAML markup extensions are available.
XAML markup extensions are so named because they are backed by code in classes that implement
IMarkupExtension . It's also possible to write your own custom markup extensions.
In many cases, XAML markup extensions are instantly recognizable in XAML files because they appear as
attribute values delimited by curly braces, { and }, but sometimes markup extensions also appear in markup as
conventional elements.
IMPORTANT
Markup extensions can have properties, but they are not set like XML attributes. In a markup extension, property settings
are separated by commas, and no quotation marks appear within the curly braces.
Shared resources
Some XAML pages contain several views with properties set to the same values. For example, many of the
property settings for these Button objects are the same:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="XamlSamples.SharedResourcesPage"
Title="Shared Resources Page">
<StackLayout>
<Button Text="Do this!"
HorizontalOptions="Center"
VerticalOptions="Center"
BorderWidth="3"
Rotation="-15"
TextColor="Red"
FontSize="24" />
<Button Text="Do that!"
HorizontalOptions="Center"
VerticalOptions="Center"
BorderWidth="3"
Rotation="-15"
TextColor="Red"
FontSize="24" />
<Button Text="Do the other thing!"
HorizontalOptions="Center"
VerticalOptions="Center"
BorderWidth="3"
Rotation="-15"
TextColor="Red"
FontSize="24" />
</StackLayout>
</ContentPage>
If one of these properties needs to be changed, you might prefer to make the change just once rather than three
times. If this were code, you’d likely be using constants and static read-only objects to help keep such values
consistent and easy to modify.
In XAML, one popular solution is to store such values or objects in a resource dictionary. The VisualElement
class defines a property named Resources of type ResourceDictionary , which is a dictionary with keys of type
string and values of type object . You can put objects into this dictionary and then reference them from
markup, all in XAML.
To use a resource dictionary on a page, include a pair of Resources property-element tags at the top of the
page, and add resources within these tags. Objects and values of various types can be added to the resource
dictionary. These types must be instantiable. They can’t be abstract classes, for example. These types must also
have a public parameterless constructor. Each item requires a dictionary key specified with the x:Key attribute:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="XamlSamples.SharedResourcesPage"
Title="Shared Resources Page">
<ContentPage.Resources>
<LayoutOptions x:Key="horzOptions"
Alignment="Center" />
<LayoutOptions x:Key="vertOptions"
Alignment="Center" />
</ContentPage.Resources>
...
</ContentPage>
In this example, the two resources are values of the structure type LayoutOptions , and each has a unique key
and one or two properties set. In code and markup, it’s much more common to use the static fields of
LayoutOptions , but here it’s more convenient to set the properties.
NOTE
Optional ResourceDictionary tags can be included as the child of the Resources tags.
The resources can then be consumed by the Button objects, by using the StaticResource XAML markup
extension to set their HorizontalOptions and VerticalOptions properties:
The StaticResource markup extension is always delimited with curly braces, and includes the dictionary key.
The name StaticResource distinguishes it from DynamicResource , which .NET MAUI also supports.
DynamicResource is for dictionary keys associated with values that might change at runtime, while
StaticResource accesses elements from the dictionary just once when the elements on the page are
constructed. Whenever the XAML parser encounters a StaticResource markup extension, it searches up the
visual tree and uses the first ResourceDictionary it encounters containing that key.
It’s necessary to store doubles in the dictionary for the BorderWidth , Rotation , and FontSize properties. XAML
conveniently defines tags for common data types like x:Double and x:Int32 :
<ContentPage.Resources>
<LayoutOptions x:Key="horzOptions"
Alignment="Center" />
<LayoutOptions x:Key="vertOptions"
Alignment="Center" />
<x:Double x:Key="borderWidth">3</x:Double>
<x:Double x:Key="rotationAngle">-15</x:Double>
<x:Double x:Key="fontSize">24</x:Double>
</ContentPage.Resources>
These additional three resources can be referenced in the same way as the LayoutOptions values:
For resources of type Color , you can use the same string representations that you use when directly assigning
attributes of these types. Type converters included in .NET MAUI are invoked when the resource is created. It's
also possible to use the OnPlatform class within the resource dictionary to define different values for the
platforms. The following example uses this class for setting different text colors:
<OnPlatform x:Key="textColor"
x:TypeArguments="Color">
<On Platform="iOS" Value="Red" />
<On Platform="Android" Value="Aqua" />
</OnPlatform>
The OnPlatformresource gets an x:Key attribute because it’s an object in the dictionary, and an
x:TypeArguments attribute because it’s a generic class. The iOS , and Android attributes are converted to Color
values when the object is initialized.
The following example shows the three buttons accessing six shared values:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="XamlSamples.SharedResourcesPage"
Title="Shared Resources Page">
<ContentPage.Resources>
<LayoutOptions x:Key="horzOptions"
Alignment="Center" />
<LayoutOptions x:Key="vertOptions"
Alignment="Center" />
<x:Double x:Key="borderWidth">3</x:Double>
<x:Double x:Key="rotationAngle">-15</x:Double>
<x:Double x:Key="fontSize">24</x:Double>
<OnPlatform x:Key="textColor"
x:TypeArguments="Color">
<On Platform="iOS" Value="Red" />
<On Platform="Android" Value="Aqua" />
<On Platform="UWP" Value="#80FF80" />
</OnPlatform>
</ContentPage.Resources>
<StackLayout>
<Button Text="Do this!"
HorizontalOptions="{StaticResource horzOptions}"
VerticalOptions="{StaticResource vertOptions}"
BorderWidth="{StaticResource borderWidth}"
Rotation="{StaticResource rotationAngle}"
TextColor="{StaticResource textColor}"
FontSize="{StaticResource fontSize}" />
<Button Text="Do that!"
HorizontalOptions="{StaticResource horzOptions}"
VerticalOptions="{StaticResource vertOptions}"
BorderWidth="{StaticResource borderWidth}"
Rotation="{StaticResource rotationAngle}"
TextColor="{StaticResource textColor}"
FontSize="{StaticResource fontSize}" />
<Button Text="Do the other thing!"
HorizontalOptions="{StaticResource horzOptions}"
VerticalOptions="{StaticResource vertOptions}"
BorderWidth="{StaticResource borderWidth}"
Rotation="{StaticResource rotationAngle}"
TextColor="{StaticResource textColor}"
FontSize="{StaticResource fontSize}" />
</StackLayout>
</ContentPage>
<StackLayout>
<StackLayout.Resources>
<Color x:Key="textColor">Blue</Color>
</StackLayout.Resources>
...
</StackLayout>
One of the most common types of objects stored in resource dictionaries is the .NET MAUI Style , which defines
a collection of property settings.
NOTE
The purpose of a resource dictionary is to share objects. Therefore, it doesn't make sense to put controls such as a
Label or Button in a resource dictionary. Visual elements can't be shared because the same instance can't appear
twice on a page.
NOTE
The StaticResource markup extension is supported by XAML implementations that define a resource dictionary, while
x:Static is an intrinsic part of XAML, as the x prefix reveals.
The following example demonstrates how x:Static can explicitly reference static fields and enumeration
members:
<Label Text="Hello, XAML!"
VerticalOptions="{x:Static LayoutOptions.Start}"
HorizontalTextAlignment="{x:Static TextAlignment.Center}"
TextColor="{x:Static Colors.Aqua}" />
The main use of the x:Static markup extension is in referencing static fields or properties from your own code.
For example, here’s an AppConstants class that contains some static fields that you might want to use on
multiple pages throughout an app:
namespace XamlSamples
{
static class AppConstants
{
public static readonly Color BackgroundColor = Colors.Aqua;
public static readonly Color ForegroundColor = Colors.Brown;
}
}
To reference the static fields of this class in a XAML file, you need to use an XML namespace declaration to
indicate where this file is located. Each additional XML namespace declaration defines a new prefix. To access
classes local to the root app namespace, such as AppConstants , you could use the prefix local . The namespace
declaration must indicate the CLR (Common Language Runtime) namespace name, also known as the .NET
namespace name, which is the name that appears in a C# namespace definition or in a using directive:
xmlns:local="clr-namespace:XamlSamples"
You can also define XML namespace declarations for .NET namespaces. For example, here’s a sys prefix for the
standard .NET System namespace, which is in the netstandard assembly. Because this is another assembly, you
must also specify the assembly name, in this case netstandard :
xmlns:sys="clr-namespace:System;assembly=netstandard"
NOTE
The keyword clr-namespace is followed by a colon and then the .NET namespace name, followed by a semicolon, the
keyword assembly , an equal sign, and the assembly name.
The static fields can then be consumed after declaring the XML namespace:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:XamlSamples"
xmlns:sys="clr-namespace:System;assembly=netstandard"
x:Class="XamlSamples.StaticConstantsPage"
Title="Static Constants Page"
Padding="5,25,5,0">
<StackLayout>
<Label Text="Hello, XAML!"
TextColor="{x:Static local:AppConstants.BackgroundColor}"
BackgroundColor="{x:Static local:AppConstants.ForegroundColor}"
FontAttributes="Bold"
FontSize="30"
HorizontalOptions="Center" />
<BoxView WidthRequest="{x:Static sys:Math.PI}"
HeightRequest="{x:Static sys:Math.E}"
Color="{x:Static local:AppConstants.ForegroundColor}"
HorizontalOptions="Center"
VerticalOptions="CenterAndExpand"
Scale="100" />
</StackLayout>
</ContentPage>
In this example, the BoxView dimensions are set to Math.PI and Math.E , but scaled by a factor of 100:
Next steps
.NET MAUI data bindings allow properties of two objects to be linked so that a change in one causes a change in
the other.
Data binding basics
Data binding basics
3/18/2022 • 9 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) data bindings allow properties of two objects to be linked so that a
change in one causes a change in the other. This is a very valuable tool, and while data bindings can be defined
entirely in code, XAML provides shortcuts and convenience.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Data bindings
Data bindings connect properties of two objects, called the source and the target. In code, two steps are
required:
1. The BindingContext property of the target object must be set to the source object,
2. The SetBinding method (often used in conjunction with the Binding class) must be called on the target
object to bind a property of that object to a property of the source object.
The target property must be a bindable property, which means that the target object must derive from
BindableObject . A property of Label , such as Text , is associated with the bindable property TextProperty .
In XAML, you must also perform the same two steps that are required in code, except that the Binding markup
extension takes the place of the SetBinding call and the Binding class. However, when you define data bindings
in XAML, there are multiple ways to set the BindingContext of the target object. Sometimes it’s set from the
code-behind file, sometimes using a StaticResource or x:Static markup extension, and sometimes as the
content of BindingContext property-element tags.
View-to-view bindings
You can define data bindings to link properties of two views on the same page. In this case, you set the
BindingContext of the target object using the x:Reference markup extension.
The following example contains a Slider and two Label views, one of which is rotated by the Slider value
and another which displays the Slider value:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="XamlSamples.SliderBindingsPage"
Title="Slider Bindings Page">
<StackLayout>
<Label Text="ROTATION"
BindingContext="{x:Reference slider}"
Rotation="{Binding Path=Value}"
FontAttributes="Bold"
FontSize="Large"
HorizontalOptions="Center"
VerticalOptions="Center" />
<Slider x:Name="slider"
Maximum="360"
VerticalOptions="Center" />
<Label BindingContext="{x:Reference slider}"
Text="{Binding Value, StringFormat='The angle is {0:F0} degrees'}"
FontAttributes="Bold"
FontSize="Large"
HorizontalOptions="Center"
VerticalOptions="Center" />
</StackLayout>
</ContentPage>
The Slider contains an x:Name attribute that is referenced by the two Label views using the x:Reference
markup extension. The x:Reference binding extension defines a property named Name to set to the name of
the referenced element, in this case slider . However, the ReferenceExtension class that defines the
x:Reference markup extension also defines a ContentProperty attribute for Name , which means that it isn’t
explicitly required.
The Binding markup extension itself can have several properties, just like the BindingBase and Binding class.
The ContentProperty for Binding is Path , but the “Path=” part of the markup extension can be omitted if the
path is the first item in the Binding markup extension.
The second Binding markup extension sets the StringFormat property. In .NET MAUI, bindings do not perform
any implicit type conversions, and if you need to display a non-string object as a string you must provide a type
converter or use StringFormat .
IMPORTANT
Formatting strings must be placed in single quotation marks.
Binding mode
A single view can have data bindings on several of its properties. However, each view can have only one
BindingContext , so multiple data bindings on that view must all reference properties of the same object.
The solution to this and other problems involves the Mode property, which is set to a member of the
BindingMode enumeration:
Default
OneWay — values are transferred from the source to the target
OneWayToSource — values are transferred from the target to the source
TwoWay — values are transferred both ways between source and target
OneTime — data goes from source to target, but only when the BindingContext changes
The following example demonstrates one common use of the OneWayToSource and TwoWay binding modes:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="XamlSamples.SliderTransformsPage"
Padding="5"
Title="Slider Transforms Page">
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="*" />
<RowDefinition Height="Auto" />
<RowDefinition Height="Auto" />
<RowDefinition Height="Auto" />
<RowDefinition Height="Auto" />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="*" />
<ColumnDefinition Width="Auto" />
</Grid.ColumnDefinitions>
In this example, four Slider views are intended to control the Scale , Rotate , RotateX , and RotateY
properties of a Label . At first, it seems as if these four properties of the Label should be data-binding targets
because each is being set by a Slider . However, the BindingContext of Label can be only one object, and
there are four different sliders. For that reason, the BindingContext of each of the four sliders is set to the
Label , and the bindings are set on the Value properties of the sliders. By using the OneWayToSource and
TwoWay modes, these Value properties can set the source properties, which are the Scale , Rotate , RotateX ,
and RotateY properties of the Label .
The bindings on three of the Slider views are OneWayToSource , meaning that the Slider value causes a
change in the property of its BindingContext , which is the Label named label . These three Slider views
cause changes to the Rotate , RotateX , and RotateY properties of the Label :
However, the binding for the Scale property is TwoWay . This is because the Scale property has a default value
of 1, and using a TwoWay binding causes the Slider initial value to be set at 1 rather than 0. If that binding were
OneWayToSource , the Scale property would initially be set to 0 from the Slider default value. The Label
would not be visible
NOTE
The VisualElement class also has ScaleX and ScaleY properties, which scale the VisualElement on the x-axis and
y-axis respectively.
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:XamlSamples;assembly=XamlSamples"
x:Class="XamlSamples.ListViewDemoPage"
Title="ListView Demo Page">
<ListView ItemsSource="{x:Static local:NamedColor.All}" />
</ContentPage>
To define a template for the items, the ItemTemplate should be set to a DataTemplate that references a
ViewCell . The ViewCell should define a layout of one or more views to display each item:
In this example, the Label element is set to the View property of the ViewCell . The ViewCell.View tags are
not needed because the View property is the content property of ViewCell . This XAML displays the
FriendlyName property of each NamedColor object:
The item template can be expanded to display more information and the actual color:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:XamlSamples"
x:Class="XamlSamples.ListViewDemoPage"
Title="ListView Demo Page">
<ContentPage.Resources>
<x:Double x:Key="boxSize">50</x:Double>
<x:Int32 x:Key="rowHeight">60</x:Int32>
<local:FloatToIntConverter x:Key="intConverter" />
</ContentPage.Resources>
namespace XamlSamples
{
public class FloatToIntConverter : IValueConverter
{
public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
{
float multiplier;
public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
{
float divider;
NOTE
The ConvertBack method does not play a role in this example because the bindings are only one way from source to
target.
A binding references a binding converter with the Converter property. A binding converter can also accept a
parameter specified with the ConverterParameter property. For some versatility, this is how the multiplier is
specified. The binding converter checks the converter parameter for a valid float value.
The converter is instantiated in the page's resource dictionary so it can be shared among multiple bindings:
The item template dsplays the color, its friendly name, and its RGB values:
The ListView can handle changes that dynamically occur in the underlying data, but only if you take certain
steps. If the collection of items assigned to the ItemsSource property of the ListView changes during runtime,
use an ObservableCollection class for these items. ObservableCollection implements the
INotifyCollectionChanged interface, and ListView will install a handler for the CollectionChanged event.
If properties of the items themselves change during runtime, then the items in the collection should implement
the INotifyPropertyChanged interface and signal changes to property values using the PropertyChanged event.
Next steps
Data bindings provide a powerful mechanism for linking properties between two objects within a page, or
between visual objects and underlying data. But when the application begins working with data sources, a
popular app architectural pattern begins to emerge as a useful paradigm.
Data binding and MVVM
Data binding and MVVM
3/18/2022 • 9 minutes to read • Edit Online
The Model-View-ViewModel (MVVM) pattern enforces a separation between three software layers — the XAML
user interface, called the view, the underlying data, called the model, and an intermediary between the view and
the model, called the viewmodel. The view and the viewmodel are often connected through data bindings
defined in XAML. The BindingContext for the view is usually an instance of the viewmodel.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Simple MVVM
In XAML markup extensions you saw how to define a new XML namespace declaration to allow a XAML file to
reference classes in other assemblies. The following example uses the x:Static markup extension to obtain the
current date and time from the static DateTime.Now property in the System namespace:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:sys="clr-namespace:System;assembly=netstandard"
x:Class="XamlSamples.OneShotDateTimePage"
Title="One-Shot DateTime Page">
<StackLayout BindingContext="{x:Static sys:DateTime.Now}"
HorizontalOptions="Center"
VerticalOptions="Center">
<Label Text="{Binding Year, StringFormat='The year is {0}'}" />
<Label Text="{Binding StringFormat='The month is {0:MMMM}'}" />
<Label Text="{Binding Day, StringFormat='The day is {0}'}" />
<Label Text="{Binding StringFormat='The time is {0:T}'}" />
</StackLayout>
</ContentPage>
In this example, the retrieved DateTime value is set as the BindingContext on a StackLayout . When you set the
BindingContext on an element, it is inherited by all the children of that element. This means that all the children
of the StackLayout have the same BindingContext , and they can contain bindings to properties of that object:
However, the problem is that the date and time are set once when the page is constructed and initialized, and
never change.
A XAML page can display a clock that always shows the current time, but it requires additional code. The MVVM
pattern is a natural choice for .NET MAUI apps when data binding from properties between visual objects and
the underlying data. When thinking in terms of MVVM, the model and viewmodel are classes written entirely in
code. The view is often a XAML file that references properties defined in the viewmodel through data bindings.
In MVVM, a model is ignorant of the viewmodel, and a viewmodel is ignorant of the view. However, often you
tailor the types exposed by the viewmodel to the types associated with the UI.
NOTE
In simple examples of MVVM, such as those shown here, often there is no model at all, and the pattern involves just a
view and viewmodel linked with data bindings.
The following example shows a viewmodel for a clock, with a single property named DateTime that's updated
every second:
using System.ComponentModel;
namespace XamlSamples
{
public class ClockViewModel : INotifyPropertyChanged
{
DateTime dateTime;
public event PropertyChangedEventHandler PropertyChanged;
public ClockViewModel()
{
this.DateTime = DateTime.Now;
Viewmodels typically implement the INotifyPropertyChanged interface, which provides the ability for a class to
fire the PropertyChanged event whenever one of its properties changes. The data binding mechanism in .NET
MAUI attaches a handler to this PropertyChanged event so it can be notified when a property changes and keep
the target updated with the new value.
The following example shows XAML that consumes ClockViewModel :
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:XamlSamples;assembly=XamlSamples"
x:Class="XamlSamples.ClockPage"
Title="Clock Page">
<ContentPage.BindingContext>
<local:ClockViewModel />
</ContentPage.BindingContext>
In this example, ClockViewModel is set to the BindingContext of the ContentPage using property element tags.
Alternatively, the code-behind file could instantiate the viewmodel.
The Binding markup extension on the Text property of the Label formats the DateTime property. The
following screenshot shows the result:
In addition, it’s possible to access individual properties of the DateTime property of the viewmodel by
separating the properties with periods:
Interactive MVVM
MVVM is often used with two-way data bindings for an interactive view based on an underlying data model.
The following example shows the HslViewModel that converts a Color value into Hue , Saturation , and
Luminosity values, and back again:
using System.ComponentModel;
namespace XamlSamples
{
public class HslViewModel : INotifyPropertyChanged
{
float hue, saturation, luminosity;
Color color;
OnPropertyChanged("Hue");
OnPropertyChanged("Saturation");
OnPropertyChanged("Luminosity");
OnPropertyChanged("Color");
}
}
}
In this example, changes to the Hue , Saturation , and Luminosity properties cause the Color property to
change, and changes to the Color property causes the other three properties to change. This might seem like
an infinite loop, except that the viewmodel doesn't invoke the PropertyChanged event unless the property has
changed.
The following XAML example contains a BoxView whose Color property is bound to the Color property of the
viewmodel, and three Slider and three Label views bound to the Hue , Saturation , and Luminosity
properties:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:XamlSamples;assembly=XamlSamples"
x:Class="XamlSamples.HslColorScrollPage"
Title="HSL Color Scroll Page">
<ContentPage.BindingContext>
<local:HslViewModel Color="Aqua" />
</ContentPage.BindingContext>
The binding on each Label is the default OneWay . It only needs to display the value. However, the default
binding on each Slider is TwoWay . This allows the Slider to be initialized from the viewmodel. When the
viewmodel is instantiated it's Color property is set to Aqua . A change in a Slider sets a new value for the
property in the viewmodel, which then calculates a new color:
Commanding
Sometimes an app has needs that go beyond property bindings by requiring the user to initiate commands that
affect something in the viewmodel. These commands are generally signaled by button clicks or finger taps, and
traditionally they are processed in the code-behind file in a handler for the Clicked event of the Button or the
Tapped event of a TapGestureRecognizer .
The commanding interface provides an alternative approach to implementing commands that is much better
suited to the MVVM architecture. The viewmodel can contain commands, which are methods that are executed
in reaction to a specific activity in the view such as a Button click. Data bindings are defined between these
commands and the Button .
To allow a data binding between a Button and a viewmodel, the Button defines two properties:
Command of type System.Windows.Input.ICommand
CommandParameter of type Object
NOTE
Many other controls also define Command and CommandParameter properties.
The ICommand interface is defined in the System.Windows.Input namespace, and consists of two methods and
one event:
void Execute(object arg)
bool CanExecute(object arg)
event EventHandler CanExecuteChanged
The viewmodel can define properties of type ICommand . You can then bind these properties to the Command
property of each Button or other element, or perhaps a custom view that implements this interface. You can
optionally set the CommandParameter property to identify individual Button objects (or other elements) that are
bound to this viewmodel property. Internally, the Button calls the Execute method whenever the user taps the
Button , passing to the Execute method its CommandParameter .
The CanExecute method and CanExecuteChanged event are used for cases where a Button tap might be
currently invalid, in which case the Button should disable itself. The Button calls CanExecute when the
Command property is first set and whenever the CanExecuteChanged event is raised. If CanExecute returns false ,
the Button disables itself and doesn’t generate Execute calls.
You can use the Command or Command<T> class included in .NET MAUI to implement the ICommand interface.
These two classes define several constructors plus a ChangeCanExecute method that the viewmodel can call to
force the Command object to fire the CanExecuteChanged event.
The following example shows a viewmodel for a simple keypad that is intended for entering telephone numbers:
using System.ComponentModel;
using System.Windows.Input;
namespace XamlSamples
{
public class KeypadViewModel : INotifyPropertyChanged
{
string inputString = "";
string displayText = "";
char[] specialChars = { '*', '#' };
public KeypadViewModel()
{
AddCharCommand = new Command<string>((key) =>
{
// Add the key to the input string.
InputString += key;
});
In this example, the Execute and CanExecute methods for the commands are defined as lambda functions in
the constructor. The viewmodel assumes that the AddCharCommand property is bound to the Command property of
several buttons (or anything other controls that have a command interface), each of which is identified by the
CommandParameter . These buttons add characters to an InputString property, which is then formatted as a
phone number for the DisplayText property. There's also a second property of type ICommand named
DeleteCharCommand . This is bound to a back-spacing button, but the button should be disabled if there are no
characters to delete.
The following example shows the XAML that consumes the KeypadViewModel :
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:XamlSamples;assembly=XamlSamples"
x:Class="XamlSamples.KeypadPage"
Title="Keypad Page">
<ContentPage.BindingContext>
<local:KeypadViewModel />
</ContentPage.BindingContext>
<Grid HorizontalOptions="Center"
VerticalOptions="Center">
<Grid.RowDefinitions>
<RowDefinition Height="Auto" />
<RowDefinition Height="Auto" />
<RowDefinition Height="Auto" />
<RowDefinition Height="Auto" />
<RowDefinition Height="Auto" />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="80" />
<ColumnDefinition Width="80" />
<ColumnDefinition Width="80" />
</Grid.ColumnDefinitions>
In this example, the Command property of the first Button that is bound to the DeleteCharCommand . The other
buttons are bound to the AddCharCommand with a CommandParameter that's the same as the character that appears
on the Button :
XAML compilation
3/18/2022 • 2 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) XAML is compiled directly into intermediate language (IL) with the
XAML compiler (XAMLC). XAML compilation offers a number of a benefits:
It performs compile-time checking of XAML, notifying you of any errors.
It removes some of the load and instantiation time for XAML elements.
It helps to reduce the file size of the final assembly by no longer including .xaml files.
XAML compilation is enabled by default in .NET MAUI apps. For apps built using debug configuration, XAML
compilation provides compile-time validation of XAML, but does not convert the XAML to IL in the assembly.
Instead, XAML files are included as embedded resources in the app package, and evaluated at runtime. For apps
built using release configuration, XAML compilation provides compile-time validation of XAML, and converts the
XAML to IL that's written to the assembly. However, XAML compilation behavior can be overridden in both
configurations with the XamlCompilationAttribute class.
Enable compilation
XAML compilation can be enabled by passing XamlCompilationOptions.Compile to the XamlCompilationAttribute :
[assembly: XamlCompilation(XamlCompilationOptions.Compile)]
In this example, XAML compilation is enabled for all of the XAML contained within the assembly, with XAML
errors being reported at compile-time rather than runtime.
TIP
While the XamlCompilationAttribute can be placed anywhere, a good place to put it is in AssemblyInfo.cs .
[XamlCompilation (XamlCompilationOptions.Compile)]
public partial class MyPage : ContentPage
{
...
}
In this example, XAML compilation is enabled only for the MyPage class.
NOTE
Compiled bindings can be enabled to improve data binding performance in .NET MAUI applications. For more
information, see Compiled Bindings.
Disable compilation
XAML compilation can be disabled by passing XamlCompilationOptions.Skip to the XamlCompilationAttribute :
[assembly: XamlCompilation(XamlCompilationOptions.Skip)]
In this example, XAML compilation is disabled within the assembly, with XAML errors being reported at runtime
rather than compile-time.
XAML compilation can also be disabled at the type level:
[XamlCompilation (XamlCompilationOptions.Skip)]
public partial class MyPage : ContentPage
{
...
}
In this example, XAML compilation is disabled only for the MyPage class.
WARNING
Disabling XAML compilation is not recommended because XAML is then parsed and interpreted at runtime, which will
reduce app performance.
Field modifiers
3/18/2022 • 2 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) x:FieldModifier attribute specifies the access level for generated
fields for named XAML elements.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
By default, if the value of the attribute isn't set, the generated field for the element will be private .
NOTE
The value of the attribute can use any casing, as it will be converted to lowercase by .NET MAUI.
IMPORTANT
The x:FieldModifier attribute cannot be used to specify the access level of a .NET MAUI XAML class.
Generics
3/18/2022 • 3 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) XAML provides support for consuming generic CLR types by specifying
the generic constraints as type arguments. This support is provided by the x:TypeArguments directive, which
passes the constraining type arguments of a generic to the constructor of the generic type.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Type arguments are specified as a string, and are typically prefixed, such as sys:String and sys:Int32 .
Prefixing is required because the typical types of CLR generic constraints come from libraries that are not
mapped to the default .NET MAUI namespaces. However, the XAML 2009 built-in types such as x:String and
x:Int32 , can also be specified as type arguments, where x is the XAML language namespace for XAML 2009.
For more information about the XAML 2009 built-in types, see XAML 2009 Language Primitives.
IMPORTANT
Defining generic classes in .NET MAUI XAML, with the x:TypeArguments directive, is unsupported.
Multiple type arguments can be specified by using a comma delimiter. In addition, if a generic constraint uses
generic types, the nested constraint type arguments should be contained in parentheses.
NOTE
The x:Type markup extension supplies a Common Language Runtime (CLR) type reference for a generic type, and has a
similar function to the typeof operator in C#.
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:scg="clr-namespace:System.Collections.Generic;assembly=netstandard"
xmlns:sys="clr-namespace:System;assembly=netstandard"
...>
<CollectionView>
<CollectionView.ItemsSource>
<scg:List x:TypeArguments="sys:String">
<sys:String>Baboon</sys:String>
<sys:String>Capuchin Monkey</sys:String>
<sys:String>Blue Monkey</sys:String>
<sys:String>Squirrel Monkey</sys:String>
<sys:String>Golden Lion Tamarin</sys:String>
<sys:String>Howler Monkey</sys:String>
<sys:String>Japanese Macaque</sys:String>
</scg:List>
</CollectionView.ItemsSource>
</CollectionView>
</ContentPage>
ImageUrl="https://upload.wikimedia.org/wikipedia/commons/thumb/f/fc/Papio_anubis_%28Serengeti%2C_2009%29.jpg
/200px-Papio_anubis_%28Serengeti%2C_2009%29.jpg" />
<models:Monkey Name="Capuchin Monkey"
Location="Central and South America"
ImageUrl="https://upload.wikimedia.org/wikipedia/commons/thumb/4/40/Capuchin_Costa_Rica.jpg/200px-
Capuchin_Costa_Rica.jpg" />
<models:Monkey Name="Blue Monkey"
Location="Central and East Africa"
ImageUrl="https://upload.wikimedia.org/wikipedia/commons/thumb/8/83/BlueMonkey.jpg/220px-BlueMonkey.jpg" />
</scg:List>
</CollectionView.ItemsSource>
<CollectionView.ItemTemplate>
<DataTemplate>
<Grid Padding="10">
<Grid.RowDefinitions>
<RowDefinition Height="Auto" />
<RowDefinition Height="Auto" />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="Auto" />
<ColumnDefinition Width="Auto" />
</Grid.ColumnDefinitions>
<Image Grid.RowSpan="2"
Source="{Binding ImageUrl}"
Aspect="AspectFill"
HeightRequest="60"
WidthRequest="60" />
<Label Grid.Column="1"
Text="{Binding Name}"
FontAttributes="Bold" />
<Label Grid.Row="1"
Grid.Column="1"
Text="{Binding Location}"
FontAttributes="Italic"
VerticalOptions="End" />
</Grid>
</DataTemplate>
</CollectionView.ItemTemplate>
</CollectionView>
</ContentPage>
ImageUrl="https://upload.wikimedia.org/wikipedia/commons/thumb/f/fc/Papio_anubis_%28Serengeti%2C_2009%29.jpg
/200px-Papio_anubis_%28Serengeti%2C_2009%29.jpg" />
</x:Arguments>
</scg:KeyValuePair>
<scg:KeyValuePair x:TypeArguments="x:String,models:Monkey">
<x:Arguments>
<x:String>Capuchin Monkey</x:String>
<models:Monkey Location="Central and South America"
ImageUrl="https://upload.wikimedia.org/wikipedia/commons/thumb/4/40/Capuchin_Costa_Rica.jpg/200px-
Capuchin_Costa_Rica.jpg" />
</x:Arguments>
</scg:KeyValuePair>
<scg:KeyValuePair x:TypeArguments="x:String,models:Monkey">
<x:Arguments>
<x:String>Blue Monkey</x:String>
<models:Monkey Location="Central and East Africa"
ImageUrl="https://upload.wikimedia.org/wikipedia/commons/thumb/8/83/BlueMonkey.jpg/220px-BlueMonkey.jpg" />
</x:Arguments>
</scg:KeyValuePair>
</scg:List>
</CollectionView.ItemsSource>
<CollectionView.ItemTemplate>
<DataTemplate>
<Grid Padding="10">
<Grid.RowDefinitions>
<RowDefinition Height="Auto" />
<RowDefinition Height="Auto" />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="Auto" />
<ColumnDefinition Width="Auto" />
</Grid.ColumnDefinitions>
<Image Grid.RowSpan="2"
Source="{Binding Value.ImageUrl}"
Aspect="AspectFill"
HeightRequest="60"
WidthRequest="60" />
<Label Grid.Column="1"
Text="{Binding Key}"
FontAttributes="Bold" />
<Label Grid.Row="1"
Grid.Column="1"
Text="{Binding Value.Location}"
FontAttributes="Italic"
VerticalOptions="End" />
</Grid>
</DataTemplate>
</CollectionView.ItemTemplate>
</CollectionView>
</ContentPage
In this example, GenericsDemo.Models is defined as the models XAML namespace, and
System.Collections.Generic is defined as the scg XAML namespace. The CollectionView.ItemsSource property
is set to a List<T> that's instantiated with a KeyValuePair<TKey, TValue> constraint, with the inner constraint
type arguments string and Monkey . The List<KeyValuePair<string,Monkey>> collection is initialized with
multiple KeyValuePair items, using the non-default KeyValuePair constructor, and a DataTemplate that defines
the appearance of each Monkey object is set as the ItemTemplate of the CollectionView . For information on
passing arguments to a non-default constructor, see Pass constructor arguments.
Consume XAML markup extensions
3/18/2022 • 14 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) XAML markup extensions help enhance the power and flexibility of
XAML by allowing element attributes to be set from a variety of sources.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
For example, you typically set the Color property of BoxView like this:
However, you might prefer instead to set the Color attribute from a value stored in a resource dictionary, or
from the value of a static property of a class that you've created, or from a property of type Color of another
element on the page, or constructed from separate hue, saturation, and luminosity values. All these options are
possible using XAML markup extensions.
A markup extension is a different way to express an attribute of an element. .NET MAUI XAML markup
extensions are usually identifiable by an attribute value that is enclosed in curly braces:
Any attribute value in curly braces is always a XAML markup extension. However, XAML markup extensions can
also be referenced without the use of curly braces.
NOTE
Several XAML markup extensions are part of the XAML 2009 specification. These appear in XAML files with the customary
x namespace prefix, and are commonly referred to with this prefix.
In addition to the markup extensions discussed in this article, the following markup extensions are included in
.NET MAUI and discussed in other articles:
StaticResource - reference objects from a resource dictionary.
DynamicResource - respond to changes in objects in a resource dictionary.
Binding - establish a link between properties of two objects. For more information, see Data binding**.
TemplateBinding - performs data binding from a control template.
RelativeSource - sets the binding source relative to the position of the binding target. For more information,
see Relative bindings.
ConstraintExpression - relates the position and size of a child in a RelativeLayout to its parent, or a sibling.
The following XAML demonstrates the most verbose approach to instantiating the StaticExtension class
between Label.FontSize property-element tags:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:sys="clr-namespace:System;assembly=netstandard"
xmlns:local="clr-namespace:MarkupExtensions"
x:Class="MarkupExtensions.StaticDemoPage"
Title="x:Static Demo">
<StackLayout Margin="10, 0">
<Label Text="Label No. 1">
<Label.FontSize>
<x:StaticExtension Member="local:AppConstants.NormalFontSize" />
</Label.FontSize>
</Label>
···
</StackLayout>
</ContentPage>
The XAML parser also allows the StaticExtension class to be abbreviated as x:Static :
This syntax can be simplified even further by putting the StaticExtension class and the member setting in curly
braces. The resulting expression is set directly to the FontSize attribute:
In this example, there are no quotation marks within the curly braces. The Member property of StaticExtension
is no longer an XML attribute. It is instead part of the expression for the markup extension.
Just as you can abbreviate x:StaticExtension to x:Static when you use it as an object element, you can also
abbreviate it in the expression within curly braces:
The StaticExtension class has a ContentProperty attribute referencing the property Member , which marks this
property as the class's default content property. For XAML markup extensions expressed with curly braces, you
can eliminate the Member= part of the expression:
In this example, both x:Reference expressions use the abbreviated version of the ReferenceExtension class
name and eliminate the Name= part of the expression. In the first example, the x:Reference markup extension is
embedded in the Binding markup extension and the Source and StringFormat properties are separated by
commas.
The following screenshot shows the XAML output:
public TypeDemoPage()
{
InitializeComponent();
BindingContext = this;
}
}
When a is pressed a new instance of the CommandParameter argument is created and added to the
Button
StackLayout . The three Button objects then share the page with dynamically created views:
x:Array markup extension
The x:Array markup extension enables you to define an array in markup. It is supported by the ArrayExtension
class, which defines two properties:
Type of type Type , which indicates the type of the elements in the array. This property should be set to an
x:Type markup extension.
Items of type IList , which is a collection of the items themselves. This is the content property of
ArrayExtension .
The x:Array markup extension itself never appears in curly braces. Instead, x:Array start and end tags delimit
the list of items.
The following XAML example shows how to use x:Array to add items to a ListView by setting the
ItemsSource property to an array:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="MarkupExtensions.ArrayDemoPage"
Title="x:Array Demo Page">
<ListView Margin="10">
<ListView.ItemsSource>
<x:Array Type="{x:Type Color}">
<Color>Aqua</Color>
<Color>Black</Color>
<Color>Blue</Color>
<Color>Fuchsia</Color>
<Color>Gray</Color>
<Color>Green</Color>
<Color>Lime</Color>
<Color>Maroon</Color>
<Color>Navy</Color>
<Color>Olive</Color>
<Color>Pink</Color>
<Color>Purple</Color>
<Color>Red</Color>
<Color>Silver</Color>
<Color>Teal</Color>
<Color>White</Color>
<Color>Yellow</Color>
</x:Array>
</ListView.ItemsSource>
<ListView.ItemTemplate>
<DataTemplate>
<ViewCell>
<BoxView Color="{Binding}"
Margin="3" />
</ViewCell>
</DataTemplate>
</ListView.ItemTemplate>
</ListView>
</ContentPage>
In this example, the ViewCell creates a simple BoxView for each color entry:
NOTE
When defining arrays of common types like strings or numbers, use the XAML language primitives tags listed in Pass
arguments.
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="MarkupExtensions.NullDemoPage"
Title="x:Null Demo">
<ContentPage.Resources>
<Style TargetType="Label">
<Setter Property="FontSize" Value="48" />
<Setter Property="FontFamily" Value="OpenSansRegular" />
</Style>
</ContentPage.Resources>
In this example, an implicit Style is defined for Label that includes a Setter that sets the FontFamily
property to a specific font. However, the third Label avoids using the font defined in the implicit style by setting
its FontFamily to x:Null :
NOTE
The XAML parser allows the OnPlatformExtension class to be abbreviated as OnPlatform .
The Default property is the content property of OnPlatformExtension . Therefore, for XAML markup expressions
expressed with curly braces, you can eliminate the Default= part of the expression provided that it's the first
argument. If the Default property isn't set, it will default to the BindableProperty.DefaultValue property value,
provided that the markup extension is targeting a BindableProperty .
IMPORTANT
The XAML parser expects that values of the correct type will be provided to properties consuming the OnPlatform
markup extension. If type conversion is necessary, the OnPlatform markup extension will attempt to perform it using
the default converters provided by Xamarin.Forms. However, there are some type conversions that can't be performed by
the default converters and in these cases the Converter property should be set to an IValueConverter
implementation.
The OnPlatform Demo page shows how to use the OnPlatform markup extension:
In this example, all three OnPlatform expressions use the abbreviated version of the OnPlatformExtension class
name. The three OnPlatform markup extensions set the Color , WidthRequest , and HeightRequest properties of
the BoxView to different values on iOS and Android. The markup extensions also provide default values for
these properties on the platforms that aren't specified, while eliminating the Default= part of the expression.
The Default property is the content property of OnIdiomExtension . Therefore, for XAML markup expressions
expressed with curly braces, you can eliminate the Default= part of the expression provided that it's the first
argument.
IMPORTANT
The XAML parser expects that values of the correct type will be provided to properties consuming the OnIdiom markup
extension. If type conversion is necessary, the OnIdiom markup extension will attempt to perform it using the default
converters provided by .NET MAUI. However, there are some type conversions that can't be performed by the default
converters and in these cases the Converter property should be set to an IValueConverter implementation.
The following XAML example shows how to use the OnIdiom markup extension:
In this example, all three OnIdiom expressions use the abbreviated version of the OnIdiomExtension class name.
The three OnIdiom markup extensions set the Color , WidthRequest , and HeightRequest properties of the
BoxView to different values on the phone, tablet, and desktop idioms. The markup extensions also provide
default values for these properties on the idioms that aren't specified, while eliminating the Default= part of
the expression.
NOTE
The XAML parser allows the DataTemplateExtension class to be abbreviated as DataTemplate .
A typical usage of this markup extension is in a Shell application, as shown in the following example:
<ShellContent Title="Monkeys"
Icon="monkey.png"
ContentTemplate="{DataTemplate views:MonkeysPage}" />
In this example, MonkeysPage is converted from a ContentPage to a DataTemplate , which is set as the value of
the ShellContent.ContentTemplate property. This ensures that MonkeysPage is only created when navigation to
the page occurs, rather than at application startup.
NOTE
The XAML parser allows the FontImageExtension class to be abbreviated as FontImage .
The Glyph property is the content property of FontImageExtension . Therefore, for XAML markup expressions
expressed with curly braces, you can eliminate the Glyph= part of the expression provided that it's the first
argument.
The following XAML example shows how to use the FontImage markup extension:
<Image BackgroundColor="#D1D1D1"
Source="{FontImage , FontFamily=Ionicons, Size=44}" />
In this example, the abbreviated version of the FontImageExtension class name is used to display an XBox icon,
from the Ionicons font family, in an Image :
While the unicode character for the icon is \uf30c , it has to be escaped in XAML and so becomes  .
NOTE
The XAML parser allows the AppThemeBindingExtension class to be abbreviated as AppBindingTheme .
The Default property is the content property of AppThemeBindingExtension . Therefore, for XAML markup
expressions expressed with curly braces, you can eliminate the Default= part of the expression provided that
it's the first argument.
The following XAML example shows how to use the AppThemeBinding markup extension:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="MarkupExtensions.AppThemeBindingDemoPage"
Title="AppThemeBinding Demo">
<ContentPage.Resources>
<Style x:Key="labelStyle"
TargetType="Label">
<Setter Property="TextColor"
Value="{AppThemeBinding Black, Light=Blue, Dark=Teal}" />
</Style>
</ContentPage.Resources>
<StackLayout Margin="20">
<Label Text="This text is green in light mode, and red in dark mode."
TextColor="{AppThemeBinding Light=Green, Dark=Red}" />
<Label Text="This text is black by default, blue in light mode, and teal in dark mode."
Style="{StaticResource labelStyle}" />
</StackLayout>
</ContentPage>
In this example, the text color of the first Label is set to green when the device is using its light theme, and is
set to red when the device is using its dark theme. The second Label has its TextColor property set through a
Style . This Style sets the text color of the Label to black by default, to blue when the device is using its light
theme, and to teal when the device is using its dark theme:
Create XAML markup extensions
3/18/2022 • 3 minutes to read • Edit Online
At the developer level, a .NET Multi-platform App UI (.NET MAUI) XAML markup extension is a class that
implements the IMarkupExtension or IMarkupExtension<T> interface. It's also possible to define your own
custom XAML markup extensions by deriving from IMarkupExtension or IMarkupExtension<T> . Use the generic
form if the markup extension obtains a value of a particular type. This is the case with several of the .NET MAUI
markup extensions:
TypeExtension derives from IMarkupExtension<Type>
ArrayExtension derives from IMarkupExtension<Array>
DynamicResourceExtension derives from IMarkupExtension<DynamicResource>
BindingExtension derives from IMarkupExtension<BindingBase>
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The two IMarkupExtension interfaces define only one method each, named ProvideValue :
Since IMarkupExtension<T> derives from IMarkupExtension and includes the new keyword on ProvideValue , it
contains both ProvideValue methods.
Often, XAML markup extensions define properties that contribute to the return value, and the ProvideValue
method has a single argument of type IServiceProvider . For more information about service providers, see
Service providers.
Because IMarkupExtension<T> derives from IMarkupExtension , the class must contain two ProvideValue
methods, one that returns a Color and another that returns an object , but the second method can call the first
method.
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:MarkupExtensions"
x:Class="MarkupExtensions.HslColorDemoPage"
Title="HSL Color Demo">
<ContentPage.Resources>
<Style TargetType="BoxView">
<Setter Property="WidthRequest" Value="80" />
<Setter Property="HeightRequest" Value="80" />
<Setter Property="HorizontalOptions" Value="Center" />
<Setter Property="VerticalOptions" Value="Center" />
</Style>
</ContentPage.Resources>
<StackLayout>
<BoxView>
<BoxView.Color>
<local:HslColorExtension H="0" S="1" L="0.5" A="1" />
</BoxView.Color>
</BoxView>
<BoxView>
<BoxView.Color>
<local:HslColor H="0.33" S="1" L="0.5" />
</BoxView.Color>
</BoxView>
<BoxView Color="{local:HslColorExtension H=0.67, S=1, L=0.5}" />
<BoxView Color="{local:HslColor H=0, S=0, L=0.5}" />
<BoxView Color="{local:HslColor A=0.5}" />
</StackLayout>
</ContentPage>
In this example, when HslColorExtension is an XML tag the four properties are set as attributes, but when it
appears between curly braces, the four properties are separated by commas without quotation marks. The
default values for H , S , and L are 0, and the default value of A is 1, so those properties can be omitted if
you want them set to default values. The last example shows an example where the luminosity is 0, which
normally results in black, but the alpha channel is 0.5, so it is half transparent and appears gray against the
white background of the page:
Service providers
By using the IServiceProvider argument to ProvideValue , XAML markup extensions can get access to data
about the XAML file in which they're being used. For example, the IProvideValueTarget service enables you to
retrieve data about the object the markup extension is applied to:
The IProvideValueTarget interface defines two properties, TargetObject and TargetProperty . When this
information is obtained in the HslColorExtension class, TargetObject is the BoxView and TargetProperty is the
Color property of BoxView . This is the property on which the XAML markup extension has been set.
XAML namespaces
3/18/2022 • 3 minutes to read • Edit Online
XAML uses the xmlns XML attribute for namespace declarations. There are two XAML namespace declarations
that are always within the root element of a XAML file. The first defines the default namespace:
xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
The default namespace specifies that elements defined within the XAML file with no prefix refer to .NET Multi-
platform App UI (.NET MAUI) classes, such as ContentPage , Label , and Button .
The second namespace declaration uses the x prefix:
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
XAML uses prefixes to declare non-default namespaces, with the prefix being used when referencing types
within the namespace. The x namespace declaration specifies that elements defined within XAML with a prefix
of x are used for elements and attributes that are intrinsic to XAML (specifically the 2009 XAML specification).
The following table outlines the x constructs supported by .NET MAUI:
x:Class Specifies the namespace and class name for a class defined in
XAML. The class name must match the class name of the
code-behind file. Note that this construct can only appear in
the root element of a XAML file.
x:DataType Specifies the type of the object that the XAML element, and
it's children, will bind to.
x:FieldModifier Specifies the access level for generated fields for named
XAML elements.
In XAML, namespace declarations inherit from parent element to child element. Therefore, when defining a
namespace in the root element of a XAML file, all elements within that file inherit the namespace declaration.
NOTE
The character separating the clr-namespace or using token from its value is a colon, whereas the character separating
the assembly token from its value is an equal sign. The character to use between the two tokens is a semicolon.
The local prefix is a convention used to indicate that the types within the namespace are local to the app.
Alternatively, if the types are in a different assembly, the assembly name should also be defined in the
namespace declaration:
The namespace prefix is then specified when declaring an instance of a type from an imported namespace:
<controls:Expander IsExpanded="True">
...
</controls:Expander>
For information about defining a custom namespace schema, see Custom namespace schemas.
Custom namespace schemas
3/18/2022 • 3 minutes to read • Edit Online
Types in a .NET Multi-platform App UI (.NET MAUI) library can be referenced in XAML by declaring an XML
namespace for the library, with the namespace declaration specifying the Common Language Runtime (CLR)
namespace name and an assembly name:
<ContentPage ...
xmlns:controls="clr-namespace:MyCompany.Controls;assembly=MyCompany.Controls">
...
</ContentPage>
However, specifying a CLR namespace and assembly name in a xmlns definition can be awkward and error
prone. In addition, multiple XML namespace declarations may be required if the library contains types in
multiple namespaces.
An alternative approach is to define a custom namespace schema, such as
http://mycompany.com/schemas/controls , that maps to one or more CLR namespaces. This enables a single XML
namespace declaration to reference all the types in an assembly, even if they are in different namespaces. It also
enables a single XML namespace declaration to reference types in multiple assemblies.
For more information about XAML namespaces, see XAML namespaces.
namespace MyCompany.Controls
{
public class CircleButton : Button
{
...
}
}
All the controls in the library reside in the MyCompany.Controls namespace. These controls can be exposed to a
calling assembly through a custom namespace schema.
A custom namespace schema is defined with the XmlnsDefinitionAttribute class, which specifies the mapping
between a XAML namespace and one or more CLR namespaces. The XmlnsDefinitionAttribute takes two
arguments: the XAML namespace name, and the CLR namespace name. The XAML namespace name is stored in
the XmlnsDefinitionAttribute.XmlNamespace property, and the CLR namespace name is stored in the
XmlnsDefinitionAttribute.ClrNamespace property.
NOTE
The XmlnsDefinitionAttribute class also has a property named AssemblyName , which can be optionally set to the
name of the assembly. This is only required when a CLR namespace referenced from a XmlnsDefinitionAttribute is in a
external assembly.
The XmlnsDefinitionAttribute should be defined at the assembly level in the project that contains the CLR
namespaces that will be mapped in the custom namespace schema. The following example shows the
AssemblyInfo.cs file from a class library:
using MyCompany.Controls;
[assembly: Preserve]
[assembly: XmlnsDefinition("http://mycompany.com/schemas/controls", "MyCompany.Controls")]
This code creates a custom namespace schema that maps the http://mycompany.com/schemas/controls URL to the
MyCompany.Controls CLR namespace. In addition, the Preserve attribute is specified on the assembly, to ensure
that the linker preserves all the types in the assembly.
IMPORTANT
The Preserve attribute should be applied to classes in the assembly that are mapped through the custom namespace
schema, or applied to the entire assembly.
The custom namespace schema can then be used for type resolution in XAML files.
namespace MyCompany.Controls
{
public static class Controls
{
public static void Init()
{
}
}
}
The Init method can then be called from the assembly that consumes types from the custom namespace
schema:
using Xamarin.Forms;
using MyCompany.Controls;
namespace CustomNamespaceSchemaDemo
{
public partial class MainPage : ContentPage
{
public MainPage()
{
Controls.Init();
InitializeComponent();
}
}
}
WARNING
Failure to include such a code reference will result in the XAML compiler being unable to locate the assembly containing
the custom namespace schema types.
To consume the CircleButton control, a XAML namespace is declared, with the namespace declaration
specifying the custom namespace schema URL:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:controls="http://mycompany.com/schemas/controls"
x:Class="CustomNamespaceSchemaDemo.MainPage">
<StackLayout>
...
<controls:CircleButton Text="+"
BackgroundColor="Fuchsia"
BorderColor="Black"
CircleDiameter="100" />
<controls:CircleButton Text="-"
BackgroundColor="Teal"
BorderColor="Silver"
CircleDiameter="70" />
...
</StackLayout>
</ContentPage>
CircleButton instances can then be added to the ContentPage by declaring them with the controls namespace
prefix.
To find the custom namespace schema types, .NET MAUI will search referenced assemblies for
XmlnsDefinitionAttribute instances. If the xmlns attribute for an element in a XAML file matches the
XmlNamespace property value in a XmlnsDefinitionAttribute , .NET MAUI will attempt to use the
XmlnsDefinitionAttribute.ClrNamespace property value for resolution of the type. If type resolution fails, .NET
MAUI will continue to attempt type resolution based on any additional matching XmlnsDefinitionAttribute
instances.
The result is that two CircleButton instances are displayed:
Custom namespace prefixes
3/18/2022 • 2 minutes to read • Edit Online
.The NET Multi-platform App UI (.NET MAUI) XmlnsPrefixAttribute class can be used by control authors to
specify a recommended prefix to associate with a XAML namespace, for XAML usage. The prefix is useful when
supporting object tree serialization to XAML, or when interacting with a design environment that has XAML
editing features. For example:
XAML text editors could use the XmlnsPrefixAttribute as a hint for an initial XAML namespace xmlns
mapping.
XAML design environments could use the XmlnsPrefixAttribute to add mappings to the XAML when
dragging objects out of a toolbox and onto a visual design surface.
Recommended namespace prefixes should be defined at the assembly level with the XmlnsPrefixAttribute
constructor, which takes two arguments: a string that specifies the identifier of a XAML namespace, and a string
that specifies a recommended prefix:
Prefixes should use short strings, because the prefix is typically applied to all serialized elements that come from
the XAML namespace. Therefore, the prefix string length can have a noticeable effect on the size of the serialized
XAML output.
NOTE
More than one XmlnsPrefixAttribute can be applied to an assembly. For example, if you have an assembly that defines
types for more than one XAML namespace, you could define different prefix values for each XAML namespace.
Pass arguments
3/18/2022 • 3 minutes to read • Edit Online
It's often necessary to instantiate objects with constructors that require arguments, or by calling a static creation
method. This can be achieved in .NET Multi-platform App UI (.NET MAUI) XAML by using the x:Arguments and
x:FactoryMethod attributes:
The x:Arguments attribute is used to specify constructor arguments for a non-default constructor, or for a
factory method object declaration. For more information, see Pass constructor arguments.
The x:FactoryMethod attribute is used to specify a factory method that can be used to initialize an object. For
more information, see Call factory methods.
In addition, the x:TypeArguments attribute can be used to specify the generic type arguments to the constructor
of a generic type. For more information, see Specify a generic type argument.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Arguments can be passed to constructors and factory methods using the following .NET MAUI XAML language
primitives:
x:Array , which corresponds to Array .
x:Boolean , which corresponds to Boolean .
x:Byte , which corresponds to Byte .
x:Char , which corresponds to Char .
x:DateTime , which corresponds to DateTime .
x:Decimal , which corresponds to Decimal .
x:Double , which corresponds to Double .
x:Int16 , which corresponds to Int16 .
x:Int32 , which corresponds to Int32 .
x:Int64 , which corresponds to Int64 .
x:Object , which corresponds to the Object .
x:Single , which corresponds to Single .
x:String , which corresponds to String .
x:TimeSpan , which corresponds to TimeSpan .
With the exception of x:DateTime , the other language primitives are in the XAML 2009 specification.
NOTE
The x:Single language primitive can be used to pass float arguments.
<BoxView HeightRequest="150"
WidthRequest="150"
HorizontalOptions="Center">
<BoxView.Color>
<Color>
<x:Arguments>
<x:Single>0.9</x:Single>
</x:Arguments>
</Color>
</BoxView.Color>
</BoxView>
<BoxView HeightRequest="150"
WidthRequest="150"
HorizontalOptions="Center">
<BoxView.Color>
<Color>
<x:Arguments>
<x:Single>0.25</x:Single>
<x:Single>0.5</x:Single>
<x:Single>0.75</x:Single>
</x:Arguments>
</Color>
</BoxView.Color>
</BoxView>
<BoxView HeightRequest="150"
WidthRequest="150"
HorizontalOptions="Center">
<BoxView.Color>
<Color>
<x:Arguments>
<x:Single>0.8</x:Single>
<x:Single>0.5</x:Single>
<x:Single>0.2</x:Single>
<x:Single>0.5</x:Single>
</x:Arguments>
</Color>
</BoxView.Color>
</BoxView>
The number of elements within the x:Arguments tag, and the types of these elements, must match one of the
Color constructors. The Color constructor with a single parameter requires a grayscale float value from 0
(black) to 1 (white). The Color constructor with three parameters requires float red, green, and blue values
ranging from 0 to 1. The Color constructor with four parameters adds a float alpha channel as the fourth
parameter.
The number of elements within the x:Arguments tag, and the types of these elements, must match the
arguments of the factory method being called. The FromRgba factory method requires four byte arguments,
which represent the red, green, blue, and alpha values, ranging from 0 to 255 respectively. The FromHsla factory
method requires four float arguments, which represent the hue, saturation, luminosity, and alpha values,
ranging from 0 to 1 respectively. The FromHex factory method requires a string argument that represents the
hexadecimal (A)RGB color.
<StackLayout>
<StackLayout.Margin>
<OnPlatform x:TypeArguments="Thickness">
<On Platform="iOS" Value="0,20,0,0" />
<On Platform="Android" Value="5, 10" />
</OnPlatform>
</StackLayout.Margin>
</StackLayout>
The OnPlatform class is a generic class and must be instantiated with an x:TypeArguments attribute that matches
the target type. In the On class, the Platform attribute can accept a single string value, or multiple comma-
delimited string values. In this example, the StackLayout.Margin property is set to a platform-specific
Thickness .
For more information about generic type arguments, see Generics in XAML.
Load XAML at runtime
3/18/2022 • 2 minutes to read • Edit Online
When a .NET Multi-platform App UI (.NET MAUI) XAML class is constructed, a LoadFromXaml method is indirectly
called. This occurs because the code-behind file for a XAML class calls the InitializeComponent method from its
constructor:
When Visual Studio builds a project containing a XAML file, a source generator generates new C# source that
contains the definition of the InitializeComponent method and adds it to the compilation object. The following
example shows the generated InitializeComponent method for the MainPage class:
The InitializeComponent method calls the LoadFromXaml method to extract the XAML compiled binary (or its
file) from the app package. After extraction, it initializes all of the objects defined in the XAML, connects them all
together in parent-child relationships, attaches event handlers defined in code to events set in the XAML file, and
sets the resultant tree of objects as the content of the page.
WARNING
Loading XAML at runtime has a significant performance cost, and generally should be avoided.
In this example, a Button instance is created, with its Text property value being set from the XAML defined in
the string . The Button is then added to a StackLayout that has been defined in the XAML for the page.
NOTE
The LoadFromXaml extension methods allow a generic type argument to be specified. However, it's rarely necessary to
specify the type argument, as it will be inferred from the type of the instance it's operating on.
The LoadFromXaml method can be used to inflate any XAML, with the following example inflating a ContentPage
and then navigating to it:
Access elements
Loading XAML at runtime with the LoadFromXaml method does not allow strongly-typed access to the XAML
elements that have specified runtime object names (using x:Name ). However, these XAML elements can be
retrieved using the FindByName method, and then accessed as required:
In this example, the XAML for a ContentPage is inflated. This XAML includes a Label named monkeyName , which
is retrieved using the FindByName method, before its Text property is set.
Build accessible apps with semantic properties
3/18/2022 • 14 minutes to read • Edit Online
Semantics for accessibility is concerned with building experiences that make your apps inclusive for people who
use technology in a wide range of environments and approach your UI with a range of needs and experiences. In
many situations, legal requirements for accessibility may provide an impetus for developers to address
accessibility issues. Regardless, it's advisable to build inclusive and accessible apps so that your apps reach the
largest possible audience.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The Web Content Accessibility Guidelines (WCAG) are the global accessibility standard and legal benchmark for
web and mobile. These guidelines describe the various ways in which apps can be made more perceivable,
operable, understandable, and robust, for all.
Many user accessibility needs are met by assistive technology products installed by the user or by tools and
settings provided by the operating system. This includes functionality such as screen readers, screen
magnification, and high-contrast settings.
Screen readers typically provide auditory descriptions of controls that are displayed on the screen. These
descriptions help users navigate through the app and provide references to controls, such as images, that have
no input or text. Screen readers are often controlled through gestures on the touchscreen, trackpad, or
keyboard. For information about enabling screen readers, see Enable screen readers.
Operating systems have their own screen readers with their own unique behavior and configuration. For
example, most screen readers read the text associated with a control when it receives focus, enabling users to
orient themselves as they navigate through the app. However, some screen readers can also read the entire app
user interface when a page appears, which enables the user to receive all of the page's available informational
content before attempting to navigate it.
Most screen readers will automatically read any text associated with a control that receives accessibility focus.
This means that controls, such as Label or Button , that have a Text property set will be accessible for the
user. However, Image , ImageButton , ActivityIndicator , and others might not be in the accessibility tree
because no text is associated with them.
.NET Multi-platform App UI (.NET MAUI) supports two approaches to providing access to the accessibility
experience of the underlying platform. Semantic properties are the .NET MAUI approach to providing
accessibility values in apps, and are the recommended approach. Automation properties are the Xamarin.Forms
approach to providing accessibility values in apps, and have been superseded by semantic properties. In both
cases, the default accessibility order of controls is the same order in which they're listed in XAML or added to the
layout. However, different layouts might have additional factors that influence accessibility order. For example,
the accessibility order of StackLayout is also based on its orientation, and the accessibility order of Grid is
based on its row and column arrangement. For more information about content ordering, see Meaningful
Content Ordering on the Xamarin blog.
NOTE
When a WebView displays a website that's accessible, it will also be accessible in a .NET MAUI app. Conversely, when a
WebView displays a website that's not accessible, it won't be accessible in a .NET MAUI app.
Semantic properties
Semantic properties are used to define information about which controls should receive accessibility focus and
which text should be read aloud to the user. Semantic properties are attached properties that can be added to
any element to set the underlying platform accessibility APIs.
IMPORTANT
Semantic properties don't try to force equivalent behavior on each platform. Instead, they rely on the accessibility
experience provided by each platform.
<Image Source="dotnet_bot.png"
SemanticProperties.Description="Image of dotnet bot." />
In addition, the SetValue method can also be used to set the SemanticProperties.Description attached
property:
The accessibility information for an element can also be defined on another element. For example, a Label next
to an Entry can be used to describe what the Entry represents. This can be accomplished in XAML as follows:
<Label x:Name="label"
Text="Enter your name: " />
<Entry SemanticProperties.Description="{Binding Source={x:Reference label} Path=Text}" />
Hint
The SemanticProperties.Hint attached property represents a string that provides additional context to the
SemanticProperties.Description attached property, such as the purpose of a control. Setting this property can
be accomplished in XAML:
<Image Source="like.png"
SemanticProperties.Description="Like"
SemanticProperties.Hint="Like this post." />
In addition, the SetValue method can also be used to set the SemanticProperties.Hint attached property:
On Android, this property behaves slightly differently depending on the control it's attached to. For example, for
controls without text values, such as Switch and CheckBox , the controls will display the hint with the control.
However, for controls with text values, the hint is not displayed and is read after the text value.
WARNING
The SemanticProperties.Hint property conflicts with the Entry.Placeholder property on Android, which both map
to the same platform property. Therefore, setting a different SemanticProperties.Hint value to the
Entry.Placeholder value isn't recommended.
Heading levels
The SemanticProperties.HeadingLevel attached property enables an element to be marked as a heading to
organize the UI and make it easier to navigate. Some screen readers enable users to quickly jump between
headings.
Headings have a level from 1 to 9, and are represented by the SemanticHeadingLevel enumeration, which
defines None , and Level1 through Level9 members.
IMPORTANT
While Windows offers 9 levels of headings, Android and iOS only offer a single heading. Therefore, when
SemanticProperties.HeadingLevel is set on Windows it maps to the correct heading level. However, when set on
Android and iOS it maps to a single heading level.
Label label1 = new Label { Text = "Get started with .NET MAUI" };
Label label2 = new Label { Text = "Paragraphs of text go here." };
Label label3 = new Label { Text = "Installation" };
Label label4 = new Label { Text = "Paragraphs of text go here." };
Label label5 = new Label { Text = "Build your first app" };
Label label6 = new Label { Text = "Paragraphs of text go here." };
Label label7 = new Label { Text = "Publish your app" };
Label label8 = new Label { Text = "Paragraphs of text go here." };
SemanticProperties.SetHeadingLevel(label1, SemanticHeadingLevel.Level1);
SemanticProperties.SetHeadingLevel(label3, SemanticHeadingLevel.Level1);
SemanticProperties.SetHeadingLevel(label5, SemanticHeadingLevel.Level1);
SemanticProperties.SetHeadingLevel(label7, SemanticHeadingLevel.Level1);
In addition, the SetValue method can also be used to set the SemanticProperties.HeadingLevel attached
property:
label1.SetValue(SemanticProperties.HeadingLevelProperty, SemanticHeadingLevel.Level1);
Semantic focus
Controls have a SetSemanticFocus extension method, defined in the Microsoft.Maui namespace, which forces
screen reader focus to a specified element. For example, given a Label named label , screen reader focus can
be forced to the element with the following code:
label.SetSemanticFocus();
Semantic screen reader
.NET MAUI Essentials includes a SemanticScreenReader class that enables you to instruct a screen reader to
announce specified text. This can be achieved by calling the SemanticScreenReader.Announce method, passing a
string argument that represents the text:
Automation properties
Automation properties are attached properties that can be added to any element to indicate how the element is
reported to the underlying platform's accessibility framework.
The AutomationProperties class defines the following attached properties:
ExcludedWithChildren , of type bool? , determines if an element and its children should be excluded from the
accessibility tree. For more information, see ExcludedWithChildren.
IsInAccessibleTree , of type bool? , indicates whether the element is available in the accessibility tree. For
more information, see IsInAccessibleTree.
Name , of type string , represents a short description of the element that serves as a speakable identifier for
that element. For more information, see Name.
HelpText , of type string , represents a longer description of the element, which can be thought of as tooltip
text that's associated with the element. For more information, see HelpText.
LabeledBy , of type VisualElement , which enables another element to define accessibility information for the
current element. For more information, see LabeledBy.
These attached properties set platform accessibility values so that a screen reader can speak about the element.
Different screen readers read different accessibility values. Therefore, when using automation properties it's
recommended that thorough accessibility testing is carried out on each platform to ensure an optimal
experience.
IMPORTANT
Automation properties are the Xamarin.Forms approach to providing accessibility values in apps, and have been
superseded by semantic properties. For more information about semantic properties, see Semantic properties.
ExcludedWithChildren
The AutomationProperties.ExcludeWithChildren attached property, of type bool? , determines if an element and
its children should be excluded from the accessibility tree. This enables scenarios such as displaying an
AbsoluteLayout over another layout such as a StackLayout , with the StackLayout being excluded from the
accessibility tree when it's not visible. It can be used from XAML as follows:
<StackLayout AutomationProperties.ExcludedWithChildren="true">
...
</StackLayout>
When this attached property is set, .NET MAUI sets the AutomationProperties.IsInAccessibleTree attached
property to false on the specified element and its children.
IsInAccessibleTree
WARNING
This attached property should typically remain unset. The majority of controls should be present in the accessibility tree,
and the AutomationProperties.ExcludedWithChildren attached property can be set in scenarios where an element
and its children need removing from the accessibility tree.
Name
IMPORTANT
The AutomationProperties.Name attached property has been superseded by the SemanticProperties.Description
attached property.
The AutomationProperties.Name attached property value should be a short, descriptive text string that a screen
reader uses to announce an element. This property should be set for elements that have a meaning that is
important for understanding the content or interacting with the user interface. This can be accomplished in
XAML as follows:
<ActivityIndicator AutomationProperties.IsInAccessibleTree="true"
AutomationProperties.Name="Progress indicator" />
HelpText
IMPORTANT
The AutomationProperties.HelpText attached property has been superseded by the SemanticProperties.Hint
attached property.
The AutomationProperties.HelpText attached property should be set to text that describes the user interface
element, and can be thought of as tooltip text associated with the element. This can be accomplished in XAML as
follows:
On some platforms, for edit controls such as an Entry , the HelpText property can sometimes be omitted and
replaced with placeholder text. For example, "Enter your name here" is a good candidate for the
Entry.Placeholder property that places the text in the control prior to the user's actual input.
LabeledBy
IMPORTANT
The AutomationProperties.LabeledBy attached property has been superseded by bindings. For more information, see
SemanticProperties: Description.
IMPORTANT
The AutomationProperties.LabeledByProperty is not supported on iOS.
Testing accessibility
.NET MAUI apps typically target multiple platforms, which means testing the accessibility features according to
the platform. Follow these links to learn how to test accessibility on each platform:
Test your app's accessibility on Android.
Verifying app accessibility on iOS.
Testing for accessibility on OS X
Accessibility testing on Windows.
The following tools can assist with your accessibility testing:
Accessibility Insights for Android and Windows apps.
Accessibility Scanner for Android apps.
Accessibility Inspector for iOS and macOS apps.
Android Studio Layout Inspector for Android apps.
Xcode View Debugger for iOS and macOS apps.
However, none of these tools can perfectly emulate the screen reader user experience, and the best way to test
and troubleshoot your apps for accessibility will always be manually on physical devices with screen readers.
NOTE
While these steps apply to most devices, you might experience some differences.
A TalkBack tutorial opens automatically the first time you enable TalkBack.
For alternative methods of enabling TalkBack, see Turn Talkback on or off.
Enable VoiceOver
VoiceOver is the primary screen reader used on iOS and macOS. On iOS, VoiceOver can be enabled as follows:
1. Open the Settings app.
2. Select Accessibility > TalkBack .
3. Turn VoiceOver on.
4. Select OK .
A VoiceOver tutorial can be opened by selecting VoiceOver Practice , once VoiceOver is enabled.
For alternative methods of enabling VoiceOver, see Turn on and practice VoiceOver on iPhone and Turn on and
practice VoiceOver on iPad.
On macOS, VoiceOver can be enabled as follows:
1. Open the System Preferences .
2. Select Accessibility > VoiceOver .
3. Select Enable VoiceOver .
4. Select Use VoiceOver .
A VoiceOver tutorial can be opened by selecting Open VoiceOver Training....
For alternative methods of enabling VoiceOver, see Turn VoiceOver on or off on Mac.
Enable Narrator
Narrator is the primary screen reader used on Windows. Narrator can be enabled by pressing the Windows
logo key + Ctrl + Enter together. These keys can be pressed again to stop Narrator.
For more information about Narrator, see Complete guide to Narrator.
Accessibility checklist
Follow these tips to ensure that your .NET MAUI apps are accessible to the widest audience possible:
Ensure your app is perceivable, operable, understandable, and robust for all by following the Web Content
Accessibility Guidelines (WCAG). WCAG is the global accessibility standard and legal benchmark for web and
mobile. For more information, see Web Content Accessibility Guidelines (WCAG) Overview.
Make sure the user interface is self-describing. Test that all the elements of your user interface are screen
reader accessible. Add descriptive text and hints when necessary.
Ensure that images and icons have alternate text descriptions.
Support large fonts and high contrast. Avoid hardcoding control dimensions, and instead prefer layouts that
resize to accommodate larger font sizes. Test color schemes in high-contrast mode to ensure they are
readable.
Design the visual tree with navigation in mind. Use appropriate layout controls so that navigating between
controls using alternate input methods follows the same logical flow as using touch. In addition, exclude
unnecessary elements from screen readers (for example, decorative images or labels for fields that are
already accessible).
Don't rely on audio or color cues alone. Avoid situations where the sole indication of progress, completion, or
some other state is a sound or color change. Either design the user interface to include clear visual cues, with
sound and color for reinforcement only, or add specific accessibility indicators. When choosing colors, try to
avoid a palette that is hard to distinguish for users with color blindness.
Provide captions for video content and a readable script for audio content. It's also helpful to provide controls
that adjust the speed of audio or video content, and ensure that volume and transport controls are easy to
find and use.
Localize your accessibility descriptions when the app supports multiple languages.
Test the accessibility features of your app on each platform it targets. For more information, see Testing
accessibility.
App lifecycle
3/18/2022 • 14 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) apps generally have four execution states: not running, running,
deactivated, and stopped. .NET MAUI raises cross-platform lifecycle events on the Window class when an app
transitions from the not running state to the running state, the running state to the deactivated state, the
deactivated state to the stopped state, the stopped state to the running state, and the stopped state to the not
running state.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The following diagram shows an overview of the .NET MAUI app lifecycle:
In the diagram, the gray oval indicates that the app isn't loaded into memory. The light blue ovals indicate that
the app is in memory. Text on arcs indicates events that are raised by .NET MAUI, that provide notifications to the
running app.
The execution state of an app depends on the app's history. For example, when an app is installed for the first
time, or a device is started, the app can be considered to be not running. When the app is started, the Created
and Activated events are raised and the app is running. If a different app window gains focus, the Deactivated
event is raised and the app is deactivated. If the user switches to a different app or returns to the device's Home
screen, so that the app window is no longer visible, the Deactivated and Stopped events are raised and the app
is stopped. If the user returns to the app, the Resuming event is raised and app is running. Alternatively, an app
might be terminated by a user while it's running. In this situation the app is deactivated then stopped, the
Destroying event is raised, and the app is not running. Similarly, a device might terminate an app while it's
stopped, due to resource restrictions, and the Destroying event is raised and the app is not running.
In addition, .NET MAUI enables apps to be notified when platform lifecycle events are raised. For more
information, see Platform lifecycle events.
Stopped This event is raised when the window Disconnect from any long running
is no longer visible. There's no processes, or cancel any pending
guarantee that an app will resume requests that might consume device
from this state, because it may be resources.
terminated by the operating system.
Resumed This event is raised when an app Subscribe to any required events, and
resumes after being stopped. This refresh any content that's on the
event won't be raised the first time visible page.
your app launches, and can only be
raised if the Stopped event has
previously been raised.
Destroying This event is raised when the native Remove any event subscriptions that
window is being destroyed and you've attached to the native window.
deallocated. The same cross-platform
window might be used against a new
native window when the app is
reopened.
These cross-platform events map to different platform events, and the following table shows this mapping:
EVEN T A N DRO ID IO S W IN DO W S
In addition to these events, the Window class also has the following overridable methods:
OnCreated , which is invoked when the Created event is raised.
OnActivated , which is invoked when the Activated event is raised.
OnDeactivated , which is invoked when the OnDeactivated event is raised.
OnStopped , which is invoked when the Stopped event is raised.
OnResumed , which is invoked when the Resumed event is raised.
OnDestroying , which is invoked when the Destroying event is raised.
To subscribe to the Window lifecycle events, override the CreateWindow method in your App class to create a
Window instance on which you can subscribe to events:
namespace MyMauiApp
{
public partial class App : Application
{
public App()
{
InitializeComponent();
return window;
}
}
}
Alternatively, to consume the lifecycle overrides, create a class that derives from the Window class
namespace MyMauiApp
{
public class MyWindow : Window
{
public MyWindow() : base()
{
}
The Window -derived class can then be consumed by overriding the CreateWindow method in your App class to
return a MyWindow instance.
WARNING
An InvalidOperationException will be thrown if the App.MainPage property is set and the CreateWindow method
creates a Window object using the override that accepts a Page argument.
IMPORTANT
The ConfigureLifecycleEvents method is in the Microsoft.Maui.LifecycleEvents namespace.
Android
The following table lists the .NET MAUI delegates that are invoked in response to Android lifecycle events being
raised:
OnApplicationConfigurationChanged
Android.App.Application Invoked when the device
, configuration changes while
your component is running.
Android.Content.Res.Configuration
OnDestroy Android.App.Activity Invoked when the activity is Always call the super class's
finishing, or because the implementation.
system is temporarily
destroying the activity
instance to save space.
OnPause Android.App.Activity Invoked when an activity is Always call the super class's
going into the background, implementation.
but has not yet been killed.
OnPostCreate Android.App.Activity , Invoked when activity Always call the super class's
Android.OS.Bundle? startup is complete, after implementation. This is a
OnStart and system-only event that
OnRestoreInstanceState generally shouldn't be used
have been called. by apps.
OnPostResume Android.App.Activity Invoked when activity Always call the super class's
resume is complete, after implementation. This is a
OnResume has been called. system-only event that
generally shouldn't be used
by apps.
OnRestart Android.App.Activity Invoked after OnStop Always call the super class's
when the current activity is implementation.
being redisplayed to the
user (the user has
navigated back to it).
DEL EGAT E A RGUM EN T S DESC RIP T IO N C O M M EN T S
OnStart Android.App.Activity Invoked after OnCreate or Always call the super class's
OnRestart when the implementation.
activity has been stopped,
but is now being displayed
to the user.
OnStop Android.App.Activity Invoked when the activity is Always call the super class's
no longer visible to the user. implementation.
IMPORTANT
Each delegate has a corresponding identically named extension method, that can be called to register a handler for the
delegate.
To respond to an Android lifecycle delegate being invoked, call the ConfigureLifecycleEvents method on the
MauiAppBuilder object in the CreateMauiapp method of your MauiProgram class. Then, on the ILifecycleBuilder
object, call the AddAndroid method and specify the Action that registers handlers for the required delegates:
using Microsoft.Maui.LifecycleEvents;
namespace PlatformLifecycleDemo
{
public static class MauiProgram
{
public static MauiApp CreateMauiApp()
{
var builder = MauiApp.CreateBuilder();
builder
.UseMauiApp<App>()
.ConfigureLifecycleEvents(events =>
{
#if ANDROID
events.AddAndroid(android => android
.OnActivityResult((activity, requestCode, resultCode, data) =>
LogEvent("OnActivityResult", requestCode.ToString()))
.OnStart((activity) => LogEvent("OnStart"))
.OnCreate((activity, bundle) => LogEvent("OnCreate"))
.OnBackPressed((activity) => LogEvent("OnBackPressed"))
.OnStop((activity) => LogEvent("OnStop")));
#endif
static void LogEvent(string eventName, string type = null)
{
System.Diagnostics.Debug.WriteLine($"Lifecycle event: {eventName}{(type == null ?
string.Empty : $" ({type})")}");
}
});
return builder.Build();
}
}
}
For more information about the Android app lifecycle, see Understand the Activity Lifecycle on
developer.android.com.
iOS
The following table lists the .NET MAUI delegates that are invoked in response to iOS lifecycle events being
raised:
IMPORTANT
Each delegate has a corresponding identically named extension method, that can be called to register a handler for the
delegate.
To respond to an iOS lifecycle delegate being invoked, call the ConfigureLifecycleEvents method on the
MauiAppBuilder object in the CreateMauiapp method of your MauiProgram class. Then, on the ILifecycleBuilder
object, call the AddiOS method and specify the Action that registers handlers for the required delegates:
using Microsoft.Maui.LifecycleEvents;
namespace PlatformLifecycleDemo
{
public static class MauiProgram
{
public static MauiApp CreateMauiApp()
{
var builder = MauiApp.CreateBuilder();
builder
.UseMauiApp<App>()
.ConfigureLifecycleEvents(events =>
{
#if IOS
events.AddiOS(ios => ios
.OnActivated((app) => LogEvent("OnActivated"))
.OnResignActivation((app) => LogEvent("OnResignActivation"))
.DidEnterBackground((app) => LogEvent("DidEnterBackground"))
.WillTerminate((app) => LogEvent("WillTerminate")));
#endif
static void LogEvent(string eventName, string type = null)
{
System.Diagnostics.Debug.WriteLine($"Lifecycle event: {eventName}{(type == null ?
string.Empty : $" ({type})")}");
}
});
return builder.Build();
}
}
}
For more information about the iOS app lifecycle, see Managing Your App's Life Cycle on developer.apple.com.
Windows
The following table lists the .NET MAUI delegates that are invoked in response to Windows lifecycle events being
raised:
The NativeMessage event is specific to .NET MAUI, and enables native Windows messages to be exposed as a
lifecycle event. The WindowsNativeMessageEventArgs object that accompanies the NativeMessage event includes a
MessageId property, of type uint . The value of this property can be examined to determine which message has
been passed to your app window. For more information about windows messages, see Windows Messages (Get
Started with Win32 and C++). For a list of window message constants, see Window notifications.
IMPORTANT
Each delegate has a corresponding identically named extension method, that can be called to register a handler for the
delegate.
To respond to a Windows lifecycle delegate being invoked, call the ConfigureLifecycleEvents method on the
MauiAppBuilder object in the CreateMauiapp method of your MauiProgram class. Then, on the ILifecycleBuilder
object, call the AddWindows method and specify the Action that registers handlers for the required delegates:
using Microsoft.Maui.LifecycleEvents;
namespace PlatformLifecycleDemo
{
public static class MauiProgram
{
public static MauiApp CreateMauiApp()
{
var builder = MauiApp.CreateBuilder();
builder
.UseMauiApp<App>()
.ConfigureLifecycleEvents(events =>
{
#if WINDOWS
events.AddWindows(windows => windows
.OnActivated((window, args) => LogEvent("OnActivated"))
.OnClosed((window, args) => LogEvent("OnClosed"))
.OnLaunched((window, args) => LogEvent("OnLaunched"))
.OnLaunching((window, args) => LogEvent("OnLaunching"))
.OnVisibilityChanged((window, args) => LogEvent("OnVisibilityChanged"))
.OnNativeMessage((window, args) =>
{
if (args.MessageId == Convert.ToUInt32("0x02E0"))
{
// DPI has changed
}
}));
#endif
static void LogEvent(string eventName, string type = null)
{
System.Diagnostics.Debug.WriteLine($"Lifecycle event: {eventName}{(type == null ?
string.Empty : $" ({type})")}");
}
});
return builder.Build();
}
}
}
namespace PlatformLifecycleDemo
{
public static class MauiProgram
{
public static MauiApp CreateMauiApp()
{
var builder = MauiApp.CreateBuilder();
builder
.UseMauiApp<App>()
.ConfigureLifecycleEvents(events =>
{
#if WINDOWS
events.AddWindows(windows => windows
.OnClosed((window, args) =>
{
IWindow appWindow = window.GetWindow();
}));
#endif
});
return builder.Build();
}
}
}
Then, apps that want to receive notification of the platform lifecycle event should modify the CreateMauiApp
method of their MauiProgram class to call the ConfigureLifecycleEvents method on the MauiAppBuilder object.
Then, on the ILifecycleBuilder object, call the AddEvent method and specify the platform event name and the
Action that will be invoked when the platform event is raised.
Example
The WinUI 3 Window.SizeChanged event occurs when the native app window has first rendered, or has changed
its rendering size. .NET MAUI doesn't expose this platform event as a lifecycle event. However, apps can receive
notification when this platform event is raised by using the following approach:
Register an event handler for the Window.SizeChanged platform lifecycle event:
using Microsoft.Maui.LifecycleEvents;
...
return builder.Build();
}
In the event handler for the platform lifecycle event, retrieve the ILifecycleEventService instance and call
its InvokeEvents method, specifying the platform event name as its argument:
using Microsoft.Maui.LifecycleEvents;
...
#if WINDOWS
static void OnSizeChanged(object sender, Microsoft.UI.Xaml.WindowSizeChangedEventArgs args)
{
ILifecycleEventService service =
MauiWinUIApplication.Current.Services.GetRequiredService<ILifecycleEventService>();
service.InvokeEvents(nameof(Microsoft.UI.Xaml.Window.SizeChanged));
}
#endif
The MauiWinUIApplicationtype on Windows can be used to access the native app instance via its
Current property. The MauiApplication type on Android can be used to access the native app instance.
Similarly, the MauiUIApplicationDelegate type on iOS can be used to access the native app instance.
WARNING
Invoking an unregistered event, with the InvokeEvents method, doesn't throw an exception.
In the CreateMauiApp method of your MauiProgram class, call the ConfigureLifecycleEvents method on
the MauiAppBuilder object. Then, on the ILifecycleBuilder object, call the AddEvent method and specify
the platform event name and the Action that will be invoked when the platform event is raised:
using Microsoft.Maui.LifecycleEvents;
namespace PlatformLifecycleDemo
{
public static class MauiProgram
{
public static MauiApp CreateMauiApp()
{
var builder = MauiApp.CreateBuilder();
builder
.UseMauiApp<App>()
.ConfigureLifecycleEvents(events =>
{
#if WINDOWS
events.AddWindows(windows => windows
.OnWindowCreated(window =>
{
window.SizeChanged += OnSizeChanged;
}));
events.AddEvent(nameof(Microsoft.UI.Xaml.Window.SizeChanged), () =>
LogEvent("Window SizeChanged"));
#endif
static void LogEvent(string eventName, string type = null)
{
System.Diagnostics.Debug.WriteLine($"Lifecycle event: {eventName}{(type ==
null ? string.Empty : $" ({type})")}");
}
});
return builder.Build();
}
}
}
The overall effect is that when a user changes the app window size on Windows, the action specified in the
AddEvent method is executed.
NOTE
The AddEvent method also has an overload that enables a delegate to be specified.
Configure fonts, services, and handlers at startup
3/18/2022 • 3 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) apps are bootstrapped using the .NET Generic Host. This enables apps
to be initialized from a single location, and provides the ability to configure fonts, services, and third-party
libraries.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Each platform entry point calls a CreateMauiApp method on the static MauiProgram class that creates and returns
a MauiApp , the entry point for your app.
The MauiProgram class must at a minimum provide an app to run:
using Microsoft.Maui.Hosting;
using Microsoft.Maui.Controls.Hosting;
namespace MyMauiApp
{
public static class MauiProgram
{
public static MauiApp CreateMauiApp()
{
var builder = MauiApp.CreateBuilder();
builder
.UseMauiApp<App>();
return builder.Build();
}
}
}
namespace MyMauiApp
{
public class App : Application
{
public App()
{
InitializeComponent();
In the example above, MainPage is a ContentPage that defines the UI for the initial page of the app. The
Application creates a Window within which to run the application and display views. You can customize this
behavior by overriding the CreateWindow method.
Register fonts
Fonts can be added to your app and referenced by filename or alias. This is accomplished by invoking the
ConfigureFonts method on the MauiAppBuilder object. Then, on the IFontCollection object, call the AddFont
method to add the required font:
using Microsoft.Maui.Hosting;
using Microsoft.Maui.Controls.Hosting;
namespace MyMauiApp
{
public static class MauiProgram
{
public static MauiApp CreateMauiApp()
{
var builder = MauiApp.CreateBuilder();
builder
.UseMauiApp<App>()
.ConfigureFonts(fonts =>
{
fonts.AddFont("OpenSans-Regular.ttf", "OpenSansRegular");
});
return builder.Build();
}
}
}
In the example above, the first argument to the AddFont method is the font filename, while the second
argument represents an optional alias by which the font can be referenced when consuming it.
Any custom fonts consumed by an app must be included in your .csproj file. This can be accomplished by
referencing their filenames, or by using a wildcard:
<ItemGroup>
<MauiFont Include="Resources\Fonts\*" />
</ItemGroup>
NOTE
Fonts added to the project through the Solution Explorer in Visual Studio will automatically be included in the .csproj file.
The font can then be consumed by referencing its name, without the file extension:
using Microsoft.Maui.Hosting;
using Microsoft.Maui.Controls.Hosting;
namespace MyMauiApp
{
public static class MauiProgram
{
public static MauiApp CreateMauiApp()
{
var builder = MauiApp.CreateBuilder();
builder
.UseMauiApp<App>()
.ConfigureMauiHandlers(handlers =>
{
handlers.AddHandler(typeof(MyEntry), typeof(MyEntryHandler));
});
return builder.Build();
}
}
}
In this example, the MyEntryHandler handler is registered against the MyEntry control. Therefore, any instances
of the MyEntry control will be handled by the MyEntryHandler .
Register renderers
To use controls backed by .NET MAUI handlers, with specific controls backed by Xamarin.Forms renderers, call
the ConfigureMauiHandlers method on the MauiAppBuilder object. Then, on the IMauiHandlersCollection object,
call the AddCompatibilityRenderer method to add the required renderer:
using Microsoft.Maui.Hosting;
using Microsoft.Maui.Controls.Hosting;
using Microsoft.Maui.Controls.Compatibility;
namespace MyMauiApp
{
public static class MauiProgram
{
public static MauiApp CreateMauiApp()
{
var builder = MauiApp.CreateBuilder();
builder
.UseMauiApp<App>()
#if ANDROID
.ConfigureMauiHandlers(handlers =>
{
handlers.AddCompatibilityRenderer(typeof(Microsoft.Maui.Controls.BoxView),
typeof(Microsoft.Maui.Controls.Compatibility.Platform.Android.BoxRenderer));
handlers.AddCompatibilityRenderer(typeof(Microsoft.Maui.Controls.Frame),
typeof(Microsoft.Maui.Controls.Compatibility.Platform.Android.FastRenderers.FrameRenderer));
});
#elif IOS
.ConfigureMauiHandlers(handlers =>
{
handlers.AddCompatibilityRenderer(typeof(Microsoft.Maui.Controls.BoxView),
typeof(Microsoft.Maui.Controls.Compatibility.Platform.iOS.BoxRenderer));
handlers.AddCompatibilityRenderer(typeof(Microsoft.Maui.Controls.Frame),
typeof(Microsoft.Maui.Controls.Compatibility.Platform.iOS.FrameRenderer));
});
#endif
return builder.Build();
}
}
}
In this example, all controls in the app will be backed by handlers, aside from the BoxView and Frame controls
that will be backed by Xamarin.Forms renderers.
Behaviors
3/18/2022 • 10 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) behaviors let you add functionality to user interface controls without
having to subclass them. Instead, the functionality is implemented in a behavior class and attached to the control
as if it was part of the control itself.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Behaviors enable you to implement code that you would normally have to write as code-behind, because it
directly interacts with the API of the control in such a way that it can be concisely attached to the control and
packaged for reuse across more than one application. They can be used to provide a full range of functionality to
controls, such as:
Adding an email validator to an Entry .
Creating a rating control using a tap gesture recognizer.
Controlling an animation.
.NET MAUI supports two different types of behaviors:
Attached behaviors are static classes with one or more attached properties. For more information about
attached behaviors, see Attached behaviors.
.NET MAUI behaviors are classes that derive from the Behavior or Behavior<T> class, where T is the type of
the control to which the behavior should apply. For more information, see .NET MAUI Behaviors.
Attached behaviors
Attached behaviors are static classes with one or more attached properties. An attached property is a special
type of bindable property. They are defined in one class but attached to other objects, and they are recognizable
in XAML as attributes that contain a class and a property name separated by a period. For more information
about attached properties, see Attached properties.
An attached property can define a propertyChanged delegate that will be executed when the value of the
property changes, such as when the property is set on a control. When the propertyChanged delegate executes,
it's passed a reference to the control on which it is being attached, and parameters that contain the old and new
values for the property. This delegate can be used to add new functionality to the control that the property is
attached to by manipulating the reference that is passed in, as follows:
1. The propertyChanged delegate casts the control reference, which is received as a BindableObject , to the
control type that the behavior is designed to enhance.
2. The propertyChanged delegate modifies properties of the control, calls methods of the control, or registers
event handlers for events exposed by the control, to implement the core behavior functionality.
WARNING
Attached behaviors are defined in a static class, with static properties and methods. This makes it difficult to create
attached behaviors that have state.
<ContentPage ...
xmlns:local="clr-namespace:BehaviorsDemos">
<Entry Placeholder="Enter a System.Double" local:AttachedNumericValidationBehavior.AttachBehavior="true"
/>
</ContentPage>
The following screenshot shows the attached behavior responding to invalid input:
NOTE
Attached behaviors are written for a specific control type (or a superclass that can apply to many controls), and they
should only be added to a compatible control.
At runtime, the OnAttachBehaviorChanged method will be executed when the value of the AttachBehavior
attached property is set to false . The OnAttachBehaviorChanged method will then de-register the event handler
for the TextChanged event, ensuring that the behavior isn't executed as you interact with the control.
// Behavior implementation
}
The OnAttachedTo method is called immediately after the behavior is attached to a control. This method receives
a reference to the control to which it is attached, and can be used to register event handlers or perform other
setup that's required to support the behavior functionality. For example, you could subscribe to an event on a
control. The behavior functionality would then be implemented in the event handler for the event.
The OnDetachingFrom method is called when the behavior is removed from the control. This method receives a
reference to the control to which it is attached, and is used to perform any required cleanup. For example, you
could unsubscribe from an event on a control to prevent memory leaks.
The behavior can then be consumed by attaching it to the Behaviors collection of the control.
Create a .NET MAUI Behavior
A .NET MAUI behavior can be implemented by creating a class that derives from the Behavior or Behavior<T>
class, and overriding the OnAttachedTo and OnDetachingFrom methods.
The following example shows the NumericValidationBehavior class, which highlights the value entered by the
user into an Entry control in red if it's not a double :
In this example, the NumericValidationBehavior class derives from the Behavior<T> class, where T is an Entry .
The OnAttachedTo method registers an event handler for the TextChanged event, with the OnDetachingFrom
method de-registering the TextChanged event to prevent memory leaks. The core functionality of the behavior is
provided by the OnEntryTextChanged method, which parses the value entered in the Entry and sets the
TextColor property to red if the value isn't a double .
IMPORTANT
.NET MAUI does not set the BindingContext of a behavior, because behaviors can be shared and applied to multiple
controls through styles.
The following screenshot shows the .NET MAUI behavior responding to invalid input:
WARNING
.NET MAUI behaviors are written for a specific control type (or a superclass that can apply to many controls), and they
should only be added to a compatible control. Attempting to attach a .NET MAUI behavior to an incompatible control will
result in an exception being thrown.
The following example shows the NumericValidationStyleBehavior class, which has an attached property that
controls adding and removing the behavior:
public class NumericValidationStyleBehavior : Behavior<Entry>
{
public static readonly BindableProperty AttachBehaviorProperty =
BindableProperty.CreateAttached("AttachBehavior", typeof(bool),
typeof(NumericValidationStyleBehavior), false, propertyChanged: OnAttachBehaviorChanged);
In this example, the NumericValidationStyleBehavior class contains an attached property named AttachBehavior
with a static getter and setter, which controls the addition or removal of the behavior to the control to which it
will be attached. This attached property registers the OnAttachBehaviorChanged method that will be executed
when the value of the property changes. This method adds or removes the behavior to the control, based on the
value of the AttachBehavior attached property.
The following code example shows an explicit style for the NumericValidationStyleBehavior that uses the
AttachBehavior attached property, and which can be applied to Entry controls:
The Style can be applied to an Entry by setting its Style property to the style using the StaticResource
markup extension:
entry.Behaviors.Clear();
NOTE
.NET MAUI behaviors are not implicitly removed from controls when pages are popped from the navigation stack.
Instead, they must be explicitly removed prior to pages going out of scope.
Data binding
3/18/2022 • 2 minutes to read • Edit Online
A .NET Multi-platform App UI (.NET MAUI) app consists of one or more pages, each of which typically contains
multiple user-interface objects called views. One of the primary tasks of the app is to keep these views
synchronized, and to keep track of the various values or selections that they represent. Often the views
represent values from an underlying data source, and users manipulate these views to change that data. When
the view changes, the underlying data must reflect that change, and similarly, when the underlying data changes,
that change must be reflected in the view.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
To handle this successfully, the app must be notified of changes in these views or the underlying data. The
common solution is to define events that signal when a change occurs. An event handler can then be installed
that is notified of these changes. It responds by transferring data from one object to another. However, when
there are many views, there must also be many event handlers, which results in a lot of boilerplate code.
Data binding automates this task, and renders the event handlers unnecessary. Data bindings can be
implemented either in XAML or code, but they are much more common in XAML where they help to reduce the
size of the code-behind file. By replacing procedural code in event handlers with declarative code or markup, the
app is simplified and clarified.
Data binding is therefore the technique of linking properties of two objects so that changes in one property are
automatically reflected in the other property. One of the two objects involved in a data binding is almost always
an element that derives from View and forms part of the visual interface of a page. The other object is either:
Another View derivative, usually on the same page.
An object in a code file.
NOTE
Data bindings between two View derivatives are often shown for purposes of clarity and simplicity. However, the same
principles can be applied to data bindings between a View and other objects. When an application is built using the
Model-View-ViewModel (MVVM) architecture, the class with underlying data is often called a viewmodel.
Basic bindings
3/18/2022 • 7 minutes to read • Edit Online
A .NET Multi-platform App UI (.NET MAUI) data binding links a pair of properties between two objects, at least
one of which is usually a user-interface object. These two objects are called the target and the source:
The target is the object (and property) on which the data binding is set.
The source is the object (and property) referenced by the data binding.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
In the simplest case, data flows from the source to the target, which means that the value of the target property
is set from the value of the source property. However, in some cases, data can alternatively flow from the target
to the source, or in both directions.
IMPORTANT
The target is always the object on which the data binding is set even if it's providing data rather than receiving data.
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="DataBindingDemos.BasicCodeBindingPage"
Title="Basic Code Binding">
<StackLayout Padding="10, 0">
<Label x:Name="label"
Text="TEXT"
FontSize="48"
HorizontalOptions="Center"
VerticalOptions="Center" />
<Slider x:Name="slider"
Maximum="360"
VerticalOptions="Center" />
</StackLayout>
</ContentPage>
Without data bindings, you would set the ValueChanged event of the Slider to an event handler that accesses
the Value property of the Slider and sets that value to the Rotation property of the Label . The data binding
automates this task, and so the event handler and the code within it are no longer necessary.
You can set a binding on an instance of any class that derives from BindableObject , which includes Element ,
VisualElement , View , and View derivatives. The binding is always set on the target object. The binding
references the source object. To set the data binding, use the following two members of the target class:
The BindingContext property specifies the source object.
The SetBinding method specifies the target property and source property.
In this example, the Label is the binding target, and the Slider is the binding source. Changes in the Slider
source affect the rotation of the Label target. Data flows from the source to the target.
The SetBindingmethod defined by BindableObject has an argument of type BindingBase from which the
Binding class derives, but there are other SetBinding methods defined by the BindableObjectExtensions class.
The code-behind for the XAML uses a simpler SetBinding extension method from the
BindableObjectExtensions class:
label.BindingContext = slider;
label.SetBinding(Label.RotationProperty, "Value");
}
}
The Label object is the binding target so that's the object on which this property is set and on which the
method is called. The BindingContext property indicates the binding source, which is the Slider . The
SetBinding method is called on the binding target but specifies both the target property and the source
property. The target property is specified as a BindableProperty object: Label.RotationProperty . The source
property is specified as a string and indicates the Value property of Slider .
IMPORTANT
The target property must be backed by a bindable property. Therefore, the target object must be an instance of a class
that derives from BindableObject .
The source property is specified as a string. Internally, reflection is used to access the actual property. In this
particular case, however, the Value property is also backed by a bindable property.
As you manipulate the Slider , the Label rotates accordingly:
<Slider x:Name="slider"
Maximum="360"
VerticalOptions="Center" />
</StackLayout>
</ContentPage>
Just as in code, the data binding is set on the target object, which is the Label . Two XAML markup extensions
are used to define the data binding:
The x:Reference markup extension is required to reference the source object, which is the Slider named
slider .
The Binding markup extension links the Rotation property of the Label to the Value property of the
Slider .
NOTE
The source property is specified with the Path property of the Binding markup extension, which corresponds with the
Path property of the Binding class.
XAML markup extensions such as x:Reference and Binding can have content property attributes defined,
which for XAML markup extensions means that the property name doesn't need to appear. The Name property
is the content property of x:Reference , and the Path property is the content property of Binding , which
means that they can be eliminated from the expressions:
<Label Text="TEXT"
FontSize="80"
HorizontalOptions="Center"
VerticalOptions="Center"
BindingContext="{x:Reference slider}"
Rotation="{Binding Value}" />
<Slider x:Name="slider"
Minimum="-2"
Maximum="2"
VerticalOptions="CenterAndExpand" />
</StackLayout>
</ContentPage>
In this example, the Slider is defined to control the Scale property of the Label . For that reason, the Slider
is set for a range of -2 to 2.
The code-behind file sets the binding with the SetBinding method, with the second argument being a
constructor for the Binding class:
The Binding constructor has 6 parameters, so the source parameter is specified with a named argument. The
argument is the slider object.
NOTE
The VisualElement class also defines ScaleX and ScaleY properties, which can scale the VisualElement differently
in the horizontal and vertical directions.
<Slider x:Name="slider"
Minimum="-2"
Maximum="2"
VerticalOptions="Center" />
</StackLayout>
</ContentPage>
In this example, the Binding markup extension has two properties set, Source and Path , separated by a
comma. The Source property is set to an embedded x:Reference markup extension that otherwise has the
same syntax as setting the BindingContext .
The content property of the Binding markup extension is Path , but the Path= part of the markup extension
can only be eliminated if it is the first property in the expression. To eliminate the Path= part, you need to swap
the two properties:
Although XAML markup extensions are usually delimited by curly braces, they can also be expressed as object
elements:
<Label Text="TEXT"
FontSize="40"
HorizontalOptions="Center"
VerticalOptions="Center">
<Label.Scale>
<Binding Source="{x:Reference slider}"
Path="Value" />
</Label.Scale>
</Label>
In this example, the Source and Path properties are regular XAML attributes. The values appear within
quotation marks and the attributes are not separated by a comma. The x:Reference markup extension can also
become an object element:
<Label Text="TEXT"
FontSize="40"
HorizontalOptions="Center"
VerticalOptions="Center">
<Label.Scale>
<Binding Path="Value">
<Binding.Source>
<x:Reference Name="slider" />
</Binding.Source>
</Binding>
</Label.Scale>
</Label>
This syntax isn't common, but sometimes it's necessary when complex objects are involved.
The examples shown so far set the BindingContext property and the Source property of Binding to an
x:Reference markup extension to reference another view on the page. These two properties are of type Object
, and they can be set to any object that includes properties that are suitable for binding sources. You can also set
the BindingContext or Source property to an x:Static markup extension to reference the value of a static
property or field, or a StaticResource markup extension to reference an object stored in a resource dictionary,
or directly to an object, which is often an instance of a viewmodel.
NOTE
The BindingContext property can also be set to a Binding object so that the Source and Path properties of
Binding define the binding context.
IMPORTANT
The BindingContext property value is inherited through the visual tree.
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="DataBindingDemos.BindingContextInheritancePage"
Title="BindingContext Inheritance">
<StackLayout Padding="10">
<StackLayout VerticalOptions="Fill"
BindingContext="{x:Reference slider}">
<Label Text="TEXT"
FontSize="80"
HorizontalOptions="Center"
VerticalOptions="End"
Rotation="{Binding Value}" />
<BoxView Color="#800000FF"
WidthRequest="180"
HeightRequest="40"
HorizontalOptions="Center"
VerticalOptions="Start"
Rotation="{Binding Value}" />
</StackLayout>
<Slider x:Name="slider"
Maximum="360" />
</StackLayout>
</ContentPage>
In this example, the BindingContext property of the StackLayout is set to the slider object. This binding
context is inherited by both the Label and the BoxView , both of which have their Rotation properties set to
the Value property of the Slider :
Binding mode
3/18/2022 • 7 minutes to read • Edit Online
Every .NET Multi-platform App UI (.NET MAUI) bindable property has a default binding mode that is set when
the bindable property is created, and which is available from the DefaultBindingMode property of the
BindableProperty object. This default binding mode indicates the mode in effect when that property is a data-
binding target. The default binding mode for most properties such as Rotation , Scale , and Opacity is OneWay
. When these properties are data-binding targets, then the target property is set from the source.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="DataBindingDemos.ReverseBindingPage"
Title="Reverse Binding">
<StackLayout Padding="10, 0">
<Label x:Name="label"
Text="TEXT"
FontSize="80"
HorizontalOptions="Center"
VerticalOptions="Center" />
<Slider x:Name="slider"
VerticalOptions="Center"
Value="{Binding Source={x:Reference label},
Path=Opacity}" />
</StackLayout>
</ContentPage>
In this example, the Label is the data-binding source, and the Slider is the target. The binding references the
Opacity property of the Label , which has a default value of 1. Therefore, the Slider is initialized to the value
1 from the initial Opacity value of Label . This is shown in the following screenshot:
In addition, the Slider continues to work. This is because the default binding mode for the Value property of
Slider is TwoWay . This means that when the Value property is a data-binding target, then the target is set
from the source but the source is also set from the target. This allows the Slider to be set from the initial
Opacity value.
NOTE
Bindable properties don't signal a property change unless the property actually changes. This prevents an infinite loop.
If the default binding mode on the target property is not suitable for a particular data binding, it's possible to
override it by setting the Mode property of Binding (or the Mode property of the Binding markup extension)
to one of the members of the BindingMode enumeration:
Default
TwoWay— data goes both ways between source and target
OneWay — data goes from source to target
OneWayToSource — data goes from target to source
OneTime — data goes from source to target, but only when the BindingContext changes (new with .NET
MAUI 3.0)
Two-way bindings
Most bindable properties have a default binding mode of OneWay but some properties have a default binding
mode of TwoWay , including the following:
Date property of DatePicker
Text property of Editor , Entry , SearchBar , and EntryCell
IsRefreshing property of ListView
SelectedItem property of MultiPage
SelectedIndex and SelectedItem properties of Picker
Value property of Slider and Stepper
IsToggled property of Switch
On property of SwitchCell
Time property of TimePicker
These properties are defined as TwoWay because when data bindings are used with the Model-View-ViewModel
(MVVM) pattern, the viewmodel class is the data-binding source, and the view, which consists of views such as
Slider , are data-binding targets. MVVM bindings resemble the example above, because it's likely that you want
each view on the page to be initialized with the value of the corresponding property in the viewmodel, but
changes in the view should also affect the viewmodel property.
One-way-to-source bindings
Read-only bindable properties have a default binding mode of OneWayToSource . For example, the SelectedItem
property of ListView has a default binding mode of OneWayToSource . This is because a binding on the
SelectedItem property should result in setting the binding source.
One-time bindings
Target properties with a binding mode of OneTime are updated only when the binding context changes. For
bindings on these target properties, this simplifies the binding infrastructure because it is not necessary to
monitor changes in the source properties.
Several properties have a default binding mode of OneTime , including the IsTextPredictionEnabled property of
Entry .
Viewmodels and property-change notifications
When using a viewmodel in a data-binding, the viewmodel is the data-binding source. The viewmodel doesn't
define bindable properties, but it does implement a notification mechanism that allows the binding
infrastructure to be notified when the value of a property changes. This notification mechanism is the
INotifyPropertyChanged interface, which defines a single event named PropertyChanged . A class that implements
this interface typically fires the event when one of its public properties changes value. The event does not need
to be raised if the property never changes. The INotifyPropertyChanged interface is also implemented by
BindableObject and a PropertyChanged event is raised whenever a bindable property changes value.
In the following example, data bindings allow you to select a color using three Slider elements for the hue,
saturation, and luminosity:
Name = NamedColor.GetNearestColorName(color);
}
}
}
In this example, the HslColorViewModel class defines Hue , Saturation , Luminosity , Color , and Name
properties. When any one of the three color components changes value, the Color property is recalculated, and
PropertyChanged events are raised for all four properties. When the Color property changes, the static
GetNearestColorName method in the NamedColor class obtains the closest named color and sets the Name
property.
When a viewmodel is set as a binding source, the binding infrastructure attaches a handler to the
PropertyChanged event. In this way, the binding can be notified of changes to properties, and can then set the
target properties from the changed values. However, when a target property (or the Binding definition on a
target property) has a BindingMode of OneTime , it is not necessary for the binding infrastructure to attach a
handler on the PropertyChanged event. The target property is updated only when the BindingContext changes
and not when the source property itself changes.
The following XAML consumes the HslColorViewModel :
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:DataBindingDemos"
x:Class="DataBindingDemos.SimpleColorSelectorPage">
<ContentPage.BindingContext>
<local:HslColorViewModel Color="MediumTurquoise" />
</ContentPage.BindingContext>
<ContentPage.Resources>
<Style TargetType="Slider">
<Setter Property="VerticalOptions" Value="CenterAndExpand" />
</Style>
</ContentPage.Resources>
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="*" />
<RowDefinition Height="*" />
</Grid.RowDefinitions>
<BoxView Color="{Binding Color}"
Grid.Row="0" />
<StackLayout Grid.Row="1"
Margin="10, 0">
<Label Text="{Binding Name}"
HorizontalTextAlignment="Center" />
<Slider Value="{Binding Hue}" />
<Slider Value="{Binding Saturation}" />
<Slider Value="{Binding Luminosity}" />
</StackLayout>
</Grid>
</ContentPage>
In this example, the HslColorViewModel is instantiated, and Color property set, and set as the page's
BindingContext . The BoxView , Label , and three Slider views inherit the binding context from the
ContentPage . These views are all binding targets that reference source properties in the viewmodel. For the
Color property of the BoxView , and the Text property of the Label , the data bindings are OneWay - the
properties in the view are set from the properties in the viewmodel. The Value property of the Slider ,
however, uses a TwoWay binding mode. This enables each Slider to be set from the viewmodel, and also for the
viewmodel to be set from each Slider .
When the example is first run, the BoxView , Label , and three Slider elements are all set from the viewmodel
based on the initial Color property set when the viewmodel was instantiated:
As you manipulate the sliders, the BoxView and Label are updated accordingly.
However, setting the Mode property doesn't always produce the expected result. For example, in the following
example setting the Mode property to TwoWay doesn't work as you might expect:
<Label Text="TEXT"
FontSize="40"
HorizontalOptions="Center"
VerticalOptions="CenterAndExpand"
Scale="{Binding Source={x:Reference slider},
Path=Value,
Mode=TwoWay}" />
In this example, it might be expected that the Slider would be initialized to the initial value of the Scale
property, which is 1, but that doesn't happen. When a TwoWay binding is initialized, the target is set from the
source first, which means that the Scale property is set to the Slider default value of 0. When the TwoWay
binding is set on the Slider , then the Slider is initially set from the source.
Alternatively, you can set the binding mode to OneWayToSource :
<Label Text="TEXT"
FontSize="40"
HorizontalOptions="Center"
VerticalOptions="CenterAndExpand"
Scale="{Binding Source={x:Reference slider},
Path=Value,
Mode=OneWayToSource}" />
Now the Slider is initialized to 1 (the default value of Scale ) but manipulating the Slider doesn't affect the
Scale property.
NOTE
The VisualElement class also defines ScaleX and ScaleY properties, which can scale the VisualElement differently
in the horizontal and vertical directions.
A very useful application of overriding the default binding mode with a TwoWay binding mode involves the
SelectedItem property of ListView . The default binding mode is OneWayToSource . When a data binding is set
on the SelectedItem property to reference a source property in a viewmodel, then that source property is set
from the ListView selection. However, in some circumstances, you might also want the ListView to be
initialized from the viewmodel.
String formatting
3/18/2022 • 3 minutes to read • Edit Online
In a .NET Multi-platform App UI (.NET MAUI) app, it's sometimes convenient to use data bindings to display the
string representation of an object or value. For example, you might want to use a Label to display the current
value of a Slider . In this data binding, the Slider is the source, and the target is the Text property of the
Label .
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
String formatting in code is typically accomplished with the static String.Format method. The formatting string
includes formatting codes specific to various types of objects, and you can include other text along with the
values being formatted. For more information, see Formatting Types in .NET for more information on string
formatting.
String formatting can also be accomplished with data bindings by setting the StringFormat property of
Binding (or the StringFormat property of the Binding markup extension) to a standard .NET formatting string
with a placeholder:
In XAML the formatting string is delimited by single-quote characters to help the XAML parser avoid treating the
curly braces as another XAML markup extension. In this example, the formatting specification of F2 causes the
value to be displayed with two decimal places.
NOTE
Using the StringFormat property only makes sense when the target property is of type string , and the binding
mode is OneWay or TwoWay . For two-way bindings, the StringFormat is only applicable for values passing from the
source to the target.
<ContentPage.Resources>
<Style TargetType="Label">
<Setter Property="HorizontalTextAlignment" Value="Center" />
</Style>
<Style TargetType="BoxView">
<Setter Property="Color" Value="Blue" />
<Setter Property="HeightRequest" Value="2" />
<Setter Property="Margin" Value="0, 5" />
</Style>
</ContentPage.Resources>
<StackLayout Margin="10">
<Slider x:Name="slider" />
<Label Text="{Binding Source={x:Reference slider},
Path=Value,
StringFormat='The slider value is {0:F2}'}" />
<BoxView />
<TimePicker x:Name="timePicker" />
<Label Text="{Binding Source={x:Reference timePicker},
Path=Time,
StringFormat='The TimeSpan is {0:c}'}" />
<BoxView />
<Entry x:Name="entry" />
<Label Text="{Binding Source={x:Reference entry},
Path=Text,
StringFormat='The Entry text is "{0}"'}" />
<BoxView />
<StackLayout BindingContext="{x:Static sys:DateTime.Now}">
<Label Text="{Binding}" />
<Label Text="{Binding Path=Ticks,
StringFormat='{0:N0} ticks since 1/1/1'}" />
<Label Text="{Binding StringFormat='The {{0:MMMM}} specifier produces {0:MMMM}'}" />
<Label Text="{Binding StringFormat='The long date is {0:D}'}" />
</StackLayout>
<BoxView />
<StackLayout BindingContext="{x:Static sys:Math.PI}">
<Label Text="{Binding}" />
<Label Text="{Binding StringFormat='PI to 4 decimal points = {0:F4}'}" />
<Label Text="{Binding StringFormat='PI in scientific notation = {0:E7}'}" />
</StackLayout>
</StackLayout>
</ContentPage>
In this example, the bindings on the Slider and TimePicker show the use of format specifications particular to
double and TimeSpan data types. The StringFormat that displays the text from the Entry view demonstrates
how to specify double quotation marks in the formatting string with the use of the " HTML entity.
The next section in the XAML file is a StackLayout with a BindingContext set to an x:Static markup extension
that references the static DateTime.Now property. The first binding has no properties:
This simply displays the DateTime value of the BindingContext with default formatting. The second binding
displays the Ticks property of DateTime , while the other two bindings display the DateTime itself with specific
formatting.
NOTE
If you need to display left or right curly braces in your formatting string, use a pair of them. For example,
StringFormat='{{0:MMMM}}' .
The last section sets the BindingContext to the value of Math.PI and displays it with default formatting and two
different types of numeric formatting:
<ContentPage.Resources>
<ResourceDictionary>
<Style TargetType="Slider">
<Setter Property="VerticalOptions" Value="Center" />
</Style>
<Style TargetType="Label">
<Setter Property="HorizontalTextAlignment" Value="Center" />
</Style>
</ResourceDictionary>
</ContentPage.Resources>
<StackLayout Margin="20">
<BoxView Color="{Binding Color}"
HeightRequest="100"
WidthRequest="100"
HorizontalOptions="Center" />
<StackLayout Margin="10, 0">
<Label Text="{Binding Name}" />
<Slider Value="{Binding Hue}" />
<Label Text="{Binding Hue, StringFormat='Hue = {0:F2}'}" />
<Slider Value="{Binding Saturation}" />
<Label Text="{Binding Saturation, StringFormat='Saturation = {0:F2}'}" />
<Slider Value="{Binding Luminosity}" />
<Label Text="{Binding Luminosity, StringFormat='Luminosity = {0:F2}'}" />
</StackLayout>
</StackLayout>
</ContentPage>
In this example, there are three pairs of Slider and Label elements that are bound to the same source
property in the HslColorViewModel object. Each Label that accompanies a Slider has a StringFormat property
to display each Slider value:
Binding path
3/18/2022 • 3 minutes to read • Edit Online
In .NET Multi-platform App UI (.NET MAUI), the Path property of the Binding class (or the Path property of
the Binding markup extension) can be set to a single property, to a sub-property (a property of a property), or
to a member of a collection.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
<TimePicker x:Name="timePicker">
The Time property of TimePicker is of type TimeSpan , and it has a TotalSeconds property. A data binding can
be created that references the TotalSeconds property of that TimeSpan value:
The Time and TotalSeconds properties are simply connected with a period.
NOTE
The items in the Path string always refer to properties and not to the types of these properties.
In the second Label , the binding source is the page itself. The Content property is of type StackLayout , which
has a Children property of type IList<View> , which has a Count property indicating the number of children.
The source is set to the static CultureInfo.CurrentCulture property, which is an object of type CultureInfo . That
class defines a property named DateTimeFormat of type DateTimeFormatInfo that contains a DayNames collection.
The index selects the fourth item.
The fourth Label does something similar but for the culture associated with France. The Source property of
the binding is set to CultureInfo object with a constructor:
<Label>
<Label.Text>
<Binding Path="DateTimeFormat.DayNames[3]"
StringFormat="The middle day of the week in France is {0}">
<Binding.Source>
<globe:CultureInfo>
<x:Arguments>
<x:String>fr-FR</x:String>
</x:Arguments>
</globe:CultureInfo>
</Binding.Source>
</Binding>
</Label.Text>
</Label>
The last Label is similar to the second, except that it references one of the children of the StackLayout :
That child is a Label , which has a Text property of type String , which has a Length property. The first
Label reports the TimeSpan set in the TimePicker , so when that text changes, the final Label changes as well:
That displays the type of the binding source, or DataBindingDemos.PathVariationsPage . You know
PathVariationsPage derives from ContentPage , so it has a Content property:
The type of the Content property is now revealed to be Microsoft.Maui.Controls.StackLayout . Add the
Children property to the Path and the type is also Microsoft.Maui.Controls.StackLayout . Add an index to that
and the type is Microsoft.Maui.Controls.Label . Continue in this way.
As .NET MAUI processes the binding path, it installs a PropertyChanged handler on any object in the path that
implements the INotifyPropertyChanged interface. For example, the final binding reacts to a change in the first
Label because the Text property changes. If a property in the binding path does not implement
INotifyPropertyChanged , any changes to that property will be ignored. Some changes could entirely invalidate
the binding path, so you should use this technique only when the string of properties and sub-properties never
become invalid.
Binding value converters
3/18/2022 • 10 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) data bindings usually transfer data from a source property to a target
property, and in some cases from the target property to the source property. This transfer is straightforward
when the source and target properties are of the same type, or when one type can be converted to the other
type through an implicit conversion. When that is not the case, a type conversion must take place.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
In the String formatting article, you saw how you can use the StringFormat property of a data binding to
convert any type into a string. For other types of conversions, you need to write some specialized code in a class
that implements the IValueConverter interface. Classes that implement IValueConverter are called value
converters, but they are also often referred to as binding converters or binding value converters.
public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
{
return (bool)value ? 1 : 0;
}
}
You then set an instance of this class to the Converter property of the Binding class or to the Converter
property of the Binding markup extension. This class becomes part of the data binding.
The Convert method is called when data moves from the source to the target in OneWay or TwoWay bindings.
The value parameter is the object or value from the data-binding source. The method must return a value of
the type of the data-binding target. The method shown here casts the value parameter to an int and then
compares it with 0 for a bool return value.
The ConvertBack method is called when data moves from the target to the source in TwoWay or OneWayToSource
bindings. ConvertBack performs the opposite conversion: It assumes the value parameter is a bool from the
target, and converts it to an int return value for the source.
NOTE
If a data binding also includes a StringFormat setting, the value converter is invoked before the result is formatted as a
string.
The following example demonstrates how to use this value converter in a data binding:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:DataBindingDemos"
x:Class="DataBindingDemos.EnableButtonsPage"
Title="Enable Buttons">
<ContentPage.Resources>
<local:IntToBoolConverter x:Key="intToBool" />
</ContentPage.Resources>
In this example, the IntToBoolConverter is instantiated in the page's resource dictionary. It's then referenced with
a StaticResource markup extension to set the Converter property in two data bindings. It is very common to
share data converters among multiple data bindings on the page. If a value converter is used in multiple pages
of your application, you can instantiate it in the application-level resource dictionary.
This example demonstrates a common need when a Button performs an operation based on text that the user
types into an Entry view. The Text property of each Entry is initialized to an empty string, because the Text
property is null by default, and the data binding will not work in that case. If nothing has been typed into the
Entry , the Button should be disabled. Each Button contains a data binding on its IsEnabled property. The
data-binding source is the Length property of the Text property of the corresponding Entry . If that Length
property is not 0, the value converter returns true and the Button is enabled:
NOTE
If you know that a value converter will only be used in OneWay bindings, then the ConvertBack method can simply
return null .
The Convert method shown above assumes that the value argument is of type int and the return value must
be of type bool . Similarly, the ConvertBack method assumes that the value argument is of type bool and the
return value is int . If that is not the case, a runtime exception will occur.
You can write value converters to be more generalized and to accept several different types of data. The
Convert and ConvertBack methods can use the as or is operators with the value parameter, or can call
GetType on that parameter to determine its type, and then do something appropriate. The expected type of each
method's return value is given by the targetType parameter. Sometimes, value converters are used with data
bindings of different target types. In this case the value converter can use the targetType argument to perform
a conversion for the correct type.
If the conversion being performed is different for different cultures, use the culture parameter for this purpose.
public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
{
return (bool)value ? TrueObject : FalseObject;
}
public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
{
return ((T)value).Equals(TrueObject);
}
}
The following example demonstrates how this converter can be used to display the value of a Switch view.
Although it's common to instantiate value converters as resources in a resource dictionary, this example
demonstrates an alternative. Here, each value converter is instantiated between Binding.Converter property-
element tags. The x:TypeArguments indicates the generic argument, and TrueObject and FalseObject are both
set to objects of that type:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:DataBindingDemos"
x:Class="DataBindingDemos.SwitchIndicatorsPage"
Title="Switch Indicators">
<ContentPage.Resources>
<Style TargetType="Label">
<Setter Property="FontSize" Value="18" />
<Setter Property="VerticalOptions" Value="Center" />
</Style>
<Style TargetType="Switch">
<Setter Property="VerticalOptions" Value="Center" />
</Style>
</ContentPage.Resources>
<StackLayout Orientation="Horizontal"
VerticalOptions="Center">
<Label Text="Allow popups?" />
<Switch x:Name="switch2" />
<Label>
<Label.Text>
<Binding Source="{x:Reference switch2}"
Path="IsToggled">
<Binding.Converter>
<local:BoolToObjectConverter x:TypeArguments="x:String"
TrueObject="Yes"
FalseObject="No" />
</Binding.Converter>
</Binding>
</Label.Text>
<Label.TextColor>
<Binding Source="{x:Reference switch2}"
Path="IsToggled">
<Binding.Converter>
<local:BoolToObjectConverter x:TypeArguments="Color"
TrueObject="Green"
FalseObject="Red" />
</Binding.Converter>
</Binding>
</Label.TextColor>
</Label>
</StackLayout>
<StackLayout Orientation="Horizontal"
VerticalOptions="Center">
<Label Text="Learn more?" />
<Label Text="Learn more?" />
<Switch x:Name="switch3" />
<Label FontSize="18"
VerticalOptions="Center">
<Label.Style>
<Binding Source="{x:Reference switch3}"
Path="IsToggled">
<Binding.Converter>
<local:BoolToObjectConverter x:TypeArguments="Style">
<local:BoolToObjectConverter.TrueObject>
<Style TargetType="Label">
<Setter Property="Text" Value="Indubitably!" />
<Setter Property="FontAttributes" Value="Italic, Bold" />
<Setter Property="TextColor" Value="Green" />
</Style>
</local:BoolToObjectConverter.TrueObject>
<local:BoolToObjectConverter.FalseObject>
<Style TargetType="Label">
<Setter Property="Text" Value="Maybe later" />
<Setter Property="FontAttributes" Value="None" />
<Setter Property="TextColor" Value="Red" />
</Style>
</local:BoolToObjectConverter.FalseObject>
</local:BoolToObjectConverter>
</Binding.Converter>
</Binding>
</Label.Style>
</Label>
</StackLayout>
</StackLayout>
</ContentPage>
In this example, in the last of the three Switch and Label pairs, the generic argument is set to a Style , and
entire Style objects are provided for the values of TrueObject and FalseObject . These override the implicit
style for Label set in the resource dictionary, so the properties in that style are explicitly assigned to the Label .
Toggling the Switch causes the corresponding Label to reflect the change:
NOTE
It's also possible to use triggers to implement changes in the user-interface based on other views.
Name = NamedColor.GetNearestColorName(color);
}
}
}
The Red , Green , and Blue property values can range between 0 and 1. However, you might prefer that the
components be displayed as two-digit hexadecimal values. To display these as hexadecimal values in XAML, they
must be multiplied by 255, converted to an integer, and then formatted with a specification of "X2" in the
StringFormat property. Multiplying by 255 and converting to an integer can be performed by the value
converter. To make the value converter as generalized as possible, the multiplication factor can be specified with
the ConverterParameter property, which means that it enters the Convert and ConvertBack methods as the
parameter argument:
public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
{
return (int)value / GetParameter(parameter);
}
return 1;
}
}
In this example, the Convert method converts from a float to int while multiplying by the parameter value.
The ConvertBack method divides the integer value argument by parameter and returns a float result.
The type of the parameter argument is likely to be different depending on whether the data binding is defined
in XAML or code. If the ConverterParameter property of Binding is set in code, it's likely to be set to a numeric
value:
binding.ConverterParameter = 255;
The ConverterParameter property is of type Object , so the C# compiler interprets the literal 255 as an integer,
and sets the property to that value.
However, in XAML the ConverterParameter is likely to be set like this:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:DataBindingDemos"
x:Class="DataBindingDemos.RgbColorSelectorPage"
Title="RGB Color Selector">
<ContentPage.BindingContext>
<local:RgbColorViewModel Color="Gray" />
</ContentPage.BindingContext>
<ContentPage.Resources>
<Style TargetType="Slider">
<Setter Property="VerticalOptions" Value="Center" />
</Style>
<Style TargetType="Label">
<Setter Property="HorizontalTextAlignment" Value="Center" />
</Style>
<StackLayout Margin="20">
<BoxView Color="{Binding Color}"
HeightRequest="100"
WidthRequest="100"
HorizontalOptions="Center" />
<StackLayout Margin="10, 0">
<Label Text="{Binding Name}" />
<Slider Value="{Binding Red}" />
<Label Text="{Binding Red,
Converter={StaticResource floatToInt},
ConverterParameter=255,
StringFormat='Red = {0:X2}'}" />
<Slider Value="{Binding Green}" />
<Label Text="{Binding Green,
Converter={StaticResource floatToInt},
ConverterParameter=255,
StringFormat='Green = {0:X2}'}" />
<Slider Value="{Binding Blue}" />
<Label>
<Label.Text>
<Binding Path="Blue"
StringFormat="Blue = {0:X2}"
Converter="{StaticResource floatToInt}">
<Binding.ConverterParameter>
<x:Single>255</x:Single>
</Binding.ConverterParameter>
</Binding>
</Label.Text>
</Label>
</StackLayout>
</StackLayout>
</ContentPage>
The values of the Red and Green properties are displayed with a Binding markup extension. The Blue
property, however, instantiates the Binding class to demonstrate how an explicit float value can be set to
ConverterParameter property:
Relative bindings
3/18/2022 • 5 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) relative bindings provide the ability to set the binding source relative to
the position of the binding target. They are created with the RelativeSource markup extension, and set as the
Source property of a binding expression.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The RelativeSource markup extension is supported by the RelativeSourceExtension class, which defines the
following properties:
, of type RelativeBindingSourceMode , describes the location of the binding source relative to the position
Mode
of the binding target.
AncestorLevel , of type int , an optional ancestor level to look for, when the Mode property is FindAncestor .
An AncestorLevel of n skips n-1 instances of the AncestorType .
AncestorType , of type Type , the type of ancestor to look for, when the Mode property is FindAncestor .
NOTE
The XAML parser allows the RelativeSourceExtension class to be abbreviated as RelativeSource .
The Mode property should be set to one of the RelativeBindingSourceMode enumeration members:
TemplatedParent indicates the element to which the template, in which the bound element exists, is applied.
For more information, see Bind to a templated parent.
Self indicates the element on which the binding is being set, allowing you to bind one property of that
element to another property on the same element. For more information, see Bind to self.
FindAncestor indicates the ancestor in the visual tree of the bound element. This mode should be used to
bind to an ancestor control represented by the AncestorType property. For more information, see Bind to an
ancestor.
FindAncestorBindingContext indicates the BindingContext of the ancestor in the visual tree of the bound
element. This mode should be used to bind to the BindingContext of an ancestor represented by the
AncestorType property. For more information, see Bind to an ancestor.
The Mode property is the content property of the RelativeSourceExtension class. Therefore, for XAML markup
expressions expressed with curly braces, you can eliminate the Mode= part of the expression.
Bind to self
The Self relative binding mode is used bind a property of an element to another property on the same
element:
<BoxView Color="Red"
WidthRequest="200"
HeightRequest="{Binding Source={RelativeSource Self}, Path=WidthRequest}"
HorizontalOptions="Center" />
In this example, the BoxView sets its WidthRequest property to a fixed size, and the HeightRequest property
binds to the WidthRequest property. Therefore, both properties are equal and so a square is drawn:
IMPORTANT
When binding a property of an element to another property on the same element, the properties must be the same type.
Alternatively, you can specify a converter on the binding to convert the value.
A common use of this binding mode is set an object's BindingContext to a property on itself. The following code
shows an example of this:
<ContentPage ...
BindingContext="{Binding Source={RelativeSource Self}, Path=DefaultViewModel}">
<StackLayout>
<ListView ItemsSource="{Binding Employees}">
...
</ListView>
</StackLayout>
</ContentPage>
In this example, the BindingContext of the page is set to the DefaultViewModel property of itself. This property is
defined in the code-behind file for the page, and provides a viewmodel instance. The ListView binds to the
Employees property of the viewmodel.
Bind to an ancestor
The FindAncestor and FindAncestorBindingContext relative binding modes are used to bind to parent elements,
of a certain type, in the visual tree. The FindAncestor mode is used to bind to a parent element, which derives
from the Element type. The FindAncestorBindingContext mode is used to bind to the BindingContext of a parent
element.
WARNING
The AncestorType property must be set to a Type when using the FindAncestor and
FindAncestorBindingContext relative binding modes, otherwise a XamlParseException is thrown.
If the Mode property isn't explicitly set, setting the AncestorType property to a type that derives from Element
will implicitly set the Mode property to FindAncestor . Similarly, setting the AncestorType property to a type that
does not derive from Element will implicitly set the Mode property to FindAncestorBindingContext .
NOTE
Relative bindings that use the FindAncestorBindingContext mode will be reapplied when the BindingContext of any
ancestors change.
The following XAML shows an example where the Mode property will be implicitly set to
FindAncestorBindingContext :
<ContentPage ...
BindingContext="{Binding Source={RelativeSource Self}, Path=DefaultViewModel}">
<StackLayout>
<ListView ItemsSource="{Binding Employees}">
<ListView.ItemTemplate>
<DataTemplate>
<ViewCell>
<StackLayout Orientation="Horizontal">
<Label Text="{Binding Fullname}"
VerticalOptions="Center" />
<Button Text="Delete"
Command="{Binding Source={RelativeSource AncestorType={x:Type
local:PeopleViewModel}}, Path=DeleteEmployeeCommand}"
CommandParameter="{Binding}"
HorizontalOptions="End" />
</StackLayout>
</ViewCell>
</DataTemplate>
</ListView.ItemTemplate>
</ListView>
</StackLayout>
</ContentPage>
In this example, the BindingContext of the page is set to the DefaultViewModel property of itself. This property is
defined in the code-behind file for the page, and provides a viewmodel instance. The ListView binds to the
Employees property of the viewmodel. The DataTemplate , which defines the appearance of each item in the
ListView , contains a Button . The button's Command property is bound to the DeleteEmployeeCommand in its
parent's viewmodel. Tapping a Button deletes an employee:
In addition, the optional AncestorLevel property can help disambiguate ancestor lookup in scenarios where
there is possibly more than one ancestor of that type in the visual tree:
In this example, the Label.Text property binds to the Text property of the second Entry that's encountered
on the upward path, starting at the target element of the binding.
NOTE
The AncestorLevel property should be set to 1 to find the ancestor nearest to the binding target element.
<ContentPage ...>
<ContentPage.Resources>
<ControlTemplate x:Key="CardViewControlTemplate">
<Frame BindingContext="{Binding Source={RelativeSource TemplatedParent}}"
BackgroundColor="{Binding CardColor}"
BorderColor="{Binding BorderColor}"
...>
<Grid>
...
<Label Text="{Binding CardTitle}"
... />
<BoxView BackgroundColor="{Binding BorderColor}"
... />
<Label Text="{Binding CardDescription}"
... />
</Grid>
</Frame>
</ControlTemplate>
</ContentPage.Resources>
<StackLayout>
<controls:CardView BorderColor="DarkGray"
CardTitle="John Doe"
CardDescription="Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nulla
elit dolor, convallis non interdum."
IconBackgroundColor="SlateGray"
IconImageSource="user.png"
ControlTemplate="{StaticResource CardViewControlTemplate}" />
<controls:CardView BorderColor="DarkGray"
CardTitle="Jane Doe"
CardDescription="Phasellus eu convallis mi. In tempus augue eu dignissim
fermentum. Morbi ut lacus vitae eros lacinia."
IconBackgroundColor="SlateGray"
IconImageSource="user.png"
ControlTemplate="{StaticResource CardViewControlTemplate}" />
<controls:CardView BorderColor="DarkGray"
CardTitle="Xamarin Monkey"
CardDescription="Aliquam sagittis, odio lacinia fermentum dictum, mi erat
scelerisque erat, quis aliquet arcu."
IconBackgroundColor="SlateGray"
IconImageSource="user.png"
ControlTemplate="{StaticResource CardViewControlTemplate}" />
</StackLayout>
</ContentPage>
In this example, the Frame , which is the root element of the ControlTemplate , has its BindingContext set to the
runtime object instance to which the template is applied. Therefore, the Frame and its children resolve their
binding expressions against the properties of each CardView object:
Binding fallbacks
3/18/2022 • 3 minutes to read • Edit Online
Sometimes data bindings fail, because the binding source can't be resolved, or because the binding succeeds but
returns a null value. While these scenarios can be handled with value converters, or other additional code,
data bindings can be made more robust by defining fallback values to use if the binding process fails. In a .NET
Multi-platform App UI (.NET MAUI) app this can be accomplished by defining the FallbackValue and
TargetNullValue properties in a binding expression. Because these properties reside in the BindingBase class,
they can be used with bindings, multi-bindings, compiled bindings, and with the Binding markup extension.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
NOTE
Use of the FallbackValue and TargetNullValue properties in a binding expression is optional.
The binding on the Label defines a FallbackValue value (delimited by single-quote characters) that will be set
on the target if the binding source can't be resolved. Therefore, the value defined by the FallbackValue property
will be displayed if the Population property doesn't exist on the bound object.
Rather than defining FallbackValue property values inline, it's recommended to define them as resources in a
ResourceDictionary . The advantage of this approach is that such values are defined once in a single location,
and are more easily localizable. The resources can then be retrieved using the StaticResource markup
extension:
NOTE
It's not possible to set the FallbackValue property with a binding expression.
When the FallbackValue property isn't set in a binding expression and the binding path or part of the path isn't
resolved, BindableProperty.DefaultValue is set on the target. However, when the FallbackValue property is set
and the binding path or part of the path isn't resolved, the value of the FallbackValue value property is set on
the target:
Therefore, in this example the Label displays "Population size unknown" because the bound object lacks a
Population property.
IMPORTANT
A defined value converter is not executed in a binding expression when the FallbackValue property is set.
The bindings on the Image and Label both define TargetNullValue values (delimited by single-quote
characters) that will be applied if the binding path returns null . Therefore, the values defined by the
TargetNullValue properties will be displayed for any objects in the collection where the ImageUrl and
Location properties are not defined.
Rather than defining TargetNullValue property values inline, it's recommended to define them as resources in a
ResourceDictionary . The advantage of this approach is that such values are defined once in a single location,
and are more easily localizable. The resources can then be retrieved using the StaticResource markup
extension:
<Image Source="{Binding ImageUrl, TargetNullValue={StaticResource fallbackImageUrl}}"
... />
<Label Text="{Binding Location, TargetNullValue={StaticResource locationUnknown}}"
... />
NOTE
It's not possible to set the TargetNullValue property with a binding expression.
When the TargetNullValue property isn't set in a binding expression, a source value of null will be converted
if a value converter is defined, formatted if a StringFormat is defined, and the result is then set on the target.
However, when the TargetNullValue property is set, a source value of null will be converted if a value
converter is defined, and if it's still null after the conversion, the value of the TargetNullValue property is set
on the target:
Therefore, in this example the Image and Label objects display their TargetNullValue when their source
objects are null .
IMPORTANT
String formatting is not applied in a binding expression when the TargetNullValue property is set.
Multi-bindings
3/18/2022 • 8 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) multi-bindings provide the ability to attach a collection of Binding
objects to a single binding target property. They are created with the MultiBinding class, which evaluates all of
its Binding objects and returns a single value through a IMultiValueConverter instance provided by your app.
In addition, MultiBinding reevaluates all of its Binding objects when any of the bound data changes.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The Bindings property is the content property of the MultiBinding class, and therefore does not need to be
explicitly set from XAML.
In addition, the MultiBinding class inherits the following properties from the BindingBase class:
FallbackValue , of type object , which represents the value to use when the multi-binding is unable to return
a value.
Mode , of type , which indicates the direction of the data flow of the multi-binding.
BindingMode
StringFormat , of type string , which specifies how to format the multi-binding result if it's displayed as a
string.
TargetNullValue , of type object , which represents the value that is used in the target wen the value of the
source is null .
A MultiBinding must use a IMultiValueConverter to produce a value for the binding target, based on the value
of the bindings in the Bindings collection. For example, a Color might be computed from red, blue, and green
values, which can be values from the same or different binding source objects. When a value moves from the
target to the sources, the target property value is translated to a set of values that are fed back into the bindings.
IMPORTANT
Individual bindings in the Bindings collection can have their own value converters.
The value of the Mode property determines the functionality of the MultiBinding , and is used as the binding
mode for all the bindings in the collection unless an individual binding overrides the property. For example, if
the Mode property on a MultiBinding object is set to TwoWay , then all the bindings in the collection are
considered TwoWay unless you explicitly set a different Mode value on one of the bindings.
Define a IMultiValueConverter
The IMultiValueConverter interface enables custom logic to be applied to a MultiBinding . To associate a
converter with a MultiBinding , create a class that implements the IMultiValueConverter interface, and then
implement the Convert and ConvertBack methods:
public object[] ConvertBack(object value, Type[] targetTypes, object parameter, CultureInfo culture)
{
if (!(value is bool b) || targetTypes.Any(t => !t.IsAssignableFrom(typeof(bool))))
{
// Return null to indicate conversion back is not possible
return null;
}
if (b)
{
return targetTypes.Select(t => (object)true).ToArray();
}
else
{
// Can't convert back from false because of ambiguity
return null;
}
}
}
The Convert method converts source values to a value for the binding target. .NET MAUI calls this method
when it propagates values from source bindings to the binding target. This method accepts four arguments:
values , of type object[] , is an array of values that the source bindings in the MultiBinding produces.
targetType , of type Type , is the type of the binding target property.
parameter , of type object , is the converter parameter to use.
culture , of type CultureInfo , is the culture to use in the converter.
The Convert method returns an object that represents a converted value. This method should return:
BindableProperty.UnsetValue to indicate that the converter did not produce a value, and that the binding will
use the FallbackValue .
Binding.DoNothing to instruct .NET MAUI not to perform any action. For example, to instruct .NET MAUI not
to transfer a value to the binding target, or not to use the FallbackValue .
null to indicate that the converter cannot perform the conversion, and that the binding will use the
TargetNullValue .
IMPORTANT
A MultiBinding that receives BindableProperty.UnsetValue from a Convert method must define its
FallbackValue property. Similarly, a MultiBinding that receives null from a Convert method must define its
TargetNullValue property.
The ConvertBack method converts a binding target to the source binding values. This method accepts four
arguments:
value , of type object , is the value that the binding target produces.
targetTypes , of type Type[] , is the array of types to convert to. The array length indicates the number and
types of values that are suggested for the method to return.
parameter , of type object , is the converter parameter to use.
culture , of type CultureInfo , is the culture to use in the converter.
The ConvertBack method returns an array of values, of type object[] , that have been converted from the
target value back to the source values. This method should return:
BindableProperty.UnsetValue at position i to indicate that the converter is unable to provide a value for the
source binding at index i , and that no value is to be set on it.
Binding.DoNothing at position i to indicate that no value is to be set on the source binding at index i .
null to indicate that the converter cannot perform the conversion or that it does not support conversion in
this direction.
Consume a IMultiValueConverter
A IMultiValueConverter is typically consumed by instantiating it in a resource dictionary, and then referencing it
using the StaticResource markup extension to set the MultiBinding.Converter property:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:DataBindingDemos"
x:Class="DataBindingDemos.MultiBindingConverterPage"
Title="MultiBinding Converter demo">
<ContentPage.Resources>
<local:AllTrueMultiConverter x:Key="AllTrueConverter" />
<local:InverterConverter x:Key="InverterConverter" />
</ContentPage.Resources>
<CheckBox>
<CheckBox.IsChecked>
<MultiBinding Converter="{StaticResource AllTrueConverter}">
<Binding Path="Employee.IsOver16" />
<Binding Path="Employee.HasPassedTest" />
<Binding Path="Employee.IsSuspended"
Converter="{StaticResource InverterConverter}" />
</MultiBinding>
</CheckBox.IsChecked>
</CheckBox>
</ContentPage>
In this example, the MultiBinding object uses the AllTrueMultiConverter instance to set the CheckBox.IsChecked
property to true , provided that the three Binding objects evaluate to true . Otherwise, the
CheckBox.IsChecked property is set to false .
By default, the CheckBox.IsCheckedproperty uses a TwoWay binding. Therefore, the ConvertBack method of the
AllTrueMultiConverter instance is executed when the CheckBox is unchecked by the user, which sets the source
binding values to the value of the CheckBox.IsChecked property.
The equivalent C# code is shown below:
Format strings
A MultiBinding can format any multi-binding result that's displayed as a string, with the StringFormat
property. This property can be set to a standard .NET formatting string, with placeholders, that specifies how to
format the multi-binding result:
<Label>
<Label.Text>
<MultiBinding StringFormat="{}{0} {1} {2}">
<Binding Path="Employee1.Forename" />
<Binding Path="Employee1.MiddleName" />
<Binding Path="Employee1.Surname" />
</MultiBinding>
</Label.Text>
</Label>
In this example, the StringFormat property combines the three bound values into a single string that's displayed
by the Label .
The equivalent C# code is shown below:
IMPORTANT
The number of parameters in a composite string format can't exceed the number of child Binding objects in the
MultiBinding .
When setting the Converter and StringFormat properties, the converter is applied to the data value first, and
then the StringFormat is applied.
For more information about string formatting in .NET MAUI, see String formatting.
<ContentPage.Resources>
<local:AllTrueMultiConverter x:Key="AllTrueConverter" />
<local:AnyTrueMultiConverter x:Key="AnyTrueConverter" />
<local:InverterConverter x:Key="InverterConverter" />
</ContentPage.Resources>
<CheckBox>
<CheckBox.IsChecked>
<MultiBinding Converter="{StaticResource AnyTrueConverter}">
<MultiBinding Converter="{StaticResource AllTrueConverter}">
<Binding Path="Employee.IsOver16" />
<Binding Path="Employee.HasPassedTest" />
<Binding Path="Employee.IsSuspended" Converter="{StaticResource InverterConverter}" />
</MultiBinding>
<Binding Path="Employee.IsMonarch" />
</MultiBinding>
</CheckBox.IsChecked>
</CheckBox>
</ContentPage>
In this example, the MultiBinding object uses its AnyTrueMultiConverter instance to set the CheckBox.IsChecked
property to true , provided that all of the Binding objects in the inner MultiBinding object evaluate to true ,
or provided that the Binding object in the outer MultiBinding object evaluates to true . Otherwise, the
CheckBox.IsChecked property is set to false .
<ControlTemplate x:Key="CardViewExpanderControlTemplate">
<local:Expander BindingContext="{Binding Source={RelativeSource TemplatedParent}}"
IsExpanded="{Binding IsExpanded, Source={RelativeSource TemplatedParent}}"
BackgroundColor="{Binding CardColor}"
RowDefinitions="Auto,Auto"
Padding="8">
<local:Expander.IsVisible>
<MultiBinding Converter="{StaticResource AllTrueConverter}">
<Binding Path="IsExpanded" />
<Binding Path="IsEnabled" />
</MultiBinding>
</local:Expander.IsVisible>
<Grid>
<!-- XAML that defines Expander header goes here -->
</Grid>
<Grid>
<!-- XAML that defines Expander content goes here -->
</Grid>
</local:Expander>
</ControlTemplate>
</ContentPage.Resources>
<StackLayout>
<controls:CardViewExpander BorderColor="DarkGray"
CardTitle="John Doe"
CardDescription="Lorem ipsum dolor sit amet, consectetur adipiscing elit.
Nulla elit dolor, convallis non interdum."
IconBackgroundColor="SlateGray"
IconImageSource="user.png"
ControlTemplate="{StaticResource CardViewExpanderControlTemplate}"
IsEnabled="True"
IsExpanded="True" />
</StackLayout>
</ContentPage>
In this example, the TemplatedParent relative binding mode is used to bind from within a control template to the
runtime object instance to which the template is applied. The Expander , which is the root element of the
ControlTemplate , has its BindingContext set to the runtime object instance to which the template is applied.
Therefore, the Expander and its children resolve their binding expressions, and Binding objects, against the
properties of the CardViewExpander object. The MultiBinding uses the AllTrueMultiConverter instance to set the
Expander.IsVisible property to true provided that the two Binding objects evaluate to true . Otherwise, the
Expander.IsVisible property is set to false .
In a .NET Multi-platform App UI (.NET MAUI) app that uses the Model-View-ViewModel (MVVM) pattern, data
bindings are defined between properties in the viewmodel, which is typically a class that derives from
INotifyPropertyChanged , and properties in the view, which is typically the XAML file. Sometimes an app has
needs that go beyond these property bindings by requiring the user to initiate commands that affect something
in the viewmodel. These commands are generally signaled by button clicks or finger taps, and traditionally they
are processed in the code-behind file in a handler for the Clicked event of the Button or the Tapped event of a
TapGestureRecognizer .
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The commanding interface provides an alternative approach to implementing commands that is much better
suited to the MVVM architecture. The viewmodel can contain commands, which are methods that are executed
in reaction to a specific activity in the view such as a Button click. Data bindings are defined between these
commands and the Button .
To allow a data binding between a Button and a viewmodel, the Button defines two properties:
Command of type System.Windows.Input.ICommand
CommandParameter of type Object
To use the command interface, you define a data binding that targets the Command property of the Button
where the source is a property in the viewmodel of type ICommand . The viewmodel contains code associated
with that ICommand property that is executed when the button is clicked. You can set the CommandParameter
property to arbitrary data to distinguish between multiple buttons if they are all bound to the same ICommand
property in the viewmodel.
Many other views also define Command and CommandParameter properties. All these commands can be handled
within a viewmodel using an approach that doesn't depend on the user-interface object in the view.
ICommands
The ICommand interface is defined in the System.Windows.Input namespace, and consists of two methods and
one event:
To use the command interface, your viewmodel should contain properties of type ICommand :
public ICommand MyCommand { private set; get; }
The viewmodel must also reference a class that implements the ICommand interface. In the view, the Command
property of a Button is bound to that property:
When the user presses the Button , the Button calls the Execute method in the ICommand object bound to its
Command property.
When the binding is first defined on the Command property of the Button , and when the data binding changes
in some way, the Button calls the CanExecute method in the ICommand object. If CanExecute returns false ,
then the Button disables itself. This indicates that the particular command is currently unavailable or invalid.
The Button also attaches a handler on the CanExecuteChanged event of ICommand . The event is raised from
within the viewmodel. When that event is raised, the Button calls CanExecute again. The Button enables itself
if CanExecute returns true and disables itself if CanExecute returns false .
WARNING
Do not use the IsEnabled property of Button if you're using the command interface.
When your viewmodel defines a property of type ICommand , the viewmodel must also contain or reference a
class that implements the ICommand interface. This class must contain or reference the Execute and CanExecute
methods, and fire the CanExecuteChanged event whenever the CanExecute method might return a different
value. You can use the Command or Command<T> class included in .NET MAUI to implement the ICommand
interface. These classes allow you to specify the bodies of the Execute and CanExecute methods in class
constructors.
TIP
Use Command<T> when you use the CommandParameter property to distinguish between multiple views bound to the
same ICommand property, and the Command class when that isn't a requirement.
Basic commanding
The following examples demonstrate basic commands implemented in a viewmodel.
The PersonViewModel class defines three properties named Name , Age , and Skills that define a person:
public class PersonViewModel : INotifyPropertyChanged
{
string name;
double age;
string skills;
storage = value;
OnPropertyChanged(propertyName);
return true;
}
The PersonCollectionViewModel class shown below creates new objects of type PersonViewModel and allows the
user to fill in the data. For that purpose, the class defines IsEditing , of type bool , and PersonEdit , of type
PersonViewModel , properties. In addition, the class defines three properties of type ICommand and a property
named Persons of type IList<PersonViewModel> :
public class PersonCollectionViewModel : INotifyPropertyChanged
{
PersonViewModel personEdit;
bool isEditing;
storage = value;
OnPropertyChanged(propertyName);
return true;
}
In this example, changes to the three ICommand properties and the Persons property do not result in
PropertyChanged events being raised. These properties are all set when the class is first created and do not
change.
The following example shows the XAML that consumes the PersonCollectionViewModel :
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:DataBindingDemos"
x:Class="DataBindingDemos.PersonEntryPage"
Title="Person Entry">
<ContentPage.BindingContext>
<local:PersonCollectionViewModel />
</ContentPage.BindingContext>
<Grid Margin="10">
<Grid.RowDefinitions>
<RowDefinition Height="Auto" />
<RowDefinition Height="Auto" />
<RowDefinition Height="Auto" />
<RowDefinition Height="*" />
</Grid.RowDefinitions>
<Button Text="Submit"
Grid.Column="0"
Command="{Binding SubmitCommand}"
VerticalOptions="Center" />
<Button Text="Cancel"
Grid.Column="1"
Command="{Binding CancelCommand}"
VerticalOptions="Center" />
</Grid>
In this example, the page's BindingContext property is set to the PersonCollectionViewModel . The Grid contains
a Button with the text New with its Command property bound to the NewCommand property in the viewmodel, an
entry form with properties bound to the IsEditing property, as well as properties of PersonViewModel , and two
more buttons bound to the SubmitCommand and CancelCommand properties of the viewmodel. The ListView
displays the collection of persons already entered:
The following screenshot shows the Submit button enabled after an age has been set:
When the user first presses the New button, this enables the entry form but disables the New button. The user
then enters a name, age, and skills. At any time during the editing, the user can press the Cancel button to start
over. Only when a name and a valid age have been entered is the Submit button enabled. Pressing this Submit
button transfers the person to the collection displayed by the ListView . After either the Cancel or Submit
button is pressed, the entry form is cleared and the New button is enabled again.
All the logic for the New , Submit , and Cancel buttons is handled in PersonCollectionViewModel through
definitions of the NewCommand , SubmitCommand , and CancelCommand properties. The constructor of the
PersonCollectionViewModel sets these three properties to objects of type Command .
A constructor of the Command class allows you to pass arguments of type Action and Func<bool>
corresponding to the Execute and CanExecute methods. This action and function can be defined as lambda
functions in the Command constructor:
public class PersonCollectionViewModel : INotifyPropertyChanged
{
···
public PersonCollectionViewModel()
{
NewCommand = new Command(
execute: () =>
{
PersonEdit = new PersonViewModel();
PersonEdit.PropertyChanged += OnPersonEditPropertyChanged;
IsEditing = true;
RefreshCanExecutes();
},
canExecute: () =>
{
return !IsEditing;
});
···
}
void RefreshCanExecutes()
{
(NewCommand as Command).ChangeCanExecute();
(SubmitCommand as Command).ChangeCanExecute();
(CancelCommand as Command).ChangeCanExecute();
}
···
}
When the user clicks the New button, the execute function passed to the Command constructor is executed. This
creates a new PersonViewModel object, sets a handler on that object's PropertyChanged event, sets IsEditing to
true , and calls the RefreshCanExecutes method defined after the constructor.
Besides implementing the ICommand interface, the Command class also defines a method named
ChangeCanExecute . A viewmodel should call ChangeCanExecute for an ICommand property whenever anything
happens that might change the return value of the CanExecute method. A call to ChangeCanExecute causes the
Command class to fire the CanExecuteChanged method. The Button has attached a handler for that event and
responds by calling CanExecute again, and then enabling itself based on the return value of that method.
When the executemethod of NewCommand calls RefreshCanExecutes , the NewCommand property gets a call to
ChangeCanExecute , and the Button calls the canExecute method, which now returns false because the
IsEditing property is now true .
The PropertyChanged handler for the new PersonViewModel object calls the ChangeCanExecute method of
SubmitCommand :
public class PersonCollectionViewModel : INotifyPropertyChanged
{
···
public PersonCollectionViewModel()
{
···
SubmitCommand = new Command(
execute: () =>
{
Persons.Add(PersonEdit);
PersonEdit.PropertyChanged -= OnPersonEditPropertyChanged;
PersonEdit = null;
IsEditing = false;
RefreshCanExecutes();
},
canExecute: () =>
{
return PersonEdit != null &&
PersonEdit.Name != null &&
PersonEdit.Name.Length > 1 &&
PersonEdit.Age > 0;
});
···
}
···
}
The canExecutefunction for SubmitCommand is called every time there's a property changed in the
PersonViewModel object being edited. It returns true only when the Name property is at least one character
long, and Age is greater than 0. At that time, the Submit button becomes enabled.
The execute function for Submit removes the property-changed handler from the PersonViewModel , adds the
object to the Persons collection, and returns everything to its initial state.
The execute function for the Cancel button does everything that the Submit button does except add the
object to the collection:
The canExecute method returns true at any time a PersonViewModel is being edited.
NOTE
It isn't necessary to define the execute and canExecute methods as lambda functions. You can write them as private
methods in the viewmodel and reference them in the Command constructors. However, this approach can result in a lot of
methods that are referenced only once in the viewmodel.
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:DataBindingDemos"
x:Class="DataBindingDemos.DecimalKeypadPage"
Title="Decimal Keyboard">
<ContentPage.BindingContext>
<local:DecimalKeypadViewModel />
</ContentPage.BindingContext>
<ContentPage.Resources>
<Style TargetType="Button">
<Setter Property="FontSize" Value="32" />
<Setter Property="BorderWidth" Value="1" />
<Setter Property="BorderColor" Value="Black" />
</Style>
</ContentPage.Resources>
<Grid WidthRequest="240"
HeightRequest="480"
ColumnDefinitions="80, 80, 80"
RowDefinitions="Auto, Auto, Auto, Auto, Auto, Auto"
ColumnSpacing="2"
RowSpacing="2"
HorizontalOptions="Center"
VerticalOptions="Center">
<Label Text="{Binding Entry}"
Grid.Row="0" Grid.Column="0" Grid.ColumnSpan="3"
Margin="0,0,10,0"
FontSize="32"
LineBreakMode="HeadTruncation"
VerticalTextAlignment="Center"
HorizontalTextAlignment="End" />
<Button Text="CLEAR"
Grid.Row="1" Grid.Column="0" Grid.ColumnSpan="2"
Command="{Binding ClearCommand}" />
<Button Text="⇦"
Grid.Row="1" Grid.Column="2"
Command="{Binding BackspaceCommand}" />
<Button Text="7"
Grid.Row="2" Grid.Column="0"
Command="{Binding DigitCommand}"
CommandParameter="7" />
<Button Text="8"
Grid.Row="2" Grid.Column="1"
Command="{Binding DigitCommand}"
CommandParameter="8" />
<Button Text="9"
Grid.Row="2" Grid.Column="2"
Command="{Binding DigitCommand}"
CommandParameter="9" />
<Button Text="4"
Grid.Row="3" Grid.Column="0"
Command="{Binding DigitCommand}"
CommandParameter="4" />
<Button Text="5"
Grid.Row="3" Grid.Column="1"
Command="{Binding DigitCommand}"
CommandParameter="5" />
<Button Text="6"
Grid.Row="3" Grid.Column="2"
Command="{Binding DigitCommand}"
CommandParameter="6" />
<Button Text="1"
Grid.Row="4" Grid.Column="0"
Command="{Binding DigitCommand}"
CommandParameter="1" />
<Button Text="2"
Grid.Row="4" Grid.Column="1"
Command="{Binding DigitCommand}"
CommandParameter="2" />
<Button Text="3"
Grid.Row="4" Grid.Column="2"
Command="{Binding DigitCommand}"
CommandParameter="3" />
<Button Text="0"
Grid.Row="5" Grid.Column="0" Grid.ColumnSpan="2"
Command="{Binding DigitCommand}"
CommandParameter="0" />
<Button Text="·"
Grid.Row="5" Grid.Column="2"
Command="{Binding DigitCommand}"
CommandParameter="." />
</Grid>
</ContentPage>
In this example, the page's BindingContext is a DecimalKeypadViewModel . The Entry property of this viewmodel
is bound to the Text property of a Label . All the Button objects are bound to commands in the viewmodel:
ClearCommand , BackspaceCommand , and DigitCommand . The 11 buttons for the 10 digits and the decimal point
share a binding to DigitCommand . The CommandParameter distinguishes between these buttons. The value set to
CommandParameter is generally the same as the text displayed by the button except for the decimal point, which
for purposes of clarity is displayed with a middle dot character:
The DecimalKeypadViewModel defines an Entry property of type string and three properties of type ICommand :
public class DecimalKeypadViewModel : INotifyPropertyChanged
{
string entry = "0";
The button corresponding to the ClearCommand is always enabled and sets the entry back to "0":
void RefreshCanExecutes()
{
((Command)BackspaceCommand).ChangeCanExecute();
((Command)DigitCommand).ChangeCanExecute();
}
···
}
Because the button is always enabled, it is not necessary to specify a canExecute argument in the Command
constructor.
The Backspace button is enabled only when the length of the entry is greater than 1, or if Entry is not equal to
the string "0":
public class DecimalKeypadViewModel : INotifyPropertyChanged
{
···
public DecimalKeypadViewModel()
{
···
BackspaceCommand = new Command(
execute: () =>
{
Entry = Entry.Substring(0, Entry.Length - 1);
if (Entry == "")
{
Entry = "0";
}
RefreshCanExecutes();
},
canExecute: () =>
{
return Entry.Length > 1 || Entry != "0";
});
···
}
···
}
The logic for the execute function for the Backspace button ensures that the Entry is at least a string of "0".
The DigitCommand property is bound to 11 buttons, each of which identifies itself with the CommandParameter
property. The DigitCommand is set to an instance of the Command<T> class. When using the commanding interface
with XAML, the CommandParameter properties are usually strings, which is type of the generic argument. The
execute and canExecute functions then have arguments of type string :
The execute method appends the string argument to the Entry property. However, if the result begins with a
zero (but not a zero and a decimal point) then that initial zero must be removed using the Substring function.
The canExecute method returns false only if the argument is the decimal point (indicating that the decimal
point is being pressed) and Entry already contains a decimal point. All the execute methods call
RefreshCanExecutes , which then calls ChangeCanExecute for both DigitCommand and ClearCommand . This ensures
that the decimal point and backspace buttons are enabled or disabled based on the current sequence of entered
digits.
Compiled bindings
3/18/2022 • 7 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) data bindings have two main issues:
1. There's no compile-time validation of binding expressions. Instead, bindings are resolved at runtime.
Therefore, any invalid bindings aren't detected until runtime when the application doesn't behave as expected
or error messages appear.
2. They aren't cost efficient. Bindings are resolved at runtime using general-purpose object inspection
(reflection), and the overhead of doing this varies from platform to platform.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Compiled bindings improve data binding performance in .NET MAUI applications by resolving binding
expressions at compile-time rather than runtime. In addition, this compile-time validation of binding expressions
enables a better developer troubleshooting experience because invalid bindings are reported as build errors.
The process for using compiled bindings is to:
1. Ensure that XAML compilation is enabled. For more information about XAML compilation, see XAML
Compilation.
2. Set an x:DataType attribute on a VisualElement to the type of the object that the VisualElement and its
children will bind to.
NOTE
It's recommended to set the x:DataType attribute at the same level in the view hierarchy as the BindingContext is set.
However, this attribute can be re-defined at any location in a view hierarchy.
To use compiled bindings, the x:DataType attribute must be set to a string literal, or a type using the x:Type
markup extension. At XAML compile time, any invalid binding expressions will be reported as build errors.
However, the XAML compiler will only report a build error for the first invalid binding expression that it
encounters. Any valid binding expressions that are defined on the VisualElement or its children will be
compiled, regardless of whether the BindingContext is set in XAML or code. Compiling a binding expression
generates compiled code that will get a value from a property on the source, and set it on the property on the
target that's specified in the markup. In addition, depending on the binding expression, the generated code may
observe changes in the value of the source property and refresh the target property, and may push changes
from the target back to the source.
IMPORTANT
Compiled bindings are disabled for any binding expressions that define the Source property. This is because the
Source property is always set using the x:Reference markup extension, which can't be resolved at compile time.
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:DataBindingDemos"
x:Class="DataBindingDemos.CompiledColorSelectorPage"
x:DataType="local:HslColorViewModel"
Title="Compiled Color Selector">
<ContentPage.BindingContext>
<local:HslColorViewModel Color="Sienna" />
</ContentPage.BindingContext>
...
<StackLayout>
<BoxView Color="{Binding Color}"
... />
<StackLayout Margin="10, 0">
<Label Text="{Binding Name}" />
<Slider Value="{Binding Hue}" />
<Label Text="{Binding Hue, StringFormat='Hue = {0:F2}'}" />
<Slider Value="{Binding Saturation}" />
<Label Text="{Binding Saturation, StringFormat='Saturation = {0:F2}'}" />
<Slider Value="{Binding Luminosity}" />
<Label Text="{Binding Luminosity, StringFormat='Luminosity = {0:F2}'}" />
</StackLayout>
</StackLayout>
</ContentPage>
The ContentPage instantiates the HslColorViewModel and initializes the Color property within property element
tags for the BindingContext property. The ContentPage also defines the x:DataType attribute as the viewmodel
type, indicating that any binding expressions in the ContentPage view hierarchy will be compiled. This can be
verified by changing any of the binding expressions to bind to a non-existent viewmodel property, which will
result in a build error. While this example sets the x:DataType attribute to a string literal, it can also be set to a
type with the x:Type markup extension.
IMPORTANT
The x:DataType attribute can be re-defined at any point in a view hierarchy.
The BoxView , Label elements, and Slider views inherit the binding context from the ContentPage . These
views are all binding targets that reference source properties in the viewmodel. For the BoxView.Color property,
and the Label.Text property, the data bindings are OneWay – the properties in the view are set from the
properties in the viewmodel. However, the Slider.Value property uses a TwoWay binding. This allows each
Slider to be set from the viewmodel, and also for the viewmodel to be set from each Slider .
When the example is first run, the BoxView , Label elements, and Slider elements are all set from the
viewmodel based on the initial Color property set when the viewmodel was instantiated. As the sliders are
manipulated, the BoxView and Label elements are updated accordingly:
For more information about this color selector, see ViewModels and property-change notifications.
The ListView.ItemsSource property is set to the static NamedColor.All property. The NamedColor class uses .NET
reflection to enumerate all the static public fields in the Colors class, and to store them with their names in a
collection that is accessible from the static All property. Therefore, the ListView is filled with all of the
NamedColor instances. For each item in the ListView , the binding context for the item is set to a NamedColor
object. The BoxView and Label elements in the ViewCell are bound to NamedColor properties.
The DataTemplate defines the x:DataType attribute to be the NamedColor type, indicating that any binding
expressions in the DataTemplate view hierarchy will be compiled. This can be verified by changing any of the
binding expressions to bind to a non-existent NamedColor property, which will result in a build error. While this
example sets the x:DataType attribute to a string literal, it can also be set to a type with the x:Type markup
extension.
When the example is first run, the ListView is populated with NamedColor instances. When an item in the
ListView is selected, the BoxView.Color property is set to the color of the selected item in the ListView :
Selecting other items in the ListView updates the color of the BoxView .
The root StackLayout sets the x:DataType attribute to be the HslColorViewModel type, indicating that any
binding expression in the root StackLayout view hierarchy will be compiled. However, the inner StackLayout
redefines the x:DataType attribute to null with the x:Null markup expression. Therefore, the binding
expressions within the inner StackLayout use classic bindings. Only the BoxView , within the root StackLayout
view hierarchy, uses compiled bindings.
Performance
Compiled bindings improve data binding performance, with the performance benefit varying:
A compiled binding that uses property-change notification (i.e. a OneWay , OneWayToSource , or TwoWay
binding) is resolved approximately 8 times quicker than a classic binding.
A compiled binding that doesn't use property-change notification (i.e. a OneTime binding) is resolved
approximately 20 times quicker than a classic binding.
Setting the BindingContext on a compiled binding that uses property change notification (i.e. a OneWay ,
OneWayToSource , or TwoWay binding) is approximately 5 times quicker than setting the BindingContext on a
classic binding.
Setting the BindingContext on a compiled binding that doesn't use property change notification (i.e. a
OneTime binding) is approximately 7 times quicker than setting the BindingContext on a classic binding.
These performance differences can be magnified on mobile devices, dependent upon the platform being used,
the version of the operating system being used, and the device on which the application is running.
Recognize a drag and drop gesture
3/18/2022 • 11 minutes to read • Edit Online
A .NET Multi-platform App UI (.NET MAUI) drag and drop gesture recognizer enables items, and their associated
data packages, to be dragged from one onscreen location to another location using a continuous gesture. Drag
and drop can take place in a single application, or it can start in one application and end in another.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The drag source, which is the element on which the drag gesture is initiated, can provide data to be transferred
by populating a data package object. When the drag source is released, drop occurs. The drop target, which is
the element under the drag source, then processes the data package.
IMPORTANT
On iOS a minimum platform of iOS 11 is required.
Enable drag
In .NET MAUI, drag gesture recognition is provided by the DragGestureRecognizer class. This class defines the
following properties:
CanDrag , of type bool , which indicates whether the element the gesture recognizer is attached to can be a
drag source. The default value of this property is true .
DragStartingCommand , of type ICommand , which is executed when a drag gesture is first recognized.
DragStartingCommandParameter , of type object , which is the parameter that's passed to the
DragStartingCommand .
DropCompletedCommand , of type ICommand , which is executed when the drag source is dropped.
DropCompletedCommandParameter , of type object , which is the parameter that's passed to the
DropCompletedCommand .
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
The class also defines DragStarting and DropCompleted events that fire provided that
DragGestureRecognizer
the CanDrag property is true . When a DragGestureRecognizer object detects a drag gesture, it executes the
DragStartingCommand and invokes the DragStarting event. Then, when the DragGestureRecognizer object detects
the completion of a drop gesture, it executes the DropCompletedCommand and invokes the DropCompleted event.
The DragStartingEventArgs object that accompanies the DragStarting event defines the following properties:
Handled, of type bool , indicates whether the event handler has handled the event or whether .NET MAUI
should continue its own processing.
Cancel , of type bool , indicates whether the event should be canceled.
Data , of type DataPackage , indicates the data package that accompanies the drag source. This is a read-only
property.
The following XAML example shows a DragGestureRecognizer attached to an Image :
<Image Source="monkeyface.png">
<Image.GestureRecognizers>
<DragGestureRecognizer />
</Image.GestureRecognizers>
</Image>
TIP
A drag gesture is initiated with a long-press followed by a drag.
The following table shows the properties that are read, and any conversion that's attempted, when a drag is
initiated on a text control:
C O N T RO L P RO P ERT Y C O N VERSIO N
Editor Text
Entry Text
Label Text
C O N T RO L P RO P ERT Y C O N VERSIO N
For content other than text and images, you'll need to build a data package yourself.
Data packages are represented by the DataPackage class, which defines the following properties:
Properties , of type DataPackagePropertySet , which is a collection of properties that comprise the data
contained in the DataPackage . This property is a read-only property.
Image , of type ImageSource , which is the image contained in the DataPackage .
Text , of type string , which is the text contained in the DataPackage .
View , of type DataPackageView , which is a read-only version of the DataPackage .
The following XAML example shows a DragGestureRecognizer that registers a handler for the DragStarting
event:
<Path Stroke="Black"
StrokeThickness="4">
<Path.GestureRecognizers>
<DragGestureRecognizer DragStarting="OnDragStarting" />
</Path.GestureRecognizers>
<Path.Data>
<!-- PathGeometry goes here -->
</Path.Data>
</Path>
In this example, the DragGestureRecognizer is attached to a Path object. The DragStarting event is raised when
a drag gesture is detected on the Path , which executes the OnDragStarting event handler:
The DragStartingEventArgs object that accompanies the DragStarting event has a Data property, of type
DataPackage . In this example, the Text property of the DataPackage object is set to a string . The DataPackage
can then be accessed on drop, to retrieve the string .
Store data in the property bag
Any data, including images and text, can be associated with a drag source by storing the data in the
DataPackage.Properties collection. This can be accomplished in the handler for the DragStarting event.
The following XAML example shows a DragGestureRecognizer that registers a handler for the DragStarting
event:
<Rectangle Stroke="Red"
Fill="DarkBlue"
StrokeThickness="4"
HeightRequest="200"
WidthRequest="200">
<Rectangle.GestureRecognizers>
<DragGestureRecognizer DragStarting="OnDragStarting" />
</Rectangle.GestureRecognizers>
</Rectangle>
In this example, the DragGestureRecognizer is attached to a Rectangle object. The DragStarting event is raised
when a drag gesture is detected on the Rectangle , which executes the OnDragStarting event handler:
The DragStartingEventArgs object that accompanies the DragStarting event has a Data property, of type
DataPackage . The Properties collection of the DataPackage object, which is a Dictionary<string, object>
collection, can be modified to store any required data. In this example, the Properties dictionary is modified to
store a Square object, that represents the size of the Rectangle , against a "Square" key.
Enable drop
In .NET MAUI, drop gesture recognition is provided by the DropGestureRecognizer class. This class defines the
following properties:
, of type bool , which indicates whether the element the gesture recognizer is attached to can be a
AllowDrop
drop target. The default value of this property is true .
DragOverCommand , of type ICommand , which is executed when the drag source is dragged over the drop target.
DragOverCommandParameter , of type object , which is the parameter that's passed to the DragOverCommand .
DragLeaveCommand , of type ICommand , which is executed when the drag source is dragged off the drop target.
DragLeaveCommandParameter , of type object , which is the parameter that's passed to the DragLeaveCommand .
DropCommand , of type ICommand , which is executed when the drag source is dropped over the drop target.
DropCommandParameter , of type object , which is the parameter that's passed to the DropCommand .
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
The DropGestureRecognizerclass also defines DragOver , DragLeave , and Drop events that fire provided that the
AllowDrop property is true . When a DropGestureRecognizer recognizes a drag source over the drop target, it
executes the DragOverCommand and invokes the DragOver event. Then, if the drag source is dragged off the drop
target, the DropGestureRecognizer executes the DragLeaveCommand and invokes the DragLeave event. Finally,
when the DropGestureRecognizer recognizes a drop gesture over the drop target, it executes the DropCommand
and invokes the Drop event.
The DragEventArgs class, that accompanies the DragOver and DragLeave events, defines the following
properties:
Data , of type DataPackage , which contains the data associated with the drag source. This property is read-
only.
AcceptedOperation , of type DataPackageOperation , that specifies which operations are allowed by the drop
target.
For information about the DataPackageOperation enumeration, see Handle the DragOver event.
The DropEventArgs class that accompanies the Drop event defines the following properties:
, of type DataPackageView , which is a read-only version of the data package.
Data
Handled , of type bool , indicates whether the event handler has handled the event or whether .NET MAUI
should continue its own processing.
The following XAML example shows a DropGestureRecognizer attached to an Image :
<Image BackgroundColor="Silver"
HeightRequest="300"
WidthRequest="250">
<Image.GestureRecognizers>
<DropGestureRecognizer />
</Image.GestureRecognizers>
</Image>
In this example, when a drag source is dropped on the Image drop target, the drag source will be copied to the
drop target, provided that the drag source is an ImageSource . This occurs because .NET MAUI automatically
copies dragged images, and text, to compatible drop targets.
IMPORTANT
When a DragEventArgs object is created, the AcceptedOperation property defaults to DataPackageOperation.Copy .
The following XAML example shows a DropGestureRecognizer that registers a handler for the DragOver event:
<Image BackgroundColor="Silver"
HeightRequest="300"
WidthRequest="250">
<Image.GestureRecognizers>
<DropGestureRecognizer DragOver="OnDragOver" />
</Image.GestureRecognizers>
</Image>
In this example, the DropGestureRecognizer is attached to an Image object. The DragOver event is raised when a
drag source is dragged over the drop target, but hasn't been dropped, which executes the OnDragOver event
handler:
void OnDragOver(object sender, DragEventArgs e)
{
e.AcceptedOperation = DataPackageOperation.None;
}
The following table shows the properties that are set, and any conversion that's attempted, when a text-based
drag source is dropped on a text control:
C O N T RO L P RO P ERT Y C O N VERSIO N
Editor Text
Entry Text
Label Text
For content other than text and images, you'll need to process the data package yourself.
The DropEventArgs class that accompanies the Drop event defines a Data property, of type DataPackageView .
This property represents a read-only version of the data package.
Retrieve image or text data
Image or text data can be retrieved from a data package in the handler for the Drop event, using methods
defined in the DataPackageView class.
The DataPackageView class includes GetImageAsync and GetTextAsync methods. The GetImageAsync method
retrieves an image from the data package, that was stored in the DataPackage.Image property, and returns
Task<ImageSource> . Similarly, the GetTextAsync method retrieves text from the data package, that was stored in
the DataPackage.Text property, and returns Task<string> .
The following example shows a Drop event handler that retrieves text from the data package for a Path :
In this example, text data is retrieved from the data package using the GetTextAsync method. An action based on
the text value can then be taken.
Retrieve data from the property bag
Any data can be retrieved from a data package in the handler for the Drop event, by accessing the Properties
collection of the data package.
The class defines a Properties property, of type DataPackagePropertySetView . The
DataPackageView
DataPackagePropertySetView class represents a read-only property bag stored as a Dictionary<string, object> .
The following example shows a Drop event handler that retrieves data from the property bag of a data package
for a Rectangle :
In this example, the Square object is retrieved from the property bag of the data package, by specifying the
"Square" dictionary key. An action based on the retrieved value can then be taken.
Recognize a pan gesture
3/18/2022 • 3 minutes to read • Edit Online
A .NET Multi-platform App UI (.NET MAUI) pan gesture recognizer detects the movement of fingers around the
screen and can be used to apply that movement to content. A typical scenario for the pan gesture is to
horizontally and vertically pan an image, so that all of the image content can be viewed when it's being
displayed in a viewport smaller than the image dimensions. This is accomplished by moving the image within
the viewport.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
In .NET MAUI, pan gesture recognition is provided by the PanGestureRecognizer class. This class defines the
TouchPoints property, of type int , which represents the number of touch points in the gesture. The default
value of this property is 1. This property is backed by a BindableProperty object, which means that it can be the
target of data bindings, and styled.
The PanGestureRecognizer class also defines a PanUpdated event that's raised when the detected pan gesture
changes. The PanUpdatedEventArgs object that accompanies this event defines the following properties:
GestureId , of type int , which represents the id of the gesture that raised the event.
StatusType , of type GestureStatus , which indicates if the event has been raised for a newly started gesture,
a running gesture, a completed gesture, or a canceled gesture.
TotalX , of type double , which indicates the total change in the X direction since the beginning of the
gesture.
TotalY , of type double , which indicates the total change in the Y direction since the beginning of the
gesture.
Create a PanGestureRecognizer
To make a View recognize a pan gesture, create a PanGestureRecognizer object, handle the PanUpdated event,
and add the new gesture recognizer to the GestureRecognizers collection on the view. The following code
example shows a PanGestureRecognizer attached to an Image :
<Image Source="monkey.jpg">
<Image.GestureRecognizers>
<PanGestureRecognizer PanUpdated="OnPanUpdated" />
</Image.GestureRecognizers>
</Image>
The code for the OnPanUpdated event handler should be added to the code-behind file:
public PanContainer()
{
// Set PanGestureRecognizer.TouchPoints to control the
// number of touch points needed to pan
PanGestureRecognizer panGesture = new PanGestureRecognizer();
panGesture.PanUpdated += OnPanUpdated;
GestureRecognizers.Add(panGesture);
}
case GestureStatus.Completed:
// Store the translation applied during the pan
x = Content.TranslationX;
y = Content.TranslationY;
break;
}
}
}
In this example, the OnPanUpdated method updates the viewable content of the wrapped view, based on the
user's pan gesture. This is achieved by using the values of the TotalX and TotalY properties of the
PanUpdatedEventArgs instance to calculate the direction and distance of the pan. The
DeviceDisplay.MainDisplayInfo.Width and DeviceDisplay.MainDisplayInfo.Height properties provide the screen
width and screen height values of the device. The wrapped user element is then panned by setting its
TranslationX and TranslationY properties to the calculated values. When panning content in an element that
does not occupy the full screen, the height and width of the viewport can be obtained from the element's
Height and Width properties.
The PanContainer class can be wrapped around a View so that a recognized pan gesture will pan the wrapped
view. The following XAML example shows the PanContainer wrapping an Image :
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:PanGesture"
x:Class="PanGesture.MainPage">
<AbsoluteLayout>
<local:PanContainer>
<Image Source="monkey.jpg" WidthRequest="1024" HeightRequest="768" />
</local:PanContainer>
</AbsoluteLayout>
</ContentPage>
In this example, when the Image receives a pan gesture, the displayed image will be panned.
Recognize a pinch gesture
3/18/2022 • 3 minutes to read • Edit Online
A .NET Multi-platform App UI (.NET MAUI) pinch gesture recognizer is used for performing interactive zoom. A
common scenario for the pinch gesture is to perform interactive zoom of an image at the pinch location. This is
accomplished by scaling the content of the viewport.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
In .NET MAUI, pinch gesture recognition is provided by the PinchGestureRecognizer class, which defines a
PinchUpdated event that's raised when the detected pinch gesture changes. The PinchGestureUpdatedEventArgs
object that accompanies the PinchUpdated event defines the following properties:
Scale , of type double , which indicates the relative size of the pinch gesture since the last update was
received.
ScaleOrigin , of type Point , which indicates the updated origin of the pinch's gesture.
Status , of type GestureStatus , which indicates if the event has been raised for a newly started gesture, a
running gesture, a completed gesture, or a canceled gesture.
Create a PinchGestureRecognizer
To make a View recognize a pinch gesture, create a PinchGestureRecognizer object, handle the PinchUpdated
event, and add the new gesture recognizer to the GestureRecognizers collection on the view. The following code
example shows a PinchGestureRecognizer attached to an Image :
<Image Source="waterfront.jpg">
<Image.GestureRecognizers>
<PinchGestureRecognizer PinchUpdated="OnPinchUpdated" />
</Image.GestureRecognizers>
</Image>
The code for the OnPinchUpdated event handler should be added to the code-behind file:
public PinchToZoomContainer()
{
PinchGestureRecognizer pinchGesture = new PinchGestureRecognizer();
pinchGesture.PinchUpdated += OnPinchUpdated;
GestureRecognizers.Add(pinchGesture);
}
In this example, the OnPinchUpdated method updates the zoom level of the wrapped view, based on the user's
pinch gesture. This is achieved by using the values of the Scale , ScaleOrigin and Status properties of the
PinchGestureUpdatedEventArgs object to calculate the scale factor to be applied at the origin of the pinch gesture.
The wrapped view is then zoomed at the origin of the pinch gesture by setting its TranslationX , TranslationY ,
and Scale properties to the calculated values.
The PinchToZoomContainer class can be wrapped around a View so that a recognized pinch gesture will zoom
the wrapped view. The following XAML example shows the PinchToZoomContainer wrapping an Image :
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:PinchGesture;assembly=PinchGesture"
x:Class="PinchGesture.HomePage">
<Grid>
<local:PinchToZoomContainer>
<Image Source="waterfront.jpg" />
</local:PinchToZoomContainer>
</Grid>
</ContentPage>
In this example, when the Image receives a pinch gesture, the displayed image will be zoomed-in or out.
Recognize a swipe gesture
3/18/2022 • 4 minutes to read • Edit Online
A .NET Multi-platform App UI (.NET MAUI) swipe gesture recognizer detects when a finger is moved across the
screen in a horizontal or vertical direction, and is often used to initiate navigation through content.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
In .NET MAUI, drag gesture recognition is provided by the SwipeGestureRecognizer class. This class defines the
following properties:
Command , of type , which is executed when a swipe gesture is recognized.
ICommand
CommandParameter , of type object , which is the parameter that's passed to the Command .
Direction , of type SwipeDirection , which defines the direction
Threshold , of type uint , which represents the minimum swipe distance that must be achieved for a swipe
to be recognized, in device-independent units. The default value of this property is 100, which means that
any swipes that are less than 100 device-independent units will be ignored.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
The SwipeGestureRecognizer also defines a Swiped event that's raised when a swipe is recognized. The
SwipedEventArgs object that accompanies the Swiped event defines the following properties:
Direction , of type SwipeDirection , indicates the direction of the swipe gesture.
Parameter , of type object , indicates the value passed by the CommandParameter property, if defined.
Create a SwipeGestureRecognizer
To make a View recognize a swipe gesture, create a SwipeGestureRecognizer object, set the Direction property
to a SwipeDirection enumeration value ( Left , Right , Up , or Down ), optionally set the Threshold property,
handle the Swiped event, and add the new gesture recognizer to the GestureRecognizers collection on the view.
The following example shows a SwipeGestureRecognizer attached to a BoxView :
boxView.GestureRecognizers.Add(leftSwipeGesture);
Similarly, swipes that occur on the vertical axis can be recognized by setting the Direction property to Up and
Down :
Alternatively, a SwipeGestureRecognizer for each swipe direction can be created to recognize swipes in every
direction:
boxView.GestureRecognizers.Add(leftSwipeGesture);
boxView.GestureRecognizers.Add(rightSwipeGesture);
boxView.GestureRecognizers.Add(upSwipeGesture);
boxView.GestureRecognizers.Add(downSwipeGesture);
Respond to a swipe
A recognized swipe can be responded to by a handler for the Swiped event:
void OnSwiped(object sender, SwipedEventArgs e)
{
switch (e.Direction)
{
case SwipeDirection.Left:
// Handle the swipe
break;
case SwipeDirection.Right:
// Handle the swipe
break;
case SwipeDirection.Up:
// Handle the swipe
break;
case SwipeDirection.Down:
// Handle the swipe
break;
}
}
The SwipedEventArgs can be examined to determine the direction of the swipe, with custom logic responding to
the swipe as required. The direction of the swipe can be obtained from the Direction property of the event
arguments, which will be set to one of the values of the SwipeDirection enumeration. In addition, the event
arguments also have a Parameter property that will be set to the value of the CommandParameter property, if
defined.
public SwipeContainer()
{
GestureRecognizers.Add(GetSwipeGestureRecognizer(SwipeDirection.Left));
GestureRecognizers.Add(GetSwipeGestureRecognizer(SwipeDirection.Right));
GestureRecognizers.Add(GetSwipeGestureRecognizer(SwipeDirection.Up));
GestureRecognizers.Add(GetSwipeGestureRecognizer(SwipeDirection.Down));
}
The SwipeContainer class creates SwipeGestureRecognizer objects for all four swipe directions, and attaches
Swipe event handlers. These event handlers invoke the Swipe event defined by the SwipeContainer .
The following XAML code example shows the SwipeContainer class wrapping a BoxView :
<StackLayout>
<local:SwipeContainer Swipe="OnSwiped" ...>
<BoxView Color="Teal" ... />
</local:SwipeContainer>
</StackLayout>
In this example, when the BoxView receives a swipe gesture, the Swiped event in the SwipeGestureRecognizer is
raised. This is handled by the SwipeContainer class, which raises its own Swipe event. This Swipe event is
handled on the page. The SwipedEventArgs can then be examined to determine the direction of the swipe, with
custom logic responding to the swipe as required.
The equivalent C# code is:
A .NET Multi-platform App UI (.NET MAUI) tap gesture recognizer is used for tap detection and is implemented
with the TapGestureRecognizer class.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
In .NET MAUI, drag gesture recognition is provided by the SwipeGestureRecognizer class. This class defines the
following properties:
Command , of type , which is executed when a tap is recognized.
ICommand
CommandParameter , of type object , which is the parameter that's passed to the Command .
NumberOfTapsRequired , of type int , which represents the number of taps required to recognize a tap
gesture. The default value of this property is 1.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
The TapGestureRecognizerclass also defines a Tapped event that's raised when a tap is recognized. The
TappedEventArgs object that accompanies the Tapped event defines a Parameter property, of type object , that
indicates the value passed by the CommandParameter property, if defined.
Create a TapGestureRecognizer
To make a View recognize a tap gesture, create a TapGestureRecognizer object, handle the Tapped event, and
add the new gesture recognizer to the GestureRecognizers collection on the view. The following code example
shows a TapGestureRecognizer attached to an Image :
<Image Source="tapped.jpg">
<Image.GestureRecognizers>
<TapGestureRecognizer Tapped="OnTapGestureRecognizerTapped"
NumberOfTapsRequired="2" />
</Image.GestureRecognizers>
</Image>
The code for the OnTapGestureRecognizerTapped event handler should be added to the code-behind file:
By default the Image will respond to single taps. When the NumberOfTapsRequired property is set above one, the
event handler will only be executed if the taps occur within a set period of time. If the second (or subsequent)
taps do not occur within that period they are effectively ignored.
Bindable properties
3/18/2022 • 7 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) bindable properties extend Common Language Runtime (CLR)
property functionality by backing a property with a BindableProperty type, instead of with a field. The purpose
of bindable properties is to provide a property system that supports data binding, styles, templates, and values
set through parent-child relationships. In addition, bindable properties can provide default values, validation of
property values, and callbacks that monitor property changes.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
In .NET MAUI apps, properties should be implemented as bindable properties to support one or more of the
following features:
Acting as a valid target property for data binding.
Setting the property through a style.
Providing a default property value that's different from the default for the type of the property.
Validating the value of the property.
Monitoring property changes.
Examples of .NET MAUI bindable properties include Label.Text , Button.BorderRadius , and
StackLayout.Orientation . Each bindable property has a corresponding public static readonly field of type
BindableProperty that is exposed on the same class and that is the identifier of the bindable property. For
example, the corresponding bindable property identifier for the Label.Text property is Label.TextProperty .
IMPORTANT
The naming convention for bindable properties is that the bindable property identifier must match the property name
specified in the Create method, with "Property" appended to it.
The following code shows an example of a bindable property, with an identifier and values for the four required
parameters:
This creates a BindableProperty instance named IsExpandedProperty , of type bool . The property is owned by
the Expander class, and has a default value of false .
Optionally, when creating a BindableProperty instance, the following parameters can be specified:
The binding mode. This is used to specify the direction in which property value changes will propagate. In the
default binding mode, changes will propagate from the source to the target.
A validation delegate that will be invoked when the property value is set. For more information, see
Validation callbacks.
A property changed delegate that will be invoked when the property value has changed. For more
information, see Detect property changes.
A property changing delegate that will be invoked when the property value will change. This delegate has the
same signature as the property changed delegate.
A coerce value delegate that will be invoked when the property value has changed. For more information,
see Coerce value callbacks.
A Func that's used to initialize a default property value. For more information, see Create a default value
with a Func.
Create accessors
Property accessors are required to use property syntax to access a bindable property. The Get accessor should
return the value that's contained in the corresponding bindable property. This can be achieved by calling the
GetValue method, passing in the bindable property identifier on which to get the value, and then casting the
result to the required type. The Set accessor should set the value of the corresponding bindable property. This
can be achieved by calling the SetValue method, passing in the bindable property identifier on which to set the
value, and the value to set.
The following code example shows accessors for the IsExpanded bindable property:
public bool IsExpanded
{
get => (bool)GetValue(IsExpandedProperty);
set => SetValue(IsExpandedProperty, value);
}
The namespace declaration is used when setting the IsExpanded bindable property, as demonstrated in the
following XAML code example:
<Expander IsExpanded="true">
...
</Expander>
Advanced scenarios
When creating a BindableProperty instance, there are a number of optional parameters that can be set to
enable advanced bindable property scenarios. This section explores these scenarios.
Detect property changes
A static property-changed callback method can be registered with a bindable property by specifying the
propertyChanged parameter for the BindableProperty.Create method. The specified callback method will be
invoked when the value of the bindable property changes.
The following code example shows how the EventName bindable property registers the OnEventNameChanged
method as a property-changed callback method:
public static readonly BindableProperty IsExpandedProperty =
BindableProperty.Create(nameof(IsExpanded), typeof(bool), typeof(Expander), false, propertyChanged:
OnIsExpandedChanged);
...
In the property-changed callback method, the BindableObject parameter is used to denote which instance of
the owning class has reported a change, and the values of the two object parameters represent the old and
new values of the bindable property.
Validation callbacks
A static validation callback method can be registered with a bindable property by specifying the
validateValue parameter for the BindableProperty.Create method. The specified callback method will be
invoked when the value of the bindable property is set.
The following code example shows how the Angle bindable property registers the IsValidValue method as a
validation callback method:
Validation callbacks are provided with a value, and should return true if the value is valid for the property,
otherwise false . An exception will be raised if a validation callback returns false , which you should be
handled. A typical use of a validation callback method is constraining the values of integers or doubles when the
bindable property is set. For example, the IsValidValue method checks that the property value is a double
within the range 0 to 360.
Coerce value callbacks
A static coerce value callback method can be registered with a bindable property by specifying the
coerceValue parameter for the BindableProperty.Create method. The specified callback method will be invoked
when the value of the bindable property changes.
IMPORTANT
The BindableObject type has a CoerceValue method that can be called to force a reevaluation of the value of its
BindableProperty argument, by invoking its coerce value callback.
Coerce value callbacks are used to force a reevaluation of a bindable property when the value of the property
changes. For example, a coerce value callback can be used to ensure that the value of one bindable property is
not greater than the value of another bindable property.
The following code example shows how the Angle bindable property registers the CoerceAngle method as a
coerce value callback method:
public static readonly BindableProperty AngleProperty =
BindableProperty.Create("Angle", typeof(double), typeof(MainPage), 0.0, coerceValue: CoerceAngle);
public static readonly BindableProperty MaximumAngleProperty =
BindableProperty.Create("MaximumAngle", typeof(double), typeof(MainPage), 360.0, propertyChanged:
ForceCoerceValue);
...
return input;
}
The CoerceAngle method checks the value of the MaximumAngle property, and if the Angle property value is
greater than it, it coerces the value to the MaximumAngle property value. In addition, when the MaximumAngle
property changes the coerce value callback is invoked on the Angle property by calling the CoerceValue
method.
Create a default value with a Func
A Func can be used to initialize the default value of a bindable property, as demonstrated in the following
example:
The defaultValueCreator parameter is set to a Func that returns a DateTime that represents today's date.
Attached properties
3/18/2022 • 4 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) attached properties enable an object to assign a value for a property
that its own class doesn't define. For example, child elements can use attached properties to inform their parent
element of how they are to be presented in the user interface. The Grid layout enables the row and column of a
child to be specified by setting the Grid.Row and Grid.Column attached properties. Grid.Row and Grid.Column
are attached properties because they are set on elements that are children of a Grid , rather than on the Grid
itself.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
An attached property can be created by declaring a public static readonly property of type BindableProperty .
The bindable property should be set to the returned value of one of the BindableProperty.CreateAttached
method overloads. The declaration should be within the body of the owning class, but outside of any member
definitions.
IMPORTANT
The naming convention for attached properties is that the attached property identifier must match the property name
specified in the CreateAttached method, with "Property" appended to it.
This creates an attached property named HasShadowProperty , of type bool . The property is owned by the
Shadow class, and has a default value of false .
For more information about creating bindable properties, including parameters that can be specified during
creation, see Create a bindable property.
Create accessors
Static Get PropertyName and Set PropertyName methods are required as accessors for the attached property,
otherwise the property system will be unable to use the attached property. The Get PropertyName accessor
should conform to the following signature:
The Get PropertyName accessor should return the value that's contained in the corresponding
BindableProperty field for the attached property. This can be achieved by calling the GetValue method, passing
in the bindable property identifier on which to get the value, and then casting the resulting value to the required
type.
The Set PropertyName accessor should conform to the following signature:
The Set PropertyName accessor should set the value of the corresponding BindableProperty field for the
attached property. This can be achieved by calling the SetValue method, passing in the bindable property
identifier on which to set the value, and the value to set.
For both accessors, the target object should be of, or derive from, BindableObject .
The following code example shows accessors for the HasShadow attached property:
The namespace declaration is then used when setting the attached property on a specific control, as
demonstrated in the following XAML:
The Stylecan be applied to a Label by setting its Style property to the Style instance using the
StaticResource markup extension, as demonstrated in the following code example:
Advanced scenarios
When creating an attached property, there are a number of optional parameters that can be set to enable
advanced attached property scenarios. This includes detecting property changes, validating property values, and
coercing property values. For more information, see Advanced scenarios.
Publish and subscribe to messages
3/18/2022 • 4 minutes to read • Edit Online
The publish-subscribe pattern is a messaging pattern in which publishers send messages without having
knowledge of any receivers, known as subscribers. Similarly, subscribers listen for specific messages, without
having knowledge of any publishers.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Events in .NET implement the publish-subscribe pattern, and are the most simple and straightforward approach
for a communication layer between components if loose coupling is not required, such as a control and the page
that contains it. However, the publisher and subscriber lifetimes are coupled by object references to each other,
and the subscriber type must have a reference to the publisher type. This can create memory management
issues, especially when there are short lived objects that subscribe to an event of a static or long-lived object. If
the event handler isn't removed, the subscriber will be kept alive by the reference to it in the publisher, and this
will prevent or delay the garbage collection of the subscriber.
The .NET Multi-platform App UI (.NET MAUI) MessagingCenter class implements the publish-subscribe pattern,
allowing message-based communication between components that are inconvenient to link by object and type
references. This mechanism allows publishers and subscribers to communicate without having a reference to
each other, helping to reduce dependencies between them.
The MessagingCenter class provides multicast publish-subscribe functionality. This means that there can be
multiple publishers that publish a single message, and there can be multiple subscribers listening for the same
message:
Publishers send messages using the MessagingCenter.Send method, while subscribers listen for messages using
the MessagingCenter.Subscribe method. In addition, subscribers can also unsubscribe from message
subscriptions, if required, with the MessagingCenter.Unsubscribe method.
IMPORTANT
Internally, the MessagingCenter class uses weak references. This means that it will not keep objects alive, and will allow
them to be garbage collected. Therefore, it should only be necessary to unsubscribe from a message when a class no
longer wishes to receive the message.
Publish a message
MessagingCenter messages are strings. Publishers notify subscribers of a message with one of the
MessagingCenter.Send overloads. The following code example publishes a Hi message:
MessagingCenter.Send<MainPage>(this, "Hi");
In this example the Send method specifies a generic argument that represents the sender. To receive the
message, a subscriber must also specify the same generic argument, indicating that they are listening for a
message from that sender. In addition, this example specifies two method arguments:
The first argument specifies the sender instance.
The second argument specifies the message.
Payload data can also be sent with a message:
In this example, the Send method specifies two generic arguments. The first is the type that's sending the
message, and the second is the type of the payload data being sent. To receive the message, a subscriber must
also specify the same generic arguments. This enables multiple messages that share a message identity but
send different payload data types to be received by different subscribers. In addition, this example specifies a
third method argument that represents the payload data to be sent to the subscriber. In this case the payload
data is a string .
The Send method will publish the message, and any payload data, using a fire-and-forget approach. Therefore,
the message is sent even if there are no subscribers registered to receive the message. In this situation, the sent
message is ignored.
Subscribe to a message
Subscribers can register to receive a message using one of the MessagingCenter.Subscribe overloads. The
following code example shows an example of this:
In this example, the Subscribe method subscribes the this object to Hi messages that are sent by the
MainPage type, and executes a callback delegate in response to receiving the message. The callback delegate,
specified as a lambda expression, could be code that updates the UI, saves some data, or triggers some other
operation.
NOTE
A subscriber might not need to handle every instance of a published message, and this can be controlled by the generic
type arguments that are specified on the Subscribe method.
The following example shows how to subscribe to a message that contains payload data:
MessagingCenter.Subscribe<MainPage, string>(this, "Hi", async (sender, arg) =>
{
await DisplayAlert("Message received", "arg=" + arg, "OK");
});
In this example, the Subscribe method subscribes to Hi messages that are sent by the MainPage type, whose
payload data is a string . A callback delegate is executed in response to receiving such a message, that displays
the payload data in an alert.
IMPORTANT
The delegate that's executed by the Subscribe method will be executed on the same thread that publishes the message
using the Send method.
MessagingCenter.Unsubscribe<MainPage>(this, "Hi");
In this example, the Unsubscribe method unsubscribes the this object from the Hi message sent by the
MainPage type.
Messages containing payload data should be unsubscribed from using the Unsubscribe overload that specifies
two generic arguments:
In this example, the Unsubscribe method unsubscribes the this object from the Hi message sent by the
MainPage type, whose payload data is a string .
Resource dictionaries
3/18/2022 • 8 minutes to read • Edit Online
A .NET Multi-platform App UI (.NET MAUI) ResourceDictionary is a repository for resources that are used by a
.NET MAUI app. Typical resources that are stored in a ResourceDictionary include styles, control templates, data
templates, converters, and colors.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
XAML resources that are stored in a ResourceDictionary can be referenced and applied to elements by using the
StaticResource or DynamicResource markup extension. In C#, resources can also be defined in a
ResourceDictionary and then referenced and applied to elements by using a string-based indexer. However,
there's little advantage to using a ResourceDictionary in C#, as shared objects can be stored as fields or
properties, and accessed directly without having to first retrieve them from a dictionary.
Create resources
Every VisualElement derived object has a Resources property, which is a ResourceDictionary that can contain
resources. Similarly, an Application derived object has a Resources property, which is a ResourceDictionary
that can contain resources.
A .NET MAUI app can contain only a single class that derives from Application , but often makes use of many
classes that derive from VisualElement , including pages, layouts, and views. Any of these objects can have its
Resources property set to a ResourceDictionary containing resources. Choosing where to put a particular
ResourceDictionary impacts where the resources can be used:
Resources in a ResourceDictionary that is attached to a view, such as Button or Label , can only be applied
to that particular object.
Resources in a ResourceDictionary attached to a layout, such as StackLayout or Grid , can be applied to the
layout and all the children of that layout.
Resources in a ResourceDictionary defined at the page level can be applied to the page and to all its children.
Resources in a ResourceDictionary defined at the application level can be applied throughout the app.
With the exception of implicit styles, each resource in resource dictionary must have a unique string key that's
defined with the x:Key attribute.
The following XAML shows resources defined in an application level ResourceDictionary in the App.xaml file:
<Application xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="ResourceDictionaryDemo.App">
<Application.Resources>
<Thickness x:Key="PageMargin">20</Thickness>
<Style TargetType="ContentPage"
ApplyToDerivedTypes="True">
<Setter Property="BackgroundColor"
Value="{StaticResource AppBackgroundColor}" />
</Style>
</Application.Resources>
</Application>
In this example, the resource dictionary defines a Thickness resource, multiple Color resources, and two
implicit Style resources.
IMPORTANT
Inserting resources directly between the Resources property-element tags automatically creates a
ResourceDictionary object. However, it's also valid to place all resources between optional ResourceDictionary tags.
Consume resources
Each resource has a key that is specified using the x:Key attribute, which becomes its dictionary key in the
ResourceDictionary . The key is used to reference a resource from the ResourceDictionary with the
StaticResource or DynamicResource XAML markup extension.
The StaticResource markup extension is similar to the DynamicResource markup extension in that both use a
dictionary key to reference a value from a resource dictionary. However, while the StaticResource markup
extension performs a single dictionary lookup, the DynamicResource markup extension maintains a link to the
dictionary key. Therefore, if the dictionary entry associated with the key is replaced, the change is applied to the
visual element. This enables runtime resource changes to be made in an app. For more information about
markup extensions, see XAML markup extensions.
The following XAML example shows how to consume resources, and also define an additional resource in a
StackLayout :
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="ResourceDictionaryDemo.MainPage"
Title="Main page">
<StackLayout Margin="{StaticResource PageMargin}"
Spacing="6">
<StackLayout.Resources>
<!-- Implicit style -->
<Style TargetType="Button">
<Setter Property="FontSize" Value="Medium" />
<Setter Property="BackgroundColor" Value="#1976D2" />
<Setter Property="TextColor" Value="White" />
<Setter Property="CornerRadius" Value="5" />
</Style>
</StackLayout.Resources>
<Label Text="This app demonstrates consuming resources that have been defined in resource
dictionaries." />
<Button Text="Navigate"
Clicked="OnNavigateButtonClicked" />
</StackLayout>
</ContentPage>
In this example, the ContentPage object consumes the implicit style defined in the application level resource
dictionary. The StackLayout object consumes the PageMargin resource defined in the application level resource
dictionary, while the Button object consumes the implicit style defined in the StackLayout resource dictionary.
This results in the appearance shown in the following screenshot:
IMPORTANT
Resources that are specific to a single page shouldn't be included in an application level resource dictionary, as such
resources will then be parsed at app startup instead of when required by a page.
The requested key is checked for in the resource dictionary, if it exists, for the element that sets the property.
If the requested key is found, its value is returned and the lookup process terminates.
If a match isn't found, the lookup process searches the visual tree upwards, checking the resource dictionary
of each parent element. If the requested key is found, its value is returned and the lookup process terminates.
Otherwise the process continues upwards until the root element is reached.
If a match isn't found at the root element, the application level resource dictionary is examined.
If a match still isn't found, a XamlParseException is thrown.
Therefore, when the XAML parser encounters a StaticResource or DynamicResource markup extension, it
searches for a matching key by traveling up through the visual tree, using the first match it finds. If this search
ends at the page and the key still hasn't been found, the XAML parser searches the ResourceDictionary attached
to the App object. If the key still isn't found, an exception is thrown.
Override resources
When resources share keys, resources defined lower in the visual tree will take precedence over those defined
higher up. For example, setting an AppBackgroundColor resource to AliceBlue at the application level will be
overridden by a page level AppBackgroundColor resource set to Teal . Similarly, a page level AppBackgroundColor
resource will be overridden by a layout or view level AppBackgroundColor resource.
NOTE
A stand-alone ResourceDictionary must have a build action of MauiXaml.
<ResourceDictionary xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml">
<DataTemplate x:Key="PersonDataTemplate">
<ViewCell>
<Grid RowSpacing="6"
ColumnSpacing="6">
<Grid.ColumnDefinitions>
<ColumnDefinition Width="0.5*" />
<ColumnDefinition Width="0.2*" />
<ColumnDefinition Width="0.3*" />
</Grid.ColumnDefinitions>
<Label Text="{Binding Name}"
TextColor="{StaticResource NormalTextColor}"
FontAttributes="Bold" />
<Label Grid.Column="1"
Text="{Binding Age}"
TextColor="{StaticResource NormalTextColor}" />
<Label Grid.Column="2"
Text="{Binding Location}"
TextColor="{StaticResource NormalTextColor}"
HorizontalTextAlignment="End" />
</Grid>
</ViewCell>
</DataTemplate>
</ResourceDictionary>
In this example, the ResourceDictionary contains a single resource, which is an object of type DataTemplate .
MyResourceDictionar y.xaml can be consumed by merging it into another resource dictionary.
<ContentPage ...>
<ContentPage.Resources>
<!-- Add more resources here -->
<ResourceDictionary Source="MyResourceDictionary.xaml" />
<!-- Add more resources here -->
</ContentPage.Resources>
...
</ContentPage>
This syntax does not instantiate the MyResourceDictionary class. Instead, it references the XAML file. For that
reason, when setting the Source property, a code-behind file isn't required, and the x:Class attribute can be
removed from the root tag of the MyResourceDictionar y.xaml file.
IMPORTANT
The ResourceDictionary.Source property can only be set from XAML.
WARNING
The ResourceDictionary class also defines a MergedWith property. However, this property has been deprecated and
should no longer be used.
The following code example shows two resource dictionaries being added to the MergedDictionaries collection
of a page level ResourceDictionary :
<ContentPage ...
xmlns:local="clr-namespace:ResourceDictionaryDemo"
xmlns:theme="clr-namespace:MyThemes;assembly=MyThemes">
<ContentPage.Resources>
<ResourceDictionary>
<!-- Add more resources here -->
<ResourceDictionary.MergedDictionaries>
<!-- Add more resource dictionaries here -->
<local:MyResourceDictionary />
<theme:DefaultTheme />
<!-- Add more resource dictionaries here -->
</ResourceDictionary.MergedDictionaries>
<!-- Add more resources here -->
</ResourceDictionary>
</ContentPage.Resources>
...
</ContentPage>
In this example, a resource dictionary from the same assembly, and a resource dictionary from an external
assembly, are merged into the page level resource dictionary. In addition, you can also add other
ResourceDictionary objects within the MergedDictionaries property-element tags, and other resources outside
of those tags.
IMPORTANT
There can be only one MergedDictionaries property-element tag in a ResourceDictionary , but you can put as many
ResourceDictionary objects in there as required.
When merged ResourceDictionary resources share identical x:Key attribute values, .NET MAUI uses the
following resource precedence:
1. The resources local to the resource dictionary.
2. The resources contained in the resource dictionaries that were merged via the MergedDictionaries collection,
in the reverse order they are listed in the MergedDictionaries property.
TIP
Searching resource dictionaries can be a computationally intensive task if an app contains multiple, large resource
dictionaries. Therefore, to avoid unnecessary searching, you should ensure that each page in an application only uses
resource dictionaries that are appropriate to the page.
Target multiple platforms from a single project
3/18/2022 • 7 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) single project takes the platform-specific development experiences you
typically encounter while developing apps and abstracts them into a single shared project that can target
Android, iOS, macOS, and Windows.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
.NET MAUI single project provides a simplified and consistent cross-platform development experience,
regardless of the platforms being targeted. .NET MAUI single project provides the following features:
A single shared project that can target Android, iOS, macOS, and Windows.
A simplified debug target selection for running your .NET MAUI apps.
Shared resource files within the single project.
Access to platform-specific APIs and tools when required.
A single cross-platform app entry point.
.NET MAUI single project is enabled using multi-targeting and the use of SDK-style projects in .NET 6.
Resource files
Resource management for cross-platform app development has traditionally been problematic. Each platform
has its own approach to managing resources, that must be implemented on each platform. For example, each
platform has differing image requirements that typically involves creating multiple versions of each image at
different resolutions. Therefore, a single image typically has to be duplicated multiple times per platform, at
different resolutions, with the resulting images having to use different filename and folder conventions on each
platform.
.NET MAUI single project enables resource files to be stored in a single location while being consumed on each
platform. This includes fonts, images, the app icon, the splash screen, raw assets, and CSS files for styling .NET
MAUI apps.
IMPORTANT
Each image resource file is used as a source image, from which images of the required resolutions are generated for each
platform at build time.
Resource files should typically be placed in the Resources folder of your .NET MAUI app project, or child folders
of the Resources folder, and must have their build action set correctly. The following table shows the build
actions for each resource file type:
Fonts MauiFont
Images MauiImage
NOTE
XAML files are also stored in your .NET MAUI app project, and are automatically assigned the MauiXaml build action
when created by project and item templates. However, XAML files will not typically be located in the Resources folder of
the app project.
When a resource file is added to a .NET MAUI app project, a corresponding entry for the resource is created in
the project (.csproj) file, with the exception of CSS files. After adding a resource file, its build action can be set in
the Proper ties window. The following screenshot shows a Resources folder containing image and font
resources in child folders:
Child folders of the Resources folder can be designated for each resource type by editing the project file for your
app:
<ItemGroup>
<!-- Images -->
<MauiImage Include="Resources\Images\*" />
The wildcard character ( * ) indicates that all the files within the folder will be treated as being of the specified
resource type. In addition, it's possible to include all files from child folders:
<ItemGroup>
<!-- Images -->
<MauiImage Include="Resources\Images\**\*" />
</ItemGroup>
In this example, the double wildcard character ('**') specifies that the Images folder can contain child folders.
Therefore, <MauiImage Include="Resources\Images\**\*" /> specifies that any files in the Resources\Images folder,
or any child folders of the Images folder, will be used as source images from which images of the required
resolution are generated for each platform.
Platform-specific resources will override their shared resource counterparts. For example, if you have an
Android-specific image located at Platforms\Android\Resources\drawable-xhdpi\logo.png, and you also provide
a shared Resources\Images\logo.svg image, the Scalable Vector Graphics (SVG) file will be used to generate the
required Android images, except for the XHDPI image that already exists as a platform-specific image.
App icons
An app icon can be added to your app project by dragging an image into the Resources\Images folder of the
project, and setting the build action of the icon to MauiIcon in the Proper ties window. This creates a
corresponding entry in your project file:
At build time, the app icon is resized to the correct sizes for the target platform and device. The resized app icons
are then added to your app package. App icons are resized to multiple resolutions because they have multiple
uses, including being used to represent the app on the device, and in the app store.
Images
Images can be added to your app project by dragging them to the Resources\Images folder of your project, and
setting their build action to MauiImage in the Proper ties window. This creates a corresponding entry per
image in your project file:
At build time, images are resized to the correct resolutions for the target platform and device. The resized
images are then added to your app package.
Fonts
True type format (TTF) and open type font (OTF) fonts can be added to your app project by dragging them into
the Resources\Fonts folder of your project, and setting their build action to MauiFont in the Proper ties
window. This creates a corresponding entry per font in your project file:
At build time, the splash screen image is resized to the correct size for the target platform and device. The
resized splash screen is then added to your app package.
For more information, see Splash screens.
Raw assets
Raw asset files, such as HTML, JSON, and videos, can be added to your app project by dragging them into the
Resources folder of your project (or a sub-folder, such as Resources\Raw ), and setting their build action to
MauiAsset in the Proper ties window. This creates a corresponding entry per asset in your project file:
<Application ...>
<Application.Resources>
<StyleSheet Source="/Resources/styles.css" />
</Application.Resources>
</Application>
Platform-specific code
A .NET MAUI app project contains a Platforms folder, with each child folder representing a platform that .NET
MAUI can target:
The folders for each platform contain platform-specific resources, and code and that starts the app on each
platform:
At build time, the build system only includes the code from each folder when building for that specific platform.
For example, when you build for Android the files in the Platforms\Android folder will be built into the app
package, but the files in the other Platforms folders won't be. This approach uses multi-targeting to target
multiple platforms from a single project. Multi-targeting can be combined with partial classes and partial
methods to invoke native platform functionality from cross-platform code. For more information, see Invoke
platform code.
In addition to this default multi-targeting approach, .NET MAUI apps can also be multi-targeted based on your
own filename and folder criteria. This enables you to structure your .NET MAUI app project so that you don't
have to place your platform code into sub-folders of the Platforms folder. For more information, see Configure
multi-targeting.
Multi-targeting can also be combined with conditional compilation so that code is targeted to specific platforms:
#if ANDROID
handler.NativeView.SetBackgroundColor(Colors.Red.ToNative());
#elif IOS
handler.NativeView.BackgroundColor = Colors.Red.ToNative();
handler.NativeView.BorderStyle = UIKit.UITextBorderStyle.Line;
#elif WINDOWS
handler.NativeView.Background = Colors.Red.ToNative();
#endif
.NET Multi-platform App UI (.NET MAUI) control templates enable you to define the visual structure of
ContentView derived custom controls, and ContentPage derived pages. Control templates separate the user
interface (UI) for a custom control, or page, from the logic that implements the control or page. Additional
content can also be inserted into the templated custom control, or templated page, at a pre-defined location.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
For example, a control template can be created that redefines the UI provided by a custom control. The control
template can then be consumed by the required custom control instance. Alternatively, a control template can be
created that defines any common UI that will be used by multiple pages in an app. The control template can then
be consumed by multiple pages, with each page still displaying its unique content.
Create a ControlTemplate
The following example shows the code for a CardView custom control:
The CardView class, which derives from the ContentView class, represents a custom control that displays data in
a card-like layout. The class contains properties, which are backed by bindable properties, for the data it
displays. However, the CardView class does not define any UI. Instead, the UI will be defined with a control
template. For more information about creating ContentView derived custom controls, see ContentView.
A control template is created with the ControlTemplate type. When you create a ControlTemplate , you combine
View objects to build the UI for a custom control, or page. A ControlTemplate must have only one View as its
root element. However, the root element usually contains other View objects. The combination of objects makes
up the control's visual structure.
While a ControlTemplate can be defined inline, the typical approach to declaring a ControlTemplate is as a
resource in a resource dictionary. Because control templates are resources, they obey the same scoping rules
that apply to all resources. For example, if you declare a control template in your app-level resource dictionary,
the template can be used anywhere in your app. If you define the template in a page, only that page can use the
control template. For more information about resources, see Resource dictionaries.
The following XAML example shows a ControlTemplate for CardView objects:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
...>
<ContentPage.Resources>
<ControlTemplate x:Key="CardViewControlTemplate">
<Frame BindingContext="{Binding Source={RelativeSource TemplatedParent}}"
BackgroundColor="{Binding CardColor}"
BorderColor="{Binding BorderColor}"
CornerRadius="5"
HasShadow="True"
Padding="8"
HorizontalOptions="Center"
VerticalOptions="Center">
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="75" />
<RowDefinition Height="4" />
<RowDefinition Height="Auto" />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="75" />
<ColumnDefinition Width="200" />
</Grid.ColumnDefinitions>
<Frame IsClippedToBounds="True"
BorderColor="{Binding BorderColor}"
BackgroundColor="{Binding IconBackgroundColor}"
CornerRadius="38"
HeightRequest="60"
WidthRequest="60"
HorizontalOptions="Center"
VerticalOptions="Center">
<Image Source="{Binding IconImageSource}"
Margin="-20"
WidthRequest="100"
HeightRequest="100"
Aspect="AspectFill" />
</Frame>
<Label Grid.Column="1"
Text="{Binding CardTitle}"
FontAttributes="Bold"
FontSize="Large"
VerticalTextAlignment="Center"
HorizontalTextAlignment="Start" />
<BoxView Grid.Row="1"
Grid.ColumnSpan="2"
BackgroundColor="{Binding BorderColor}"
HeightRequest="2"
HorizontalOptions="Fill" />
<Label Grid.Row="2"
Grid.ColumnSpan="2"
Text="{Binding CardDescription}"
VerticalTextAlignment="Start"
VerticalOptions="Fill"
HorizontalOptions="Fill" />
</Grid>
</Frame>
</ControlTemplate>
</ContentPage.Resources>
...
</ContentPage>
When a ControlTemplate is declared as a resource, it must have a key specified with the x:Key attribute so that
it can be identified in the resource dictionary. In this example, the root element of the CardViewControlTemplate
is a Frame object. The Frame object uses the RelativeSource markup extension to set its BindingContext to the
runtime object instance to which the template will be applied, which is known as the templated parent. The
Frame object uses a combination of Grid , Frame , Image , Label , and BoxView objects to define the visual
structure of a CardView object. The binding expressions of these objects resolve against CardView properties,
due to inheriting the BindingContext from the root Frame element. For more information about the
RelativeSource markup extension, see Relative bindings.
Consume a ControlTemplate
A ControlTemplate can be applied to a ContentView derived custom control by setting its ControlTemplate
property to the control template object. Similarly, a ControlTemplate can be applied to a ContentPage derived
page by setting its ControlTemplate property to the control template object. At runtime, when a
ControlTemplate is applied, all of the controls that are defined in the ControlTemplate are added to the visual
tree of the templated custom control, or templated page.
The following example shows the CardViewControlTemplate being assigned to the ControlTemplate property of
each CardView object:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:controls="clr-namespace:ControlTemplateDemos.Controls"
...>
<StackLayout Margin="30">
<controls:CardView BorderColor="DarkGray"
CardTitle="John Doe"
CardDescription="Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nulla
elit dolor, convallis non interdum."
IconBackgroundColor="SlateGray"
IconImageSource="user.png"
ControlTemplate="{StaticResource CardViewControlTemplate}" />
<controls:CardView BorderColor="DarkGray"
CardTitle="Jane Doe"
CardDescription="Phasellus eu convallis mi. In tempus augue eu dignissim
fermentum. Morbi ut lacus vitae eros lacinia."
IconBackgroundColor="SlateGray"
IconImageSource="user.png"
ControlTemplate="{StaticResource CardViewControlTemplate}" />
</StackLayout>
</ContentPage>
In this example, the controls in the CardViewControlTemplate become part of the visual tree for each CardView
object. Because the root Frame object for the control template sets its BindingContext to the templated parent,
the Frame and its children resolve their binding expressions against the properties of each CardView object.
The following screenshot shows the CardViewControlTemplate applied to the three CardView objects:
IMPORTANT
The point in time that a ControlTemplate is applied to a control instance can be detected by overriding the
OnApplyTemplate method in the templated custom control, or templated page. For more information, see Get a named
element from a template.
IMPORTANT
The TemplateBinding markup expression enables the RelativeSource binding from the previous control template to
be removed, and replaces the Binding expressions.
The ContentProperty for the TemplateBinding markup extension is Path . Therefore, the "Path=" part of the
markup extension can be omitted if the path is the first item in the TemplateBinding expression. For more
information about using these properties in a binding expression, see Data binding.
WARNING
The TemplateBinding markup extension should only be used within a ControlTemplate . However, attempting to use a
TemplateBinding expression outside of a ControlTemplate will not result in a build error or an exception being
thrown.
The following XAML example shows a ControlTemplate for CardView objects, that uses the TemplateBinding
markup extension:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
...>
<ContentPage.Resources>
<ControlTemplate x:Key="CardViewControlTemplate">
<Frame BackgroundColor="{TemplateBinding CardColor}"
BorderColor="{TemplateBinding BorderColor}"
CornerRadius="5"
HasShadow="True"
Padding="8"
HorizontalOptions="Center"
VerticalOptions="Center">
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="75" />
<RowDefinition Height="4" />
<RowDefinition Height="Auto" />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="75" />
<ColumnDefinition Width="200" />
</Grid.ColumnDefinitions>
<Frame IsClippedToBounds="True"
BorderColor="{TemplateBinding BorderColor}"
BackgroundColor="{TemplateBinding IconBackgroundColor}"
CornerRadius="38"
HeightRequest="60"
WidthRequest="60"
HorizontalOptions="Center"
VerticalOptions="Center">
<Image Source="{TemplateBinding IconImageSource}"
Margin="-20"
WidthRequest="100"
HeightRequest="100"
Aspect="AspectFill" />
</Frame>
<Label Grid.Column="1"
Text="{TemplateBinding CardTitle}"
FontAttributes="Bold"
FontSize="Large"
VerticalTextAlignment="Center"
HorizontalTextAlignment="Start" />
<BoxView Grid.Row="1"
Grid.ColumnSpan="2"
BackgroundColor="{TemplateBinding BorderColor}"
HeightRequest="2"
HorizontalOptions="Fill" />
<Label Grid.Row="2"
Grid.ColumnSpan="2"
Text="{TemplateBinding CardDescription}"
VerticalTextAlignment="Start"
VerticalOptions="Fill"
HorizontalOptions="Fill" />
</Grid>
</Frame>
</ControlTemplate>
</ContentPage.Resources>
...
</ContentPage>
In this example, the TemplateBinding markup extension resolves binding expressions against the properties of
each CardView object. The following screenshot shows the CardViewControlTemplate applied to the CardView
objects:
IMPORTANT
Using the TemplateBinding markup extension is equivalent to setting the BindingContext of the root element in the
template to its templated parent with the RelativeSource markup extension, and then resolving bindings of child
objects with the Binding markup extension. In fact, the TemplateBinding markup extension creates a Binding
whose Source is RelativeBindingSource.TemplatedParent .
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:controls="clr-namespace:ControlTemplateDemos.Controls"
...>
<ContentPage.Resources>
<ControlTemplate x:Key="CardViewControlTemplate">
...
</ControlTemplate>
<Style TargetType="controls:CardView">
<Setter Property="ControlTemplate"
Value="{StaticResource CardViewControlTemplate}" />
</Style>
</ContentPage.Resources>
<StackLayout Margin="30">
<controls:CardView BorderColor="DarkGray"
CardTitle="John Doe"
CardDescription="Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nulla
elit dolor, convallis non interdum."
IconBackgroundColor="SlateGray"
IconImageSource="user.png" />
<controls:CardView BorderColor="DarkGray"
CardTitle="Jane Doe"
CardDescription="Phasellus eu convallis mi. In tempus augue eu dignissim
fermentum. Morbi ut lacus vitae eros lacinia."
IconBackgroundColor="SlateGray"
IconImageSource="user.png"/>
</StackLayout>
</ContentPage>
In this example, the implicit Style is automatically applied to each CardView object, and sets the
ControlTemplate property of each CardView to CardViewControlTemplate .
Redefine a control’s UI
When a ControlTemplate is instantiated and assigned to the ControlTemplate property of a ContentView
derived custom control, or a ContentPage derived page, the visual structure defined for the custom control or
page is replaced with the visual structure defined in the ControlTemplate .
For example, the CardViewUI custom control defines its user interface using the following XAML:
<ContentView xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="ControlTemplateDemos.Controls.CardViewUI"
x:Name="this">
<Frame BindingContext="{x:Reference this}"
BackgroundColor="{Binding CardColor}"
BorderColor="{Binding BorderColor}"
CornerRadius="5"
HasShadow="True"
Padding="8"
HorizontalOptions="Center"
VerticalOptions="Center">
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="75" />
<RowDefinition Height="4" />
<RowDefinition Height="Auto" />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="75" />
<ColumnDefinition Width="200" />
</Grid.ColumnDefinitions>
<Frame IsClippedToBounds="True"
BorderColor="{Binding BorderColor, FallbackValue='Black'}"
BackgroundColor="{Binding IconBackgroundColor, FallbackValue='Gray'}"
CornerRadius="38"
HeightRequest="60"
WidthRequest="60"
HorizontalOptions="Center"
VerticalOptions="Center">
<Image Source="{Binding IconImageSource}"
Margin="-20"
WidthRequest="100"
HeightRequest="100"
Aspect="AspectFill" />
</Frame>
<Label Grid.Column="1"
Text="{Binding CardTitle, FallbackValue='Card title'}"
FontAttributes="Bold"
FontSize="Large"
VerticalTextAlignment="Center"
HorizontalTextAlignment="Start" />
<BoxView Grid.Row="1"
Grid.ColumnSpan="2"
BackgroundColor="{Binding BorderColor, FallbackValue='Black'}"
HeightRequest="2"
HorizontalOptions="Fill" />
<Label Grid.Row="2"
Grid.ColumnSpan="2"
Text="{Binding CardDescription, FallbackValue='Card description'}"
VerticalTextAlignment="Start"
VerticalOptions="Fill"
HorizontalOptions="Fill" />
</Grid>
</Frame>
</ContentView>
However, the controls that comprise this UI can be replaced by defining a new visual structure in a
ControlTemplate , and assigning it to the ControlTemplate property of a CardViewUI object:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
...>
<ContentPage.Resources>
<ControlTemplate x:Key="CardViewCompressed">
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="100" />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="100" />
<ColumnDefinition Width="*" />
</Grid.ColumnDefinitions>
<Image Source="{TemplateBinding IconImageSource}"
BackgroundColor="{TemplateBinding IconBackgroundColor}"
WidthRequest="100"
HeightRequest="100"
Aspect="AspectFill"
HorizontalOptions="Center"
VerticalOptions="Center" />
<StackLayout Grid.Column="1">
<Label Text="{TemplateBinding CardTitle}"
FontAttributes="Bold" />
<Label Text="{TemplateBinding CardDescription}" />
</StackLayout>
</Grid>
</ControlTemplate>
</ContentPage.Resources>
<StackLayout Margin="30">
<controls:CardViewUI BorderColor="DarkGray"
CardTitle="John Doe"
CardDescription="Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nulla
elit dolor, convallis non interdum."
IconBackgroundColor="SlateGray"
IconImageSource="user.png"
ControlTemplate="{StaticResource CardViewCompressed}" />
<controls:CardViewUI BorderColor="DarkGray"
CardTitle="Jane Doe"
CardDescription="Phasellus eu convallis mi. In tempus augue eu dignissim
fermentum. Morbi ut lacus vitae eros lacinia."
IconBackgroundColor="SlateGray"
IconImageSource="user.png"
ControlTemplate="{StaticResource CardViewCompressed}" />
</StackLayout>
</ContentPage>
In this example, the visual structure of the CardViewUI object is redefined in a ControlTemplate that provides a
more compact visual structure that's suitable for a condensed list:
The following XAML shows a control template named TealTemplate that contains a ContentPresenter in its
visual structure:
<ControlTemplate x:Key="TealTemplate">
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="0.1*" />
<RowDefinition Height="0.8*" />
<RowDefinition Height="0.1*" />
</Grid.RowDefinitions>
<BoxView Color="Teal" />
<Label Margin="20,0,0,0"
Text="{TemplateBinding HeaderText}"
TextColor="White"
FontSize="Title"
VerticalOptions="Center" />
<ContentPresenter Grid.Row="1" />
<BoxView Grid.Row="2"
Color="Teal" />
<Label x:Name="changeThemeLabel"
Grid.Row="2"
Margin="20,0,0,0"
Text="Change Theme"
TextColor="White"
HorizontalOptions="Start"
VerticalOptions="Center">
<Label.GestureRecognizers>
<TapGestureRecognizer Tapped="OnChangeThemeLabelTapped" />
</Label.GestureRecognizers>
</Label>
<controls:HyperlinkLabel Grid.Row="2"
Margin="0,0,20,0"
Text="Help"
TextColor="White"
Url="https://docs.microsoft.com/dotnet/maui/"
HorizontalOptions="End"
VerticalOptions="Center" />
</Grid>
</ControlTemplate>
The following example shows TealTemplate assigned to the ControlTemplate property of a ContentPage
derived page:
<controls:HeaderFooterPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:controls="clr-namespace:ControlTemplateDemos.Controls"
ControlTemplate="{StaticResource TealTemplate}"
HeaderText="MyApp"
...>
<StackLayout Margin="10">
<Entry Placeholder="Enter username" />
<Entry Placeholder="Enter password"
IsPassword="True" />
<Button Text="Login" />
</StackLayout>
</controls:HeaderFooterPage>
At runtime, when TealTemplate is applied to the page, the page content is substituted into the ContentPresenter
defined in the control template:
Get a named element from a template
Named elements within a control template can be retrieved from the templated custom control or templated
page. This can be achieved with the GetTemplateChild method, which returns the named element in the
instantiated ControlTemplate visual tree, if found. Otherwise, it returns null .
After a control template has been instantiated, the template's OnApplyTemplate method is called. The
GetTemplateChild method should therefore be called from a OnApplyTemplate override in the templated control
or templated page.
IMPORTANT
The GetTemplateChild method should only be called after the OnApplyTemplate method has been called.
The following XAML shows a control template named TealTemplate that can be applied to ContentPage derived
pages:
<ControlTemplate x:Key="TealTemplate">
<Grid>
...
<Label x:Name="changeThemeLabel"
Grid.Row="2"
Margin="20,0,0,0"
Text="Change Theme"
TextColor="White"
HorizontalOptions="Start"
VerticalOptions="Center">
<Label.GestureRecognizers>
<TapGestureRecognizer Tapped="OnChangeThemeLabelTapped" />
</Label.GestureRecognizers>
</Label>
...
</Grid>
</ControlTemplate>
In this example, the Label element is named, and can be retrieved in the code for the templated page. This is
achieved by calling the GetTemplateChild method from the OnApplyTemplate override for the templated page:
public AccessTemplateElementPage()
{
InitializeComponent();
}
In this example, the Label object named changeThemeLabel is retrieved once the ControlTemplate has been
instantiated. changeThemeLabel can then be accessed and manipulated by the AccessTemplateElementPage class.
The following screenshot shows that the text displayed by the Label has been changed:
Bind to a viewmodel
A ControlTemplate can data bind to a viewmodel, even when the ControlTemplate binds to the templated
parent (the runtime object instance to which the template is applied).
The following XAML example shows a page that consumes a viewmodel named PeopleViewModel :
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:ControlTemplateDemos"
xmlns:controls="clr-namespace:ControlTemplateDemos.Controls"
...>
<ContentPage.BindingContext>
<local:PeopleViewModel />
</ContentPage.BindingContext>
<ContentPage.Resources>
<DataTemplate x:Key="PersonTemplate">
<controls:CardView BorderColor="DarkGray"
CardTitle="{Binding Name}"
CardDescription="{Binding Description}"
ControlTemplate="{StaticResource CardViewControlTemplate}" />
</DataTemplate>
</ContentPage.Resources>
<StackLayout Margin="10"
BindableLayout.ItemsSource="{Binding People}"
BindableLayout.ItemTemplate="{StaticResource PersonTemplate}" />
</ContentPage>
In this example, the BindingContext of the page is set to a PeopleViewModel instance. This viewmodel exposes a
People collection and an ICommand named DeletePersonCommand . The StackLayout on the page uses a bindable
layout to data bind to the People collection, and the ItemTemplate of the bindable layout is set to the
PersonTemplate resource. This DataTemplate specifies that each item in the People collection will be displayed
using a object. The visual structure of the
CardView CardView object is defined using a ControlTemplate named
CardViewControlTemplate :
<ControlTemplate x:Key="CardViewControlTemplate">
<Frame BindingContext="{Binding Source={RelativeSource TemplatedParent}}"
BackgroundColor="{Binding CardColor}"
BorderColor="{Binding BorderColor}"
CornerRadius="5"
HasShadow="True"
Padding="8"
HorizontalOptions="Center"
VerticalOptions="Center">
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="75" />
<RowDefinition Height="4" />
<RowDefinition Height="Auto" />
</Grid.RowDefinitions>
<Label Text="{Binding CardTitle}"
FontAttributes="Bold"
FontSize="Large"
VerticalTextAlignment="Center"
HorizontalTextAlignment="Start" />
<BoxView Grid.Row="1"
BackgroundColor="{Binding BorderColor}"
HeightRequest="2"
HorizontalOptions="Fill" />
<Label Grid.Row="2"
Text="{Binding CardDescription}"
VerticalTextAlignment="Start"
VerticalOptions="Fill"
HorizontalOptions="Fill" />
<Button Text="Delete"
Command="{Binding Source={RelativeSource AncestorType={x:Type local:PeopleViewModel}},
Path=DeletePersonCommand}"
CommandParameter="{Binding CardTitle}"
HorizontalOptions="End" />
</Grid>
</Frame>
</ControlTemplate>
In this example, the root element of the ControlTemplate is a Frame object. The Frame object uses the
RelativeSource markup extension to set its BindingContext to the templated parent. The binding expressions of
the Frame object and its children resolve against CardView properties, due to inheriting the BindingContext
from the root Frame element. The following screenshot shows the page displaying the People collection:
While the objects in the ControlTemplate bind to properties on its templated parent, the Button within the
control template binds to both its templated parent, and to the DeletePersonCommand in the viewmodel. This is
because the Button.Command property redefines its binding source to be the binding context of the ancestor
whose binding context type is PeopleViewModel , which is the StackLayout . The Path part of the binding
expressions can then resolve the DeletePersonCommand property. However, the Button.CommandParameter property
doesn't alter its binding source, instead inheriting it from its parent in the ControlTemplate . Therefore, the
CommandParameter property binds to the CardTitle property of the CardView .
The overall effect of the Button bindings is that when the Button is tapped, the DeletePersonCommand in the
PeopleViewModel class is executed, with the value of the CardName property being passed to the
DeletePersonCommand . This results in the specified CardView being removed from the bindable layout.
.NET Multi-platform App UI (.NET MAUI) data templates provide the ability to define the presentation of data on
supported controls.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Consider a CollectionView that displays a collection of Person objects. The following example shows the
definition of the Person class:
The Person class defines Name , Age , and Location properties, which can be set when a Person object is
created. A control that displays collections, such as CollectionView , can be used to display Person objects:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:DataTemplates"
x:Class=""DataTemplates.WithoutDataTemplatePage>
<StackLayout>
<CollectionView>
<CollectionView.ItemsSource>
<x:Array Type="{x:Type local:Person}">
<local:Person Name="Steve" Age="21" Location="USA" />
<local:Person Name="John" Age="37" Location="USA" />
<local:Person Name="Tom" Age="42" Location="UK" />
<local:Person Name="Lucas" Age="29" Location="Germany" />
<local:Person Name="Tariq" Age="39" Location="UK" />
<local:Person Name="Jane" Age="30" Location="USA" />
</x:Array>
</CollectionView.ItemsSource>
</CollectionView>
</StackLayout>
</ContentPage>
In this example, items are added to the CollectionView by initializing its ItemsSource property from an array of
Person objects. CollectionView calls ToString when displaying the objects in the collection. However, because
there is no Person.ToString override, ToString returns the type name of each object:
The Person object can override the ToString method to display meaningful data:
This results in the CollectionView displaying the Person.Name property value for each object in the collection:
The Person.ToString override could return a formatted string consisting of the Name , Age , and Location
properties. However, this approach only offers limited control over the appearance of each item of data. For
more flexibility, a DataTemplate can be created that defines the appearance of the data.
Create a DataTemplate
A DataTemplate is used to specify the appearance of data, and typically uses data binding to display data. A
common usage scenario for data templates is when displaying data from a collection of objects in a control such
as a CollectionView or CarouselView . For example, when a CollectionView is bound to a collection of Person
objects, the CollectionView.ItemTemplate property can be set to a DataTemplate that defines the appearance of
each Person object in the CollectionView . The DataTemplate will contain objects that bind to property values
of each Person object. For more information about data binding, see Data binding.
A DataTemplate that's defined inline in a control is known as an inline template. Alternatively, data templates can
be defined as a control-level, page-level, or app-level resource. Choosing where to define a DataTemplate
impacts where it can be used:
A DataTemplate defined at the control-level can only be applied to the control.
A DataTemplate defined at the page-level can be applied to multiple controls on the page.
A DataTemplate defined at the app-level can be applied to valid controls throughout the app.
NOTE
Data templates lower in the view hierarchy take precedence over those defined higher up when they share x:Key
attributes. For example, an app-level data template will be overridden by a page-level data template, and a page-level
data template will be overridden by a control-level data template, or an inline data template.
A DataTemplate can be created inline, with a type, or as a resource, regardless of where it's defined.
Create an inline DataTemplate
An inline data template, which is one that's defined inline in a control, should be used if there's no need to reuse
the data template elsewhere. The objects specified in the DataTemplate define the appearance of each item of
data. A control such as CollectionView can then set its ItemTemplate property to the inline DataTemplate :
<CollectionView>
<CollectionView.ItemsSource>
<x:Array Type="{x:Type local:Person}">
<local:Person Name="Steve" Age="21" Location="USA" />
<local:Person Name="John" Age="37" Location="USA" />
<local:Person Name="Tom" Age="42" Location="UK" />
<local:Person Name="Lucas" Age="29" Location="Germany" />
<local:Person Name="Tariq" Age="39" Location="UK" />
<local:Person Name="Jane" Age="30" Location="USA" />
</x:Array>
</CollectionView.ItemsSource>
<CollectionView.ItemTemplate>
<DataTemplate>
<Grid>
...
<Label Text="{Binding Name}" FontAttributes="Bold" />
<Label Grid.Column="1" Text="{Binding Age}" />
<Label Grid.Column="2" Text="{Binding Location}" HorizontalTextAlignment="End" />
</Grid>
</DataTemplate>
</CollectionView.ItemTemplate>
</CollectionView>
In a CollectionView, the child of an inline DataTemplate must derive from BindableObject . In this example, a
Grid , which derives from Layout is used. The Grid contains three Label objects that bind their Text
properties to properties of each Person object in the collection. The following screenshot shows the resulting
appearance:
In this example, the CollectionView.ItemTemplate property is set to a DataTemplate that's created from a custom
type that defines the view appearance. The custom type must derive from ContentView :
<ContentView xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="DataTemplates.PersonView">
<Grid>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="0.5*" />
<ColumnDefinition Width="0.2*" />
<ColumnDefinition Width="0.3*" />
</Grid.ColumnDefinitions>
<Label Text="{Binding Name}" FontAttributes="Bold" />
<Label Grid.Column="1" Text="{Binding Age}" />
<Label Grid.Column="2" Text="{Binding Location}" HorizontalTextAlignment="End" />
</Grid>
</ContentView>
In this example, layout within the ContentView is managed by a Grid . The Grid contains three Label objects
that bind their Text properties to properties of each Person object in the collection.
For more information about creating custom views, see ContentView.
Create a DataTemplate as a resource
Data templates can be created as reusable objects in a ResourceDictionary . This is achieved by giving each
DataTemplate a unique x:Key value, which provides it with a descriptive key in the ResourceDictionary . A
control such as CollectionView can then set its ItemTemplate property to the DataTemplate :
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:DataTemplates"
x:Class="DataTemplates.WithDataTemplateResource">
<ContentPage.Resources>
<ResourceDictionary>
<DataTemplate x:Key="personTemplate">
<Grid>
...
</Grid>
</DataTemplate>
</ResourceDictionary>
</ContentPage.Resources>
<StackLayout>
<CollectionView ItemTemplate="{StaticResource personTemplate}">
<CollectionView.ItemsSource>
<x:Array Type="{x:Type local:Person}">
<local:Person Name="Steve" Age="21" Location="USA" />
...
</x:Array>
</CollectionView.ItemsSource>
</CollectionView>
</StackLayout>
</ContentPage>
In this example, the DataTemplate is assigned to the CollectionView.ItemTemplate property by using the
StaticResource markup extension. While the DataTemplate is defined in the page's ResourceDictionary , it could
also be defined at the control-level or app-level.
Create a DataTemplateSelector
A DataTemplateSelector can be used to choose a DataTemplate at runtime based on the value of a data-bound
property. This enables multiple data templates to be applied to the same type of object, to choose their
appearance at runtime. A data template selector enables scenarios such as a CollectionView or CarouselView
binding to a collection of objects where the appearance of each object can be chosen at runtime by the data
template selector returning a specific DataTemplate .
A data template selector is implemented by creating a class that inherits from DataTemplateSelector . The
OnSelectTemplate method should then be overridden to return a specific DataTemplate :
In this example, the OnSelectTemplate method returns a specific data template based on the value of the
DateOfBirth property. The returned data template is defined by the ValidTemplate or InvalidTemplate
property, which are set when consuming the data template selector.
Limitations
DataTemplateSelector objects have the following limitations:
The DataTemplateSelector subclass must always return the same template for the same data if queried
multiple times.
The DataTemplateSelector subclass must not return another DataTemplateSelector subclass.
The DataTemplateSelector subclass must not return new instances of a DataTemplate on each call. Instead,
the same instance must be returned. Failure to do so will create a memory leak and will disable control
virtualization.
Consume a DataTemplateSelector
A data template selector can be consumed by creating it as a resource and assigning its instance to .NET MAUI
control properties of type DataTemplate , such as CollectionView.ItemTemplate .
The following example shows declaring PersonDataTemplateSelector as a page-level resource:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:local="clr-namespace:Selector"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="Selector.MainPage">
<ContentPage.Resources>
<DataTemplate x:Key="validPersonTemplate">
<Grid>
...
</Grid>
</DataTemplate>
<DataTemplate x:Key="invalidPersonTemplate">
<Grid>
...
</Grid>
</DataTemplate>
<local:PersonDataTemplateSelector x:Key="personDataTemplateSelector"
ValidTemplate="{StaticResource validPersonTemplate}"
InvalidTemplate="{StaticResource invalidPersonTemplate}" />
</ContentPage.Resources>
...
</ContentPage>
In this example, the page-level ResourceDictionary defines two DataTemplate objects and a
PersonDataTemplateSelector object. The PersonDataTemplateSelector object sets its ValidTemplate and
InvalidTemplate properties to the DataTemplate objects using the StaticResource markup extension. While the
resources are defined in the page's ResourceDictionary , they could also be defined at the control-level or app-
level.
The PersonDataTemplateSelector object can be consumed by assigning it to the CollectionView.ItemTemplate
property:
<CollectionView x:Name="collectionView"
ItemTemplate="{StaticResource personDataTemplateSelector}" />
At runtime, the CollectionView calls the PersonDataTemplateSelector.OnSelectTemplate method for each of the
items in the underlying collection, with the call passing the data object as the item parameter. The returned
DataTemplate is then applied to that object.
The following screenshot shows the result of the CollectionView applying the PersonDataTemplateSelector to
each object in the underlying collection:
In this example, any Person object that has a DateOfBirth property value greater than or equal to 1980 is
displayed in green, with the remaining objects being displayed in red.
Triggers
3/18/2022 • 11 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) triggers allow you to express actions declaratively in XAML that change
the appearance of controls based on events or data changes. In addition, state triggers, which are a specialized
group of triggers, define when a VisualState should be applied.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
You can assign a trigger directly to a control's Triggers collection, or add it to a page-level or app-level
resource dictionary to be applied to multiple controls.
Property triggers
A Trigger represents a trigger that applies property values, or performs actions, when the specified property
meets a specified condition.
The following example shows a Trigger that changes an Entry background color when it receives focus:
In addition, optional EnterActions and ExitActions collections can be specified. For more information, see
EnterActions and ExitActions.
Apply a trigger using a style
Triggers can also be added to a Style declaration on a control, in a page, or an application ResourceDictionary .
The following example declares an implicit style for all Entry controls on the page:
<ContentPage.Resources>
<Style TargetType="Entry">
<Style.Triggers>
<Trigger TargetType="Entry"
Property="IsFocused"
Value="True">
<Setter Property="BackgroundColor"
Value="Yellow" />
<!-- Multiple Setter elements are allowed -->
</Trigger>
</Style.Triggers>
</Style>
</ContentPage.Resources>
Data triggers
A DataTrigger represents a trigger that applies property values, or performs actions, when the bound data
meets a specified condition. The Binding markup extension is used to monitor for the specified condition.
The following example shows a DataTrigger that disables a Button when the Entry is empty:
<Entry x:Name="entry"
Text=""
Placeholder="Enter text" />
<Button Text="Save">
<Button.Triggers>
<DataTrigger TargetType="Button"
Binding="{Binding Source={x:Reference entry},
Path=Text.Length}"
Value="0">
<Setter Property="IsEnabled"
Value="False" />
<!-- Multiple Setter elements are allowed -->
</DataTrigger>
</Button.Triggers>
</Button>
In this example, when the length of the Entry is zero, the trigger is activated.
TIP
When evaluating Path=Text.Length always provide a default value for the target property (eg. Text="" > because
otherwise it will be null and the trigger won't work like you expect.
In addition, optional EnterActions and ExitActions collections can be specified. For more information, see
EnterActions and ExitActions.
Event triggers
An EventTrigger represents a trigger that applies a set of actions in response to an event. Unlike Trigger ,
EventTrigger has no concept of termination of state, so the actions will not be undone once the condition that
raised the event is no longer true.
An EventTrigger only requires an Event property to be set:
<EventTrigger Event="TextChanged">
<local:NumericValidationTriggerAction />
</EventTrigger>
In this example, there are no Setter elements. Instead, there's a NumericalValidationTriggerAction object.
NOTE
Event triggers don't support EnterActions and ExitActions .
WARNING
Be careful when sharing triggers in a ResourceDictionary . One instance will be shared among controls so any state that
is configured once will apply to them all.
Multi-triggers
A MultiTrigger represents a trigger that applies property values, or performs actions, when a set of conditions
are satisfied. All the conditions must be true before the Setter objects are applied.
The following example shows a MultiTrigger that binds to two Entry objects:
<Entry x:Name="email"
Text="" />
<Entry x:Name="phone"
Text="" />
<Button Text="Save">
<Button.Triggers>
<MultiTrigger TargetType="Button">
<MultiTrigger.Conditions>
<BindingCondition Binding="{Binding Source={x:Reference email},
Path=Text.Length}"
Value="0" />
<BindingCondition Binding="{Binding Source={x:Reference phone},
Path=Text.Length}"
Value="0" />
</MultiTrigger.Conditions>
<Setter Property="IsEnabled" Value="False" />
<!-- multiple Setter elements are allowed -->
</MultiTrigger>
</Button.Triggers>
</Button>
<PropertyCondition Property="Text"
Value="OK" />
The EnterActions collection, of type IList<TriggerAction> , defines a collection that will be invoked when the
trigger condition is met. The ExitActions collection, of type IList<TriggerAction> , defines a collection that will
be invoked after the trigger condition is no longer met.
NOTE
The TriggerAction objects defined in the EnterActions and ExitActions collections are ignored by the
EventTrigger class.
The following example shows a property trigger that specifies an EnterAction and an ExitAction :
<Trigger.ExitActions>
<local:FadeTriggerAction StartsFrom="1" />
</Trigger.ExitActions>
</Trigger>
</Entry.Triggers>
</Entry>
A trigger action implementation must:
Implement the generic TriggerAction<T> class, with the generic parameter corresponding with the type of
control the trigger will be applied to. You can use classes such as VisualElement to write trigger actions that
work with a variety of controls, or specify a control type like Entry .
Override the Invoke method. This method is called whenever the trigger event occurs.
Optionally expose properties that can be set in XAML when the trigger is declared.
The following example shows the FadeTriggerAction class:
NOTE
You can provide EnterActions and ExitActions as well as Setter objects in a trigger, but be aware that the
Setter objects are called immediately (they do not wait for the EnterAction or ExitAction to complete).
State triggers
State triggers are a specialized group of triggers that define the conditions under which a VisualState should
be applied.
State triggers are added to the StateTriggers collection of a VisualState . This collection can contain a single
state trigger, or multiple state triggers. A VisualState will be applied when any state triggers in the collection
are active.
When using state triggers to control visual states, .NET MAUI uses the following precedence rules to determine
which trigger (and corresponding VisualState ) will be active:
1. Any trigger that derives from StateTriggerBase .
2. An AdaptiveTrigger activated due to the MinWindowWidth condition being met.
3. An AdaptiveTrigger activated due to the MinWindowHeight condition being met.
If multiple triggers are simultaneously active (for example, two custom triggers) then the first trigger declared in
the markup takes precedence.
NOTE
State triggers can be set in a Style , or directly on elements.
State trigger
The StateTrigger class, which derives from the StateTriggerBase class, has an IsActive bindable property. A
StateTrigger triggers a VisualState change when the IsActive property changes value.
The class, which is the base class for all state triggers, has an IsActive property and an
StateTriggerBase
IsActiveChanged event. This event fires whenever a VisualState change occurs. In addition, the
StateTriggerBase class has overridable OnAttached and OnDetached methods.
IMPORTANT
The StateTrigger.IsActive bindable property hides the inherited StateTriggerBase.IsActive property.
The following XAML example shows a Style that includes StateTrigger objects:
<Style TargetType="Grid">
<Setter Property="VisualStateManager.VisualStateGroups">
<VisualStateGroupList>
<VisualStateGroup>
<VisualState x:Name="Checked">
<VisualState.StateTriggers>
<StateTrigger IsActive="{Binding IsToggled}"
IsActiveChanged="OnCheckedStateIsActiveChanged" />
</VisualState.StateTriggers>
<VisualState.Setters>
<Setter Property="BackgroundColor"
Value="Black" />
</VisualState.Setters>
</VisualState>
<VisualState x:Name="Unchecked">
<VisualState.StateTriggers>
<StateTrigger IsActive="{Binding IsToggled, Converter={StaticResource
inverseBooleanConverter}}"
IsActiveChanged="OnUncheckedStateIsActiveChanged" />
</VisualState.StateTriggers>
<VisualState.Setters>
<Setter Property="BackgroundColor"
Value="White" />
</VisualState.Setters>
</VisualState>
</VisualStateGroup>
</VisualStateGroupList>
</Setter>
</Style>
In this example, the implicit Style targets Grid objects. When the IsToggled property of the bound object is
true , the background color of the Grid is set to black. When the IsToggled property of the bound object
becomes false , a VisualState change is triggered, and the background color of the Grid becomes white.
In addition, every time a VisualState change occurs, the IsActiveChanged event for the VisualState is raised.
Each VisualState registers an event handler for this event:
void OnCheckedStateIsActiveChanged(object sender, EventArgs e)
{
StateTriggerBase stateTrigger = sender as StateTriggerBase;
Console.WriteLine($"Checked state active: {stateTrigger.IsActive}");
}
In this example, when a handler for the IsActiveChanged event is raised, the handler outputs whether the
VisualState is active or not. For example, the following messages are output to the console window when
changing from the Checked visual state to the Unchecked visual state:
NOTE
Custom state triggers can be created by deriving from the StateTriggerBase class, and overriding the OnAttached
and OnDetached methods to perform any required registrations and cleanup.
Adaptive trigger
An AdaptiveTrigger triggers a VisualState change when the window is a specified height or width. This trigger
has two bindable properties:
MinWindowHeight , of type double , which indicates the minimum window height at which the VisualState
should be applied.
MinWindowWidth , of type double , which indicates the minimum window width at which the VisualState
should be applied.
NOTE
The AdaptiveTrigger derives from the StateTriggerBase class and can therefore attach an event handler to the
IsActiveChanged event.
The following XAML example shows a Style that includes AdaptiveTrigger objects:
<Style TargetType="StackLayout">
<Setter Property="VisualStateManager.VisualStateGroups">
<VisualStateGroupList>
<VisualStateGroup>
<VisualState x:Name="Vertical">
<VisualState.StateTriggers>
<AdaptiveTrigger MinWindowWidth="0" />
</VisualState.StateTriggers>
<VisualState.Setters>
<Setter Property="Orientation"
Value="Vertical" />
</VisualState.Setters>
</VisualState>
<VisualState x:Name="Horizontal">
<VisualState.StateTriggers>
<AdaptiveTrigger MinWindowWidth="800" />
</VisualState.StateTriggers>
<VisualState.Setters>
<Setter Property="Orientation"
Value="Horizontal" />
</VisualState.Setters>
</VisualState>
</VisualStateGroup>
</VisualStateGroupList>
</Setter>
</Style>
In this example, the implicit Style targets StackLayout objects. When the window width is between 0 and 800
device-independent units, StackLayout objects to which the Style is applied will have a vertical orientation.
When the window width is >= 800 device-independent units, the VisualState change is triggered, and the
StackLayout orientation changes to horizontal.
The MinWindowHeight and MinWindowWidth properties can be used independently or in conjunction with each
other. The following XAML shows an example of setting both properties:
<AdaptiveTrigger MinWindowWidth="800"
MinWindowHeight="1200"/>
In this example, the AdaptiveTrigger indicates that the corresponding VisualState will be applied when the
current window width is >= 800 device-independent units and the current window height is >= 1200 device-
independent units.
Compare state trigger
The CompareStateTrigger triggers a VisualState change when a property is equal to a specific value. This
trigger has two bindable properties:
Property , of type object , which indicates the property being compared by the trigger.
Value , of type object , which indicates the value at which the VisualState should be applied.
NOTE
The CompareStateTrigger derives from the StateTriggerBase class and can therefore attach an event handler to the
IsActiveChanged event.
The following XAML example shows a Style that includes CompareStateTrigger objects:
<Style TargetType="Grid">
<Setter Property="VisualStateManager.VisualStateGroups">
<VisualStateGroupList>
<VisualStateGroup>
<VisualState x:Name="Checked">
<VisualState.StateTriggers>
<CompareStateTrigger Property="{Binding Source={x:Reference checkBox},
Path=IsChecked}"
Value="True" />
</VisualState.StateTriggers>
<VisualState.Setters>
<Setter Property="BackgroundColor"
Value="Black" />
</VisualState.Setters>
</VisualState>
<VisualState x:Name="Unchecked">
<VisualState.StateTriggers>
<CompareStateTrigger Property="{Binding Source={x:Reference checkBox},
Path=IsChecked}"
Value="False" />
</VisualState.StateTriggers>
<VisualState.Setters>
<Setter Property="BackgroundColor"
Value="White" />
</VisualState.Setters>
</VisualState>
</VisualStateGroup>
</VisualStateGroupList>
</Setter>
</Style>
...
<Grid>
<Frame BackgroundColor="White"
CornerRadius="12"
Margin="24"
HorizontalOptions="Center"
VerticalOptions="Center">
<StackLayout Orientation="Horizontal">
<CheckBox x:Name="checkBox"
VerticalOptions="Center" />
<Label Text="Check the CheckBox to modify the Grid background color."
VerticalOptions="Center" />
</StackLayout>
</Frame>
</Grid>
In this example, the implicit Style targets Grid objects. When the IsChecked property of the CheckBox is
false , the background color of the Grid is set to white. When the CheckBox.IsChecked property becomes
true , a VisualState change is triggered, and the background color of the Grid becomes black.
NOTE
The DeviceStateTrigger derives from the StateTriggerBase class and can therefore attach an event handler to the
IsActiveChanged event.
The following XAML example shows a Style that includes DeviceStateTrigger objects:
<Style x:Key="DeviceStateTriggerPageStyle"
TargetType="ContentPage">
<Setter Property="VisualStateManager.VisualStateGroups">
<VisualStateGroupList>
<VisualStateGroup>
<VisualState x:Name="iOS">
<VisualState.StateTriggers>
<DeviceStateTrigger Device="iOS" />
</VisualState.StateTriggers>
<VisualState.Setters>
<Setter Property="BackgroundColor"
Value="Silver" />
</VisualState.Setters>
</VisualState>
<VisualState x:Name="Android">
<VisualState.StateTriggers>
<DeviceStateTrigger Device="Android" />
</VisualState.StateTriggers>
<VisualState.Setters>
<Setter Property="BackgroundColor"
Value="#2196F3" />
</VisualState.Setters>
</VisualState>
</VisualStateGroup>
</VisualStateGroupList>
</Setter>
</Style>
In this example, the explicit Style targets ContentPage objects. ContentPage objects that consume the style set
their background color to silver on iOS, and to pale blue on Android.
Orientation state trigger
The OrientationStateTrigger triggers a VisualState change when the orientation of the device changes. This
trigger has a single bindable property:
Orientation , of type DeviceOrientation , which indicates the orientation to which the VisualState should be
applied.
NOTE
The OrientationStateTrigger derives from the StateTriggerBase class and can therefore attach an event handler to
the IsActiveChanged event.
The following XAML example shows a Style that includes OrientationStateTrigger objects:
<Style x:Key="OrientationStateTriggerPageStyle"
TargetType="ContentPage">
<Setter Property="VisualStateManager.VisualStateGroups">
<VisualStateGroupList>
<VisualStateGroup>
<VisualState x:Name="Portrait">
<VisualState.StateTriggers>
<OrientationStateTrigger Orientation="Portrait" />
</VisualState.StateTriggers>
<VisualState.Setters>
<Setter Property="BackgroundColor"
Value="Silver" />
</VisualState.Setters>
</VisualState>
<VisualState x:Name="Landscape">
<VisualState.StateTriggers>
<OrientationStateTrigger Orientation="Landscape" />
</VisualState.StateTriggers>
<VisualState.Setters>
<Setter Property="BackgroundColor"
Value="White" />
</VisualState.Setters>
</VisualState>
</VisualStateGroup>
</VisualStateGroupList>
</Setter>
</Style>
In this example, the explicit Style targets ContentPage objects. ContentPage objects that consume the style set
their background color to silver when the orientation is portrait, and set their background color to white when
the orientation is landscape.
Basic animation
3/18/2022 • 8 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) animation classes target different properties of visual elements,
with a typical basic animation progressively changing a property from one value to another over a period of
time.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Basic animations can be created with extension methods provided by the ViewExtensions class, in the
Microsoft.Maui.Controls namespace, which operate on VisualElement objects:
By default, each animation will take 250 milliseconds. However, a duration for each animation can be specified
when creating the animation.
NOTE
The ViewExtensions class also provides a LayoutTo extension method. However, this method is intended to be used
by layouts to animate transitions between layout states that contain size and position changes.
The animation extension methods in the ViewExtensions class are all asynchronous and return a Task<bool>
object. The return value is false if the animation completes, and true if the animation is cancelled. Therefore,
when animation operations are combined with the await operator it becomes possible to create sequential
animations with subsequent animation methods executing after the previous method has completed. For more
information, see Compound animations.
If there's a requirement to let an animation complete in the background, then the await operator can be
omitted. In this scenario, the animation extension methods will quickly return after initiating the animation, with
the animation occurring in the background. This operation can be taken advantage of when creating composite
animations. For more information, see Composite animations.
Single animations
Each extension method in the ViewExtensions class implements a single animation operation that progressively
changes a property from one value to another value over a period of time.
Rotation
Rotation is performed with the RotateTo method, which progressively changes the Rotation property of an
element:
In this example, an Image instance is rotated up to 360 degrees over 2 seconds (2000 milliseconds). The
RotateTo method obtains the current Rotation property value of the element for the start of the animation,
and then rotates from that value to its first argument (360). Once the animation is complete, the image's
Rotation property is reset to 0. This ensures that the Rotation property doesn't remain at 360 after the
animation concludes, which would prevent additional rotations.
NOTE
In addition to the RotateTo method, there are also RotateXTo and RotateYTo methods that animate the
RotationX and RotationY properties, respectively.
Relative rotation
Relative rotation is performed with the RelRotateTo method, which progressively changes the Rotation
property of an element:
In this example, an Image instance is rotated 360 degrees from its starting position over 2 seconds (2000
milliseconds). The RelRotateTo method obtains the current Rotation property value of the element for the
start of the animation, and then rotates from that value to the value plus its first argument (360). This ensures
that each animation will always be a 360 degrees rotation from the starting position. Therefore, if a new
animation is invoked while an animation is already in progress, it will start from the current position and may
end at a position that is not an increment of 360 degrees.
Scaling
Scaling is performed with the ScaleTo method, which progressively changes the Scale property of an
element:
In this example, an Image instance is scaled up to twice its size over 2 seconds (2000 milliseconds). The
ScaleTo method obtains the current Scale property value of the element for the start of the animation, and
then scales from that value to its first argument. This has the effect of expanding the size of the image to twice
its size.
NOTE
In addition to the ScaleTo method, there are also ScaleXTo and ScaleYTo methods that animate the ScaleX and
ScaleY properties, respectively.
Relative scaling
Relative scaling is performed with the RelScaleTo method, which progressively changes the Scale property of
an element:
In this example, an Image instance is scaled up to twice its size over 2 seconds (2000 milliseconds). The
RelScaleTo method obtains the current Scale property value of the element for the start of the animation, and
then scales from that value to the value plus its first argument. This ensures that each animation will always be a
scaling of 2 from the starting position.
Scaling and rotation with anchors
The AnchorX and AnchorY properties of a visual element set the center of scaling or rotation for the Rotation
and Scale properties. Therefore, their values also affect the RotateTo and ScaleTo methods.
Given an Image that has been placed at the center of a layout, the following code example demonstrates
rotating the image around the center of the layout by setting its AnchorY property:
To rotate the Image instance around the center of the layout, the AnchorX and AnchorY properties must be set
to values that are relative to the width and height of the Image . In this example, the center of the Image is
defined to be at the center of the layout, and so the default AnchorX value of 0.5 does not require changing.
However, the AnchorY property is redefined to be a value from the top of the Image to the center point of the
layout. This ensures that the Image makes a full rotation of 360 degrees around the center point of the layout.
Translation
Translation is performed with the TranslateTo method, which progressively changes the TranslationX and
TranslationY properties of an element:
In this example, the Image instance is translated horizontally and vertically over 1 second (1000 milliseconds).
The TranslateTo method simultaneously translates the image 100 device-independent units to the left, and 100
device-independent units upwards. This is because the first and second arguments are both negative numbers.
Providing positive numbers would translate the image to the right, and down.
IMPORTANT
If an element is initially laid out off screen and then translated onto the screen, after translation the element's input layout
remains off screen and the user can't interact with it. Therefore, it's recommended that a view should be laid out in its final
position, and then any required translations performed.
Fading
Fading is performed with the FadeTo method, which progressively changes the Opacity property of an
element:
image.Opacity = 0;
await image.FadeTo(1, 4000);
In this example, the Image instance fades in over 4 seconds (4000 milliseconds). The FadeTo method obtains
the current Opacity property value of the element for the start of the animation, and then fades in from that
value to its first argument.
Compound animations
A compound animation is a sequential combination of animations, and can be created with the await operator:
In this example, the Image instance is translated over 6 seconds (6000 milliseconds). The translation of the
Image uses five animations, with the await operator indicating that each animation executes sequentially.
Therefore, subsequent animation methods execute after the previous method has completed.
Composite animations
A composite animation is a combination of animations where two or more animations run simultaneously.
Composite animations can be created by combining awaited and non-awaited animations:
image.RotateTo(360, 4000);
await image.ScaleTo(2, 2000);
await image.ScaleTo(1, 2000);
In this example, the Image instance is scaled and simultaneously rotated over 4 seconds (4000 milliseconds).
The scaling of the Image uses two sequential animations that occur at the same time as the rotation. The
RotateTo method executes without an await operator and returns immediately, with the first ScaleTo
animation then beginning. The await operator on the first ScaleTo method delays the second ScaleTo
method until the first ScaleTo method has completed. At this point the RotateTo animation is half completed
and the Image will be rotated 180 degrees. During the final 2 seconds (2000 milliseconds), the second ScaleTo
animation and the RotateTo animation both complete.
Run multiple animations concurrently
The Task.WhenAny and Task.WhenAll methods can be used to run multiple asynchronous methods concurrently,
and therefore can create composite animations. Both methods return a Task object and accept a collection of
methods that each return a Task object. The Task.WhenAny method completes when any method in its
collection completes execution, as demonstrated in the following code example:
await Task.WhenAny<bool>
(
image.RotateTo(360, 4000),
image.ScaleTo(2, 2000)
);
await image.ScaleTo(1, 2000);
In this example, the Task.WhenAny method contains two tasks. The first task rotates an Image instance over 4
seconds (4000 milliseconds), and the second task scales the image over 2 seconds (2000 milliseconds). When
the second task completes, the Task.WhenAny method call completes. However, even though the RotateTo
method is still running, the second ScaleTo method can begin.
The Task.WhenAll method completes when all the methods in its collection have completed, as demonstrated in
the following code example:
// 10 minute animation
uint duration = 10 * 60 * 1000;
await Task.WhenAll
(
image.RotateTo(307 * 360, duration),
image.RotateXTo(251 * 360, duration),
image.RotateYTo(199 * 360, duration)
);
In this example, the Task.WhenAll method contains three tasks, each of which executes over 10 minutes. Each
Task makes a different number of 360 degree rotations – 307 rotations for RotateTo , 251 rotations for
RotateXTo , and 199 rotations for RotateYTo . These values are prime numbers, therefore ensuring that the
rotations aren't synchronized and hence won't result in repetitive patterns.
Canceling animations
An app can cancel one or more animations with a call to the CancelAnimations extension method:
image.CancelAnimations();
In this example, all animations that are running on the Image instance are immediately canceled.
Easing functions
3/18/2022 • 3 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) includes an Easing class that enables you to specify a transfer function
that controls how animations speed up or slow down as they're running.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The Easing class defines a number of easing functions that can be consumed by animations:
The BounceIn easing function bounces the animation at the beginning.
The BounceOut easing function bounces the animation at the end.
The CubicIn easing function slowly accelerates the animation.
The CubicInOut easing function accelerates the animation at the beginning, and decelerates the animation at
the end.
The CubicOut easing function quickly decelerates the animation.
The Linear easing function uses a constant velocity, and is the default easing function.
The SinIn easing function smoothly accelerates the animation.
The SinInOut easing function smoothly accelerates the animation at the beginning, and smoothly
decelerates the animation at the end.
The SinOut easing function smoothly decelerates the animation.
The SpringIn easing function causes the animation to very quickly accelerate towards the end.
The SpringOut easing function causes the animation to quickly decelerate towards the end.
The In and Out suffixes indicate if the effect provided by the easing function is noticeable at the beginning of
the animation, at the end, or both.
In addition, custom easing functions can be created. For more information, see Custom easing functions.
By specifying an easing function for an animation, the animation velocity becomes non-linear and produces the
effect provided by the easing function. Omitting an easing function when creating an animation causes the
animation to use the default Linear easing function, which produces a linear velocity.
For more information about using the animation extension methods in the ViewExtensions class, see Basic
animation. Easing functions can also be consumed by the Animation class. For more information, see Custom
animation.
In this example, the CustomEase method truncates the incoming value to the values 0, 0.2, 0.4, 0.6, 0.8, and 1.
Therefore, the Image instance is translated in discrete jumps, rather than smoothly.
Custom easing func
A custom easing function can also be defined as a Func<double, double> :
In this example, the CustomEaseFunc represents an easing function that starts off fast, slows down and reverses
course, and then reverses course again to accelerate quickly towards the end. Therefore, while the overall
movement of the Image instance is downwards, it also temporarily reverses course halfway through the
animation.
Custom easing constructor
A custom easing function can also be defined as the argument to the Easing constructor:
await image.TranslateTo(0, 200, 2000, new Easing (t => 1 - Math.Cos (10 * Math.PI * t) * Math.Exp (-5 *
t)));
In this example, the custom easing function is specified as a lambda function argument to the Easing
constructor, and uses the Math.Cos method to create a slow drop effect that's dampened by the Math.Exp
method. Therefore, the Image instance is translated so that it appears to drop to its final position.
Custom animation
3/18/2022 • 9 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) Animation class is the building block of all .NET MAUI animations,
with the extension methods in the ViewExtensions class creating one or more Animation objects.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
A number of parameters must be specified when creating an Animation object, including start and end values
of the property being animated, and a callback that changes the value of the property. An Animation object can
also maintain a collection of child animations that can be run and synchronized. For more information, see Child
animations.
Running an animation created with the Animation class, which may or may not include child animations, is
achieved by calling the Commit method. This method specifies the duration of the animation, and amongst other
items, a callback that controls whether to repeat the animation.
NOTE
The Animation class has an IsEnabled property that can be examined to determine if animations have been disabled
by the operating system, such as when power saving mode is activated.
Create an animation
When creating an Animation object, typically, a minimum of three parameters are required, as demonstrated in
the following code example:
In this example, an animation of the Scale property of an Image instance is defined from a value of 1 to a
value of 2. The animated value is passed to the callback specified as the first argument, where it's used to change
the value of the Scale property.
The animation is started with a call to the Commit method:
animation.Commit(this, "SimpleAnimation", 16, 2000, Easing.Linear, (v, c) => image.Scale = 1, () => true);
NOTE
The Commit method doesn't return a Task object. Instead, notifications are provided through callback methods.
In the above example, the overall effect is to create an animation that increases the Scale property of an Image
instance from 1 to 2, over 2 seconds (2000 milliseconds), using the Linear easing function. Each time the
animation completes, its Scale property is reset to 1 and the animation repeats.
NOTE
Concurrent animations, that run independently of each other can be constructed by creating an Animation object for
each animation, and then calling the Commit method on each animation.
Child animations
The Animation class also supports child animations, which are Animation objects to which other Animation
objects are added as children. This enables a series of animations to be run and synchronized. The following
code example demonstrates creating and running child animations:
In both examples, a parent Animation object is created, to which additional Animation objects are then added.
The first two arguments to the Add method specify when to begin and finish the child animation. The argument
values must be between 0 and 1, and represent the relative period within the parent animation that the specified
child animation will be active. Therefore, in this example the scaleUpAnimation will be active for the first half of
the animation, the scaleDownAnimation will be active for the second half of the animation, and the
rotateAnimation will be active for the entire duration.
The overall effect of this example is that the animation occurs over 4 seconds (4000 milliseconds). The
scaleUpAnimation animates the Scale property from 1 to 2, over 2 seconds. The scaleDownAnimation then
animates the Scale property from 2 to 1, over 2 seconds. While both scale animations are occurring, the
rotateAnimation animates the Rotation property from 0 to 360, over 4 seconds. Both scaling animations also
use easing functions. The SpringIn easing function causes the Image instance to initially shrink before getting
larger, and the SpringOut easing function causes the Image to become smaller than its actual size towards the
end of the complete animation.
There are a number of differences between an Animation object that uses child animations, and one that
doesn't:
When using child animations, the finished callback on a child animation indicates when the child has
completed, and the finished callback passed to the Commit method indicates when the entire animation
has completed.
When using child animations, returning true from the repeat callback on the Commit method will not
cause the animation to repeat, but the animation will continue to run without new values.
When including an easing function in the Commit method, and the easing function returns a value greater
than 1, the animation will be terminated. If the easing function returns a value less than 0, the value is
clamped to 0. To use an easing function that returns a value less than 0 or greater than 1, it must specified in
one of the child animations, rather than in the Commit method.
The Animation class also includes WithConcurrent methods that can be used to add child animations to a parent
Animation object. However, their begin and finish argument values aren't restricted to 0 to 1, but only that
part of the child animation that corresponds to a range of 0 to 1 will be active. For example, if a WithConcurrent
method call defines a child animation that targets a Scale property from 1 to 6, but with begin and finish
values of -2 and 3, the begin value of -2 corresponds to a Scale value of 1, and the finish value of 3
corresponds to a Scale value of 6. Because values outside the range of 0 and 1 play no part in an animation,
the Scale property will only be animated from 3 to 6.
Cancel an animation
An app can cancel a custom animation with a call to the AbortAnimation extension method:
this.AbortAnimation ("SimpleAnimation");
Because animations are uniquely identified by a combination of the animation owner, and the animation name,
the owner and name specified when running the animation must be specified to cancel it. Therefore, this
example will immediately cancel the animation named SimpleAnimation that's owned by the page.
Create a custom animation
The examples shown here so far have demonstrated animations that could equally be achieved with the
methods in the ViewExtensions class. However, the advantage of the Animation class is that it has access to the
callback method, which is executed when the animated value changes. This allows the callback to implement any
desired animation. For example, the following code example animates the BackgroundColor property of a page
by setting it to Color values created by the Color.FromHsla method, with hue values ranging from 0 to 1:
The resulting animation provides the appearance of advancing the page background through the colors of the
rainbow.
The Animate method requires a transform argument, which is a callback method. The input to this callback is
always a double ranging from 0 to 1. Therefore, in this example the ColorTo method defines its own transform
Func that accepts a double ranging from 0 to 1, and that returns a Color value corresponding to that value.
The Color value is calculated by interpolating the Red , Green , Blue , and Alpha values of the two supplied
Color arguments. The Color value is then passed to the callback method to be applied to a property. This
approach allows the ColorTo method to animate any specified Color property:
await Task.WhenAll(
label.ColorTo(Colors.Red, Colors.Blue, c => label.TextColor = c, 5000),
label.ColorTo(Colors.Blue, Colors.Red, c => label.BackgroundColor = c, 5000));
await this.ColorTo(Color.FromRgb(0, 0, 0), Color.FromRgb(255, 255, 255), c => BackgroundColor = c, 5000);
await boxView.ColorTo(Colors.Blue, Colors.Red, c => boxView.Color = c, 4000);
In this code example, the ColorTo method animates the TextColor and BackgroundColor properties of a Label
, the BackgroundColor property of a page, and the Color property of a BoxView .
Brushes
3/18/2022 • 2 minutes to read • Edit Online
A .NET Multi-platform App UI (.NET MAUI) brush enables you to paint an area, such as the background of a
control, using different approaches.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The Brush class is an abstract class that paints an area with its output. Classes that derive from Brush describe
different ways of painting an area. The following list describes the different brush types available in .NET MAUI:
SolidColorBrush , which paints an area with a solid color. For more information, see Solid color brushes.
LinearGradientBrush , which paints an area with a linear gradient. For more information, see Linear gradient
brushes.
RadialGradientBrush , which paints an area with a radial gradient. For more information, see Radial gradient
brushes.
Instances of these brush types can be assigned to the Stroke and Fill properties of a Shape , the Stoke
property of a Border , the Brush property of a Shadow , and the Background property of a VisualElement .
NOTE
The VisualElement.Background property enables brushes to be used as the background in any control.
The Brush class also has an IsNullOrEmpty method that returns a bool that represents whether the brush is
defined or not.
Solid color brushes
3/18/2022 • 2 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) SolidColorBrush class derives from the Brush class, and is used to
paint an area with a solid color. There are a variety of approaches to specifying the color of a SolidColorBrush .
For example, you can specify its color with a Color value or by using one of the predefined SolidColorBrush
objects provided by the Brush class.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The SolidColorBrush class defines the Color property, of type Color , which represents the color of the brush.
This property is backed by a BindableProperty object, which means that it can be the target of data bindings,
and styled.
The SolidColorBrush class also has an IsEmpty method that returns a bool that represents whether the brush
has been assigned a color.
Create a SolidColorBrush
There are three main techniques for creating a SolidColorBrush . You can create a SolidColorBrush from a
Color , use a predefined brush, or create a SolidColorBrush using hexadecimal notation.
<Frame Background="DarkBlue"
BorderColor="LightGray"
HasShadow="True"
CornerRadius="12"
HeightRequest="120"
WidthRequest="120" />
In this example, the background of the Frame is painted with a dark blue SolidColorBrush :
Alternatively, the Color value can be specified using property tag syntax:
<Frame BorderColor="LightGray"
HasShadow="True"
CornerRadius="12"
HeightRequest="120"
WidthRequest="120">
<Frame.Background>
<SolidColorBrush Color="DarkBlue" />
</Frame.Background>
</Frame>
In this example, the background of the Frame is painted with a SolidColorBrush whose color is specified by
setting the SolidColorBrush.Color property.
Use a predefined Brush
The Brush class defines a set of commonly used SolidColorBrush objects. The following example uses one of
these predefined SolidColorBrush objects:
In this example, the background of the Frame is painted with an indigo SolidColorBrush :
For a list of predefined SolidColorBrush objects provided by the Brush class, see Solid color brushes.
Use hexadecimal notation
SolidColorBrush objects can also be created using hexadecimal notation. With this approach, a color is specified
in terms of the amount of red, green, and blue to combine into a single color. The main format for specifying a
color using hexadecimal notation is #rrggbb , where:
rr is a two-digit hexadecimal number specifying the relative amount of red.
gg is a two-digit hexadecimal number specifying the relative amount of green.
bb is a two-digit hexadecimal number specifying the relative amount of blue.
In addition, a color can be specified as #aarrggbb where aa specifies the alpha value, or transparency, of the
color. This approach enables you to create colors that are partially transparent.
The following example sets the color value of a SolidColorBrush using hexadecimal notation:
<Frame Background="#FF9988"
BorderColor="LightGray"
HasShadow="True"
CornerRadius="12"
HeightRequest="120"
WidthRequest="120" />
In this example, the background of the Frame is painted with a salmon-colored SolidColorBrush :
The .NET Multi-platform App UI (.NET MAUI) GradientBrush class derives from the Brush class, and is an
abstract class that describes a gradient, which is composed of gradient stops. A gradient brush paints an area
with multiple colors that blend into each other along an axis.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Classes that derive from GradientBrush describe different ways of interpreting gradient stops, and .NET MAUI
provides the following gradient brushes:
LinearGradientBrush , which paints an area with a linear gradient. For more information, see Linear gradient
brushes.
RadialGradientBrush , which paints an area with a radial gradient. For more information, see Radial gradient
brushes.
The GradientBrush class defines the GradientStops property, of type GradientStopsCollection , which
represents the brush's gradient stops, each of which specifies a color and an offset along the brush's gradient
axis. A GradientStopsCollection is an ObservableCollection of GradientStop objects. The GradientStops
property is backed by a BindableProperty object, which means that it can be the target of data bindings, and
styled.
NOTE
The GradientStops property is the ContentProperty of the GradientBrush class, and so does not need to be
explicitly set from XAML.
Gradient stops
Gradient stops are the building blocks of a gradient brush, and specify the colors in the gradient and their
location along the gradient axis. Gradient stops are specified using GradientStop objects.
The GradientStop class defines the following properties:
, of type Color , which represents the color of the gradient stop.
Color
Offset , of type float , which represents the location of the gradient stop within the gradient vector. The
default value of this property is 0, and valid values are in the range 0.0-1.0. The closer this value is to 0, the
closer the color is to the start of the gradient. Similarly, the closer this value is to 1, the closer the color is to
the end of the gradient.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
IMPORTANT
The coordinate system used by gradients is relative to a bounding box for the output area. 0 indicates 0 percent of the
bounding box, and 1 indicates 100 percent of the bounding box. Therefore, (0.5,0.5) describes a point in the middle of the
bounding box, and (1,1) describes a point at the bottom right of the bounding box.
The following XAML example creates a diagonal LinearGradientBrush with four colors:
<LinearGradientBrush StartPoint="0,0"
EndPoint="1,1">
<GradientStop Color="Yellow"
Offset="0.0" />
<GradientStop Color="Red"
Offset="0.25" />
<GradientStop Color="Blue"
Offset="0.75" />
<GradientStop Color="LimeGreen"
Offset="1.0" />
</LinearGradientBrush>
The color of each point between gradient stops is interpolated as a combination of the color specified by the two
bounding gradient stops. The following diagram shows the gradient stops from the previous example:
In this diagram, the circles mark the position of gradient stops, and the dashed line shows the gradient axis. The
first gradient stop specifies the color yellow at an offset of 0.0. The second gradient stop specifies the color red
at an offset of 0.25. The points between these two gradient stops gradually change from yellow to red as you
move from left to right along the gradient axis. The third gradient stop specifies the color blue at an offset of
0.75. The points between the second and third gradient stops gradually change from red to blue. The fourth
gradient stop specifies the color lime green at at offset of 1.0. The points between the third and fourth gradient
stops gradually change from blue to lime green.
Linear gradient brushes
3/18/2022 • 3 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) LinearGradientBrush class derives from the GradientBrush class,
and paints an area with a linear gradient, which blends two or more colors along a line known as the gradient
axis. GradientStop objects are used to specify the colors in the gradient and their positions. For more
information about GradientStop objects, see Gradients.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
NOTE
Linear gradients can also be created with the linear-gradient() CSS function.
Create a LinearGradientBrush
A linear gradient brush's gradient stops are positioned along the gradient axis. The orientation and size of the
gradient axis can be changed using the brush's StartPoint and EndPoint properties. By manipulating these
properties, you can create horizontal, vertical, and diagonal gradients, reverse the gradient direction, condense
the gradient spread, and more.
The StartPoint and EndPoint properties are relative to the area being painted. (0,0) represents the top-left
corner of the area being painted, and (1,1) represents the bottom-right corner of the area being painted. The
following diagram shows the gradient axis for a diagonal linear gradient brush:
In this diagram, the dashed line shows the gradient axis, which highlights the interpolation path of the gradient
from the start point to the end point.
Create a horizontal linear gradient
To create a horizontal linear gradient, create a LinearGradientBrush object and set its StartPoint to (0,0) and its
EndPoint to (1,0). Then, add two or more GradientStop objects to the LinearGradientBrush.GradientStops
collection, that specify the colors in the gradient and their positions.
The following XAML example shows a horizontal LinearGradientBrush that's set as the Background of a Frame :
<Frame BorderColor="LightGray"
HasShadow="True"
CornerRadius="12"
HeightRequest="120"
WidthRequest="120">
<Frame.Background>
<!-- StartPoint defaults to (0,0) -->
<LinearGradientBrush EndPoint="1,0">
<GradientStop Color="Yellow"
Offset="0.1" />
<GradientStop Color="Green"
Offset="1.0" />
</LinearGradientBrush>
</Frame.Background>
</Frame>
In this example, the background of the Frame is painted with a LinearGradientBrush that interpolates from
yellow to green horizontally:
In this example, the background of the Frame is painted with a LinearGradientBrush that interpolates from
yellow to green vertically:
<Frame BorderColor="LightGray"
HasShadow="True"
CornerRadius="12"
HeightRequest="120"
WidthRequest="120">
<Frame.Background>
<!-- StartPoint defaults to (0,0)
Endpoint defaults to (1,1) -->
<LinearGradientBrush>
<GradientStop Color="Yellow"
Offset="0.1" />
<GradientStop Color="Green"
Offset="1.0" />
</LinearGradientBrush>
</Frame.Background>
</Frame>
In this example, the background of the Frame is painted with a LinearGradientBrush that interpolates from
yellow to green diagonally:
Radial gradient brushes
3/18/2022 • 2 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) RadialGradientBrush class derives from the GradientBrush class,
and paints an area with a radial gradient, which blends two or more colors across a circle. GradientStop objects
are used to specify the colors in the gradient and their positions. For more information about GradientStop
objects, see Gradients.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
NOTE
Radial gradients can also be created with the radial-gradient() CSS function.
Create a RadialGradientBrush
A radial gradient brush's gradient stops are positioned along a gradient axis defined by a circle. The gradient axis
radiates from the center of the circle to its circumference. The position and size of the circle can be changed
using the brush's Center and Radius properties. The circle defines the end point of the gradient. Therefore, a
gradient stop at 1.0 defines the color at the circle's circumference. A gradient stop at 0.0 defines the color at the
center of the circle.
To create a radial gradient, create a RadialGradientBrush object and set its Center and Radius properties. Then,
add two or more GradientStop objects to the RadialGradientBrush.GradientStops collection, that specify the
colors in the gradient and their positions.
The following XAML example shows a RadialGradientBrush that's set as the Background of a Frame :
<Frame BorderColor="LightGray"
HasShadow="True"
CornerRadius="12"
HeightRequest="120"
WidthRequest="120">
<Frame.Background>
<!-- Center defaults to (0.5,0.5)
Radius defaults to (0.5) -->
<RadialGradientBrush>
<GradientStop Color="Red"
Offset="0.1" />
<GradientStop Color="DarkBlue"
Offset="1.0" />
</RadialGradientBrush>
</Frame.Background>
</Frame>
In this example, the background of the Frame is painted with a RadialGradientBrush that interpolates from red
to dark blue. The center of the radial gradient is positioned in the center of the Frame :
The following XAML example moves the center of the radial gradient to the top-left corner of the Frame :
In this example, the background of the Frame is painted with a RadialGradientBrush that interpolates from red
to dark blue. The center of the radial gradient is positioned in the top-left of the Frame :
The following XAML example moves the center of the radial gradient to the bottom-right corner of the Frame :
<!-- Radius defaults to (0.5) -->
<RadialGradientBrush Center="1.0,1.0">
<GradientStop Color="Red"
Offset="0.1" />
<GradientStop Color="DarkBlue"
Offset="1.0" />
</RadialGradientBrush>
In this example, the background of the Frame is painted with a RadialGradientBrush that interpolates from red
to dark blue. The center of the radial gradient is positioned in the bottom-right of the Frame :
Controls
3/18/2022 • 6 minutes to read • Edit Online
The user interface of a .NET Multi-platform App UI (.NET MAUI) app is constructed of objects that map to the
native controls of each target platform.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The main control groups used to create the user interface of a .NET MAUI app are pages, layouts, and views. A
.NET MAUI page generally occupies the full screen or window. The page usually contains a layout, which
contains views and possibly other layouts. Pages, layouts, and views derive from the VisualElement class. This
class provides a variety of properties, methods, and events that are useful in derived classes.
NOTE
ListView and TableView also support the use of cells. Cells are specialized elements used for items in a table, that
describe how each item should be rendered.
Pages
.NET MAUI apps consist of one or more pages. A page usually occupies all of the screen, or window, and each
page typically contains at least one layout.
.NET MAUI contains the following pages:
PA GE DESC RIP T IO N
Layouts
.NET MAUI layouts are used to compose user-interface controls into visual structures, and each layout typically
contains multiple views. Layout classes typically contain logic to set the position and size of child elements.
.NET MAUI contains the following layouts:
L AY O UT DESC RIP T IO N
Views
.NET MAUI views are the UI objects such as labels, buttons, and sliders that are commonly known as controls or
widgets in other environments.
.NET MAUI contains the following views:
Editor Editor enables you to enter and edit multiple lines of text.
For more information, see Editor.
Entry Entry enables you to enter and edit a single line of text.
For more information, see Entry.
Line Line displays a line from a start point to an end point. For
more information, see Line.
.NET Multi-platform App UI (.NET MAUI) provides a collection of controls that can be used to display data,
initiate actions, indicate activity, display collections, pick data, and more. By default, handlers map these cross-
platform controls to native controls on each platform.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
For example, on iOS a .NET MAUI handler will map a .NET MAUI Button to an iOS UIButton . On Android, the
Button will be mapped to a AppCompatButton :
Handlers can be accessed through a control-specific interface provided by .NET MAUI, such as IButton for a
Button . This avoids the cross-platform control having to reference its handler, and the handler having to
reference the cross-platform control. The mapping of the cross-platform control API to the platform API is
provided by a mapper.
Handlers expose the native control that implements the .NET MAUI cross-platform control via the NativeView
property. This property can be accessed to set platform properties, invoke platform methods, and subscribe to
platform events.
Handlers are global and therefore customization can occur anywhere in your .NET MAUI app. For example,
customizing a handler for a control in your App class will result in all controls of that type being customized in
the app. Similarly, customizing a handler for a control in the first page of your app will also result in all controls
of that type in the app being customized.
Examples
Handlers are typically customized to augment the appearance and behavior of a native platform control.
Handlers can be customized per platform by using the #ifdef compiler directive, to multi-target code based on
the platform. However, you can just as easily use platform-specific folders or files to organize such code.
NOTE
Handler customization can range from the simple to the complex. More complex examples will appear in time.
#if ANDROID
Microsoft.Maui.Handlers.ViewHandler.ViewMapper.AppendToMapping(nameof(IView.Background), (h, v) =>
{
(h.NativeView as
Android.Views.View).SetBackgroundColor(Microsoft.Maui.Graphics.Colors.Cyan.ToNative());
});
#endif
}
}
namespace MyMauiApp
{
public class MyEntry : Entry
{
}
}
You can then customize the EntryHandler to perform the desired modification to MyEntry instances:
namespace MauiApp1
{
public partial class App : Application
{
public App()
{
InitializeComponent();
Microsoft.Maui.Handlers.EntryHandler.EntryMapper.AppendToMapping(nameof(IView.Background),
(handler, view) =>
{
if (view is MyEntry)
{
#if ANDROID
handler.NativeView.SetBackgroundColor(Colors.Red.ToNative());
#elif IOS
handler.NativeView.BackgroundColor = Colors.Red.ToNative();
handler.NativeView.BorderStyle = UIKit.UITextBorderStyle.Line;
#elif WINDOWS
handler.NativeView.Background = Colors.Red.ToNative();
#endif
}
});
}
}
}
Any MyEntry instances will then be customized as per the handler modification.
Layouts
3/18/2022 • 9 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) layout classes allow you to arrange and group UI controls in your
application. Choosing a layout class requires knowledge of how the layout positions its child elements, and how
the layout sizes its child elements. In addition, it may be necessary to nest layouts to create your desired layout.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
StackLayout
A StackLayout organizes elements in a one-dimensional stack, either horizontally or vertically. The Orientation
property specifies the direction of the elements, and the default orientation is Vertical . StackLayout is
typically used to arrange a subsection of the UI on a page.
The following XAML shows how to create a vertical StackLayout containing three Label objects:
<StackLayout Margin="20,35,20,25">
<Label Text="The StackLayout has its Margin property set, to control the rendering position of the
StackLayout." />
<Label Text="The Padding property can be set to specify the distance between the StackLayout and its
children." />
<Label Text="The Spacing property can be set to specify the distance between views in the StackLayout."
/>
</StackLayout>
In a StackLayout, if an element's size is not explicitly set, it expands to fill the available width, or height if the
Orientation property is set to Horizontal .
A StackLayout is often used as a parent layout, which contains other child layouts. However, a StackLayout
should not be used to reproduce a Grid layout by using a combination of StackLayout objects. The following
code shows an example of this bad practice:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="Details.HomePage"
Padding="0,20,0,0">
<StackLayout>
<StackLayout Orientation="Horizontal">
<Label Text="Name:" />
<Entry Placeholder="Enter your name" />
</StackLayout>
<StackLayout Orientation="Horizontal">
<Label Text="Age:" />
<Entry Placeholder="Enter your age" />
</StackLayout>
<StackLayout Orientation="Horizontal">
<Label Text="Occupation:" />
<Entry Placeholder="Enter your occupation" />
</StackLayout>
<StackLayout Orientation="Horizontal">
<Label Text="Address:" />
<Entry Placeholder="Enter your address" />
</StackLayout>
</StackLayout>
</ContentPage>
This is wasteful because unnecessary layout calculations are performed. Instead, the desired layout can be better
achieved by using a Grid .
For more information, see StackLayout.
HorizontalStackLayout
A HorizontalStackLayout organizes child views in a one-dimensional horizontal stack, and is a more performant
alternative to a StackLayout . HorizontalStackLayout is typically used to arrange a subsection of the UI on a
page.
The following XAML shows how to create a HorizontalStackLayout containing different child views:
<HorizontalStackLayout Margin="20">
<Rectangle Fill="Red"
HeightRequest="30"
WidthRequest="30" />
<Label Text="Red"
FontSize="Large" />
</HorizontalStackLayout>
In a HorizontalStackLayout , if an element's size is not explicitly set, it expands to fill the available height.
For more information, see HorizontalStackLayout.
VerticalStackLayout
A VerticalStackLayout organizes child views in a one-dimensional vertical stack, and is a more performant
alternative to a StackLayout . VerticalStackLayout is typically used to arrange a subsection of the UI on a page.
The following XAML shows how to create a VerticalStackLayout containing three Label objects:
<VerticalStackLayout Margin="20,35,20,25">
<Label Text="The VericalStackLayout has its Margin property set, to control the rendering position of
the VerticalStackLayout." />
<Label Text="The Padding property can be set to specify the distance between the VerticalStackLayout and
its children." />
<Label Text="The Spacing property can be set to specify the distance between views in the
VerticalStackLayout." />
</VerticalStackLayout>
In a VerticalStackLayout , if an element's size is not explicitly set, it expands to fill the available width.
For more information, see VerticalStackLayout.
Grid
A Grid is used for displaying elements in rows and columns, which can have proportional or absolute sizes. A
grid's rows and columns are specified with the RowDefinitions and ColumnDefinitions properties.
To position elements in specific Grid cells, use the Grid.Column and Grid.Row attached properties. To make
elements span across multiple rows and columns, use the Grid.RowSpan and Grid.ColumnSpan attached
properties.
NOTE
A Grid layout should not be confused with tables, and is not intended to present tabular data.
The following XAML shows how to create a Grid with two rows and two columns:
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="50" />
<RowDefinition Height="50" />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="Auto" />
<ColumnDefinition />
</Grid.ColumnDefinitions>
<Label Text="Column 0, Row 0"
WidthRequest="200" />
<Label Grid.Column="1"
Text="Column 1, Row 0" />
<Label Grid.Row="1"
Text="Column 0, Row 1" />
<Label Grid.Column="1"
Grid.Row="1"
Text="Column 1, Row 1" />
</Grid>
Space can be distributed within a column or row by using auto sizing, which lets columns and rows size to fit
their content. This is achieved by setting the height of a RowDefinition , or the width of a ColumnDefinition , to
Auto . Proportional sizing can also be used to distribute available space among the rows and columns of the
grid by weighted proportions. This is achieved by setting the height of a RowDefinition , or the width of a
ColumnDefinition , to a value that uses the * operator.
Cau t i on
Try to ensure that as few rows and columns as possible are set to Auto size. Each auto-sized row or column will
cause the layout engine to perform additional layout calculations. Instead, use fixed size rows and columns if
possible. Alternatively, set rows and columns to occupy a proportional amount of space with the
GridUnitType.Star enumeration value.
FlexLayout
A FlexLayout is similar to a StackLayout in that it displays child elements either horizontally or vertically in a
stack. However, a FlexLayout can also wrap its children if there are too many to fit in a single row or column,
and also enables more granular control of the size, orientation, and alignment of its child elements.
The following XAML shows how to create a FlexLayout that displays its views in a single column:
<FlexLayout Direction="Column"
AlignItems="Center"
JustifyContent="SpaceEvenly">
<Label Text="FlexLayout in Action" />
<Button Text="Button" />
<Label Text="Another Label" />
</FlexLayout>
AbsoluteLayout
An AbsoluteLayout is used to position and size elements using explicit values, or values relative to the size of the
layout. The position is specified by the upper-left corner of the child relative to the upper-left corner of the
AbsoluteLayout .
An AbsoluteLayout should be regarded as a special-purpose layout to be used only when you can impose a size
on children, or when the element's size doesn't affect the positioning of other children. A standard use of this
layout is to create an overlay, which covers the page with other controls, perhaps to protect the user from
interacting with the normal controls on the page.
IMPORTANT
The HorizontalOptions and VerticalOptions properties have no effect on children of an AbsoluteLayout .
Within an AbsoluteLayout , the AbsoluteLayout.LayoutBounds attached property is used to specify the horizontal
position, vertical position, width and height of an element. In addition, the AbsoluteLayout.LayoutFlags attached
property specifies how the layout bounds will be interpreted.
The following XAML shows how to arrange elements in an AbsoluteLayout :
<AbsoluteLayout Margin="40">
<BoxView Color="Red"
AbsoluteLayout.LayoutFlags="PositionProportional"
AbsoluteLayout.LayoutBounds="0.5, 0, 100, 100"
Rotation="30" />
<BoxView Color="Green"
AbsoluteLayout.LayoutFlags="PositionProportional"
AbsoluteLayout.LayoutBounds="0.5, 0, 100, 100"
Rotation="60" />
<BoxView Color="Blue"
AbsoluteLayout.LayoutFlags="PositionProportional"
AbsoluteLayout.LayoutBounds="0.5, 0, 100, 100" />
</AbsoluteLayout>
Avoid using the AbsoluteLayout.AutoSize property whenever possible, as it will cause the layout engine to
perform additional layout calculations.
For more information, see AbsoluteLayout.
RelativeLayout
A RelativeLayout is used to position and size elements relative to properties of the layout or sibling elements.
By default, an element is positioned in the upper left corner of the layout. A RelativeLayout can be used to
create UIs that scale proportionally across device sizes.
Within a , positions and sizes are specified as constraints. Constraints have Factor and
RelativeLayout
Constant properties, which can be used to define positions and sizes as multiples (or fractions) of properties of
other objects, plus a constant. In addition, constants can be negative.
NOTE
A RelativeLayout supports positioning elements outside of its own bounds.
WARNING
Avoid using a RelativeLayout whenever possible. It will result in the CPU having to perform significantly more work.
BindableLayout
A BindableLayout enables any layout class that derives from the Layout class to generate its content by binding
to a collection of items, with the option to set the appearance of each item with a DataTemplate .
A bindable layout is populated with data by setting its ItemsSource property to any collection that implements
IEnumerable , and attaching it to a Layout -derived class. The appearance of each item in the bindable layout can
be defined by setting the BindableLayout.ItemTemplate attached property to a DataTemplate .
The following XAML shows how to bind a StackLayout to a collection of items, and define their appearance with
a DataTemplate :
Bindable layouts should only be used when the collection of items to be displayed is small, and scrolling and
selection isn't required.
For more information, see BindableLayout.
AbsoluteLayout
3/18/2022 • 7 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) AbsoluteLayout is used to position and size children using explicit
values. The position is specified by the upper-left corner of the child relative to the upper-left corner of the
AbsoluteLayout , in device-independent units. AbsoluteLayout also implements a proportional positioning and
sizing feature. In addition, unlike some other layout classes, AbsoluteLayout is able to position children so that
they overlap.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
An AbsoluteLayout should be regarded as a special-purpose layout to be used only when you can impose a size
on children, or when the element's size doesn't affect the positioning of other children.
The AbsoluteLayout class defines the following properties:
LayoutBounds, of type Rect , which is an attached property that represents the position and size of a child.
The default value of this property is (0,0,AutoSize,AutoSize).
LayoutFlags , of type AbsoluteLayoutFlags , which is an attached property that indicates whether properties
of the layout bounds used to position and size the child are interpreted proportionally. The default value of
this property is AbsoluteLayoutFlags.None .
These properties are backed by BindableProperty objects, which means that the properties can be targets of
data bindings and styled.
IMPORTANT
The HorizontalOptions and VerticalOptions properties have no effect on children of an AbsoluteLayout .
WARNING
Using absolute values for positioning and sizing children can be problematic, because different devices have different
screen sizes and resolutions. Therefore, the coordinates for the center of the screen on one device may be offset on other
devices.
The following XAML shows an AbsoluteLayout whose children are positioned using absolute values:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="AbsoluteLayoutDemos.Views.XAML.StylishHeaderDemoPage"
Title="Stylish header demo">
<AbsoluteLayout Margin="20">
<BoxView Color="Silver"
AbsoluteLayout.LayoutBounds="0, 10, 200, 5" />
<BoxView Color="Silver"
AbsoluteLayout.LayoutBounds="0, 20, 200, 5" />
<BoxView Color="Silver"
AbsoluteLayout.LayoutBounds="10, 0, 5, 65" />
<BoxView Color="Silver"
AbsoluteLayout.LayoutBounds="20, 0, 5, 65" />
<Label Text="Stylish Header"
FontSize="24"
AbsoluteLayout.LayoutBounds="30, 25" />
</AbsoluteLayout>
</ContentPage>
In this example, the position of each BoxView object is defined using the first two absolute values that are
specified in the AbsoluteLayout.LayoutBounds attached property. The size of each BoxView is defined using the
third and forth values. The position of the Label object is defined using the two absolute values that are
specified in the AbsoluteLayout.LayoutBounds attached property. Size values are not specified for the Label , and
so it's unconstrained and sizes itself. In all cases, the absolute values represent device-independent units.
The following screenshot shows the resulting layout:
The equivalent C# code is shown below:
absoluteLayout.Add(new BoxView
{
Color = Colors.Silver
}, new Rect(0, 10, 200, 5));
absoluteLayout.Add(new BoxView
{
Color = Colors.Silver
}, new Rect(0, 20, 200, 5));
absoluteLayout.Add(new BoxView
{
Color = Colors.Silver
}, new Rect(10, 0, 5, 65));
absoluteLayout.Add(new BoxView
{
Color = Colors.Silver
}, new Rect(20, 0, 5, 65));
absoluteLayout.Add(new Label
{
Text = "Stylish Header",
FontSize = 24
}, new Point(30,25));
In this example, the position and size of each BoxView is defined using a Rect object. The position of the Label
is defined using a Point object.
In C#, it's also possible to set the position and size of a child of an AbsoluteLayout after it has been added to the
layout, using the AbsoluteLayout.SetLayoutBounds method. The first argument to this method is the child, and the
second is a Rect object.
NOTE
An AbsoluteLayout that uses absolute values can position and size children so that they don't fit within the bounds of
the layout.
TIP
The AbsoluteLayoutFlags enumeration is a Flags enumeration, which means that enumeration members can be
combined. This is accomplished in XAML with a comma-separated list, and in C# with the bitwise OR operator.
For example, if you use the SizeProportional flag and set the width of a child to 0.25 and the height to 0.1, the
child will be one-quarter of the width of the AbsoluteLayout and one-tenth the height. The
PositionProportional flag is similar. A position of (0,0) puts the child in the upper-left corner, while a position of
(1,1) puts the child in the lower-right corner, and a position of (0.5,0.5) centers the child within the
AbsoluteLayout .
The following XAML shows an AbsoluteLayout whose children are positioned using proportional values:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="AbsoluteLayoutDemos.Views.XAML.ProportionalDemoPage"
Title="Proportional demo">
<AbsoluteLayout>
<BoxView Color="Blue"
AbsoluteLayout.LayoutBounds="0.5,0,100,25"
AbsoluteLayout.LayoutFlags="PositionProportional" />
<BoxView Color="Green"
AbsoluteLayout.LayoutBounds="0,0.5,25,100"
AbsoluteLayout.LayoutFlags="PositionProportional" />
<BoxView Color="Red"
AbsoluteLayout.LayoutBounds="1,0.5,25,100"
AbsoluteLayout.LayoutFlags="PositionProportional" />
<BoxView Color="Black"
AbsoluteLayout.LayoutBounds="0.5,1,100,25"
AbsoluteLayout.LayoutFlags="PositionProportional" />
<Label Text="Centered text"
AbsoluteLayout.LayoutBounds="0.5,0.5,110,25"
AbsoluteLayout.LayoutFlags="PositionProportional" />
</AbsoluteLayout>
</ContentPage>
In this example, each child is positioned using proportional values but sized using absolute values. This is
accomplished by setting the AbsoluteLayout.LayoutFlags attached property of each child to
PositionProportional . The first two values that are specified in the AbsoluteLayout.LayoutBounds attached
property, for each child, define the position using proportional values. The size of each child is defined with the
third and forth absolute values, using device-independent units.
The following screenshot shows the resulting layout:
The equivalent C# code is shown below:
In this example, the position and size of each child is set with the AbsoluteLayout.SetLayoutBounds method. The
first argument to the method is the child, and the second is a Rect object. The position of each child is set with
proportional values, while the size of each child is set with absolute values, using device-independent units.
NOTE
An AbsoluteLayout that uses proportional values can position and size children so that they don't fit within the bounds
of the layout by using values outside the 0-1 range.
BindableLayout
3/18/2022 • 7 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) bindable layouts enable any layout class that derives from the Layout
class to generate its content by binding to a collection of items, with the option to set the appearance of each
item with a DataTemplate .
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Bindable layouts are provided by the BindableLayout class, which exposes the following attached properties:
ItemsSource – specifies the collection of IEnumerable items to be displayed by the layout.
ItemTemplate – specifies the DataTemplate to apply to each item in the collection of items displayed by the
layout.
ItemTemplateSelector – specifies the DataTemplateSelector that will be used to choose a DataTemplate for
an item at runtime.
NOTE
The ItemTemplate property takes precedence when both the ItemTemplate and ItemTemplateSelector properties
are set.
NOTE
The EmptyViewTemplate property takes precedence when both the EmptyView and EmptyViewTemplate properties
are set.
All of these properties can be attached to the AbsoluteLayout , FlexLayout , Grid , HorizontalStackLayout ,
RelativeLayout , StackLayout , and VerticalStackLayout classes, which all derive from the Layout class.
When the BinableLayout.ItemsSource property is set to a collection of items and attached to a Layout -derived
class, each item in the collection is added to the Layout -derived class for display. The Layout -derived class will
then update its child views when the underlying collection changes.
Bindable layouts should only be used when the collection of items to be displayed is small, and scrolling and
selection isn't required. While scrolling can be provided by wrapping a bindable layout in a ScrollView , this is
not recommended as bindable layouts lack UI virtualization. When scrolling is required, a scrollable view that
includes UI virtualization, such as ListView or CollectionView , should be used. Failure to observe this
recommendation can lead to performance issues.
IMPORTANT
While it's technically possible to attach a bindable layout to any layout class that derives from the Layout class, it's not
always practical to do so, particularly for the AbsoluteLayout , Grid , and RelativeLayout classes. For example,
consider the scenario of wanting to display a collection of data in a Grid using a bindable layout, where each item in the
collection is an object containing multiple properties. Each row in the Grid should display an object from the collection,
with each column in the Grid displaying one of the object's properties. Because the DataTemplate for the bindable
layout can only contain a single object, it's necessary for that object to be a layout class containing multiple views that
each display one of the object's properties in a specific Grid column. While this scenario can be realised with bindable
layouts, it results in a parent Grid containing a child Grid for each item in the bound collection, which is a highly
inefficient and problematic use of the Grid layout.
When the BindableLayout.ItemsSource attached property is set on a layout, but the BindableLayout.ItemTemplate
attached property isn't set, every item in the IEnumerable collection will be displayed by a Label that's created
by the BindableLayout class.
In this example, every item in the TopFollowers collection will be displayed by an Image view defined in the
DataTemplate :
The TechItemTemplateSelector class defines DefaultTemplate and MAUITemplate DataTemplate properties that
are set to different data templates. The OnSelectTemplate method returns the MAUITemplate , which displays an
item in dark red with a heart next to it, when the item is equal to ".NET MAUI". When the item isn't equal to ".NET
MAUI", the OnSelectTemplate method returns the DefaultTemplate , which displays an item using the default
color of a Label :
The result is that when the data bound collection is null , the string set as the EmptyView property value is
displayed:
The result is that when the data bound collection is null , the StackLayout and its child views are displayed.
Similarly, the EmptyViewTemplate can be set to a DataTemplate , which will be displayed when the ItemsSource
property is null , or when the collection specified by the ItemsSource property is null or empty. The
DataTemplate can contain a single view, or a view that contains multiple child views. In addition, the
BindingContext of the EmptyViewTemplate will be inherited from the BindingContext of the BindableLayout . The
following XAML example shows the EmptyViewTemplate property set to a DataTemplate that contains a single
view:
<StackLayout BindableLayout.ItemsSource="{Binding UserWithoutAchievements.Achievements}">
<BindableLayout.EmptyViewTemplate>
<DataTemplate>
<Label Text="{Binding Source={x:Reference usernameLabel}, Path=Text, StringFormat='{0} has no
achievements.'}" />
</DataTemplate>
</BindableLayout.EmptyViewTemplate>
...
</StackLayout>
The result is that when the data bound collection is null , the Label in the DataTemplate is displayed:
NOTE
The EmptyViewTemplate property can't be set via a DataTemplateSelector .
<ContentPage ...>
<ContentPage.Resources>
...
<ContentView x:Key="BasicEmptyView">
<StackLayout>
<Label Text="No achievements."
FontSize="14" />
</StackLayout>
</ContentView>
<ContentView x:Key="AdvancedEmptyView">
<StackLayout>
<Label Text="None."
FontAttributes="Italic"
FontSize="14" />
<Label Text="Try harder and return later?"
FontAttributes="Italic"
FontSize="14" />
</StackLayout>
</ContentView>
</ContentPage.Resources>
<StackLayout>
...
<Switch Toggled="OnEmptyViewSwitchToggled" />
<StackLayout x:Name="stackLayout"
BindableLayout.ItemsSource="{Binding UserWithoutAchievements.Achievements}">
...
</StackLayout>
</StackLayout>
</ContentPage>
The XAML defines two ContentView objects in the page-level ResourceDictionary , with the Switch object
controlling which ContentView object will be set as the EmptyView property value. When the Switch is toggled,
the OnEmptyViewSwitchToggled event handler executes the ToggleEmptyView method:
The ToggleEmptyView method sets the EmptyView property of the stackLayout object to one of the two
ContentView objects stored in the ResourceDictionary , based on the value of the Switch.IsToggled property.
Then, when the data bound collection is null , the ContentView object set as the EmptyView property is
displayed.
FlexLayout
3/18/2022 • 14 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) FlexLayout is a layout that can arrange its children horizontally
and vertically in a stack, and can also wrap its children if there are too many to fit in a single row or column. In
addition, FlexLayout can control orientation and alignment, and adapt to different screen sizes. FlexLayout is
based on the Cascading Style Sheets (CSS) Flexible Box Layout Module.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
IMPORTANT
When items in a FlexLayout are arranged in a column, the FlexLayout has a vertical main axis and a horizontal cross
axis. When items in a FlexLayout are arranged in a row, the FlexLayout has a horizontal main axis and a vertical cross
axis.
NOTE
In XAML, you can specify the value of this property using the enumeration member names in lowercase, uppercase, or
mixed case, or you can use the two additional strings shown in parentheses.
Wrap
The Wrap property, of type FlexWrap , controls whether children are laid out in a single line or in multiple lines.
The FlexWrap enumeration defines the following members:
NoWrap , which indicates that children are laid out in a single line. This is the default value of the Wrap
property.
Wrap , which indicates that items are laid out in multiple lines if needed.
Reverse (or "wrap-reverse" in XAML), which indicates that items are laid out in multiple lines if needed, in
reverse order.
When the Wrap property is set to NoWrap and the main axis is constrained, and the main axis is not wide or tall
enough to fit all the children, the FlexLayout attempts to make the items smaller. You can control the shrink
factor of children with the Shrink attached bindable property.
When the Wrap property is set to Wrap or WrapReverse , the AlignContent property can be used to specify how
the lines should be distributed.
JustifyContent
The JustifyContent property, of type FlexJustify , specifies how space is distributed between and around
children along the main axis. The FlexJustify enumeration defines the following members:
Start (or "flex-start" in XAML), which indicates that children should be aligned at the start. This is the
default value of the JustifyContent property.
Center , which indicates that children should be aligned around the center.
End (or "flex-end" in XAML), which indicates that children should be aligned at the end.
SpaceBetween (or "space-between" in XAML), which indicates that children should be evenly distributed, with
the first child being at the start and the last child being at the end.
SpaceAround (or "space-around" in XAML), which indicates that children should be evenly distributed, with
the first and last children having a half-size space.
SpaceEvenly , which indicates that children should be evenly distributed, with all children having equal space
around them.
AlignItems
The AlignItems property, of type FlexAlignItems , indicates how the layout engine will distribute space between
and around children along the cross axis. The FlexAlignItems enumeration defines the following members:
Stretch , which indicates that children should be stretched out. This is the default value of the AlignItems
property.
Center , which indicates that children should be aligned around the center.
Start (or "flex-start" in XAML), which indicates that children should be aligned at the start.
End (or "flex-end" in XAML), which indicates that children should be aligned at the end.
This is one of two properties that indicates how children are aligned on the cross axis. Within each row, children
are stretched or aligned on the start, center, or end of each item.
For any individual child, the AlignItems setting can be overridden with the AlignSelf attached bindable
property.
AlignContent
The AlignContent property, of type FlexAlignContent , determines how the layout engine will distribute space
between and around children that have been laid out on multiple lines. The FlexAlignContent enumeration
defines the following members:
Stretch , which indicates that children should be stretched out. This is the default value of the AlignContent
property.
Center , which indicates that children should be aligned around the center.
Start (or "flex-start" in XAML), which indicates that children should be aligned at the start.
End (or "flex-end" in XAML), which indicates that children should be aligned at the end.
SpaceBetween (or "space-between" in XAML), which indicates that children should be evenly distributed, with
the first child being at the start and the last child being at the end.
SpaceAround (or "space-around" in XAML), which indicates that children should be evenly distributed, with
the first and last children having a half-size space.
, which indicates that children should be evenly distributed, with all children having equal space
SpaceEvenly
around them.
The AlignContent property has no effect when there is only one row or column.
For any individual child of the FlexLayout , this property overrides the AlignItems property set on the
FlexLayout . The default setting of Auto means to use the AlignItems setting.
In XAML, this property is set on a child without any reference to its FlexLayout parent:
<Label FlexLayout.AlignSelf="Center"
... />
Order
The Order property, of type int , enables you to change the order that children of the FlexLayout are
arranged. The default value of this property is 0.
Usually, children are arranged in the order in which they are added to the FlexLayout . However, this order can
be overridden by setting this property to a non-zero integer value on one or more children. The FlexLayout
then arranges its children based on their Order property values. Children with the same Order property values
are arranged in the order in which they are added to the FlexLayout .
Basis
The Basis property, of type FlexBasis , defines the amount of space that's allocated to a child on the main axis.
The value specified by this property is the size along the main axis of the parent FlexLayout . Therefore, this
property indicates the width of a child when children are arranged in rows, or the height of a child when
children are arranged in columns. This property is called basis because it specifies a size that is the basis of all
subsequent layout.
The FlexBasis type is a structure that enables size to be specified in device-independent units, or as a
percentage of the size of the FlexLayout . The default value of the Basis property is Auto , which means that
the child's requested width or height is used.
In XAML, you can use a number for a size in device-independent units:
<Label FlexLayout.Basis="40"
... />
FlexLayout.SetBasis(label, 40);
<Label FlexLayout.Basis="25%"
... />
The first argument to the FlexBasis constructor is a fractional float value that must be in the range of 0 to 1.
The second argument indicates that the size is relative, rather than absolute.
Grow
The Grow property, of type float , specifies the amount of available space the child should use on the main
axis. The default value of this property is 0.0, and its value must be greater than or equal to 0.
The Grow property is used when the Wrap property is set to NoWrap and a row of children has a total width
less than the width of the FlexLayout , or a column of children has a shorter height than the FlexLayout . The
Grow property indicates how to apportion the leftover space among the children. If a single child is given a
positive Grow value, then that child takes up all the remaining space. Alternatively, the remaining space can also
be allocated among two or more children.
Shrink
The Shrink property, of type float , controls how a child should shrink so that all children can fit inside the
container. The default value of this property is 1.0, and its value must be greater than or equal to 0.
The Shrink property is used when the Wrap property is set to NoWrap and the aggregate width of a row of
children is greater than the width of the FlexLayout , or the aggregate height of a single column of children is
greater than the height of the FlexLayout . Normally the FlexLayout will display these children by constricting
their sizes. The Shrink property can indicate which children are given priority in being displayed at their full
sizes.
TIP
The Grow and Shrink values can both be set to accommodate situations where the aggregate child sizes might
sometimes be less than or sometimes greater than the size of the FlexLayout .
Examples
The following examples demonstrate common uses of FlexLayout .
Stack
A FlexLayout can substitute for a StackLayout :
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="FlexLayoutDemos.Views.SimpleStackPage"
Title="Simple Stack">
<FlexLayout Direction="Column"
AlignItems="Center"
JustifyContent="SpaceEvenly">
<Label Text="FlexLayout in Action"
FontSize="Large" />
<Image Source="dotnet_bot_branded.png"
HeightRequest="300" />
<Button Text="Do-Nothing Button" />
<Label Text="Another Label" />
</FlexLayout>
</ContentPage>
In this example, the Direction property is set to Column , which causes the children of the FlexLayout to be
arranged in a single column. The AlignItems property is set to Center , which causes each child to be
horizontally centered. The JustifyContent property is set to SpaceEvenly which allocates all leftover vertical
space equally between all the children, above the first child and below the last child:
NOTE
The AlignSelf attached property can be used to override the AlignItems property for a specific child.
Wrap items
A FlexLayout can wrap its children to additional rows or columns:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="FlexLayoutDemos.Views.PhotoWrappingPage"
Title="Photo Wrapping">
<Grid>
<ScrollView>
<FlexLayout x:Name="flexLayout"
Wrap="Wrap"
JustifyContent="SpaceAround" />
</ScrollView>
...
</Grid>
</ContentPage>
In this example, the Direction property of the FlexLayout is not set, so it has the default setting of Row ,
meaning that the children are arranged in rows and the main axis is horizontal. The Wrap property is set to to
Wrap , which causes children to wrap to the next row if there are too many children to fit on a row. The
JustifyContent property is set to SpaceAround which allocates all leftover space on the main axis so that each
child is surrounded by the same amount of space:
The code-behind file for this example retrieves a collection of photos and adds them to the FlexLayout .
In addition, the FlexLayout is a child of a ScrollView . Therefore, if there are too many rows to fit on the page,
then the ScrollView has a default Orientation property of Vertical and allows vertical scrolling.
Page layout
There is a standard layout in web design called the holy grail because it's a layout format that is very desirable,
but often hard to realize with perfection. The layout consists of a header at the top of the page and a footer at
the bottom, both extending to the full width of the page. Occupying the center of the page is the main content,
but often with a columnar menu to the left of the content and supplementary information (sometimes called an
aside area) at the right. This layout can be realized with a FlexLayout .
The following example shows an implementation of this layout using a FlexLayout nested in another:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="FlexLayoutDemos.Views.HolyGrailLayoutPage"
Title="Holy Grail Layout">
<FlexLayout Direction="Column">
</FlexLayout>
The navigation and aside areas are rendered with a BoxView on the left and right. The first FlexLayout has a
vertical main axis and contains three children arranged in a column. These are the header, the body of the page,
and the footer. The nested FlexLayout has a horizontal main axis with three children arranged in a row:
In this example, the Order property is set on the first BoxView to a value less than its siblings to cause it to
appear as the first item in the row. The Basis property is set on both BoxView objects to give them a width of
50 device-independent units. The Grow property is set on the nested FlexLayout to indicate that this
FlexLayout should occupy all of the unused vertical space within the outer FlexLayout . In addition, the Grow
property is set on the Label representing the content, to indicate that this content is to occupy all of the unused
horizontal space within the nested FlexLayout .
NOTE
There's also a Shrink property that you can use when the size of children exceeds the size of the FlexLayout but
wrapping is not desired.
Grid
3/18/2022 • 14 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) Grid , is a layout that organizes its children into rows and columns,
which can have proportional or absolute sizes. By default, a Grid contains one row and one column. In addition,
a Grid can be used as a parent layout that contains other child layouts.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The Grid should not be confused with tables, and is not intended to present tabular data. Unlike HTML tables, a
Grid is intended for laying out content.
The Grid class defines the following properties:
Column , of type int , which is an attached property that indicates the column alignment of a view within a
parent Grid . The default value of this property is 0. A validation callback ensures that when the property is
set, its value is greater than or equal to 0.
ColumnDefinitions , of type ColumnDefinitionCollection , is a list of ColumnDefinition objects that define the
width of the grid columns.
ColumnSpacing , of type double , indicates the distance between grid columns. The default value of this
property is 0.
ColumnSpan , of type int , which is an attached property that indicates the total number of columns that a
view spans within a parent Grid . The default value of this property is 1. A validation callback ensures that
when the property is set, its value is greater than or equal to 1.
Row , of type int , which is an attached property that indicates the row alignment of a view within a parent
Grid . The default value of this property is 0. A validation callback ensures that when the property is set, its
value is greater than or equal to 0.
RowDefinitions , of type RowDefinitionCollection , is a list of RowDefintion objects that define the height of
the grid rows.
RowSpacing , of type double , indicates the distance between grid rows. The default value of this property is 0.
RowSpan , of type int , which is an attached property that indicates the total number of rows that a view
spans within a parent Grid . The default value of this property is 1. A validation callback ensures that when
the property is set, its value is greater than or equal to 1.
These properties are backed by BindableProperty objects, which means that the properties can be targets of
data bindings and styled.
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="GridTutorial.MainPage">
<Grid Margin="20,35,20,20">
<Label Text="By default, a Grid contains one row and one column." />
</Grid>
</ContentPage>
In this example, the Grid contains a single child Label that's automatically positioned in a single location:
The layout behavior of a Grid can be defined with the RowDefinitions and ColumnDefinitions properties,
which are collections of RowDefinition and ColumnDefinition objects, respectively. These collections define the
row and column characteristics of a Grid , and should contain one RowDefinition object for each row in the
Grid , and one ColumnDefinition object for each column in the Grid .
The RowDefinition class defines a Height property, of type GridLength , and the ColumnDefinition class
defines a Width property, of type GridLength . The GridLength struct specifies a row height or a column width
in terms of the GridUnitType enumeration, which has three members:
Absolute – the row height or column width is a value in device-independent units (a number in XAML).
Auto – the row height or column width is autosized based on the cell contents ( Auto in XAML).
Star – leftover row height or column width is allocated proportionally (a number followed by * in XAML).
A Grid row with a Height property of Auto constrains the height of views in that row in the same way as a
vertical StackLayout . Similarly, a column with a Width property of Auto works much like a horizontal
StackLayout .
Cau t i on
Try to ensure that as few rows and columns as possible are set to Auto size. Each auto-sized row or column will
cause the layout engine to perform additional layout calculations. Instead, use fixed size rows and columns if
possible. Alternatively, set rows and columns to occupy a proportional amount of space with the
GridUnitType.Star enumeration value.
The following XAML shows how to create a Grid with three rows and two columns:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="GridDemos.Views.XAML.BasicGridPage"
Title="Basic Grid demo">
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="2*" />
<RowDefinition Height="*" />
<RowDefinition Height="100" />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="*" />
<ColumnDefinition Width="*" />
</Grid.ColumnDefinitions>
...
</Grid>
</ContentPage>
In this example, the Grid has an overall height that is the height of the page. The Grid knows that the height of
the third row is 100 device-independent units. It subtracts that height from its own height, and allocates the
remaining height proportionally between the first and second rows based on the number before the star. In this
example, the height of the first row is twice that of the second row.
The two ColumnDefinition objects both set the Width to * , which is the same as 1* , meaning that the width
of the screen is divided equally beneath the two columns.
IMPORTANT
The default value of the RowDefinition.Height property is * . Similarly, the default value of the
ColumnDefinition.Width property is * . Therefore, it's not necessary to set these properties in cases where these
defaults are acceptable.
Child views can be positioned in specific Grid cells with the Grid.Column and Grid.Row attached properties. In
addition, to make child views span across multiple rows and columns, use the Grid.RowSpan and
Grid.ColumnSpan attached properties.
The following XAML shows the same Grid definition, and also positions child views in specific Grid cells:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="GridDemos.Views.XAML.BasicGridPage"
Title="Basic Grid demo">
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="2*" />
<RowDefinition />
<RowDefinition Height="100" />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition />
<ColumnDefinition />
</Grid.ColumnDefinitions>
<BoxView Color="Green" />
<Label Text="Row 0, Column 0"
HorizontalOptions="Center"
VerticalOptions="Center" />
<BoxView Grid.Column="1"
Color="Blue" />
<Label Grid.Column="1"
Text="Row 0, Column 1"
HorizontalOptions="Center"
VerticalOptions="Center" />
<BoxView Grid.Row="1"
Color="Teal" />
<Label Grid.Row="1"
Text="Row 1, Column 0"
HorizontalOptions="Center"
VerticalOptions="Center" />
<BoxView Grid.Row="1"
Grid.Column="1"
Color="Purple" />
<Label Grid.Row="1"
Grid.Column="1"
Text="Row1, Column 1"
HorizontalOptions="Center"
VerticalOptions="Center" />
<BoxView Grid.Row="2"
Grid.ColumnSpan="2"
Color="Red" />
<Label Grid.Row="2"
Grid.ColumnSpan="2"
Text="Row 2, Columns 0 and 1"
HorizontalOptions="Center"
VerticalOptions="Center" />
</Grid>
</ContentPage>
NOTE
The Grid.Row and Grid.Column properties are both indexed from 0, and so Grid.Row="2" refers to the third row
while Grid.Column="1" refers to the second column. In addition, both of these properties have a default value of 0, and
so don't need to be set on child views that occupy the first row or first column of a Grid .
In this example, all three Grid rows are occupied by BoxView and Label views. The third row is 100 device-
independent units high, with the first two rows occupying the remaining space (the first row is twice as high as
the second row). The two columns are equal in width and divide the Grid in half. The BoxView in the third row
spans both columns:
In addition, child views in a Grid can share cells. The order that the children appear in the XAML is the order
that the children are placed in the Grid . In the previous example, the Label objects are only visible because
they are rendered on top of the BoxView objects. The Label objects would not be visible if the BoxView objects
were rendered on top of them.
The equivalent C# code is:
// Row 0
// The BoxView and Label are in row 0 and column 0, and so only need to be added to the
// Grid to obtain the default row and column settings.
grid.Add(new BoxView
{
Color = Colors.Green
});
grid.Add(new Label
{
Text = "Row 0, Column 0",
HorizontalOptions = LayoutOptions.Center,
VerticalOptions = LayoutOptions.Center
VerticalOptions = LayoutOptions.Center
});
// This BoxView and Label are in row 0 and column 1, which are specified as arguments
// to the Add method.
grid.Add(new BoxView
{
Color = Colors.Blue
}, 1, 0);
grid.Add(new Label
{
Text = "Row 0, Column 1",
HorizontalOptions = LayoutOptions.Center,
VerticalOptions = LayoutOptions.Center
}, 1, 0);
// Row 1
// This BoxView and Label are in row 1 and column 0, which are specified as arguments
// to the Add method overload.
grid.Add(new BoxView
{
Color = Colors.Teal
}, 0, 1);
grid.Add(new Label
{
Text = "Row 1, Column 0",
HorizontalOptions = LayoutOptions.Center,
VerticalOptions = LayoutOptions.Center
}, 0, 1);
// This BoxView and Label are in row 1 and column 1, which are specified as arguments
// to the Add method overload.
grid.Add(new BoxView
{
Color = Colors.Purple
}, 1, 1);
grid.Add(new Label
{
Text = "Row1, Column 1",
HorizontalOptions = LayoutOptions.Center,
VerticalOptions = LayoutOptions.Center
}, 1, 1);
// Row 2
// Alternatively, the BoxView and Label can be positioned in cells with the Grid.SetRow
// and Grid.SetColumn methods.
BoxView boxView = new BoxView { Color = Colors.Red };
Grid.SetRow(boxView, 2);
Grid.SetColumnSpan(boxView, 2);
Label label = new Label
{
Text = "Row 2, Column 0 and 1",
HorizontalOptions = LayoutOptions.Center,
VerticalOptions = LayoutOptions.Center
};
Grid.SetRow(label, 2);
Grid.SetColumnSpan(label, 2);
grid.Add(boxView);
grid.Add(label);
In code, to specify the height of a RowDefinition object, and the width of a ColumnDefinition object, you use
values of the GridLength structure, often in combination with the GridUnitType enumeration.
Simplify row and column definitions
In XAML, the row and column characteristics of a Grid can be specified using a simplified syntax that avoids
having to define RowDefinition and ColumnDefinition objects for each row and column. Instead, the
RowDefinitions and ColumnDefinitions properties can be set to strings containing comma-delimited
GridUnitType values, from which type converters built into .NET MAUI create RowDefinition and
ColumnDefinition objects:
In this example, the Grid has five rows and four columns. The third, forth, and fifth rows are set to absolute
heights, with the second row auto-sizing to its content. The remaining height is then allocated to the first row.
The forth column is set to an absolute width, with the third column auto-sizing to its content. The remaining
width is allocated proportionally between the first and second columns based on the number before the star. In
this example, the width of the second column is twice that of the first column (because * is identical to 1* ).
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="GridDemos.Views.XAML.GridSpacingPage"
Title="Grid spacing demo">
<Grid RowSpacing="6"
ColumnSpacing="6">
...
</Grid>
</ContentPage>
This example creates a Grid whose rows and columns are separated by 6 device-independent units of space:
TIP
The RowSpacing and ColumnSpacing properties can be set to negative values to make cell contents overlap.
Content = grid;
}
}
Alignment
Child views in a Grid can be positioned within their cells by the HorizontalOptions and VerticalOptions
properties. These properties can be set to the following fields from the LayoutOptions struct:
Start
Center
End
Fill
The following XAML creates a Grid with nine equal-size cells, and places a Label in each cell with a different
alignment:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="GridDemos.Views.XAML.GridAlignmentPage"
Title="Grid alignment demo">
<Grid>
<Grid.RowDefinitions>
<RowDefinition />
<RowDefinition />
<RowDefinition />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition />
<ColumnDefinition />
<ColumnDefinition />
</Grid.ColumnDefinitions>
In this example, the Label objects in each row are all identically aligned vertically, but use different horizontal
alignments. Alternatively, this can be thought of as the Label objects in each column being identically aligned
horizontally, but using different vertical alignments:
// Row 0
grid.Add(new BoxView
{
Color = Colors.AliceBlue
});
grid.Add(new Label
{
Text = "Upper left",
HorizontalOptions = LayoutOptions.Start,
VerticalOptions = LayoutOptions.Start
});
grid.Add(new BoxView
{
Color = Colors.LightSkyBlue
}, 1, 0);
grid.Add(new Label
{
Text = "Upper center",
HorizontalOptions = LayoutOptions.Center,
VerticalOptions = LayoutOptions.Start
}, 1, 0);
grid.Add(new BoxView
{
Color = Colors.CadetBlue
}, 2, 0);
grid.Add(new Label
{
Text = "Upper right",
HorizontalOptions = LayoutOptions.End,
VerticalOptions = LayoutOptions.Start
}, 2, 0);
// Row 1
grid.Add(new BoxView
{
Color = Colors.CornflowerBlue
}, 0, 1);
grid.Add(new Label
{
Text = "Center left",
HorizontalOptions = LayoutOptions.Start,
VerticalOptions = LayoutOptions.Center
}, 0, 1);
grid.Add(new BoxView
{
Color = Colors.DodgerBlue
}, 1, 1);
grid.Add(new Label
{
Text = "Center center",
HorizontalOptions = LayoutOptions.Center,
VerticalOptions = LayoutOptions.Center
}, 1, 1);
grid.Add(new BoxView
{
Color = Colors.DarkSlateBlue
}, 2, 1);
grid.Add(new Label
{
Text = "Center right",
HorizontalOptions = LayoutOptions.End,
VerticalOptions = LayoutOptions.Center
VerticalOptions = LayoutOptions.Center
}, 2, 1);
// Row 2
grid.Add(new BoxView
{
Color = Colors.SteelBlue
}, 0, 2);
grid.Add(new Label
{
Text = "Lower left",
HorizontalOptions = LayoutOptions.Start,
VerticalOptions = LayoutOptions.End
}, 0, 2);
grid.Add(new BoxView
{
Color = Colors.LightBlue
}, 1, 2);
grid.Add(new Label
{
Text = "Lower center",
HorizontalOptions = LayoutOptions.Center,
VerticalOptions = LayoutOptions.End
}, 1, 2);
grid.Add(new BoxView
{
Color = Colors.BlueViolet
}, 2, 2);
grid.Add(new Label
{
Text = "Lower right",
HorizontalOptions = LayoutOptions.End,
VerticalOptions = LayoutOptions.End
}, 2, 2);
<ContentPage.Resources>
<converters:DoubleToIntConverter x:Key="doubleToInt" />
<Style TargetType="Label">
<Setter Property="HorizontalTextAlignment"
Value="Center" />
</Style>
</ContentPage.Resources>
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="500" />
<RowDefinition Height="Auto" />
</Grid.RowDefinitions>
<BoxView x:Name="boxView"
Color="Black" />
<Grid Grid.Row="1"
Margin="20">
<Grid.RowDefinitions>
<RowDefinition />
<RowDefinition />
<RowDefinition />
<RowDefinition />
<RowDefinition />
<RowDefinition />
</Grid.RowDefinitions>
<Slider x:Name="redSlider"
ValueChanged="OnSliderValueChanged" />
<Label Grid.Row="1"
Text="{Binding Source={x:Reference redSlider},
Path=Value,
Converter={StaticResource doubleToInt},
ConverterParameter=255,
StringFormat='Red = {0}'}" />
<Slider x:Name="greenSlider"
Grid.Row="2"
ValueChanged="OnSliderValueChanged" />
<Label Grid.Row="3"
Text="{Binding Source={x:Reference greenSlider},
Path=Value,
Converter={StaticResource doubleToInt},
ConverterParameter=255,
StringFormat='Green = {0}'}" />
<Slider x:Name="blueSlider"
Grid.Row="4"
ValueChanged="OnSliderValueChanged" />
<Label Grid.Row="5"
Text="{Binding Source={x:Reference blueSlider},
Path=Value,
Converter={StaticResource doubleToInt},
ConverterParameter=255,
StringFormat='Blue = {0}'}" />
</Grid>
</Grid>
</ContentPage>
In this example, the root Grid contains a BoxView in its first row, and a child Grid in its second row. The child
Grid contains Slider objects that manipulate the color displayed by the BoxView , and Label objects that
display the value of each Slider :
IMPORTANT
The deeper you nest Grid objects and other layouts, the more the nested layouts will impact performance.
public ColorSlidersGridPage()
{
// Create an implicit style for the Labels
Style labelStyle = new Style(typeof(Label))
{
Setters =
{
new Setter { Property = Label.HorizontalTextAlignmentProperty, Value = TextAlignment.Center
}
}
};
Resources.Add(labelStyle);
The .NET Multi-platform App UI (.NET MAUI) HorizontalStackLayout organizes child views in a one-dimensional
horizontal stack, and is a more performant alternative to a StackLayout . In addition, a HorizontalStackLayout
can be used as a parent layout that contains other child layouts.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="StackLayoutDemos.Views.HorizontalStackLayoutPage">
<HorizontalStackLayout Margin="20">
<Rectangle Fill="Red"
HeightRequest="30"
WidthRequest="30" />
<Label Text="Red"
FontSize="Large" />
</HorizontalStackLayout>
</ContentPage>
This example creates a HorizontalStackLayout containing a Rectangle and a Label object. By default, there is
no space between the child views:
NOTE
The value of the Margin property represents the distance between an element and its adjacent elements.
This example creates a HorizontalStackLayout containing a Rectangle and a Label object, that have ten
device-independent units of space between them:
TIP
The Spacing property can be set to negative values to make child views overlap.
The VerticalOptions properties of a HorizontalStackLayout , and its child views, can be set to fields from the
LayoutOptions struct, which encapsulates an alignment layout preference. This layout preference determines the
position and size of a child view within its parent layout.
The following XAML example sets alignment preferences on each child view in the HorizontalStackLayout :
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="StackLayoutDemos.Views.HorizontalStackLayoutPage">
<HorizontalStackLayout Margin="20"
HeightRequest="200">
<Label Text="Start"
BackgroundColor="Gray"
VerticalOptions="Start" />
<Label Text="Center"
BackgroundColor="Gray"
VerticalOptions="Center" />
<Label Text="End"
BackgroundColor="Gray"
VerticalOptions="End" />
<Label Text="Fill"
BackgroundColor="Gray"
VerticalOptions="Fill" />
</HorizontalStackLayout>
</ContentPage>
In this example, alignment preferences are set on the Label objects to control their position within the
HorizontalStackLayout . The Start , Center , End , and Fill fields are used to define the alignment of the
Label objects within the parent HorizontalStackLayout :
A HorizontalStackLayout only respects the alignment preferences on child views that are in the opposite
direction to the orientation of the layout. Therefore, the Label child views within the HorizontalStackLayout set
their VerticalOptions properties to one of the alignment fields:
Start , which positions the Label at the start of the HorizontalStackLayout .
Center , which vertically centers the Label in the HorizontalStackLayout .
End , which positions the Label at the end of the HorizontalStackLayout .
Fill , which ensures that the Label fills the height of the HorizontalStackLayout .
<ContentPage xmlns="http://xamarin.com/schemas/2014/forms"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="StackLayoutDemos.Views.HorizontalStackLayoutPage">
<HorizontalStackLayout Margin="20"
Spacing="6">
<Label Text="Primary colors:" />
<VerticalStackLayout Spacing="6">
<Rectangle Fill="Red"
WidthRequest="30"
HeightRequest="30" />
<Rectangle Fill="Yellow"
WidthRequest="30"
HeightRequest="30" />
<Rectangle Fill="Blue"
WidthRequest="30"
HeightRequest="30" />
</VerticalStackLayout>
<Label Text="Secondary colors:" />
<VerticalStackLayout Spacing="6">
<Rectangle Fill="Green"
WidthRequest="30"
HeightRequest="30" />
<Rectangle Fill="Orange"
WidthRequest="30"
HeightRequest="30" />
<Rectangle Fill="Purple"
WidthRequest="30"
HeightRequest="30" />
</VerticalStackLayout>
</HorizontalStackLayout>
</ContentPage>
In this example, the parent HorizontalStackLayout contains two nested HorizontalStackLayout objects:
IMPORTANT
The deeper you nest layout objects, the more the nested layouts will impact performance.
RelativeLayout
3/18/2022 • 10 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) RelativeLayout , which is available in the
Microsoft.Maui.Controls.Compatibility namespace, is used to position and size children relative to properties of
the layout or sibling elements. This allows UIs to be created that scale proportionally across device sizes. In
addition, unlike some other layout classes, RelativeLayout is able to position children so that they overlap.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Constraints
Within a RelativeLayout , the position and size of children are specified as constraints using absolute values or
relative values. When constraints aren't specified, a child will be positioned in the upper left corner of the layout.
The following table shows how to specify constraints in XAML and C#:
XA M L C#
Absolute values Absolute constraints are specified by Absolute constraints are specified by
setting the RelativeLayout attached the Constraint.Constant method,
properties to double values. or by using the Children.Add
overload that requires a Func<Rect>
argument.
Relative values Relative constraints are specified by Relative constraints are specified by
setting the RelativeLayout attached Constraint objects that are returned
properties to Constraint objects by methods of the Constraint class.
that are returned by the
ConstraintExpression markup
extension.
For more information about specifying constraints using absolute values, see Absolute positioning and sizing.
For more information about specifying constraints using relative values, see Relative positioning and sizing.
In C#, children can be added to RelativeLayout by three Add overloads. The first overload requires a
Expression<Func<Rect>> to specify the position and size of a child. The second overload requires optional
Expression<Func<double>> objects for the x , y , width , and height arguments. The third overload requires
optional Constraint objects for the x , y , width , and height arguments.
It's possible to change the position and size of a child in a RelativeLayout with the SetXConstraint ,
SetYConstraint , SetWidthConstraint , and SetHeightConstraint methods. The first argument to each of these
methods is the child, and the second is a Constraint object. In addition, the SetBoundsConstraint method can
also be used to change the position and size of a child. The first argument to this method is the child, and the
second is a BoundsConstraint object.
WARNING
Using absolute values for positioning and sizing children can be problematic, because different devices have different
screen sizes and resolutions. Therefore, the coordinates for the center of the screen on one device may be offset on other
devices.
The following XAML shows a RelativeLayout whose children are positioned using absolute values:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:compat="clr-
namespace:Microsoft.Maui.Controls.Compatibility;assembly=Microsoft.Maui.Controls"
x:Class="RelativeLayoutDemos.Views.XAML.StylishHeaderDemoPage"
Title="Stylish header demo">
<compat:RelativeLayout Margin="20">
<BoxView Color="Silver"
compat:RelativeLayout.XConstraint="0"
compat:RelativeLayout.YConstraint="10"
compat:RelativeLayout.WidthConstraint="200"
compat:RelativeLayout.HeightConstraint="5" />
<BoxView Color="Silver"
compat:RelativeLayout.XConstraint="0"
compat:RelativeLayout.YConstraint="20"
compat:RelativeLayout.WidthConstraint="200"
compat:RelativeLayout.HeightConstraint="5" />
<BoxView Color="Silver"
compat:RelativeLayout.XConstraint="10"
compat:RelativeLayout.YConstraint="0"
compat:RelativeLayout.WidthConstraint="5"
compat:RelativeLayout.HeightConstraint="65" />
<BoxView Color="Silver"
compat:RelativeLayout.XConstraint="20"
compat:RelativeLayout.YConstraint="0"
compat:RelativeLayout.WidthConstraint="5"
compat:RelativeLayout.HeightConstraint="65" />
<Label Text="Stylish header"
FontSize="24"
compat:RelativeLayout.XConstraint="30"
compat:RelativeLayout.YConstraint="25" />
</compat:RelativeLayout>
</ContentPage>
In this example, the position of each BoxView object is defined using the values specified in the XConstraint
and YConstraint attached properties. The size of each BoxView is defined using the values specified in the
WidthConstraint and HeightConstraint attached properties. The position of the Label object is also defined
using the values specified in the XConstraint and YConstraint attached properties. However, size values are
not specified for the Label , and so it's unconstrained and sizes itself. In all cases, the absolute values represent
device-independent units.
The following screenshot shows the resulting layout:
relativeLayout.Children.Add(new BoxView
{
Color = Colors.Silver
}, () => new Rect(0, 10, 200, 5));
relativeLayout.Children.Add(new BoxView
{
Color = Colors.Silver
}, () => new Rect(0, 20, 200, 5));
relativeLayout.Children.Add(new BoxView
{
Color = Colors.Silver
}, () => new Rect(10, 0, 5, 65));
relativeLayout.Children.Add(new BoxView
{
Color = Colors.Silver
}, () => new Rect(20, 0, 5, 65));
relativeLayout.Children.Add(new Label
{
Text = "Stylish Header",
FontSize = 24
}, Constraint.Constant(30), Constraint.Constant(25));
In this example, objects are added to the RelativeLayout using an Add overload that requires a
BoxView
Expression<Func<Rect>> to specify the position and size of each child. The position of the Label is defined using
an Add overload that requires optional Constraint objects, in this case created by the Constraint.Constant
method.
NOTE
A RelativeLayout that uses absolute values can position and size children so that they don't fit within the bounds of
the layout.
The following XAML shows a RelativeLayout whose children are constrained by the ConstraintExpression
markup extension:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:compat="clr-
namespace:Microsoft.Maui.Controls.Compatibility;assembly=Microsoft.Maui.Controls"
x:Class="RelativeLayoutDemos.Views.XAML.RelativePositioningAndSizingDemoPage"
Title="RelativeLayout demo">
<compat:RelativeLayout>
<!-- Four BoxView's -->
<BoxView Color="Red"
compat:RelativeLayout.XConstraint="{compat:ConstraintExpression Type=Constant, Constant=0}"
compat:RelativeLayout.YConstraint="{compat:ConstraintExpression Type=Constant, Constant=0}"
/>
<BoxView Color="Green"
compat:RelativeLayout.XConstraint="{compat:ConstraintExpression Type=RelativeToParent,
Property=Width, Constant=-40}"
compat:RelativeLayout.YConstraint="{compat:ConstraintExpression Type=Constant, Constant=0}"
/>
<BoxView Color="Blue"
compat:RelativeLayout.XConstraint="{compat:ConstraintExpression Type=Constant, Constant=0}"
compat:RelativeLayout.YConstraint="{compat:ConstraintExpression Type=RelativeToParent,
Property=Height, Constant=-40}" />
<BoxView Color="Yellow"
compat:RelativeLayout.XConstraint="{compat:ConstraintExpression Type=RelativeToParent,
Property=Width, Constant=-40}"
compat:RelativeLayout.YConstraint="{compat:ConstraintExpression Type=RelativeToParent,
Property=Height, Constant=-40}" />
In this example, the position of each BoxView object is defined by setting the XConstraint and YConstraint
attached properties. The first BoxView has its XConstraint and YConstraint attached properties set to
constants, which are absolute values. The remaining BoxView objects all have their position set by using at least
one relative value. For example, the yellow BoxView object sets the XConstraint attached property to the width
of its parent (the RelativeLayout minus 40). Similarly, this BoxView sets the YConstraint attached property to
the height of its parent minus 40. This ensures that the yellow BoxView appears in the lower-right corner of the
screen.
NOTE
BoxView objects that don't specify a size are automatically sized to 40x40 by Xamarin.Forms.
The silver BoxView named oneThird is positioned centrally, relative to its parent. It's also sized relative to its
parent, being one third of its width and height. This is achieved by setting the XConstraint and WidthConstraint
attached properties to the width of the parent (the RelativeLayout , multiplied by 0.33). Similarly, the
YConstraint and HeightConstraint attached properties are set to the height of the parent, multiplied by 0.33.
The black BoxView is positioned and sized relative to the oneThird BoxView . This is achieved by setting its
XConstraint and YConstraint attached properties to the X and Y values, respectively, of the sibling element.
Similarly, its size is set to one third of the width and height of its sibling element. This is achieved by setting its
WidthConstraint and HeightConstraint attached properties to the Width and Height values of the sibling
element, respectively, which are then multiplied by 0.33.
The following screenshot shows the resulting layout:
Constraint objects
The Constraint class defines the following public static methods, which return Constraint objects:
Constant , which constrains a child to a size specified with a double .
FromExpression , which constrains a child using a lambda expression.
RelativeToParent , which constrains a child relative to its parent's size.
RelativeToView , which constrains a child relative to the size of a view.
In addition, the BoundsConstraint class defines a single method, FromExpression , which returns a
BoundsConstraint that constrains a child's position and size with a Expression<Func<Rect>> . This method can be
used to set the BoundsConstraint attached property.
The following C# code shows a RelativeLayout whose children are constrained by Constraint objects:
// Four BoxView's
relativeLayout.Children.Add(
new BoxView { Color = Colors.Red },
Constraint.Constant(0),
Constraint.Constant(0));
relativeLayout.Children.Add(
new BoxView { Color = Colors.Green },
Constraint.RelativeToParent((parent) =>
{
return parent.Width - 40;
}), Constraint.Constant(0));
relativeLayout.Children.Add(
new BoxView { Color = Colors.Blue },
Constraint.Constant(0),
Constraint.RelativeToParent((parent) =>
{
return parent.Height - 40;
}));
relativeLayout.Children.Add(
new BoxView { Color = Colors.Yellow },
Constraint.RelativeToParent((parent) =>
{
return parent.Width - 40;
}),
Constraint.RelativeToParent((parent) =>
{
return parent.Height - 40;
}));
In this example, children are added to the RelativeLayout using the Add overload that requires an optional
Constraint object for the x , y , width , and height arguments.
NOTE
A RelativeLayout that uses relative values can position and size children so that they don't fit within the bounds of the
layout.
StackLayout
3/18/2022 • 7 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) StackLayout organizes child views in a one-dimensional stack,
either horizontally or vertically. By default, a StackLayout is oriented vertically. In addition, a StackLayout can
be used as a parent layout that contains other child layouts.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Vertical orientation
The following XAML shows how to create a vertically oriented StackLayout that contains different child views:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="StackLayoutDemos.Views.XAML.VerticalStackLayoutPage"
Title="Vertical StackLayout demo">
<StackLayout Margin="20">
<Label Text="Primary colors" />
<BoxView Color="Red"
HeightRequest="40" />
<BoxView Color="Yellow"
HeightRequest="40" />
<BoxView Color="Blue"
HeightRequest="40" />
<Label Text="Secondary colors" />
<BoxView Color="Green"
HeightRequest="40" />
<BoxView Color="Orange"
HeightRequest="40" />
<BoxView Color="Purple"
HeightRequest="40" />
</StackLayout>
</ContentPage>
This example creates a vertical StackLayout containing Label and BoxView objects. By default, there's no space
between the child views:
Content = stackLayout;
}
}
NOTE
The value of the Margin property represents the distance between an element and its adjacent elements.
Horizontal orientation
The following XAML shows how to create a horizontally oriented StackLayout by setting its Orientation
property to Horizontal :
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="StackLayoutDemos.Views.XAML.HorizontalStackLayoutPage"
Title="Horizontal StackLayout demo">
<StackLayout Margin="20"
Orientation="Horizontal"
HorizontalOptions="Center">
<BoxView Color="Red"
WidthRequest="40" />
<BoxView Color="Yellow"
WidthRequest="40" />
<BoxView Color="Blue"
WidthRequest="40" />
<BoxView Color="Green"
WidthRequest="40" />
<BoxView Color="Orange"
WidthRequest="40" />
<BoxView Color="Purple"
WidthRequest="40" />
</StackLayout>
</ContentPage>
This example creates a horizontal StackLayout containing BoxView objects, with no space between the child
views:
Content = stackLayout;
}
}
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="StackLayoutDemos.Views.XAML.StackLayoutSpacingPage"
Title="StackLayout Spacing demo">
<StackLayout Margin="20"
Spacing="6">
<Label Text="Primary colors" />
<BoxView Color="Red"
HeightRequest="40" />
<BoxView Color="Yellow"
HeightRequest="40" />
<BoxView Color="Blue"
HeightRequest="40" />
<Label Text="Secondary colors" />
<BoxView Color="Green"
HeightRequest="40" />
<BoxView Color="Orange"
HeightRequest="40" />
<BoxView Color="Purple"
HeightRequest="40" />
</StackLayout>
</ContentPage>
This example creates a vertical StackLayout containing Label and BoxView objects that have six device-
independent units of vertical space between them:
TIP
The Spacing property can be set to negative values to make child views overlap.
Content = stackLayout;
}
}
In this example, alignment preferences are set on the Label objects to control their position within the
StackLayout . The Start , Center , End , and Fill fields are used to define the alignment of the Label objects
within the parent StackLayout :
A only respects the alignment preferences on child views that are in the opposite direction to the
StackLayout
StackLayout orientation. Therefore, the Label child views within the vertically oriented StackLayout set their
HorizontalOptions properties to one of the alignment fields:
Start , which positions the Label on the left-hand side of the StackLayout .
Center , which centers the Label in the StackLayout .
End , which positions the Label on the right-hand side of the StackLayout .
Fill , which ensures that the Label fills the width of the StackLayout .
Content = stackLayout;
}
}
In this example, the parent StackLayout contains nested StackLayout objects inside Frame objects. The parent
StackLayout is oriented vertically, while the child StackLayout objects are oriented horizontally:
IMPORTANT
The deeper you nest StackLayout objects and other layouts, the more the nested layouts will impact performance.
...
Content = stackLayout;
}
}
VerticalStackLayout
3/18/2022 • 3 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) VerticalStackLayout organizes child views in a one-dimensional
vertical stack, and is a more performant alternative to a StackLayout . In addition, a VerticalStackLayout can be
used as a parent layout that contains other child layouts.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="StackLayoutDemos.Views.VerticalStackLayoutPage">
<VerticalStackLayout Margin="20">
<Label Text="Primary colors" />
<Rectangle Fill="Red"
HeightRequest="30"
WidthRequest="300" />
<Rectangle Fill="Yellow"
HeightRequest="30"
WidthRequest="300" />
<Rectangle Fill="Blue"
HeightRequest="30"
WidthRequest="300" />
<Label Text="Secondary colors" />
<Rectangle Fill="Green"
HeightRequest="30"
WidthRequest="300" />
<Rectangle Fill="Orange"
HeightRequest="30"
WidthRequest="300" />
<Rectangle Fill="Purple"
HeightRequest="30"
WidthRequest="300" />
</VerticalStackLayout>
</ContentPage>
This example creates a VerticalStackLayout containing Label and Rectangle objects. By default, there is no
space between the child views:
NOTE
The value of the Margin property represents the distance between an element and its adjacent elements.
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="StackLayoutDemos.Views.VerticalStackLayoutPage">
<VerticalStackLayout Margin="20"
Spacing="10">
<Label Text="Primary colors" />
<Rectangle Fill="Red"
HeightRequest="30"
WidthRequest="300" />
<Rectangle Fill="Yellow"
HeightRequest="30"
WidthRequest="300" />
<Rectangle Fill="Blue"
HeightRequest="30"
WidthRequest="300" />
<Label Text="Secondary colors" />
<Rectangle Fill="Green"
HeightRequest="30"
WidthRequest="300" />
<Rectangle Fill="Orange"
HeightRequest="30"
WidthRequest="300" />
<Rectangle Fill="Purple"
HeightRequest="30"
WidthRequest="300" />
</VerticalStackLayout>
</ContentPage>
This example creates a VerticalStackLayout containing Label and Rectangle objects that have ten device-
independent units of space between the child views:
TIP
The Spacing property can be set to negative values to make child views overlap.
The HorizontalOptions properties of a VerticalStackLayout , and its child views, can be set to fields from the
LayoutOptions struct, which encapsulates an alignment layout preference. This layout preference determines the
position and size of a child view within its parent layout.
TIP
Don't set the HorizontalOptions property of a VerticalStackLayout unless you need to. The default value of
LayoutOptions.Fill allows for the best layout optimization. Changing this property has a cost and consumes memory,
even when setting it back to its default value.
The following XAML example sets alignment preferences on each child view in the VerticalStackLayout :
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="StackLayoutDemos.Views.VerticalStackLayoutPage">
<VerticalStackLayout Margin="20"
Spacing="6">
<Label Text="Start"
BackgroundColor="Gray"
HorizontalOptions="Start" />
<Label Text="Center"
BackgroundColor="Gray"
HorizontalOptions="Center" />
<Label Text="End"
BackgroundColor="Gray"
HorizontalOptions="End" />
<Label Text="Fill"
BackgroundColor="Gray"
HorizontalOptions="Fill" />
</VerticalStackLayout>
</ContentPage>
In this example, alignment preferences are set on the Label objects to control their position within the
VerticalStackLayout . The Start , Center , End , and Fill fields are used to define the alignment of the Label
objects within the parent VerticalStackLayout :
A VerticalStackLayout only respects the alignment preferences on child views that are in the opposite direction
to the orientation of the layout. Therefore, the Label child views within the VerticalStackLayout set their
HorizontalOptions properties to one of the alignment fields:
Start , which positions the Label on the left-hand side of the VerticalStackLayout .
Center , which centers the Label in the VerticalStackLayout .
End , which positions the Label on the right-hand side of the VerticalStackLayout .
Fill , which ensures that the Label fills the width of the VerticalStackLayout .
In this example, the parent VerticalStackLayout contains nested HorizontalStackLayout objects inside Frame
objects:
IMPORTANT
The deeper you nest layout objects, the more the nested layouts will impact performance.
ContentPage
3/18/2022 • 2 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) ContentPage displays a single view, which is often a layout such as
as Grid or StackLayout , and is the most common page type.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
ContentPage defines a Content property, of type View , which defines the view that represents the page's
content. This property is backed by a BindableProperty object, which means that it can be the target of data
bindings, and styled. In addition, ContentPage inherits Title , IconImageSource , BackgroundImageSource , IsBusy
, and Padding bindable properties from the Page class.
NOTE
The Content property is the content property of the ContentPage class, and therefore does not need to be explicitly
set from XAML.
.NET MAUI apps typically contain multiple pages that derive from ContentPage , and navigation between these
pages can be performed. For more information about page navigation, see NavigationPage.
A ContentPage can be templated with a control template. For more information, see Control templates.
Create a ContentPage
To add a ContentPage to a .NET MAUI app:
1. In Solution Explorer right-click on your project or folder in your project, and select New Item....
2. In the Add New Item dialog, expand Installed > C# Items , select .NET MAUI , and select the .NET
MAUI ContentPage (XAML) item template, enter a suitable page name, and click the Add button:
Visual Studio then creates a new ContentPage -derived page, which will be similar to the following example:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="MyMauiApp.MyPage"
Title="MyPage"
BackgroundColor="White">
<StackLayout>
<Label Text="Welcome to .NET MAUI!"
VerticalOptions="Center"
HorizontalOptions="Center" />
<!-- Other views go here -->
</StackLayout>
</ContentPage>
The child of a ContentPage is typically a layout, such as Grid or StackLayout , with the layout typically
containing multiple views. However, the child of the ContentPage can be a view that displays a collection, such as
CollectionView .
NOTE
The value of the Title property will be shown on the navigation bar, when the app performs navigation using a
NavigationPage . For more information, see NavigationPage.
FlyoutPage
3/18/2022 • 4 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) FlyoutPage is a page that manages two related pages of
information – a flyout page that presents items, and a detail page that presents details about items on the flyout
page. Selecting an item on the flyout page will navigate to the corresponding detail page.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Create a FlyoutPage
To create a flyout page, create a FlyoutPage object and set it's Flyout and Detail properties. The Flyout
property should be set to ContentPage object, and the Detail property should be set to a TabbedPage ,
NavigationPage , or ContentPage object. This will help to ensure a consistent user experience across all
platforms.
IMPORTANT
A FlyoutPage is designed to be the root page of an app, and using it as a child page in other page types could result in
unexpected and inconsistent behavior.
The following example shows a FlyoutPage that sets the Flyout and Detail properties:
<FlyoutPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:FlyoutPageNavigation"
x:Class="FlyoutPageNavigation.MainPage">
<FlyoutPage.Flyout>
<local:FlyoutMenuPage x:Name="flyoutPage" />
</FlyoutPage.Flyout>
<FlyoutPage.Detail>
<NavigationPage>
<x:Arguments>
<local:ContactsPage />
</x:Arguments>
</NavigationPage>
</FlyoutPage.Detail>
</FlyoutPage>
In this example, the Flyout property is set to a ContentPage object, and the Detail property is set to a
NavigationPage containing a ContentPage object.
The following example shows the definition of the FlyoutMenuPage object, which is of type ContentPage :
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:FlyoutPageNavigation"
x:Class="FlyoutPageNavigation.FlyoutMenuPage"
Padding="0,40,0,0"
IconImageSource="hamburger.png"
Title="Personal Organiser">
<ListView x:Name="listView"
x:FieldModifier="public">
<ListView.ItemsSource>
<x:Array Type="{x:Type local:FlyoutPageItem}">
<local:FlyoutPageItem Title="Contacts"
IconSource="contacts.png"
TargetType="{x:Type local:ContactsPage}" />
<local:FlyoutPageItem Title="TodoList"
IconSource="todo.png"
TargetType="{x:Type local:TodoListPage}" />
<local:FlyoutPageItem Title="Reminders"
IconSource="reminders.png"
TargetType="{x:Type local:ReminderPage}" />
</x:Array>
</ListView.ItemsSource>
<ListView.ItemTemplate>
<DataTemplate>
<ViewCell>
<Grid Padding="5,10">
<Grid.ColumnDefinitions>
<ColumnDefinition Width="30"/>
<ColumnDefinition Width="*" />
</Grid.ColumnDefinitions>
<Image Source="{Binding IconSource}" />
<Label Grid.Column="1"
Text="{Binding Title}" />
</Grid>
</ViewCell>
</DataTemplate>
</ListView.ItemTemplate>
</ListView>
</ContentPage>
In this example, the flyout page consists of a ListView that's populated with data by setting its ItemsSource
property to an array of FlyoutPageItem objects. Each FlyoutPageItem defines Title , IconSource , and
TargetType properties. A DataTemplate is assigned to the ListView.ItemTemplate property, to display each
FlyoutPageItem . The DataTemplate contains a ViewCell that consists of an Image and a Label . The Image
displays the IconSource property value, and the Label displays the Title property value, for each
FlyoutPageItem . In addition, the flyout page has its Title and IconImageSource properties set. The icon will
appear on the detail page, provided that the detail page has a title bar.
NOTE
The Flyout page must have its Title property set, or an exception will occur.
In this example, the OnItemSelected event handler retrieves the SelectedItem from the ListView object and
sets the detail page to an instance of the page type stored in the TargetType property of the FlyoutPageItem .
The detail page is displayed by setting the FlyoutPage.IsPresented property to false .
The following example shows how to set the FlyoutLayoutBehavior property on a FlyoutPage :
<FlyoutPage ...
FlyoutLayoutBehavior="Split">
...
</FlyoutPage>
IMPORTANT
The value of the FlyoutLayoutBehavior property only affects apps running on tablets or the desktop. Apps running on
phones always have the Popover behavior.
NavigationPage
3/18/2022 • 10 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) NavigationPage provides a hierarchical navigation experience
where you're able to navigate through pages, forwards and backwards, as desired. NavigationPage provides
navigation as a last-in, first-out (LIFO) stack of Page objects.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
All three events receive NavigationEventArgs objects that define a read-only Page property, which retrieves the
page that was popped from the navigation stack, or the newly visible page on the stack.
When the second page returns back to the first page, a page is popped from the stack, and the new topmost
page then becomes active:
A NavigationPage consists of a navigation bar, with the active page being displayed below the navigation bar.
The following diagram shows the main components of the navigation bar:
An optional icon can be displayed between the back button and the title.
Navigation methods are exposed by the Navigation property on any Page derived types. These methods
provide the ability to push pages onto the navigation stack, to pop pages from the stack, and to manipulate the
stack.
TIP
It's recommended that a NavigationPage should only be populated with ContentPage objects.
NOTE
The RootPage property of a NavigationPage provides access to the first page in the navigation stack.
In this example, the DetailsPage object is pushed onto the navigation stack, where it becomes the active page.
NOTE
The PushAsync method has an override that includes a bool argument that specifies whether to display a page
transition during navigation. The PushAsync method that lacks the bool argument enables the page transition by
default.
await Navigation.PopAsync();
In this example, the current page is removed from the navigation stack, with the new topmost page becoming
the active page.
NOTE
The PopAsync method has an override that includes a bool argument that specifies whether to display a page
transition during navigation. The PopAsync method that lacks the bool argument enables the page transition by
default.
In addition, the Navigation property of each page also exposes a PopToRootAsync method that pops all but the
root page off the navigation stack, therefore making the app's root page the active page.
The RemovePage method removes the specified page from the navigation stack, as shown in the following
diagram:
Together, these methods enable a custom navigation experience, such as replacing a login page with a new page
following a successful login.
To return to the previous page the app should pop the current page from the modal stack, and the new topmost
page becomes the active page:
Modal navigation methods are exposed by the Navigation property on any Page derived types. These methods
provide the ability to push pages onto the modal stack, and pop pages from the modal stack. The Navigation
property also exposes a ModalStack property from which pages in the modal stack can be obtained. However,
there is no concept of performing modal stack manipulation, or popping to the root page in modal navigation.
This is because these operations are not universally supported on the underlying platforms.
NOTE
A NavigationPage object is not required for performing modal page navigation.
In this example, the DetailsPage object is pushed onto the modal stack, where it becomes the active page.
NOTE
The PushModalAsync method has an override that includes a bool argument that specifies whether to display a page
transition during navigation. The PushModalAsync method that lacks the bool argument enables the page transition
by default.
await Navigation.PopModalAsync();
In this example, the current page is removed from the modal stack, with the new topmost page becoming the
active page.
NOTE
The PopModalAsync method has an override that includes a bool argument that specifies whether to display a page
transition during navigation. The PopModalAsync method that lacks the bool argument enables the page transition by
default.
In this example, a Contact object is passed as a constructor argument to DetailPage . The Contact object can
then be displayed by DetailsPage .
Pass data through a BindingContext
An alternative approach for passing data to another page during navigation is by setting the new page's
BindingContext to the data:
The advantage of passing navigation data via a page's BindingContext is that the new page can use data binding
to display the data:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="MyMauiApp.DetailsPage"
Title="Details">
<StackLayout>
<Label Text="{Binding Name}" />
<Label Text="{Binding Occupation}" />
</StackLayout>
</ContentPage>
In this example, a Slider is displayed in the navigation bar of the NavigationPage , to control zooming.
IMPORTANT
Many views won't appear in the navigation bar unless the size of the view is specified with the WidthRequest and
HeightRequest properties.
Because the Layout class derives from the View class, the TitleView attached property can be set to display a
layout class that contains multiple views. However, this can result in clipping if the view displayed in the
navigation bar is larger than the default size of the navigation bar. However, on Android, the height of the
navigation bar can be changed by setting the NavigationPage.BarHeight bindable property to a double
representing the new height.
Alternatively, an extended navigation bar can be suggested by placing some of the content in the navigation bar,
and some in a view at the top of the page content that you color match to the navigation bar. In addition, on iOS
the separator line and shadow that's at the bottom of the navigation bar can be removed by setting the
NavigationPage.HideNavigationBarSeparator bindable property to true .
TIP
The BackButtonTitle , Title , TitleIconImageSource , and TitleView properties can all define values that occupy
space on the navigation bar. While the navigation bar size varies by platform and screen size, setting all of these
properties will result in conflicts due to the limited space available. Instead of attempting to use a combination of these
properties, you may find that you can better achieve your desired navigation bar design by only setting the TitleView
property.
TabbedPage
3/18/2022 • 4 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) TabbedPage maintains a collection of children of type Page , only
one of which is fully visible at a time. Each child is identified by a series of tabs across the top or bottom of the
page. Typically, each child will be a ContentPage and when its tab is selected the page content is displayed.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
WARNING
In a TabbedPage , each Page object is created when the TabbedPage is constructed. This can lead to a poor user
experience, particularly if the TabbedPage is the root page of the app. However, .NET MAUI Shell enables pages accessed
through a tab bar to be created on demand, in response to navigation.
Create a TabbedPage
Two approaches can be used to create a TabbedPage :
Populate the TabbedPage with a collection of child Page objects, such as a collection of ContentPage objects.
For more information, see Populate a TabbedPage with a Page collection.
Assign a collection to the ItemsSource property and assign a DataTemplate to the ItemTemplate property to
return pages for objects in the collection. For more information, see Populate a TabbedPage with a
DataTemplate.
IMPORTANT
A TabbedPage should only be populated with NavigationPage and ContentPage objects.
Regardless of the approach taken, the location of the tab bar in a TabbedPage is platform-dependent:
On iOS, the list of tabs appears at the bottom of the screen, and the page content is above. Each tab consists
of a title and an icon. In portrait orientation, tab bar icons appear above tab titles. In landscape orientation,
icons and titles appear side by side. In addition, a regular or compact tab bar may be displayed, depending on
the device and orientation. If there are more than five tabs, a More tab will appear, which can be used to
access the additional tabs.
On Android, the list of tabs appears at the top of the screen, and the page content is below. Each tab consists
of a title and an icon. However, the tabs can be moved to the bottom of the screen with a platform-specific. If
there are more than five tabs, and the tab list is at the bottom of the screen, a More tab will appear that can
be used to access the additional tabs.
On Windows, the list of tabs appears at the top of the screen, and the page content is below. Each tab consists
of a title. However, icons can be added to each tab with a platform-specific.
Populate a TabbedPage with a Page collection
A TabbedPage can be populated with a collection of child Page objects, which will typically be ContentPage
objects. This is achieved by adding ContentPage objects as children of the TabbedPage :
<TabbedPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:TabbedPageWithNavigationPage"
x:Class="TabbedPageWithNavigationPage.MainPage">
<local:TodayPage />
<local:SchedulePage />
<local:SettingsPage />
</TabbedPage>
Page objects that are added as child elements of TabbedPage are added to the Children collection. The
Children property of the MultiPage<T> class, from which TabbedPage derives, is the ContentProperty of
MultiPage<T> . Therefore, in XAML it's not necessary to explicitly assign the Page objects to the Children
property.
The following screenshot shows the appearance of the resulting tab bar on the TabbedPage :
The page content for a tab appears when the tab is selected.
Populate a TabbedPage with a DataTemplate
TabbedPage inherits ItemsSource , ItemTemplate , and SelectedItem bindable properties from the MultiPage<T>
class. These properties enable you to generate TabbedPage children dynamically, by setting the ItemsSource
property to an IEnumerable collection of objects with public properties suitable for data bindings, and by setting
the ItemTemplate property to a DataTemplate with a page type as the root element.
The following example shows generating TabbedPage children dynamically:
<TabbedPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:TabbedPageDemo"
x:Class="TabbedPageDemo.MainPage"
ItemsSource="{x:Static local:MonkeyDataModel.All}">
<TabbedPage.ItemTemplate>
<DataTemplate>
<ContentPage Title="{Binding Name}"
IconImageSource="monkeyicon.png">
<StackLayout Padding="5, 25">
<Label Text="{Binding Name}"
FontAttributes="Bold"
FontSize="Large"
HorizontalOptions="Center" />
<Image Source="{Binding PhotoUrl}"
HorizontalOptions="Center"
WidthRequest="200"
HeightRequest="200" />
<StackLayout Padding="50, 10">
<StackLayout Orientation="Horizontal">
<Label Text="Family: "
FontAttributes="Bold" />
<Label Text="{Binding Family}" />
</StackLayout>
...
</StackLayout>
</StackLayout>
</ContentPage>
</DataTemplate>
</TabbedPage.ItemTemplate>
</TabbedPage>
In this example, each tab consists of a ContentPage object that uses Image and Label objects to display data
for the tab:
In this example, the TabbedPage is populated with two Page objects. The first child is a ContentPage object, and
the second child is a NavigationPage object containing a ContentPage object.
When a ContentPage is wrapped in a NavigationPage , forwards page navigation can be performed by calling
the PushAsync method on the Navigation property of the ContentPage object:
For more information about performing navigation using the NavigationPage class, see NavigationPage.
WARNING
While a NavigationPage can be placed in a TabbedPage , it's not recommended to place a TabbedPage into a
NavigationPage .
Host a Blazor web app in a .NET MAUI app using
BlazorWebView
3/18/2022 • 3 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) BlazorWebView is a control that enables you to host a Blazor web
app in your .NET MAUI app. These apps, known as .NET MAUI Blazor apps, enable a Blazor web app to be
integrated with platform features and UI controls. The BlazorWebView control can be added to any page of a
.NET MAUI app, and pointed to the root of the Blazor app. The Razor components run natively in the .NET
process and render web UI to an embedded web view control. .NET MAUI Blazor apps can run on all the
platforms supported by .NET MAUI.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
NOTE
While Visual Studio installs all the required tooling to develop .NET MAUI Blazor apps, end users of .NET MAUI Blazor apps
on Windows must install the WebView2 runtime.
<ContentPage xmlns=http://schemas.microsoft.com/dotnet/2021/maui
xmlns:x=http://schemas.microsoft.com/winfx/2009/xaml
xmlns:b="clr-
namespace:Microsoft.AspNetCore.Components.WebView.Maui;assembly=Microsoft.AspNetCore.Components.WebView.Maui
"
xmlns:local="clr-namespace:BlazorWebViewDemo"
x:Class="BlazorWebViewDemo.MainPage"
BackgroundColor="{DynamicResource PageBackgroundColor}">
<b:BlazorWebView HostPage="wwwroot/index.html">
<b:BlazorWebView.RootComponents>
<b:RootComponent Selector="app" ComponentType="{x:Type local:Main}" />
</b:BlazorWebView.RootComponents>
</b:BlazorWebView>
</ContentPage>
The root Razor component for the app is in Main.razor, which Razor compiles into a type named Main in the
application's root namespace. The rest of the Razor components are in the Pages and Shared project folders, and
are identical to the components used in the default Blazor web template. Static web assets for the app are in the
wwwroot folder.
The Razor SDK is required to build and package projects containing Razor files for Blazor projects.
2. Add the root Razor component for the app to the project.
3. Add your Razor components to project folders named Pages and Shared.
4. Add your static web assets to a project folder named wwwroot.
5. Add any optional _Imports.razor files to your project.
6. Add a BlazorWebView to a page in your .NET MAUI app, and point it to the root of the Blazor app:
<ContentPage xmlns=http://schemas.microsoft.com/dotnet/2021/maui
xmlns:x=http://schemas.microsoft.com/winfx/2009/xaml
xmlns:b="clr-
namespace:Microsoft.AspNetCore.Components.WebView.Maui;assembly=Microsoft.AspNetCore.Components.WebVi
ew.Maui"
xmlns:local="clr-namespace:MyBlazorApp"
x:Class="MyBlazorApp.MainPage">
<b:BlazorWebView HostPage="wwwroot/index.html">
<b:BlazorWebView.RootComponents>
<b:RootComponent Selector="app" ComponentType="{x:Type local:Main}" />
</b:BlazorWebView.RootComponents>
</b:BlazorWebView>
</ContentPage>
using Microsoft.AspNetCore.Components.WebView.Maui;
8. Modify the CreateMauiApp method of your MauiProgram class to register the BlazorWebView control for
use in your app. To do this, call the RegisterBlazorMauiWebView method on the MauiAppBuilder object.
Then, on the IServiceCollection object, call the AddBlazorWebView method to add component web view
services to the services collection:
public static class MauiProgram
{
public static MauiApp CreateMauiApp()
{
var builder = MauiApp.CreateBuilder();
builder
.RegisterBlazorMauiWebView()
.UseMauiApp<App>()
.ConfigureFonts(fonts =>
{
fonts.AddFont("OpenSans-Regular.ttf", "OpenSansRegular");
});
builder.Services.AddBlazorWebView();
// Register any app services on the IServiceCollection object
// e.g. builder.Services.AddSingleton<WeatherForecastService>();
return builder.Build();
}
}
Border
3/18/2022 • 2 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) Border is a container control that draws a border, background, or
both, around another control. A Border can only contain one child object. If you want to put a border around
multiple objects, wrap them in a container object such as a layout.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Create a Border
To draw a border, create a Border object and set its properties to define its appearance. Then, set its child to the
control to which the border should be added.
The following XAML example shows how to draw a border around a Label :
<Border Stroke="#C49B33"
StrokeThickness="4"
Background="#2B0B98"
Padding="16,8"
HorizontalOptions="Center">
<Border.StrokeShape>
<RoundRectangle CornerRadius="40,0,0,40" />
</Border.StrokeShape>
<Label Text=".NET MAUI"
TextColor="White"
FontSize="18"
FontAttributes="Bold" />
</Border>
In this example, a border with rounded top-left and bottom-right corners is drawn around a Label . The border
shape is defined as a RoundRectangle object, whose CornerRadius property is set to a Thickness value that
enables independent control of each corner of the rectangle:
Because the Stroke property is of type Brush , borders can also be drawn using gradients:
<Border StrokeThickness="4"
Background="#2B0B98"
Padding="16,8"
HorizontalOptions="Center">
<Border.StrokeShape>
<RoundRectangle CornerRadius="40,0,0,40" />
</Border.StrokeShape>
<Border.Stroke>
<LinearGradientBrush EndPoint="0,1">
<GradientStop Color="Orange"
Offset="0.1" />
<GradientStop Color="Brown"
Offset="1.0" />
</LinearGradientBrush>
</Border.Stroke>
<Label Text=".NET MAUI"
TextColor="White"
FontSize="18"
FontAttributes="Bold" />
</Border>
In this example, a border that uses a linear gradient is drawn around a Label :
BoxView
3/18/2022 • 2 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) BoxView draws a simple rectangle or square, of a specified width,
height, and color.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
NOTE
Although BoxView can mimic simple graphics, a better alternative is to use .NET MAUI Shapes or .NET MAUI Graphics.
Create a BoxView
To draw a rectangle or square, create a BoxView object and set its Color , WidthRequest , and HeightRequest
properties. Optionally, you can also set its CornerRadius property.
The following XAML example shows how to create a BoxView :
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:BasicBoxView"
x:Class="BasicBoxView.MainPage">
<BoxView Color="CornflowerBlue"
CornerRadius="10"
WidthRequest="160"
HeightRequest="160"
VerticalOptions="Center"
HorizontalOptions="Center" />
</ContentPage>
In this example, a cornflower blue BoxView is displayed in the center of the page:
The WidthRequest and HeightRequest properties are measured in device-independent units.
NOTE
A BoxView can also be a child of an AbsoluteLayout . In this case, both the location and size of the BoxView are set
using the LayoutBounds attached bindable property.
A BoxView can also be sized to resemble a line of a specific width and thickness.
Frame
3/18/2022 • 2 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) Frame class is used to wrap a view or layout with a border that can
be configured with color, shadow, and other options. Frames can be used to create borders around controls but
can also be used to create more complex UI.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
The Frame class inherits from ContentView , which provides a Content bindable property. The Content
property is the ContentProperty of the Frame class, and therefore does not need to be explicitly set from XAML.
Create a Frame
A Frame object typically wraps another control, such as a Label :
<Frame>
<Label Text="Frame wrapped around a Label" />
</Frame>
<Frame BorderColor="Gray"
CornerRadius="10">
<Label Text="Frame wrapped around a Label" />
</Frame>
<Frame BorderColor="Gray"
CornerRadius="5"
Padding="8">
<StackLayout>
<Label Text="Card Example"
FontSize="Medium"
FontAttributes="Bold" />
<BoxView Color="Gray"
HeightRequest="2"
HorizontalOptions="Fill" />
<Label Text="Frames can wrap more complex layouts to create more complex UI components, such as this
card!"/>
</StackLayout>
</Frame>
Round elements
The CornerRadius property of the Frame control is one approach to creating a circle image. The following XAML
shows how to create a circle image with a Frame :
<Frame Margin="10"
BorderColor="Black"
CornerRadius="50"
HeightRequest="60"
WidthRequest="60"
IsClippedToBounds="True"
HorizontalOptions="Center"
VerticalOptions="Center">
<Image Source="outdoors.jpg"
Aspect="AspectFill"
Margin="-20"
HeightRequest="100"
WidthRequest="100" />
</Frame>
The .NET Multi-platform App UI (.NET MAUI) GraphicsView is a graphics canvas on which 2D graphics can be
drawn using types from the Microsoft.Maui.Graphics namespace. For more information about
Microsoft.Maui.Graphics , see Graphics.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
GraphicsView defines the Drawable property, of type IDrawable , which specifies the content that will be drawn.
This property is backed by a BindableProperty , which means it can be the target of data binding, and styled.
Create a GraphicsView
A GraphicsView must define an IDrawable object that specifies the content that will be drawn on the control.
This can be achieved by creating an object that derives from IDrawable , and by implementing its Draw method:
namespace MyMauiApp
{
public class GraphicsDrawable : IDrawable
{
public void Draw(ICanvas canvas, RectF dirtyRect)
{
// Drawing code goes here
}
}
}
The Draw method has ICanvas and RectF arguments. The ICanvas argument is the drawing canvas on which
you draw graphical objects. The RectF argument is a struct that contains data about the size and location of
the drawing canvas. For more information about drawing on an ICanvas , see Draw graphical objects.
In XAML, the IDrawable object should be declared as a resource, and then consumed by a GraphicsView by
specifying its key:
<ContentPage xmlns=http://schemas.microsoft.com/dotnet/2021/maui
xmlns:x=http://schemas.microsoft.com/winfx/2009/xaml
xmlns:drawable="clr-namespace:MyMauiApp"
x:Class="MyMauiApp.MainPage">
<ContentPage.Resources>
<drawable:GraphicsDrawable x:Key="drawable" />
</ContentPage.Resources>
<VerticalStackLayout>
<GraphicsView Drawable="{StaticResource drawable}"
HeightRequest="300"
WidthRequest="400" />
</VerticalStackLayout>
</ContentPage>
Position and size graphical objects
The location and size of the ICanvas on a page can be determined by examining properties of the RectF
argument in the Draw method.
The RectF struct defines the following properties:
Bottom , of type float , which represents the y-coordinate of the bottom edge of the canvas.
Center , of type PointF . which specifies the coordinates of the center of the canvas.
Height , of type float , which defines the height of the canvas.
IsEmpty , of type bool , which indicates whether the canvas has a zero size and location.
Left , of type float , which represents the x-coordinate of the left edge of the canvas.
Location , of type PointF , which defines the coordinates of the upper-left corner of the canvas.
Right , of type float , which represents the x-coordinate of the right edge of the canvas.
Size , of type SizeF , which defines the width and height of the canvas.
Top , of type float , which represents the y-coordinate of the top edge of the canvas.
Width , of type float , which defines the width of the canvas.
X , of type float , which defines the x-coordinate of the upper-left corner of the canvas.
Y , of type float , which defines the y-coordinate of the upper-left corner of the canvas.
These properties can be used to position and size graphical objects on the ICanvas . For example, graphical
objects can be placed at the center of the Canvas by using the Center.X and Center.Y values as arguments to
a drawing method. For information about drawing on an ICanvas , see Draw graphical objects.
graphicsView.Invalidate();
<GraphicsView x:Name="graphicsView"
Drawable="{StaticResource drawable}"
HeightRequest="300"
WidthRequest="400" />
In code, the Drawable property of the GraphicsView object can then be accessed, and the ToImage method
called:
The .NET Multi-platform App UI (.NET MAUI) Image displays an image that can be loaded from a local file, a URI,
an embedded resource, or a stream. The standard platform image formats are supported, including animated
GIFs, and local Scalable Vector Graphics (SVG) files are also supported.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
These properties are backed by BindableProperty objects, which means that they can be styled, and be the
target of data bindings.
NOTE
Font icons can be displayed by an Image by specifying the font icon data as a FontImageSource object.
The ImageSource class defines the following methods that can be used to load an image from different sources:
FromFile returns a FileImageSource that reads an image from a local file.
FromUri returns an UriImageSource that downloads and reads an image from a specified URI.
FromResource returns a StreamImageSource that reads an image file embedded in an assembly.
FromStream returns a StreamImageSource that reads an image from a stream that supplies image data.
In XAML, images can be loaded from files and URIs by specifying the filename or URI as a string value for the
Source property. Images can also be loaded from resources or streams in XAML through custom markup
extensions.
IMPORTANT
Images will be displayed at their full resolution unless the size of the Image is constrained by its layout, or the
HeightRequest or WidthRequest property of the Image is specified.
For information about adding app icons and a splash screen to your app, see App icons and Splash screen.
Load a local image
Images can be added to your app project by dragging them to the Resources\Images folder of your project, and
ensuring their build action is set to MauiImage in the Proper ties window. At build time, images are resized to
the correct resolutions for the target platform and device and added to your app package. This is necessary
because different platforms support different image resolutions, and the operating system chooses the
appropriate image resolution at runtime based on the device's capabilities.
To comply with Android resource naming rules, all local image filenames must be lowercase, start and end with
a letter character, and contain only alphanumeric characters or underscores. For more information, see App
resources overview on developer.android.com.
Adhering to these rules for file naming and placement enables the following XAML to load and display an
image:
The ImageSource.FromFile method requires a string argument, and returns a new FileImageSource object that
reads the image from the file. There's also an implicit conversion operator that enables the filename to be
specified as a string argument to the Image.Source property:
The ImageSource.FromUri method requires a Uri argument, and returns a new UriImageSource object that
reads the image from the Uri . There's also an implicit conversion for string-based URIs:
Image caching
Caching of downloaded images is enabled by default, with cached images being stored for 1 day. This behavior
can be changed by setting properties of the UriImageSource class.
The UriImageSource class defines the following properties:
Uri , of type , represents the URI of the image to be downloaded for display.
Uri
CacheValidity , of type TimeSpan , specifies how long the image will be stored locally for. The default value of
this property is 1 day.
CachingEnabled , of type bool , defines whether image caching is enabled. The default value of this property
is true .
These properties are backed by BindableProperty objects, which means that they can be styled, and be the
target of data bindings.
To set a specific cache period, set the Source property to an UriImageSource object that sets its CacheValidity
property:
<Image>
<Image.Source>
<UriImageSource Uri="https://aka.ms/campus.jpg"
CacheValidity="10:00:00.0" />
</Image.Source>
</Image>
By default, the ImageSource.FromResource method only looks for images in the same assembly as the calling
code. However, the assembly containing the embedded image can be specified as the second argument to the
ImageSource.FromResource method:
Image image = new Image
{
Source = ImageSource.FromResource("MyLibrary.MyFolder.myimage.png",
typeof(MyLibrary.MyClass).GetTypeInfo().Assembly)
};
using System.Reflection;
using System.Xml;
namespace ImageDemos
{
[ContentProperty("Source")]
public class ImageResourceExtension : IMarkupExtension<ImageSource>
{
public string Source { set; get; }
The ImageResourceExtension class is used to access an embedded image in XAML, and uses the Source property
to call the ImageSource.FromResource method. The argument to the ProvideValue method is an
IServiceProvider object that can be used to obtain an IXmlLineInfoProvider object that can provide line and
character information indicating where an error has been detected. This object is used to raise an exception
when the Image.Source property hasn't been set.
The markup extension can be consumed in XAML to load an embedded image:
<ContentPage ...
xmlns:local="clr-namespace:ImageDemos">
<StackLayout>
<Image Source="{local:ImageResource monkey.png}"
HeightRequest="100" />
</StackLayout>
</ContentPage>
For more information about XAML markup extensions, see Create XAML markup extensions.
Load an image from a stream
Images can be loaded from streams with the ImageSource.FromStream method:
IMPORTANT
While the animated GIF support in .NET MAUI includes the ability to download files, it does not support caching or
streaming animated GIFs.
By default, when an animated GIF is loaded it will not be played. This is because the IsAnimationPlaying
property, that controls whether an animated GIF is playing or stopped, has a default value of false . Therefore,
when an animated GIF is loaded it will not be played until the IsAnimationPlaying property is set to true .
Playback can be stopped by reseting the IsAnimationPlaying property to false . Note that this property has no
effect when displaying a non-GIF image source.
The .NET Multi-platform App UI (.NET MAUI) Label displays single-line and multi-line text. Text displayed by a
Label can be colored, spaced, and can have text decorations.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
Create a Label
The following example shows how to create a Label :
Set colors
Labels can be set to use a specific text color via the TextColor property.
The following example sets the text color of a Label :
<Label TextColor="#77d065"
Text="This is a green label." />
The result is that characters in the text displayed by the Label are spaced CharacterSpacing device-
independent units apart.
In C#, text can be forced onto a new line with the "\n" character:
NoWrap — does not wrap text, displaying only as much text as can fit on one line. This is the default value of
the property.
LineBreakMode
WordWrap — wraps text at the word boundary.
CharacterWrap — wraps text onto a new line at a character boundary.
HeadTruncation — truncates the head of the text, showing the end.
MiddleTruncation — displays the beginning and end of the text, with the middle replace by an ellipsis.
TailTruncation — shows the beginning of the text, truncating the end.
<Label Text="Lorem ipsum dolor sit amet, consectetur adipiscing elit. In facilisis nulla eu felis fringilla
vulputate. Nullam porta eleifend lacinia. Donec at iaculis tellus."
LineBreakMode="WordWrap"
MaxLines="2" />
NOTE
On iOS, the Label.LineHeight property changes the line height of text that fits on a single line, and text that wraps
onto multiple lines.
On Android, the Label.LineHeight property only changes the line height of text that wraps onto multiple lines.
On Windows, the Label.LineHeight property changes the line height of text that wraps onto multiple lines.
<Label Text="Lorem ipsum dolor sit amet, consectetur adipiscing elit. In facilisis nulla eu felis fringilla
vulputate. Nullam porta eleifend lacinia. Donec at iaculis tellus."
LineBreakMode="WordWrap"
LineHeight="1.8" />
The following screenshot shows the result of setting the Label.LineHeight property to 1.8:
Display HTML
The Label class has a TextType property, which determines whether the Label object should display plain
text, or HTML text. This property should be set to one of the members of the TextType enumeration:
Text indicates that the Label will display plain text, and is the default value of the TextType property.
Html indicates that the Label will display HTML text.
Therefore, Label objects can display HTML by setting the TextType property to Html , and the Text property
to a HTML string:
In the example above, the double quote characters in the HTML have to be escaped using the \ symbol.
In XAML, HTML strings can become unreadable due to additionally escaping the < and > symbols:
Alternatively, for greater readability the HTML can be inlined in a CDATA section:
<Label TextType="Html">
<![CDATA[
This is <strong style="color:red">HTML</strong> text.
]]>
</Label>
In this example, the Text property is set to the HTML string that's inlined in the CDATA section. This works
because the Text property is the ContentProperty for the Label class.
IMPORTANT
Displaying HTML in a Label is limited to the HTML tags that are supported by the underlying platform.
Decorate text
Underline and strikethrough text decorations can be applied to Label objects by setting the TextDecorations
property to one or more TextDecorations enumeration members:
None
Underline
Strikethrough
The following example demonstrates setting the TextDecorations property:
The following screenshot shows the TextDecorations enumeration members applied to Label instances:
NOTE
Text decorations can also be applied to Span instances. For more information about the Span class, see Use formatted
text.
Transform text
A Label can transform the casing of its text, stored in the Text property, by setting the TextTransform
property to a value of the TextTransform enumeration. This enumeration has four values:
None indicates that the text won't be transformed.
Default indicates that the default behavior for the platform will be used. This is the default value of the
TextTransform property.
Lowercase indicates that the text will be transformed to lowercase.
Uppercase indicates that the text will be transformed to uppercase.
NOTE
It's not possible to display HTML in a Span .
Span defines the following properties:
BackgroundColor , of type Color , which represents the color of the span background.
CharacterSpacing , of type double , sets the spacing between characters in the displayed text.
FontAttributes , of type FontAttributes , determines text style.
FontAutoScalingEnabled , of type bool , defines whether the text will reflect scaling preferences set in the
operating system. The default value of this property is true .
FontFamily , of type string , defines the font family.
FontSize , of type double , defines the font size.
LineHeight , of type double , specifies the multiplier to apply to the default line height when displaying text.
Style , of type Style , which is the style to apply to the span.
Text , of type string , defines the text displayed as the content of the Span .
TextColor , of type Color , defines the color of the displayed text.
TextDecorations , of type TextDecorations , specifies the text decorations (underline and strikethrough) that
can be applied.
TextTransform , of type TextTransform , specifies the casing of the displayed text.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
NOTE
The Span.LineHeight property has no effect on Windows.
In addition, the GestureRecognizers property can be used to define a collection of gesture recognizers that will
respond to gestures on the Span .
The following XAML example demonstrates a FormattedText property that consists of three Span instances:
<Label LineBreakMode="WordWrap">
<Label.FormattedText>
<FormattedString>
<Span Text="Red Bold, " TextColor="Red" FontAttributes="Bold" />
<Span Text="default, " Style="{DynamicResource BodyStyle}">
<Span.GestureRecognizers>
<TapGestureRecognizer Command="{Binding TapCommand}" />
</Span.GestureRecognizers>
</Span>
<Span Text="italic small." FontAttributes="Italic" FontSize="Small" />
</FormattedString>
</Label.FormattedText>
</Label>
The following screenshot shows the resulting Label that contains three Span objects:
A Span can also respond to any gestures that are added to the span's GestureRecognizers collection. For
example, a TapGestureRecognizer has been added to the second Span in the above examples. Therefore, when
this Span is tapped the TapGestureRecognizer will respond by executing the ICommand defined by the Command
property. For more information about tap gesture recognition, see Recognize a tap gesture.
Create a hyperlink
The text displayed by Label and Span instances can be turned into hyperlinks with the following approach:
1. Set the TextColor and TextDecoration properties of the Label or Span .
2. Add a TapGestureRecognizer to the GestureRecognizers collection of the Label or Span , whose Command
property binds to a ICommand , and whose CommandParameter property contains the URL to open.
3. Define the ICommand that will be executed by the TapGestureRecognizer .
4. Write the code that will be executed by the ICommand .
The following example, shows a Label whose content is set from multiple Span objects:
<Label>
<Label.FormattedText>
<FormattedString>
<Span Text="Alternatively, click " />
<Span Text="here"
TextColor="Blue"
TextDecorations="Underline">
<Span.GestureRecognizers>
<TapGestureRecognizer Command="{Binding TapCommand}"
CommandParameter="https://docs.microsoft.com/dotnet/maui/" />
</Span.GestureRecognizers>
</Span>
<Span Text=" to view .NET MAUI documentation." />
</FormattedString>
</Label.FormattedText>
</Label>
In this example, the first and third Span instances contain text, while the second Span represents a tappable
hyperlink. It has its text color set to blue, and has an underline text decoration. This creates the appearance of a
hyperlink, as shown in the following screenshot:
When the hyperlink is tapped, the TapGestureRecognizer will respond by executing the ICommand defined by its
Command property. In addition, the URL specified by the CommandParameter property will be passed to the
ICommand as a parameter.
The code-behind for the XAML page contains the TapCommand implementation:
using System.Windows.Input;
public MainPage()
{
InitializeComponent();
BindingContext = this;
}
}
public HyperlinkSpan()
{
TextDecorations = TextDecorations.Underline;
TextColor = Color.Blue;
GestureRecognizers.Add(new TapGestureRecognizer
{
// Launcher.OpenAsync is provided by Essentials.
Command = new Command(async () => await Launcher.OpenAsync(Url))
});
}
}
The HyperlinkSpan class defines a Url property, and associated BindableProperty , and the constructor sets the
hyperlink appearance and the TapGestureRecognizer that will respond when the hyperlink is tapped. When a
HyperlinkSpan is tapped, the TapGestureRecognizer will respond by executing the Launcher.OpenAsync method
to open the URL, specified by the Url property, in a web browser.
The HyperlinkSpan class can be consumed by adding an instance of the class to the XAML:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:HyperlinkDemo"
x:Class="HyperlinkDemo.MainPage">
<StackLayout>
...
<Label>
<Label.FormattedText>
<FormattedString>
<Span Text="Alternatively, click " />
<local:HyperlinkSpan Text="here"
Url="https://docs.microsoft.com/dotnet/" />
<Span Text=" to view .NET documentation." />
</FormattedString>
</Label.FormattedText>
</Label>
</StackLayout>
</ContentPage>
ScrollView
3/18/2022 • 9 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) ScrollView is a view that's capable of scrolling its content. By
default, ScrollView scrolls its content vertically. A ScrollView can only have a single child, although this can be
other layouts.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
These properties are backed by BindableProperty objects, with the exception of the Content property, which
means that they can be targets of data bindings and styled.
The Content property is the ContentProperty of the ScrollView class, and therefore does not need to be
explicitly set from XAML.
WARNING
ScrollView objects should not be nested. In addition, ScrollView objects should not be nested with other controls
that provide scrolling, such as CollectionView , ListView , and WebView .
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:ScrollViewDemos"
x:Class="ScrollViewDemos.Views.XAML.ColorListPage"
Title="ScrollView demo">
<ScrollView Margin="20">
<StackLayout BindableLayout.ItemsSource="{x:Static local:NamedColor.All}">
<BindableLayout.ItemTemplate>
<DataTemplate>
<StackLayout Orientation="Horizontal">
<BoxView Color="{Binding Color}"
HeightRequest="32"
WidthRequest="32"
VerticalOptions="Center" />
<Label Text="{Binding FriendlyName}"
FontSize="24"
VerticalOptions="Center" />
</StackLayout>
</DataTemplate>
</BindableLayout.ItemTemplate>
</StackLayout>
</ScrollView>
</ContentPage>
In this example, the ScrollView has its content set to a StackLayout that uses a bindable layout to display the
Color fields defined by .NET MAUI. By default, a ScrollView scrolls vertically, which reveals more content:
The equivalent C# code is:
return horizontalStackLayout;
});
In this example, the root layout is a Grid that has a Label , ScrollView , and Button as its children. The
ScrollView has a StackLayout as its content, with the StackLayout containing multiple Label objects. This
arrangement ensures that the first Label is always on-screen, while text displayed by the other Label objects
can be scrolled:
The equivalent C# code is:
public class BlackCatPage : ContentPage
{
public BlackCatPage()
{
Label titleLabel = new Label
{
Text = "THE BLACK CAT by Edgar Allan Poe",
// More properties set here to define the Label appearance
};
Content = grid;
}
}
Orientation
ScrollView has an Orientation property, which represents the scrolling direction of the ScrollView . This
property is of type ScrollOrientation , which defines the following members:
Vertical indicates that the ScrollView will scroll vertically. This member is the default value of the
Orientation property.
Horizontal indicates that the ScrollView will scroll horizontally.
Both indicates that the ScrollView will scroll horizontally and vertically.
Neither indicates that the ScrollView won't scroll.
TIP
Scrolling can be disabled by setting the Orientation property to Neither .
Detect scrolling
ScrollView defines a Scrolled event that's raised to indicate that scrolling occurred. The ScrolledEventArgs
object that accompanies the Scrolled event has ScrollX and ScrollY properties, both of type double .
IMPORTANT
The ScrolledEventArgs.ScrollX and ScrolledEventArgs.ScrollY properties can have negative values, due to the
bounce effect that occurs when scrolling back to the start of a ScrollView .
The following XAML example shows a ScrollView that sets an event handler for the Scrolled event:
<ScrollView Scrolled="OnScrollViewScrolled">
...
</ScrollView>
In this example, the OnScrollViewScrolled event handler is executed when the Scrolled event fires:
In this example, the OnScrollViewScrolled event handler outputs the values of the ScrolledEventArgs object that
accompanies the event.
NOTE
The Scrolled event is raised for user initiated scrolls, and for programmatic scrolls.
Scroll programmatically
ScrollView defines two ScrollToAsync methods, that asynchronously scroll the ScrollView . One of the
overloads scrolls to a specified position in the ScrollView , while the other scrolls a specified element into view.
Both overloads have an additional argument that can be used to indicate whether to animate the scroll.
IMPORTANT
The ScrollToAsync methods will not result in scrolling when the ScrollView.Orientation property is set to
Neither .
The third argument to the ScrollToAsync is the animated argument, which determines whether a scrolling
animation is displayed when programmatically scrolling a ScrollView .
Scroll an element into view
An element within a ScrollView can be scrolled into view with the ScrollToAsync method that accepts Element
and ScrollToPosition arguments. Given a vertical ScrollView named scrollView , and a Label named label ,
the following example shows how to scroll an element into view:
The third argument to the ScrollToAsync is the animated argument, which determines whether a scrolling
animation is displayed when programmatically scrolling a ScrollView .
When scrolling an element into view, the exact position of the element after the scroll has completed can be set
with the second argument, position , of the ScrollToAsync method. This argument accepts a ScrollToPosition
enumeration member:
MakeVisible indicates that the element should be scrolled until it's visible in the ScrollView .
Start indicates that the element should be scrolled to the start of the ScrollView .
Center indicates that the element should be scrolled to the center of the ScrollView .
End indicates that the element should be scrolled to the end of the ScrollView .
A .NET Multi-platform App UI (.NET MAUI) Shape is a type of View that enables you to draw a shape to the
screen. Shape objects can be used inside layout classes and most controls, because the Shape class derives
from the View class. .NET MAUI Shapes is available in the Microsoft.Maui.Controls.Shapes namespace.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Paint shapes
Brush objects are used to paint a shapes's Stroke and Fill :
<Ellipse Fill="DarkBlue"
Stroke="Red"
StrokeThickness="4"
WidthRequest="150"
HeightRequest="50"
HorizontalOptions="Start" />
In this example, the stroke and fill of an Ellipse are specified:
IMPORTANT
Brush objects use a type converter that enables Color values to specified for the Stroke property.
If you don't specify a Brush object for Stroke , or if you set StrokeThickness to 0, then the border around the
shape is not drawn.
For more information about Brush objects, see Brushes. For more information about valid Color values, see
Colors.
Stretch shapes
Shape objects have an Aspect property, of type Stretch . This property determines how a Shape object's
contents is stretched to fill the Shape object's layout space. A Shape object's layout space is the amount of
space the Shape is allocated by the .NET MAUI layout system, because of either an explicit WidthRequest and
HeightRequest setting or because of its HorizontalOptions and VerticalOptions settings.
<Path Aspect="Uniform"
Stroke="Yellow"
Fill="Red"
BackgroundColor="LightGray"
HorizontalOptions="Start"
HeightRequest="100"
WidthRequest="100">
<Path.Data>
<!-- Path data goes here -->
</Path.Data>
</Path>
In this example, a Path object draws a heart. The Path object's WidthRequest and HeightRequest properties
are set to 100 device-independent units, and its Aspect property is set to Uniform . As a result, the object's
contents are resized to fit the destination dimensions, while preserving the aspect ratio:
Draw dashed shapes
Shape objects have a StrokeDashArray property, of type DoubleCollection . This property represents a
collection of double values that indicate the pattern of dashes and gaps that are used to outline a shape. A
DoubleCollection is an ObservableCollection of double values. Each double in the collection specifies the
length of a dash or gap. The first item in the collection, which is located at index 0, specifies the length of a dash.
The second item in the collection, which is located at index 1, specifies the length of a gap. Therefore, objects
with an even index value specify dashes, while objects with an odd index value specify gaps.
Shape objects also have a StrokeDashOffset property , of type double , which specifies the distance within the
dash pattern where a dash begins. Failure to set this property will result in the Shape having a solid outline.
Dashed shapes can be drawn by setting both the StrokeDashArray and StrokeDashOffset properties. The
StrokeDashArray property should be set to one or more double values, with each pair delimited by a single
comma and/or one or more spaces. For example, "0.5 1.0" and "0.5,1.0" are both valid.
The following XAML example shows how to draw a dashed rectangle:
<Rectangle Fill="DarkBlue"
Stroke="Red"
StrokeThickness="4"
StrokeDashArray="1,1"
StrokeDashOffset="6"
WidthRequest="150"
HeightRequest="50"
HorizontalOptions="Start" />
<Line X1="0"
Y1="20"
X2="300"
Y2="20"
StrokeLineCap="Round"
Stroke="Red"
StrokeThickness="12" />
In this example, the red line is rounded at the start and end of the line:
NOTE
When the StrokeLineJoin property is set to Miter , the StrokeMiterLimit property can be set to a double to
limit the miter length of line joins in the shape.
In this example, the dark blue polyline has rounded joins at its vertices:
Ellipse
3/18/2022 • 2 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) Ellipse class derives from the Shape class, and can be used to
draw ellipses and circles. For information on the properties that the Ellipse class inherits from the Shape
class, see Shapes.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The Ellipse class sets the Aspect property, inherited from the Shape class, to Stretch.Fill . For more
information about the Aspect property, see Stretch shapes.
Create an Ellipse
To draw an ellipse, create an Ellipse object and set its WidthRequest and HeightRequest properties. To paint
the inside of the ellipse, set its Fill property to a Brush -derived object. To give the ellipse an outline, set its
Stroke property to a Brush -derived object. The StrokeThickness property specifies the thickness of the ellipse
outline. For more information about Brush objects, see Brushes.
To draw a circle, make the WidthRequest and HeightRequest properties of the Ellipse object equal.
The following XAML example shows how to draw a filled ellipse:
<Ellipse Fill="Red"
WidthRequest="150"
HeightRequest="50"
HorizontalOptions="Start" />
In this example, a red filled ellipse with dimensions 150x50 (device-independent units) is drawn:
<Ellipse Stroke="Red"
StrokeThickness="4"
WidthRequest="150"
HeightRequest="150"
HorizontalOptions="Start" />
In this example, a red circle with dimensions 150x150 (device-independent units) is drawn:
For information about drawing a dashed ellipse, see Draw dashed shapes.
Fill rules
3/18/2022 • 4 minutes to read • Edit Online
Several .NET Multi-platform App UI (.NET MAUI) Shapes classes have FillRule properties, of type FillRule .
These include Polygon , Polyline , and GeometryGroup .
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The FillRule enumeration defines EvenOdd and Nonzero members. Each member represents a different rule
for determining whether a point is in the fill region of a shape.
IMPORTANT
All shapes are considered closed for the purposes of fill rules.
EvenOdd
The EvenOdd fill rule draws a ray from the point to infinity in any direction and counts the number of segments
within the shape that the ray crosses. If this number is odd, the point is inside. If this number is even, the point is
outside.
The following XAML example creates and renders a composite shape, with the FillRule defaulting to EvenOdd :
<Path Stroke="Black"
Fill="#CCCCFF"
Aspect="Uniform"
HorizontalOptions="Start">
<Path.Data>
<!-- FillRule doesn't need to be set, because EvenOdd is the default. -->
<GeometryGroup>
<EllipseGeometry RadiusX="50"
RadiusY="50"
Center="75,75" />
<EllipseGeometry RadiusX="70"
RadiusY="70"
Center="75,75" />
<EllipseGeometry RadiusX="100"
RadiusY="100"
Center="75,75" />
<EllipseGeometry RadiusX="120"
RadiusY="120"
Center="75,75" />
</GeometryGroup>
</Path.Data>
</Path>
In the image above, the red circles represent points, and the lines represent arbitrary rays. For the upper point,
the two arbitrary rays each pass through an even number of line segments. Therefore, the ring the point is in
isn't filled. For the lower point, the two arbitrary rays each pass through an odd number of line segments.
Therefore, the ring the point is in is filled.
Nonzero
The Nonzero fill rule draws a ray from the point to infinity in any direction and then examines the places where
a segment of the shape crosses the ray. Starting with a count of zero, the count is incremented each time a
segment crosses the ray from left to right and decremented each time a segment crosses the ray from right to
left. After counting the crossings, if the result is zero then the point is outside the polygon. Otherwise, it's inside.
The following XAML example creates and renders a composite shape, with the FillRule set to Nonzero :
<Path Stroke="Black"
Fill="#CCCCFF"
Aspect="Uniform"
HorizontalOptions="Start">
<Path.Data>
<GeometryGroup FillRule="Nonzero">
<EllipseGeometry RadiusX="50"
RadiusY="50"
Center="75,75" />
<EllipseGeometry RadiusX="70"
RadiusY="70"
Center="75,75" />
<EllipseGeometry RadiusX="100"
RadiusY="100"
Center="75,75" />
<EllipseGeometry RadiusX="120"
RadiusY="120"
Center="75,75" />
</GeometryGroup>
</Path.Data>
</Path>
In this example, a composite shape made up of a series of concentric rings is displayed:
In the composite shape, notice that all rings are filled. This is because all the segments are running in the same
direction, and so a ray drawn from any point will cross one or more segments and the sum of the crossings will
not equal zero:
In the image above the red arrows represent the direction the segments are drawn, and black arrow represents
an arbitrary ray running from a point in the innermost ring. Starting with a value of zero, for each segment that
the ray crosses, a value of one is added because the segment crosses the ray from left to right.
A more complex shape with segments running in different directions is required to better demonstrate the
behavior of the Nonzero fill rule. The following XAML example creates a similar shape to the previous example,
except that it's created with a PathGeometry rather than an EllipseGeometry :
<Path Stroke="Black"
Fill="#CCCCFF">
<Path.Data>
<GeometryGroup FillRule="Nonzero">
<PathGeometry>
<PathGeometry.Figures>
<!-- Inner ring -->
<PathFigure StartPoint="120,120">
<PathFigure.Segments>
<PathSegmentCollection>
<ArcSegment Size="50,50"
IsLargeArc="True"
SweepDirection="CounterClockwise"
Point="140,120" />
</PathSegmentCollection>
</PathFigure.Segments>
</PathFigure>
In this example, a series of arc segments are drawn, that aren't closed:
In the image above, the third arc from the center is not filled. This is because the sum of the values from a given
ray crossing the segments in its path is zero:
In the image above, the red circle represents a point, the black lines represent arbitrary rays that move out from
the point in the non-filled region, and the red arrows represent the direction the segments are drawn. As can be
seen, the sum of the values from the rays crossing the segments is zero:
The arbitrary ray that travels diagonally right crosses two segments that run in different directions.
Therefore, the segments cancel each other out giving a value of zero.
The arbitrary ray that travels diagonally left crosses a total of six segments. However, the crossings cancel
each other out so that zero is the final sum.
A sum of zero results in the ring not being filled.
Geometries
3/18/2022 • 17 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) Geometry class, and the classes that derive from it, enable you to
describe the geometry of a 2D shape. Geometry objects can be simple, such as rectangles and circles, or
composite, created from two or more geometry objects. In addition, more complex geometries can be created
that include arcs and curves.
The Geometry class is the parent class for several classes that define different categories of geometries:
EllipseGeometry , which represents the geometry of an ellipse or circle.
GeometryGroup , which represents a container that can combine multiple geometry objects into a single
object.
LineGeometry , which represents the geometry of a line.
PathGeometry , which represents the geometry of a complex shape that can be composed of arcs, curves,
ellipses, lines, and rectangles.
RectangleGeometry , which represents the geometry of a rectangle or square.
NOTE
There's also a RoundedRectangleGeometry class that derives from the GeometryGroup class. For more information, see
RoundRectangleGeometry.
The Geometry and Shape classes seem similar, in that they both describe 2D shapes, but have an important
difference. The Geometry class derives from the BindableObject class, while the Shape class derives from the
View class. Therefore, Shape objects can render themselves and participate in the layout system, while
Geometry objects cannot. While Shape objects are more readily usable than Geometry objects, Geometry
objects are more versatile. While a Shape object is used to render 2D graphics, a Geometry object can be used
to define the geometric region for 2D graphics, and define a region for clipping.
The following classes have properties that can be set to Geometry objects:
The class uses a Geometry to describe its contents. You can render a Geometry by setting the
Path
Path.Data property to a Geometry object, and setting the Path object's Fill and Stroke properties.
The VisualElement class has a Clip property, of type Geometry , that defines the outline of the contents of
an element. When the Clip property is set to a Geometry object, only the area that is within the region of
the Geometry will be visible. For more information, see Clip with a Geometry.
The classes that derive from the Geometry class can be grouped into three categories: simple geometries, path
geometries, and composite geometries.
Simple geometries
The simple geometry classes are EllipseGeometry , LineGeometry , and RectangleGeometry . They are used to
create basic geometric shapes, such as circles, lines, and rectangles. These same shapes, as well as more complex
shapes, can be created using a PathGeometry or by combining geometry objects together, but these classes
provide a simpler approach for producing these basic geometric shapes.
EllipseGeometry
An ellipse geometry represents the geometry or an ellipse or circle, and is defined by a center point, an x-radius,
and a y-radius.
The EllipseGeometry class defines the following properties:
, of type Point , which represents the center point of the geometry.
Center
RadiusX , of type double , which represents the x-radius value of the geometry. The default value of this
property is 0.0.
RadiusY , of type double , which represents the y-radius value of the geometry. The default value of this
property is 0.0.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
The following example shows how to create and render an EllipseGeometry in a Path object:
<Path Fill="Blue"
Stroke="Red">
<Path.Data>
<EllipseGeometry Center="50,50"
RadiusX="50"
RadiusY="50" />
</Path.Data>
</Path>
In this example, the center of the EllipseGeometry is set to (50,50) and the x-radius and y-radius are both set to
50. This creates a red circle with a diameter of 100 device-independent units, whose interior is painted blue:
LineGeometry
A line geometry represents the geometry of a line, and is defined by specifying the start point of the line and the
end point.
The LineGeometry class defines the following properties:
StartPoint , of type Point , which represents the start point of the line.
EndPoint , of type Point , which represents the end point of the line.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
The following example shows how to create and render a LineGeometry in a Path object:
<Path Stroke="Black">
<Path.Data>
<LineGeometry StartPoint="10,20"
EndPoint="100,130" />
</Path.Data>
</Path>
RectangleGeometry
A rectangle geometry represents the geometry of a rectangle or square, and is defined with a Rect structure
that specifies its relative position and its height and width.
The RectangleGeometry class defines the Rect property, of type Rect , which represents the dimensions of the
rectangle. This property is backed by a BindableProperty object, which means that it can be the target of data
bindings, and styled.
The following example shows how to create and render a RectangleGeometry in a Path object:
<Path Fill="Blue"
Stroke="Red">
<Path.Data>
<RectangleGeometry Rect="10,10,150,100" />
</Path.Data>
</Path>
The position and dimensions of the rectangle are defined by a Rect structure. In this example, the position is
(10,10), the width is 150, and the height is 100 device-independent units:
Path geometries
A path geometry describes a complex shape that can be composed of arcs, curves, ellipses, lines, and rectangles.
The PathGeometry class defines the following properties:
, of type PathFigureCollection , which represents the collection of PathFigure objects that describe
Figures
the path's contents.
FillRule , of type FillRule , which determines how the intersecting areas contained in the geometry are
combined. The default value of this property is FillRule.EvenOdd .
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
For more information about the FillRule enumeration, see .NET MAUI Shapes: Fill rules.
NOTE
The Figures property is the ContentProperty of the PathGeometry class, and so does not need to be explicitly set
from XAML.
A PathGeometry is made up of a collection of PathFigure objects, with each PathFigure describing a shape in
the geometry. Each PathFigure is itself comprised of one or more PathSegment objects, each of which describes
a segment of the shape. There are many types of segments:
ArcSegment , which creates an elliptical arc between two points.
BezierSegment , which creates a cubic Bezier curve between two points.
LineSegment , which creates a line between two points.
PolyBezierSegment , which creates a series of cubic Bezier curves.
PolyLineSegment , which creates a series of lines.
PolyQuadraticBezierSegment , which creates a series of quadratic Bezier curves.
QuadraticBezierSegment , which creates a quadratic Bezier curve.
All the above classes derive from the abstract PathSegment class.
The segments within a PathFigure are combined into a single geometric shape with the end point of each
segment being the start point of the next segment. The StartPoint property of a PathFigure specifies the point
from which the first segment is drawn. Each subsequent segment starts at the end point of the previous
segment. For example, a vertical line from 10,50 to 10,150 can be defined by setting the StartPoint property
to 10,50 and creating a LineSegment with a Point property setting of 10,150 :
<Path Stroke="Black">
<Path.Data>
<PathGeometry>
<PathGeometry.Figures>
<PathFigureCollection>
<PathFigure StartPoint="10,50">
<PathFigure.Segments>
<PathSegmentCollection>
<LineSegment Point="10,150" />
</PathSegmentCollection>
</PathFigure.Segments>
</PathFigure>
</PathFigureCollection>
</PathGeometry.Figures>
</PathGeometry>
</Path.Data>
</Path>
More complex geometries can be created by using a combination of PathSegment objects, and by using multiple
PathFigure objects within a PathGeometry .
Create an ArcSegment
An ArcSegment creates an elliptical arc between two points. An elliptical arc is defined by its start and end points,
x- and y-radius, x-axis rotation factor, a value indicating whether the arc should be greater than 180 degrees, and
a value describing the direction in which the arc is drawn.
The ArcSegment class defines the following properties:
Point , of type Point , which represents the endpoint of the elliptical arc. The default value of this property is
(0,0).
Size , of type Size , which represents the x- and y-radius of the arc. The default value of this property is
(0,0).
RotationAngle , of type double , which represents the amount in degrees by which the ellipse is rotated
around the x-axis. The default value of this property is 0.
SweepDirection , of type SweepDirection , which specifies the direction in which the arc is drawn. The default
value of this property is SweepDirection.CounterClockwise .
IsLargeArc , of type bool , which indicates whether the arc should be greater than 180 degrees. The default
value of this property is false .
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
NOTE
The ArcSegment class does not contain a property for the starting point of the arc. It only defines the end point of the
arc it represents. The start point of the arc is the current point of the PathFigure to which the ArcSegment is added.
The following example shows how to create and render an ArcSegment in a Path object:
<Path Stroke="Black">
<Path.Data>
<PathGeometry>
<PathGeometry.Figures>
<PathFigureCollection>
<PathFigure StartPoint="10,10">
<PathFigure.Segments>
<PathSegmentCollection>
<ArcSegment Size="100,50"
RotationAngle="45"
IsLargeArc="True"
SweepDirection="CounterClockwise"
Point="200,100" />
</PathSegmentCollection>
</PathFigure.Segments>
</PathFigure>
</PathFigureCollection>
</PathGeometry.Figures>
</PathGeometry>
</Path.Data>
</Path>
NOTE
The BezierSegment class does not contain a property for the starting point of the curve. The start point of the curve is
the current point of the PathFigure to which the BezierSegment is added.
The two control points of a cubic Bezier curve behave like magnets, attracting portions of what would otherwise
be a straight line toward themselves and producing a curve. The first control point affects the start portion of
the curve. The second control point affects the end portion of the curve. The curve doesn't necessarily pass
through either of the control points. Instead, each control point moves its portion of the line toward itself, but
not through itself.
The following example shows how to create and render a BezierSegment in a Path object:
<Path Stroke="Black">
<Path.Data>
<PathGeometry>
<PathGeometry.Figures>
<PathFigureCollection>
<PathFigure StartPoint="10,10">
<PathFigure.Segments>
<PathSegmentCollection>
<BezierSegment Point1="100,0"
Point2="200,200"
Point3="300,10" />
</PathSegmentCollection>
</PathFigure.Segments>
</PathFigure>
</PathFigureCollection>
</PathGeometry.Figures>
</PathGeometry>
</Path.Data>
</Path>
In this example, a cubic Bezier curve is drawn from (10,10) to (300,10). The curve has two control points at
(100,0) and (200,200):
Create a LineSegment
A LineSegment creates a line between two points.
The LineSegment class defines the Point property, of type Point , which represents the end point of the line
segment. The default value of this property is (0,0), and it's backed by a BindableProperty object, which means
that it can be the target of data bindings, and styled.
NOTE
The LineSegment class does not contain a property for the starting point of the line. It only defines the end point. The
start point of the line is the current point of the PathFigure to which the LineSegment is added.
The following example shows how to create and render LineSegment objects in a Path object:
<Path Stroke="Black"
Aspect="Uniform"
HorizontalOptions="Start">
<Path.Data>
<PathGeometry>
<PathGeometry.Figures>
<PathFigureCollection>
<PathFigure IsClosed="True"
StartPoint="10,100">
<PathFigure.Segments>
<PathSegmentCollection>
<LineSegment Point="100,100" />
<LineSegment Point="100,50" />
</PathSegmentCollection>
</PathFigure.Segments>
</PathFigure>
</PathFigureCollection>
</PathGeometry.Figures>
</PathGeometry>
</Path.Data>
</Path>
In this example, a line segment is drawn from (10,100) to (100,100), and from (100,100) to (100,50). In addition,
the PathFigure is closed because its IsClosed property is set to true . This results in a triangle being drawn:
Create a PolyBezierSegment
A PolyBezierSegment creates one or more cubic Bezier curves.
The PolyBezierSegment class defines the Points property, of type PointCollection , which represents the points
that define the PolyBezierSegment . A PointCollection is an ObservableCollection of Point objects. This
property is backed by a BindableProperty object, which means that it can be the target of data bindings, and
styled.
NOTE
The PolyBezierSegment class does not contain a property for the starting point of the curve. The start point of the
curve is the current point of the PathFigure to which the PolyBezierSegment is added.
The following example shows how to create and render a PolyBezierSegment in a Path object:
<Path Stroke="Black">
<Path.Data>
<PathGeometry>
<PathGeometry.Figures>
<PathFigureCollection>
<PathFigure StartPoint="10,10">
<PathFigure.Segments>
<PathSegmentCollection>
<PolyBezierSegment Points="0,0 100,0 150,100 150,0 200,0 300,10" />
</PathSegmentCollection>
</PathFigure.Segments>
</PathFigure>
</PathFigureCollection>
</PathGeometry.Figures>
</PathGeometry>
</Path.Data>
</Path>
In this example, the PolyBezierSegment specifies two cubic Bezier curves. The first curve is from (10,10) to
(150,100) with a control point of (0,0), and another control point of (100,0). The second curve is from (150,100)
to (300,10) with a control point of (150,0) and another control point of (200,0):
Create a PolyLineSegment
A PolyLineSegment creates one or more line segments.
The PolyLineSegment class defines the Points property, of type PointCollection , which represents the points
that define the PolyLineSegment . A PointCollection is an ObservableCollection of Point objects. This property
is backed by a BindableProperty object, which means that it can be the target of data bindings, and styled.
NOTE
The PolyLineSegment class does not contain a property for the starting point of the line. The start point of the line is
the current point of the PathFigure to which the PolyLineSegment is added.
The following example shows how to create and render a PolyLineSegment in a Path object:
<Path Stroke="Black">
<Path.Data>
<PathGeometry>
<PathGeometry.Figures>
<PathFigure StartPoint="10,10">
<PathFigure.Segments>
<PolyLineSegment Points="50,10 50,50" />
</PathFigure.Segments>
</PathFigure>
</PathGeometry.Figures>
</PathGeometry>
</Path.Data>
</Path>
In this example, the PolyLineSegment specifies two lines. The first line is from (10,10) to (50,10), and the second
line is from (50,10) to (50,50):
Create a PolyQuadraticBezierSegment
A PolyQuadraticBezierSegment creates one or more quadratic Bezier curves.
The PolyQuadraticBezierSegment class defines the Points property, of type PointCollection , which represents
the points that define the PolyQuadraticBezierSegment . A PointCollection is an ObservableCollection of Point
objects. This property is backed by a BindableProperty object, which means that it can be the target of data
bindings, and styled.
NOTE
The PolyQuadraticBezierSegment class does not contain a property for the starting point of the curve. The start point
of the curve is the current point of the PathFigure to which the PolyQuadraticBezierSegment is added.
The following example shows to create and render a PolyQuadraticBezierSegment in a Path object:
<Path Stroke="Black">
<Path.Data>
<PathGeometry>
<PathGeometry.Figures>
<PathFigureCollection>
<PathFigure StartPoint="10,10">
<PathFigure.Segments>
<PathSegmentCollection>
<PolyQuadraticBezierSegment Points="100,100 150,50 0,100 15,200" />
</PathSegmentCollection>
</PathFigure.Segments>
</PathFigure>
</PathFigureCollection>
</PathGeometry.Figures>
</PathGeometry>
</Path.Data>
</Path>
In this example, the PolyQuadraticBezierSegment specifies two Bezier curves. The first curve is from (10,10) to
(150,50) with a control point at (100,100). The second curve is from (100,100) to (15,200) with a control point at
(0,100):
Create a QuadraticBezierSegment
A QuadraticBezierSegment creates a quadratic Bezier curve between two points.
The QuadraticBezierSegment class defines the following properties:
Point1 , of type Point , which represents the control point of the curve. The default value of this property is
(0,0).
Point2 , of type Point , which represents the end point of the curve. The default value of this property is
(0,0).
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
NOTE
The QuadraticBezierSegment class does not contain a property for the starting point of the curve. The start point of
the curve is the current point of the PathFigure to which the QuadraticBezierSegment is added.
The following example shows how to create and render a QuadraticBezierSegment in a Path object:
<Path Stroke="Black">
<Path.Data>
<PathGeometry>
<PathGeometry.Figures>
<PathFigureCollection>
<PathFigure StartPoint="10,10">
<PathFigure.Segments>
<PathSegmentCollection>
<QuadraticBezierSegment Point1="200,200"
Point2="300,10" />
</PathSegmentCollection>
</PathFigure.Segments>
</PathFigure>
</PathFigureCollection>
</PathGeometry.Figures>
</PathGeometry>
</Path.Data>
</Path>
In this example, a quadratic Bezier curve is drawn from (10,10) to (300,10). The curve has a control point at
(200,200):
In this example, a BezierSegment is first defined using four points. The example then adds a LineSegment , which
is drawn between the end point of the BezierSegment to the point specified by the LineSegment . Finally, an
ArcSegment is drawn from the end point of the LineSegment to the point specified by the ArcSegment .
Even more complex geometries can be created by using multiple PathFigure objects within a PathGeometry .
The following example creates a PathGeometry from seven PathFigure objects, some of which contain multiple
PathSegment objects:
<Path Stroke="Red"
StrokeThickness="12"
StrokeLineJoin="Round">
<Path.Data>
<PathGeometry>
<!-- H -->
<PathFigure StartPoint="0,0">
<LineSegment Point="0,100" />
</PathFigure>
<PathFigure StartPoint="0,50">
<LineSegment Point="50,50" />
</PathFigure>
<PathFigure StartPoint="50,0">
<LineSegment Point="50,100" />
</PathFigure>
<!-- E -->
<PathFigure StartPoint="125, 0">
<BezierSegment Point1="60, -10"
Point2="60, 60"
Point3="125, 50" />
<BezierSegment Point1="60, 40"
Point2="60, 110"
Point3="125, 100" />
</PathFigure>
<!-- L -->
<PathFigure StartPoint="150, 0">
<LineSegment Point="150, 100" />
<LineSegment Point="200, 100" />
</PathFigure>
<!-- L -->
<PathFigure StartPoint="225, 0">
<LineSegment Point="225, 100" />
<LineSegment Point="275, 100" />
</PathFigure>
<!-- O -->
<PathFigure StartPoint="300, 50">
<ArcSegment Size="25, 50"
Point="300, 49.9"
IsLargeArc="True" />
</PathFigure>
</PathGeometry>
</Path.Data>
</Path>
In this example, the word "Hello" is drawn using a combination of LineSegment and BezierSegment objects,
along with a single ArcSegment object:
Composite geometries
Composite geometry objects can be created using a GeometryGroup . The GeometryGroup class creates a
composite geometry from one or more Geometry objects. Any number of Geometry objects can be added to a
GeometryGroup .
The GeometryGroup class defines the following properties:
Children , of type , which species the objects that define the GeomtryGroup . A
GeometryCollection
GeometryCollection is an ObservableCollection of Geometry objects.
FillRule , of type FillRule , which specifies how the intersecting areas in the GeometryGroup are combined.
The default value of this property is FillRule.EvenOdd .
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
NOTE
The Children property is the ContentProperty of the GeometryGroup class, and so does not need to be explicitly set
from XAML.
For more information about the FillRule enumeration, see Fill rules.
To draw a composite geometry, set the required Geometry objects as the children of a GeometryGroup , and
display them with a Path object. The following XAML shows an example of this:
<Path Stroke="Green"
StrokeThickness="2"
Fill="Orange">
<Path.Data>
<GeometryGroup>
<EllipseGeometry RadiusX="100"
RadiusY="100"
Center="150,150" />
<EllipseGeometry RadiusX="100"
RadiusY="100"
Center="250,150" />
<EllipseGeometry RadiusX="100"
RadiusY="100"
Center="150,250" />
<EllipseGeometry RadiusX="100"
RadiusY="100"
Center="250,250" />
</GeometryGroup>
</Path.Data>
</Path>
In this example, four EllipseGeometry objects with identical x-radius and y-radius coordinates, but with different
center coordinates, are combined. This creates four overlapping circles, whose interiors are filled orange due to
the default EvenOdd fill rule:
RoundRectangleGeometry
A round rectangle geometry represents the geometry of a rectangle, or square, with rounded corners, and is
defined by a corner radius and a Rect structure that specifies its relative position and its height and width.
The RoundRectangleGeometry class, which derives from the GeometryGroup class, defines the following properties:
CornerRadius , of type CornerRadius , which is the corner radius of the geometry.
Rect , of type Rect , which represents the dimensions of the rectangle.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
NOTE
The fill rule used by the RoundRectangleGeometry is FillRule.Nonzero . For more information about fill rules, see Fill
rules.
The following example shows how to create and render a RoundRectangleGeometry in a Path object:
<Path Fill="Blue"
Stroke="Red">
<Path.Data>
<RoundRectangleGeometry CornerRadius="5"
Rect="10,10,150,100" />
</Path.Data>
</Path>
The position and dimensions of the rectangle are defined by a Rect structure. In this example, the position is
(10,10), the width is 150, and the height is 100 device-independent units. In addition, the rectangle corners are
rounded with a radius of 5 device-independent units.
The following example shows how to use a Geometry object as the clip region for an Image :
<Image Source="monkeyface.png">
<Image.Clip>
<EllipseGeometry RadiusX="100"
RadiusY="100"
Center="180,180" />
</Image.Clip>
</Image>
In this example, an EllipseGeometry with RadiusX and RadiusY values of 100, and a Center value of (180,180)
is set to the Clip property of an Image . Only the part of the image that is within the area of the ellipse will be
displayed:
NOTE
Simple geometries, path geometries, and composite geometries can all be used to clip VisualElement objects.
Other features
The GeometryHelper class provides the following helper methods:
FlattenGeometry , which flattens a Geometry into a PathGeometry .
FlattenCubicBezier , which flattens a cubic Bezier curve into a List<Point> collection.
FlattenQuadraticBezier , which flattens a quadratic Bezier curve into a List<Point> collection.
FlattenArc , which flattens an elliptical arc into a List<Point> collection.
Line
3/18/2022 • 2 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) Line class derives from the Shape class, and can be used to draw
lines. For information on the properties that the Line class inherits from the Shape class, see Shapes.
Line defines the following properties:
X1 , of type double, indicates the x-coordinate of the start point of the line. The default value of this property
is 0.0.
Y1 , of type double, indicates the y-coordinate of the start point of the line. The default value of this property
is 0.0.
X2 , of type double, indicates the x-coordinate of the end point of the line. The default value of this property
is 0.0.
Y2 , of type double, indicates the y-coordinate of the end point of the line. The default value of this property
is 0.0.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
For information about controlling how line ends are drawn, see Control line ends.
Create a Line
To draw a line, create a Line object and set its X1 and Y1 properties to its start point, and its X2 and Y
properties to its end point. In addition, set its Stroke property to a Brush -derived object because a line without
a stroke is invisible. For more information about Brush objects, see Brushes.
NOTE
Setting the Fill property of a Line has no effect, because a line has no interior.
<Line X1="40"
Y1="0"
X2="0"
Y2="120"
Stroke="Red" />
Because the X1 , Y1 , X2 , and Y2 properties have default values of 0, it's possible to draw some lines with
minimal syntax:
<Line Stroke="Red"
X2="200" />
In this example, a horizontal line that's 200 device-independent units long is defined. Because the other
properties are 0 by default, a line is drawn from (0,0) to (200,0).
The following XAML example shows how to draw a dashed line:
<Line X1="40"
Y1="0"
X2="0"
Y2="120"
Stroke="DarkBlue"
StrokeDashArray="1,1"
StrokeDashOffset="6" />
In this example, a dark blue dashed diagonal line is drawn from (40,0) to (0,120):
For more information about drawing a dashed line, see Draw dashed shapes.
Path
3/18/2022 • 2 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) Path class derives from the Shape class, and can be used to draw
curves and complex shapes. These curves and shapes are often described using Geometry objects. For
information on the properties that the Path class inherits from the Shape class, see Shapes.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Create a Path
To draw a path, create a Path object and set its Data property. There are two techniques for setting the Data
property:
You can set a string value for Data in XAML, using path markup syntax. With this approach, the Path.Data
value is consuming a serialization format for graphics. Typically, you don't edit this string value by hand after
it's created. Instead, you use design tools to manipulate the data, and export it as a string fragment that's
consumable by the Data property.
You can set the Data property to a Geometry object. This can be a specific Geometry object, or a
GeometryGroup which acts as a container that can combine multiple geometry objects into a single object.
The Data string begins with the move command, indicated by M , which establishes an absolute start point for
the path. L is the line command, which creates a straight line from the start point to the specified end point. Z
is the close command, which creates a line that connects the current point to the starting point. The result is a
triangle:
For more information about path markup syntax, see Path markup syntax.
Create a Path with Geometry objects
Curves and shapes can be described using Geometry objects, which are used to set the Path object's Data
property. There are a variety of Geometry objects to choose from. The EllipseGeometry , LineGeometry , and
RectangleGeometry classes describe relatively simple shapes. To create more complex shapes or create curves,
use a PathGeometry .
PathGeometry objects are comprised of one or more PathFigure objects. Each PathFigure object represents a
different shape. Each PathFigure object is itself comprised of one or more PathSegment objects, each
representing a connection portion of the shape. Segment types include the following the LineSegment ,
BezierSegment , and ArcSegment classes.
The following XAML example shows how to draw a triangle using a PathGeometry object:
<Path Stroke="Black"
Aspect="Uniform"
HorizontalOptions="Start">
<Path.Data>
<PathGeometry>
<PathGeometry.Figures>
<PathFigureCollection>
<PathFigure IsClosed="True"
StartPoint="10,100">
<PathFigure.Segments>
<PathSegmentCollection>
<LineSegment Point="100,100" />
<LineSegment Point="100,50" />
</PathSegmentCollection>
</PathFigure.Segments>
</PathFigure>
</PathFigureCollection>
</PathGeometry.Figures>
</PathGeometry>
</Path.Data>
</Path>
In this example, the start point of the triangle is (10,100). A line segment is drawn from (10,100) to (100,100),
and from (100,100) to (100,50). Then the figures first and last segments are connected, because the
PathFigure.IsClosed property is set to true . The result is a triangle:
.NET Multi-platform App UI (.NET MAUI) path markup syntax enables you to compactly specify path geometries
in XAML.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
<Path Stroke="Black"
Data="M13.908992,16.207977 L32.000049,16.207977 32.000049,31.999985 13.908992,30.109983Z" />
Path markup syntax is composed of an optional FillRule value, and one or more figure descriptions. This
syntax can be expressed as: <Path Data=" [fillRule] figureDescription [figureDescription] * " ... />
In this syntax:
fillRule is an optional FillRule that specifies whether the geometry should use the EvenOdd or Nonzero fill
rule. F0 is used to specify the EvenOdd fill rule, while F1 is used to specify the Nonzero fill rule. For more
information about fill rules, see Fill rules.
figureDescription represents a figure composed of a move command, draw commands, and an optional
close command. A move command specifies the start point of the figure. Draw commands describe the
figure's contents, and the optional close command closes the figure.
In the example above, the path markup syntax specifies a start point using the move command ( M ), a series of
straight lines using the line command ( L ), and closes the path with the close command ( Z ).
In path markup syntax, spaces are not required before or after commands. In addition, two numbers don't have
to be separated by a comma or white space, but this can only be achieved when the string is unambiguous.
TIP
Path markup syntax is compatible with Scalable Vector Graphics (SVG) image path definitions, and so it can be useful for
porting graphics from SVG format.
While path markup syntax is intended for consumption in XAML, it can be converted to a Geometry object in
code by invoking the ConvertFromInvariantString method in the PathGeometryConverter class:
Move command
The move command specifies the start point of a new figure. The syntax for this command is: M startPoint or
m startPoint.
In this syntax, startPoint is a Point structure that specifies the start point of a new figure. If you list multiple
points after the move command, a line is drawn to those points.
M 10,10 is an example of a valid move command.
Draw commands
A draw command can consist of several shape commands. The following draw commands are available:
Line ( L or l ).
Horizontal line ( H or h ).
Vertical line ( V or v ).
Elliptical arc ( A or a ).
Cubic Bezier curve ( C or c ).
Quadratic Bezier curve ( Q or q ).
Smooth cubic Bezier curve ( S or s ).
Smooth quadratic Bezier curve ( T or t ).
Each draw command is specified with a case-insensitive letter. When sequentially entering more than one
command of the same type, you can omit the duplicate command entry. For example L 100,200 300,400 is
equivalent to L 100,200 L 300,400 .
Line command
The line command creates a straight line between the current point and the specified end point. The syntax for
this command is: L endPoint or l endPoint.
In this syntax, endPoint is a Point that represents the end point of the line.
L 20,30 and L 20 30 are examples of valid line commands.
For information about creating a straight line as a PathGeometry object, see Create a LineSegment.
Horizontal line command
The horizontal line command creates a horizontal line between the current point and the specified x-coordinate.
The syntax for this command is: H x or h x.
In this syntax, x is a double that represents the x-coordinate of the end point of the line.
H 90 is an example of a valid horizontal line command.
Vertical line command
The vertical line command creates a vertical line between the current point and the specified y-coordinate. The
syntax for this command is: V y or v y.
In this syntax, y is a double that represents the y-coordinate of the end point of the line.
V 90 is an example of a valid vertical line command.
Elliptical arc command
The elliptical arc command creates an elliptical arc between the current point and the specified end point. The
syntax for this command is: A size rotationAngle isLargeArcFlag sweepDirectionFlag endPoint or a size
rotationAngle isLargeArcFlag sweepDirectionFlag endPoint.
In this syntax:
size is a that represents the x- and y-radius of the arc.
Size
rotationAngle is a double that represents the rotation of the ellipse, in degrees.
isLargeArcFlag should be set to 1 if the angle of the arc should be 180 degrees or greater, otherwise set it to
0.
sweepDirectionFlag should be set to 1 if the arc is drawn in a positive-angle direction, otherwise set it to 0.
endPoint is a Point to which the arc is drawn.
Close command
The close command ends the current figure and creates a line that connects the current point to the starting
point of the figure. Therefore, this command creates a line-join between the last segment and the first segment
of the figure.
The syntax for the close command is: Z or z .
Additional values
Instead of a standard numerical value, you can also use the following case-sensitive special values:
Infinity represents double.PositiveInfinity .
-Infinity represents double.NegativeInfinity .
NaN represents double.NaN .
In addition, you may also use case-insensitive scientific notation. Therefore, +1.e17 is a valid value.
Path transforms
3/18/2022 • 12 minutes to read • Edit Online
A .NET Multi-platform App UI (.NET MAUI) Transform defines how to transform a Path object from one
coordinate space to another coordinate space. When a transform is applied to a Path object, it changes how the
object is rendered in the UI.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Transforms can be categorized into four general classifications: rotation, scaling, skew, and translation. .NET
MAUI defines a class for each of these transform classifications:
RotateTransform , which rotates a Path by a specified Angle .
ScaleTransform , which scales a Path object by specified ScaleX and ScaleY amounts.
SkewTransform , which skews a Path object by specified AngleX and AngleY amounts.
TranslateTransform , which moves a Path object by specified X and Y amounts.
.NET MAUI also provides the following classes for creating more complex transformations:
TransformGroup , which represents a composite transform composed of multiple transform objects.
CompositeTransform , which applies multiple transform operations to a Path object.
MatrixTransform , which creates custom transforms that are not provided by the other transform classes.
All of these classes derive from the Transform class, which defines a Value property of type Matrix , which
represents the current transformation as a Matrix object. This property is backed by a BindableProperty object,
which means that it can be the target of data bindings, and styled. For more information about the Matrix
struct, see Transform matrix.
To apply a transform to a Path , you create a transform class and set it as the value of the Path.RenderTransform
property.
Rotation transform
A rotate transform rotates a Path object clockwise about a specified point in a 2D x-y coordinate system.
The RotateTransform class, which derives from the Transform class, defines the following properties:
, of type double , represents the angle, in degrees, of clockwise rotation. The default value of this
Angle
property is 0.0.
CenterX , of type double , represents the x-coordinate of the rotation center point. The default value of this
property is 0.0.
CenterY , of type double , represents the y-coordinate of the rotation center point. The default value of this
property is 0.0.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
The CenterX and CenterY properties specify the point about which the Path object is rotated. This center point
is expressed in the coordinate space of the object that's transformed. By default, the rotation is applied to (0,0),
which is the upper-left corner of the Path object.
The following example shows how to rotate a Path object:
<Path Stroke="Black"
Aspect="Uniform"
HorizontalOptions="Center"
HeightRequest="100"
WidthRequest="100"
Data="M13.908992,16.207977L32.000049,16.207977 32.000049,31.999985 13.908992,30.109983z">
<Path.RenderTransform>
<RotateTransform CenterX="0"
CenterY="0"
Angle="45" />
</Path.RenderTransform>
</Path>
In this example, the Path object is rotated 45 degrees about its upper-left corner.
Scale transform
A scale transform scales a Path object in the 2D x-y coordinate system.
The ScaleTransform class, which derives from the Transform class, defines the following properties:
, of type double , which represents the x-axis scale factor. The default value of this property is 1.0.
ScaleX
ScaleY , of type double , which represents the y-axis scale factor. The default value of this property is 1.0.
CenterX , of type double , which represents the x-coordinate of the center point of this transform. The default
value of this property is 0.0.
CenterY , of type double , which represents the y-coordinate of the center point of this transform. The default
value of this property is 0.0.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
The value of ScaleX and ScaleY have a huge impact on the resulting scaling:
Values between 0 and 1 decrease the width and height of the scaled object.
Values greater than 1 increase the width and height of the scaled object.
Values of 1 indicate that the object is not scaled.
Negative values flip the scale object horizontally and vertically.
Values between 0 and -1 flip the scale object and decrease its width and height.
Values less than -1 flip the object and increase its width and height.
Values of -1 flip the scaled object but do not change its horizontal or vertical size.
The CenterX and CenterY properties specify the point about which the Path object is scaled. This center point
is expressed in the coordinate space of the object that's transformed. By default, scaling is applied to (0,0), which
is the upper-left corner of the Path object. This has the effect of moving the Path object and making it appear
larger, because when you apply a transform you change the coordinate space in which the Path object resides.
The following example shows how to scale a Path object:
<Path Stroke="Black"
Aspect="Uniform"
HorizontalOptions="Center"
HeightRequest="100"
WidthRequest="100"
Data="M13.908992,16.207977L32.000049,16.207977 32.000049,31.999985 13.908992,30.109983z">
<Path.RenderTransform>
<ScaleTransform CenterX="0"
CenterY="0"
ScaleX="1.5"
ScaleY="1.5" />
</Path.RenderTransform>
</Path>
In this example, the Path object is scaled to 1.5 times the size.
Skew transform
A skew transform skews a Path object in the 2D x-y coordinate system, and is useful for creating the illusion of
3D depth in a 2D object.
The SkewTransform class, which derives from the Transform class, defines the following properties:
, of type double , which represents the x-axis skew angle, which is measured in degrees
AngleX
counterclockwise from the y-axis. The default value of this property is 0.0.
AngleY , of type double , which represents the y-axis skew angle, which is measured in degrees
counterclockwise from the x-axis. The default value of this property is 0.0.
CenterX , of type double , which represents the x-coordinate of the transform center. The default value of this
property is 0.0.
CenterY , of type double , which represents the y-coordinate of the transform center. The default value of this
property is 0.0.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
To predict the effect of a skew transformation, consider that AngleX skews x-axis values relative to the original
coordinate system. Therefore, for an AngleX of 30, the y-axis rotates 30 degrees through the origin and skews
the values in x by 30 degrees from that origin. Similarly, an AngleY of 30 skews the y values of the Path object
by 30 degrees from the origin.
NOTE
To skew a Path object in place, set the CenterX and CenterY properties to the object's center point.
In this example, a horizontal skew of 45 degrees is applied to the Path object, from a center point of (0,0).
Translate transform
A translate transform moves an object in the 2D x-y coordinate system.
The TranslateTransform class, which derives from the Transform class, defines the following properties:
X , of type double , which represents the distance to move along the x-axis. The default value of this
property is 0.0.
Y , of type double , which represents the distance to move along the y-axis. The default value of this
property is 0.0.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
Negative X values move an object to the left, while positive values move an object to the right. Negative Y
values move an object up, while positive values move an object down.
The following example shows how to translate a Path object:
<Path Stroke="Black"
Aspect="Uniform"
HorizontalOptions="Center"
HeightRequest="100"
WidthRequest="100"
Data="M13.908992,16.207977L32.000049,16.207977 32.000049,31.999985 13.908992,30.109983z">
<Path.RenderTransform>
<TranslateTransform X="50"
Y="50" />
</Path.RenderTransform>
</Path>
In this example, the Path object is moved 50 device-independent units to the right, and 50 device-independent
units down.
Multiple transforms
.NET MAUI has two classes that support applying multiple transforms to a Path object. These are
TransformGroup , and CompositeTransform . A TransformGroup performs transforms in any desired order, while a
CompositeTransform performs transforms in a specific order.
Transform groups
Transform groups represent composite transforms composed of multiple Transform objects.
The class, which derives from the Transform class, defines a Children property, of type
TransformGroup
TransformCollection , which represents a collection of Transform objects. This property is backed by a
BindableProperty object, which means that it can be the target of data bindings, and styled.
The order of transformations is important in a composite transform that uses the TransformGroup class. For
example, if you first rotate, then scale, then translate, you get a different result than if you first translate, then
rotate, then scale. One reason order is significant is that transforms like rotation and scaling are performed
respect to the origin of the coordinate system. Scaling an object that is centered at the origin produces a
different result to scaling an object that has been moved away from the origin. Similarly, rotating an object that
is centered at the origin produces a different result than rotating an object that has been moved away from the
origin.
The following example shows how to perform a composite transform using the TransformGroup class:
<Path Stroke="Black"
Aspect="Uniform"
HorizontalOptions="Center"
HeightRequest="100"
WidthRequest="100"
Data="M13.908992,16.207977L32.000049,16.207977 32.000049,31.999985 13.908992,30.109983z">
<Path.RenderTransform>
<TransformGroup>
<ScaleTransform ScaleX="1.5"
ScaleY="1.5" />
<RotateTransform Angle="45" />
</TransformGroup>
</Path.RenderTransform>
</Path>
In this example, the Path object is scaled to 1.5 times its size, and then rotated by 45 degrees.
Composite transforms
A composite transform applies multiple transforms to an object.
The CompositeTransform class, which derives from the Transform class, defines the following properties:
CenterX, of type double , which represents the x-coordinate of the center point of this transform. The default
value of this property is 0.0.
CenterY , of type double , which represents the y-coordinate of the center point of this transform. The default
value of this property is 0.0.
ScaleX , of type double , which represents the x-axis scale factor. The default value of this property is 1.0.
ScaleY , of type double , which represents the y-axis scale factor. The default value of this property is 1.0.
SkewX , of type double , which represents the x-axis skew angle, which is measured in degrees
counterclockwise from the y-axis. The default value of this property is 0.0.
SkewY , of type double , which represents the y-axis skew angle, which is measured in degrees
counterclockwise from the x-axis. The default value of this property is 0.0.
Rotation , of type double , represents the angle, in degrees, of clockwise rotation. The default value of this
property is 0.0.
TranslateX , of type double , which represents the distance to move along the x-axis. The default value of this
property is 0.0.
TranslateY , of type double , which represents the distance to move along the y-axis. The default value of this
property is 0.0.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
A CompositeTransform applies transforms in this order:
1. Scale ( ScaleX and ScaleY ).
2. Skew ( SkewX and SkewY ).
3. Rotate ( Rotation ).
4. Translate ( TranslateX , TranslateY ).
If you want to apply multiple transforms to an object in a different order, you should create a TransformGroup
and insert the transforms in your intended order.
IMPORTANT
A CompositeTransform uses the same center points, CenterX and CenterY , for all transformations. If you want to
specify different center points per transform, use a TransformGroup ,
The following example shows how to perform a composite transform using the CompositeTransform class:
<Path Stroke="Black"
Aspect="Uniform"
HorizontalOptions="Center"
HeightRequest="100"
WidthRequest="100"
Data="M13.908992,16.207977L32.000049,16.207977 32.000049,31.999985 13.908992,30.109983z">
<Path.RenderTransform>
<CompositeTransform ScaleX="1.5"
ScaleY="1.5"
Rotation="45"
TranslateX="50"
TranslateY="50" />
</Path.RenderTransform>
</Path>
In this example, the Path object is scaled to 1.5 times its size, then rotated by 45 degrees, and then translated
by 50 device-independent units.
Transform matrix
A transform can be described in terms of a 3x3 affine transformation matrix, that performs transformations in
2D space. This 3x3 matrix is represented by the Matrix struct, which is a collection of three rows and three
columns of double values.
The Matrix struct defines the following properties:
Determinant , of type double , which gets the determinant of the matrix.
HasInverse , of type bool , which indicates whether the matrix is invertible.
Identity , of type Matrix , which gets an identity matrix.
HasIdentity , of type bool , which indicates whether the matrix is an identity matrix.
M11 , of type double , which represents the value of the first row and first column of the matrix.
M12 , of type double , which represents the value of the first row and second column of the matrix.
M21 , of type double , which represents the value of the second row and first column of the matrix.
M22 , of type double , which represents the value of the second row and second column of the matrix.
OffsetX , of type double , which represents the value of the third row and first column of the matrix.
OffsetY , of type double , which represents the value of the third row and second column of the matrix.
The OffsetX and OffsetY properties are so named because they specify the amount to translate the coordinate
space along the x-axis, and y-axis, respectively.
In addition, the Matrix struct exposes a series of methods that can be used to manipulate the matrix values,
including Append , Invert , Multiply , Prepend and many more.
The following table shows the structure of a .NET MAUI matrix:
M11
M12
0.0
M21
M22
0.0
OffsetX
OffsetY
1.0
NOTE
An affine transformation matrix has its final column equal to (0,0,1), so only the members in the first two columns need to
be specified.
By manipulating matrix values, you can rotate, scale, skew, and translate Path objects. For example, if you
change the OffsetX value to 100, you can use it move a Path object 100 device-independent units along the x-
axis. If you change the M22 value to 3, you can use it to stretch a Path object to three times its current height. If
you change both values, you move the Path object 100 device-independent units along the x-axis and stretch
its height by a factor of 3. In addition, affine transformation matrices can be multiplied to form any number of
linear transformations, such as rotation and skew, followed by translation.
Custom transforms
The MatrixTransform class, which derives from the Transform class, defines a Matrix property, of type Matrix ,
which represents the matrix that defines the transformation. This property is backed by a BindableProperty
object, which means that it can be the target of data bindings, and styled.
Any transform that you can describe with a TranslateTransform , ScaleTransform , RotateTransform , or
SkewTransform object can equally be described by a MatrixTransform . However, the TranslateTransform ,
ScaleTransform , RotateTransform , and SkewTransform classes are easier to conceptualize than setting the vector
components in a Matrix . Therefore, the MatrixTransform class is typically used to create custom
transformations that aren't provided by the RotateTransform , ScaleTransform , SkewTransform , or
TranslateTransform classes.
The following example shows how to transform a Path object using a MatrixTransform :
<Path Stroke="Black"
Aspect="Uniform"
HorizontalOptions="Center"
Data="M13.908992,16.207977L32.000049,16.207977 32.000049,31.999985 13.908992,30.109983z">
<Path.RenderTransform>
<MatrixTransform>
<MatrixTransform.Matrix>
<!-- M11 stretches, M12 skews -->
<Matrix OffsetX="10"
OffsetY="100"
M11="1.5"
M12="1" />
</MatrixTransform.Matrix>
</MatrixTransform>
</Path.RenderTransform>
</Path>
In this example, the Path object is stretched, skewed, and offset in both the X and Y dimensions.
Alternatively, this can be written in a simplified form that uses a type converter that's built into .NET MAUI:
<Path Stroke="Black"
Aspect="Uniform"
HorizontalOptions="Center"
Data="M13.908992,16.207977L32.000049,16.207977 32.000049,31.999985 13.908992,30.109983z">
<Path.RenderTransform>
<MatrixTransform Matrix="1.5,1,0,1,10,100" />
</Path.RenderTransform>
</Path>
In this example, the Matrix property is specified as a comma-delimited string consisting of six members: M11 ,
M12 , M21 , M22 , OffsetX , OffsetY . While the members are comma-delimited in this example, they can also be
delimited by one or more spaces.
In addition, the previous example can be simplified even further by specifying the same six members as the
value of the RenderTransform property:
<Path Stroke="Black"
Aspect="Uniform"
HorizontalOptions="Center"
RenderTransform="1.5 1 0 1 10 100"
Data="M13.908992,16.207977L32.000049,16.207977 32.000049,31.999985 13.908992,30.109983z" />
Polygon
3/18/2022 • 2 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) Polygon class derives from the Shape class, and can be used to
draw polygons, which are connected series of lines that form closed shapes. For information on the properties
that the Polygon class inherits from the Shape class, see Shapes.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Create a Polygon
To draw a polygon, create a Polygon object and set its Points property to the vertices of a shape. A line is
automatically drawn that connects the first and last points. To paint the inside of the polygon, set its Fill
property to a Brush -derived object. To give the polygon an outline, set its Stroke property to a Brush -derived
object. The StrokeThickness property specifies the thickness of the polygon outline. For more information about
Brush objects, see Brushes.
For more information about drawing a dashed polygon, see Draw dashed shapes.
The following XAML example shows a polygon that uses the default fill rule:
<Polygon Points="0 48, 0 144, 96 150, 100 0, 192 0, 192 96, 50 96, 48 192, 150 200 144 48"
Fill="Blue"
Stroke="Red"
StrokeThickness="3" />
In this example, the fill behavior of each polygon is determined using the EvenOdd fill rule.
The following XAML example shows a polygon that uses the Nonzero fill rule:
<Polygon Points="0 48, 0 144, 96 150, 100 0, 192 0, 192 96, 50 96, 48 192, 150 200 144 48"
Fill="Black"
FillRule="Nonzero"
Stroke="Yellow"
StrokeThickness="3" />
In this example, the fill behavior of each polygon is determined using the Nonzero fill rule.
Polyline
3/18/2022 • 2 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) Polyline class derives from the Shape class, and can be used to
draw a series of connected straight lines. A polyline is similar to a polygon, except the last point in a polyline is
not connected to the first point. For information on the properties that the Polyline class inherits from the
Shape class, see Shapes.
Create a Polyline
To draw a polyline, create a Polyline object and set its Points property to the vertices of a shape. To give the
polyline an outline, set its Stroke property to a Brush -derived object. The StrokeThickness property specifies
the thickness of the polyline outline. For more information about Brush objects, see Brushes.
IMPORTANT
If you set the Fill property of a Polyline to a Brush -derived object, the interior space of the polyline is painted,
even if the start point and end point do not intersect.
<Polyline Points="0,0 10,30, 15,0 18,60 23,30 35,30 40,0 43,60 48,30 100,30"
Stroke="Red" />
For more information about drawing a dashed polyline, see Draw dashed shapes.
The following XAML example shows a polyline that uses the default fill rule:
<Polyline Points="0 48, 0 144, 96 150, 100 0, 192 0, 192 96, 50 96, 48 192, 150 200 144 48"
Fill="Blue"
Stroke="Red"
StrokeThickness="3" />
In this example, the fill behavior of the polyline is determined using the EvenOdd fill rule.
The following XAML example shows a polyline that uses the Nonzero fill rule:
<Polyline Points="0 48, 0 144, 96 150, 100 0, 192 0, 192 96, 50 96, 48 192, 150 200 144 48"
Fill="Black"
FillRule="Nonzero"
Stroke="Yellow"
StrokeThickness="3" />
In this example, the fill behavior of the polyline is determined using the Nonzero fill rule.
Rectangle
3/18/2022 • 2 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) Rectangle class derives from the Shape class, and can be used to
draw rectangles and squares. For information on the properties that the Rectangle class inherits from the
Shape class, see .NET MAUI Shapes.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Create a Rectangle
To draw a rectangle, create a Rectangle object and sets its WidthRequest and HeightRequest properties. To paint
the inside of the rectangle, set its Fill property to a Brush -derived object. To give the rectangle an outline, set
its Stroke property to a Brush -derived object. The StrokeThickness property specifies the thickness of the
rectangle outline. For more information about Brush objects, see Brushes.
To give the rectangle rounded corners, set its RadiusX and RadiusY properties. These properties set the x-axis
and y-axis radii that's used to round the corners of the rectangle.
NOTE
There's also a RoundRectangle class, that has a CornerRadius BindableProperty , which can be used to draw
rectangles with rounded corners.
To draw a square, make the WidthRequest and HeightRequest properties of the Rectangle object equal.
The following XAML example shows how to draw a filled rectangle:
<Rectangle Fill="Red"
WidthRequest="150"
HeightRequest="50"
HorizontalOptions="Start" />
In this example, a red filled rectangle with dimensions 150x50 (device-independent units) is drawn:
The following XAML example shows how to draw a filled rectangle, with rounded corners:
<Rectangle Fill="Blue"
Stroke="Black"
StrokeThickness="3"
RadiusX="50"
RadiusY="10"
WidthRequest="200"
HeightRequest="100"
HorizontalOptions="Start" />
For information about drawing a dashed rectangle, see Draw dashed shapes.
WebView
3/18/2022 • 6 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) WebView displays remote web pages, local HTML files, and HTML
strings, in an app. The content displayed a WebView includes support for Cascading Style Sheets (CSS), and
JavaScript. By default, .NET MAUI projects include the platform permissions required for a WebView to display a
remote web page.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
The Source property can be set to an UrlWebViewSource object or a HtmlWebViewSource object, which both
derive from WebViewSource . A UrlWebViewSource is used for loading a web page specified with a URL, while a
HtmlWebViewSource object is used for loading a local HTML file, or local HTML.
WebView defines a Navigating event that's raised when page navigation starts, and a Navigated event that's
raised when page navigation completes. The WebNavigatingEventArgs object that accompanies the Navigating
event defines a Cancel property, of type bool , that can be used to cancel navigation. The
WebNavigatedEventArgs object that accompanies the Navigated event defines a Result property, of type
WebNavigationResult , that indicates the navigation result.
IMPORTANT
A WebView must specify its HeightRequest and WidthRequest properties when contained in a
HorizontalStackLayout , RelativeLayout , StackLayout , or VerticalStackLayout . If you fail to specify these
properties, the WebView will not render.
NOTE
Despite the Source property being of type WebViewSource , the property can be set to a string-based URI. This is
because .NET MAUI includes a type converter, and an implicit conversion operator, that converts the string-based URI to
a UrlWebViewSource object.
<key>NSAppTransportSecurity</key>
<dict>
<key>NSExceptionDomains</key>
<dict>
<key>mydomain.com</key>
<dict>
<key>NSIncludesSubdomains</key>
<true/>
<key>NSTemporaryExceptionAllowsInsecureHTTPLoads</key>
<true/>
<key>NSTemporaryExceptionMinimumTLSVersion</key>
<string>TLSv1.1</string>
</dict>
</dict>
</dict>
...
</key>
It's best practice to only enable specific domains to bypass ATS, allowing you to use trusted sites while
benefitting from additional security on untrusted domains.
The following Info.plist configuration shows how to disable ATS for an app:
<key>NSAppTransportSecurity</key>
<dict>
<key>NSAllowsArbitraryLoads</key>
<true/>
</dict>
...
</key>
IMPORTANT
If your app requires a connection to an insecure website, you should always enter the domain as an exception using the
NSExceptionDomains key instead of turning ATS off completely using the NSAllowsArbitraryLoads key.
Display local HTML
To display inline HTML, set the Source property to a HtmlWebViewSource object:
<WebView>
<WebView.Source>
<HtmlWebViewSource Html="<HTML><BODY><H1>.NET MAUI</H1><P>Welcome to
WebView.</P></BODY><HTML>" />
</WebView.Source>
</WebView>
In XAML, HTML strings can become unreadable due to escaping the < and > symbols. Therefore, for greater
readability the HTML can be inlined in a CDATA section:
<WebView>
<WebView.Source>
<HtmlWebViewSource>
<HtmlWebViewSource.Html>
<![CDATA[
<HTML>
<BODY>
<H1>.NET MAUI</H1>
<P>Welcome to WebView.</P>
</BODY>
</HTML>
]]>
</HtmlWebViewSource.Html>
</HtmlWebViewSource>
</WebView.Source>
</WebView>
The local HTML file can load Cascading Style Sheets (CSS), JavaScript, and images, provided that they've also
been added to your app project with the MauiAsset build action.
For more information about raw assets, see Raw assets.
Reload content
WebView has a Reload method that can be called to reload its source:
Perform navigation
WebView supports programmatic navigation with the GoBack and GoForward methods. These methods enable
navigation through the WebView page stack, and should only be called after inspecting the values of the
CanGoBack and CanGoForward properties:
// Go backwards, if allowed.
if (webView.CanGoBack)
{
webView.GoBack();
}
// Go forwards, if allowed.
if (webView.CanGoForward)
{
webView.GoForward();
}
When page navigation occurs in a WebView , either initiated programmatically or by the user, the following
events occur:
, which is raised when page navigation starts. The WebNavigatingEventArgs object that
Navigating
accompanies the Navigating event defines a Cancel property, of type bool , that can be used to cancel
navigation.
Navigated , which is raised when page navigation completes. The WebNavigatedEventArgs object that
accompanies the Navigated event defines a Result property, of type WebNavigationResult , that indicates
the navigation result.
Set cookies
Cookies can be set on a WebView , which are then sent with the web request to the specified URL. This is
accomplished by adding Cookie objects to a CookieContainer , which is then set as the value of the
WebView.Cookies bindable property. The following code shows an example of this:
using System.Net;
In this example, a single Cookie is added to the CookieContainer object, which is then set as the value of the
WebView.Cookies property. When the WebView sends a web request to the specified URL, the cookie is sent with
the request.
Invoke JavaScript
WebView includes the ability to invoke a JavaScript function from C#, and return any result to the calling C#
code. This is accomplished with the EvaluateJavaScriptAsync method, which is shown in the following example:
The WebView.EvaluateJavaScriptAsync method evaluates the JavaScript that's specified as the argument, and
returns any result as a string . In this example, the factorial JavaScript function is invoked, which returns the
factorial of number as a result. This JavaScript function is defined in the local HTML file that the WebView loads,
and is shown in the following example:
<html>
<body>
<script type="text/javascript">
function factorial(num) {
if (num === 0 || num === 1)
return 1;
for (var i = num - 1; i >= 1; i--) {
num *= i;
}
return num;
}
</script>
</body>
</html>
await Launcher.OpenAsync("https://docs.microsoft.com/dotnet/maui");
Button
3/18/2022 • 9 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) Button displays text and responds to a tap or click that directs the
app to carry out a task. A Button usually displays a short text string indicating a command, but it can also
display a bitmap image, or a combination of text and an image. When the Button is pressed with a finger or
clicked with a mouse it initiates that command.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
NOTE
While Button defines an ImageSource property, that allows you to display a image on the Button , this property is
intended to be used when displaying a small icon next to the Button text.
In addition, Button defines Clicked , Pressed , and Released events. The Clicked event is raised when a
Button tap with a finger or mouse pointer is released from the button's surface. The Pressed event is raised
when a finger presses on a Button , or a mouse button is pressed with the pointer positioned over the Button .
The Released event is raised when the finger or mouse button is released. Generally, a Clicked event is also
raised at the same time as the Released event, but if the finger or mouse pointer slides away from the surface
of the Button before being released, the Clicked event might not occur.
IMPORTANT
A Button must have its IsEnabled property set to true for it to respond to taps.
Create a Button
To create a button, create a Button object and handle it's Clicked event.
The following XAML example show how to create a Button :
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="ButtonDemos.BasicButtonClickPage"
Title="Basic Button Click">
<StackLayout>
<Button Text="Click to Rotate Text!"
VerticalOptions="Center"
HorizontalOptions="Center"
Clicked="OnButtonClicked" />
<Label x:Name="label"
Text="Click the Button above"
FontSize="Large"
VerticalOptions="Center"
HorizontalOptions="Center" />
</StackLayout>
</ContentPage>
The Text property specifies the text that appears in the Button . The Clicked event is set to an event handler
named OnButtonClicked . This handler is located in the code-behind file:
In this example, when the Button is tapped, the OnButtonClicked method executes. The sender argument is the
Button object responsible for this event. You can use this to access the Button object, or to distinguish between
multiple Button objects sharing the same Clicked event. The Clicked handler calls an animation function that
rotates the Label 360 degrees in 1000 milliseconds:
The equivalent C# code to create a Button is:
This approach is particularly suitable in connection with data-binding, and particularly when implementing the
Model-View-ViewModel (MVVM) pattern. In an MVVM application, the viewmodel defines properties of type
ICommand that are then connected to Button objects with data bindings. .NET MAUI also defines Command and
Command<T> classes that implement the ICommand interface and assist the viewmodel in defining properties of
type ICommand . For more information about commanding, see Commanding.
The following example shows a very simple viewmodel class that defines a property of type double named
Number , and two properties of type ICommand named MultiplyBy2Command and DivideBy2Command :
public class CommandDemoViewModel : INotifyPropertyChanged
{
double number = 1;
public CommandDemoViewModel()
{
MultiplyBy2Command = new Command(() => Number *= 2);
DivideBy2Command = new Command(() => Number /= 2);
}
In this example, the two ICommand properties are initialized in the class's constructor with two objects of type
Command . The Command constructors include a little function (called the execute constructor argument) that
either doubles or halves the value of the Number property.
The following XAML example consumes the CommandDemoViewModel class:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:ButtonDemos"
x:Class="ButtonDemos.BasicButtonCommandPage"
Title="Basic Button Command">
<ContentPage.BindingContext>
<local:CommandDemoViewModel />
</ContentPage.BindingContext>
<StackLayout>
<Label Text="{Binding Number, StringFormat='Value is now {0}'}"
FontSize="Large"
VerticalOptions="Center"
HorizontalOptions="Center" />
<Button Text="Multiply by 2"
VerticalOptions="Center"
HorizontalOptions="Center"
Command="{Binding MultiplyBy2Command}" />
<Button Text="Divide by 2"
VerticalOptions="Center"
HorizontalOptions="Center"
Command="{Binding DivideBy2Command}" />
</StackLayout>
</ContentPage>
In this example, the Label element and two Button objects contain bindings to the three properties in the
CommandDemoViewModel class. As the two Button objects are tapped, the commands are executed, and the
number changes value. The advantage of this approach over Clicked handlers is that all the logic involving the
functionality of this page is located in the viewmodel rather than the code-behind file, achieving a better
separation of the user interface from the business logic.
It's also possible for the Command objects to control the enabling and disabling of the Button objects. For
example, suppose you want to limit the range of number values between 210 and 2–10. You can add another
function to the constructor (called the canExecute argument) that returns true if the Button should be
enabled:
In this example, the calls to the ChangeCanExecute method of Command are required so that the Command method
can call the canExecute method and determine whether the Button should be disabled or not. With this code
change, as the number reaches the limit, the Button is disabled.
It's also possible for two or more Button elements to be bound to the same ICommand property. The Button
elements can be distinguished using the CommandParameter property of Button . In this case, you'll want to use
the generic Command<T> class. The CommandParameter object is then passed as an argument to the execute and
canExecute methods. For more information, see Commanding.
The code-behind file animates the Label when a Pressed event occurs, but suspends the rotation when a
Released event occurs:
public PressAndReleaseButtonPage()
{
InitializeComponent();
timer = Dispatcher.CreateTimer();
timer.Interval = TimeSpan.FromMilliseconds(16);
timer.Tick += (s, e) =>
{
label.Rotation = 360 * (stopwatch.Elapsed.TotalSeconds % 1);
};
}
The result is that the Label only rotates while a finger is in contact with the Button , and stops when the finger
is released.
In this example, the Pressed VisualState specifies that when the Button is pressed, its Scale property will be
changed from its default value of 1 to 0.8. The Normal VisualState specifies that when the Button is in a
normal state, its Scale property will be set to 1. Therefore, the overall effect is that when the Button is pressed,
it's rescaled to be slightly smaller, and when the Button is released, it's rescaled to its default size.
In XAML, you can create a Button and set the ContentLayout property by specifying only the enumeration
member, or the spacing, or both in any order separated by commas:
Disable a Button
Sometimes an app enters a state where a Button click is not a valid operation. In such cases, the Button can be
disabled by setting its IsEnabled property to false .
ImageButton
3/18/2022 • 5 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) ImageButton view combines the Button view and Image view to
create a button whose content is an image. When you press the ImageButton with a finger or click it with a
mouse, it directs the app to carry out a task. However, unlike the Button the ImageButton view has no concept
of text and text appearance.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
The Aspect property can be set to one of the members of the Aspect enumeration:
Fill- stretches the image to completely and exactly fill the ImageButton . This may result in the image being
distorted.
AspectFill - clips the image so that it fills the ImageButton while preserving the aspect ratio.
AspectFit - letterboxes the image (if necessary) so that the entire image fits into the ImageButton , with
blank space added to the top/bottom or sides depending on whether the image is wide or tall. This is the
default value of the Aspect enumeration.
Center - centers the image in the ImageButton while preserving the aspect ratio.
In addition, defines Clicked , Pressed , and Released events. The Clicked event is raised when an
ImageButton
ImageButton tap with a finger or mouse pointer is released from the button's surface. The Pressed event is
raised when a finger presses on an ImageButton , or a mouse button is pressed with the pointer positioned over
the ImageButton . The Released event is raised when the finger or mouse button is released. Generally, a
Clicked event is also raised at the same time as the Released event, but if the finger or mouse pointer slides
away from the surface of the ImageButton before being released, the Clicked event might not occur.
IMPORTANT
An ImageButton must have its IsEnabled property set to true for it to respond to taps.
Create an ImageButton
To create an image button, create an ImageButton object, set its Source property and handle it's Clicked event.
The following XAML example show how to create an ImageButton :
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="ControlGallery.Views.XAML.ImageButtonDemoPage"
Title="ImageButton Demo">
<StackLayout>
<ImageButton Source="image.png"
Clicked="OnImageButtonClicked"
HorizontalOptions="Center"
VerticalOptions="Center" />
</StackLayout>
</ContentPage>
The Source property specifies the image that appears in the ImageButton . The Clicked event is set to an event
handler named OnImageButtonClicked . This handler is located in the code-behind file:
public ImageButtonDemoPage()
{
InitializeComponent();
}
In this example, when the ImageButton is tapped, the OnImageButtonClicked method executes. The sender
argument is the ImageButton responsible for this event. You can use this to access the ImageButton object, or to
distinguish between multiple ImageButton objects sharing the same Clicked event. The Clicked handler
increments a counter and displays the counter value in a Label :
The equivalent C# code to create an ImageButton is:
Label label;
int clickTotal = 0;
...
<ImageButton Source="image.png"
...>
<VisualStateManager.VisualStateGroups>
<VisualStateGroup x:Name="CommonStates">
<VisualState x:Name="Normal">
<VisualState.Setters>
<Setter Property="Scale"
Value="1" />
</VisualState.Setters>
</VisualState>
<VisualState x:Name="Pressed">
<VisualState.Setters>
<Setter Property="Scale"
Value="0.8" />
</VisualState.Setters>
</VisualState>
</VisualStateGroup>
</VisualStateManager.VisualStateGroups>
</ImageButton>
In this example, the Pressed VisualState specifies that when the ImageButton is pressed, its Scale property
will be changed from its default value of 1 to 0.8. The Normal VisualState specifies that when the ImageButton
is in a normal state, its Scale property will be set to 1. Therefore, the overall effect is that when the
ImageButton is pressed, it's rescaled to be slightly smaller, and when the ImageButton is released, it's rescaled to
its default size.
Disable an ImageButton
Sometimes an app enters a state where an ImageButton click is not a valid operation. In those cases, the
ImageButton should be disabled by setting its IsEnabled property to false .
RadioButton
3/18/2022 • 10 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) RadioButton is a type of button that allows users to select one
option from a set. Each option is represented by one radio button, and you can only select one radio button in a
group. By default, each RadioButton displays text:
However, on some platforms a RadioButton can display a View , and on all platforms the appearance of each
RadioButton can be redefined with a ControlTemplate :
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
For more information about the GroupName attached property, see Group RadioButtons. For more information
about the SelectedValue attached property, see Respond to RadioButton state changes.
Create RadioButtons
The appearance of a RadioButton is defined by the type of data assigned to the RadioButton.Content property:
When the RadioButton.Content property is assigned a string , it will be displayed on each platform,
horizontally aligned next to the radio button circle.
When the RadioButton.Content is assigned a View , it will be displayed on supported platforms (iOS,
Windows), while unsupported platforms will fallback to a string representation of the View object (Android).
In both cases, the content is displayed horizontally aligned next to the radio button circle.
When a ControlTemplate is applied to a RadioButton , a View can be assigned to the RadioButton.Content
property on all platforms. For more information, see Redefine RadioButton appearance.
Display string-based content
A RadioButton displays text when the Content property is assigned a string :
<StackLayout>
<Label Text="What's your favorite animal?" />
<RadioButton Content="Cat" />
<RadioButton Content="Dog" />
<RadioButton Content="Elephant" />
<RadioButton Content="Monkey"
IsChecked="true" />
</StackLayout>
In this example, RadioButton objects are implicitly grouped inside the same parent container. This XAML results
in the appearance shown in the following screenshot:
In this example, RadioButton objects are implicitly grouped inside the same parent container. This XAML results
in the appearance shown in the following screenshot:
On Android, RadioButton objects will display a string-based representation of the View object that's been set as
content.
NOTE
When a ControlTemplate is applied to a RadioButton , a View can be assigned to the RadioButton.Content
property on all platforms. For more information, see Redefine RadioButton appearance.
In this example, each RadioButton has an Image as its content, while also defining a string-based value. This
enables the value of the checked radio button to be easily identified.
Group RadioButtons
Radio buttons work in groups, and there are three approaches to grouping radio buttons:
Place them inside the same parent container. This is known as implicit grouping.
Set the GroupName property on each radio button in the group to the same value. This is known as explicit
grouping.
Set the RadioButtonGroup.GroupName attached property on a parent container, which in turn sets the
GroupName property of any RadioButton objects in the container. This is also known as explicit grouping.
IMPORTANT
RadioButton objects don't have to belong to the same parent to be grouped. They are mutually exclusive provided that
they share a group name.
In this example, each RadioButton is mutually exclusive because it shares the same GroupName value.
Explicit grouping with the RadioButtonGroup.GroupName attached property
The RadioButtonGroup class defines a GroupName attached property, of type string , which can be set on a
Layout<View> object. This enables any layout to be turned into a radio button group:
<StackLayout RadioButtonGroup.GroupName="colors">
<Label Text="What's your favorite color?" />
<RadioButton Content="Red" />
<RadioButton Content="Green" />
<RadioButton Content="Blue" />
<RadioButton Content="Other" />
</StackLayout>
In this example, each RadioButton in the StackLayout will have its GroupName property set to fruits , and will
be mutually exclusive.
NOTE
When an ILayout object that sets the RadioButtonGroup.GroupName attached property contains a RadioButton that
sets its GroupName property, the value of the RadioButton.GroupName property will take precedence.
<RadioButton Content="Red"
GroupName="colors"
CheckedChanged="OnColorsRadioButtonCheckedChanged" />
The sender argument is the RadioButton responsible for this event. You can use this to access the RadioButton
object, or to distinguish between multiple RadioButton objects sharing the same CheckedChanged event handler.
Respond to a property change
The RadioButtonGroup class defines a SelectedValue attached property, of type object , which can be set on an
ILayout object. This attached property represents the value of the checked RadioButton within a group defined
on a layout.
When the IsChecked property changes, either through user or programmatic manipulation, the
RadioButtonGroup.SelectedValue attached property also changes. Therefore, the RadioButtonGroup.SelectedValue
attached property can be data bound to a property that stores the user's selection:
<StackLayout RadioButtonGroup.GroupName="{Binding GroupName}"
RadioButtonGroup.SelectedValue="{Binding Selection}">
<Label Text="What's your favorite animal?" />
<RadioButton Content="Cat"
Value="Cat" />
<RadioButton Content="Dog"
Value="Dog" />
<RadioButton Content="Elephant"
Value="Elephant" />
<RadioButton Content="Monkey"
Value="Monkey"/>
<Label x:Name="animalLabel">
<Label.FormattedText>
<FormattedString>
<Span Text="You have chosen:" />
<Span Text="{Binding Selection}" />
</FormattedString>
</Label.FormattedText>
</Label>
</StackLayout>
In this example, the value of the RadioButtonGroup.GroupName attached property is set by the GroupName property
on the binding context. Similarly, the value of the RadioButtonGroup.SelectedValue attached property is set by
the Selection property on the binding context. In addition, the Selection property is updated to the Value
property of the checked RadioButton .
In this example, the implicit Style targets RadioButton objects. The Checked VisualState specifies that when a
RadioButton is checked, its TextColor property will be set to green with an Opacity value of 1. The Unchecked
VisualState specifies that when a RadioButton is in a unchecked state, its TextColor property will be set to red
with an Opacity value of 0.5. Therefore, the overall effect is that when a RadioButton is unchecked it's red and
partially transparent, and is green without transparency when it's checked:
The following XAML shows a ControlTemplate that can be used to redefine the visual structure of RadioButton
objects:
<ContentPage ...>
<ContentPage.Resources>
<ContentPage.Resources>
<ControlTemplate x:Key="RadioButtonTemplate">
<Frame BorderColor="#F3F2F1"
BackgroundColor="#F3F2F1"
HasShadow="False"
HeightRequest="100"
WidthRequest="100"
HorizontalOptions="Start"
VerticalOptions="Start"
Padding="0">
<VisualStateManager.VisualStateGroups>
<VisualStateGroupList>
<VisualStateGroup x:Name="CheckedStates">
<VisualState x:Name="Checked">
<VisualState.Setters>
<Setter Property="BorderColor"
Value="#FF3300" />
<Setter TargetName="check"
Property="Opacity"
Value="1" />
</VisualState.Setters>
</VisualState>
<VisualState x:Name="Unchecked">
<VisualState.Setters>
<Setter Property="BackgroundColor"
Value="#F3F2F1" />
<Setter Property="BorderColor"
Value="#F3F2F1" />
<Setter TargetName="check"
Property="Opacity"
Value="0" />
</VisualState.Setters>
</VisualState>
</VisualStateGroup>
</VisualStateGroupList>
</VisualStateManager.VisualStateGroups>
<Grid Margin="4"
WidthRequest="100">
<Grid WidthRequest="18"
HeightRequest="18"
HorizontalOptions="End"
VerticalOptions="Start">
<Ellipse Stroke="Blue"
Fill="White"
WidthRequest="16"
HeightRequest="16"
HorizontalOptions="Center"
VerticalOptions="Center" />
<Ellipse x:Name="check"
Fill="Blue"
WidthRequest="8"
HeightRequest="8"
HorizontalOptions="Center"
VerticalOptions="Center" />
</Grid>
<ContentPresenter />
</Grid>
</Frame>
</ControlTemplate>
<Style TargetType="RadioButton">
<Setter Property="ControlTemplate"
Value="{StaticResource RadioButtonTemplate}" />
</Style>
</ContentPage.Resources>
<!-- Page content -->
</ContentPage>
In this example, the root element of the ControlTemplate is a Frame object that defines Checked and Unchecked
visual states. The Frame object uses a combination of Grid , Ellipse , and ContentPresenter objects to define
the visual structure of a RadioButton . The example also includes an implicit style that will assign the
RadioButtonTemplate to the ControlTemplate property of any RadioButton objects on the page.
NOTE
The ContentPresenter object marks the location in the visual structure where RadioButton content will be displayed.
The following XAML shows RadioButton objects that consume the ControlTemplate via the implicit style:
<StackLayout>
<Label Text="What's your favorite animal?" />
<StackLayout RadioButtonGroup.GroupName="animals"
Orientation="Horizontal">
<RadioButton Value="Cat">
<RadioButton.Content>
<StackLayout>
<Image Source="cat.png"
HorizontalOptions="Center"
VerticalOptions="Center" />
<Label Text="Cat"
HorizontalOptions="Center"
VerticalOptions="End" />
</StackLayout>
</RadioButton.Content>
</RadioButton>
<RadioButton Value="Dog">
<RadioButton.Content>
<StackLayout>
<Image Source="dog.png"
HorizontalOptions="Center"
VerticalOptions="Center" />
<Label Text="Dog"
HorizontalOptions="Center"
VerticalOptions="End" />
</StackLayout>
</RadioButton.Content>
</RadioButton>
<RadioButton Value="Elephant">
<RadioButton.Content>
<StackLayout>
<Image Source="elephant.png"
HorizontalOptions="Center"
VerticalOptions="Center" />
<Label Text="Elephant"
HorizontalOptions="Center"
VerticalOptions="End" />
</StackLayout>
</RadioButton.Content>
</RadioButton>
<RadioButton Value="Monkey">
<RadioButton.Content>
<StackLayout>
<Image Source="monkey.png"
HorizontalOptions="Center"
VerticalOptions="Center" />
<Label Text="Monkey"
HorizontalOptions="Center"
VerticalOptions="End" />
</StackLayout>
</RadioButton.Content>
</RadioButton>
</StackLayout>
</StackLayout>
In this example, the visual structure defined for each RadioButton is replaced with the visual structure defined in
the ControlTemplate , and so at runtime the objects in the ControlTemplate become part of the visual tree for
each RadioButton . In addition, the content for each RadioButton is substituted into the ContentPresenter
defined in the control template. This results in the following RadioButton appearance:
Disable a RadioButton
Sometimes an app enters a state where a RadioButton being checked is not a valid operation. In such cases, the
RadioButton can be disabled by setting its IsEnabled property to false .
RefreshView
3/18/2022 • 2 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) RefreshView is a container control that provides pull to refresh
functionality for scrollable content. Therefore, the child of a RefreshView must be a scrollable control, such as
ScrollView , CollectionView , or ListView .
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
Create a RefreshView
To add a RefreshView to a page, create a RefreshView object and set its IsRefreshing and Command properties.
Then set its child to a scrollable control.
The following example shows how to instantiate a RefreshView in XAML:
In this example, the RefreshView provides pull to refresh functionality to a ScrollView whose child is a
FlexLayout . The FlexLayout uses a bindable layout to generate its content by binding to a collection of items,
and sets the appearance of each item with a DataTemplate .
The value of the RefreshView.IsRefreshing property indicates the current state of the RefreshView . When a
refresh is triggered by the user, this property will automatically transition to true . Once the refresh completes,
you should reset the property to false .
When the user initiates a refresh, the ICommand defined by the Command property is executed, which should
refresh the items being displayed. A refresh visualization is shown while the refresh occurs, which consists of an
animated progress circle. The following screenshot shows the progress circle on iOS:
NOTE
Manually setting the IsRefreshing property to true will trigger the refresh visualization, and will execute the
ICommand defined by the Command property.
RefreshView appearance
In addition to the properties that RefreshView inherits from the VisualElement class, RefreshView also defines
the RefreshColor property. This property can be set to define the color of the progress circle that appears
during the refresh:
<RefreshView RefreshColor="Teal"
... />
The following Android screenshot shows a RefreshView with the RefreshColor property:
In addition, the BackgroundColor property can be set to a Color that represents the background color of the
progress circle.
NOTE
On iOS, the BackgroundColor property sets the background color of the UIView that contains the progress circle.
Disable a RefreshView
An app may enter a state where pull to refresh is not a valid operation. In such cases, the RefreshView can be
disabled by setting its IsEnabled property to false . This will prevent users from being able to trigger pull to
refresh.
Alternatively, when defining the Command property, the CanExecute delegate of the ICommand can be specified to
enable or disable the command.
SearchBar
3/18/2022 • 4 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) SearchBar is a user input control used to initiating a search. The
SearchBar control supports placeholder text, query input, search execution, and cancellation. The following iOS
screenshot shows a SearchBar query with results displayed in a ListView :
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
In addition, SearchBar defines a SearchButtonPressed event, which is raised when the search button is clicked,
or the enter key is pressed.
NOTE
SearchBar derives from the InputView class, from which it inherits additional properties and events.
Create a SearchBar
To create a search bar, create a SearchBar object and set its Placeholder property to text that instructs the user
to enter a search term.
The following XAML example shows how to create a SearchBar :
The following XAML example shows an event handler attached to the TextChanged event and uses a ListView
to display search results:
In this example, the TextChanged event is set to an event handler named OnTextChanged . This handler is located
in the code-behind file:
In this example, a DataService class with a GetSearchResults method is used to returnitems that match a query.
The SearchBar control's Text property value is passed to the GetSearchResults method and the result is used
to update the ListView control's ItemsSource property. The overall effect is that search results are displayed in
the ListView .
NOTE
The viewmodel assumes the existence of a DataService class capable of performing searches.
<ContentPage ...>
<ContentPage.BindingContext>
<viewmodels:SearchViewModel />
</ContentPage.BindingContext>
<StackLayout>
<SearchBar x:Name="searchBar"
SearchCommand="{Binding PerformSearch}"
SearchCommandParameter="{Binding Text, Source={x:Reference searchBar}}"/>
<ListView x:Name="searchResults"
ItemsSource="{Binding SearchResults}" />
</StackLayout>
</ContentPage>
In this example, the BindingContext is set to an instance of the SearchViewModel class. The
SearchBar.SearchCommand property binds to PerformSearch viewmodel property, and the
SearchCommandParameter property binds to the SearchBar.Text property. Similarly, the ListView.ItemsSource
property is bound to the SearchResults property of the viewmodel.
SwipeView
3/18/2022 • 10 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) SwipeView is a container control that wraps around an item of
content, and provides context menu items that are revealed by a swipe gesture:
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Create a SwipeView
A SwipeView must define the content that the SwipeView wraps around, and the swipe items that are revealed
by the swipe gesture. The swipe items are one or more SwipeItem objects that are placed in one of the four
SwipeView directional collections - LeftItems , RightItems , TopItems , or BottomItems .
<SwipeView>
<SwipeView.LeftItems>
<SwipeItems>
<SwipeItem Text="Favorite"
IconImageSource="favorite.png"
BackgroundColor="LightGreen"
Invoked="OnFavoriteSwipeItemInvoked" />
<SwipeItem Text="Delete"
IconImageSource="delete.png"
BackgroundColor="LightPink"
Invoked="OnDeleteSwipeItemInvoked" />
</SwipeItems>
</SwipeView.LeftItems>
<!-- Content -->
<Grid HeightRequest="60"
WidthRequest="300"
BackgroundColor="LightGray">
<Label Text="Swipe right"
HorizontalOptions="Center"
VerticalOptions="Center" />
</Grid>
</SwipeView>
// SwipeView content
Grid grid = new Grid
{
HeightRequest = 60,
WidthRequest = 300,
BackgroundColor = Colors.LightGray
};
grid.Add(new Label
{
Text = "Swipe right",
HorizontalOptions = LayoutOptions.Center,
VerticalOptions = LayoutOptions.Center
});
The swipe items are used to perform actions on the SwipeView content, and are revealed when the control is
swiped from the left side:
By default, a swipe item is executed when it is tapped by the user. However, this behavior can be changed. For
more information, see Swipe mode.
Once a swipe item has been executed the swipe items are hidden and the SwipeView content is re-displayed.
However, this behavior can be changed. For more information, see Swipe behavior.
NOTE
Swipe content and swipe items can be placed inline, or defined as resources.
Swipe items
The LeftItems , RightItems , TopItems , and BottomItems collections are all of type SwipeItems . The SwipeItems
class defines the following properties:
Mode , of type SwipeMode , which indicates the effect of a swipe interaction. For more information about swipe
mode, see Swipe mode.
SwipeBehaviorOnInvoked , of type SwipeBehaviorOnInvoked , which indicates how a SwipeView behaves after a
swipe item is invoked. For more information about swipe behavior, see Swipe behavior.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
Each swipe item is defined as a SwipeItem object that's placed into one of the four SwipeItems directional
collections. The SwipeItem class derives from the MenuItem class, and adds the following members:
A BackgroundColor property, of type Color , that defines the background color of the swipe item. This
property is backed by a bindable property.
An Invoked event, which is raised when the swipe item is executed.
IMPORTANT
The MenuItem class defines several properties, including Command , CommandParameter , IconImageSource , and Text .
These properties can be set on a SwipeItem object to define its appearance, and to define an ICommand that executes
when the swipe item is invoked.
The following example shows two SwipeItem objects in the LeftItems collection of a SwipeView :
<SwipeView>
<SwipeView.LeftItems>
<SwipeItems>
<SwipeItem Text="Favorite"
IconImageSource="favorite.png"
BackgroundColor="LightGreen"
Invoked="OnFavoriteSwipeItemInvoked" />
<SwipeItem Text="Delete"
IconImageSource="delete.png"
BackgroundColor="LightPink"
Invoked="OnDeleteSwipeItemInvoked" />
</SwipeItems>
</SwipeView.LeftItems>
<!-- Content -->
</SwipeView>
The appearance of each SwipeItem is defined by a combination of the Text , IconImageSource , and
BackgroundColor properties:
When a SwipeItem is tapped, its Invoked event fires and is handled by its registered event handler. In addition,
the MenuItem.Clicked event fires. Alternatively, the Command property can be set to an ICommand
implementation that will be executed when the SwipeItem is invoked.
NOTE
When the appearance of a SwipeItem is defined only using the Text or IconImageSource properties, the content is
always centered.
In addition to defining swipe items as SwipeItem objects, it's also possible to define custom swipe item views.
For more information, see Custom swipe items.
Swipe direction
SwipeView supports four different swipe directions, with the swipe direction being defined by the directional
SwipeItems collection the SwipeItem objects are added to. Each swipe direction can hold its own swipe items.
For example, the following example shows a SwipeView whose swipe items depend on the swipe direction:
<SwipeView>
<SwipeView.LeftItems>
<SwipeItems>
<SwipeItem Text="Delete"
IconImageSource="delete.png"
BackgroundColor="LightPink"
Command="{Binding DeleteCommand}" />
</SwipeItems>
</SwipeView.LeftItems>
<SwipeView.RightItems>
<SwipeItems>
<SwipeItem Text="Favorite"
IconImageSource="favorite.png"
BackgroundColor="LightGreen"
Command="{Binding FavoriteCommand}" />
<SwipeItem Text="Share"
IconImageSource="share.png"
BackgroundColor="LightYellow"
Command="{Binding ShareCommand}" />
</SwipeItems>
</SwipeView.RightItems>
<!-- Content -->
</SwipeView>
In this example, the SwipeView content can be swiped right or left. Swiping to the right will show the Delete
swipe item, while swiping to the left will show the Favorite and Share swipe items.
WARNING
Only one instance of a directional SwipeItems collection can be set at a time on a SwipeView . Therefore, you cannot
have two LeftItems definitions on a SwipeView .
The SwipeStarted , SwipeChanging , and SwipeEnded events report the swipe direction via the SwipeDirection
property in the event arguments. This property is of type SwipeDirection , which is an enumeration consisting of
four members:
Right indicates that a right swipe occurred.
Left indicates that a left swipe occurred.
Up indicates that an upwards swipe occurred.
Down indicates that a downwards swipe occurred.
Swipe threshold
SwipeView includes a Threshold property, of type double , which represents the number of device-independent
units that trigger a swipe gesture to fully reveal swipe items.
The following example shows a SwipeView that sets the Threshold property:
<SwipeView Threshold="200">
<SwipeView.LeftItems>
<SwipeItems>
<SwipeItem Text="Favorite"
IconImageSource="favorite.png"
BackgroundColor="LightGreen" />
</SwipeItems>
</SwipeView.LeftItems>
<!-- Content -->
</SwipeView>
In this example, the SwipeView must be swiped for 200 device-independent units before the SwipeItem is fully
revealed.
Swipe mode
The SwipeItems class has a Mode property, which indicates the effect of a swipe interaction. This property
should be set to one of the SwipeMode enumeration members:
Reveal indicates that a swipe reveals the swipe items. This is the default value of the SwipeItems.Mode
property.
Execute indicates that a swipe executes the swipe items.
In reveal mode, the user swipes a SwipeView to open a menu consisting of one or more swipe items, and must
explicitly tap a swipe item to execute it. After the swipe item has been executed the swipe items are closed and
the SwipeView content is re-displayed. In execute mode, the user swipes a SwipeView to open a menu consisting
of one more swipe items, which are then automatically executed. Following execution, the swipe items are
closed and the SwipeView content is re-displayed.
The following example shows a SwipeView configured to use execute mode:
<SwipeView>
<SwipeView.LeftItems>
<SwipeItems Mode="Execute">
<SwipeItem Text="Delete"
IconImageSource="delete.png"
BackgroundColor="LightPink"
Command="{Binding DeleteCommand}" />
</SwipeItems>
</SwipeView.LeftItems>
<!-- Content -->
</SwipeView>
In this example, the SwipeView content can be swiped right to reveal the swipe item, which is executed
immediately. Following execution, the SwipeView content is re-displayed.
Swipe behavior
The SwipeItems class has a SwipeBehaviorOnInvoked property, which indicates how a SwipeView behaves after a
swipe item is invoked. This property should be set to one of the SwipeBehaviorOnInvoked enumeration members:
Auto indicates that in reveal mode the SwipeView closes after a swipe item is invoked, and in execute mode
the SwipeView remains open after a swipe item is invoked. This is the default value of the
SwipeItems.SwipeBehaviorOnInvoked property.
Close indicates that the SwipeView closes after a swipe item is invoked.
RemainOpen indicates that the SwipeView remains open after a swipe item is invoked.
The following example shows a SwipeView configured to remain open after a swipe item is invoked:
<SwipeView>
<SwipeView.LeftItems>
<SwipeItems SwipeBehaviorOnInvoked="RemainOpen">
<SwipeItem Text="Favorite"
IconImageSource="favorite.png"
BackgroundColor="LightGreen"
Invoked="OnFavoriteSwipeItemInvoked" />
<SwipeItem Text="Delete"
IconImageSource="delete.png"
BackgroundColor="LightPink"
Invoked="OnDeleteSwipeItemInvoked" />
</SwipeItems>
</SwipeView.LeftItems>
<!-- Content -->
</SwipeView>
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
The SwipeItemView class also defines an Invoked event that's raised when the item is tapped, after the Command
is executed.
The following example shows a SwipeItemView object in the LeftItems collection of a SwipeView :
<SwipeView>
<SwipeView.LeftItems>
<SwipeItems>
<SwipeItemView Command="{Binding CheckAnswerCommand}"
CommandParameter="{Binding Source={x:Reference resultEntry}, Path=Text}">
<StackLayout Margin="10"
WidthRequest="300">
<Entry x:Name="resultEntry"
Placeholder="Enter answer"
HorizontalOptions="CenterAndExpand" />
<Label Text="Check"
FontAttributes="Bold"
HorizontalOptions="Center" />
</StackLayout>
</SwipeItemView>
</SwipeItems>
</SwipeView.LeftItems>
<!-- Content -->
</SwipeView>
In this example, the SwipeItemView comprises a StackLayout containing an Entry and a Label . After the user
enters input into the Entry , the rest of the SwipeViewItem can be tapped which executes the ICommand defined
by the SwipeItemView.Command property.
swipeView.Open(OpenSwipeItem.LeftItems);
swipeView.Close();
NOTE
The Close method also accepts an optional bool argument that defines whether the SwipeView will be animated
when it closes.
Disable a SwipeView
An app may enter a state where swiping an item of content is not a valid operation. In such cases, the SwipeView
can be disabled by setting its IsEnabled property to false . This will prevent users from being able to swipe
content to reveal swipe items.
In addition, when defining the Command property of a SwipeItem or SwipeItemView , the CanExecute delegate of
the ICommand can be specified to enable or disable the swipe item.
CheckBox
3/18/2022 • 3 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) CheckBox is a type of button that can either be checked or empty.
When a checkbox is checked, it's considered to be on. When a checkbox is empty, it's considered to be off.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Create a CheckBox
The following example shows how to instantiate a CheckBox in XAML:
<CheckBox />
By default, the CheckBox is empty. The CheckBox can be checked by user manipulation, or by setting the
IsChecked property to true :
The code-behind file contains the handler for the CheckedChanged event:
The sender argument is the CheckBox responsible for this event. You can use this to access the CheckBox
object, or to distinguish between multiple CheckBox objects sharing the same CheckedChanged event handler.
Alternatively, an event handler for the CheckedChanged event can be registered in code:
In this example, the Label uses a binding expression in a data trigger to monitor the IsChecked property of the
CheckBox . When this property becomes true , the FontAttributes and FontSize properties of the Label
change. When the IsChecked property returns to false , the FontAttributes and FontSize properties of the
Label are reset to their initial state.
The following screenshot shows the Label formatting when the CheckBox is checked:
Disable a Checkbox
Sometimes an application enters a state where a CheckBox being checked is not a valid operation. In such cases,
the CheckBox can be disabled by setting its IsEnabled property to false .
CheckBox appearance
In addition to the properties that CheckBox inherits from the View class, CheckBox also defines a Color
property that sets its color to a Color :
The following screenshot shows a series of checked CheckBox objects, where each object has its Color
property set to a different Color :
<CheckBox ...>
<VisualStateManager.VisualStateGroups>
<VisualStateGroup x:Name="CommonStates">
<VisualState x:Name="Normal">
<VisualState.Setters>
<Setter Property="Color"
Value="Red" />
</VisualState.Setters>
</VisualState>
<VisualState x:Name="IsChecked">
<VisualState.Setters>
<Setter Property="Color"
Value="Green" />
</VisualState.Setters>
</VisualState>
</VisualStateGroup>
</VisualStateManager.VisualStateGroups>
</CheckBox>
In this example, the IsChecked VisualState specifies that when the CheckBox is checked, its Color property
will be set to green. The Normal VisualState specifies that when the CheckBox is in a normal state, its Color
property will be set to red. Therefore, the overall effect is that the CheckBox is red when it's empty, and green
when it's checked.
DatePicker
3/18/2022 • 3 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) DatePicker invokes the platform's date-picker control and allows
you to select a date.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
All eight properties are backed by BindableProperty objects, which means that they can be styled, and the
properties can be targets of data bindings. The Date property has a default binding mode of
BindingMode.TwoWay , which means that it can be a target of a data binding in an application that uses the Model-
View-ViewModel (MVVM) pattern.
WARNING
When setting MinimumDate and MaximumDate , make sure that MinimumDate is always less than or equal to
MaximumDate . Otherwise, DatePicker will raise an exception.
The DatePicker ensures that Date is between MinimumDate and MaximumDate , inclusive. If MinimumDate or
MaximumDate is set so that Date is not between them, DatePicker will adjust the value of Date .
The DatePicker fires a DateSelected event when the user selects a date.
Create a DatePicker
When a DateTimevalue is specified in XAML, the XAML parser uses the DateTime.Parse method with a
CultureInfo.InvariantCulture argument to convert the string to a DateTime value. The dates must be specified
in a precise format: two-digit months, two-digit days, and four-digit years separated by slashes:
<DatePicker MinimumDate="01/01/2022"
MaximumDate="12/31/2022"
Date="06/21/2022" />
In this example, all three properties are initialized to the corresponding properties in the viewmodel. Because
the Date property has a binding mode of TwoWay , any new date that the user selects is automatically reflected
in the viewmodel.
If the DatePickerdoes not contain a binding on its Date property, your app should attach a handler to the
DateSelected event to be informed when the user selects a new date.
In code, you can initialize the MinimumDate , MaximumDate , and Date properties to values of type DateTime :
<DatePicker ···
HorizontalOptions="Center" />
However, this is not recommended. Depending on the setting of the Format property, selected dates might
require different display widths. For example, the "D" format string causes DateTime to display dates in a long
format, and "Wednesday, September 12, 2018" requires a greater display width than "Friday, May 4, 2018".
Depending on the platform, this difference might cause the DateTime view to change width in layout, or for the
display to be truncated.
TIP
It's best to use the default HorizontalOptions setting of Fill with DatePicker , and not to use a width of Auto
when putting DatePicker in a Grid cell.
Slider
3/18/2022 • 6 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) Slider is a horizontal bar that you can manipulate to select a
double value from a continuous range.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
These properties are backed by BindableProperty objects. The Value property has a default binding mode of
BindingMode.TwoWay , which means that it's suitable as a binding source in an application that uses the Model-
View-ViewModel (MVVM) pattern.
NOTE
The ThumbColor and ThumbImageSource properties are mutually exclusive. If both properties are set, the
ThumbImageSource property will take precedence.
The Slider coerces the Value property so that it is between Minimum and Maximum , inclusive. If the Minimum
property is set to a value greater than the Value property, the Slider sets the Value property to Minimum .
Similarly, if Maximum is set to a value less than Value , then Slider sets the Value property to Maximum .
Internally, the Slider ensures that Minimum is less than Maximum . If Minimum or Maximum are ever set so that
Minimum is not less than Maximum , an exception is raised. For more information on setting the Minimum and
Maximum properties, see Precautions.
Slider defines a ValueChanged event that's raised when the Value changes, either through user manipulation
of the Slider or when the program sets the Value property directly. A ValueChanged event is also raised when
the Value property is coerced as described in the previous paragraph. The ValueChangedEventArgs object that
accompanies the ValueChanged event has OldValue and NewValue properties, of type double . At the time the
event is raised, the value of NewValue is the same as the Value property of the Slider object.
Slider also defines DragStarted and DragCompleted events, that are raised at the beginning and end of the
drag action. Unlike the ValueChanged event, the DragStarted and DragCompleted events are only raised through
user manipulation of the Slider . When the DragStarted event fires, the DragStartedCommand , of type ICommand ,
is executed. Similarly, when the DragCompleted event fires, the DragCompletedCommand , of type ICommand , is
executed.
WARNING
Do not use unconstrained horizontal layout options of Center , Start , or End with Slider . Keep the default
HorizontalOptions setting of Fill , and don't use a width of Auto when putting Slider in a Grid layout.
Create a Slider
The following example shows how to create a Slider , with two Label objects:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="SliderDemos.BasicSliderXamlPage"
Title="Basic Slider XAML"
Padding="10, 0">
<StackLayout>
<Label x:Name="rotatingLabel"
Text="ROTATING TEXT"
FontSize="Large"
HorizontalOptions="Center"
VerticalOptions="Center" />
<Slider Maximum="360"
ValueChanged="OnSliderValueChanged" />
<Label x:Name="displayLabel"
Text="(uninitialized)"
HorizontalOptions="Center"
VerticalOptions="Center" />
</StackLayout>
</ContentPage>
In this example, the Slider is initialized to have a Maximum property of 360. The second Label displays the text
"(uninitialized)" until the Slider is manipulated, which causes the first ValueChanged event to be raised.
The code-behind file contains the handler for the ValueChanged event:
The ValueChanged handler of the Slider uses the Value property of the slider object to set the Rotation
property of the first Label and uses the String.Format method with the NewValue property of the event
arguments to set the Text property of the second Label :
It's also possible for the event handler to obtain the Slider that is firing the event through the sender
argument. The Value property contains the current value:
If the Slider object were given a name in the XAML file with an x:Name attribute (for example, "slider"), then
the event handler could reference that object directly:
In this example, the Rotation property of the first Label is bound to the Value property of the Slider , as is
the Text property of the second Label with a StringFormat specification. When the page first appears, the
second Label displays the text string with the value. To display text without data binding, you'd need to
specifically initialize the Text property of the Label or simulate a firing of the ValueChanged event by calling
the event handler from the class constructor.
Precautions
The value of the Minimum property must always be less than the value of the Maximum property. The following
example causes the Slider to raise an exception:
// Throws an exception!
Slider slider = new Slider
{
Minimum = 10,
Maximum = 20
};
The C# compiler generates code that sets these two properties in sequence, and when the Minimum property is
set to 10, it is greater than the default Maximum value of 1. You can avoid the exception in this case by setting the
Maximum property first:
In this example, setting Maximum to 20 is not a problem because it is greater than the default Minimum value of 0.
When Minimum is set, the value is less than the Maximum value of 20.
The same problem exists in XAML. The properties must be set in an order that ensures that Maximum is always
greater than Minimum :
<Slider Maximum="20"
Minimum="10" ... />
You can set the Minimum and Maximum values to negative numbers, but only in an order where Minimum is
always less than Maximum :
<Slider Minimum="-20"
Maximum="-10" ... />
The Value property is always greater than or equal to the Minimum value and less than or equal to Maximum . If
Value is set to a value outside that range, the value will be coerced to lie within the range, but no exception is
raised. For example, the following example won't raise an exception:
<Slider ValueChanged="OnSliderValueChanged"
Maximum="20"
Minimum="10" />
When Minimum is set to 10, Value is also set to 10, and the ValueChanged event is raised. This might occur
before the rest of the page has been constructed, and the handler might attempt to reference other elements on
the page that have not yet been created. You might want to add some code to the ValueChanged handler that
checks for null values of other elements on the page. Or, you can set the ValueChanged event handler after the
Slider values have been initialized.
Stepper
3/18/2022 • 6 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) Stepper enables a numeric value to be selected from a range of
values. It consists of two buttons labeled with minus and plus signs. These buttons can be manipulated by the
user to incrementally select a double value from a range of values.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
All of these properties are backed by BindableProperty objects. The Value property has a default binding mode
of BindingMode.TwoWay , which means that it's suitable as a binding source in an application that uses the Model-
View-ViewModel (MVVM) pattern.
The Stepper coerces the Value property so that it is between Minimum and Maximum , inclusive. If the Minimum
property is set to a value greater than the Value property, the Stepper sets the Value property to Minimum .
Similarly, if Maximum is set to a value less than Value , then Stepper sets the Value property to Maximum .
Internally, the Stepper ensures that Minimum is less than Maximum . If Minimum or Maximum are ever set so that
Minimum is not less than Maximum , an exception is raised. For more information on setting the Minimum and
Maximum properties, see Precautions.
Stepper defines a ValueChanged event that's raised when the Value changes, either through user manipulation
of the Stepper or when the application sets the Value property directly. A ValueChanged event is also raised
when the Value property is coerced as previously described. The ValueChangedEventArgs object that
accompanies the ValueChanged event has OldValue and NewValue , of type double . At the time the event is
raised, the value of NewValue is the same as the Value property of the Stepper object.
Create a Stepper
The following example shows how to create a Stepper , with two Label objects:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="StepperDemo.BasicStepperXAMLPage"
Title="Basic Stepper XAML">
<StackLayout Margin="20">
<Label x:Name="_rotatingLabel"
Text="ROTATING TEXT"
FontSize="Large"
HorizontalOptions="Center"
VerticalOptions="Center" />
<Stepper Maximum="360"
Increment="30"
HorizontalOptions="Center"
ValueChanged="OnStepperValueChanged" />
<Label x:Name="_displayLabel"
Text="(uninitialized)"
HorizontalOptions="Center"
VerticalOptions="Center" />
</StackLayout>
</ContentPage>
In this example, the Stepper is initialized to have a Maximum property of 360, and an Increment property of 30.
Manipulating the Stepper changes the selected value incrementally between Minimum to Maximum based on the
value of the Increment property. The second Label displays the text "(uninitialized)" until the Stepper is
manipulated, which causes the first ValueChanged event to be raised.
The code-behind file contains the handler for the ValueChanged event:
The ValueChanged handler of the Stepper uses the Value property of the stepper object to set the Rotation
property of the first Label and uses the string.Format method with the NewValue property of the event
arguments to set the Text property of the second Label :
It's also possible for the event handler to obtain the Stepper that is firing the event through the sender
argument. The Value property contains the current value:
If the Stepper object were given a name in the XAML file with an x:Name attribute (for example, "stepper"), then
the event handler could reference that object directly:
In this example, the Rotation property of the first Label is bound to the Value property of the Stepper , as is
the Text property of the second Label with a StringFormat specification. When the page first appears, the
second Label displays the text string with the value. To display text without data binding, you'd need to
specifically initialize the Text property of the Label or simulate a firing of the ValueChanged event by calling
the event handler from the class constructor.
Precautions
The value of the Minimum property must always be less than the value of the Maximum property. The following
code example causes the Stepper to raise an exception:
// Throws an exception!
Stepper stepper = new Stepper
{
Minimum = 180,
Maximum = 360
};
The C# compiler generates code that sets these two properties in sequence, and when the Minimum property is
set to 180, it is greater than the default Maximum value of 100. You can avoid the exception in this case by setting
the Maximum property first:
In this example, setting Maximum to 360 is not a problem because it is greater than the default Minimum value of
0. When Minimum is set, the value is less than the Maximum value of 360.
The same problem exists in XAML. Set the properties in an order that ensures that Maximum is always greater
than Minimum :
<Stepper Maximum="360"
Minimum="180" ... />
You can set the Minimum and Maximum values to negative numbers, but only in an order where Minimum is
always less than Maximum :
<Stepper Minimum="-360"
Maximum="-180" ... />
The Value property is always greater than or equal to the Minimum value and less than or equal to Maximum . If
Value is set to a value outside that range, the value will be coerced to lie within the range, but no exception is
raised. For example, this code won't raise an exception:
<Stepper ValueChanged="OnStepperValueChanged"
Maximum="360"
Minimum="180" />
When Minimum is set to 180, Value is also set to 180, and the ValueChanged event is raised. This might occur
before the rest of the page has been constructed, and the handler might attempt to reference other elements on
the page that have not yet been created. You might want to add some code to the ValueChanged handler that
checks for null values of other elements on the page. Or, you can set the ValueChanged event handler after the
Stepper values have been initialized.
Switch
3/18/2022 • 4 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) Switch control is a horizontal toggle button that can be
manipulated by the user to toggle between on and off states, which are represented by a boolean value.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The following screenshot shows a Switch control in its on and off toggle states:
These properties are backed by BindableProperty objects, which means they can be styled and be the target of
data bindings.
The Switch control defines a Toggled event that's raised when the IsToggled property changes, either
through user manipulation or when an application sets the IsToggled property. The ToggledEventArgs object
that accompanies the Toggled event has a single property named Value , of type bool . When the event is
raised, the value of the Value property reflects the new value of the IsToggled property.
Create a Switch
A Switchcan be instantiated in XAML. Its IsToggled property can be set to toggle the Switch . By default, the
IsToggled property is false . The following example shows how to instantiate a Switch in XAML with the
optional IsToggled property set:
<Switch IsToggled="true"/>
Switch appearance
In addition to the properties that Switch inherits from the View class, Switch also defines OnColor and
ThumbColor properties. The OnColor property can be set to define the Switch color when it is toggled to its on
state, and the ThumbColor property can be set to define the Color of the switch thumb. The following example
shows how to instantiate a Switch in XAML with these properties set:
<Switch OnColor="Orange"
ThumbColor="Green" />
The following screenshot shows the Switch in its on and off toggle states, with the OnColor and ThumbColor
properties set:
The code-behind file contains the handler for the Toggled event:
The senderargument in the event handler is the Switch responsible for firing this event. You can use the
sender property to access the Switch object, or to distinguish between multiple Switch objects sharing the
same Toggled event handler.
The Toggled event handler can also be assigned in code:
In this example, the Label uses a binding expression in a DataTrigger to monitor the IsToggled property of
the Switch named styleSwitch . When this property becomes true , the FontAttributes and FontSize
properties of the Label are changed. When the IsToggled property returns to false , the FontAttributes and
FontSize properties of the Label are reset to their initial state.
<Switch IsToggled="True">
<VisualStateManager.VisualStateGroups>
<VisualStateGroup x:Name="CommonStates">
<VisualState x:Name="On">
<VisualState.Setters>
<Setter Property="ThumbColor"
Value="MediumSpringGreen" />
</VisualState.Setters>
</VisualState>
<VisualState x:Name="Off">
<VisualState.Setters>
<Setter Property="ThumbColor"
Value="Red" />
</VisualState.Setters>
</VisualState>
</VisualStateGroup>
</VisualStateManager.VisualStateGroups>
</Switch>
In this example, the On VisualState specifies that when the IsToggled property is true , the ThumbColor
property will be set to medium spring green. The Off VisualState specifies that when the IsToggled property
is false , the ThumbColor property will be set to red. Therefore, the overall effect is that when the Switch is in
an off position its thumb is red, and its thumb is medium spring green when the Switch is in an on position:
Disable a Switch
An app may enter a state where the Switch being toggled is not a valid operation. In such cases, the Switch
can be disabled by setting its IsEnabled property to false . This will prevent users from being able to
manipulate the Switch .
TimePicker
3/18/2022 • 2 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) TimePicker invokes the platform's time-picker control and allows
you to select a time.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
All of these properties are backed by BindableProperty objects, which means that they can be styled, and the
properties can be targets of data bindings. The Time property has a default binding mode of
BindingMode.TwoWay , which means that it can be a target of a data binding in an application that uses the Model-
View-ViewModel (MVVM) pattern.
NOTE
The TimePicker doesn't include an event to indicate a new selected Time value. If you need to be notified of this, you
can add a handler for the PropertyChanged event.
Create a TimePicker
When the Time property is specified in XAML, the value is converted to a TimeSpan and validated to ensure
that the number of milliseconds is greater than or equal to 0, and that the number of hours is less than 24. The
time components should be separated by colons:
<TimePicker ···
HorizontalOptions="Center" />
However, this is not recommended. Depending on the setting of the Format property, selected times might
require different display widths. For example, the "T" format string causes the TimePicker view to display times
in a long format, and "4:15:26 AM" requires a greater display width than the short time format ("t") of "4:15
AM". Depending on the platform, this difference might cause the TimePicker view to change width in layout, or
for the display to be truncated.
TIP
It's best to use the default HorizontalOptions setting of Fill with TimePicker , and not to use a width of Auto
when putting TimePicker in a Grid cell.
Editor
3/18/2022 • 7 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) Editor allows you to enter and edit multiple lines of text.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
In addition, Editor defines a Completed event, which is raised when the user finalizes text in the Editor with
the return key.
Editor derives from the InputView class, from which it inherits the following properties:
IsReadOnly , of type bool , defines whether the user should be prevented from modifying text. The default
value of this property is false .
IsSpellCheckEnabled , of type bool , controls whether spell checking is enabled.
Keyboard , of type Keyboard , specifies the virtual keyboard that's displayed when entering text.
MaxLength , of type int , defines the maximum input length.
TextTransform , of type TextTransform , specifies the casing of the entered text.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
In addition, defines a TextChanged event, which is raised when the text in the Editor changes. The
InputView
TextChangedEventArgs object that accompanies the TextChanged event has NewTextValue and OldTextValue
properties, which specify the new and old text, respectively.
Create an Editor
The following example shows how to create an Editor :
<Editor x:Name="editor"
Placeholder="Enter your response here"
HeightRequest="250"
TextChanged="OnTextChanged"
Completed="OnTextCompleted" />
Entered text can be accessed by reading the Text property, and the TextChanged and Completed events signal
that the text has changed or been completed.
The TextChanged event is raised when the text in the Editor changes, and the TextChangedEventArgs provide
the text before and after the change via the OldTextValue and NewTextValue properties:
The Completed event is raised when the user has ended input by pressing the return key on the keyboard, or by
pressing the Tab key on Windows. The handler for the event is a generic event handler:
The result is that characters in the text displayed by the Editor are spaced CharacterSpacing device-
independent units apart.
NOTE
The CharacterSpacing property value is applied to the text displayed by the Text and Placeholder properties.
A MaxLength property value of 0 indicates that no input will be allowed, and a value of int.MaxValue , which is
the default value for an Editor , indicates that there is no effective limit on the number of characters that may
be entered.
Auto-size an Editor
An Editor can be made to auto-size to its content by setting the Editor.AutoSize property to TextChanges ,
which is a value of the EditorAutoSizeOption enumeration. This enumeration has two values:
Disabled indicates that automatic resizing is disabled, and is the default value.
TextChanges indicates that automatic resizing is enabled.
When auto-resizing is enabled, the height of the Editor will increase when the user fills it with text, and the
height will decrease as the user deletes text.
NOTE
An Editor will not auto-size if the HeightRequest property has been set.
Transform text
An Editor can transform the casing of its text, stored in the Text property, by setting the TextTransform
property to a value of the TextTransform enumeration. This enumeration has four values:
None indicates that the text won't be transformed.
Default indicates that the default behavior for the platform will be used. This is the default value of the
TextTransform property.
Lowercase indicates that the text will be transformed to lowercase.
Uppercase indicates that the text will be transformed to uppercase.
Chat – used for texting and places where emoji are useful.
Default – the default keyboard.
Email – used when entering email addresses.
Numeric – used when entering numbers.
Plain – used when entering text, without any KeyboardFlags specified.
Telephone – used when entering telephone numbers.
Text – used when entering text.
Url – used for entering file paths & web addresses.
The Keyboard class also has a Create factory method that can be used to customize a keyboard by specifying
capitalization, spellcheck, and suggestion behavior. KeyboardFlags enumeration values are specified as
arguments to the method, with a customized Keyboard being returned. The KeyboardFlags enumeration
contains the following values:
None – no features are added to the keyboard.
CapitalizeSentence – indicates that the first letter of the first word of each entered sentence will be
automatically capitalized.
Spellcheck – indicates that spellcheck will be performed on entered text.
Suggestions – indicates that word completions will be offered on entered text.
CapitalizeWord – indicates that the first letter of each word will be automatically capitalized.
CapitalizeCharacter – indicates that every character will be automatically capitalized.
CapitalizeNone – indicates that no automatic capitalization will occur.
All – indicates that spellcheck, word completions, and sentence capitalization will occur on entered text.
The following XAML code example shows how to customize the default Keyboard to offer word completions
and capitalize every entered character:
<Editor>
<Editor.Keyboard>
<Keyboard x:FactoryMethod="Create">
<x:Arguments>
<KeyboardFlags>Suggestions,CapitalizeCharacter</KeyboardFlags>
</x:Arguments>
</Keyboard>
</Editor.Keyboard>
</Editor>
However, for some text entry scenarios, such as entering a username, spell checking provides a negative
experience and so should be disabled by setting the IsSpellCheckEnabled property to false :
NOTE
When the IsSpellCheckEnabled property is set to false , and a custom keyboard isn't being used, the native spell
checker will be disabled. However, if a Keyboard has been set that disables spell checking, such as Keyboard.Chat , the
IsSpellCheckEnabled property is ignored. Therefore, the property cannot be used to enable spell checking for a
Keyboard that explicitly disables it.
NOTE
When the IsTextPredictionEnabled property is set to false , and a custom keyboard isn't being used, text prediction
and automatic text correction is disabled. However, if a Keyboard has been set that disables text prediction, the
IsTextPredictionEnabled property is ignored. Therefore, the property cannot be used to enable text prediction for a
Keyboard that explicitly disables it.
Prevent text entry
Users can be prevented from modifying the text in an Editor by setting the IsReadOnly property, which has a
default value of false , to true :
NOTE
The IsReadonly property does not alter the visual appearance of an Editor , unlike the IsEnabled property that also
changes the visual appearance of the Editor to gray.
Entry
3/18/2022 • 9 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) Entry allows you to enter and edit a single line of text. In addition,
the Entry can be used as a password field.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
In addition, Entry defines a Completed event, which is raised when the user finalizes text in the Entry with the
return key.
Entry derives from the InputView class, from which it inherits the following properties:
IsReadOnly , of type bool , defines whether the user should be prevented from modifying text. The default
value of this property is false .
IsSpellCheckEnabled , of type bool , controls whether spell checking is enabled.
MaxLength , of type int , defines the maximum input length.
TextTransform , of type TextTransform , specifies the casing of the entered text.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
In addition, defines a TextChanged event, which is raised when the text in the Entry changes. The
InputView
TextChangedEventArgs object that accompanies the TextChanged event has NewTextValue and OldTextValue
properties, which specify the new and old text, respectively.
Create an Entry
The following example shows how to create an Entry :
<Entry x:Name="entry"
Placeholder="Enter text"
TextChanged="OnTextChanged"
Completed="OnTextCompleted" />
Entered text can be accessed by reading the Text property, and the TextChanged and Completed events signal
that the text has changed or been completed.
The TextChanged event is raised when the text in the Entry changes, and the TextChangedEventArgs provide the
text before and after the change via the OldTextValue and NewTextValue properties:
The Completed event is raised when the user has ended input by pressing the return key on the keyboard, or by
pressing the Tab key on Windows. The handler for the event is a generic event handler:
After the Completed event fires, any ICommand specified by the ReturnCommand property is executed, with the
object specified by the ReturnCommandParameter property being passed to the ReturnCommand .
NOTE
The VisualElement class, which is in the Entry inheritance hierarchy, also has Focused and Unfocused events.
<Entry ...
CharacterSpacing="10" />
The result is that characters in the text displayed by the Entry are spaced CharacterSpacing device-
independent units apart.
NOTE
The CharacterSpacing property value is applied to the text displayed by the Text and Placeholder properties.
<Entry ...
MaxLength="10" />
A MaxLength property value of 0 indicates that no input will be allowed, and a value of int.MaxValue , which is
the default value for an Entry , indicates that there is no effective limit on the number of characters that may be
entered.
The default value of the CursorPosition property is 0, which indicates that text will be inserted at the start of the
Entry .
In addition, the SelectionLength property can be used to return or set the length of text selection within the
Entry :
The default value of the SelectionLength property is 0, which indicates that no text is selected.
Display a clear button
The ClearButtonVisibility property can be used to control whether an Entry displays a clear button, which
enables the user to clear the text. This property should be set to a ClearButtonVisibility enumeration member:
Never indicates that a clear button will never be displayed. This is the default value for the
ClearButtonVisibility property.
WhileEditing indicates that a clear button will be displayed in the Entry , while it has focus and text.
The following screenshot shows an Entry on Android with the clear button enabled:
Transform text
An Entry can transform the casing of its text, stored in the Text property, by setting the TextTransform
property to a value of the TextTransform enumeration. This enumeration has four values:
None indicates that the text won't be transformed.
Default indicates that the default behavior for the platform will be used. This is the default value of the
TextTransform property.
Lowercase indicates that the text will be transformed to lowercase.
Uppercase indicates that the text will be transformed to uppercase.
The following screenshot shows an Entry whose input has been obscured:
The Keyboard class also has a Create factory method that can be used to customize a keyboard by specifying
capitalization, spellcheck, and suggestion behavior. KeyboardFlags enumeration values are specified as
arguments to the method, with a customized Keyboard being returned. The KeyboardFlags enumeration
contains the following values:
None – no features are added to the keyboard.
CapitalizeSentence – indicates that the first letter of the first word of each entered sentence will be
automatically capitalized.
Spellcheck – indicates that spellcheck will be performed on entered text.
Suggestions – indicates that word completions will be offered on entered text.
CapitalizeWord – indicates that the first letter of each word will be automatically capitalized.
CapitalizeCharacter – indicates that every character will be automatically capitalized.
CapitalizeNone – indicates that no automatic capitalization will occur.
All – indicates that spellcheck, word completions, and sentence capitalization will occur on entered text.
The following XAML code example shows how to customize the default Keyboard to offer word completions
and capitalize every entered character:
The following XAML example shows how to set the return key:
NOTE
The exact appearance of the return key is dependent upon the platform. On iOS, the return key is a text-based button.
However, on Android and Windows, the return key is a icon-based button.
When the return key is pressed, the Completed event fires and any ICommand specified by the ReturnCommand
property is executed. In addition, any object specified by the ReturnCommandParameter property will be passed
to the ICommand as a parameter. For more information about commands, see Commanding.
However, for some text entry scenarios, such as entering a username, spell checking provides a negative
experience and should be disabled by setting the IsSpellCheckEnabled property to false :
NOTE
When the IsSpellCheckEnabled property is set to false , and a custom keyboard isn't being used, the native spell
checker will be disabled. However, if a Keyboard has been set that disables spell checking, such as Keyboard.Chat , the
IsSpellCheckEnabled property is ignored. Therefore, the property cannot be used to enable spell checking for a
Keyboard that explicitly disables it.
NOTE
The IsReadonly property does not alter the visual appearance of an Entry , unlike the IsEnabled property that also
changes the visual appearance of the Entry to gray.
ActivityIndicator
3/18/2022 • 2 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) ActivityIndicator displays an animation to show that the
application is engaged in a lengthy activity. Unlike ProgressBar , ActivityIndicator gives no indication of
progress.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Create an ActivityIndicator
To indicate a lengthy activity, create an ActivityIndicator object and sets its properties to define its appearance.
The following XAML example shows how to display an ActivityIndicator :
The following XAML example shows how to change the color of an ActivityIndicator :
<ActivityIndicator IsRunning="true"
Color="Orange" />
The equivalent C# code is:
The .NET Multi-platform App UI (.NET MAUI) ProgressBar indicates to users that the app is progressing through
a lengthy activity. The progress bar is a horizontal bar that is filled to a percentage represented by a double
value.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The appearance of a ProgressBar is platform-dependent, and the following screenshot shows a ProgressBar on
iOS and Android:
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
ProgressBar also defines a ProgressTo method that animates the bar from its current value to a specified value.
For more information, see Animate a ProgressBar.
Create a ProgressBar
To indicate progress through a lengthy activity, create a ProgressBar object and set its properties to define its
appearance.
The following XAML example shows how to display a ProgressBar :
The following XAML example shows how to change the color of a ProgressBar :
<ProgressBar Progress="0.5"
ProgressColor="Orange" />
Animate a ProgressBar
The ProgressTo method animates the ProgressBar from its current Progress value to a provided value over
time. The method accepts a double progress value, a uint duration in milliseconds, an Easing enum value
and returns a Task<bool> . The following example demonstrates how to animate a ProgressBar :
For more information about the Easing enumeration, see Easing functions.
CarouselView
3/18/2022 • 2 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) CarouselView is a view for presenting data in a scrollable layout,
where users can swipe to move through a collection of items.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
By default, CarouselView will display its items in a horizontal orientation. A single item will be displayed on
screen, with swipe gestures resulting in forwards and backwards navigation through the collection of items. In
addition, indicators can be displayed that represent each item in the CarouselView :
By default, CarouselView provides looped access to its collection of items. Therefore, swiping backwards from
the first item in the collection will display the last item in the collection. Similarly, swiping forwards from the last
item in the collection will return to the first item in the collection.
CarouselView shares much of its implementation with CollectionView . However, the two controls have different
use cases. CollectionView is typically used to present lists of data of any length, whereas CarouselView is
typically used to highlight information in a list of limited length.
Populate a CarouselView with data
3/18/2022 • 9 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) CarouselView includes the following properties that define the data
to be displayed, and its appearance:
ItemsSource , of type IEnumerable , specifies the collection of items to be displayed, and has a default value of
null .
ItemTemplate , of type DataTemplate , specifies the template to apply to each item in the collection of items to
be displayed.
These properties are backed by BindableProperty objects, which means that the properties can be targets of
data bindings.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
CarouselView defines a ItemsUpdatingScrollMode property that represents the scrolling behavior of the
CarouselView when new items are added to it. For more information about this property, see Control scroll
position when new items are added.
CarouselView supports incremental data virtualization as the user scrolls. For more information, see Load data
incrementally.
IMPORTANT
If the CarouselView is required to refresh as items are added, removed, or changed in the underlying collection, the
underlying collection should be an IEnumerable collection that sends property change notifications, such as
ObservableCollection .
CarouselView can be populated with data by using data binding to bind its ItemsSource property to an
IEnumerable collection. In XAML, this is achieved with the Binding markup extension:
NOTE
Compiled bindings can be enabled to improve data binding performance in .NET MAUI applications.
For information on how to change the CarouselView orientation, see Specify CarouselView layout. For
information on how to define the appearance of each item in the CarouselView , see Define item appearance.
return rootStackLayout;
});
The elements specified in the DataTemplate define the appearance of each item in the CarouselView . In the
example, layout within the DataTemplate is managed by a StackLayout , and the data is displayed with an Image
object, and three Label objects, that all bind to properties of the Monkey class:
<ContentPage ...
xmlns:controls="clr-namespace:CarouselViewDemos.Controls"
x:Class="CarouselViewDemos.Views.HorizontalLayoutDataTemplateSelectorPage">
<ContentPage.Resources>
<DataTemplate x:Key="AmericanMonkeyTemplate">
...
</DataTemplate>
<DataTemplate x:Key="OtherMonkeyTemplate">
...
</DataTemplate>
<controls:MonkeyDataTemplateSelector x:Key="MonkeySelector"
AmericanMonkey="{StaticResource AmericanMonkeyTemplate}"
OtherMonkey="{StaticResource OtherMonkeyTemplate}" />
</ContentPage.Resources>
The MonkeyDataTemplateSelector class defines AmericanMonkey and OtherMonkey DataTemplate properties that
are set to different data templates. The OnSelectTemplate override returns the AmericanMonkey template when
the monkey name contains "America". When the monkey name doesn't contain "America", the
OnSelectTemplate override returns the OtherMonkey template, which displays its data grayed out:
IMPORTANT
When using CarouselView , never set the root element of your DataTemplate objects to a ViewCell . This will result in
an exception being thrown because CarouselView has no concept of cells.
Display indicators
Indicators, that represent the number of items and current position in a CarouselView , can be displayed next to
the CarouselView . This can be accomplished with the IndicatorView control:
<StackLayout>
<CarouselView ItemsSource="{Binding Monkeys}"
IndicatorView="indicatorView">
<CarouselView.ItemTemplate>
<!-- DataTemplate that defines item appearance -->
</CarouselView.ItemTemplate>
</CarouselView>
<IndicatorView x:Name="indicatorView"
IndicatorColor="LightGray"
SelectedIndicatorColor="DarkGray"
HorizontalOptions="Center" />
</StackLayout>
In this example, the IndicatorView is rendered beneath the CarouselView , with an indicator for each item in the
CarouselView . The IndicatorView is populated with data by setting the CarouselView.IndicatorView property to
the IndicatorView object. Each indicator is a light gray circle, while the indicator that represents the current item
in the CarouselView is dark gray:
IMPORTANT
Setting the CarouselView.IndicatorView property results in the IndicatorView.Position property binding to the
CarouselView.Position property, and the IndicatorView.ItemsSource property binding to the
CarouselView.ItemsSource property.
Context menus
CarouselView supports context menus for items of data through the SwipeView , which reveals the context menu
with a swipe gesture. The SwipeView is a container control that wraps around an item of content, and provides
context menu items for that item of content. Therefore, context menus are implemented for a CarouselView by
creating a SwipeView that defines the content that the SwipeView wraps around, and the context menu items
that are revealed by the swipe gesture. This is achieved by adding a SwipeView to the DataTemplate that defines
the appearance of each item of data in the CarouselView :
<CarouselView x:Name="carouselView"
ItemsSource="{Binding Monkeys}">
<CarouselView.ItemTemplate>
<DataTemplate>
<StackLayout>
<Frame HasShadow="True"
BorderColor="DarkGray"
CornerRadius="5"
Margin="20"
HeightRequest="300"
HorizontalOptions="Center"
VerticalOptions="CenterAndExpand">
<SwipeView>
<SwipeView.TopItems>
<SwipeItems>
<SwipeItem Text="Favorite"
IconImageSource="favorite.png"
BackgroundColor="LightGreen"
Command="{Binding Source={x:Reference carouselView},
Path=BindingContext.FavoriteCommand}"
CommandParameter="{Binding}" />
</SwipeItems>
</SwipeView.TopItems>
<SwipeView.BottomItems>
<SwipeItems>
<SwipeItem Text="Delete"
IconImageSource="delete.png"
BackgroundColor="LightPink"
Command="{Binding Source={x:Reference carouselView},
Path=BindingContext.DeleteCommand}"
CommandParameter="{Binding}" />
</SwipeItems>
</SwipeView.BottomItems>
<StackLayout>
<!-- Define item appearance -->
</StackLayout>
</SwipeView>
</Frame>
</StackLayout>
</DataTemplate>
</CarouselView.ItemTemplate>
</CarouselView>
return stackLayout;
});
In this example, the SwipeView content is a StackLayout that defines the appearance of each item that's
surrounded by a Frame in the CarouselView . The swipe items are used to perform actions on the SwipeView
content, and are revealed when the control is swiped from the bottom and from the top:
SwipeView supports four different swipe directions, with the swipe direction being defined by the directional
SwipeItems collection the SwipeItems objects are added to. By default, a swipe item is executed when it's tapped
by the user. In addition, once a swipe item has been executed the swipe items are hidden and the SwipeView
content is re-displayed. However, these behaviors can be changed.
Pull to refresh
CarouselView supports pull to refresh functionality through the RefreshView , which enables the data being
displayed to be refreshed by pulling down on the items. The RefreshView is a container control that provides
pull to refresh functionality to its child, provided that the child supports scrollable content. Therefore, pull to
refresh is implemented for a CarouselView by setting it as the child of a RefreshView :
When the user initiates a refresh, the ICommand defined by the Command property is executed, which should
refresh the items being displayed. A refresh visualization is shown while the refresh occurs, which consists of an
animated progress circle:
The value of the RefreshView.IsRefreshing property indicates the current state of the RefreshView . When a
refresh is triggered by the user, this property will automatically transition to true . Once the refresh completes,
you should reset the property to false .
CarouselView also defines a RemainingItemsThresholdReached event that is fired when the CarouselView is
scrolled far enough that RemainingItemsThreshold items have not been displayed. This event can be handled to
load more items. In addition, when the RemainingItemsThresholdReached event is fired, the
RemainingItemsThresholdReachedCommand is executed, enabling incremental data loading to take place in a
viewmodel.
The default value of the RemainingItemsThresholdproperty is -1, which indicates that the
RemainingItemsThresholdReached event will never be fired. When the property value is 0, the
RemainingItemsThresholdReached event will be fired when the final item in the ItemsSource is displayed. For
values greater than 0, the RemainingItemsThresholdReached event will be fired when the ItemsSource contains
that number of items not yet scrolled to.
NOTE
CarouselView validates the RemainingItemsThreshold property so that its value is always greater than or equal to -1.
The following XAML example shows a CarouselView that loads data incrementally:
In this code example, the RemainingItemsThresholdReached event fires when there are 2 items not yet scrolled to,
and in response executes the OnCollectionViewRemainingItemsThresholdReached event handler:
The .NET Multi-platform App UI (.NET MAUI) CarouselView defines the following properties that control layout:
ItemsLayout , of type LinearItemsLayout , specifies the layout to be used.
PeekAreaInsets , of type Thickness , specifies how much to make adjacent items partially visible by.
These properties are backed by BindableProperty objects, which means that the properties can be targets of
data bindings.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
By default, a CarouselView will display its items in a horizontal orientation. A single item will be displayed on
screen, with swipe gestures resulting in forwards and backwards navigation through the collection of items.
However, a vertical orientation is also possible. This is because the ItemsLayout property is of type
LinearItemsLayout , which inherits from the ItemsLayout class. The ItemsLayout class defines the following
properties:
, of type
Orientation ItemsLayoutOrientation , specifies the direction in which the CarouselView expands as
items are added.
SnapPointsAlignment , of type SnapPointsAlignment , specifies how snap points are aligned with items.
SnapPointsType , of type SnapPointsType , specifies the behavior of snap points when scrolling.
These properties are backed by BindableProperty objects, which means that the properties can be targets of
data bindings. For more information about snap points, see Snap points in Control scrolling in a CarouselView
guide.
The ItemsLayoutOrientation enumeration defines the following members:
Vertical indicates that the CarouselView will expand vertically as items are added.
Horizontal indicates that the CarouselView will expand horizontally as items are added.
The LinearItemsLayout class inherits from the ItemsLayout class, and defines an ItemSpacing property, of type
double , that represents the empty space around each item. The default value of this property is 0, and its value
must always be greater than or equal to 0. The LinearItemsLayout class also defines static Vertical and
Horizontal members. These members can be used to create vertical or horizontal lists, respectively.
Alternatively, a LinearItemsLayout object can be created, specifying an ItemsLayoutOrientation enumeration
member as an argument.
NOTE
CarouselView uses the native layout engines to perform layout.
Horizontal layout
By default, CarouselView will display its items horizontally. Therefore, it's not necessary to set the ItemsLayout
property to use this layout:
Alternatively, this layout can also be accomplished by setting the ItemsLayout property to a LinearItemsLayout
object, specifying the Horizontal ItemsLayoutOrientation enumeration member as the Orientation property
value:
This results in a layout that grows horizontally as new items are added.
Vertical layout
CarouselView can display its items vertically by setting the ItemsLayout property to a LinearItemsLayout
object, specifying the Vertical ItemsLayoutOrientation enumeration member as the Orientation property
value:
This results in a layout that grows vertically as new items are added.
Item spacing
By default, there is no space between each item in a CarouselView . This behavior can be changed by setting the
ItemSpacing property on the items layout used by the CarouselView .
NOTE
The LinearItemsLayout.ItemSpacing property has a validation callback set, which ensures that the value of the
property is always greater than or equal to 0.
This code results in a vertical layout, that has a spacing of 20 between items.
The OnImageTapped event handler is executed in response to an Image object being tapped, and changes the
dimensions of the image (and its parent Frame , so that it's more easily viewed:
Right-to-left layout
CarouselView can layout its content in a right-to-left flow direction by setting its FlowDirection property to
RightToLeft . However, the FlowDirection property should ideally be set on a page or root layout, which causes
all the elements within the page, or root layout, to respond to the flow direction:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="CarouselViewDemos.Views.HorizontalTemplateLayoutRTLPage"
Title="Horizontal layout (RTL FlowDirection)"
FlowDirection="RightToLeft">
<CarouselView ItemsSource="{Binding Monkeys}">
...
</CarouselView>
</ContentPage>
The default FlowDirection for an element with a parent is MatchParent . Therefore, the CarouselView inherits
the FlowDirection property value from the ContentPage .
Configure CarouselView interaction
3/18/2022 • 8 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) CarouselView defines the following properties that control user
interaction:
CurrentItem , of type object , the current item being displayed. This property has a default binding mode of
TwoWay , and has a null value when there isn't any data to display.
CurrentItemChangedCommand , of type ICommand , which is executed when the current item changes.
CurrentItemChangedCommandParameter , of type object , which is the parameter that's passed to the
CurrentItemChangedCommand .
IsBounceEnabled , of type bool , which specifies whether the CarouselView will bounce at a content
boundary. The default value is true .
IsSwipeEnabled , of type bool , which determines whether a swipe gesture will change the displayed item.
The default value is true .
Loop , of type bool , which determines whether the CarouselView provides looped access to its collection of
items. The default value is true .
Position , of type int , the index of the current item in the underlying collection. This property has a default
binding mode of TwoWay , and has a 0 value when there isn't any data to display.
PositionChangedCommand , of type ICommand , which is executed when the position changes.
PositionChangedCommandParameter , of type object , which is the parameter that's passed to the
PositionChangedCommand .
VisibleViews , of type ObservableCollection<View> , which is a read-only property that contains the objects
for the items that are currently visible.
All of these properties are backed by BindableProperty objects, which means that the properties can be targets
of data bindings.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
CarouselView defines a CurrentItemChanged event that's fired when the CurrentItem property changes, either
due to user scrolling, or when an application sets the property. The CurrentItemChangedEventArgs object that
accompanies the CurrentItemChanged event has two properties, both of type object :
PreviousItem – the previous item, after the property change.
CurrentItem – the current item, after the property change.
CarouselView also defines a PositionChanged event that's fired when the Position property changes, either
due to user scrolling, or when an application sets the property. The PositionChangedEventArgs object that
accompanies the PositionChanged event has two properties, both of type int :
PreviousPosition – the previous position, after the property change.
CurrentPosition – the current position, after the property change.
Respond to the current item changing
When the currently displayed item changes, the CurrentItem property will be set to the value of the item. When
this property changes, the CurrentItemChangedCommand is executed with the value of the
CurrentItemChangedCommandParameter being passed to the ICommand . The Position property is then updated, and
the CurrentItemChanged event fires.
IMPORTANT
The Position property changes when the CurrentItem property changes. This will result in the
PositionChangedCommand being executed, and the PositionChanged event firing.
Event
The following XAML example shows a CarouselView that uses an event handler to respond to the current item
changing:
In this example, the OnCurrentItemChanged event handler is executed when the CurrentItemChanged event fires:
In this example, the OnCurrentItemChanged event handler exposes the previous and current items:
Command
The following XAML example shows a CarouselView that uses a command to respond to the current item
changing:
In this example, the CurrentItemChangedCommand property binds to the ItemChangedCommand property, passing the
CurrentItem property value to it as an argument. The ItemChangedCommand can then respond to the current item
changing, as required:
In this example, the ItemChangedCommand updates objects that store the previous and current items.
NOTE
Setting the Position property to 0 will result in the first item in the underlying collection being displayed.
Event
The following XAML example shows a CarouselView that uses an event handler to respond to the Position
property changing:
In this example, the OnPositionChanged event handler is executed when the PositionChanged event fires:
In this example, the OnCurrentItemChanged event handler exposes the previous and current positions:
Command
The following XAML example shows a CarouselView that uses a command to respond to the Position property
changing:
In this example, the PositionChangedCommand property binds to the PositionChangedCommand property, passing the
Position property value to it as an argument. The PositionChangedCommand can then respond to the position
changing, as required:
public ICommand PositionChangedCommand => new Command<int>((position) =>
{
PreviousPosition = CurrentPosition;
CurrentPosition = position;
});
In this example, the PositionChangedCommand updates objects that store the previous and current positions.
NOTE
The CurrentItem property has a default binding mode of TwoWay .
The CarouselView.CurrentItem property data binds to the CurrentItem property of the connected view model,
which is of type Monkey . By default, a TwoWay binding is used so that if the user changes the current item, the
value of the CurrentItem property will be set to the current Monkey object. The CurrentItem property is defined
in the MonkeysViewModel class:
public MonkeysViewModel()
{
// ...
CurrentItem = Monkeys.Skip(3).FirstOrDefault();
OnPropertyChanged("CurrentItem");
}
}
In this example, the CurrentItem property is set to the fourth item in the Monkeys collection:
Preset the position
The displayed item CarouselView can be programmatically set by setting the Position property to the index of
the item in the underlying collection. The following XAML example shows a CarouselView that sets the displayed
item:
NOTE
The Position property has a default binding mode of TwoWay .
The CarouselView.Position property data binds to the Position property of the connected view model, which
is of type int . By default, a TwoWay binding is used so that if the user scrolls through the CarouselView , the
value of the Position property will be set to the index of the displayed item. The Position property is defined
in the MonkeysViewModel class:
public MonkeysViewModel()
{
// ...
Position = 3;
OnPropertyChanged("Position");
}
}
In this example, the Position property is set to the fourth item in the Monkeys collection:
These visual states can be used to initiate visual changes to the items displayed by the CarouselView .
The following XAML example shows how to define the CurrentItem , PreviousItem , NextItem , and DefaultItem
visual states:
<CarouselView ItemsSource="{Binding Monkeys}"
PeekAreaInsets="100">
<CarouselView.ItemTemplate>
<DataTemplate>
<StackLayout>
<VisualStateManager.VisualStateGroups>
<VisualStateGroup x:Name="CommonStates">
<VisualState x:Name="CurrentItem">
<VisualState.Setters>
<Setter Property="Scale"
Value="1.1" />
</VisualState.Setters>
</VisualState>
<VisualState x:Name="PreviousItem">
<VisualState.Setters>
<Setter Property="Opacity"
Value="0.5" />
</VisualState.Setters>
</VisualState>
<VisualState x:Name="NextItem">
<VisualState.Setters>
<Setter Property="Opacity"
Value="0.5" />
</VisualState.Setters>
</VisualState>
<VisualState x:Name="DefaultItem">
<VisualState.Setters>
<Setter Property="Opacity"
Value="0.25" />
</VisualState.Setters>
</VisualState>
</VisualStateGroup>
</VisualStateManager.VisualStateGroups>
In this example, the CurrentItem visual state specifies that the current item displayed by the CarouselView will
have its Scale property changed from its default value of 1 to 1.1. The PreviousItem and NextItem visual
states specify that the items surrounding the current item will be displayed with an Opacity value of 0.5. The
DefaultItem visual state specifies that the remainder of the items displayed by the CarouselView will be
displayed with an Opacity value of 0.25.
NOTE
Alternatively, the visual states can be defined in a Style that has a TargetType property value that's the type of the
root element of the DataTemplate , which is set as the ItemTemplate property value.
The following screenshot shows the CurrentItem , PreviousItem , and NextItem visual states:
Clear the current item
The CurrentItem property can be cleared by setting it, or the object it binds to, to null .
Disable bounce
By default, CarouselView bounces items at content boundaries. This can be disabled by setting the
IsBounceEnabled property to false .
Disable loop
By default, CarouselView provides looped access to its collection of items. Therefore, swiping backwards from
the first item in the collection will display the last item in the collection. Similarly, swiping forwards from the last
item in the collection will return to the first item in the collection. This behavior can be disabled by setting the
Loop property to false .
The .NET Multi-platform App UI (.NET MAUI) CarouselView defines the following properties that can be used to
provide user feedback when there's no data to display:
EmptyView , of type object , the string, binding, or view that will be displayed when the ItemsSource
property is null , or when the collection specified by the ItemsSource property is null or empty. The
default value is null .
EmptyViewTemplate , of type DataTemplate , the template to use to format the specified EmptyView . The default
value is null .
These properties are backed by BindableProperty objects, which means that the properties can be targets of
data bindings.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The main usage scenarios for setting the EmptyView property are displaying user feedback when a filtering
operation on a CarouselView yields no data, and displaying user feedback while data is being retrieved from a
web service.
NOTE
The EmptyView property can be set to a view that includes interactive content if required.
The result is that, because the data bound collection is null , the string set as the EmptyView property value is
displayed.
Display views when data is unavailable
The EmptyView property can be set to a view, which will be displayed when the ItemsSource property is null ,
or when the collection specified by the ItemsSource property is null or empty. This can be a single view, or a
view that contains multiple child views. The following XAML example shows the EmptyView property set to a
view that contains multiple child views:
<StackLayout Margin="20">
<SearchBar SearchCommand="{Binding FilterCommand}"
SearchCommandParameter="{Binding Source={RelativeSource Self}, Path=Text}"
Placeholder="Filter" />
<CarouselView ItemsSource="{Binding Monkeys}">
<CarouselView.EmptyView>
<ContentView>
<StackLayout HorizontalOptions="CenterAndExpand"
VerticalOptions="CenterAndExpand">
<Label Text="No results matched your filter."
Margin="10,25,10,10"
FontAttributes="Bold"
FontSize="18"
HorizontalOptions="Fill"
HorizontalTextAlignment="Center" />
<Label Text="Try a broader filter?"
FontAttributes="Italic"
FontSize="12"
HorizontalOptions="Fill"
HorizontalTextAlignment="Center" />
</StackLayout>
</ContentView>
</CarouselView.EmptyView>
<CarouselView.ItemTemplate>
...
</CarouselView.ItemTemplate>
</CarouselView>
</StackLayout>
In this example, what looks like a redundant has been added as the root element of the [ EmptyView . This is
because internally, the EmptyView is added to a native container that doesn't provide any context for .NET MAUI
layout. Therefore, to position the views that comprise your EmptyView , you must add a root layout, whose child
is a layout that can position itself within the root layout.
The equivalent C# code is:
When the SearchBar executes the FilterCommand , the collection displayed by the CarouselView is filtered for
the search term stored in the SearchBar.Text property. If the filtering operation yields no data, the StackLayout
set as the EmptyView property value is displayed.
Display a templated custom type when data is unavailable
The EmptyView property can be set to a custom type, whose template is displayed when the ItemsSource
property is null , or when the collection specified by the ItemsSource property is null or empty. The
EmptyViewTemplate property can be set to a DataTemplate that defines the appearance of the EmptyView . The
following XAML shows an example of this scenario:
<StackLayout Margin="20">
<SearchBar x:Name="searchBar"
SearchCommand="{Binding FilterCommand}"
SearchCommandParameter="{Binding Source={RelativeSource Self}, Path=Text}"
Placeholder="Filter" />
<CarouselView ItemsSource="{Binding Monkeys}">
<CarouselView.EmptyView>
<controls:FilterData Filter="{Binding Source={x:Reference searchBar}, Path=Text}" />
</CarouselView.EmptyView>
<CarouselView.EmptyViewTemplate>
<DataTemplate>
<Label Text="{Binding Filter, StringFormat='Your filter term of {0} did not match any
records.'}"
Margin="10,25,10,10"
FontAttributes="Bold"
FontSize="18"
HorizontalOptions="Fill"
HorizontalTextAlignment="Center" />
</DataTemplate>
</CarouselView.EmptyViewTemplate>
<CarouselView.ItemTemplate>
...
</CarouselView.ItemTemplate>
</CarouselView>
</StackLayout>
The property is set to a FilterData object, and the Filter property data binds to the
EmptyView
SearchBar.Text property. When the SearchBar executes the FilterCommand , the collection displayed by the
CarouselView is filtered for the search term stored in the Filter property. If the filtering operation yields no
data, the Label defined in the DataTemplate , that's set as the EmptyViewTemplate property value, is displayed.
NOTE
When displaying a templated custom type when data is unavailable, the EmptyViewTemplate property can be set to a
view that contains multiple child views.
This XAML defines two ContentView objects in the page-level ResourceDictionary , with the Switch object
controlling which ContentView object will be set as the EmptyView property value. When the Switch is toggled,
the OnEmptyViewSwitchToggled event handler executes the ToggleEmptyView method:
The ToggleEmptyView method sets the EmptyView property of the carouselView object to one of the two
ContentView objects stored in the ResourceDictionary , based on the value of the Switch.IsToggled property.
When the SearchBar executes the FilterCommand , the collection displayed by the CarouselView is filtered for
the search term stored in the SearchBar.Text property. If the filtering operation yields no data, the ContentView
object set as the EmptyView property is displayed.
<ContentPage ...
xmlns:controls="clr-namespace:CarouselViewDemos.Controls">
<ContentPage.Resources>
<DataTemplate x:Key="AdvancedTemplate">
...
</DataTemplate>
<DataTemplate x:Key="BasicTemplate">
...
</DataTemplate>
<controls:SearchTermDataTemplateSelector x:Key="SearchSelector"
DefaultTemplate="{StaticResource AdvancedTemplate}"
OtherTemplate="{StaticResource BasicTemplate}" />
</ContentPage.Resources>
<StackLayout Margin="20">
<SearchBar x:Name="searchBar"
SearchCommand="{Binding FilterCommand}"
SearchCommandParameter="{Binding Source={RelativeSource Self}, Path=Text}"
Placeholder="Filter" />
<CarouselView ItemsSource="{Binding Monkeys}"
EmptyView="{Binding Source={x:Reference searchBar}, Path=Text}"
EmptyViewTemplate="{StaticResource SearchSelector}">
<CarouselView.ItemTemplate>
...
</CarouselView.ItemTemplate>
</CarouselView>
</StackLayout>
</ContentPage>
The EmptyViewproperty is set to the SearchBar.Text property, and the EmptyViewTemplate property is set to a
SearchTermDataTemplateSelector object.
When the SearchBar executes the FilterCommand , the collection displayed by the CarouselView is filtered for
the search term stored in the SearchBar.Text property. If the filtering operation yields no data, the
DataTemplate chosen by the SearchTermDataTemplateSelector object is set as the EmptyViewTemplate property
and displayed.
The following example shows the SearchTermDataTemplateSelector class:
public class SearchTermDataTemplateSelector : DataTemplateSelector
{
public DataTemplate DefaultTemplate { get; set; }
public DataTemplate OtherTemplate { get; set; }
The .NET Multi-platform App UI (.NET MAUI) CarouselView defines the following scroll related properties:
HorizontalScrollBarVisibility , of type ScrollBarVisibility , which specifies when the horizontal scroll bar
is visible.
IsDragging , of type bool , which indicates whether the CarouselView is scrolling. This is a read only
property, whose default value is false .
IsScrollAnimated , of type bool , which specifies whether an animation will occur when scrolling the
CarouselView . The default value is true .
ItemsUpdatingScrollMode , of type ItemsUpdatingScrollMode , which represents the scrolling behavior of the
CarouselView when new items are added to it.
VerticalScrollBarVisibility , of type ScrollBarVisibility , which specifies when the vertical scroll bar is
visible.
All of these properties are backed by BindableProperty objects, which means that they can be targets of data
bindings.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
CarouselView also defines two ScrollTo methods, that scroll items into view. One of the overloads scrolls the
item at the specified index into view, while the other scrolls the specified item into view. Both overloads have
additional arguments that can be specified to indicate the exact position of the item after the scroll has
completed, and whether to animate the scroll.
CarouselView defines a event that is fired when one of the ScrollTo methods is invoked.
ScrollToRequested
The ScrollToRequestedEventArgs object that accompanies the ScrollToRequested event has many properties,
including IsAnimated , Index , Item , and ScrollToPosition . These properties are set from the arguments
specified in the ScrollTo method calls.
In addition, defines a Scrolled event that is fired to indicate that scrolling occurred. The
CarouselView
ItemsViewScrolledEventArgs object that accompanies the Scrolled event has many properties. For more
information, see Detect scrolling.
When a user swipes to initiate a scroll, the end position of the scroll can be controlled so that items are fully
displayed. This feature is known as snapping, because items snap to position when scrolling stops. For more
information, see Snap points.
CarouselView can also load data incrementally as the user scrolls. For more information, see Load data
incrementally.
Detect scrolling
The IsDragging property can be examined to determine whether the CarouselView is currently scrolling
through items.
In addition, CarouselView defines a Scrolled event which is fired to indicate that scrolling occurred. This event
should be consumed when data about the scroll is required.
The following XAML example shows a CarouselView that sets an event handler for the Scrolled event:
<CarouselView Scrolled="OnCollectionViewScrolled">
...
</CarouselView>
In this code example, the OnCarouselViewScrolled event handler is executed when the Scrolled event fires:
In this example, the OnCarouselViewScrolled event handler outputs the values of the ItemsViewScrolledEventArgs
object that accompanies the event.
IMPORTANT
The Scrolled event is fired for user initiated scrolls, and for programmatic scrolls.
carouselView.ScrollTo(6);
NOTE
The ScrollToRequested event is fired when the ScrollTo method is invoked.
NOTE
The ScrollToRequested event is fired when the ScrollTo method is invoked.
This example code results in the minimal scrolling required to scroll the item into view.
NOTE
The ScrollToPosition.MakeVisible member is used by default, if the position argument is not specified when
calling the ScrollTo method.
Start
The ScrollToPosition.Start member indicates that the item should be scrolled to the start of the view:
This example code results in the item being scrolled to the start of the view.
Center
The ScrollToPosition.Center member indicates that the item should be scrolled to the center of the view:
This example code results in the item being scrolled to the end of the view.
<CarouselView ItemsUpdatingScrollMode="KeepLastItemInView">
...
</CarouselView>
Snap points
When a user swipes to initiate a scroll, the end position of the scroll can be controlled so that items are fully
displayed. This feature is known as snapping, because items snap to position when scrolling stops, and is
controlled by the following properties from the ItemsLayout class:
SnapPointsType , of type SnapPointsType , specifies the behavior of snap points when scrolling.
SnapPointsAlignment , of type SnapPointsAlignment , specifies how snap points are aligned with items.
These properties are backed by BindableProperty objects, which means that the properties can be targets of
data bindings.
NOTE
When snapping occurs, it will occur in the direction that produces the least amount of motion.
IMPORTANT
The value of the SnapPointsAlignment property is only respected when the SnapPointsType property is set to
Mandatory , or MandatorySingle .
Start
The SnapPointsAlignment.Start member indicates that snap points are aligned with the leading edge of items.
The following XAML example shows how to set this enumeration member:
When a user swipes to initiate a scroll in a horizontally scrolling CarouselView , the left item will be aligned with
the left of the view:
Center
The SnapPointsAlignment.Center member indicates that snap points are aligned with the center of items.
By default on a CarouselView , the SnapPointsAlignment property is set to Center . However, for completeness,
the following XAML example shows how to set this enumeration member:
When a user swipes to initiate a scroll in a horizontally scrolling CarouselView , the center item will be aligned
with the center of the view:
End
The SnapPointsAlignment.End member indicates that snap points are aligned with the trailing edge of items. The
following XAML example shows how to set this enumeration member:
When a user swipes to initiate a scroll in a horizontally scrolling CarouselView , the right item will be aligned
with the right of the view.
CollectionView
3/18/2022 • 2 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) CollectionView is a view for presenting lists of data using different
layout specifications. It aims to provide a more flexible, and performant alternative to ListView .
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The following screenshot shows a CollectionView that uses a two column vertical grid, and which allows
multiple selection:
CollectionView should be used for presenting lists of data that require scrolling or selection. A bindable layout
can be used when the data to be displayed doesn't require scrolling or selection.
The .NET Multi-platform App UI (.NET MAUI) CollectionView includes the following properties that define the
data to be displayed, and its appearance:
ItemsSource , of type IEnumerable , specifies the collection of items to be displayed, and has a default value of
null .
ItemTemplate , of type DataTemplate , specifies the template to apply to each item in the collection of items to
be displayed.
These properties are backed by BindableProperty objects, which means that the properties can be targets of
data bindings.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
CollectionView defines a ItemsUpdatingScrollMode property that represents the scrolling behavior of the
CollectionView when new items are added to it. For more information about this property, see Control scroll
position when new items are added.
CollectionView supports incremental data virtualization as the user scrolls. For more information, see Load
data incrementally.
IMPORTANT
If the CollectionView is required to refresh as items are added, removed, or changed in the underlying collection, the
underlying collection should be an IEnumerable collection that sends property change notifications, such as
ObservableCollection .
CollectionView can be populated with data by using data binding to bind its ItemsSource property to an
IEnumerable collection. In XAML, this is achieved with the Binding markup extension:
NOTE
Compiled bindings can be enabled to improve data binding performance in .NET MAUI applications.
For information on how to change the CollectionView layout, see Specify CollectionView layout. For
information on how to define the appearance of each item in the CollectionView , see Define item appearance.
WARNING
CollectionView will throw an exception if its ItemsSource is updated off the UI thread.
Grid.SetRowSpan(image, 2);
grid.Add(image);
grid.Add(nameLabel, 1, 0);
grid.Add(locationLabel, 1, 1);
return grid;
});
The elements specified in the DataTemplate define the appearance of each item in the list. In the example, layout
within the DataTemplate is managed by a Grid . The Grid contains an Image object, and two Label objects,
that all bind to properties of the Monkey class:
The following screenshot shows the result of templating each item in the list:
Choose item appearance at runtime
The appearance of each item in the CollectionView can be chosen at runtime, based on the item value, by
setting the CollectionView.ItemTemplate property to a DataTemplateSelector object:
<ContentPage ...
xmlns:controls="clr-namespace:CollectionViewDemos.Controls">
<ContentPage.Resources>
<DataTemplate x:Key="AmericanMonkeyTemplate">
...
</DataTemplate>
<DataTemplate x:Key="OtherMonkeyTemplate">
...
</DataTemplate>
<controls:MonkeyDataTemplateSelector x:Key="MonkeySelector"
AmericanMonkey="{StaticResource AmericanMonkeyTemplate}"
OtherMonkey="{StaticResource OtherMonkeyTemplate}" />
</ContentPage.Resources>
The MonkeyDataTemplateSelector class defines AmericanMonkey and OtherMonkey DataTemplate properties that
are set to different data templates. The OnSelectTemplate override returns the AmericanMonkey template, which
displays the monkey name and location in teal, when the monkey name contains "America". When the monkey
name doesn't contain "America", the OnSelectTemplate override returns the OtherMonkey template, which
displays the monkey name and location in silver:
IMPORTANT
When using CollectionView , never set the root element of your DataTemplate objects to a ViewCell . This will result
in an exception being thrown because CollectionView has no concept of cells.
Context menus
CollectionView supports context menus for items of data through the SwipeView , which reveals the context
menu with a swipe gesture. The SwipeView is a container control that wraps around an item of content, and
provides context menu items for that item of content. Therefore, context menus are implemented for a
CollectionView by creating a SwipeView that defines the content that the SwipeView wraps around, and the
context menu items that are revealed by the swipe gesture. This is achieved by setting the SwipeView as the root
view in the DataTemplate that defines the appearance of each item of data in the CollectionView :
<CollectionView x:Name="collectionView"
ItemsSource="{Binding Monkeys}">
<CollectionView.ItemTemplate>
<DataTemplate>
<SwipeView>
<SwipeView.LeftItems>
<SwipeItems>
<SwipeItem Text="Favorite"
IconImageSource="favorite.png"
BackgroundColor="LightGreen"
Command="{Binding Source={x:Reference collectionView},
Path=BindingContext.FavoriteCommand}"
CommandParameter="{Binding}" />
<SwipeItem Text="Delete"
IconImageSource="delete.png"
BackgroundColor="LightPink"
Command="{Binding Source={x:Reference collectionView},
Path=BindingContext.DeleteCommand}"
CommandParameter="{Binding}" />
</SwipeItems>
</SwipeView.LeftItems>
<Grid BackgroundColor="White"
Padding="10">
<!-- Define item appearance -->
</Grid>
</SwipeView>
</DataTemplate>
</CollectionView.ItemTemplate>
</CollectionView>
In this example, the SwipeView content is a Grid that defines the appearance of each item in the
CollectionView . The swipe items are used to perform actions on the SwipeView content, and are revealed when
the control is swiped from the left side:
SwipeView supports four different swipe directions, with the swipe direction being defined by the directional
SwipeItems collection the SwipeItems objects are added to. By default, a swipe item is executed when it's tapped
by the user. In addition, once a swipe item has been executed the swipe items are hidden and the SwipeView
content is re-displayed. However, these behaviors can be changed.
Pull to refresh
CollectionView supports pull to refresh functionality through the RefreshView , which enables the data being
displayed to be refreshed by pulling down on the list of items. The RefreshView is a container control that
provides pull to refresh functionality to its child, provided that the child supports scrollable content. Therefore,
pull to refresh is implemented for a CollectionView by setting it as the child of a RefreshView :
When the user initiates a refresh, the ICommand defined by the Command property is executed, which should
refresh the items being displayed. A refresh visualization is shown while the refresh occurs, which consists of an
animated progress circle:
The value of the RefreshView.IsRefreshing property indicates the current state of the RefreshView . When a
refresh is triggered by the user, this property will automatically transition to true . Once the refresh completes,
you should reset the property to false .
CollectionView also defines a RemainingItemsThresholdReached event that is fired when the CollectionView is
scrolled far enough that RemainingItemsThreshold items have not been displayed. This event can be handled to
load more items. In addition, when the RemainingItemsThresholdReached event is fired, the
RemainingItemsThresholdReachedCommand is executed, enabling incremental data loading to take place in a
viewmodel.
The default value of the property is -1, which indicates that the
RemainingItemsThreshold
RemainingItemsThresholdReached event will never be fired. When the property value is 0, the
RemainingItemsThresholdReached event will be fired when the final item in the ItemsSource is displayed. For
values greater than 0, the RemainingItemsThresholdReached event will be fired when the ItemsSource contains
that number of items not yet scrolled to.
NOTE
CollectionView validates the RemainingItemsThreshold property so that its value is always greater than or equal to -
1.
The following XAML example shows a CollectionView that loads data incrementally:
In this code example, the RemainingItemsThresholdReached event fires when there are 5 items not yet scrolled to,
and in response executes the OnCollectionViewRemainingItemsThresholdReached event handler:
NOTE
Data can also be loaded incrementally by binding the RemainingItemsThresholdReachedCommand to an ICommand
implementation in the viewmodel.
Specify CollectionView layout
3/18/2022 • 11 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) CollectionView defines the following properties that control
layout:
ItemsLayout , of type IItemsLayout , specifies the layout to be used.
ItemSizingStrategy , of type ItemSizingStrategy , specifies the item measure strategy to be used.
These properties are backed by BindableProperty objects, which means that the properties can be targets of
data bindings.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
By default, a CollectionView will display its items in a vertical list. However, any of the following layouts can be
used:
Vertical list – a single column list that grows vertically as new items are added.
Horizontal list – a single row list that grows horizontally as new items are added.
Vertical grid – a multi-column grid that grows vertically as new items are added.
Horizontal grid – a multi-row grid that grows horizontally as new items are added.
These layouts can be specified by setting the ItemsLayout property to class that derives from the ItemsLayout
class. This class defines the following properties:
Orientation , of type ItemsLayoutOrientation , specifies the direction in which the CollectionView expands as
items are added.
SnapPointsAlignment , of type SnapPointsAlignment , specifies how snap points are aligned with items.
SnapPointsType , of type SnapPointsType , specifies the behavior of snap points when scrolling.
These properties are backed by BindableProperty objects, which means that the properties can be targets of
data bindings. For more information about snap points, see Snap points in Control scrolling in a CollectionView.
The ItemsLayoutOrientation enumeration defines the following members:
Vertical indicates that the CollectionView will expand vertically as items are added.
Horizontal indicates that the CollectionView will expand horizontally as items are added.
The LinearItemsLayout class inherits from the ItemsLayout class, and defines an ItemSpacing property, of type
double , that represents the empty space around each item. The default value of this property is 0, and its value
must always be greater than or equal to 0. The LinearItemsLayout class also defines static Vertical and
Horizontal members. These members can be used to create vertical or horizontal lists, respectively.
Alternatively, a LinearItemsLayout object can be created, specifying an ItemsLayoutOrientation enumeration
member as an argument.
The GridItemsLayout class inherits from the ItemsLayout class, and defines the following properties:
VerticalItemSpacing , of type double , that represents the vertical empty space around each item. The default
value of this property is 0, and its value must always be greater than or equal to 0.
HorizontalItemSpacing , of type double , that represents the horizontal empty space around each item. The
default value of this property is 0, and its value must always be greater than or equal to 0.
Span , of type int , that represents the number of columns or rows to display in the grid. The default value
of this property is 1, and its value must always be greater than or equal to 1.
These properties are backed by BindableProperty objects, which means that the properties can be targets of
data bindings.
NOTE
CollectionView uses the native layout engines to perform layout.
Vertical list
By default, CollectionView will display its items in a vertical list layout. Therefore, it's not necessary to set the
ItemsLayout property to use this layout:
However, for completeness, in XAML a CollectionView can be set to display its items in a vertical list by setting
its ItemsLayout property to VerticalList :
Alternatively, this can also be accomplished by setting the ItemsLayout property to a LinearItemsLayout object,
specifying the Vertical ItemsLayoutOrientation enumeration member as the Orientation property value:
This results in a single column list, which grows vertically as new items are added:
Horizontal list
In XAML, a CollectionView can display its items in a horizontal list by setting its ItemsLayout property to
HorizontalList :
<CollectionView ItemsSource="{Binding Monkeys}"
ItemsLayout="HorizontalList">
<CollectionView.ItemTemplate>
<DataTemplate>
<Grid Padding="10">
<Grid.RowDefinitions>
<RowDefinition Height="35" />
<RowDefinition Height="35" />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="70" />
<ColumnDefinition Width="140" />
</Grid.ColumnDefinitions>
<Image Grid.RowSpan="2"
Source="{Binding ImageUrl}"
Aspect="AspectFill"
HeightRequest="60"
WidthRequest="60" />
<Label Grid.Column="1"
Text="{Binding Name}"
FontAttributes="Bold"
LineBreakMode="TailTruncation" />
<Label Grid.Row="1"
Grid.Column="1"
Text="{Binding Location}"
LineBreakMode="TailTruncation"
FontAttributes="Italic"
VerticalOptions="End" />
</Grid>
</DataTemplate>
</CollectionView.ItemTemplate>
</CollectionView>
Alternatively, this layout can also be accomplished by setting the ItemsLayout property to a LinearItemsLayout
object, specifying the Horizontal ItemsLayoutOrientation enumeration member as the Orientation property
value:
This results in a single row list, which grows horizontally as new items are added:
Vertical grid
In XAML, a CollectionView can display its items in a vertical grid by setting its ItemsLayout property to
VerticalGrid :
Alternatively, this layout can also be accomplished by setting the ItemsLayout property to a GridItemsLayout
object whose Orientation property is set to Vertical :
By default, a vertical GridItemsLayout will display items in a single column. However, this example sets the
GridItemsLayout.Span property to 2. This results in a two-column grid, which grows vertically as new items are
added:
Horizontal grid
In XAML, a CollectionView can display its items in a horizontal grid by setting its ItemsLayout property to
HorizontalGrid :
<CollectionView ItemsSource="{Binding Monkeys}"
ItemsLayout="HorizontalGrid, 4">
<CollectionView.ItemTemplate>
<DataTemplate>
<Grid Padding="10">
<Grid.RowDefinitions>
<RowDefinition Height="35" />
<RowDefinition Height="35" />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="70" />
<ColumnDefinition Width="140" />
</Grid.ColumnDefinitions>
<Image Grid.RowSpan="2"
Source="{Binding ImageUrl}"
Aspect="AspectFill"
HeightRequest="60"
WidthRequest="60" />
<Label Grid.Column="1"
Text="{Binding Name}"
FontAttributes="Bold"
LineBreakMode="TailTruncation" />
<Label Grid.Row="1"
Grid.Column="1"
Text="{Binding Location}"
LineBreakMode="TailTruncation"
FontAttributes="Italic"
VerticalOptions="End" />
</Grid>
</DataTemplate>
</CollectionView.ItemTemplate>
</CollectionView>
Alternatively, this layout can also be accomplished by setting the ItemsLayout property to a GridItemsLayout
object whose Orientation property is set to Horizontal :
By default, a horizontal GridItemsLayout will display items in a single row. However, this example sets the
GridItemsLayout.Span property to 4. This results in a four-row grid, which grows horizontally as new items are
added:
Headers and footers
CollectionView can present a header and footer that scroll with the items in the list. The header and footer can
be strings, views, or DataTemplate objects.
CollectionView defines the following properties for specifying the header and footer:
Header , of type , specifies the string, binding, or view that will be displayed at the start of the list.
object
HeaderTemplate , of type DataTemplate , specifies the DataTemplate to use to format the Header .
Footer , of type object , specifies the string, binding, or view that will be displayed at the end of the list.
FooterTemplate , of type DataTemplate , specifies the DataTemplate to use to format the Footer .
These properties are backed by BindableProperty objects, which means that the properties can be targets of
data bindings.
When a header is added to a layout that grows horizontally, from left to right, the header is displayed to the left
of the list. Similarly, when a footer is added to a layout that grows horizontally, from left to right, the footer is
displayed to the right of the list.
Display strings in the header and footer
The Header and Footer properties can be set to string values, as shown in the following example:
This code results in the following screenshots, with the header shown in the iOS screenshot, and the footer
shown in the Android screenshot:
This code results in the following screenshots, with the header shown in the iOS screenshot, and the footer
shown in the Android screenshot:
This code results in the following screenshots, with the header shown in the iOS screenshot, and the footer
shown in the Android screenshot:
Item spacing
By default, there is no space between each item in a CollectionView . This behavior can be changed by setting
properties on the items layout used by the CollectionView .
When a CollectionView sets its ItemsLayout property to a LinearItemsLayout object, the
LinearItemsLayout.ItemSpacing property can be set to a double value that represents the space between items:
NOTE
The LinearItemsLayout.ItemSpacing property has a validation callback set, which ensures that the value of the
property is always greater than or equal to 0.
NOTE
The GridItemsLayout.VerticalItemSpacing and GridItemsLayout.HorizontalItemSpacing properties have
validation callbacks set, which ensure that the values of the properties are always greater than or equal to 0.
This code results in a vertical two-column grid, that has a vertical spacing of 20 between items, and a horizontal
spacing of 30 between items:
Item sizing
By default, each item in a CollectionView is individually measured and sized, provided that the UI elements in
the DataTemplate don't specify fixed sizes. This behavior, which can be changed, is specified by the
CollectionView.ItemSizingStrategy property value. This property value can be set to one of the
ItemSizingStrategy enumeration members:
IMPORTANT
The MeasureFirstItem sizing strategy will result in increased performance when used in situations where the item size is
intended to be uniform across all items.
The following code example shows setting the ItemSizingStrategy property:
<CollectionView ...
ItemSizingStrategy="MeasureFirstItem">
...
</CollectionView>
The OnImageTapped event handler is executed in response to an Image object being tapped, and changes the
dimensions of the image so that it's more easily viewed:
Right-to-left layout
CollectionView can layout its content in a right-to-left flow direction by setting its FlowDirection property to
RightToLeft . However, the FlowDirection property should ideally be set on a page or root layout, which causes
all the elements within the page, or root layout, to respond to the flow direction:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="CollectionViewDemos.Views.VerticalListFlowDirectionPage"
Title="Vertical list (RTL FlowDirection)"
FlowDirection="RightToLeft">
<StackLayout Margin="20">
<CollectionView ItemsSource="{Binding Monkeys}">
...
</CollectionView>
</StackLayout>
</ContentPage>
The default FlowDirection for an element with a parent is MatchParent . Therefore, the CollectionView inherits
the FlowDirection property value from the StackLayout , which in turn inherits the FlowDirection property
value from the ContentPage . This results in the right-to-left layout shown in the following screenshot:
Configure CollectionView item selection
3/18/2022 • 6 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) CollectionView defines the following properties that control item
selection:
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
All of these properties are backed by BindableProperty objects, which means that the properties can be targets
of data bindings.
By default, CollectionView selection is disabled. However, this behavior can be changed by setting the
SelectionMode property value to one of the SelectionMode enumeration members:
None – indicates that items cannot be selected. This is the default value.
Single – indicates that a single item can be selected, with the selected item being highlighted.
Multiple – indicates that multiple items can be selected, with the selected items being highlighted.
CollectionView defines a SelectionChanged event that is fired when the SelectedItem property changes, either
due to the user selecting an item from the list, or when an application sets the property. In addition, this event is
also fired when the SelectedItems property changes. The SelectionChangedEventArgs object that accompanies
the SelectionChanged event has two properties, both of type IReadOnlyList<object> :
PreviousSelection – the list of items that were selected, before the selection changed.
CurrentSelection – the list of items that are selected, after the selection change.
In addition, CollectionView has a UpdateSelectedItems method that updates the SelectedItems property with a
list of selected items, while only firing a single change notification.
Single selection
When the SelectionMode property is set to Single , a single item in the CollectionView can be selected. When
an item is selected, the SelectedItem property will be set to the value of the selected item. When this property
changes, the SelectionChangedCommand is executed (with the value of the SelectionChangedCommandParameter
being passed to the ICommand , and the SelectionChanged event fires.
The following XAML example shows a CollectionView that can respond to single item selection:
In this code example, the OnCollectionViewSelectionChanged event handler is executed when the
SelectionChanged event fires, with the event handler retrieving the previously selected item, and the current
selected item:
IMPORTANT
The SelectionChanged event can be fired by changes that occur as a result of changing the SelectionMode property.
In this code example, the OnCollectionViewSelectionChanged event handler is executed when the
SelectionChanged event fires, with the event handler retrieving the previously selected items, and the current
selected items:
void OnCollectionViewSelectionChanged(object sender, SelectionChangedEventArgs e)
{
var previous = e.PreviousSelection;
var current = e.CurrentSelection;
...
}
IMPORTANT
The SelectionChanged event can be fired by changes that occur as a result of changing the SelectionMode property.
Single pre-selection
When the SelectionMode property is set to Single , a single item in the CollectionView can be pre-selected by
setting the SelectedItem property to the item. The following XAML example shows a CollectionView that pre-
selects a single item:
The SelectedItem property data binds to the SelectedMonkey property of the connected view model, which is of
type Monkey . By default, a TwoWay binding is used so that if the user changes the selected item, the value of the
SelectedMonkey property will be set to the selected Monkey object. The SelectedMonkey property is defined in
the MonkeysViewModel class, and is set to the fourth item of the Monkeys collection:
Monkey selectedMonkey;
public Monkey SelectedMonkey
{
get
{
return selectedMonkey;
}
set
{
if (selectedMonkey != value)
{
selectedMonkey = value;
}
}
}
public MonkeysViewModel()
{
...
selectedMonkey = Monkeys.Skip(3).FirstOrDefault();
}
...
}
Therefore, when the CollectionView appears, the fourth item in the list is pre-selected:
Multiple pre-selection
When the SelectionMode property is set to Multiple , multiple items in the CollectionView can be pre-selected.
The following XAML example shows a CollectionView that will enable the pre-selection of multiple items:
<CollectionView x:Name="collectionView"
ItemsSource="{Binding Monkeys}"
SelectionMode="Multiple"
SelectedItems="{Binding SelectedMonkeys}">
...
</CollectionView>
NOTE
The SelectedItems property has a default binding mode of OneWay .
The SelectedItems property data binds to the SelectedMonkeys property of the connected view model, which is
of type ObservableCollection<object> . The SelectedMonkeys property is defined in the MonkeysViewModel class,
and is set to the second, fourth, and fifth items in the Monkeys collection:
namespace CollectionViewDemos.ViewModels
{
public class MonkeysViewModel : INotifyPropertyChanged
{
...
ObservableCollection<object> selectedMonkeys;
public ObservableCollection<object> SelectedMonkeys
{
get
{
return selectedMonkeys;
}
set
{
if (selectedMonkeys != value)
{
selectedMonkeys = value;
}
}
}
public MonkeysViewModel()
{
...
SelectedMonkeys = new ObservableCollection<object>()
{
Monkeys[1], Monkeys[3], Monkeys[4]
};
}
...
}
}
Therefore, when the CollectionView appears, the second, fourth, and fifth items in the list are pre-selected:
Clear selections
The SelectedItem and SelectedItems properties can be cleared by setting them, or the objects they bind to, to
null .
<ContentPage ...>
<ContentPage.Resources>
<Style TargetType="Grid">
<Setter Property="VisualStateManager.VisualStateGroups">
<VisualStateGroupList>
<VisualStateGroup x:Name="CommonStates">
<VisualState x:Name="Normal" />
<VisualState x:Name="Selected">
<VisualState.Setters>
<Setter Property="BackgroundColor"
Value="LightSkyBlue" />
</VisualState.Setters>
</VisualState>
</VisualStateGroup>
</VisualStateGroupList>
</Setter>
</Style>
</ContentPage.Resources>
<StackLayout Margin="20">
<CollectionView ItemsSource="{Binding Monkeys}"
SelectionMode="Single">
<CollectionView.ItemTemplate>
<DataTemplate>
<Grid Padding="10">
...
</Grid>
</DataTemplate>
</CollectionView.ItemTemplate>
</CollectionView>
</StackLayout>
</ContentPage>
IMPORTANT
The Style that contains the Selected VisualState must have a TargetType property value that's the type of the
root element of the DataTemplate , which is set as the ItemTemplate property value.
In this example, the Style.TargetType property value is set to Grid because the root element of the
ItemTemplate is a Grid . The Selected VisualState specifies that when an item in the CollectionView is
selected, the BackgroundColor of the item will be set to LightSkyBlue :
Disable selection
CollectionView selection is disabled by default. However, if a CollectionView has selection enabled, it can be
disabled by setting the SelectionMode property to None :
<CollectionView ...
SelectionMode="None" />
When the SelectionMode property is set to None , items in the CollectionView cannot be selected, the
SelectedItem property will remain null , and the SelectionChanged event will not be fired.
NOTE
When an item has been selected and the SelectionMode property is changed from Single to None , the
SelectedItem property will be set to null and the SelectionChanged event will be fired with an empty
CurrentSelection property.
Define an EmptyView for a CollectionView
3/18/2022 • 6 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) CollectionView defines the following properties that can be used
to provide user feedback when there's no data to display:
EmptyView , of type object , the string, binding, or view that will be displayed when the ItemsSource
property is null , or when the collection specified by the ItemsSource property is null or empty. The
default value is null .
EmptyViewTemplate , of type DataTemplate , the template to use to format the specified EmptyView . The default
value is null .
These properties are backed by BindableProperty objects, which means that the properties can be targets of
data bindings.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The main usage scenarios for setting the EmptyView property are displaying user feedback when a filtering
operation on a CollectionView yields no data, and displaying user feedback while data is being retrieved from a
web service.
NOTE
The EmptyView property can be set to a view that includes interactive content if required.
The result is that, because the data bound collection is null , the string set as the EmptyView property value is
displayed:
Display views when data is unavailable
The EmptyView property can be set to a view, which will be displayed when the ItemsSource property is null ,
or when the collection specified by the ItemsSource property is null or empty. This can be a single view, or a
view that contains multiple child views. The following XAML example shows the EmptyView property set to a
view that contains multiple child views:
<StackLayout Margin="20">
<SearchBar x:Name="searchBar"
SearchCommand="{Binding FilterCommand}"
SearchCommandParameter="{Binding Source={x:Reference searchBar}, Path=Text}"
Placeholder="Filter" />
<CollectionView ItemsSource="{Binding Monkeys}">
<CollectionView.ItemTemplate>
<DataTemplate>
...
</DataTemplate>
</CollectionView.ItemTemplate>
<CollectionView.EmptyView>
<ContentView>
<StackLayout HorizontalOptions="CenterAndExpand"
VerticalOptions="CenterAndExpand">
<Label Text="No results matched your filter."
Margin="10,25,10,10"
FontAttributes="Bold"
FontSize="18"
HorizontalOptions="Fill"
HorizontalTextAlignment="Center" />
<Label Text="Try a broader filter?"
FontAttributes="Italic"
FontSize="12"
HorizontalOptions="Fill"
HorizontalTextAlignment="Center" />
</StackLayout>
</ContentView>
</CollectionView.EmptyView>
</CollectionView>
</StackLayout>
In this example, what looks like a redundant has been added as the root element of the EmptyView . This is
because internally, the EmptyView is added to a native container that doesn't provide any context for .NET MAUI
layout. Therefore, to position the views that comprise your EmptyView , you must add a root layout, whose child
is a layout that can position itself within the root layout.
The equivalent C# code is:
StackLayout stackLayout = new StackLayout();
stackLayout.Add(new Label { Text = "No results matched your filter.", ... } );
stackLayout.Add(new Label { Text = "Try a broader filter?", ... } );
When the SearchBar executes the FilterCommand , the collection displayed by the CollectionView is filtered for
the search term stored in the SearchBar.Text property. If the filtering operation yields no data, the StackLayout
set as the EmptyView property value is displayed:
The property is set to a FilterData object, and the Filter property data binds to the
EmptyView
SearchBar.Text property. When the SearchBar executes the FilterCommand , the collection displayed by the
CollectionView is filtered for the search term stored in the Filter property. If the filtering operation yields no
data, the Label defined in the DataTemplate , that's set as the EmptyViewTemplate property value, is displayed:
NOTE
When displaying a templated custom type when data is unavailable, the EmptyViewTemplate property can be set to a
view that contains multiple child views.
<StackLayout Margin="20">
<SearchBar x:Name="searchBar"
SearchCommand="{Binding FilterCommand}"
SearchCommandParameter="{Binding Source={x:Reference searchBar}, Path=Text}"
Placeholder="Filter" />
<StackLayout Orientation="Horizontal">
<Label Text="Toggle EmptyViews" />
<Switch Toggled="OnEmptyViewSwitchToggled" />
</StackLayout>
<CollectionView x:Name="collectionView"
ItemsSource="{Binding Monkeys}">
<CollectionView.ItemTemplate>
<DataTemplate>
...
</DataTemplate>
</CollectionView.ItemTemplate>
</CollectionView>
</StackLayout>
</ContentPage>
This XAML defines two ContentView objects in the page-level ResourceDictionary , with the Switch object
controlling which ContentView object will be set as the EmptyView property value. When the Switch is toggled,
the OnEmptyViewSwitchToggled event handler executes the ToggleEmptyView method:
The ToggleEmptyView method sets the EmptyView property of the collectionView object to one of the two
ContentView objects stored in the ResourceDictionary , based on the value of the Switch.IsToggled property.
When the SearchBar executes the FilterCommand , the collection displayed by the CollectionView is filtered for
the search term stored in the SearchBar.Text property. If the filtering operation yields no data, the ContentView
object set as the EmptyView property is displayed:
<ContentPage ...
xmlns:controls="clr-namespace:CollectionViewDemos.Controls">
<ContentPage.Resources>
<DataTemplate x:Key="AdvancedTemplate">
...
</DataTemplate>
<DataTemplate x:Key="BasicTemplate">
...
</DataTemplate>
<controls:SearchTermDataTemplateSelector x:Key="SearchSelector"
DefaultTemplate="{StaticResource AdvancedTemplate}"
OtherTemplate="{StaticResource BasicTemplate}" />
</ContentPage.Resources>
<StackLayout Margin="20">
<SearchBar x:Name="searchBar"
SearchCommand="{Binding FilterCommand}"
SearchCommandParameter="{Binding Source={x:Reference searchBar}, Path=Text}"
Placeholder="Filter" />
<CollectionView ItemsSource="{Binding Monkeys}"
EmptyView="{Binding Source={x:Reference searchBar}, Path=Text}"
EmptyViewTemplate="{StaticResource SearchSelector}" />
</StackLayout>
</ContentPage>
The EmptyViewproperty is set to the SearchBar.Text property, and the EmptyViewTemplate property is set to a
SearchTermDataTemplateSelector object.
When the SearchBar executes the FilterCommand , the collection displayed by the CollectionView is filtered for
the search term stored in the SearchBar.Text property. If the filtering operation yields no data, the
DataTemplate chosen by the SearchTermDataTemplateSelector object is set as the EmptyViewTemplate property
and displayed.
The following example shows the SearchTermDataTemplateSelector class:
The .NET Multi-platform App UI (.NET MAUI) CollectionView defines two ScrollTo methods, that scroll items
into view. One of the overloads scrolls the item at the specified index into view, while the other scrolls the
specified item into view. Both overloads have additional arguments that can be specified to indicate the group
the item belongs to, the exact position of the item after the scroll has completed, and whether to animate the
scroll.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
CollectionView defines a event that is fired when one of the ScrollTo methods is invoked.
ScrollToRequested
The ScrollToRequestedEventArgs object that accompanies the ScrollToRequested event has many properties,
including IsAnimated , Index , Item , and ScrollToPosition . These properties are set from the arguments
specified in the ScrollTo method calls.
In addition, CollectionViewdefines a Scrolled event that is fired to indicate that scrolling occurred. The
ItemsViewScrolledEventArgs object that accompanies the Scrolled event has many properties. For more
information, see Detect scrolling.
CollectionView also defines a ItemsUpdatingScrollMode property that represents the scrolling behavior of the
CollectionView when new items are added to it. For more information about this property, see Control scroll
position when new items are added.
When a user swipes to initiate a scroll, the end position of the scroll can be controlled so that items are fully
displayed. This feature is known as snapping, because items snap to position when scrolling stops. For more
information, see Snap points.
CollectionView can also load data incrementally as the user scrolls. For more information, see Load data
incrementally.
Detect scrolling
CollectionView defines a Scrolled event which is fired to indicate that scrolling occurred. The
ItemsViewScrolledEventArgs class, which represents the object that accompanies the Scrolled event, defines
the following properties:
HorizontalDelta , of type double , represents the change in the amount of horizontal scrolling. This is a
negative value when scrolling left, and a positive value when scrolling right.
VerticalDelta , of type double , represents the change in the amount of vertical scrolling. This is a negative
value when scrolling upwards, and a positive value when scrolling downwards.
HorizontalOffset , of type double , defines the amount by which the list is horizontally offset from its origin.
VerticalOffset , of type double , defines the amount by which the list is vertically offset from its origin.
FirstVisibleItemIndex , of type int , is the index of the first item that's visible in the list.
CenterItemIndex , of type int , is the index of the the center item that's visible in the list.
LastVisibleItemIndex , of type int , is the index of the last item that's visible in the list.
The following XAML example shows a CollectionView that sets an event handler for the Scrolled event:
<CollectionView Scrolled="OnCollectionViewScrolled">
...
</CollectionView>
In this code example, the OnCollectionViewScrolled event handler is executed when the Scrolled event fires:
IMPORTANT
The Scrolled event is fired for user initiated scrolls, and for programmatic scrolls.
collectionView.ScrollTo(12);
Alternatively, an item in grouped data can be scrolled into view by specifying the item and group indexes. The
following example shows how to scroll the third item in the second group into view:
NOTE
The ScrollToRequested event is fired when the ScrollTo method is invoked.
NOTE
The ScrollToRequested event is fired when the ScrollTo method is invoked.
This example code results in the minimal scrolling required to scroll the item into view:
NOTE
The ScrollToPosition.MakeVisible member is used by default, if the position argument is not specified when
calling the ScrollTo method.
Start
The ScrollToPosition.Start member indicates that the item should be scrolled to the start of the view:
This example code results in the item being scrolled to the start of the view:
Center
The ScrollToPosition.Center member indicates that the item should be scrolled to the center of the view:
This example code results in the item being scrolled to the center of the view:
End
The ScrollToPosition.End member indicates that the item should be scrolled to the end of the view:
collectionView.ScrollTo(monkey, position: ScrollToPosition.End);
This example code results in the item being scrolled to the end of the view:
<CollectionView ItemsUpdatingScrollMode="KeepLastItemInView">
...
</CollectionView>
Snap points
When a user swipes to initiate a scroll, the end position of the scroll can be controlled so that items are fully
displayed. This feature is known as snapping, because items snap to position when scrolling stops, and is
controlled by the following properties from the ItemsLayout class:
SnapPointsType , of type SnapPointsType , specifies the behavior of snap points when scrolling.
SnapPointsAlignment , of type SnapPointsAlignment , specifies how snap points are aligned with items.
These properties are backed by BindableProperty objects, which means that the properties can be targets of
data bindings.
NOTE
When snapping occurs, it will occur in the direction that produces the least amount of motion.
By default, the SnapPointsType property is set to SnapPointsType.None , which ensures that scrolling does not
snap items, as shown in the following screenshot:
Snap points alignment
The SnapPointsAlignment enumeration defines Start , Center , and End members.
IMPORTANT
The value of the SnapPointsAlignment property is only respected when the SnapPointsType property is set to
Mandatory , or MandatorySingle .
Start
The SnapPointsAlignment.Start member indicates that snap points are aligned with the leading edge of items.
By default, the SnapPointsAlignment property is set to SnapPointsAlignment.Start . However, for completeness,
the following XAML example shows how to set this enumeration member:
When a user swipes to initiate a scroll, the top item will be aligned with the top of the view:
Center
The SnapPointsAlignment.Center member indicates that snap points are aligned with the center of items. The
following XAML example shows how to set this enumeration member:
When a user swipes to initiate a scroll, the top item will be center aligned at the top of the view:
End
The SnapPointsAlignment.End member indicates that snap points are aligned with the trailing edge of items. The
following XAML example shows how to set this enumeration member:
When a user swipes to initiate a scroll, the bottom item will be aligned with the bottom of the view:
Display grouped data in a CollectionView
3/18/2022 • 5 minutes to read • Edit Online
Large data sets can often become unwieldy when presented in a continually scrolling list. In this scenario,
organizing the data into groups can improve the user experience by making it easier to navigate the data.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The .NET Multi-platform App UI (.NET MAUI) CollectionView supports displaying grouped data, and defines the
following properties that control how it will be presented:
IsGrouped , of type bool , indicates whether the underlying data should be displayed in groups. The default
value of this property is false .
GroupHeaderTemplate , of type DataTemplate , the template to use for the header of each group.
GroupFooterTemplate , of type DataTemplate , the template to use for the footer of each group.
These properties are backed by BindableProperty objects, which means that the properties can be targets of
data bindings.
The following screenshot shows a CollectionView displaying grouped data:
Group data
Data must be grouped before it can be displayed. This can be accomplished by creating a list of groups, where
each group is a list of items. The list of groups should be an IEnumerable<T> collection, where T defines two
pieces of data:
A group name.
An IEnumerable collection that defines the items belonging to the group.
The process for grouping data, therefore, is to:
Create a type that models a single item.
Create a type that models a single group of items.
Create an IEnumerable<T> collection, where T is the type that models a single group of items. This collection
is a collection of groups, which stores the grouped data.
Add data to the IEnumerable<T> collection.
Example
When grouping data, the first step is to create a type that models a single item. The following example shows
the Animal class from the sample application:
The Animal class models a single item. A type that models a group of items can then be created. The following
example shows the AnimalGroup class from the sample application:
The AnimalGroup class inherits from the List<T> class and adds a Name property that represents the group
name.
An IEnumerable<T> collection of groups can then be created:
This code defines a collection named Animals , where each item in the collection is an AnimalGroup object. Each
AnimalGroup object comprises a name, and a List<Animal> collection that defines the Animal objects in the
group.
Grouped data can then be added to the Animals collection:
Animals.Add(new AnimalGroup("Bears", new List<Animal>
{
new Animal
{
Name = "American Black Bear",
Location = "North America",
Details = "Details about the bear go here.",
ImageUrl = "https://upload.wikimedia.org/wikipedia/commons/0/08/01_Schwarzbär.jpg"
},
new Animal
{
Name = "Asian Black Bear",
Location = "Asia",
Details = "Details about the bear go here.",
ImageUrl =
"https://upload.wikimedia.org/wikipedia/commons/thumb/b/b7/Ursus_thibetanus_3_%28Wroclaw_zoo%29.JPG/180px-
Ursus_thibetanus_3_%28Wroclaw_zoo%29.JPG"
},
// ...
}));
This code creates two groups in the Animals collection. The first AnimalGroup is named Bears , and contains a
List<Animal> collection of bear details. The second AnimalGroup is named Monkeys , and contains a
List<Animal> collection of monkey details.
The appearance of each item in the CollectionView is defined by setting the CollectionView.ItemTemplate
property to a DataTemplate . For more information, see Define item appearance.
NOTE
By default, CollectionView will display the group name in the group header and footer. This behavior can be changed
by customizing the group header and group footer.
In this example, each group footer is set to a Label that displays the number of items in the group. The
following screenshot shows the customized group footer:
Empty groups
When a CollectionView displays grouped data, it will display any groups that are empty. Such groups will be
displayed with a group header and footer, indicating that the group is empty. The following screenshot shows an
empty group:
NOTE
On iOS 10, group headers and footers for empty groups may all be displayed at the top of the CollectionView .
In this scenario, meaningful data can be displayed by overriding the ToString method in the type that models a
single item, and the type that models a single group of items.
IndicatorView
3/18/2022 • 4 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) IndicatorView is a control that displays indicators that represent
the number of items, and current position, in a CarouselView :
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
Create an IndicatorView
To add indicators to a page, create an IndicatorView object and set its IndicatorColor and
SelectedIndicatorColor properties. In addition, set the CarouselView.IndicatorView property to the name of the
IndicatorView object.
<StackLayout>
<CarouselView ItemsSource="{Binding Monkeys}"
IndicatorView="indicatorView">
<CarouselView.ItemTemplate>
<!-- DataTemplate that defines item appearance -->
</CarouselView.ItemTemplate>
</CarouselView>
<IndicatorView x:Name="indicatorView"
IndicatorColor="LightGray"
SelectedIndicatorColor="DarkGray"
HorizontalOptions="Center" />
</StackLayout>
In this example, the IndicatorView is rendered beneath the CarouselView , with an indicator for each item in the
CarouselView . The IndicatorView is populated with data by setting the CarouselView.IndicatorView property to
the IndicatorView object. Each indicator is a light gray circle, while the indicator that represents the current item
in the CarouselView is dark gray.
IMPORTANT
Setting the CarouselView.IndicatorView property results in the IndicatorView.Position property binding to the
CarouselView.Position property, and the IndicatorView.ItemsSource property binding to the
CarouselView.ItemsSource property.
<IndicatorView x:Name="indicatorView"
IndicatorsShape="Square"
IndicatorColor="LightGray"
SelectedIndicatorColor="DarkGray" />
<IndicatorView x:Name="indicatorView"
MaximumVisible="6" />
<StackLayout>
<CarouselView ItemsSource="{Binding Monkeys}"
IndicatorView="indicatorView">
<CarouselView.ItemTemplate>
<!-- DataTemplate that defines item appearance -->
</CarouselView.ItemTemplate>
</CarouselView>
<IndicatorView x:Name="indicatorView"
Margin="0,0,0,40"
IndicatorColor="Transparent"
SelectedIndicatorColor="Transparent"
HorizontalOptions="Center">
<IndicatorView.IndicatorTemplate>
<DataTemplate>
<Label Text=""
FontFamily="ionicons"
FontSize="12" />
</DataTemplate>
</IndicatorView.IndicatorTemplate>
</IndicatorView>
</StackLayout>
The elements specified in the DataTemplate define the appearance of each indicator. In this example, each
indicator is a Label that displays a font icon.
The following screenshot shows indicators rendered using a font icon:
Set visual states
IndicatorView has a Selected visual state that can be used to initiate a visual change to the indicator for the
current position in the IndicatorView . A common use case for this VisualState is to change the color of the
indicator that represents the current position:
<ContentPage ...>
<ContentPage.Resources>
<Style x:Key="IndicatorLabelStyle"
TargetType="Label">
<Setter Property="VisualStateManager.VisualStateGroups">
<VisualStateGroupList>
<VisualStateGroup x:Name="CommonStates">
<VisualState x:Name="Normal">
<VisualState.Setters>
<Setter Property="TextColor"
Value="LightGray" />
</VisualState.Setters>
</VisualState>
<VisualState x:Name="Selected">
<VisualState.Setters>
<Setter Property="TextColor"
Value="Black" />
</VisualState.Setters>
</VisualState>
</VisualStateGroup>
</VisualStateGroupList>
</Setter>
</Style>
</ContentPage.Resources>
<StackLayout>
...
<IndicatorView x:Name="indicatorView"
Margin="0,0,0,40"
IndicatorColor="Transparent"
SelectedIndicatorColor="Transparent"
HorizontalOptions="Center">
<IndicatorView.IndicatorTemplate>
<DataTemplate>
<Label Text=""
FontFamily="ionicons"
FontSize="12"
Style="{StaticResource IndicatorLabelStyle}" />
</DataTemplate>
</IndicatorView.IndicatorTemplate>
</IndicatorView>
</StackLayout>
</ContentPage>
In this example, the Selected visual state specifies that the indicator that represents the current position will
have its TextColor set to black. Otherwise the TextColor of the indicator will be light gray:
ListView
3/18/2022 • 27 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) ListView displays a scrollable vertical list of selectable data items.
While ListView manages the appearance of the list, the appearance of each item in the list is defined by a
DataTemplate that uses a Cell to display items. .NET MAUI includes cell types to display combinations of text
and images, and you can also define custom cells that display any content you want. ListView also includes
support for displaying headers and footers, grouped data, pull-to-refresh, and context menu items.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The ListView class derives from the ItemsView<Cell> class, from which it inherits the following properties:
ItemsSource , of type IEnumerable , specifies the collection of items to be displayed, and has a default value of
null .
ItemTemplate , of type DataTemplate , specifies the template to apply to each item in the collection of items to
be displayed.
ListView defines the following properties:
Footer , of type , specifies the string or view that will be displayed at the end of the list.
object
FooterTemplate , of type DataTemplate , specifies the DataTemplate to use to format the Footer .
GroupHeaderTemplate , of type DataTemplate , defines the DataTemplate used to define the appearance of the
header of each group.
HasUnevenRows , of type bool , indicates whether items in the list can have rows of different heights. The
default value of this property is false .
Header , of type object , specifies the string or view that will be displayed at the start of the list.
HeaderTemplate , of type DataTemplate , specifies the DataTemplate to use to format the Header .
HorizontalScrollBarVisibility , of type ScrollBarVisibility , indicates when the horizontal scroll bar will be
visible.
IsGroupedEnabled , of type bool , indicates whether the underlying data should be displayed in groups. The
default value of this property is false .
IsPullToRefreshEnabled , of type bool , indicates whether the user can swipe down to cause the ListView to
refresh its data. The default value of this property is false .
IsRefreshing , of type bool , indicates whether the ListView is currently refreshing. The default value of this
property is false .
RefreshCommand , of type ICommand , represents the command that will be executed when a refresh is
triggered.
RefreshControlColor , of type Color , determines the color of the refresh visualization that's shown while a
refresh occurs.
RowHeight , of type int , determines the height of each row when HasUnevenRows is false .
SelectedItem , of type object , represents the currently selected item in the ListView .
SelectionMode , of type ListViewSelectionMode , indicates whether items can be selected in the ListView or
not. The default value of this property is Single .
SeparatorColor , of type Color , defines the color of the bar that separates items in the list.
SeparatorVisibility , of type SeparatorVisibility , defines whether separators are visible between items.
VerticalScrollBarVisibility , of type ScrollBarVisibility , indicates when the vertical scroll bar will be
visible.
All of these properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
In addition, ListView defines the following properties that aren't backed by BindableProperty objects:
GroupDisplayBinding , of type BindingBase , the binding to use for displaying the group header.
GroupShortNameBinding , of type BindingBase , the binding for the name to display in grouped jump lists.
CachingStrategy , of type ListViewCachingStrategy , defines the cell reuse strategy of the ListView . This is a
read-only property.
ListView defines the following events:
ItemAppearing , which is raised when the visual representation of an item is being added to the visual layout
of the ListView . The ItemVisibilityEventArgs object that accompanies this event defines Item and Index
properties.
ItemDisappearing , which is raised when the visual representation of an item is being removed from the
visual layout of the ListView . The ItemVisibilityEventArgs object that accompanies this event defines Item
and Index properties.
ItemSelected , which is raised when a new item in the list is selected. The SelectedItemChangedEventArgs
object that accompanies this event defines SelectedItem and SelectedItemIndex properties.
ItemTapped , which raised when an item in the ListView is tapped. The ItemTappedEventArgs object that
accompanies this event defines Group , Item , and ItemIndex properties.
Refreshing , which is raised when a pull to refresh operation is triggered on the ListView .
Scrolled , . The ScrolledEventArgs object that accompanies this event defines ScrollX and ScrollY
properties.
ScrollToRequested . The ScrollToRequestedEventArgs object that accompanies this event defines Element ,
Mode , Position , ScrollX , ScrollY , and ShouldAnimate properties.
IMPORTANT
If the ListView is required to refresh as items are added, removed, or changed in the underlying collection, the
underlying collection should be an IEnumerable collection that sends property change notifications, such as
ObservableCollection .
ListView can be populated with data by using data binding to bind its ItemsSource property to an
IEnumerable collection. In XAML, this is achieved with the Binding markup extension:
In this example, the ItemsSource property data binds to the Monkeys property of the connected viewmodel.
NOTE
Compiled bindings can be enabled to improve data binding performance in .NET MAUI applications. For more
information, see Compiled bindings.
The elements specified in the DataTemplate define the appearance of each item in the list, and the child of the
DataTemplate must be a Cell object. In the example, layout within the DataTemplate is managed by a Grid .
The Grid contains an Image object, and two Label objects, that all bind to properties of the Monkey class:
public class Monkey
{
public string Name { get; set; }
public string Location { get; set; }
public string Details { get; set; }
public string ImageUrl { get; set; }
}
The following screenshot shows the result of templating each item in the list:
Typically, SwitchCell and EntryCell will only be used in a TableView and won't be used in a ListView . For
more information about SwitchCell and EntryCell , see TableView.
Text cell
A TextCell displays primary and secondary text on separate lines. TextCell defines the following properties:
Text , of type string , defines the primary text to be displayed.
TextColor , of type Color , represents the color of the primary text.
Detail , of type string , defines the secondary text to be displayed.
DetailColor , of type Color , indicates the color of the secondary text.
Command , of type ICommand , defines the command that's executed when the cell is tapped.
CommandParameter , of type object , represents the parameter that's passed to the command.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
The following example shows using a TextCell to define the appearance of items in a ListView :
Image cell
An ImageCell displays an image with primary and secondary text on separate lines. ImageCell inherits the
properties from TextCell , and defines the ImageSource property, of type ImageSource , which specifies the
image to be displayed in the cell. This property is backed by a BindableProperty object, which means it can be
the target of data bindings, and be styled.
The following example shows using an ImageCell to define the appearance of items in a ListView :
View cell
A ViewCell is a custom cell whose appearance is defined by a View . ViewCell defines a View property, of
type View , which defines the view that represents the content of the cell. This property is backed by a
BindableProperty object, which means it can be the target of data bindings, and be styled.
NOTE
The View property is the content property of the ViewCell class, and therefore does not need to be explicitly set from
XAML.
The following example shows using a ViewCell to define the appearance of items in a ListView :
Inside the ViewCell , layout can be managed by any .NET MAUI layout. In this example, layout is managed by a
Grid . The Grid contains an Image object, and two Label objects, that all bind to properties of the Monkey
class.
The following screenshot shows the result of templating each item in the list:
Choose item appearance at runtime
The appearance of each item in the ListView can be chosen at runtime, based on the item value, by setting the
ItemTemplate property to a DataTemplateSelector object:
<ContentPage ...
xmlns:templates="clr-namespace:ListViewDemos.Templates">
<ContentPage.Resources>
<DataTemplate x:Key="AmericanMonkeyTemplate">
<ViewCell>
...
</ViewCell>
</DataTemplate>
<DataTemplate x:Key="OtherMonkeyTemplate">
<ViewCell>
...
</ViewCell>
</DataTemplate>
<templates:MonkeyDataTemplateSelector x:Key="MonkeySelector"
AmericanMonkey="{StaticResource AmericanMonkeyTemplate}"
OtherMonkey="{StaticResource OtherMonkeyTemplate}" />
</ContentPage.Resources>
<StackLayout Margin="20">
<ListView ItemsSource="{Binding Monkeys}"
ItemTemplate="{StaticResource MonkeySelector}" />
</StackLayout>
</ContentPage>
The ItemTemplateproperty is set to a MonkeyDataTemplateSelector object. The following example shows the
MonkeyDataTemplateSelector class:
public class MonkeyDataTemplateSelector : DataTemplateSelector
{
public DataTemplate AmericanMonkey { get; set; }
public DataTemplate OtherMonkey { get; set; }
The MonkeyDataTemplateSelector class defines AmericanMonkey and OtherMonkey DataTemplate properties that
are set to different data templates. The OnSelectTemplate override returns the AmericanMonkey template, which
displays the monkey name and location in teal, when the monkey name contains "America". When the monkey
name doesn't contain "America", the OnSelectTemplate override returns the OtherMonkey template, which
displays the monkey name and location in silver:
For more information about data template selectors, see Create a DataTemplateSelector.
In this example, the OnItemSelected event handler is executed when the ItemSelected event fires, with the event
handler retrieving the selected item:
<ListView ...
SelectionMode="None" />
When the SelectionMode property is set to None , items in the ListView cannot be selected, the SelectedItem
property will remain null , and the ItemSelected event will not be fired.
Cache data
ListView is a powerful view for displaying data, but it has some limitations. Scrolling performance can suffer
when using custom cells, especially when they contain deeply nested view hierarchies or use certain layouts that
require complex measurement. Fortunately, there are techniques you can use to avoid poor performance.
A ListView is often used to display much more data than fits onscreen. For example, a music app might have a
library of songs with thousands of entries. Creating an item for every entry would waste valuable memory and
perform poorly. Creating and destroying rows constantly would require the app to instantiate and cleanup
objects constantly, which would also perform poorly.
To conserve memory, the native ListView equivalents for each platform have built-in features for reusing rows.
Only the cells visible on screen are loaded in memory and the content is loaded into existing cells. This pattern
prevents the app from instantiating thousands of objects, saving time and memory.
.NET MAUI permits ListView cell reuse through the ListViewCachingStrategy enumeration, which defines the
following members:
RetainElement , specifies that the ListView will generate a cell for each item in the list.
RecycleElement , specifies that the ListView will attempt to minimize its memory footprint and execution
speed by recycling list cells.
RecycleElementAndDataTemplate , as RecycleElement while also ensuring that when a ListView uses a
DataTemplateSelector , DataTemplate objects are cached by the type of item in the list.
Retain elements
The RetainElement caching strategy specifies that the ListView will generate a cell for each item in the list, and
is the default ListView behavior. It should be used in the following circumstances:
Each cell has a large number of bindings (20-30+).
The cell template changes frequently.
Testing reveals that the RecycleElement caching strategy results in a reduced execution speed.
It's important to recognize the consequences of the RetainElement caching strategy when working with custom
cells. Any cell initialization code will need to run for each cell creation, which may be multiple times per second.
In this circumstance, layout techniques that were fine on a page, like using multiple nested StackLayout objects,
become performance bottlenecks when they're set up and destroyed in real time as the user scrolls.
Recycle elements
The RecycleElement caching strategy specifies that the ListView will attempt to minimize its memory footprint
and execution speed by recycling list cells. This mode doesn't always offer a performance improvement, and
testing should be performed to determine any improvements. However, it's the preferred choice, and should be
used in the following circumstances:
Each cell has a small to moderate number of bindings.
Each cell's BindingContext defines all of the cell data.
Each cell is largely similar, with the cell template unchanging.
During virtualization the cell will have its binding context updated, and so if an app uses this mode it must
ensure that binding context updates are handled appropriately. All data about the cell must come from the
binding context or consistency errors may occur. This problem can be avoided by using data binding to display
cell data. Alternatively, cell data should be set in the OnBindingContextChanged override, rather than in the custom
cell's constructor, as shown in the following example:
public class CustomCell : ViewCell
{
Image image = null;
public CustomCell()
{
image = new Image();
View = image;
}
NOTE
The RecycleElement caching strategy requires that when a DataTemplateSelector is asked to select a DataTemplate
that each DataTemplate must return the same ViewCell type. For example, given a ListView with a
DataTemplateSelector that can return either MyDataTemplateA (where MyDataTemplateA returns a ViewCell of
type MyViewCellA ), or MyDataTemplateB (where MyDataTemplateB returns a ViewCell of type MyViewCellB ),
when MyDataTemplateA is returned it must return MyViewCellA or an exception will be thrown.
NOTE
The RecycleElementAndDataTemplate caching strategy requires that DataTemplate objects returned by the
DataTemplateSelector must use the DataTemplate constructor that takes a Type .
Then the ListViewCachingStrategy enumeration value can be specified from XAML by using the x:Arguments
attribute:
<local:CustomListView>
<x:Arguments>
<ListViewCachingStrategy>RecycleElement</ListViewCachingStrategy>
</x:Arguments>
</local:CustomListView>
These properties are backed by BindableProperty objects, which means that the properties can be targets of
data bindings.
Display strings in the header and footer
The Header and Footer properties can be set to string values, as shown in the following example:
<ListView ItemsSource="{Binding Monkeys}"
Header="Monkeys"
Footer="2022">
...
</ListView>
<ListView ...
SeparatorVisibility="None" />
In addition, when the separator is enabled, it's color can be set with the SeparatorColor property:
<ListView ...
SeparatorColor="Blue" />
Size items
By default, all items in a ListView have the same height, which is derived from the contents of the
DataTemplate that defines the appearance of each item. However, this behavior can be changed with the
HasUnevenRows and RowHeight properties. By default, the HasUnevenRows property is false .
The RowHeight property can be set to an int that represents the height of each item in the ListView , provided
that HasUnevenRows is false . When HasUnevenRows is set to true , each item in the ListView can have a
different height. The height of each item will be derived from the contents of the item's DataTemplate , and so
each item will be sized to its content.
Individual ListView items can be programmatically resized at runtime by changing layout related properties of
elements within the DataTemplate , provided that the HasUnevenRows property is true . The following example
changes the height of an Image object when it's tapped:
void OnImageTapped(object sender, EventArgs args)
{
Image image = sender as Image;
ViewCell viewCell = image.Parent.Parent as ViewCell;
In this example, the OnImageTapped event handler is executed in response to an Image object being tapped. The
event handler updates the height of the Image and the Cell.ForceUpdateSize method updates the cell's size,
even when it isn't currently visible.
WARNING
Overuse of dynamic item sizing can cause ListView performance to degrade.
Right-to-left layout
ListView can layout its content in a right-to-left flow direction by setting its FlowDirection property to
RightToLeft . However, the FlowDirection property should ideally be set on a page or root layout, which causes
all the elements within the page, or root layout, to respond to the flow direction:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="ListViewDemos.RightToLeftListPage"
Title="Right to left list"
FlowDirection="RightToLeft">
<StackLayout Margin="20">
<ListView ItemsSource="{Binding Monkeys}">
...
</ListView>
</StackLayout>
</ContentPage>
The default for an element with a parent is MatchParent . Therefore, the ListView inherits the
FlowDirection
FlowDirection property value from the StackLayout , which in turn inherits the FlowDirection property value
from the ContentPage .
The Animal class models a single item. A type that models a group of items can then be created. The following
example shows the AnimalGroup class:
The AnimalGroup class inherits from the List<T> class and adds a Name property that represents the group
name.
An IEnumerable<T> collection of groups can then be created:
This code defines a collection named Animals , where each item in the collection is an AnimalGroup object. Each
AnimalGroup object comprises a name, and a List<Animal> collection that defines the Animal objects in the
group.
Grouped data can then be added to the Animals collection:
Animals.Add(new AnimalGroup("Bears", new List<Animal>
{
new Animal
{
Name = "American Black Bear",
Location = "North America",
Details = "Details about the bear go here.",
ImageUrl = "https://upload.wikimedia.org/wikipedia/commons/0/08/01_Schwarzbär.jpg"
},
new Animal
{
Name = "Asian Black Bear",
Location = "Asia",
Details = "Details about the bear go here.",
ImageUrl =
"https://upload.wikimedia.org/wikipedia/commons/thumb/b/b7/Ursus_thibetanus_3_%28Wroclaw_zoo%29.JPG/180px-
Ursus_thibetanus_3_%28Wroclaw_zoo%29.JPG"
},
// ...
}));
This code creates two groups in the Animals collection. The first AnimalGroup is named Bears , and contains a
List<Animal> collection of bear details. The second AnimalGroup is named Monkeys , and contains a
List<Animal> collection of monkey details.
ListView will display grouped data, provided that the data has been grouped correctly, by setting the
IsGroupingEnabled property to true :
<ListView ItemsSource="{Binding Animals}"
IsGroupingEnabled="True">
<ListView.ItemTemplate>
<DataTemplate>
<ViewCell>
<Grid Padding="10">
<Grid.RowDefinitions>
<RowDefinition Height="Auto" />
<RowDefinition Height="Auto" />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="Auto" />
<ColumnDefinition Width="Auto" />
</Grid.ColumnDefinitions>
<Image Grid.RowSpan="2"
Source="{Binding ImageUrl}"
Aspect="AspectFill"
HeightRequest="60"
WidthRequest="60" />
<Label Grid.Column="1"
Text="{Binding Name}"
FontAttributes="Bold" />
<Label Grid.Row="1"
Grid.Column="1"
Text="{Binding Location}"
FontAttributes="Italic"
VerticalOptions="End" />
</Grid>
</ViewCell>
</DataTemplate>
</ListView.ItemTemplate>
</ListView>
The appearance of each item in the ListView is defined by setting its ItemTemplate property to a DataTemplate .
For more information, see Define item appearance.
The following screenshot shows the ListView displaying grouped data:
NOTE
By default, ListView will display the group name in the group header. This behavior can be changed by customizing the
group header.
In this example, each group header is set to a Label that displays the group name, and that has other
appearance properties set. The following screenshot shows the customized group header:
In this scenario, meaningful data can be displayed by overriding the ToString method in the type that models a
single item, and the type that models a single group of items.
Control scrolling
ListView defines two ScrollTo methods, that scroll items into view. One of the overloads scrolls the specified
item into view, while the other scrolls the specified item in the specified group into view. Both overloads have
additional arguments that allow the exact position of the item after the scroll has completed to be specified, and
whether to animate the scroll.
ListView defines a event that is fired when one of the ScrollTo methods is invoked. The
ScrollToRequested
ScrollToRequestedEventArgs object that accompanies the ScrollToRequested event has many properties,
including ShouldAnimate , Element , Mode , and Position . Some of these properties are set from the arguments
specified in the ScrollTo method calls.
In addition, ListView defines a Scrolled event that is fired to indicate that scrolling occurred. The
ScrolledEventArgs object that accompanies the Scrolled event has ScrollX and ScrollY properties.
Detect scrolling
ListView defines a Scrolled event which is fired to indicate that scrolling occurred. The
ItemsViewScrolledEventArgs class, which represents the object that accompanies the Scrolled event, defines
the following properties:
ScrollX , of type double , represents the X position of the scroll
ScrollY , of type double , represents the Y position of the scroll.
The following XAML example shows a ListView that sets an event handler for the Scrolled event:
<ListView Scrolled="OnListViewScrolled">
...
</ListView>
In this code example, the OnListViewScrolled event handler is executed when the Scrolled event fires:
IMPORTANT
The Scrolled event is fired for user initiated scrolls, and for programmatic scrolls.
Alternatively, an item in grouped data can be scrolled into view by specifying the item and the group. The
following example shows how to scroll the Proboscis Monkey item in the Monkeys group into view:
GroupedAnimalsViewModel viewModel = BindingContext as GroupedAnimalsViewModel;
AnimalGroup group = viewModel.Animals.FirstOrDefault(a => a.Name == "Monkeys");
Animal monkey = group.FirstOrDefault(m => m.Name == "Proboscis Monkey");
listView.ScrollTo(monkey, group, ScrollToPosition.MakeVisible, true);
NOTE
The ScrollToRequested event is fired when the ScrollTo method is invoked.
Start
The ScrollToPosition.Start member indicates that the item should be scrolled to the start of the view:
Center
The ScrollToPosition.Center member indicates that the item should be scrolled to the center of the view:
End
The ScrollToPosition.End member indicates that the item should be scrolled to the end of the view:
<ListView x:Name="listView"
ItemsSource="{Binding Monkeys}">
<ListView.ItemTemplate>
<DataTemplate>
<ViewCell>
<ViewCell.ContextActions>
<MenuItem Text="Favorite"
Command="{Binding Source={x:Reference listView},
Path=BindingContext.FavoriteCommand}"
CommandParameter="{Binding}" />
<MenuItem Text="Delete"
Command="{Binding Source={x:Reference listView},
Path=BindingContext.DeleteCommand}"
CommandParameter="{Binding}" />
</ViewCell.ContextActions>
...
</ViewCell>
</DataTemplate>
</ListView.ItemTemplate>
</ListView>
The MenuItem objects are revealed when an item in the ListView is right-clicked:
Pull to refresh
ListView supports pull to refresh functionality, which enables the data being displayed to be refreshed by
pulling down on the list of items.
To enable pull to refresh, set the IsPullToRefreshEnabled property to true . When a refresh is triggered,
ListView raises the Refreshing event, and the IsRefreshing property will be set to true . The code required to
refresh the contents of the ListView should then be executed by the handler for the Refreshing event, or by
the ICommand implementation the RefreshCommand executes. Once the ListView is refreshed, the IsRefreshing
property should be set to false , or the EndRefresh method should be called on the ListView , to indicate that
the refresh is complete.
The following example shows a ListView that uses pull to refresh:
<ListView ItemsSource="{Binding Animals}"
IsPullToRefreshEnabled="true"
RefreshCommand="{Binding RefreshCommand}"
IsRefreshing="{Binding IsRefreshing}">
<ListView.ItemTemplate>
<DataTemplate>
<ViewCell>
...
</ViewCell>
</DataTemplate>
</ListView.ItemTemplate>
</ListView>
In this example, when the user initiates a refresh, the ICommand defined by the RefreshCommand property is
executed, which should refresh the items being displayed. A refresh visualization is shown while the refresh
occurs, which consists of an animated progress circle. The value of the IsRefreshing property indicates the
current state of the refresh operation. When a refresh is triggered, this property will automatically transition to
true . Once the refresh completes, you should reset the property to false .
Picker
3/18/2022 • 7 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) Picker displays a short list of items, from which the user can
select an item.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
All of the properties are backed by BindableProperty objects, which means that they can be styled, and the
properties can be targets of data bindings. The SelectedIndex and SelectedItem properties have a default
binding mode of BindingMode.TwoWay , which means that they can be targets of data bindings in an application
that uses the Model-View-ViewModel (MVVM) pattern.
A Picker doesn't show any data when it's first displayed. Instead, the value of its Title property is shown as a
placeholder, as shown in the following iOS screenshot:
When the Picker gains focus, its data is displayed and the user can select an item:
The Picker fires a SelectedIndexChanged event when the user selects an item. Following selection, the selected
item is displayed by the Picker :
<Picker x:Name="picker"
Title="Select a monkey">
<Picker.ItemsSource>
<x:Array Type="{x:Type x:String}">
<x:String>Baboon</x:String>
<x:String>Capuchin Monkey</x:String>
<x:String>Blue Monkey</x:String>
<x:String>Squirrel Monkey</x:String>
<x:String>Golden Lion Tamarin</x:String>
<x:String>Howler Monkey</x:String>
<x:String>Japanese Macaque</x:String>
</x:Array>
</Picker.ItemsSource>
</Picker>
NOTE
The x:Array element requires a Type attribute indicating the type of the items in the array.
In addition, an event handler can be executed when the SelectedIndexChanged event fires:
if (selectedIndex != -1)
{
monkeyNameLabel.Text = (string)picker.ItemsSource[selectedIndex];
}
}
In this example, the event handler obtains the SelectedIndex property value, and uses the value to retrieve the
selected item from the ItemsSource collection. This is functionally equivalent to retrieving the selected item
from the SelectedItem property. Each item in the ItemsSource collection is of type object , and so must be cast
to a string for display.
NOTE
A Picker can be initialized to display a specific item by setting the SelectedIndex or SelectedItem properties.
However, these properties must be set after initializing the ItemsSource collection.
In this example, the ItemsSource property data binds to the Monkeys property of the binding context, which
returns an IList<Monkey> collection. The following code example shows the Monkey class, which contains four
properties:
When binding to a list of objects, the Picker must be told which property to display from each object. This is
achieved by setting the ItemDisplayBinding property to the required property from each object. In the code
examples above, the Picker is set to display each Monkey.Name property value.
Respond to item selection
Data binding can be used to set an object to the SelectedItem property value when it changes:
NOTE
The SelectedItem and SelectedIndex properties both support two-way bindings by default.
In addition to adding data using the Items.Add method, data can also be inserted into the collection by using
the Items.Insert method.
Respond to item selection
A Picker supports selection of one item at a time. When a user selects an item, the SelectedIndexChanged event
fires, and the SelectedIndex property is updated to an integer representing the index of the selected item in the
list. The SelectedIndex property is a zero-based number indicating the item that the user selected. If no item is
selected, which is the case when the Picker is first created and initialized, SelectedIndex will be -1.
The following code example shows the OnPickerSelectedIndexChanged event handler method, which is executed
when the SelectedIndexChanged event fires:
void OnPickerSelectedIndexChanged(object sender, EventArgs e)
{
var picker = (Picker)sender;
int selectedIndex = picker.SelectedIndex;
if (selectedIndex != -1)
{
monkeyNameLabel.Text = picker.Items[selectedIndex];
}
}
This method obtains the SelectedIndex property value, and uses the value to retrieve the selected item from the
Items collection. Because each item in the Items collection is a string , they can be displayed by a Label
without requiring a cast.
NOTE
A Picker can be initialized to display a specific item by setting the SelectedIndex property. However, the
SelectedIndex property must be set after initializing the Items collection.
TableView
3/18/2022 • 8 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) TableView displays a table of scrollable items that can be grouped
into sections. A TableView is typically used for displaying items where each row has a different appearance,
such as presenting a table of settings.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
While TableView manages the appearance of the table, the appearance of each item in the table is defined by a
Cell . .NET MAUI includes five cell types that are used to display different combinations of data, and you can
also define custom cells that display any content you want.
TableView defines the following properties:
Intent , of type TableIntent , defines the purpose of the table.
HasUnevenRows , of type bool , indicates whether items in the table can have rows of different heights. The
default value of this property is false .
Root , of type TableRoot , defines the child of the TableView .
RowHeight , of type int , determines the height of each row when HasUnevenRows is false .
The HasUnevenRows and RowHeight properties are backed by BindableProperty objects, which means that they
can be targets of data bindings, and styled.
The value of the Intent property helps to define the TableView appearance on each platform. This property
should be set to a value of the TableIntent enumeration, which defines the following members:
Menu , for presenting a selectable menu.
Settings , for presenting a table of configuration settings.
Form , for presenting a data input form.
Data , for presenting data.
Create a TableView
To create a table, create a TableView object and set its Intent property to a TableIntent member. The child of
a TableView must be a TableRoot object, which is parent to one or more TableSection objects. Each
TableSection consists of an optional title whose color can also be set, and one or more Cell objects.
NOTE
Each TextCell can execute a command when tapped, provided that the Command property is set to a valid ICommand
implementation.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
The following example shows using a TextCell to define the appearance of items in a TableView :
<TableView Intent="Menu">
<TableRoot>
<TableSection Title="Chapters">
<TextCell Text="1. Introduction to .NET MAUI"
Detail="Learn about .NET MAUI and what it provides." />
<TextCell Text="2. Anatomy of an app"
Detail="Learn about the visual elements in .NET MAUI" />
<TextCell Text="3. Text"
Detail="Learn about the .NET MAUI controls that display text." />
<TextCell Text="4. Dealing with sizes"
Detail="Learn how to size .NET MAUI controls on screen." />
<TextCell Text="5. XAML vs code"
Detail="Learn more about creating your UI in XAML." />
</TableSection>
</TableRoot>
</TableView>
Image cell
An ImageCell displays an image with primary and secondary text on separate lines. ImageCell inherits the
properties from TextCell , and defines the ImageSource property, of type ImageSource , which specifies the
image to be displayed in the cell. This property is backed by a BindableProperty object, which means it can be
the target of data bindings, and be styled.
The following example shows using an ImageCell to define the appearance of items in a TableView :
<TableView Intent="Menu">
<TableRoot>
<TableSection Title="Learn how to use your XBox">
<ImageCell Text="1. Introduction"
Detail="Learn about your XBox and its capabilities."
ImageSource="xbox.png" />
<ImageCell Text="2. Turn it on"
Detail="Learn how to turn on your XBox."
ImageSource="xbox.png" />
<ImageCell Text="3. Connect your controller"
Detail="Learn how to connect your wireless controller."
ImageSource="xbox.png" />
<ImageCell Text="4. Launch a game"
Detail="Learn how to launch a game."
ImageSource="xbox.png" />
</TableSection>
</TableRoot>
</TableView>
Switch cell
A SwitchCell displays text and a switch that can be switched on or off. SwitchCell defines the following
properties:
Text , of type string , defines the text to display next to the switch.
On , of type bool , represents whether the switch is on or off.
OnColor , of type Color , indicates the color of the switch when in it's on position.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
SwitchCell also defines an OnChanged event that's raised when the switch changes state. The ToggledEventArgs
object that accompanies this event defines a Value property, that indicates whether the switch is on or off.
The following example shows using a SwitchCell to define the appearance of items in a TableView :
<TableView Intent="Settings">
<TableRoot>
<TableSection>
<SwitchCell Text="Airplane Mode"
On="False" />
<SwitchCell Text="Notifications"
On="True" />
</TableSection>
</TableRoot>
</TableView>
The following screenshot shows the resulting cell appearance:
Entry cell
An EntryCell displays a label and text data that's editable. EntryCell defines the following properties:
HorizontalTextAlignment , of type TextAlignment , represents the horizontal alignment of the text.
Keyboard , of type Keyboard , determines the keyboard to display when entering text.
Label , of type string , represents the text to display to the left of the editable text.
LabelColor , of type Color , defines the color of the label text.
Placeholder , of type string , represents the text that's displayed when the Text property is empty.
Text , of type string , defines the text that's editable.
VerticalTextAlignment , of type TextAlignment , represents the vertical alignment of the text.
These properties are backed by BindableProperty objects, which means that they can be targets of data
bindings, and styled.
EntryCell also defines a Completed event that's raised when the user hits the return key, to indicate that editing
is complete.
The following example shows using an EntryCell to define the appearance of items in a TableView :
<TableView Intent="Settings">
<TableRoot>
<TableSection>
<EntryCell Label="Login"
Placeholder="username" />
<EntryCell Label="Password"
Placeholder="password" />
</TableSection>
</TableRoot>
</TableView>
View cell
A ViewCell is a custom cell whose appearance is defined by a View . ViewCell defines a View property, of
type View , which defines the view that represents the content of the cell. This property is backed by a
BindableProperty object, which means it can be the target of data bindings, and be styled.
NOTE
The View property is the content property of the ViewCell class, and therefore does not need to be explicitly set from
XAML.
The following example shows using a ViewCell to define the appearance of an item in a TableView :
<TableView Intent="Settings">
<TableRoot>
<TableSection Title="Silent">
<ViewCell>
<Grid RowDefinitions="Auto,Auto"
ColumnDefinitions="0.5*,0.5*">
<Label Text="Vibrate"
Margin="10,10,0,0"/>
<Switch Grid.Column="1"
HorizontalOptions="End" />
<Slider Grid.Row="1"
Grid.ColumnSpan="2"
Margin="10"
Minimum="0"
Maximum="10"
Value="3" />
</Grid>
</ViewCell>
</TableSection>
</TableRoot>
</TableView>
Inside the ViewCell , layout can be managed by any .NET MAUI layout. The following screenshot shows the
resulting cell appearance:
Size items
By default, all cells of the same type in a TableView have the same height. However, this behavior can be
changed with the HasUnevenRows and RowHeight properties. By default, the HasUnevenRows property is false .
The RowHeight property can be set to an int that represents the height of each item in the TableView ,
provided that HasUnevenRows is false . When HasUnevenRows is set to true , each item in the TableView can
have a different height. The height of each item will be derived from the contents of each cell, and so each item
will be sized to its content.
Individual cells can be programmatically resized at runtime by changing layout related properties of elements
within the cell, provided that the HasUnevenRows property is true . The following example changes the height of
the cell when it's tapped:
In this example, the OnViewCellTapped event handler is executed in response to the cell being tapped. The event
handler updates the visibility of the Label object and the Cell.ForceUpdateSize method updates the cell's size.
If the Label has been made visible the cell's height will increase. If the Label has been made invisible the cell's
height will decrease.
WARNING
Overuse of dynamic item sizing can cause TableView performance to degrade.
Right-to-left layout
TableView can layout its content in a right-to-left flow direction by setting its FlowDirection property to
RightToLeft . However, the FlowDirection property should ideally be set on a page or root layout, which causes
all the elements within the page, or root layout, to respond to the flow direction:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="TableViewDemos.RightToLeftTablePage"
Title="Right to left TableView"
FlowDirection="RightToLeft">
<TableView Intent="Settings">
...
</TableView>
</ContentPage>
The default FlowDirectionfor an element with a parent is MatchParent . Therefore, the TableView inherits the
FlowDirection property value from the ContentPage .
ContentView
3/18/2022 • 4 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) ContentView is a control that enables the creation of custom,
reusable controls.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The ContentView class defines a Content property, of type View , which represents the content of the
ContentView . This property is backed by a BindableProperty object, which means that it can be the target of
data bindings, and styled.
The ContentView class derives from the TemplatedView class, which defines the ControlTemplate bindable
property, of type ControlTemplate , which defines the appearance of the control. For more information about the
ControlTemplate property, see Customize appearance with a ControlTemplate.
NOTE
A ContentView can only contain a single child.
public CardView()
{
InitializeComponent();
}
}
Define the UI
The custom control UI can be defined in the XAML file for the ContentView -derived class, which uses a
ContentView as the root element of the control:
<ContentView ...
x:Name="this"
x:Class="CardViewDemo.Controls.CardView">
<Frame BindingContext="{x:Reference this}"
BackgroundColor="{Binding CardColor}"
BorderColor="{Binding BorderColor}"
...>
<Grid>
...
<Frame BorderColor="{Binding BorderColor, FallbackValue='Black'}"
BackgroundColor="{Binding IconBackgroundColor, FallbackValue='Grey'}"
...>
<Image Source="{Binding IconImageSource}"
.. />
</Frame>
<Label Text="{Binding CardTitle, FallbackValue='Card Title'}"
... />
<BoxView BackgroundColor="{Binding BorderColor, FallbackValue='Black'}"
... />
<Label Text="{Binding CardDescription, FallbackValue='Card description text.'}"
... />
</Grid>
</Frame>
</ContentView>
The ContentView element sets the x:Name property to this , which can be used to access the object bound to
the CardView instance. Elements in the layout set bindings on their properties to values defined on the bound
object.
NOTE
The FallbackValue property in the Binding expression provides a default value in case the binding is null .
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:controls="clr-namespace:CardViewDemo.Controls"
x:Class="CardViewDemo.CardViewXamlPage">
<ScrollView>
<StackLayout>
<controls:CardView BorderColor="DarkGray"
CardTitle="Slavko Vlasic"
CardDescription="Lorem ipsum dolor sit amet, consectetur adipiscing elit.
Nulla elit dolor, convallis non interdum."
IconBackgroundColor="SlateGray"
IconImageSource="user.png" />
<!-- More CardView objects -->
</StackLayout>
</ScrollView>
</ContentPage>
Data binding in a uses the TemplateBinding markup extension to specify bindings. The
ControlTemplate
ControlTemplate property can then be set to the defined ControlTemplate object, by using its x:Key value. The
following example shows the ControlTemplate property set on a CardView instance:
The following screenshot shows a standard CardView instance, and multiple CardView instances whose control
templates have been overridden:
Display pop-ups
3/18/2022 • 3 minutes to read • Edit Online
Displaying an alert, asking a user to make a choice, or displaying a prompt is a common UI task. .NET Multi-
platform App UI (.NET MAUI) has three methods on the Page class for interacting with the user via a pop-up:
DisplayAlert , DisplayActionSheet , and DisplayPromptAsync . Pop-ups are rendered with native controls on each
platform.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Display an alert
All .NET MAUI-supported platforms have a modal pop-up to alert the user or ask simple questions of them. To
display alerts, use the DisplayAlert method on any Page . The following example shows a simple message to
the user:
The alert is displayed modally, and once dismissed the user continues interacting with the app:
The DisplayAlert method can also be used to capture a user's response by presenting two buttons and
returning a bool . To get a response from an alert, supply text for both buttons and await the method:
bool answer = await DisplayAlert("Question?", "Would you like to play a game", "Yes", "No");
Debug.WriteLine("Answer: " + answer);
After the user selects one of the options the response will be returned as a bool .
The DisplayAlert method also has overloads that accept a FlowDirection argument that specifies the direction
in which UI elements flow within the alert.
string action = await DisplayActionSheet("ActionSheet: Send to?", "Cancel", null, "Email", "Twitter",
"Facebook");
Debug.WriteLine("Action: " + action);
After the user taps one of the buttons, the button label will be returned as a string .
Action sheets also support a destroy button, which is a button that represents destructive behavior. The destroy
button can be specified as the third string argument to the DisplayActionSheet method, or can be left null . The
following example specifies a destroy button:
NOTE
On iOS, the destroy button is rendered differently to the other buttons in the action sheet.
The DisplayActionSheet method also has an overload that accepts a FlowDirection argument that specifies the
direction in which UI elements flow within the action sheet.
Display a prompt
To display a prompt, call the DisplayPromptAsync on any Page , passing a title and message as string
arguments:
string result = await DisplayPromptAsync("Question 2", "What's 5 + 5?", initialValue: "10", maxLength: 2,
keyboard: Keyboard.Numeric);
This code displays a predefined response of 10, limits the number of characters that can be input to 2, and
displays the numeric keyboard for user input:
Graphics
3/18/2022 • 2 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) provides a cross-platform graphics canvas on which 2D graphics can be
drawn using types from the Microsoft.Maui.Graphics namespace. This canvas supports drawing and painting
shapes and images, compositing operations, and graphical object transforms.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
There are many similarities between the functionality provided by Microsoft.Maui.Graphics , and the
functionality provided by .NET MAUI shapes and brushes. However, each is aimed at different scenarios:
Microsoft.Maui.Graphics functionality must be consumed on a drawing canvas, enables performant graphics
to be drawn, and provides a convenient approach for writing graphics-based controls. For example, a control
that replicates the GitHub contribution profile can be more easily implemented using
Microsoft.Maui.Graphics than by using .NET MAUI shapes.
.NET MAUI shapes can be consumed directly on a page, and brushes can be consumed by all controls. This
functionality is provided to help you produce an attractive UI.
Drawing canvas
In .NET MAUI, the GraphicsView enables consumption of the Microsoft.Maui.Graphics functionality, via a
drawing canvas that's exposed as an ICanvas object. For more information about the GraphicsView , see
GraphicsView.
ICanvas defines the following properties, that affect the appearance of object's that are drawn on the canvas:
Alpha , of type float , indicates the opacity of an object.
Antialias , of type bool , specifies whether anti-aliasing is enabled.
BlendMode , of type BlendMode , defines the blend mode, which determines what happens when an object is
rendered on top of an existing object.
DisplayScale , of type float , represents the scaling factor to scale the UI by on a canvas.
FillColor , of type Color , indicates the color used to paint an object's interior.
Font , of type IFont , defines the font when drawing text.
FontColor , of type Color , specifies the font color when drawing text.
FontSize , of type float , defines the size of the font when drawing text.
MiterLimit , of type float , specifies the limit of the miter length of line joins in an object.
StrokeColor , of type Color , indicates the color used to paint an object's outline.
StrokeDashPattern , of type float[] , specifies the pattern of dashes and gaps that are used to outline an
object.
StrokeLineCap , of type LineCap , describes the shape at the start and end of a line.
StrokeLineJoin , of type LineJoin , specifies the type of join that is used at the vertices of a shape.
StrokeSize , of type float , indicates the width of an object's outline.
By default, an ICanvas sets StrokeSize to 1, StrokeColor to black, StrokeLineJoin to LineJoin.Miter , and
StrokeLineCap to LineJoin.Cap .
Drawing canvas state
The drawing canvas on each platform has the ability to maintain its state. This enables you to persist the current
graphics state, and restore it when required.
However, not all elements of the canvas are elements of the graphics state. The graphics state does not include
drawing objects, such as paths, and paint objects, such as gradients. Typical elements of the graphics state on
each platform include stroke and fill data, and font data.
The graphics state of each ICanvas can be manipulated with the following methods:
SaveState , which saves the current graphics state.
RestoreState , which sets the graphics state to the most recently saved state.
ResetState , which resets the graphics state to its default values.
NOTE
The state that's persisted by these methods is platform dependent.
Blend modes
3/18/2022 • 7 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) graphics enables different compositing operations for graphical objects
to be specified by the ICanvas.BlendMode property. This property determines what happens when a graphical
object (called the source), is rendered on top of an existing graphical object (called the destination).
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
By default, the last drawn object obscures the objects drawn underneath it:
In this example, the cyan circle is drawn first, followed by the magenta circle, then the yellow circle. Each circle
obscures the circle drawn underneath it. This occurs because the default blend mode is Normal , which means
that the source is drawn over the destination. However, it's possible to specify a different blend mode for a
different result. For example, if you specify DestinationOver , then in the area where the source and destination
intersect, the destination is drawn over the source.
The 28 members of the BlendMode enumeration can be divided into three categories:
Darken SourceAtop
Lighten DestinationOver
SEPA RA B L E N O N - SEPA RA B L E P O RT ER- DUF F
ColorDodge DestinationIn
ColorBurn DestinationOut
SoftLight DestinationAtop
HardLight Xor
Difference PlusDarker
Exclusion PlusLighter
The order that the members are listed in the table above is the same as in the BlendMode enumeration. The first
column lists the 12 separable blend modes, while the second column lists the non-separable blend modes.
Finally, the third column lists the Porter-Duff blend modes.
In the image above, the destination is a transparent rectangle except for a brown area that occupies the left and
top two-thirds of the display surface. The source is also a transparent rectangle except for a blue area that
occupies the right and bottom two-thirds of the display surface. Displaying the source on the destination
produces the following result:
The transparent pixels of the source allow the background to show through, while the blue source pixels obscure
the background. This is the normal case, using the default blend mode of Normal . However, it's possible to
specify that in the area where the source and destination intersect, the destination appears instead of the source,
using the DestinationOver blend mode:
The DestinationIn blend mode displays only the area where the destination and source intersect, using the
destination color:
The Xor blend mode causes nothing to appear where the two areas overlap:
The colored destination and source rectangles effectively divide the display surface into four unique areas that
can be colored in different ways, corresponding to the presence of the destination and source rectangles:
The upper-right and lower-left rectangles are always blank because both the destination and source are
transparent in those areas. The destination color occupies the upper-left area, so that area can either be colored
with the destination color or not at all. Similarly, the source color occupies the lower-right area, so that area can
be colored with the source color or not at all.
The following table lists the Porter-Duff blend modes provided by Microsoft.Maui.Graphics , and how they color
each of the three non-blank areas in the diagram above:
Clear
Copy Source X
SourceIn Source
SourceOut X
SourceAtop X Source
DestinationOver X Destination X
DestinationIn Destination
DestinationOut X
DestinationAtop Destination X
Xor X X
PlusDarker X Sum X
PlusLighter X Sum X
NOTE
These blend modes are symmetrical. The source and destination can be exchanged, and all the modes are still available.
The PlusLighter blend mode sums the source and destination. Then, for values above 1, white is displayed.
Similarly, the PlusDarker blend mode sums the source and destination, but subtracts 1 from the resulting
values, with values below 0 becoming black.
B L EN D M O DE N O C H A N GE O P ERAT IO N
NOTE
If the source is transparent, then the separable blend modes have no effect.
The following example uses the Multiply blend mode to draw three overlapping circles of cyan, magenta, and
yellow:
canvas.BlendMode = BlendMode.Multiply;
canvas.FillColor = Colors.Cyan;
canvas.FillCircle(center1, radius);
canvas.FillColor = Colors.Magenta;
canvas.FillCircle(center2, radius);
canvas.FillColor = Colors.Yellow;
canvas.FillCircle(center3, radius);
The result is that a combination of any two colors produces red, green, and blue, and a combination of all three
colors produces black.
B L EN D M O DE SO URC E C O M P O N EN T S DEST IN AT IO N C O M P O N EN T S
The Color class, in the Microsoft.Maui.Graphics namespace, lets you specify colors as Red-Green-Blue (RGB)
values, Hue-Saturation-Luminosity (HSL) values, Hue-Saturation-Value (HSV) values, or with a color name. An
Alpha channel is also available to indicate transparency.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Color objects can be created with Color constructors, which can be used to specify a gray shade, an RGB
value, or an RGB value with transparency. In all cases, arguments are float values ranging from 0 to 1.
NOTE
The default Color constructor creates a black Color object.
You can also use the following static methods to create Color objects:
Color.FromRgb from float RGB values that range from 0 to 1.
Color.FromRgb from double RGB values that range from 0 to 1.
Color.FromRgb from byte RGB values that range from 0 to 255.
Color.FromRgba from float RGBA values that range from 0 to 1.
Color.FromRgba from double RGBA values that range from 0 to 1.
Color.FromRgba from byte RGBA values that range from 0 to 255.
Color.FromRgba from a string -based hexadecimal value in the form "#RRGGBBAA" or "#RRGGBB" or
"#RGBA" or "#RGB", where each letter corresponds to a hexadecimal digit for the alpha, red, green, and blue
channels.
Color.FromHsla from float HSLA values.
Color.FromHsla from double HSLA values.
Color.FromHsv from float HSV values that range from 0 to 1.
Color.FromHsv from int HSV values that range from 0 to 255.
Color.FromHsva from float HSVA values.
Color.FromHsva from int HSV values.
Color.FromInt from an int value calculated as (B + 256 * (G + 256 * (R + 256 * A))).
Color.FromUint from a uint value calculated as (B + 256 * (G + 256 * (R + 256 * A))).
Color.FromArgb from a string -based hexadecimal value in the form "#AARRGGBB" or "#RRGGBB" or
"#ARGB" or "RGB", where each letter corresponds to a hexadecimal digit for the alpha, red, green, and blue
channels.
NOTE
In addition to the methods listed above, the Color class also has Parse and TryParse methods that create Color
objects from string arguments.
Once created, a Color object is immutable. The characteristics of the color can be obtained from the following
float properties, that range from 0 to 1:
In addition, the characteristics of the color can be obtained from the following methods:
GetHue , which returns a float that represents the hue channel of the color.
GetSaturation , which returns a float that represents the saturation channel of the color.
Luminosity , which returns a float that represents the luminosity channel of the color.
Named colors
The class defines 148 public static read-only fields for common colors, such as
Colors AntiqueWhite ,
MidnightBlue , and YellowGreen .
Modify a color
The following instance methods modify an existing color to create a new color:
AddLuminosity returns a Color by adding the luminosity value to the supplied delta value.
GetComplementary returns the complementary Color .
MultiplyAlpha returns a Color by multiplying the alpha value by the supplied float value.
WithAlpha returns a Color , replacing the alpha value with the supplied float value.
WithHue returns a Color , replacing the hue value with the supplied float value.
WithLuminosity returns a Color , replacing the luminosity value with the supplied float value.
WithSaturation returns a Color , replacing the saturation value with the supplied float value.
Conversions
The following instance methods convert a Color to an alternative representation:
AsPaint returns a SolidPaint object whose Color property is set to the color.
ToHex returns a hexadecimal string representation of a Color .
ToArgbHex returns an ARGB hexadecimal string representation of a Color .
ToRgbaHex returns an RGBA hexadecimal string representation of a Color .
ToInt returns an ARGB int representation of a Color .
ToUint returns an ARGB uint representation of a Color .
ToRgb converts a Color to RGB byte values that are returned as out arguments.
ToRgba converts a Color to RGBA byte values that are returned as out arguments.
ToHsl converts a Color to HSL float values that are passed as out arguments.
Examples
In XAML, colors are typically referenced using their named values, or with hexadecimal:
In C#, colors are typically referenced using their named values, or with their static methods:
The following example uses the OnPlatform markup extension to selectively set the color of an
ActivityIndicator :
.NET Multi-platform App UI (.NET MAUI) graphics, in the Microsoft.Maui.Graphics namespace, enables you to
draw graphical objects on a canvas that's defined as an ICanvas object.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The .NET MAUI GraphicsView control provides access to an ICanvas object, on which properties can be set and
methods invoked to draw graphical objects. For more information about the GraphicsView , see GraphicsView.
IMPORTANT
Graphical objects are drawn on an ICanvas in units of pixels.
Draw a line
Lines can be drawn on an ICanvas using the DrawLine method, which requires four float arguments that
represent the start and end points of the line.
The following example shows how to draw a line:
canvas.StrokeColor = Colors.Red;
canvas.StrokeSize = 6;
canvas.DrawLine(10, 10, 90, 100);
NOTE
There's also a DrawLine overload that takes two PointF arguments.
canvas.StrokeColor = Colors.Red;
canvas.StrokeSize = 4;
canvas.StrokeDashPattern = new float[] { 2, 2 };
canvas.DrawLine(10, 10, 90, 100);
In this example, a red dashed diagonal line is drawn from (10,10) to (90,100):
For more information about dashed lines, see Draw dashed objects.
Draw an ellipse
Ellipses and circles can be drawn on an ICanvas using the DrawEllipse method, which requires x , y , width ,
and height arguments, of type float .
The following example shows how to draw an ellipse:
canvas.StrokeColor = Colors.Red;
canvas.StrokeSize = 4;
canvas.DrawEllipse(10, 10, 100, 50);
To draw a circle, make the width and height arguments to the DrawEllipse method equal:
canvas.StrokeColor = Colors.Red;
canvas.StrokeSize = 4;
canvas.DrawEllipse(10, 10, 100, 100);
NOTE
Circles can also be drawn with the DrawCircle method.
For information about drawing a dashed ellipse, see Draw dashed objects.
A filled ellipse can be drawn with the FillEllipse method, which also requires x , y , width , and height
arguments, of type float :
canvas.FillColor = Colors.Red;
canvas.FillEllipse(10, 10, 150, 50);
In this example, a red filled ellipse with dimensions 150x50 is drawn at (10,10):
The FillColor property of the ICanvas object must be set to a Color before invoking the FillEllipse
method.
Filled circles can also be drawn with the FillCircle method.
NOTE
There are DrawEllipse and FillEllipse overloads that take Rect and RectF arguments. In addition, there are
also DrawCircle and FillCircle overloads.
Draw a rectangle
Rectangles and squares can be drawn on an ICanvas using the DrawRectangle method, which requires x , y ,
width , and height arguments, of type float .
canvas.StrokeColor = Colors.DarkBlue;
canvas.StrokeSize = 4;
canvas.DrawRectangle(10, 10, 100, 50);
In this example, a dark blue rectangle with dimensions 100x50 is drawn at (10,10):
To draw a square, make the width and height arguments to the DrawRectangle method equal:
canvas.StrokeColor = Colors.DarkBlue;
canvas.StrokeSize = 4;
canvas.DrawRectangle(10, 10, 100, 100);
In this example, a dark blue square with dimensions 100x100 is drawn at (10,10):
For information about drawing a dashed rectangle, see Draw dashed objects.
A filled rectangle can be drawn with the FillRectangle method, which also requires x , y , width , and height
arguments, of type float :
canvas.FillColor = Colors.DarkBlue;
canvas.FillRectangle(10, 10, 100, 50);
In this example, a dark blue filled rectangle with dimensions 100x50 is drawn at (10,10):
The FillColor property of the ICanvas object must be set to a Color before invoking the FillRectangle
method.
NOTE
There are DrawRectangle and FillRectangle overloads that take Rect and RectF arguments.
canvas.StrokeColor = Colors.Green;
canvas.StrokeSize = 4;
canvas.DrawRoundedRectangle(10, 10, 100, 50, 12);
In this example, a green rectangle with rounded corners and dimensions 100x50 is drawn at (10,10):
For information about drawing a dashed rounded rectangle, see Draw dashed objects.
A filled rounded rectangle can be drawn with the FillRoundedRectangle method, which also requires x , y ,
width , height , and cornerRadius arguments, of type float :
canvas.FillColor = Colors.Green;
canvas.FillRoundedRectangle(10, 10, 100, 50, 12);
In this example, a green filled rectangle with rounded corners and dimensions 100x50 is drawn at (10,10):
The FillColorproperty of the ICanvas object must be set to a Color before invoking the
FillRoundedRectangle method.
NOTE
There are DrawRoundedRectangle and FillRoundedRectangle overloads that take Rect and RectF arguments, and
overloads that enable the radius of each corner to be separately specified.
Draw an arc
Arcs can be drawn on an ICanvas using the DrawArc method, which requires x , y , width , height ,
startAngle , and endAngle arguments of type float , and clockwise and closed arguments of type bool .
The startAngle argument specifies the angle from the x-axis to the starting point of the arc. The endAngle
argument specifies the angle from the x-axis to the end point of the arc. The clockwise argument specifies the
direction in which the arc is drawn, and the closed argument specifies whether the end point of the arc will be
connected to the start point.
The following example shows how to draw an arc:
canvas.StrokeColor = Colors.Teal;
canvas.StrokeSize = 4;
canvas.DrawArc(10, 10, 100, 100, 0, 180, true, false);
In this example, a teal arc of dimensions 100x100 is drawn at (10,10). The arc is drawn in a clockwise direction
from 0 degrees to 180 degrees, and isn't closed:
For information about drawing a dashed arc, see Draw dashed objects.
A filled arc can be drawn with the FillArc method, which requires x , y , width , height , startAngle , and
endAngle arguments of type float , and a clockwise argument of type bool :
canvas.FillColor = Colors.Teal;
canvas.FillArc(10, 10, 100, 100, 0, 180, true);
In this example, a filled teal arc of dimensions 100x100 is drawn at (10,10). The arc is drawn in a clockwise
direction from 0 degrees to 180 degrees, and is closed automatically:
The FillColor property of the ICanvas object must be set to a Color before invoking the FillArc method.
NOTE
There are DrawArc and FillArc overloads that take Rect and RectF arguments.
Draw a path
A path is a collection of one or more contours. Each contour is a collection of connected straight lines and
curves. Contours are not connected to each other but they might visually overlap. Sometimes a single contour
can overlap itself.
Paths are used to draw curves and complex shapes and can be drawn on an ICanvas using the DrawPath
method, which requires a PathF argument.
A contour generally begins with a call to the PathF.MoveTo method, which you can express either as a PointF
value or as separate x and y coordinates. The MoveTo call establishes a point at the beginning of the contour
and an initial current point. You can then call the following methods to continue the contour with a line or curve
from the current point to a point specified in the method, which then becomes the new current point:
LineTo to add a straight line to the path.
AddArc to add an arc, which is a line on the circumference of a circle or ellipse.
CurveTo to add a cubic Bezier spline.
QuadTo to add a quadratic Bezier spline.
None of these methods contain all of the data necessary to describe the line or curve. Instead, each method
works with the current point established by the method call immediately preceding it. For example, the LineTo
method adds a straight line to the contour based on the current point.
A contour ends with another call to MoveTo , which begins a new contour, or a call to Close , which closes the
contour. The Close method automatically appends a straight line from the current point to the first point of the
contour, and marks the path as closed.
The PathF class also defines other methods and properties. The following methods add entire contours to the
path:
AppendEllipse appends a closed ellipse contour to the path.
AppendCircle appends a closed circle contour to the path.
AppendRectangle appends a closed rectangle contour to the path.
AppendRoundedRectangle appends a closed rectangle with rounded corners to the path.
A filled path can be drawn with the FillPath , which also requires a PathF argument:
The FillColor property of the ICanvas object must be set to a Color before invoking the FillPath method.
IMPORTANT
The FillPath method has an overload that enables a WindingMode to be specified, which sets the fill algorithm that's
used. For more information, see Winding modes.
Draw an image
Images can be drawn on an ICanvas using the DrawImage method, which requires an IImage argument, and
x , y , width , and height arguments, of type float .
The following example shows how to load an image and draw it to the canvas:
using Microsoft.Maui.Graphics.Platform;
...
IImage image;
Assembly assembly = GetType().GetTypeInfo().Assembly;
using (Stream stream =
assembly.GetManifestResourceStream("GraphicsViewDemos.Resources.Images.dotnet_bot.png"))
{
image = PlatformImage.FromStream(stream);
}
if (image != null)
{
canvas.DrawImage(image, 10, 10, image.Width, image.Height);
}
In this example, an image is retrieved from the assembly and loaded as a stream. It's then drawn at actual size at
(10,10):
IMPORTANT
Loading an image that's embedded in an assembly requires the image to have its build action set to Embedded
Resource rather than MauiImage .
Draw a string
Strings can be drawn on an ICanvas using one of the DrawString overloads. The appearance of each string can
be defined by setting the Font , FontColor , and FontSize properties. String alignment can be specified by
horizontal and vertical alignment options that perform alignment within the string's bounding box.
NOTE
The bounding box for a string is defined by its x , y , width , and height arguments.
canvas.Font = Font.Default;
canvas.DrawString("Text is left aligned.", 20, 20, 380, 100, HorizontalAlignment.Left,
VerticalAlignment.Top);
canvas.DrawString("Text is centered.", 20, 60, 380, 100, HorizontalAlignment.Center, VerticalAlignment.Top);
canvas.DrawString("Text is right aligned.", 20, 100, 380, 100, HorizontalAlignment.Right,
VerticalAlignment.Top);
canvas.Font = Font.DefaultBold;
canvas.DrawString("This text is displayed using the bold system font.", 20, 140, 350, 100,
HorizontalAlignment.Left, VerticalAlignment.Top);
In this example, strings with different appearance and alignment options are displayed:
NOTE
The DrawString overloads also enable truncation and line spacing to be specified.
using Microsoft.Maui.Graphics.Text;
...
string markdownText = @"This is *italic text*, **bold text**, __underline text__, and ***bold italic
text***.";
IAttributedText attributedText = MarkdownAttributedTextReader.Read(markdownText); // Requires the
Microsoft.Maui.Graphics.Text.Markdig package
canvas.DrawText(attributedText, 10, 10, 400, 400);
In this example, markdown is converted to attributed text and displayed with the correct styling:
IMPORTANT
Drawing attributed text requires you to have added the Microsoft.Maui.Graphics.Text.Markdig NuGet package to
your project.
canvas.StrokeColor = Colors.Blue;
canvas.StrokeSize = 10;
canvas.FillColor = Colors.Red;
canvas.FillPath(path);
canvas.DrawPath(path);
In this example, the stroke and fill colors for a PathF object are specified. The filled circle is drawn, then the
outline stroke of the circle:
WARNING
Calling a draw method before a fill method will result in an incorrect z-order. The fill will be drawn over the stroke, and the
stroke won't be visible.
Draw a shadow
Graphical objects drawn on an ICanvas can have a shadow applied using the SetShadow method, which takes
the following arguments:
offset , of type SizeF , specifies an offset for the shadow, which represents the position of a light source
that creates the shadow.
blur , of type float , represents the amount of blur to apply to the shadow.
color , of type Color , defines the color of the shadow.
The following examples show how to add shadows to filled objects:
canvas.FillColor = Colors.Red;
canvas.SetShadow(new SizeF(10, 10), 4, Colors.Grey);
canvas.FillRectangle(10, 10, 90, 100);
canvas.FillColor = Colors.Green;
canvas.SetShadow(new SizeF(10, -10), 4, Colors.Grey);
canvas.FillEllipse(110, 10, 90, 100);
canvas.FillColor = Colors.Blue;
canvas.SetShadow(new SizeF(-10, 10), 4, Colors.Grey);
canvas.FillRoundedRectangle(210, 10, 90, 100, 25);
In these examples, shadows whose light sources are in different positions are added to the filled objects, with
identical amounts of blur:
canvas.StrokeColor = Colors.Red;
canvas.StrokeSize = 4;
canvas.StrokeDashPattern = new float[] { 2, 2 };
canvas.DrawRectangle(10, 10, 90, 100);
The following example shows how to draw a dashed square, using an irregular dash:
canvas.StrokeColor = Colors.Red;
canvas.StrokeSize = 4;
canvas.StrokeDashPattern = new float[] { 4, 4, 1, 4 };
canvas.DrawRectangle(10, 10, 90, 100);
canvas.StrokeSize = 10;
canvas.StrokeColor = Colors.Red;
canvas.StrokeLineCap = LineCap.Round;
canvas.DrawLine(10, 10, 110, 110);
In this example, the red line is rounded at the start and end of the line:
NOTE
When the StrokeLineJoin property is set to Miter , the MiterLimit property can be set to a float to limit the
miter length of line joins in the object.
canvas.StrokeSize = 20;
canvas.StrokeColor = Colors.Blue;
canvas.StrokeLineJoin = LineJoin.Round;
canvas.DrawPath(path);
In this example, the blue PathF object has rounded joins at its vertices:
Clip objects
Graphical objects that are drawn to an ICanvas can be clipped prior to drawing, with the following methods:
ClipPath clips an object so that only the area that's within the region of a PathF object will be visible.
ClipRectangle clips an object so that only the area that's within the region of a rectangle will be visible. The
rectangle can be specified using float arguments, or by a Rect or RectF argument.
SubtractFromClip clips an object so that only the area that's outside the region of a rectangle will be visible.
The rectangle can be specified using float arguments, or by a Rect or RectF argument.
The following example shows how to use the ClipPath method to clip an image:
using Microsoft.Maui.Graphics.Platform;
...
IImage image;
var assembly = GetType().GetTypeInfo().Assembly;
using (var stream = assembly.GetManifestResourceStream("GraphicsViewDemos.Resources.Images.dotnet_bot.png"))
{
image = PlatformImage.FromStream(stream);
}
if (image != null)
{
PathF path = new PathF();
path.AppendCircle(100, 90, 80);
canvas.ClipPath(path); // Must be called before DrawImage
canvas.DrawImage(image, 10, 10, image.Width, image.Height);
}
In this example, the image is clipped using a PathF object that defines a circle that's centered at (100,90) with a
radius of 80. The result is that only the part of the image within the circle is visible:
IMPORTANT
The ClipPath method has an overload that enables a WindingMode to be specified, which sets the fill algorithm that's
used when clipping. For more information, see Winding modes.
The following example shows how to use the SubtractFromClip method to clip an image:
using Microsoft.Maui.Graphics.Platform;
...
IImage image;
var assembly = GetType().GetTypeInfo().Assembly;
using (var stream = assembly.GetManifestResourceStream("MyMauiApp.Resources.Images.dotnet_bot.png"))
{
image = PlatformImage.FromStream(stream);
}
if (image != null)
{
canvas.SubtractFromClip(60, 60, 90, 90);
canvas.DrawImage(image, 10, 10, image.Width, image.Height);
}
In this example, the area defined by the rectangle that's specified by the arguments supplied to the
SubtractFromClip method is clipped from the image. The result is that only the parts of the image outside the
rectangle are visible:
Images
3/18/2022 • 4 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) graphics includes functionality to load, save, resize, and downsize
images. Supported image formats are dependent on the underlying platform.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Images are represented by the IImage type, which defines the following properties:
Width , of type float , that defines the width of an image.
Height , of type float , that defines the height of an image.
An optional ImageFormat argument can be specified when loading and saving images. The ImageFormat
enumeration defines Png , Jpeg , Gif , Tiff , and Bmp members. However, this argument is only used when
the image format is supported by the underlying platform.
Load an image
Image loading functionality is provided by the GraphicsService class. Images can be loaded from a stream by
the LoadFromStream method, or from a byte array using the LoadImageFromBytes method.
The following example shows how to load an image:
using Microsoft.Maui.Graphics.Platform;
...
IImage image;
Assembly assembly = GetType().GetTypeInfo().Assembly;
using (Stream stream =
assembly.GetManifestResourceStream("GraphicsViewDemos.Resources.Images.dotnet_bot.png"))
{
image = PlatformImage.FromStream(stream);
}
if (image != null)
{
canvas.DrawImage(image, 10, 10, image.Width, image.Height);
}
In this example, the image is retrieved from the assembly, loaded as a stream, and displayed.
IMPORTANT
Loading an image that's embedded in an assembly requires the image to have its build action set to Embedded
Resource rather than MauiImage .
Resize an image
Images can be resized using the IImage.Resize method, which requires width and height arguments, of type
float , which represent the target dimensions of the image. The Resize method also accepts two optional
arguments:
A ResizeMode argument, that controls how the image will be resized to fit its target dimensions.
A bool argument that controls whether the source image will be disposed after performing the resize
operation. This argument defaults to false , indicating that the source image won't be disposed.
The ResizeMode enumeration defines the following members, which specify how to resize the image to the
target size:
Fit , which letterboxes the image so that it fits its target size.
Bleed , which clips the image so that it fits its target size, while preserving its aspect ratio.
Stretch , which stretches the image so it fills the available space. This can result in a change in the image
aspect ratio.
The following example shows how to resize an image:
using Microsoft.Maui.Graphics.Platform;
...
IImage image;
Assembly assembly = GetType().GetTypeInfo().Assembly;
using (Stream stream =
assembly.GetManifestResourceStream("GraphicsViewDemos.Resources.Images.dotnet_bot.png"))
{
image = PlatformImage.FromStream(stream);
}
if (image != null)
{
IImage newImage = image.Resize(100, 60, ResizeMode.Stretch, true);
canvas.DrawImage(newImage, 10, 10, newImage.Width, newImage.Height);
}
In this example, the image is retrieved from the assembly and loaded as a stream. The image is resized using the
Resize method, with its arguments specifying the new size, and that it should be stretched to fill the available
space. In addition, the source image is disposed. The resized image is then drawn at actual size at (10,10).
Downsize an image
Images can be downsized by one of the IImage.Downsize overloads. The first overload requires a single float
value that represents the maximum width or height of the image, and downsizes the image while maintaining
its aspect ratio. The second overload requires two float arguments, that represent the maximum width and
maximum height of the image.
The Downsize overloads also accept an optional bool argument that controls whether the source image should
be disposed after performing the downsizing operation. This argument defaults to false , indicating that the
source image won't be disposed.
The following example shows how to downsize an image:
using Microsoft.Maui.Graphics.Platform;
...
IImage image;
Assembly assembly = GetType().GetTypeInfo().Assembly;
using (Stream stream =
assembly.GetManifestResourceStream("GraphicsViewDemos.Resources.Images.dotnet_bot.png"))
{
image = PlatformImage.FromStream(stream);
}
if (image != null)
{
IImage newImage = image.Downsize(100, true);
canvas.DrawImage(newImage, 10, 10, newImage.Width, newImage.Height);
}
In this example, the image is retrieved from the assembly and loaded as a stream. The image is downsized using
the Downsize method, with the argument specifying that its largest dimension should be set to 100 pixels. In
addition, the source image is disposed. The downsized image is then drawn at actual size at (10,10).
Save an image
Images can be saved by the IImage.Save and IImage.SaveAsync methods. Each method saves the IImage to a
Stream , and enables optional ImageFormat and quality values to be specified.
using Microsoft.Maui.Graphics.Platform;
...
IImage image;
Assembly assembly = GetType().GetTypeInfo().Assembly;
using (Stream stream =
assembly.GetManifestResourceStream("GraphicsViewDemos.Resources.Images.dotnet_bot.png"))
{
image = PlatformImage.FromStream(stream);
}
In this example, the image is retrieved from the assembly and loaded as a stream. The image is downsized using
the Downsize method, with the argument specifying that its largest dimension should be set to 150 pixels. In
addition, the source image is disposed. The downsized image is then saved to a stream.
Paint graphical objects
3/18/2022 • 12 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) graphics includes the ability to paint graphical objects with solid colors,
gradients, repeating images, and patterns.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The Paint class is an abstract class that paints an object with its output. Classes that derive from Paint
describe different ways of painting an object. The following list describes the different paint types available in
.NET MAUI graphics:
SolidPaint , which paints an object with a solid color. For more information, see Paint a solid color.
ImagePaint , which paints an object with an image. For more information, see Paint an image.
PatternPaint , which paints an object with a pattern. For more information, see Paint a pattern.
GradientPaint , which paints an object with a gradient. For more information, see Paint a gradient.
Instances of these types can be painted on an ICanvas , typically by using the SetFillPaint method to set the
paint as the fill of a graphical object.
The Paint class also defines BackgroundColor , and ForegroundColor properties, of type Color , that can be
used to optionally define background and foreground colors for a Paint object.
The SolidPaint object is specified as the first argument to the SetFillPaint method. Therefore, a filled
rounded rectangle is painted with a silver SolidPaint object:
Alternatively, the color can be specified with the Color property:
Paint an image
The ImagePaint class, that's derived from the Paint class, is used to paint a graphical object with an image.
The ImagePaint class defines an Image property, of type IImage , which represents the image to paint. The class
also has an IsTransparent property that returns false .
Create an ImagePaint object
To paint an object with an image, load the image and assign it to the Image property of the ImagePaint object.
NOTE
Loading an image that's embedded in an assembly requires the image to have its build action set to Embedded
Resource .
The following example shows how to load an image and fill a rectangle with it:
using Microsoft.Maui.Graphics.Platform;
...
IImage image;
var assembly = GetType().GetTypeInfo().Assembly;
using (var stream = assembly.GetManifestResourceStream("GraphicsViewDemos.Resources.Images.dotnet_bot.png"))
{
image = PlatformImage.FromStream(stream);
}
if (image != null)
{
ImagePaint imagePaint = new ImagePaint
{
Image = image.Downsize(100)
};
canvas.SetFillPaint(imagePaint, RectF.Zero);
canvas.FillRectangle(0, 0, 240, 300);
}
In this example, the image is retrieved from the assembly and loaded as a stream. The image is resized using the
Downsize method, with the argument specifying that its largest dimension should be set to 100 pixels. For more
information about downsizing an image, see Downsize an image.
The Image property of the ImagePaint object is set to the downsized version of the image, and the ImagePaint
object is set as the paint to fill an object with. A rectangle is then drawn that's filled with the paint:
NOTE
An ImagePaint object can also be created from an IImage object by the AsPaint extension method.
Alternatively, the SetFillImage extension method can be used to simplify the code:
if (image != null)
{
canvas.SetFillImage(image.Downsize(100));
canvas.FillRectangle(0, 0, 240, 300);
}
Paint a pattern
The PatternPaint class, that's derived from the Paint class, is used to paint a graphical object with a pattern.
The PatternPaint class defines a Pattern property, of type IPattern , which represents the pattern to paint.
The class also has an IsTransparent property that returns a bool that represents whether the background or
foreground color of the paint has an alpha value of less than 1.
Create a PatternPaint object
To paint an area with a pattern, create the pattern and assign it to the Pattern property of a PatternPaint
object.
The following example shows how to create a pattern and fill an object with it:
IPattern pattern;
In this example, the pattern is a 10x10 area that contains a diagonal line from (0,0) to (10,10), and a diagonal line
from (0,10) to (10,0). The Pattern property of the PatternPaint object is set to the pattern, and the
PatternPaint object is set as the paint to fill an object with. A rectangle is then drawn that's filled with the paint:
NOTE
A PatternPaint object can also be created from a PicturePattern object by the AsPaint extension method.
Paint a gradient
The GradientPaint class, that's derived from the Paint class, is an abstract base class that describes a gradient,
which is composed of gradient steps. A GradientPaint paints a graphical object with multiple colors that blend
into each other along an axis. Classes that derive from GradientPaint describe different ways of interpreting
gradients stops, and .NET MAUI graphics provides the following gradient paints:
LinearGradientPaint , which paints an object with a linear gradient. For more information, see Paint a linear
gradient.
RadialGradientPaint , which paints an object with a radial gradient. For more information, see Paint a radial
gradient.
The GradientPaint class defines the GradientStops property, of type GradientStop , which represents the
brush's gradient stops, each of which specifies a color and an offset along the gradient axis.
Gradient stops
Gradient stops are the building blocks of a gradient, and specify the colors in the gradient and their location
along the gradient axis. Gradient stops are specified using GradientStop objects.
The GradientStop class defines the following properties:
, of type Color , which represents the color of the gradient stop.
Color
Offset , of type float , which represents the location of the gradient stop within the gradient vector. Valid
values are in the range 0.0-1.0. The closer this value is to 0, the closer the color is to the start of the gradient.
Similarly, the closer this value is to 1, the closer the color is to the end of the gradient.
IMPORTANT
The coordinate system used by gradients is relative to a bounding box for the graphical object. 0 indicates 0 percent of
the bounding box, and 1 indicates 100 percent of the bounding box. Therefore, (0.5,0.5) describes a point in the middle of
the bounding box, and (1,1) describes a point at the bottom right of the bounding box.
Gradient stops can be added to a GradientPoint object with the AddOffset method.
The following example creates a diagonal LinearGradientPaint with four colors:
linearGradientPaint.AddOffset(0.25f, Colors.Red);
linearGradientPaint.AddOffset(0.75f, Colors.Blue);
The color of each point between gradient stops is interpolated as a combination of the color specified by the two
bounding gradient stops. The following diagram shows the gradient stops from the previous example:
In this diagram, the circles mark the position of gradient stops, and the dashed line shows the gradient axis. The
first gradient stop specifies the color yellow at an offset of 0.0. The second gradient stop specifies the color red
at an offset of 0.25. The points between these two gradient stops gradually change from yellow to red as you
move from left to right along the gradient axis. The third gradient stop specifies the color blue at an offset of
0.75. The points between the second and third gradient stops gradually change from red to blue. The fourth
gradient stop specifies the color lime green at offset of 1.0. The points between the third and fourth gradient
stops gradually change from blue to lime green.
Paint a linear gradient
The LinearGradientPaint class, that's derived from the GradientPaint class, paints a graphical object with a
linear gradient. A linear gradient blends two or more colors along a line known as the gradient axis.
GradientStop objects are used to specify the colors in the gradient and their positions. For more information
about GradientStop objects, see Paint a gradient.
The LinearGradientPaint class defines the following properties:
StartPoint, of type Point , which represents the starting two-dimensional coordinates of the linear
gradient. The class constructor initializes this property to (0,0).
EndPoint , of type Point , which represents the ending two-dimensional coordinates of the linear gradient.
The class constructor initializes this property to (1,1).
Create a LinearGradientPaint object
A linear gradient's gradient stops are positioned along the gradient axis. The orientation and size of the gradient
axis can be changed using the StartPoint and EndPoint properties. By manipulating these properties, you can
create horizontal, vertical, and diagonal gradients, reverse the gradient direction, condense the gradient spread,
and more.
The StartPoint and EndPoint properties are relative to the graphical object being painted. (0,0) represents the
top-left corner of the object being painted, and (1,1) represents the bottom-right corner of the object being
painted. The following diagram shows the gradient axis for a diagonal linear gradient brush:
In this diagram, the dashed line shows the gradient axis, which highlights the interpolation path of the gradient
from the start point to the end point.
Create a horizontal linear gradient
To create a horizontal linear gradient, create a LinearGradientPaint object and set its StartColor and EndColor
properties. Then, set its EndPoint to (1,0).
The following example shows how to create a horizontal LinearGradientPaint :
In this example, the rounded rectangle is painted with a linear gradient that interpolates horizontally from
yellow to green:
In this example, the rounded rectangle is painted with a linear gradient that interpolates vertically from yellow to
green:
In this example, the rounded rectangle is painted with a linear gradient that interpolates diagonally from yellow
to green:
In this example, the rounded rectangle is painted with a radial gradient that interpolates from red to dark blue.
The center of the radial gradient is positioned in the center of the rectangle:
The following example moves the center of the radial gradient to the top-left corner of the rectangle:
In this example, the rounded rectangle is painted with a radial gradient that interpolates from red to dark blue.
The center of the radial gradient is positioned in the top-left of the rectangle:
The following example moves the center of the radial gradient to the bottom-right corner of the rectangle:
RadialGradientPaint radialGradientPaint = new RadialGradientPaint
{
StartColor = Colors.Red,
EndColor = Colors.DarkBlue,
Center = new Point(1.0, 1.0)
// Radius is already 0.5
};
In this example, the rounded rectangle is painted with a radial gradient that interpolates from red to dark blue.
The center of the radial gradient is positioned in the bottom-right of the rectangle:
Transforms
3/18/2022 • 8 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) graphics supports traditional graphics transforms, which are
implemented as methods on the ICanvas object. Mathematically, transforms alter the coordinates and sizes that
you specify in ICanvas drawing methods, when the graphical objects are rendered.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
Translate transform
The translate transform shifts graphical objects in the horizontal and vertical directions. Translation can be
considered unnecessary because the same result can be accomplished by changing the coordinates of the
drawing method you're using. However, when displaying a path, all the coordinates are encapsulated in the path,
and so it's often easier to apply a translate transform to shift the entire path.
The Translate method requires x and y arguments, of type float , that cause subsequently drawn graphic
objects to be shifted horizontally and vertically. Negative x values move an object to the left, while positive
values move an object to the right. Negative y values move an object up, while positive values move an object
down.
A common use of the translate transform is for rendering a graphical object that has been originally created
using coordinates that are convenient for drawing. The following example creates a PathF object for an 11-
pointed star:
if (i == 0)
path.MoveTo(point);
else
path.LineTo(point);
}
canvas.FillColor = Colors.Red;
canvas.Translate(150, 150);
canvas.FillPath(path);
The center of the star is at (0,0), and the points of the star are on a circle surrounding that point. Each point is a
combination of sine and cosine values of an angle that increases by 5/11ths of 360 degrees. The radius of the
circle is set as 100. If the PathF object is displayed without any transforms, the center of the star will be
positioned at the upper-left corner of the ICanvas , and only a quarter of it will be visible. Therefore, a translate
transform is used to shift the star horizontally and vertically to (150,150):
Scale transform
The scale transform changes the size of a graphical object, and can also often cause coordinates to move when a
graphical object is made larger.
The Scale method requires x and y arguments, of type float , that let you specify different values for
horizontal and vertical scaling, otherwise known as anisotropic scaling. The values of x and y have a
significant impact on the resulting scaling:
Values between 0 and 1 decrease the width and height of the scaled object.
Values greater than 1 increase the width and height of the scaled object.
Values of 1 indicate that the object is not scaled.
The following example demonstrates the Scale method:
canvas.StrokeColor = Colors.Red;
canvas.StrokeSize = 4;
canvas.StrokeDashPattern = new float[] { 2, 2 };
canvas.FontColor = Colors.Blue;
canvas.FontSize = 18;
canvas.Scale(2, 2);
canvas.DrawRoundedRectangle(50, 100, 80, 20, 5);
canvas.DrawString(".NET MAUI", 50, 100, 80, 20, HorizontalAlignment.Left, VerticalAlignment.Top);
In this example, ".NET MAUI" is displayed inside a rounded rectangle stroked with a dashed line. The same
graphical objects drawn after the Scale call increase in size proportionally:
The text and the rounded rectangle are both subject to the same scaling factors.
NOTE
Anisotropic scaling causes the stroke size to become different for lines aligned with the horizontal and vertical axes.
Order matters when you combine Translate and Scale calls. If the Translate call comes after the Scale call,
the translation factors are scaled by the scaling factors. If the Translate call comes before the Scale call, the
translation factors are not scaled.
Rotate transform
The rotate transform rotates a graphical object around a point. Rotation is clockwise for increasing angles.
Negative angles and angles greater than 360 degrees are allowed.
There are two Rotate overloads. The first requires a degrees argument, of type float , that defines the
rotation angle, and centers the rotation around the upper-left corner of the canvas (0,0). The following example
demonstrates this Rotate method:
canvas.FontColor = Colors.Blue;
canvas.FontSize = 18;
canvas.Rotate(45);
canvas.DrawString(".NET MAUI", 50, 50, HorizontalAlignment.Left);
canvas.FontColor = Colors.Blue;
canvas.FontSize = 18;
In this example, .NET MAUI is rotated 45 degrees around the center of the canvas.
Combine transforms
The simplest way to combine transforms is to begin with global transforms, followed by local transforms. For
example, translation, scaling, and rotation can be combined to draw an analog clock. The clock can be drawn
using an arbitrary coordinate system based on a circle that's centered at (0,0) with a radius of 100. Translation
and scaling expand and center the clock on the canvas, and rotation can then be used to draw the minute and
hour marks of the clock and to rotate the hands:
canvas.StrokeLineCap = LineCap.Round;
canvas.FillColor = Colors.Gray;
// Hour hand
canvas.StrokeSize = 20;
canvas.SaveState();
canvas.Rotate(30 * now.Hour + now.Minute / 2f);
canvas.DrawLine(0, 0, 0, -50);
canvas.RestoreState();
// Minute hand
canvas.StrokeSize = 10;
canvas.SaveState();
canvas.Rotate(6 * now.Minute + now.Second / 10f);
canvas.DrawLine(0, 0, 0, -70);
canvas.RestoreState();
// Second hand
canvas.StrokeSize = 2;
canvas.SaveState();
canvas.Rotate(6 * now.Second);
canvas.DrawLine(0, 10, 0, -80);
canvas.RestoreState();
In this example, the Translate and Scale calls apply globally to the clock, and so are called before the Rotate
method
There are 60 marks of two different sizes that are drawn in a circle around the clock. The FillCircle call draws
that circle at (0,-90), which relative to the center of the clock corresponds to 12:00. The Rotate call increments
the rotation angle by 6 degrees after every tick mark. The angle variable is used solely to determine if a large
circle or a small circle is drawn. Finally, the current time is obtained and rotation degrees are calculated for the
hour, minute, and second hands. Each hand is drawn in the 12:00 position so that the rotation angle is relative to
that:
Concatenate transforms
A transform can be described in terms of a 3x3 affine transformation matrix, which performs transformations in
2D space. The following table shows the structure of a 3x3 affine transformation matrix:
M11
M12
0.0
M21
M22
0.0
M31
M32
1.0
An affine transformation matrix has its final column equal to (0,0,1), so only members in the first two columns
need to be specified. Therefore, the 3x3 matrix is represented by the Matrix3x2 struct, from the
System.Numerics namespace, which is a collection of three rows and two columns of float values.
The six cells in the first two columns of the transform matrix represent values that performing scaling, shearing,
and translation:
ScaleX
ShearY
0.0
ShearX
ScaleY
0.0
TranslateX
TranslateY
1.0
For example, if you change the M31 value to 100, you can use it to translate a graphical object 100 pixels along
the x-axis. If you change the M22 value to 3, you can use it to stretch a graphical object to three times its current
height. If you change both values, you move the graphical object 100 pixels along the x-axis and stretch its
height by a factor of 3.
You can define a new transform matrix with the Matrix3x2 constructor. The advantage of specifying transforms
with a transform matrix is that composite transforms can be applied as a single transform, which is referred to
as concatenation. The Matrix3x2 struct also defines methods that can be used to manipulate matrix values.
The only ICanvas method that accepts a Matrix3x2 argument is the ConcatenateTransform method, which
combines multiple transforms into a single transform. The following example shows how to use this method to
transform a PathF object:
PathF path = new PathF();
for (int i = 0; i < 11; i++)
{
double angle = 5 * i * 2 * Math.PI / 11;
PointF point = new PointF(100 * (float)Math.Sin(angle), -100 * (float)Math.Cos(angle));
if (i == 0)
path.MoveTo(point);
else
path.LineTo(point);
}
In this example, the PathF object is scaled and sheared on the x-axis, and translated on the x-axis and the y-axis.
Winding modes
3/18/2022 • 2 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) graphics provides a WindingMode enumeration, that enables you to
specify the fill algorithm to be used by the FillPath method. Contours in a path can overlap, and any enclosed
area can potentially be filled, but you might not want to fill all the enclosed areas. For more information about
paths, see Draw a path.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The WindingMode enumeration defines NonZero and EvenOdd members. Each member represents a different
algorithm for determining whether a point is in the fill region of an enclosed area.
NOTE
The ClipPath method has an overload that enables a WindingMode argument to be specified. By default, this
argument is set to WindingMode.NonZero .
NonZero
The NonZero winding mode draws a hypothetical ray from the point to infinity in any direction and then
examines the places where a path contour crosses the ray. Starting with a count of zero, the count is
incremented each time a contour crosses the ray from left to right and decremented each time a contour crosses
the ray from right to left. After counting the crossings, if the result is zero then the area isn't filled. Otherwise, the
area is filled.
The following example fills a five-pointed star using the NonZero winding mode:
canvas.StrokeSize = 15;
canvas.StrokeLineJoin = LineJoin.Round;
canvas.StrokeColor = Colors.Red;
canvas.FillColor = Colors.Blue;
canvas.FillPath(path); // Overload automatically uses a NonZero winding mode
canvas.DrawPath(path);
In this example, the path is drawn twice. The FillPath method is used to fill the path with blue, while the
DrawPath method outlines the path with a red stroke. The FillPath overload used omits the WindingMode
argument, and instead automatically uses the NonZero winding mode. This results in all the enclosed areas of
the path being filled:
NOTE
For many paths, the NonZero winding mode often fills all the enclosed areas of a path.
EvenOdd
The EvenOdd winding mode draws a hypothetical ray from the point to infinity in any direction and counts the
number of path contours that the ray crosses. If this number is odd, then the area is filled. Otherwise, the area
isn't filled.
The following example fills a five-pointed star using the EvenOdd winding mode:
canvas.StrokeSize = 15;
canvas.StrokeLineJoin = LineJoin.Round;
canvas.StrokeColor = Colors.Red;
canvas.FillColor = Colors.Blue;
canvas.FillPath(path, WindingMode.EvenOdd);
canvas.DrawPath(path);
In this example, the path is drawn twice. The FillPath method is used to fill the path with blue, while the
DrawPath method outlines the path with a red stroke. The FillPath overload used specifies that the EvenOdd
winding mode is used. This results in the central area of the star not being filled:
App icons
3/18/2022 • 2 minutes to read • Edit Online
Every app has a logo icon that represents it, and that icon typically appears in multiple places. For example, on
iOS the app icon appears on the Home screen and throughout the system, such as in Settings, notifications, and
search results, and in the App Store. On Android, the app icon appears as a launcher icon and throughout the
system, such as on the action bar and in notifications, and in the Google Play Store. On Windows the app icon
appears in the app list in the start menu, on the taskbar and task manager, on the app's tile, and in the Microsoft
Store.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
A .NET Multi-platform App UI (.NET MAUI) app icon is a composite of an image and a background color. The
standard platform image formats are supported, including Scalable Vector Graphics (SVG) files.
TIP
The SVG format is the recommended image format for .NET MAUI app icons.
An app icon can be added to your app project by dragging an image into the Resources\Images folder of the
project, and setting the build action of the image to MauiIcon in the Proper ties window. This creates a
corresponding entry in your project file:
<ItemGroup>
<MauiIcon Include="Resources\Images\appicon.svg" />
</ItemGroup>
To comply with Android resource naming rules, app icon filenames must be lowercase, start and end with a
letter character, and contain only alphanumeric characters or underscores. For more information, see App
resources overview on developer.android.com.
A background color for the app icon can also be specified:
Color values can be specified in hexadecimal, or as a .NET MAUI color. For example, Color="Red" is valid.
At build time, the app icon is resized to the correct sizes for the target platform and device. The resized app icons
are then added to your app package.
Android
iOS
.NET MAUI supports creating an adaptive launcher icon on Android 8.0 and higher, from the app icon. Adaptive
launcher icons can display as a variety of shapes across different device models, including circular and square.
For more information about adaptive icons, see Adaptive icons on developer.android.com.
Adaptive launcher icons are defined using a background layer and a foreground layer, and an optional scaling
value:
The ForegroundFile attribute defines the main content of the icon. The Include attribute is used to specify the
background layer, and is typically set to a color or pattern. In addition, the ForegroundScale attribute can be
optionally specified to change the scaling of the foreground layer being rendered over the background layer.
IMPORTANT
The Color attribute can still be specified when producing an adaptive launcher icon on Android so that a background
color is generated for the app icon on other platforms.
On Android, drawable XML resources and bitmaps are generated from the app icon. An adaptive launcher icon
is generated when the ForegroundFile attribute is defined.
Splash screens
3/18/2022 • 2 minutes to read • Edit Online
On Android and iOS, .NET Multi-platform App UI (.NET MAUI) apps can display a splash screen while their
initialization process completes. The splash screen is displayed immediately when an app is launched, providing
immediate feedback to users while app resources are initialized:
Once the app is ready for interaction, its splash screen is dismissed.
Splash screens are a composite of an image and a background color. The standard platform image formats are
supported, including Scalable Vector Graphics (SVG) files.
TIP
The SVG format is the recommended image format for .NET MAUI splash screens.
.NET MAUI splash screens can be added to your app project by dragging an image into the Resources\Images
folder of the project, and setting the build action of the image to MauiSplashScreen in the Proper ties
window. This creates a corresponding entry in your project file:
<ItemGroup>
<MauiSplashScreen Include="Resources\Images\splashscreen.svg" />
</ItemGroup>
To comply with Android resource naming rules, splash screen files names must be lowercase, start and end with
a letter character, and contain only alphanumeric characters or underscores. For more information, see App
resources overview on developer.android.com.
A background color for your splash screen can also be specified:
Color values can be specified in hexadecimal, or as a .NET MAUI color. For example, Color="Red" is valid.
At build time, the splash screen image is resized to the correct size for the target platform and device. The splash
screen is then added to your app package.
Android
iOS
On Android, the splash screen is added to your app package as Resourcs/values/maui_colors.xml and
Resources/drawable/maui_splash_image.xml . .NET MAUI apps use the Maui.SplashTheme by default, which
ensures that a splash screen will be displayed if present. Therefore, you should not specify a different theme in
your manifest file or in your MainActivity class:
using Android.App;
using Android.Content.PM;
namespace MyMauiApp
{
[Activity(Theme = "@style/Maui.SplashTheme", MainLauncher = true, ConfigurationChanges =
ConfigChanges.ScreenSize | ConfigChanges.Orientation | ConfigChanges.UiMode | ConfigChanges.ScreenLayout |
ConfigChanges.SmallestScreenSize)]
public class MainActivity : MauiAppCompatActivity
{
}
}
The .NET Multi-platform App UI (.NET MAUI) Shadow class paints a shadow around a layout or view. The
VisualElement class has a Shadow bindable property, of type Shadow , that enables a shadow to be added to any
layout or view.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
IMPORTANT
The Brush property only currently supports a SolidColorBrush .
Create a Shadow
To add a shadow to a control, set the control's Shadow property to a Shadow object whose properties define its
appearance.
The following XAML example shows how to add a shadow to an Image :
<Image Source="dotnet_bot.png"
WidthRequest="250"
HeightRequest="310">
<Image.Shadow>
<Shadow Brush="Black"
Offset="20,20"
Radius="40"
Opacity="0.8" />
</Image.Shadow>
</Image>
In this example, a black shadow is painted around the outline of the image, with its offset specifying that it
appears at the right and bottom of the image:
Shadows can also be added to clipped objects, as shown in the following example:
<Image Source=https://aka.ms/campus.jpg
Aspect="AspectFill"
HeightRequest="220"
WidthRequest="220"
HorizontalOptions="Center">
<Image.Clip>
<EllipseGeometry Center="220,250"
RadiusX="220"
RadiusY="220" />
</Image.Clip>
<Image.Shadow>
<Shadow Brush="Black"
Offset="10,10"
Opacity="0.8" />
</Image.Shadow>
</Image>
In this example, a black shadow is painted around the outline of the EllipseGeometry that clips the image:
Style apps using XAML
3/18/2022 • 13 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) apps often contain multiple controls that have an identical appearance.
For example, an app may have multiple Label instances that have the same font options and layout options:
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
In this example, each Label object has identical property values for controlling the appearance of the text
displayed by the Label . However, setting the appearance of each individual control can be repetitive and error
prone. Instead, a style can be created that defines the appearance, and then applied to the required controls.
Introduction to styles
An app can be styled by using the Style class to group a collection of property values into one object that can
then be applied to multiple visual elements. This helps to reduce repetitive markup, and allows an apps
appearance to be more easily changed.
Although styles are designed primarily for XAML-based apps, they can also be created in C#:
Style objects created in XAML are typically defined in a ResourceDictionary that's assigned to the
Resources collection of a control, page, or to the Resources collection of the app.
Style objects created in C# are typically defined in the page's class, or in a class that can be globally
accessed.
Choosing where to define a Style impacts where it can be used:
Style instances defined at the control-level can only be applied to the control and to its children.
Style instances defined at the page-level can only be applied to the page and to its children.
Style instances defined at the app-level can be applied throughout the app.
Each Style object contains a collection of one or more Setter objects, with each Setter having a Property
and a Value . The Property is the name of the bindable property of the element the style is applied to, and the
Value is the value that is applied to the property.
Each Style object can be explicit, or implicit:
An explicit Style object is defined by specifying a TargetType and an x:Key value, and by setting the target
element's Style property to the x:Key reference. For more information, see Explicit styles.
An implicit Style object is defined by specifying only a TargetType . The Style object will then
automatically be applied to all elements of that type. However, the subclasses of the TargetType do not
automatically have the Style applied. For more information, see Implicit styles.
When creating a Style , the TargetType property is always required. The following example shows an explicit
style:
To apply a Style , the target object must be a VisualElement that matches the TargetType property value of the
Style :
Styles lower in the view hierarchy take precedence over those defined higher up. For example, setting a Style
that sets Label.TextColor to Red at the app-level will be overridden by a page-level style that sets
Label.TextColor to Green . Similarly, a page-level style will be overridden by a control-level style. In addition, if
Label.TextColor is set directly on a control property, this takes precedence over any styles.
Styles do not respond to property changes, and remain unchanged for the duration of an app. However, apps
can respond to style changes dynamically at runtime by using dynamic resources. For more information, see
Dynamic styles.
Explicit styles
To create a Style at the page-level, a ResourceDictionary must be added to the page and then one or more
Style declarations can be included in the ResourceDictionary . A Style is made explicit by giving its
declaration an x:Key attribute, which gives it a descriptive key in the ResourceDictionary . Explicit styles must
then be applied to specific visual elements by setting their Style properties.
The following example shows explicit styles in a page's ResourceDictionary , and applied to the page's Label
objects:
<ContentPage ...>
<ContentPage.Resources>
<Style x:Key="labelRedStyle"
TargetType="Label">
<Setter Property="HorizontalOptions" Value="Center" />
<Setter Property="VerticalOptions" Value="Center" />
<Setter Property="FontSize" Value="Large" />
<Setter Property="TextColor" Value="Red" />
</Style>
<Style x:Key="labelGreenStyle"
TargetType="Label">
<Setter Property="HorizontalOptions" Value="Center" />
<Setter Property="VerticalOptions" Value="Center" />
<Setter Property="FontSize" Value="Large" />
<Setter Property="TextColor" Value="Green" />
</Style>
<Style x:Key="labelBlueStyle"
TargetType="Label">
<Setter Property="HorizontalOptions" Value="Center" />
<Setter Property="VerticalOptions" Value="Center" />
<Setter Property="FontSize" Value="Large" />
<Setter Property="TextColor" Value="Blue" />
</Style>
</ContentPage.Resources>
<StackLayout>
<Label Text="These labels"
Style="{StaticResource labelRedStyle}" />
<Label Text="are demonstrating"
Style="{StaticResource labelGreenStyle}" />
<Label Text="explicit styles,"
Style="{StaticResource labelBlueStyle}" />
<Label Text="and an explicit style override"
Style="{StaticResource labelBlueStyle}"
TextColor="Teal" />
</StackLayout>
</ContentPage>
In this example, the ResourceDictionary defines three styles that are explicitly set on the page's Label objects.
Each Style is used to display text in a different color, while also setting the font size, and horizontal and vertical
layout options. Each Style is applied to a different Label by setting its Style properties using the
StaticResource markup extension. In addition, while the final Label has a Style set on it, it also overrides the
TextColor property to a different Color value.
Implicit styles
To create a Style at the page-level, a ResourceDictionary must be added to the page and then one or more
Style declarations can be included in the ResourceDictionary . A Style is made implicit by not specifying an
x:Key attribute. The style will then be applied to in scope visual elements that match the TargetType exactly,
but not to elements that are derived from the TargetType value.
The following code example shows an implicit style in a page's ResourceDictionary , and applied to the page's
Entry objects:
<ContentPage ...>
<ContentPage.Resources>
<Style TargetType="Entry">
<Setter Property="HorizontalOptions" Value="Fill" />
<Setter Property="VerticalOptions" Value="Center" />
<Setter Property="BackgroundColor" Value="Yellow" />
<Setter Property="FontAttributes" Value="Italic" />
<Setter Property="TextColor" Value="Blue" />
</Style>
</ContentPage.Resources>
<StackLayout>
<Entry Text="These entries" />
<Entry Text="are demonstrating" />
<Entry Text="implicit styles," />
<Entry Text="and an implicit style override"
BackgroundColor="Lime"
TextColor="Red" />
<local:CustomEntry Text="Subclassed Entry is not receiving the style" />
</StackLayout>
</ContentPage>
In this example, the ResourceDictionary defines a single implicit style that are implicitly set on the page's Entry
objects. The Style is used to display blue text on a yellow background, while also setting other appearance
options. The Style is added to the page's ResourceDictionary without specifying an x:Key attribute. Therefore,
the Style is applied to all the Entry objects implicitly as they match the TargetType property of the Style
exactly. However, the Style is not applied to the CustomEntry object, which is a subclassed Entry . In addition,
the fourth Entry overrides the BackgroundColor and TextColor properties of the style to different Color
values.
<Style TargetType="Button"
ApplyToDerivedTypes="True">
<Setter Property="BackgroundColor"
Value="Red" />
</Style>
Placing this style in a page-level ResourceDictionary will result in it being applied to all Button objects on the
page, and also to any controls that derive from Button . However, if the ApplyToDerivedTypes property remained
unset, the style would only be applied to Button objects.
Global styles
Styles can be defined globally by adding them to the app's resource dictionary. These styles can then be
consumed throughout an app, and help to avoid style duplication across pages and controls.
The following example shows a Style defined at the app-level:
<Application xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:Styles"
x:Class="Styles.App">
<Application.Resources>
<Style x:Key="buttonStyle" TargetType="Button">
<Setter Property="HorizontalOptions"
Value="Center" />
<Setter Property="VerticalOptions"
Value="CenterAndExpand" />
<Setter Property="BorderColor"
Value="Lime" />
<Setter Property="CornerRadius"
Value="5" />
<Setter Property="BorderWidth"
Value="5" />
<Setter Property="WidthRequest"
Value="200" />
<Setter Property="TextColor"
Value="Teal" />
</Style>
</Application.Resources>
</Application>
In this example, the ResourceDictionary defines a single explicit style, buttonStyle , which will be used to set the
appearance of Button objects.
NOTE
Global styles can be explicit or implicit.
The following example shows a page consuming the buttonStyle on the page's Button objects:
<ContentPage ...>
<StackLayout>
<Button Text="These buttons"
Style="{StaticResource buttonStyle}" />
<Button Text="are demonstrating"
Style="{StaticResource buttonStyle}" />
<Button Text="application styles"
Style="{StaticResource buttonStyle}" />
</StackLayout>
</ContentPage>
Style inheritance
Styles can inherit from other styles to reduce duplication and enable reuse. This is achieved by setting the
Style.BasedOn property to an existing Style . In XAML, this can be achieved by setting the BasedOn property to
a StaticResource markup extension that references a previously created Style .
Styles that inherit from a base style can include Setter instances for new properties, or use them to override
setters from the base style. In addition, styles that inherit from a base style must target the same type, or a type
that derives from the type targeted by the base style. For example, if a base style targets View objects, styles
that are based on the base style can target View objects or types that derive from the View class, such as
Label and Button objects.
A style can only inherit from styles at the same level, or above, in the view hierarchy. This means that:
An app-level style can only inherit from other app-level styles.
A page-level style can inherit from app-level styles, and other page-level styles.
A control-level style can inherit from app-level styles, page-level styles, and other control-level styles.
The following example shows explicit style inheritance:
<ContentPage ...>
<ContentPage.Resources>
<Style x:Key="baseStyle"
TargetType="View">
<Setter Property="HorizontalOptions" Value="Center" />
<Setter Property="VerticalOptions" Value="Center" />
</Style>
</ContentPage.Resources>
<StackLayout>
<StackLayout.Resources>
<Style x:Key="labelStyle"
TargetType="Label"
BasedOn="{StaticResource baseStyle}">
<Setter Property="FontSize" Value="Large" />
<Setter Property="FontAttributes" Value="Italic" />
<Setter Property="TextColor" Value="Teal" />
</Style>
<Style x:Key="buttonStyle"
TargetType="Button"
BasedOn="{StaticResource baseStyle}">
<Setter Property="BorderColor" Value="Lime" />
<Setter Property="CornerRadius" Value="5" />
<Setter Property="BorderWidth" Value="5" />
<Setter Property="WidthRequest" Value="200" />
<Setter Property="TextColor" Value="Teal" />
</Style>
</StackLayout.Resources>
<Label Text="This label uses style inheritance"
Style="{StaticResource labelStyle}" />
<Button Text="This button uses style inheritance"
Style="{StaticResource buttonStyle}" />
</StackLayout>
</ContentPage>
In this example, the baseStyle targets View objects, and sets the HorizontalOptions and VerticalOptions
properties. The baseStyle is not set directly on any controls. Instead, labelStyle and buttonStyle inherit from
it, setting additional bindable property values. The labelStyle and buttonStyle objects are then set on a
Label and Button .
IMPORTANT
An implicit style can be derived from an explicit style, but an explicit style can't be derived from an implicit style.
Dynamic styles
Styles do not respond to property changes, and remain unchanged for the duration of an app. For example, after
assigning a Style to a visual element, if one of the Setter objects is modified, removed, or a new Setter
added, the changes won't be applied to the visual element. However, apps can respond to style changes
dynamically at runtime by using dynamic resources.
The DynamicResource markup extension is similar to the StaticResource markup extension in that both use a
dictionary key to fetch a value from a ResourceDictionary . However, while the StaticResource performs a single
dictionary lookup, the DynamicResource maintains a link to the dictionary key. Therefore, if the dictionary entry
associated with the key is replaced, the change is applied to the visual element. This enables runtime style
changes to be made in an app.
The following example shows dynamic styles:
<ContentPage ...>
<ContentPage.Resources>
<Style x:Key="baseStyle"
TargetType="View">
<Setter Property="VerticalOptions" Value="Center" />
</Style>
<Style x:Key="blueSearchBarStyle"
TargetType="SearchBar"
BasedOn="{StaticResource baseStyle}">
<Setter Property="FontAttributes" Value="Italic" />
<Setter Property="PlaceholderColor" Value="Blue" />
</Style>
<Style x:Key="greenSearchBarStyle"
TargetType="SearchBar">
<Setter Property="FontAttributes" Value="None" />
<Setter Property="PlaceholderColor" Value="Green" />
</Style>
</ContentPage.Resources>
<StackLayout>
<SearchBar Placeholder="SearchBar demonstrating dynamic styles"
Style="{DynamicResource blueSearchBarStyle}" />
</StackLayout>
</ContentPage>
In this example, the object use the DynamicResource markup extension to set a
SearchBar Style named
blueSearchBarStyle . The SearchBar can then have its Style definition updated in code:
Resources["blueSearchBarStyle"] = Resources["greenSearchBarStyle"];
In this example, the blueSearchBarStyle definition is updated to use the values from the greenSearchBarStyle
definition. When this code is executed, the SearchBar will be updated to use the Setter objects defined in
greenSearchBarStyle .
In this example, the SearchBarobject uses the StaticResource markup extension to reference a Style named
tealSearchBarStyle . This Style sets some additional properties and uses the BaseResourceKey property to
reference blueSearchBarStyle . The DynamicResource markup extension is not required because
tealSearchBarStyle will not change, except for the Style it derives from. Therefore, tealSearchBarStyle
maintains a link to blueSearchBarStyle and is updated when the base style changes.
The blueSearchBarStyle definition can be updated in code:
Resources["blueSearchBarStyle"] = Resources["greenSearchBarStyle"];
In this example, the blueSearchBarStyle definition is updated to use the values from the greenSearchBarStyle
definition. When this code is executed, the SearchBar will be updated to use the Setter objects defined in
greenSearchBarStyle .
Style classes
Style classes enable multiple styles to be applied to a control, without resorting to style inheritance.
A style class can be created by setting the Class property on a Style to a string that represents the class
name. The advantage this offers, over defining an explicit style using the x:Key attribute, is that multiple style
classes can be applied to a VisualElement .
IMPORTANT
Multiple styles can share the same class name, provided they target different types. This enables multiple style classes,
that are identically named, to target different types.
The following example shows three BoxView style classes, and a VisualElement style class:
<ContentPage ...>
<ContentPage.Resources>
<Style TargetType="BoxView"
Class="Separator">
<Setter Property="BackgroundColor"
Value="#CCCCCC" />
<Setter Property="HeightRequest"
Value="1" />
</Style>
<Style TargetType="BoxView"
Class="Rounded">
<Setter Property="BackgroundColor"
Value="#1FAECE" />
<Setter Property="HorizontalOptions"
Value="Start" />
<Setter Property="CornerRadius"
Value="10" />
</Style>
<Style TargetType="BoxView"
Class="Circle">
<Setter Property="BackgroundColor"
Value="#1FAECE" />
<Setter Property="WidthRequest"
Value="100" />
<Setter Property="HeightRequest"
Value="100" />
<Setter Property="HorizontalOptions"
Value="Start" />
<Setter Property="CornerRadius"
Value="50" />
</Style>
<Style TargetType="VisualElement"
Class="Rotated"
ApplyToDerivedTypes="true">
<Setter Property="Rotation"
Value="45" />
</Style>
</ContentPage.Resources>
</ContentPage>
In this example, the Separator , Rounded , and Circle style classes each set BoxView properties to specific
values. The Rotated style class has a TargetType of VisualElement , which means it can only be applied to
VisualElement instances. However, its ApplyToDerivedTypes property is set to true , which ensures that it can be
applied to any controls that derive from VisualElement , such as BoxView . For more information about applying
a style to a derived type, see Apply a style to derived types.
Style classes can be consumed by setting the StyleClass property of the control, which is of type
IList<string> , to a list of style class names. The style classes will be applied, provided that the type of the
control matches the TargetType of the style classes.
The following example shows three BoxView instances, each set to different style classes:
<ContentPage ...>
<ContentPage.Resources>
...
</ContentPage.Resources>
<StackLayout>
<BoxView StyleClass="Separator" />
<BoxView WidthRequest="100"
HeightRequest="100"
HorizontalOptions="Center"
StyleClass="Rounded, Rotated" />
<BoxView HorizontalOptions="Center"
StyleClass="Circle" />
</StackLayout>
</ContentPage>
In this example, the first BoxView is styled to be a line separator, while the third BoxView is circular. The second
BoxView has two style classes applied to it, which give it rounded corners and rotate it 45 degrees:
IMPORTANT
Multiple style classes can be applied to a control because the StyleClass property is of type IList<string> . When
this occurs, style classes are applied in ascending list order. Therefore, when multiple style classes set identical properties,
the property in the style class that's in the highest list position will take precedence.
Style apps using Cascading Style Sheets
3/18/2022 • 13 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) apps can be styled using Cascading Style Sheets (CSS). A style sheet
consists of a list of rules, with each rule consisting of one or more selectors and a declaration block. A
declaration block consists of a list of declarations in braces, with each declaration consisting of a property, a
colon, and a value. When there are multiple declarations in a block, a semi-colon is inserted as a separator.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
^contentpage {
background-color: lightgray;
}
#listView {
background-color: lightgray;
}
stacklayout {
margin: 20;
-maui-spacing: 6;
}
grid {
row-gap: 6;
column-gap: 6;
}
.mainPageTitle {
font-style: bold;
font-size: medium;
}
.mainPageSubtitle {
margin-top: 15;
}
.detailPageTitle {
font-style: bold;
font-size: medium;
text-align: center;
}
.detailPageSubtitle {
text-align: center;
font-style: italic;
}
listview image {
height: 60;
width: 60;
}
stacklayout>image {
height: 200;
width: 200;
}
In .NET MAUI, CSS style sheets are parsed and evaluated at runtime, rather than compile time, and style sheets
are re-parsed on use.
IMPORTANT
It's not possible to fully style a .NET MAUI app using CSS. However, XAML styles can be used to supplement CSS. For
more information about XAML styles, see Style apps using XAML.
NOTE
It's not possible to change a style sheet at runtime and have the new style sheet applied.
<Application ...>
<Application.Resources>
<StyleSheet Source="/Resources/styles.css" />
</Application.Resources>
</Application>
The StyleSheet.Source property specifies the style sheet as a URI relative to the location of the enclosing XAML
file, or relative to the project root if the URI starts with a / .
WARNING
The CSS file will fail to load if its build action is not set to MauiCss .
Alternatively, a style sheet can be loaded and parsed with the StyleSheet class, before being added to a
ResourceDictionary , by inlining it in a CDATA section:
<ContentPage ...>
<ContentPage.Resources>
<StyleSheet>
<![CDATA[
^contentpage {
background-color: lightgray;
}
]]>
</StyleSheet>
</ContentPage.Resources>
...
</ContentPage>
The argument to the StyleSheet.FromReader method is the TextReader that has read the style sheet.
NOTE
CSS variables are unsupported.
stacklayout {
margin: 20;
}
This selector identifies any StackLayout elements on pages that consume the style sheet, and sets their margins
to a uniform thickness of 20.
NOTE
The element selector does not identify subclasses of the specified type.
Selecting elements by base class
Elements in the visual tree can be selected by base class with the case insensitive ^base selector:
^contentpage {
background-color: lightgray;
}
This selector identifies any ContentPage elements that consume the style sheet, and sets their background color
to lightgray .
NOTE
The ^base selector is specific to .NET MAUI, and isn't part of the CSS specification.
#listView {
background-color: lightgray;
}
This selector identifies the element whose StyleId property is set to listView . However, if the StyleId
property is not set, the selector will fall back to using the x:Name of the element. Therefore, in the following
example, the #listView selector will identify the ListView whose x:Name attribute is set to listView , and will
set it's background color to lightgray .
<ContentPage ...>
<ContentPage.Resources>
<StyleSheet Source="/Resources/styles.css" />
</ContentPage.Resources>
<StackLayout>
<ListView x:Name="listView">
...
</ListView>
</StackLayout>
</ContentPage>
.detailPageTitle {
font-style: bold;
font-size: medium;
text-align: center;
}
.detailPageSubtitle {
text-align: center;
font-style: italic;
}
A CSS class can be assigned to a XAML element by setting the StyleClass property of the element to the CSS
class name. Therefore, in the following example, the styles defined by the .detailPageTitle class are assigned
to the first Label , while the styles defined by the .detailPageSubtitle class are assigned to the second Label .
<ContentPage ...>
<ContentPage.Resources>
<StyleSheet Source="/Resources/styles.css" />
</ContentPage.Resources>
<ScrollView>
<StackLayout>
<Label ... StyleClass="detailPageTitle" />
<Label ... StyleClass="detailPageSubtitle"/>
</StackLayout>
</ScrollView>
</ContentPage>
listview image {
height: 60;
width: 60;
}
This selector identifies any Image elements that are children of ListView elements, and sets their height and
width to 60. Therefore, in the following XAML example, the listview image selector will identify the Image
that's a child of the ListView , and sets its height and width to 60.
<ContentPage ...>
<ContentPage.Resources>
<StyleSheet Source="/Resources/styles.css" />
</ContentPage.Resources>
<StackLayout>
<ListView ...>
<ListView.ItemTemplate>
<DataTemplate>
<ViewCell>
<Grid>
...
<Image ... />
...
</Grid>
</ViewCell>
</DataTemplate>
</ListView.ItemTemplate>
</ListView>
</StackLayout>
</ContentPage>
NOTE
The element element selector does not require the child element to be a direct child of the parent – the child element
may have a different parent. Selection occurs provided that an ancestor is the specified first element.
stacklayout>image {
height: 200;
width: 200;
}
This selector identifies any Image elements that are direct children of StackLayout elements, and sets their
height and width to 200. Therefore, in the following example, the stacklayout>image selector will identify the
Image that's a direct child of the StackLayout , and sets its height and width to 200.
<ContentPage ...>
<ContentPage.Resources>
<StyleSheet Source="/Resources/styles.css" />
</ContentPage.Resources>
<ScrollView>
<StackLayout>
...
<Image ... />
...
</StackLayout>
</ScrollView>
</ContentPage>
NOTE
The element>element selector requires that the child element is a direct child of the parent.
Selector reference
The following CSS selectors are supported by .NET MAUI:
Styles with matching selectors are applied consecutively, in definition order. Styles defined on a specific item are
always applied last.
TIP
Selectors can be combined without limitation, such as StackLayout>ContentView>label.email .
NOTE
Specificity, and specificity overrides are unsupported.
Property reference
The following CSS properties are supported by .NET MAUI (in the Values column, types are italic, while string
literals are gray ):
spacebetween |
spacearound |
spaceevenly |
flex-start | flex-end |
space-between |
space-around | initial
P RO P ERT Y A P P L IES TO VA L UES EXA M P L E
flex-start | flex-end |
initial
CheckBox , DatePicker ,
Editor , Entry , Label ,
Picker , ProgressBar ,
SearchBar , Switch ,
TimePicker
row-reverse |
column-reverse |
initial
| initial
spacearound |
spaceevenly |
flex-start | flex-end |
space-between |
space-around | initial
SearchHandler initial
NOTE
initial is a valid value for all properties. It clears the value (resets to default) that was set from another style.
In addition, there's no inherit value and so inheritance isn't supported. Therefore you can't, for example, set the
font-size property on a layout and expect all the Label instances in the layout to inherit the value. The one
exception is the direction property, which has a default value of inherit .
IMPORTANT
Span elements can't be targeted using CSS.
only supported on a
ScrollView .
Color
The following color values are supported:
X11 colors, which match CSS colors and .NET MAUI colors. These color values are case insensitive.
hex colors: #rgb , #argb , #rrggbb , #aarrggbb
rgb colors: rgb(255,0,0) , rgb(100%,0%,0%) . Values are in the range 0-255, or 0%-100%.
rgba colors: rgba(255, 0, 0, 0.8) , rgba(100%, 0%, 0%, 0.8) . The opacity value is in the range 0.0-1.0.
hsl colors: hsl(120, 100%, 50%) . The h value is in the range 0-360, while s and l are in the range 0%-100%.
hsla colors: hsla(120, 100%, 50%, .8) . The opacity value is in the range 0.0-1.0.
Thickness
One, two, three, or four thickness values are supported, each separated by white space:
A single value indicates uniform thickness.
Two values indicate vertical then horizontal thickness.
Three values indicate top, then horizontal (left and right), then bottom thickness.
Four values indicate top, then right, then bottom, then left thickness.
NOTE
CSS thickness values differ from XAML Thickness values. For example, in XAML a two-value Thickness indicates
horizontal then vertical thickness, while a four-value Thickness indicates left, then top, then right, then bottom
thickness. In addition, XAML Thickness values are comma delimited.
NamedSize
The following case insensitive namedsize values are supported:
default
micro
small
medium
large
body
header
title
subtitle
caption
Functions
Linear and radial gradients can be specified using the linear-gradient() and radial-gradient() CSS functions,
respectively. The result of these functions should be assigned to the background property of a control.
Theme an app
3/18/2022 • 4 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) apps can respond to style changes dynamically at runtime by using the
DynamicResource markup extension. This markup extension is similar to the StaticResource markup extension,
in that both use a dictionary key to fetch a value from a ResourceDictionary . However, while the StaticResource
markup extension performs a single dictionary lookup, the DynamicResource markup extension maintains a link
to the dictionary key. Therefore, if the value associated with the key is replaced, the change is applied to the
VisualElement . This enables runtime theming to be implemented in .NET MAUI apps.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The process for implementing runtime theming in a .NET MAUI app is as follows:
1. Define the resources for each theme in a ResourceDictionary . For more information, see Define themes.
2. Set a default theme in the app's App.xaml file. For more information, see Set a default theme.
3. Consume theme resources in the app, using the DynamicResource markup extension. For more information,
see Consume theme resources.
4. Add code to load a theme at runtime. For more information, see Load a theme at runtime.
IMPORTANT
Use the StaticResource markup extension if you don't need to change the app theme at runtime.
The following screenshot shows themed pages, with the iOS app using a light theme and the Android app using
a dark theme:
NOTE
Changing a theme at runtime requires the use of XAML styles, and is not possible using CSS.
Define themes
A theme is defined as a collection of resource objects stored in a ResourceDictionary .
The following example shows a ResourceDictionary for a light theme named LightTheme :
<ResourceDictionary xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="ThemingDemo.LightTheme">
<Color x:Key="PageBackgroundColor">White</Color>
<Color x:Key="NavigationBarColor">WhiteSmoke</Color>
<Color x:Key="PrimaryColor">WhiteSmoke</Color>
<Color x:Key="SecondaryColor">Black</Color>
<Color x:Key="PrimaryTextColor">Black</Color>
<Color x:Key="SecondaryTextColor">White</Color>
<Color x:Key="TertiaryTextColor">Gray</Color>
<Color x:Key="TransparentColor">Transparent</Color>
</ResourceDictionary>
The following example shows a ResourceDictionary for a dark theme named DarkTheme :
<ResourceDictionary xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="ThemingDemo.DarkTheme">
<Color x:Key="PageBackgroundColor">Black</Color>
<Color x:Key="NavigationBarColor">Teal</Color>
<Color x:Key="PrimaryColor">Teal</Color>
<Color x:Key="SecondaryColor">White</Color>
<Color x:Key="PrimaryTextColor">White</Color>
<Color x:Key="SecondaryTextColor">White</Color>
<Color x:Key="TertiaryTextColor">WhiteSmoke</Color>
<Color x:Key="TransparentColor">Transparent</Color>
</ResourceDictionary>
Each ResourceDictionary contains Color resources that define their respective themes, with each
ResourceDictionary using identical key values. For more information about resource dictionaries, see Resource
Dictionaries.
IMPORTANT
A code behind file is required for each ResourceDictionary , which calls the InitializeComponent method. This is
necessary so that a CLR object representing the chosen theme can be created at runtime.
For more information about merging resource dictionaries, see Merged resource dictionaries.
<Application xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="ThemingDemo.App">
<Application.Resources>
<Style x:Key="LargeLabelStyle"
TargetType="Label">
<Setter Property="TextColor"
Value="{DynamicResource SecondaryTextColor}" />
<Setter Property="FontSize"
Value="30" />
</Style>
<Style x:Key="MediumLabelStyle"
TargetType="Label">
<Setter Property="TextColor"
Value="{DynamicResource PrimaryTextColor}" />
<Setter Property="FontSize"
Value="25" />
</Style>
<Style x:Key="SmallLabelStyle"
TargetType="Label">
<Setter Property="TextColor"
Value="{DynamicResource TertiaryTextColor}" />
<Setter Property="FontSize"
Value="15" />
</Style>
</Application.Resources>
</Application>
These styles are defined in the app-level resource dictionary, so that they can be consumed by multiple pages.
Each style consumes theme resources with the DynamicResource markup extension.
These styles are then consumed by pages:
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:ThemingDemo"
x:Class="ThemingDemo.UserSummaryPage"
Title="User Summary"
BackgroundColor="{DynamicResource PageBackgroundColor}">
...
<ScrollView>
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="200" />
<RowDefinition Height="120" />
<RowDefinition Height="70" />
</Grid.RowDefinitions>
<Grid BackgroundColor="{DynamicResource PrimaryColor}">
<Label Text="Face-Palm Monkey"
VerticalOptions="Center"
Margin="15"
Style="{StaticResource MediumLabelStyle}" />
...
</Grid>
<StackLayout Grid.Row="1"
Margin="10">
<Label Text="This monkey reacts appropriately to ridiculous assertions and actions."
Style="{StaticResource SmallLabelStyle}" />
<Label Text=" • Cynical but not unfriendly."
Style="{StaticResource SmallLabelStyle}" />
<Label Text=" • Seven varieties of grimaces."
Style="{StaticResource SmallLabelStyle}" />
<Label Text=" • Doesn't laugh at your jokes."
Style="{StaticResource SmallLabelStyle}" />
</StackLayout>
...
</Grid>
</ScrollView>
</ContentPage>
When a theme resource is consumed directly, it should be consumed with the DynamicResource markup
extension. However, when a style that uses the DynamicResource markup extension is consumed, it should be
consumed with the StaticResource markup extension.
Any VisualElement objects that set properties with the DynamicResource markup extension will then apply the
new theme values. This occurs because the DynamicResource markup extension maintains a link to dictionary
keys. Therefore, when the values associated with keys are replaced, the changes are applied to the
VisualElement objects.
In the sample application, a theme is selected via a modal page that contains a Picker . The following code
shows the OnPickerSelectionChanged method, which is executed when the selected theme changes:
The following example shows removing the current theme and loading a new theme:
ICollection<ResourceDictionary> mergedDictionaries = Application.Current.Resources.MergedDictionaries;
if (mergedDictionaries != null)
{
mergedDictionaries.Clear();
mergedDictionaries.Add(new DarkTheme());
}
Respond to system theme changes
3/18/2022 • 4 minutes to read • Edit Online
Devices typically include light and dark themes, which each refer to a broad set of appearance preferences that
can be set at the operating system level. Apps should respect these system themes, and respond immediately
when the system theme changes.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The system theme may change for a variety of reasons, depending on the device configuration. This includes the
system theme being explicitly changed by the user, it changing due to the time of day, and it changing due to
environmental factors such as low light.
.NET Multi-platform App UI (.NET MAUI) apps can respond to system theme changes by consuming resources
with the AppThemeBinding markup extension, and the SetAppThemeColor and SetOnAppTheme<T> extension
methods.
NOTE
.NET MAUI apps can respond to system theme changes on iOS 13 or greater, Android 10 (API 29) or greater, macOS
10.14 or greater, and Windows 10 or greater.
The following screenshot shows themed pages, for the light system theme on iOS and the dark system theme
on Android:
<StackLayout>
<Label Text="This text is green in light mode, and red in dark mode."
TextColor="{AppThemeBinding Light=Green, Dark=Red}" />
<Image Source="{AppThemeBinding Light=lightlogo.png, Dark=darklogo.png}" />
</StackLayout>
In this example, the text color of the first Label is set to green when the device is using its light theme, and is
set to red when the device is using its dark theme. Similarly, the Image displays a different image file based
upon the current system theme.
In addition, resources defined in a ResourceDictionary can be consumed with the StaticResource markup
extension:
<ContentPage ...>
<ContentPage.Resources>
<Style x:Key="ButtonStyle"
TargetType="Button">
<Setter Property="BackgroundColor"
Value="{AppThemeBinding Light={StaticResource LightPrimaryColor}, Dark={StaticResource
DarkPrimaryColor}}" />
<Setter Property="TextColor"
Value="{AppThemeBinding Light={StaticResource LightSecondaryColor}, Dark={StaticResource
DarkSecondaryColor}}" />
</Style>
</ContentPage.Resources>
In this example, the background color of the Grid and the Button style changes based on whether the device is
using its light theme or dark theme.
For more information about the AppThemeBinding markup extension, see AppThemeBinding markup extension.
Extension methods
.NET MAUI includes SetAppThemeColor and SetOnAppTheme<T> extension methods that enable VisualElement
objects to respond to system theme changes.
The SetAppThemeColor method enables Color objects to be specified that will be set on a target property based
on the current system theme:
In this example, the text color of the Label is set to green when the device is using its light theme, and is set to
red when the device is using its dark theme.
The SetOnAppTheme<T> method enables objects of type T to be specified that will be set on a target property
based on the current system theme:
In this example, the Image displays lightlogo.png when the device is using its light theme, and darklogo.png
when the device is using its dark theme.
The RequestedTheme property returns an AppTheme enumeration member. The AppTheme enumeration defines
the following members:
Unspecified , which indicates that the device is using an unspecified theme.
Light , which indicates that the device is using its light theme.
Dark , which indicates that the device is using its dark theme.
Application.Current.UserAppTheme = AppTheme.Dark;
In this example, the app is set to use the theme defined for the system dark mode, regardless of which system
theme is currently operational.
NOTE
Set the UserAppTheme property to AppTheme.Unspecified to default to the operational system theme.
The AppThemeChangedEventArgs object, which accompanies the RequestedThemeChanged event, has a single
property named RequestedTheme , of type AppTheme . This property can be examined to detect the requested
system theme.
IMPORTANT
To respond to theme changes on Android your MainActivity class must include the ConfigChanges.UiMode flag in the
Activity attribute. .NET MAUI apps created with the Visual Studio project templates automatically include this flag.
Visual states
3/18/2022 • 9 minutes to read • Edit Online
The .NET Multi-platform App UI (.NET MAUI) Visual State Manager provides a structured way to make visual
changes to the user interface from code. In most cases, the user interface of an app is defined in XAML, and this
XAML can include markup describing how the Visual State Manager affects the visuals of the user interface.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The Visual State Manager introduces the concept of visual states. A .NET MAUI view such as a Button can have
several different visual appearances depending on its underlying state — whether it's disabled, or pressed, or
has input focus. These are the button's states. Visual states are collected in visual state groups. All the visual
states within a visual state group are mutually exclusive. Both visual states and visual state groups are identified
by simple text strings.
The .NET MAUI Visual State Manager defines a visual state group named CommonStates with the following visual
states:
Normal
Disabled
Focused
Selected
The Normal , Disabled , and Focused visual states are supported on all classes that derive from VisualElement ,
which is the base class for View and Page . In addition, you can also define your own visual state groups and
visual states.
The advantage of using the Visual State Manager to define appearance, rather than accessing visual elements
directly from code-behind, is that you can control how visual elements react to different state entirely in XAML,
which keeps all of the UI design in one location.
NOTE
Triggers can also make changes to visuals in the user interface based on changes in a view's properties or the firing of
events. However, using triggers to deal with various combinations of these changes can become confusing. With the
Visual State Manager, the visual states within a visual state group are always mutually exclusive. At any time, only one
state in each group is the current state.
IMPORTANT
In order for visual state Setter objects to function correctly, a VisualStateGroup must contain a VisualState object
for the Normal state. If this visual state does not have any Setter objects, it should be included as an empty visual
state ( <VisualState x:Name="Normal" /> ).
<Entry FontSize="18">
<VisualStateManager.VisualStateGroups>
<VisualStateGroup x:Name="CommonStates">
<VisualState x:Name="Normal">
<VisualState.Setters>
<Setter Property="BackgroundColor" Value="Lime" />
</VisualState.Setters>
</VisualState>
<VisualState x:Name="Focused">
<VisualState.Setters>
<Setter Property="FontSize" Value="36" />
</VisualState.Setters>
</VisualState>
<VisualState x:Name="Disabled">
<VisualState.Setters>
<Setter Property="BackgroundColor" Value="Pink" />
</VisualState.Setters>
</VisualState>
</VisualStateGroup>
</VisualStateManager.VisualStateGroups>
</Entry>
The following screenshot shows the Entry in its three defined visual states:
When the Entry is in the Normal state, its background is lime. When the the Entry gains input focus its font
size doubles. When the Entry becomes disabled, its background becomes pink. The Entry does not retain
doesn't retain its lime background when it gains input focus. As the Visual State Manager switches between the
visual states, the properties set by the previous state are unset. Therefore, the visual states are mutually
exclusive.
If you want the Entry to have a lime background in the Focused state, add another Setter to that visual state:
<VisualState x:Name="Focused">
<VisualState.Setters>
<Setter Property="FontSize" Value="36" />
<Setter Property="BackgroundColor" Value="Lime" />
</VisualState.Setters>
</VisualState>
The following example shows an implicit style for an Entry that defines the common visual states:
<Style TargetType="Entry">
<Setter Property="FontSize" Value="18" />
<Setter Property="VisualStateManager.VisualStateGroups">
<VisualStateGroupList>
<VisualStateGroup x:Name="CommonStates">
<VisualState x:Name="Normal">
<VisualState.Setters>
<Setter Property="BackgroundColor" Value="Lime" />
</VisualState.Setters>
</VisualState>
<VisualState x:Name="Focused">
<VisualState.Setters>
<Setter Property="FontSize" Value="36" />
<Setter Property="BackgroundColor" Value="Lime" />
</VisualState.Setters>
</VisualState>
<VisualState x:Name="Disabled">
<VisualState.Setters>
<Setter Property="BackgroundColor" Value="Pink" />
</VisualState.Setters>
</VisualState>
</VisualStateGroup>
</VisualStateGroupList>
</Setter>
</Style>
When this style is included in a page-level resource dictionary, the Style object will be applied to all Entry
objects on the page. Therefore, all Entry objects on the page will respond in the same way to their visual states.
Visual states in .NET MAUI
The following table lists the visual states that are defined in .NET MAUI:
C L A SS STAT ES M O RE IN F O RM AT IO N
<Setter TargetName="label"
Property="Label.TextColor"
Value="Red" />
In this example, a Label named label will have its TextColor property set to Red . When setting the
TargetName property you must specify the full path to the property in Property . Therefore, to set the
TextColor property on a Label , Property is specified as Label.TextColor .
NOTE
Any property referenced by a Setter object must be backed by a bindable property.
The following example shows how to set state on multiple objects, from a single visual state group:
<StackLayout>
<Label Text="What is the capital of France?" />
<Entry x:Name="entry"
Placeholder="Enter answer" />
<Button Text="Reveal answer">
<VisualStateManager.VisualStateGroups>
<VisualStateGroup x:Name="CommonStates">
<VisualState x:Name="Normal" />
<VisualState x:Name="Pressed">
<VisualState.Setters>
<Setter Property="Scale"
Value="0.8" />
<Setter TargetName="entry"
Property="Entry.Text"
Value="Paris" />
</VisualState.Setters>
</VisualState>
</VisualStateGroup>
</VisualStateManager.VisualStateGroups>
</Button>
</StackLayout>
In this example, the Normal state is active when the Button isn't pressed, and a response can be entered into
the Entry . The Pressed state becomes active when the Button is pressed, and specifies that its Scale
property will be changed from the default value of 1 to 0.8. In addition, the Entry named entry will have its
Text property set to Paris. Therefore, the result is that when the Button is pressed it's rescaled to be slightly
smaller, and the Entry displays Paris:
Then, when the Button is released it's rescaled to its default value of 1 ,and the Entry displays any previously
entered text.
IMPORTANT
Property paths are unsupported in Setter elements that specify the TargetName property.
In this example, visual states are attached to the StackLayout , and there are two mutually-exclusive states
named Valid and Invalid . If the Entry does not contain a valid phone number, then the current state is
Invalid , and so the Entry has a pink background, the second Label is visible, and the Button is disabled.
When a valid phone number is entered, then the current state becomes Valid . The Entry gets a lime
background, the second Label disappears, and the Button is now enabled:
The code-behind file is responsible for handling the TextChanged event from the Entry . The handler uses a
regular expression to determine if the input string is valid or not. The GoToState method in the code-behind file
calls the static VisualStateManager.GoToState method on the StackLayout object:
GoToState(false);
}
In this example, the GoToState method is called from the constructor to initialize the state. There should always
be a current state. The code-behind file then calls VisualStateManager.GoToState , with a state name, on the object
that defines the visual states.
If multiple triggers are simultaneously active (for example, two custom triggers) then the first trigger declared in
the markup takes precedence.
For more information about state triggers, see State triggers.
Configure multi-targeting
3/18/2022 • 3 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) apps use multi-targeting to target multiple platforms from a single
project.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The project for a .NET MAUI app contains a Platforms folder, with each child folder representing a platform that
.NET MAUI can target:
The folders for each target platform contain platform-specific code that starts the app on each platform, plus
any additional platform code you add. At build time, the build system only includes the code from each folder
when building for that specific platform. For example, when you build for Android the files in the Platforms >
Android folder will be built into the app package, but the files in the other Platforms folders won't be.
In addition to this default multi-targeting approach, .NET MAUI apps can also be multi-targeted based on your
own filename and folder criteria. This enables you to structure your .NET MAUI app project so that you don't
have to place your platform code into sub-folders of the Platforms folder.
IMPORTANT
Filename-based multi-targeting can be combined with folder-based multi-targeting. For more information, see Combine
filename and folder multi-targeting.
This XML configures the build system to remove platform-based folder patterns under specific conditions:
Don't compile C# code that's located in the Android folder, or sub-folder of the Android folder, if you aren't
building for Android.
Don't compile C# code that's located in the iOS folder, or sub-folder of the iOS folder, if you aren't building
for iOS or MacCatalyst.
Don't compile C# code that's located in the Windows folder, or sub-folder of the Windows folder, if you aren't
building for Windows.
IMPORTANT
Folder-based multi-targeting can be combined with filename-based multi-targeting. For more information, see Combine
filename and folder multi-targeting.
This XML configures the build system to remove platform-based filename and folder patterns under specific
conditions:
Don't compile C# code whose filename ends with .Android.cs, or that's located in the Android folder or sub-
folder of the Android folder, if you aren't building for Android.
Don't compile C# code whose filename ends with .iOS.cs, or that's located in the iOS folder or sub-folder of
the iOS folder, if you aren't building for iOS or MacCatalyst.
Don't compile C# code whose filename ends with .Windows.cs, or that's located in the Windows folder or
sub-folder of the Windows folder, if you aren't building for Windows.
Invoke platform code
3/18/2022 • 4 minutes to read • Edit Online
.NET Multi-platform App UI (.NET MAUI) apps can combine multi-targeting with partial classes and partial
methods to invoke platform code from cross-platform code.
IMPORTANT
.NET Multi-platform App UI (.NET MAUI) is currently in preview. This content relates to a pre-release product that may be
substantially modified before it's released. Microsoft makes no warranties, express or implied, with respect to the
information provided here.
The project for a .NET MAUI app contains a Platforms folder, with each child folder representing a platform that
.NET MAUI can target:
The folders for each target platform contain platform-specific code that starts the app on each platform, plus
any additional platform code you add. At build time, the build system only includes the code from each folder
when building for that specific platform. For example, when you build for Android the files in the Platforms >
Android folder will be built into the app package, but the files in the other Platforms folders won't be. This
approach uses a feature called multi-targeting to target multiple platforms from a single project.
Multi-targeting can be combined with partial classes and partial methods to invoke platform functionality from
cross-platform code. The process for invoking platform code from cross-platform code is to:
1. Define the cross-platform API as a partial class that defines partial method signatures for any operations you
want to invoke on each platform. For more information, see Define the cross-platform API.
2. Implement the cross-platform API per platform, by defining the same partial class and the same partial
method signatures, while also providing the method implementations. For more information, see Implement
the API per platform.
3. Invoke the cross-platform API by creating an instance of the partial class and invoking its methods as
required. For more information, see Invoke the cross-platform API.
The partial class is named DeviceOrientationService , which includes a partial method named GetOrientation .
P L AT F O RM F O L DER
IMPORTANT
Platform implementations must be in the same namespace and same class that the cross-platform API was defined in.
The following screenshot shows the DeviceOrientationService classes in the Android and iOS folders:
Alternatively, multi-targeting can be performed based on your own filename and folder criteria, rather than
using the Platforms folders. For more information, see Configure multi-targeting.
Android
The following example shows the implementation of the GetOrientation method on Android:
using Android.Content;
using Android.Runtime;
using Android.Views;
namespace InvokePlatformCodeDemos.Services
{
public partial class DeviceOrientationService
{
public partial DeviceOrientation GetOrientation()
{
IWindowManager windowManager =
Android.App.Application.Context.GetSystemService(Context.WindowService).JavaCast<IWindowManager>();
SurfaceOrientation orientation = windowManager.DefaultDisplay.Rotation;
bool isLandscape = orientation == SurfaceOrientation.Rotation90 || orientation ==
SurfaceOrientation.Rotation270;
return isLandscape ? DeviceOrientation.Landscape : DeviceOrientation.Portrait;
}
}
}
iOS
The following example shows the implementation of the GetOrientation method on iOS:
using UIKit;
namespace InvokePlatformCodeDemos.Services
{
public partial class DeviceOrientationService
{
public partial DeviceOrientation GetOrientation()
{
UIInterfaceOrientation orientation = UIApplication.SharedApplication.StatusBarOrientation;
bool isPortrait = orientation == UIInterfaceOrientation.Portrait || orientation ==
UIInterfaceOrientation.PortraitUpsideDown;
return isPortrait ? DeviceOrientation.Portrait : DeviceOrientation.Landscape;
}
}
}
using InvokePlatformCodeDemos.Services;
...
At build time the build system will use multi-targeting to combine the cross-platform partial class with the
partial class for the target platform, and build it into the app package.
Configure multi-targeting
.NET MAUI apps can also be multi-targeted based on your own filename and folder criteria. This enables you to
structure your .NET MAUI app project so that you don't have to place your platform code into child folders of the
Platforms folder.
For example, a standard multi-targeting pattern is to include the platform as an extension in the filename for the
platform code. The build system can be configured to combine cross-platform partial classes with platform
partial classes based on this pattern:
Another standard multi-targeting pattern is to include the platform as a folder name. The build system can be
configured to combine cross-platform partial classes with platform partial classes based on this pattern: