From d9cf7a92311921d21cbe6e4d938836aaed16012e Mon Sep 17 00:00:00 2001 From: Ivan Povazan <55002338+ivanpovazan@users.noreply.github.com> Date: Wed, 13 Nov 2024 16:17:55 +0100 Subject: [PATCH 01/11] Correct documentation on observability support with Native AOT on iOS and Mac Catalyst (#2619) Co-authored-by: David Britch --- docs/deployment/nativeaot.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/docs/deployment/nativeaot.md b/docs/deployment/nativeaot.md index cc05f58d7..1aee13bda 100644 --- a/docs/deployment/nativeaot.md +++ b/docs/deployment/nativeaot.md @@ -147,7 +147,7 @@ The following table shows the diagnostics support with Native AOT on iOS and Mac | Feature | Fully supported | Partially supported | Not supported | | - | - | - | - | -| [Observability and telemetry](#observability-and-telemetry) | | | Not supported | +| [Observability and telemetry](#observability-and-telemetry) | | Partially supported | | | [Development-time diagnostics](#development-time-diagnostics) | Fully supported | | | | [Native debugging](#native-debugging) | | Partially supported | | | [CPU Profiling](#cpu-profiling) | | Partially supported | | @@ -157,7 +157,7 @@ The following sections provide additional information about this diagnostics sup ### Observability and telemetry -Tracing of .NET MAUI applications on mobile platforms is enabled through [dotnet-dsrouter](/dotnet/core/diagnostics/dotnet-dsrouter) which connects diagnostic tooling with .NET applications running on iOS and Mac Catalyst, over TCP/IP. However, Native AOT is currently not compatible with this scenario as it doesn't support EventPipe/DiagnosticServer components built with the TCP/IP stack. +Tracing of .NET MAUI applications on mobile platforms is enabled through [dotnet-dsrouter](/dotnet/core/diagnostics/dotnet-dsrouter) which connects diagnostic tooling with .NET applications running on iOS and Mac Catalyst, over TCP/IP. However, Native AOT is currently not compatible with this scenario as it doesn't support EventPipe/DiagnosticServer components built with the TCP/IP stack. Observability is still achievable explicitly in the code. ### Development-time diagnostics From dc648aaabc15538767f91243fb64ffd19998b79a Mon Sep 17 00:00:00 2001 From: David Britch Date: Wed, 13 Nov 2024 15:56:29 +0000 Subject: [PATCH 02/11] Fire and forget enabled by default (#2628) * Fire and forget enabled by default. * Edits. --- docs/user-interface/controls/blazorwebview.md | 18 +++++++++--------- docs/whats-new/dotnet-9.md | 11 ++++++++--- 2 files changed, 17 insertions(+), 12 deletions(-) diff --git a/docs/user-interface/controls/blazorwebview.md b/docs/user-interface/controls/blazorwebview.md index 534460bc2..0edd2ed7f 100644 --- a/docs/user-interface/controls/blazorwebview.md +++ b/docs/user-interface/controls/blazorwebview.md @@ -1,7 +1,7 @@ --- title: "Host a Blazor web app in a .NET MAUI app using BlazorWebView" description: "The .NET MAUI BlazorWebView control enables you to host a Blazor web app in your .NET MAUI app, and integrate the app with device features." -ms.date: 10/01/2024 +ms.date: 11/13/2024 --- # Host a Blazor web app in a .NET MAUI app using BlazorWebView @@ -218,20 +218,20 @@ To play inline video in a Blazor hybrid app on iOS, in a performs async-over-sync disposal, which means that it blocks the thread until the async disposal is complete. However, this can cause deadlocks if the disposal needs to run code on the same thread (because the thread is blocked while waiting). +By default, fires and forgets the async disposal of the underlying `WebViewManager`. This reduces the potential for disposal deadlocks to occur on Android. -If you encounter hangs on Android with you should enable an switch in the `CreateMauiApp` method in *MauiProgram.cs*: +> [!WARNING] +> This fire-and-forget default behavior means that disposal can return before all objects are disposed, which can cause behavioral changes in your app. The items that are disposed are partially Blazor's own internal types, but also app-defined types such as scoped services used within the portion of your app. + +To opt out of this behavior, you should configure your app to block on dispose via an switch in the `CreateMauiApp` method in your `MauiProgram` class: ```csharp -AppContext.SetSwitch("BlazorWebView.AndroidFireAndForgetAsync", true); +AppContext.SetSwitch("BlazorWebView.AndroidFireAndForgetAsync", false); ``` -This switch enables to fire and forget the async disposal that occurs, and as a result fixes the majority of the disposal deadlocks that occur on Android. - -> [!WARNING] -> Enabling this switch means that disposal can return before all objects are disposed, which can cause behavioral changes in your app. The items that are disposed are partially Blazor's own internal types, but also app-defined types such as scoped services used within the portion of your app. +If your app is configured to block on dispose via this switch, performs async-over-sync disposal, which means that it blocks the thread until the async disposal is complete. However, this can cause deadlocks if the disposal needs to run code on the same thread (because the thread is blocked while waiting). ## Host content using the legacy behavior on iOS and Mac Catalyst diff --git a/docs/whats-new/dotnet-9.md b/docs/whats-new/dotnet-9.md index e8c4b9474..6a87e5574 100644 --- a/docs/whats-new/dotnet-9.md +++ b/docs/whats-new/dotnet-9.md @@ -153,13 +153,18 @@ To opt into using the `0.0.0.1` address, add the following code to the `CreateMa AppContext.SetSwitch("BlazorWebView.AppHostAddressAlways0000", true); ``` -If you encounter hangs on Android with you should enable an switch in the `CreateMauiApp` method in your `MauiProgram` class: +By default, now fires and forgets the async disposal of the underlying `WebViewManager`. This reduces the potential for disposal deadlocks to occur on Android. + +> [!WARNING] +> This fire-and-forget default behavior means that disposal can return before all objects are disposed, which can cause behavioral changes in your app. The items that are disposed are partially Blazor's own internal types, but also app-defined types such as scoped services used within the portion of your app. + +To opt out of this behavior, you should configure your app to block on dispose via an switch in the `CreateMauiApp` method in your `MauiProgram` class: ```csharp -AppContext.SetSwitch("BlazorWebView.AndroidFireAndForgetAsync", true); +AppContext.SetSwitch("BlazorWebView.AndroidFireAndForgetAsync", false); ``` -This switch enables to fire and forget the async disposal that occurs, and as a result fixes the majority of the disposal deadlocks that occur on Android. For more information, see [Fix disposal deadlocks on Android](~/user-interface/controls/blazorwebview.md#fix-disposal-deadlocks-on-android). +If your app is configured to block on dispose via this switch, performs async-over-sync disposal, which means that it blocks the thread until the async disposal is complete. However, this can cause deadlocks if the disposal needs to run code on the same thread (because the thread is blocked while waiting). ### Buttons on iOS From 2e12ba442ed11fe2dccfd3c0008726650aaeb37d Mon Sep 17 00:00:00 2001 From: David Britch Date: Thu, 14 Nov 2024 09:10:42 +0000 Subject: [PATCH 03/11] Add final release notes. (#2621) --- docs/whats-new/dotnet-9.md | 3 +++ 1 file changed, 3 insertions(+) diff --git a/docs/whats-new/dotnet-9.md b/docs/whats-new/dotnet-9.md index 6a87e5574..4dcd438f1 100644 --- a/docs/whats-new/dotnet-9.md +++ b/docs/whats-new/dotnet-9.md @@ -8,6 +8,7 @@ ms.date: 11/13/2024 The focus of .NET Multi-platform App UI (.NET MAUI) in .NET 9 is to improve product quality. This includes expanding test coverage, end to end scenario testing, and bug fixing. For more information about the product quality improvements in .NET MAUI 9, see the following release notes: +- [.NET MAUI 9](https://github.com/dotnet/maui/releases/tag/9.0.0) - [.NET MAUI 9 RC2](https://github.com/dotnet/maui/releases/tag/9.0.0-rc.2.24503.2) - [.NET MAUI 9 RC1](https://github.com/dotnet/maui/releases/tag/9.0.0-rc.1.24453.9) - [.NET MAUI 9 Preview 7](https://github.com/dotnet/maui/releases/tag/9.0.0-preview.7.24407.4) @@ -584,6 +585,7 @@ You can opt into Native AOT deployment on iOS and Mac Catalyst. Native AOT deplo .NET for Android in .NET 9, which adds support for API 35, includes work to reduce build times, and to improve the trimability of apps to reduce size and improve performance. For more information about .NET for Android in .NET 9, see the following release notes: +- [.NET for Android 9](https://github.com/dotnet/android/releases/tag/35.0.7) - [.NET for Android 9 RC2](https://github.com/dotnet/android/releases/tag/35.0.0-rc.2.152) - [.NET for Android 9 RC1](https://github.com/dotnet/android/releases/tag/35.0.0-rc.1.80) - [.NET for Android 9 Preview 7](https://github.com/xamarin/xamarin-android/releases/tag/35.0.0-preview.7.41) @@ -704,6 +706,7 @@ For more information, see [Trimming granularity](/dotnet/core/deploying/trimming For more information about .NET 9 on iOS, tvOS, Mac Catalyst, and macOS, see the following release notes: +- [.NET 9](https://github.com/xamarin/xamarin-macios/releases/tag/dotnet-9.0.1xx-xcode16.0-9617) - [.NET 9.0.1xx RC2](https://github.com/xamarin/xamarin-macios/releases/tag/dotnet-9.0.1xx-rc2-9600) - [.NET 9.0.1xx RC1](https://github.com/xamarin/xamarin-macios/releases/tag/dotnet-9.0.1xx-rc1-9270) - [.NET 9.0.1xx Preview 7](https://github.com/xamarin/xamarin-macios/releases/tag/dotnet-9.0.1xx-preview7-9231) From 168bd97a576dfc7958827f24085c59939ed973b0 Mon Sep 17 00:00:00 2001 From: David Britch Date: Thu, 14 Nov 2024 09:54:56 +0000 Subject: [PATCH 04/11] Add developer tools. (#2630) --- docs/data-cloud/database-sqlite.md | 2 +- docs/user-interface/controls/blazorwebview.md | 4 +- docs/user-interface/controls/hybridwebview.md | 46 +++++++++++++++---- 3 files changed, 40 insertions(+), 12 deletions(-) diff --git a/docs/data-cloud/database-sqlite.md b/docs/data-cloud/database-sqlite.md index 6c2a00bad..1653e0ec9 100644 --- a/docs/data-cloud/database-sqlite.md +++ b/docs/data-cloud/database-sqlite.md @@ -160,7 +160,7 @@ public class TodoItemDatabase ## Access data -The `TodoItemDatabase` class can be registered as a singleton that can be used throughout the app if you are using dependency injection. For example, you can register your pages and the database access class as services on the `IServiceCollection` object, in **MauiProgram.cs**, with the `AddSingleton` and `AddTransient` methods: +The `TodoItemDatabase` class can be registered as a singleton that can be used throughout the app if you are using dependency injection. For example, you can register your pages and the database access class as services on the object, in **MauiProgram.cs**, with the `AddSingleton` and `AddTransient` methods: ```csharp builder.Services.AddSingleton(); diff --git a/docs/user-interface/controls/blazorwebview.md b/docs/user-interface/controls/blazorwebview.md index 0edd2ed7f..407cd118c 100644 --- a/docs/user-interface/controls/blazorwebview.md +++ b/docs/user-interface/controls/blazorwebview.md @@ -96,7 +96,7 @@ The process to add a ``` -1. Modify the `CreateMauiApp` method of your `MauiProgram` class to register the control for use in your app. To do this, on the `IServiceCollection` object, call the `AddMauiBlazorWebView` method to add component web view services to the services collection: +1. Modify the `CreateMauiApp` method of your `MauiProgram` class to register the control for use in your app. To do this, on the object, call the `AddMauiBlazorWebView` method to add component web view services to the services collection: ```csharp public static class MauiProgram @@ -123,6 +123,8 @@ The process to add a has a method that can call a specified `Action` asynchronously and pass in the scoped services available in Razor components. This enables code from the native UI to access scoped services such as : diff --git a/docs/user-interface/controls/hybridwebview.md b/docs/user-interface/controls/hybridwebview.md index e92bb9547..a1ff389f0 100644 --- a/docs/user-interface/controls/hybridwebview.md +++ b/docs/user-interface/controls/hybridwebview.md @@ -2,7 +2,7 @@ title: HybridWebView description: Learn how to use a HybridWebView to host HTML/JS/CSS content in a WebView, and communicate between that content and .NET. ms.topic: concept-article -ms.date: 10/28/2024 +ms.date: 11/14/2024 monikerRange: ">=net-maui-9.0" #customer intent: As a developer, I want to host HTML/JS/CSS content in a web view so that I can publish the web app as a mobile app. @@ -10,7 +10,7 @@ monikerRange: ">=net-maui-9.0" # HybridWebView - +[![Browse sample.](~/media/code-sample.png) Browse the sample](/samples/dotnet/maui-samples/userinterface-hybridwebview) The .NET Multi-platform App UI (.NET MAUI) enables hosting arbitrary HTML/JS/CSS content in a web view, and enables communication between the code in the web view (JavaScript) and the code that hosts the web view (C#/.NET). For example, if you have an existing React JS app, you could host it in a cross-platform .NET MAUI native app, and build the back-end of the app using C# and .NET. @@ -21,7 +21,7 @@ The .NET Multi-platform App UI (.NET MAUI) defines a event that's raised when a raw message is received. The object that accompanies the event defines a property that contains the message. -Your app's C# code can invoke synchronous and asynchronous JavaScript methods within the with the `InvokeJavaScriptAsync` and `EvaluateJavaScriptAsync` methods. For more information, see [Invoke JavaScript from C#](#invoke-javascript-from-c). +Your app's C# code can invoke synchronous and asynchronous JavaScript methods within the with the and methods. For more information, see [Invoke JavaScript from C#](#invoke-javascript-from-c). To create a .NET MAUI app with you need: @@ -175,6 +175,32 @@ To create a .NET MAUI app with a : ``` +1. Modify the `CreateMauiApp` method of your `MauiProgram` class to enable developer tools on the underlying WebView controls when your app is running in debug configuration. To do this, call the method on the object: + + ```csharp + public static class MauiProgram + { + public static MauiApp CreateMauiApp() + { + var builder = MauiApp.CreateBuilder(); + builder + .UseMauiApp() + .ConfigureFonts(fonts => + { + fonts.AddFont("OpenSans-Regular.ttf", "OpenSansRegular"); + fonts.AddFont("OpenSans-Semibold.ttf", "OpenSansSemibold"); + }); + + #if DEBUG + builder.Services.AddHybridWebViewDeveloperTools(); + #endif + // Register any app services on the IServiceCollection object + + return builder.Build(); + } + } + ``` + 1. Use the APIs to send messages between the JavaScript and C# code: ```csharp @@ -193,14 +219,14 @@ To create a .NET MAUI app with a : ## Invoke JavaScript from C\# -Your app's C# code can synchronously and asynchronously invoke JavaScript methods within the , with optional parameters and an optional return value. This can be achieved with the `InvokeJavaScriptAsync` and `EvaluateJavaScriptAsync` methods: +Your app's C# code can synchronously and asynchronously invoke JavaScript methods within the , with optional parameters and an optional return value. This can be achieved with the and methods: -- The `EvaluateJavaScriptAsync` method runs the JavaScript code provided via a parameter and returns the result as a string. -- The `InvokeJavaScriptAsync` method invokes a specified JavaScript method, optionally passing in parameter values, and specifies a generic argument that indicates the type of the return value. It returns an object of the generic argument type that contains the return value of the called JavaScript method. Internally, parameters and return values are JSON encoded. +- The method runs the JavaScript code provided via a parameter and returns the result as a string. +- The method invokes a specified JavaScript method, optionally passing in parameter values, and specifies a generic argument that indicates the type of the return value. It returns an object of the generic argument type that contains the return value of the called JavaScript method. Internally, parameters and return values are JSON encoded. ### Invoke synchronous JavaScript -Synchronous JavaScript methods can be invoked with the `EvaluateJavaScriptAsync` and `InvokeJavaScriptAsync` methods. In the following example the `InvokeJavaScriptAsync` method is used to demonstrate invoking JavaScript that's embedded in an app's web content. For example, a simple Javascript method to add two numbers could be defined in your web content: +Synchronous JavaScript methods can be invoked with the and methods. In the following example the method is used to demonstrate invoking JavaScript that's embedded in an app's web content. For example, a simple Javascript method to add two numbers could be defined in your web content: ```javascript function AddNumbers(a, b) { @@ -208,7 +234,7 @@ function AddNumbers(a, b) { } ``` -The `AddNumbers` JavaScript method can be invoked from C# with the `InvokeJavaScriptAsync` method: +The `AddNumbers` JavaScript method can be invoked from C# with the method: ```csharp double x = 123d; @@ -238,7 +264,7 @@ internal partial class HybridSampleJsContext : JsonSerializerContext ### Invoke asynchronous JavaScript -Asynchronous JavaScript methods can be invoked with the `EvaluateJavaScriptAsync` and `InvokeJavaScriptAsync` methods. In the following example the `InvokeJavaScriptAsync` method is used to demonstrate invoking JavaScript that's embedded in an app's web content. For example, a Javascript method that asynchronously retrieves data could be defined in your web content: +Asynchronous JavaScript methods can be invoked with the and methods. In the following example the method is used to demonstrate invoking JavaScript that's embedded in an app's web content. For example, a Javascript method that asynchronously retrieves data could be defined in your web content: ```javascript async function EvaluateMeWithParamsAndAsyncReturn(s1, s2) { @@ -253,7 +279,7 @@ async function EvaluateMeWithParamsAndAsyncReturn(s1, s2) { } ``` -The `EvaluateMeWithParamsAndAsyncReturn` JavaScript method can be invoked from C# with the `InvokeJavaScriptAsync` method: +The `EvaluateMeWithParamsAndAsyncReturn` JavaScript method can be invoked from C# with the method: ```csharp Dictionary asyncResult = await hybridWebView.InvokeJavaScriptAsync>( From 0346de81a4867f3fd3f0fbb156ee9d8d3bce703d Mon Sep 17 00:00:00 2001 From: David Britch Date: Thu, 14 Nov 2024 15:58:37 +0000 Subject: [PATCH 05/11] Invoke C# from JavaScript (#2632) * Invoke C# from JavaScript. * Edits. * Edit. * Edit. --- docs/deployment/includes/feature-switches.md | 16 +- .../includes/trimming-incompatibilities.md | 3 +- docs/user-interface/controls/hybridwebview.md | 221 ++++++++++++++++-- 3 files changed, 219 insertions(+), 21 deletions(-) diff --git a/docs/deployment/includes/feature-switches.md b/docs/deployment/includes/feature-switches.md index 707dc398d..ea0195354 100644 --- a/docs/deployment/includes/feature-switches.md +++ b/docs/deployment/includes/feature-switches.md @@ -1,18 +1,19 @@ --- ms.topic: include -ms.date: 10/28/2024 +ms.date: 11/14/2024 --- -.NET MAUI has trimmer directives, known as feature switches, that make it possible to preserve the code for features that aren't trim safe. These trimmer directives can be used when the `$(TrimMode)` build property is set to `full`, as well as for NativeAOT: +.NET MAUI has trimmer directives, known as feature switches, that make it possible to preserve the code for features that aren't trim safe. These trimmer directives can be used when the `$(TrimMode)` build property is set to `full`, as well as for Native AOT: | MSBuild property | Description | | ---------------- | ----------- | -| `MauiEnableVisualAssemblyScanning` | When set to `true`, .NET MAUI will scan assemblies for types implementing `IVisual` and for `[assembly:Visual(...)]` attributes, and will register these types. By default, this build property is set to `false`. | -| `MauiShellSearchResultsRendererDisplayMemberNameSupported` | When set to `false`, the value of `SearchHandler.DisplayMemberName` will be ignored. Instead, you should provide an to define the appearance of results. By default, this build property is set to `true`.| -| `MauiQueryPropertyAttributeSupport` | When set to `false`, `[QueryProperty(...)]` attributes won't be used to set property values when navigating. Instead, you should implement the interface to accept query parameters. By default, this build property is set to `true`. | -| `MauiImplicitCastOperatorsUsageViaReflectionSupport` | When set to `false`, .NET MAUI won't look for implicit conversion operators when converting values from one type to another. This can affect bindings between properties with different types, and setting a property value of a bindable object with a value of a different type. Instead, you should define a for your type and attach it to the type using the attribute. By default, this build property is set to `true`.| +| `MauiEnableVisualAssemblyScanning` | When set to `true`, .NET MAUI will scan assemblies for types implementing `IVisual` and for `[assembly:Visual(...)]` attributes, and will register these types. By default, this build property is set to `false` when full trimming is enabled. | +| `MauiShellSearchResultsRendererDisplayMemberNameSupported` | When set to `false`, the value of `SearchHandler.DisplayMemberName` will be ignored. Instead, you should provide an to define the appearance of results. By default, this build property is set to `false` when full trimming or Native AOT is enabled.| +| `MauiQueryPropertyAttributeSupport` | When set to `false`, `[QueryProperty(...)]` attributes won't be used to set property values when navigating. Instead, you should implement the interface to accept query parameters. By default, this build property is set to `false` when full trimming or Native AOT is enabled. | +| `MauiImplicitCastOperatorsUsageViaReflectionSupport` | When set to `false`, .NET MAUI won't look for implicit conversion operators when converting values from one type to another. This can affect bindings between properties with different types, and setting a property value of a bindable object with a value of a different type. Instead, you should define a for your type and attach it to the type using the attribute. By default, this build property is set to `false` when full trimming or Native AOT is enabled. | | `_MauiBindingInterceptorsSupport` | When set to `false`, .NET MAUI won't intercept any calls to the `SetBinding` methods and won't try to compile them. By default, this build property is set to `true`. | -| `MauiEnableXamlCBindingWithSourceCompilation` | When set to `true`, .NET MAUI will compile all bindings, including those where the `Source` property is used. If you enable this feature ensure that all bindings have the correct `x:DataType` so that they compile, or clear the data type with `x:Data={x:Null}}` if the binding shouldn't be compiled. By default, this build property is only set to `true` when full trimming or Native AOT deployment is enabled. | +| `MauiEnableXamlCBindingWithSourceCompilation` | When set to `true`, .NET MAUI will compile all bindings, including those where the `Source` property is used. If you enable this feature ensure that all bindings have the correct `x:DataType` so that they compile, or clear the data type with `x:Data={x:Null}}` if the binding shouldn't be compiled. By default, this build property is set to `true` when full trimming or Native AOT is enabled. | +| `MauiHybridWebViewSupported` | When set to `false`, the control won't be available. By default, this build property is set to `false` when full trimming or Native AOT is enabled. | These MSBuild properties also have equivalent switches: @@ -22,5 +23,6 @@ These MSBuild properties also have equivalent switches: - The `MauiImplicitCastOperatorsUsageViaReflectionSupport` MSBuild property has an equivalent switch named `Microsoft.Maui.RuntimeFeature.IsImplicitCastOperatorsUsageViaReflectionSupported`. - The `_MauiBindingInterceptorsSupport` MSBuild property has an equivalent switch named `Microsoft.Maui.RuntimeFeature.AreBindingInterceptorsSupported`. - The `MauiEnableXamlCBindingWithSourceCompilation` MSBuild property has an equivalent switch named `Microsoft.Maui.RuntimeFeature.MauiEnableXamlCBindingWithSourceCompilationEnabled`. +- The `MauiHybridWebViewSupported` MSBuild property has an equivalent switch named `Microsoft.Maui.RuntimeFeature.IsHybridWebViewSupported`. The easiest way to consume a feature switch is by putting the corresponding MSBuild property into your app's project file (*.csproj), which causes the related code to be trimmed from the .NET MAUI assemblies. diff --git a/docs/deployment/includes/trimming-incompatibilities.md b/docs/deployment/includes/trimming-incompatibilities.md index 16e4dcf94..0054005a4 100644 --- a/docs/deployment/includes/trimming-incompatibilities.md +++ b/docs/deployment/includes/trimming-incompatibilities.md @@ -1,6 +1,6 @@ --- ms.topic: include -ms.date: 10/23/2024 +ms.date: 11/14/2024 monikerRange: ">=net-maui-9.0" --- @@ -11,3 +11,4 @@ The following .NET MAUI features are incompatible with full trimming and will be - Loading XAML at runtime with the extension method. This XAML can be made trim safe by annotating all types that could be loaded at runtime with the [`DynamicallyAccessedMembers`](xref:System.Diagnostics.CodeAnalysis.DynamicallyAccessedMembersAttribute) attribute or the [`DynamicDependency`](xref:System.Diagnostics.CodeAnalysis.DynamicDependencyAttribute) attribute. However, this is very error prone and isn't recommended. - Receiving navigation data using the . Instead, you should implement the interface on types that need to accept query parameters. For more information, see [Process navigation data using a single method](~/fundamentals/shell/navigation.md#process-navigation-data-using-a-single-method). - The `SearchHandler.DisplayMemberName` property. Instead, you should provide an to define the appearance of results. For more information, see [Define search results item appearance](~/fundamentals/shell/search.md#define-search-results-item-appearance). +- The control, due to its use of dynamic `System.Text.Json` serialization features. diff --git a/docs/user-interface/controls/hybridwebview.md b/docs/user-interface/controls/hybridwebview.md index a1ff389f0..1b2379e8a 100644 --- a/docs/user-interface/controls/hybridwebview.md +++ b/docs/user-interface/controls/hybridwebview.md @@ -21,7 +21,7 @@ The .NET Multi-platform App UI (.NET MAUI) defines a event that's raised when a raw message is received. The object that accompanies the event defines a property that contains the message. -Your app's C# code can invoke synchronous and asynchronous JavaScript methods within the with the and methods. For more information, see [Invoke JavaScript from C#](#invoke-javascript-from-c). +Your app's C# code can invoke synchronous and asynchronous JavaScript methods within the with the and methods. Your app's JavaScript code can also synchronously invoke C# methods. For more information, see [Invoke JavaScript from C#](#invoke-javascript-from-c) and [Invoke C# from JavaScript](#invoke-c-from-javascript). To create a .NET MAUI app with you need: @@ -31,6 +31,9 @@ To create a .NET MAUI app with you The entire app, including the web content, is packaged and runs locally on a device, and can be published to applicable app stores. The web content is hosted within a native web view control and runs within the context of the app. Any part of the app can access external web services, but isn't required to. +> [!IMPORTANT] +> By default, the control won't be available when full trimming or Native AOT is enabled. To change this behavior, see [Trimming feature switches](~/deployment/trimming.md#trimming-feature-switches). + ## Create a .NET MAUI HybridWebView app To create a .NET MAUI app with a : @@ -52,23 +55,89 @@ To create a .NET MAUI app with a : + -

HybridWebView app!

- + Hybrid sample! +
+
+
- Messages from C#: + + + + +
+
+ Log: +
+
+ Consider checking out this PDF: sample.pdf
@@ -78,7 +147,7 @@ To create a .NET MAUI app with a : ```js window.HybridWebView = { - "Init": function () { + "Init": function Init() { function DispatchHybridWebViewMessage(message) { const event = new CustomEvent("HybridWebViewMessageReceived", { detail: { message: message } }); window.dispatchEvent(event); @@ -106,11 +175,53 @@ To create a .NET MAUI app with a : } }, - "SendRawMessage": function (message) { - window.HybridWebView.__SendMessageInternal('RawMessage', message); + "SendRawMessage": function SendRawMessage(message) { + window.HybridWebView.__SendMessageInternal('__RawMessage', message); + }, + + "InvokeDotNet": async function InvokeDotNetAsync(methodName, paramValues) { + const body = { + MethodName: methodName + }; + + if (typeof paramValues !== 'undefined') { + if (!Array.isArray(paramValues)) { + paramValues = [paramValues]; + } + + for (var i = 0; i < paramValues.length; i++) { + paramValues[i] = JSON.stringify(paramValues[i]); + } + + if (paramValues.length > 0) { + body.ParamValues = paramValues; + } + } + + const message = JSON.stringify(body); + + var requestUrl = `${window.location.origin}/__hwvInvokeDotNet?data=${encodeURIComponent(message)}`; + + const rawResponse = await fetch(requestUrl, { + method: 'GET', + headers: { + 'Accept': 'application/json' + } + }); + const response = await rawResponse.json(); + + if (response) { + if (response.IsJson) { + return JSON.parse(response.Result); + } + + return response.Result; + } + + return null; }, - "__SendMessageInternal": function (type, message) { + "__SendMessageInternal": function __SendMessageInternal(type, message) { const messageToSend = type + '|' + message; @@ -128,7 +239,7 @@ To create a .NET MAUI app with a : } }, - "InvokeMethod": function (taskId, methodName, args) { + "__InvokeJavaScript": function __InvokeJavaScript(taskId, methodName, args) { if (methodName[Symbol.toStringTag] === 'AsyncFunction') { // For async methods, we need to call the method and then trigger the callback when it's done const asyncPromise = methodName(...args); @@ -144,13 +255,13 @@ To create a .NET MAUI app with a : } }, - "__TriggerAsyncCallback": function (taskId, result) { + "__TriggerAsyncCallback": function __TriggerAsyncCallback(taskId, result) { // Make sure the result is a string if (result && typeof (result) !== 'string') { result = JSON.stringify(result); } - window.HybridWebView.__SendMessageInternal('InvokeMethodCompleted', taskId + '|' + result); + window.HybridWebView.__SendMessageInternal('__InvokeJavaScriptCompleted', taskId + '|' + result); } } @@ -178,6 +289,8 @@ To create a .NET MAUI app with a : 1. Modify the `CreateMauiApp` method of your `MauiProgram` class to enable developer tools on the underlying WebView controls when your app is running in debug configuration. To do this, call the method on the object: ```csharp + using Microsoft.Extensions.Logging; + public static class MauiProgram { public static MauiApp CreateMauiApp() @@ -193,6 +306,7 @@ To create a .NET MAUI app with a : #if DEBUG builder.Services.AddHybridWebViewDeveloperTools(); + builder.Logging.AddDebug(); #endif // Register any app services on the IServiceCollection object @@ -306,3 +420,84 @@ internal partial class HybridSampleJSContext : JsonSerializerContext > [!IMPORTANT] > The `HybridSampleJsContext` class must be `partial` so that code generation can provide the implementation when the project is compiled. If the type is nested into another type, then that type must also be `partial`. + +## Invoke C\# from JavaScript + +Your app's JavaScript code within the can synchronously invoke C# methods, with optional parameters and an optional return value. This can be achieved by: + +- Defining public C# methods that will be invoked from JavaScript. +- Calling the method to set the object that will be the target of JavaScript calls from the . +- Calling the C# methods from JavaScript. + +> [!IMPORTANT] +> Asynchronously invoking C# methods from JavaScript isn't currently supported. + +The following example defines four public methods for invoking from JavaScript: + +```csharp +public partial class MainPage : ContentPage +{ + ... + + public void DoSyncWork() + { + Debug.WriteLine("DoSyncWork"); + } + + public void DoSyncWorkParams(int i, string s) + { + Debug.WriteLine($"DoSyncWorkParams: {i}, {s}"); + } + + public string DoSyncWorkReturn() + { + Debug.WriteLine("DoSyncWorkReturn"); + return "Hello from C#!"; + } + + public SyncReturn DoSyncWorkParamsReturn(int i, string s) + { + Debug.WriteLine($"DoSyncWorkParamReturn: {i}, {s}"); + return new SyncReturn + { + Message = "Hello from C#!" + s, + Value = i + }; + } + + public class SyncReturn + { + public string? Message { get; set; } + public int Value { get; set; } + } +} +``` + +You must then call the method to set the object that will be the target of JavaScript calls from the : + +```csharp +public partial class MainPage : ContentPage +{ + public MainPage() + { + InitializeComponent(); + hybridWebView.SetInvokeJavaScriptTarget(this); + } + + ... +} +``` + +The public methods on the object set via the method can then be invoked from JavaScript with the `window.HybridWebView.InvokeDotNet` function: + +```js +await window.HybridWebView.InvokeDotNet('DoSyncWork'); +await window.HybridWebView.InvokeDotNet('DoSyncWorkParams', [123, 'hello']); +const retValue = await window.HybridWebView.InvokeDotNet('DoSyncWorkReturn'); +const retValue = await window.HybridWebView.InvokeDotNet('DoSyncWorkParamsReturn', [123, 'hello']); +``` + +The `window.HybridWebView.InvokeDotNet` JavaScript function invokes a specified C# method, with optional parameters and an optional return value. + +> [!NOTE] +> Invoking the `window.HybridWebView.InvokeDotNet` JavaScript function requires your app to include the *HybridWebView.js* JavaScript library listed earlier in this article. From 4e790d0541b8ad2bffa29f003625450a48c70d75 Mon Sep 17 00:00:00 2001 From: David Britch Date: Thu, 14 Nov 2024 16:43:53 +0000 Subject: [PATCH 06/11] Fix typo (#2633) * Fix typo. * Edit. --- docs/ios/includes/distribution-certificate.md | 2 +- docs/ios/wireless-deployment.md | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/docs/ios/includes/distribution-certificate.md b/docs/ios/includes/distribution-certificate.md index 33100e721..c37b44db6 100644 --- a/docs/ios/includes/distribution-certificate.md +++ b/docs/ios/includes/distribution-certificate.md @@ -7,7 +7,7 @@ ms.topic: include A distribution certificate is used to confirm your identity. Before creating a distribution certificate, you should ensure that you've added your Apple Developer Account to Visual Studio. For more information, see [Apple account management](~/ios/apple-account-management.md). -You only need to create a distribution certificate if you don't already one. The distribution certificate must be created using the Apple ID for your Apple Developer Account. +You only need to create a distribution certificate if you don't already have one. The distribution certificate must be created using the Apple ID for your Apple Developer Account. To create a distribution certificate in Visual Studio: diff --git a/docs/ios/wireless-deployment.md b/docs/ios/wireless-deployment.md index 65629989f..c1690210b 100644 --- a/docs/ios/wireless-deployment.md +++ b/docs/ios/wireless-deployment.md @@ -6,7 +6,7 @@ ms.date: 08/27/2024 # Wireless deployment for .NET MAUI iOS apps -Rather than having to use a USB cable to connect an iOS device to your Mac to deploy and debug a .NET Multi-platform App UI (.NET MAUI) app, Visual Studio can deploy .NET MAUI iOS apps to devices wirelessly, and debug them wirelessly. To do this you must pair your iOS device with Xcode on your Mac. Once paired, the device can be selected from the device target list in Visual Studio. +Rather than having to use a USB cable to connect an iOS device to your Mac to deploy and debug a .NET Multi-platform App UI (.NET MAUI) app, Visual Studio can deploy .NET MAUI iOS apps to devices wirelessly, and debug them wirelessly. To do this, you must pair your iOS device with Xcode on your Mac. Once paired, the device can be selected from the device target list in Visual Studio. > [!IMPORTANT] > .NET MAUI iOS apps must have been provisioned before they can be deployed to a device for testing and debugging. For more information, see [Device provisioning for iOS](~/ios/device-provisioning/index.md). From 9e8d41f5ec4923f9f401629520bc71d63a53eb95 Mon Sep 17 00:00:00 2001 From: David Britch Date: Thu, 14 Nov 2024 17:05:44 +0000 Subject: [PATCH 07/11] Use the sample example in compiled bindings (#2634) * Use the sample example. * Edit. * Edit. --- .../data-binding/compiled-bindings.md | 54 +++++------------- .../compiled-bindings/compiledcolorlist.png | Bin 29095 -> 0 bytes 2 files changed, 15 insertions(+), 39 deletions(-) delete mode 100755 docs/fundamentals/data-binding/media/compiled-bindings/compiledcolorlist.png diff --git a/docs/fundamentals/data-binding/compiled-bindings.md b/docs/fundamentals/data-binding/compiled-bindings.md index afccfa913..d2c60455d 100644 --- a/docs/fundamentals/data-binding/compiled-bindings.md +++ b/docs/fundamentals/data-binding/compiled-bindings.md @@ -1,7 +1,7 @@ --- title: "Compiled bindings" description: "Compiled bindings can be used to improve data binding performance in .NET MAUI applications." -ms.date: 10/29/2024 +ms.date: 11/14/2024 --- # Compiled bindings @@ -120,45 +120,21 @@ Bindings in a are interpreted in the The following example demonstrates correctly setting the `x:DataType` on a : ```xaml - - - ... - - - - - - - - - - - - - - + + + + + + + + + ``` -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 class, and to store them with their names in a collection that is accessible from the static `All` property. Therefore, the is filled with all of the `NamedColor` instances. For each item in the , the binding context for the item is set to a `NamedColor` object. The and elements in the are bound to `NamedColor` properties. - -The defines the `x:DataType` attribute to be the `NamedColor` type, indicating that any binding expressions in the 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. For more information about the `x:Type` markup extension, see [x:Type Markup Extension](~/xaml/markup-extensions/consume.md#xtype-markup-extension). - -When the example is first run, the is populated with `NamedColor` instances. When an item in the is selected, the `BoxView.Color` property is set to the color of the selected item in the : - -:::image type="content" source="media/compiled-bindings/compiledcolorlist.png" alt-text="Compiled color list."::: - -Selecting other items in the updates the color of the . +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. For more information about the `x:Type` markup extension, see [x:Type Markup Extension](~/xaml/markup-extensions/consume.md#xtype-markup-extension). ::: moniker range=">=net-maui-9.0" @@ -174,8 +150,8 @@ Then, ensure that all your bindings are annotated with the correct `x:DataType` ```xaml - ``` diff --git a/docs/fundamentals/data-binding/media/compiled-bindings/compiledcolorlist.png b/docs/fundamentals/data-binding/media/compiled-bindings/compiledcolorlist.png deleted file mode 100755 index f4f08e5cb6815a9820fcadd6103ab4a5743a42da..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 29095 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 Date: Mon, 25 Nov 2024 04:05:17 -0500 Subject: [PATCH 08/11] Added note to warn about macOS requirement (#2640) --- docs/macios/xcsync.md | 3 +++ 1 file changed, 3 insertions(+) diff --git a/docs/macios/xcsync.md b/docs/macios/xcsync.md index 6e1fe1718..408a538c3 100644 --- a/docs/macios/xcsync.md +++ b/docs/macios/xcsync.md @@ -21,6 +21,9 @@ Supported file types include: The tool has two commands: `generate` and `sync`. Use `generate` to create an Xcode project from a .NET project and `sync` to bring changes in the Xcode project back to the .NET project. +> [!NOTE] +> This tool is only available for macOS due to the interactions with Xcode. + # [Visual Studio Code](#tab/visual-studio-code) From 6f9e0194da7c949a6c7e0c6dea1f1682611f38e0 Mon Sep 17 00:00:00 2001 From: Jeffrey Stedfast Date: Mon, 25 Nov 2024 05:38:43 -0500 Subject: [PATCH 09/11] Added manual steps for downloading iOS provisioning profiles (#2641) * Added manual steps for downloading iOS provisioning profiles * Fixed subtitles so links continue to work * Edits. * Fix linting error. * Fix bookmark. --------- Co-authored-by: David Britch --- docs/ios/capabilities.md | 2 +- .../manual-provisioning.md | 23 +++++++++++++++++-- 2 files changed, 22 insertions(+), 3 deletions(-) diff --git a/docs/ios/capabilities.md b/docs/ios/capabilities.md index c2cf31b82..d1b6dd38a 100644 --- a/docs/ios/capabilities.md +++ b/docs/ios/capabilities.md @@ -86,7 +86,7 @@ A provisioning profile can be created with the following steps: Once you've created a provisioning profile it must be downloaded by Visual Studio, and set as the provisioning profile for your project: -1. In Visual Studio, download the provisioning profile you've just created so that it's available for signing your app. For more information, see [Download provisioning profiles in Visual Studio](~/ios/device-provisioning/manual-provisioning.md#download-provisioning-profiles-in-visual-studio). +1. In Visual Studio, download the provisioning profile you've just created so that it's available for signing your app. For more information, see [Download provisioning profiles](~/ios/device-provisioning/manual-provisioning.md#download-provisioning-profiles). 1. In Visual Studio, enable manual provisioning for your project. For more information, see [Enable manual provisioning](~/ios/device-provisioning/manual-provisioning.md#enable-manual-provisioning). ## Troubleshoot diff --git a/docs/ios/device-provisioning/manual-provisioning.md b/docs/ios/device-provisioning/manual-provisioning.md index c4d3a5fe0..a6db723d9 100644 --- a/docs/ios/device-provisioning/manual-provisioning.md +++ b/docs/ios/device-provisioning/manual-provisioning.md @@ -1,7 +1,7 @@ --- title: "Manual provisioning for .NET MAUI iOS apps" description: "Learn how to use manual provisioning to create development certificates and profiles for .NET MAUI iOS apps." -ms.date: 08/28/2024 +ms.date: 11/25/2024 --- # Manual provisioning for iOS apps @@ -99,7 +99,11 @@ A development provisioning profile can be created with the following steps: 1. In the **Generate a Provisioning Profile** page, you can optionally click the **Download** button to download the provisioning profile. -## Download provisioning profiles in Visual Studio +## Download provisioning profiles + + +# [Visual Studio](#tab/vswin) + After creating a development provisioning profile in your Apple Developer Account, Visual Studio can download it so that it's available for signing your app: @@ -110,6 +114,21 @@ After creating a development provisioning profile in your Apple Developer Accoun The provisioning profiles will be downloaded on Windows, and exported to your Mac build host if the IDE is paired to it. For more information, see [Pair to Mac for iOS development](~/ios/pair-to-mac.md). + +# [Visual Studio Code](#tab/visual-studio-code) + + +After creating a development provisioning profile in your Apple Developer Account, you will need to download it in Xcode so that it's available for signing your app: + +1. Open the **Xcode** app. +2. Select the **Settings...** item in the **Xcode** menu. +3. Select the **@ Accounts** tab. +4. If you haven't already, add your Apple Developer Account. Otherwise, select your account. +5. In the right-hand pane with your account selected, select the appropriate **Team**. +6. Click the **Download Manual Profiles** button. + +--- + ## Enable manual provisioning After manually creating the development provisioning profile, and installing it in Visual Studio, your .NET MAUI app project should be configured to use manual provisioning: From 6635a1249c2c069abeaf22c995a2425c7e44aa37 Mon Sep 17 00:00:00 2001 From: David Britch Date: Wed, 27 Nov 2024 16:24:08 +0000 Subject: [PATCH 10/11] Replace Frame with Border (#2648) * Replace Frame with Border. * Edit. * EDit. * Edit. * Edit. * Edits. --- docs/fundamentals/controltemplate.md | 36 +++++----- .../data-binding/relative-bindings.md | 10 +-- docs/fundamentals/triggers.md | 13 ++-- docs/user-interface/brushes/gradient.md | 2 +- docs/user-interface/brushes/lineargradient.md | 65 +++++++++--------- docs/user-interface/brushes/radialgradient.md | 33 +++++---- docs/user-interface/brushes/solidcolor.md | 68 +++++++++---------- .../controls/carouselview/interaction.md | 4 +- .../controls/carouselview/layout.md | 40 +++++------ .../controls/carouselview/populate-data.md | 44 ++++++------ docs/user-interface/controls/contentview.md | 14 ++-- docs/user-interface/layouts/stacklayout.md | 68 +++++++++---------- .../layouts/verticalstacklayout.md | 20 +++--- docs/whats-new/dotnet-9.md | 27 +++++++- 14 files changed, 230 insertions(+), 214 deletions(-) diff --git a/docs/fundamentals/controltemplate.md b/docs/fundamentals/controltemplate.md index 813a02d5b..6bf44144c 100644 --- a/docs/fundamentals/controltemplate.md +++ b/docs/fundamentals/controltemplate.md @@ -53,19 +53,19 @@ The following XAML example shows a - - + ... ``` -When a 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 object. The object uses the [`RelativeSource`](xref:Microsoft.Maui.Controls.Xaml.RelativeSourceExtension) markup extension to set its to the runtime object instance to which the template will be applied, which is known as the *templated parent*. The object uses a combination of controls to define the visual structure of a `CardView` object. The binding expressions of these objects resolve against `CardView` properties, due to inheriting the from the root element. For more information about the [`RelativeSource`](xref:Microsoft.Maui.Controls.Xaml.RelativeSourceExtension) markup extension, see [Relative bindings](~/fundamentals/data-binding/relative-bindings.md). +When a 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 object. The object uses the [`RelativeSource`](xref:Microsoft.Maui.Controls.Xaml.RelativeSourceExtension) markup extension to set its to the runtime object instance to which the template will be applied, which is known as the *templated parent*. The object uses a combination of controls to define the visual structure of a `CardView` object. The binding expressions of these objects resolve against `CardView` properties, due to inheriting the from the root element. For more information about the [`RelativeSource`](xref:Microsoft.Maui.Controls.Xaml.RelativeSourceExtension) markup extension, see [Relative bindings](~/fundamentals/data-binding/relative-bindings.md). ## Consume a ControlTemplate @@ -95,7 +95,7 @@ The following example shows the `CardViewControlTemplate` being assigned to the ``` -In this example, the controls in the `CardViewControlTemplate` become part of the visual tree for each `CardView` object. Because the root object for the control template sets its to the templated parent, the and its children resolve their binding expressions against the properties of each `CardView` object. +In this example, the controls in the `CardViewControlTemplate` become part of the visual tree for each `CardView` object. Because the root object for the control template sets its to the templated parent, the and its children resolve their binding expressions against the properties of each `CardView` object. The following screenshot shows the `CardViewControlTemplate` applied to the the `CardView` objects: @@ -132,11 +132,11 @@ The following XAML example shows a - - + ... @@ -197,12 +197,12 @@ For example, the `CardViewUI` custom control defines its user interface using th xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml" x:Class="ControlTemplateDemos.Controls.CardViewUI" x:Name="this"> - - + ``` @@ -384,16 +384,16 @@ In this example, the - - + ``` -In this example, the root element of the is a object. The object uses the [`RelativeSource`](xref:Microsoft.Maui.Controls.Xaml.RelativeSourceExtension) markup extension to set its to the templated parent. The binding expressions of the object and its children resolve against `CardView` properties, due to inheriting the from the root element. The following screenshot shows the page displaying the `People` collection: +In this example, the root element of the is a object. The object uses the [`RelativeSource`](xref:Microsoft.Maui.Controls.Xaml.RelativeSourceExtension) markup extension to set its to the templated parent. The binding expressions of the object and its children resolve against `CardView` properties, due to inheriting the from the root element. The following screenshot shows the page displaying the `People` collection: :::image type="content" source="media/controltemplate/viewmodel-controltemplate.png" alt-text="Screenshot of three templated CardView objects that bind to a viewmodel."::: diff --git a/docs/fundamentals/data-binding/relative-bindings.md b/docs/fundamentals/data-binding/relative-bindings.md index 7d8f6998a..5a0b954d1 100644 --- a/docs/fundamentals/data-binding/relative-bindings.md +++ b/docs/fundamentals/data-binding/relative-bindings.md @@ -126,9 +126,9 @@ The following XAML shows an example of the `TemplatedParent` relative binding mo - ... @@ -139,7 +139,7 @@ The following XAML shows an example of the `TemplatedParent` relative binding mo - + @@ -165,7 +165,7 @@ The following XAML shows an example of the `TemplatedParent` relative binding mo ``` -In this example, the , which is the root element of the , has its `BindingContext` set to the runtime object instance to which the template is applied. Therefore, the and its children resolve their binding expressions against the properties of each `CardView` object: +In this example, the , which is the root element of the , has its `BindingContext` set to the runtime object instance to which the template is applied. Therefore, the and its children resolve their binding expressions against the properties of each `CardView` object: :::image type="content" source="media/relative-bindings/templatedparent-relative-binding.png" alt-text="Screenshot of a TemplatedParent mode relative binding."::: diff --git a/docs/fundamentals/triggers.md b/docs/fundamentals/triggers.md index 319604f92..c6d21d3f0 100644 --- a/docs/fundamentals/triggers.md +++ b/docs/fundamentals/triggers.md @@ -410,18 +410,19 @@ The following XAML example shows a that inc ... - + - + ``` diff --git a/docs/user-interface/brushes/gradient.md b/docs/user-interface/brushes/gradient.md index 4ccf7f439..7ec6f7886 100644 --- a/docs/user-interface/brushes/gradient.md +++ b/docs/user-interface/brushes/gradient.md @@ -52,6 +52,6 @@ The following XAML example creates a diagonal object and set its `StartPoint` to (0,0) and its `EndPoint` to (1,0). Then, add two or more objects to the `LinearGradientBrush.GradientStops` collection, that specify the colors in the gradient and their positions. -The following XAML example shows a horizontal that's set as the `Background` of a : +The following XAML example shows a horizontal that's set as the `Background` of a : ```xaml - - + + - - + + ``` -In this example, the background of the is painted with a that interpolates from yellow to green horizontally: +In this example, the background of the is painted with a that interpolates from yellow to green horizontally: -:::image type="content" source="media/lineargradient/horizontal.png" alt-text="Screenshot of a Frame painted with a horizontal LinearGradientBrush."::: +:::image type="content" source="media/lineargradient/horizontal.png" alt-text="Screenshot of a Border painted with a horizontal LinearGradientBrush."::: ### Create a vertical linear gradient To create a vertical linear gradient, create a object and set its `StartPoint` to (0,0) and its `EndPoint` to (0,1). Then, add two or more objects to the `LinearGradientBrush.GradientStops` collection, that specify the colors in the gradient and their positions. -The following XAML example shows a vertical that's set as the `Background` of a : +The following XAML example shows a vertical that's set as the `Background` of a : ```xaml - - + + - - + + ``` -In this example, the background of the is painted with a that interpolates from yellow to green vertically: +In this example, the background of the is painted with a that interpolates from yellow to green vertically: -:::image type="content" source="media/lineargradient/vertical.png" alt-text="Screenshot of a Frame painted with a vertical LinearGradientBrush."::: +:::image type="content" source="media/lineargradient/vertical.png" alt-text="Screenshot of a Border painted with a vertical LinearGradientBrush."::: ### Create a diagonal linear gradient To create a diagonal linear gradient, create a object and set its `StartPoint` to (0,0) and its `EndPoint` to (1,1). Then, add two or more objects to the `LinearGradientBrush.GradientStops` collection, that specify the colors in the gradient and their positions. -The following XAML example shows a diagonal that's set as the `Background` of a : +The following XAML example shows a diagonal that's set as the `Background` of a : ```xaml - - + + @@ -109,10 +106,10 @@ The following XAML example shows a diagonal - - + + ``` -In this example, the background of the is painted with a that interpolates from yellow to green diagonally: +In this example, the background of the is painted with a that interpolates from yellow to green diagonally: -:::image type="content" source="media/lineargradient/diagonal.png" alt-text="Screenshot of a Frame painted with a diagonal LinearGradientBrush."::: +:::image type="content" source="media/lineargradient/diagonal.png" alt-text="Screenshot of a Border painted with a diagonal LinearGradientBrush."::: diff --git a/docs/user-interface/brushes/radialgradient.md b/docs/user-interface/brushes/radialgradient.md index 6cbe34981..99ae771b5 100644 --- a/docs/user-interface/brushes/radialgradient.md +++ b/docs/user-interface/brushes/radialgradient.md @@ -28,15 +28,14 @@ A radial gradient brush's gradient stops are positioned along a gradient axis de To create a radial gradient, create a object and set its `Center` and `Radius` properties. Then, add two or more objects to the `RadialGradientBrush.GradientStops` collection, that specify the colors in the gradient and their positions. -The following XAML example shows a that's set as the `Background` of a : +The following XAML example shows a that's set as the `Background` of a : ```xaml - - + + @@ -45,15 +44,15 @@ The following XAML example shows a - - + + ``` -In this example, the background of the is painted with a that interpolates from red to dark blue. The center of the radial gradient is positioned in the center of the : +In this example, the background of the is painted with a that interpolates from red to dark blue. The center of the radial gradient is positioned in the center of the : -:::image type="content" source="media/radialgradient/center.png" alt-text="Screenshot of a Frame painted with a centered RadialGradientBrush."::: +:::image type="content" source="media/radialgradient/center.png" alt-text="Screenshot of a Border painted with a centered RadialGradientBrush."::: -The following XAML example moves the center of the radial gradient to the top-left corner of the : +The following XAML example moves the center of the radial gradient to the top-left corner of the : ```xaml @@ -65,11 +64,11 @@ The following XAML example moves the center of the radial gradient to the top-le ``` -In this example, the background of the is painted with a that interpolates from red to dark blue. The center of the radial gradient is positioned in the top-left of the : +In this example, the background of the is painted with a that interpolates from red to dark blue. The center of the radial gradient is positioned in the top-left of the : -:::image type="content" source="media/radialgradient/top-left.png" alt-text="Screenshot of a Frame painted with a top-left RadialGradientBrush."::: +:::image type="content" source="media/radialgradient/top-left.png" alt-text="Screenshot of a Border painted with a top-left RadialGradientBrush."::: -The following XAML example moves the center of the radial gradient to the bottom-right corner of the : +The following XAML example moves the center of the radial gradient to the bottom-right corner of the : ```xaml @@ -81,6 +80,6 @@ The following XAML example moves the center of the radial gradient to the bottom ``` -In this example, the background of the is painted with a that interpolates from red to dark blue. The center of the radial gradient is positioned in the bottom-right of the : +In this example, the background of the is painted with a that interpolates from red to dark blue. The center of the radial gradient is positioned in the bottom-right of the : -:::image type="content" source="media/radialgradient/bottom-right.png" alt-text="Screenshot of a Frame painted with a bottom-right RadialGradientBrush."::: +:::image type="content" source="media/radialgradient/bottom-right.png" alt-text="Screenshot of a Border painted with a bottom-right RadialGradientBrush."::: diff --git a/docs/user-interface/brushes/solidcolor.md b/docs/user-interface/brushes/solidcolor.md index 970c33152..176538f1a 100644 --- a/docs/user-interface/brushes/solidcolor.md +++ b/docs/user-interface/brushes/solidcolor.md @@ -23,61 +23,58 @@ There are three main techniques for creating a from a value. In XAML, this enables a to be created from a predefined value: ```xaml - + ``` -In this example, the background of the is painted with a dark blue : +In this example, the background of the is painted with a dark blue : -:::image type="content" source="media/solidcolor/predefined-color.png" alt-text="Screenshot of a Frame painted with a predefined color."::: +:::image type="content" source="media/solidcolor/predefined-color.png" alt-text="Screenshot of a Border painted with a predefined color."::: Alternatively, the value can be specified using property tag syntax: ```xaml - - - - - + + + + + ``` -In this example, the background of the is painted with a whose color is specified by setting the `SolidColorBrush.Color` property. +In this example, the background of the is painted with a whose color is specified by setting the `SolidColorBrush.Color` property. ### Use a predefined Brush The class defines a set of commonly used objects. The following example uses one of these predefined objects: ```xaml - + ``` The equivalent C# code is: ```csharp -Frame frame = new Frame +Border border = new Border { Background = Brush.Indigo, - BorderColor = Colors.LightGray, + Stroke = Colors.LightGray, // ... }; ``` -In this example, the background of the is painted with an indigo : +In this example, the background of the is painted with an indigo : -:::image type="content" source="media/solidcolor/predefined-brush.png" alt-text="Screenshot of a Frame painted with a predefined SolidColorBrush."::: +:::image type="content" source="media/solidcolor/predefined-brush.png" alt-text="Screenshot of a Border painted with a predefined SolidColorBrush."::: For a list of predefined objects provided by the class, see [Solid color brushes](#solid-color-brushes). @@ -94,17 +91,16 @@ In addition, a color can be specified as `#aarrggbb` where `aa` specifies the al The following example sets the color value of a using hexadecimal notation: ```xaml - + ``` -In this example, the background of the is painted with a salmon-colored : +In this example, the background of the is painted with a salmon-colored : -:::image type="content" source="media/solidcolor/hex.png" alt-text="Screenshot of a Frame painted with a SolidColorBrush created with hexadecimal notation."::: +:::image type="content" source="media/solidcolor/hex.png" alt-text="Screenshot of a Border painted with a SolidColorBrush created with hexadecimal notation."::: For other ways of describing color, see [Colors](~/user-interface/graphics/colors.md). diff --git a/docs/user-interface/controls/carouselview/interaction.md b/docs/user-interface/controls/carouselview/interaction.md index c9c83b0d5..3eb22f0de 100644 --- a/docs/user-interface/controls/carouselview/interaction.md +++ b/docs/user-interface/controls/carouselview/interaction.md @@ -316,9 +316,9 @@ The following XAML example shows how to define the `CurrentItem`, `PreviousItem` - + ... - + diff --git a/docs/user-interface/controls/carouselview/layout.md b/docs/user-interface/controls/carouselview/layout.md index 651e5bfe3..4ca89c2d8 100644 --- a/docs/user-interface/controls/carouselview/layout.md +++ b/docs/user-interface/controls/carouselview/layout.md @@ -42,13 +42,13 @@ By default, will display its items h - + - + @@ -110,13 +110,13 @@ This results in a layout that grows horizontally as new items are added. - + - + @@ -216,19 +216,19 @@ This code results in a vertical layout that has a spacing of 20 between items. ## Dynamic resizing of items -Items in a can be dynamically resized at runtime by changing layout related properties of elements within the . For example, the following code example changes the and properties of an object, and the property of its parent : +Items in a can be dynamically resized at runtime by changing layout related properties of elements within the . For example, the following code example changes the and properties of an object, and the property of its parent : ```csharp void OnImageTapped(object sender, EventArgs e) { Image image = sender as Image; image.HeightRequest = image.WidthRequest = image.HeightRequest.Equals(150) ? 200 : 150; - Frame frame = ((Frame)image.Parent.Parent); - frame.HeightRequest = frame.HeightRequest.Equals(300) ? 350 : 300; + Border border = ((Border)image.Parent.Parent); + border.HeightRequest = border.HeightRequest.Equals(360) ? 410 : 360; } ``` -The `OnImageTapped` event handler is executed in response to an object being tapped, and changes the dimensions of the image (and its parent , so that it's more easily viewed: +The `OnImageTapped` event handler is executed in response to an object being tapped, and changes the dimensions of the image (and its parent ), so that it's more easily viewed: :::image type="content" source="media/layout/runtime-resizing.png" alt-text="Screenshot of a CarouselView with dynamic item sizing."::: diff --git a/docs/user-interface/controls/carouselview/populate-data.md b/docs/user-interface/controls/carouselview/populate-data.md index fc0533727..420732f99 100644 --- a/docs/user-interface/controls/carouselview/populate-data.md +++ b/docs/user-interface/controls/carouselview/populate-data.md @@ -55,13 +55,13 @@ The appearance of each item in the c - + - + @@ -114,9 +114,9 @@ carouselView.ItemTemplate = new DataTemplate(() => stackLayout.Add(locationLabel); stackLayout.Add(detailsLabel); - Frame frame = new Frame { ... }; + Border border = new Border { ... }; StackLayout rootStackLayout = new StackLayout(); - rootStackLayout.Add(frame); + rootStackLayout.Add(border); return rootStackLayout; }); @@ -239,13 +239,13 @@ For more information about indicators, see [IndicatorView](~/user-interface/cont - + @@ -269,7 +269,7 @@ For more information about indicators, see [IndicatorView](~/user-interface/cont - + @@ -285,7 +285,7 @@ carouselView.SetBinding(ItemsView.ItemsSourceProperty, "Monkeys"); carouselView.ItemTemplate = new DataTemplate(() => { StackLayout stackLayout = new StackLayout(); - Frame frame = new Frame { ... }; + Border border = new Border { ... }; SwipeView swipeView = new SwipeView(); SwipeItem favoriteSwipeItem = new SwipeItem @@ -311,14 +311,14 @@ carouselView.ItemTemplate = new DataTemplate(() => StackLayout swipeViewStackLayout = new StackLayout { ... }; swipeView.Content = swipeViewStackLayout; - frame.Content = swipeView; - stackLayout.Add(frame); + border.Content = swipeView; + stackLayout.Add(border); return stackLayout; }); ``` -In this example, the content is a that defines the appearance of each item that's surrounded by a in the . The swipe items are used to perform actions on the content, and are revealed when the control is swiped from the bottom and from the top: +In this example, the content is a that defines the appearance of each item that's surrounded by a in the . The swipe items are used to perform actions on the content, and are revealed when the control is swiped from the bottom and from the top: :::image type="content" source="media/populate-data/swipeview-bottom.png" alt-text="Screenshot of a CarouselView bottom context menu item."::: :::image type="content" source="media/populate-data/swipeview-top.png" alt-text="Screenshot of a CarouselView top context menu item."::: diff --git a/docs/user-interface/controls/contentview.md b/docs/user-interface/controls/contentview.md index 4c634a6d3..7dc6baf5e 100644 --- a/docs/user-interface/controls/contentview.md +++ b/docs/user-interface/controls/contentview.md @@ -75,18 +75,18 @@ The custom control UI can be defined in the XAML file for the - ... - - + ``` diff --git a/docs/user-interface/layouts/stacklayout.md b/docs/user-interface/layouts/stacklayout.md index 7abfaa835..2aec4a12e 100644 --- a/docs/user-interface/layouts/stacklayout.md +++ b/docs/user-interface/layouts/stacklayout.md @@ -286,8 +286,8 @@ The following XAML shows an example of nesting ... - + - - + + - - + + - + ... ``` -In this example, the parent contains nested objects inside objects. The parent is oriented vertically, while the child objects are oriented horizontally: +In this example, the parent contains nested objects inside objects. The parent is oriented vertically, while the child objects are oriented horizontally: :::image type="content" source="media/stacklayout/nested.png" alt-text="Nested .NET MAUI StackLayouts."::: @@ -340,59 +340,59 @@ public class CombinedStackLayoutPage : ContentPage { Title = "Combined StackLayouts demo"; - Frame frame1 = new Frame + Border border1 = new Border { - BorderColor = Colors.Black, + Stroke = Colors.Black, Padding = new Thickness(5) }; - StackLayout frame1StackLayout = new StackLayout + StackLayout border1StackLayout = new StackLayout { Orientation = StackOrientation.Horizontal, Spacing = 15 }; - frame1StackLayout.Add(new BoxView { Color = Colors.Red, WidthRequest = 40 }); - frame1StackLayout.Add(new Label { Text = "Red", FontSize = 22, VerticalOptions = LayoutOptions.Center }); - frame1.Content = frame1StackLayout; + border1StackLayout.Add(new BoxView { Color = Colors.Red, WidthRequest = 40 }); + border1StackLayout.Add(new Label { Text = "Red", FontSize = 20, VerticalOptions = LayoutOptions.Center }); + border1.Content = border1StackLayout; - Frame frame2 = new Frame + Border border2 = new Border { - BorderColor = Colors.Black, + Stroke = Colors.Black, Padding = new Thickness(5) }; - StackLayout frame2StackLayout = new StackLayout + StackLayout border2StackLayout = new StackLayout { Orientation = StackOrientation.Horizontal, Spacing = 15 }; - frame2StackLayout.Add(new BoxView { Color = Colors.Yellow, WidthRequest = 40 }); - frame2StackLayout.Add(new Label { Text = "Yellow", FontSize = 22, VerticalOptions = LayoutOptions.Center }); - frame2.Content = frame2StackLayout; + border2StackLayout.Add(new BoxView { Color = Colors.Yellow, WidthRequest = 40 }); + border2StackLayout.Add(new Label { Text = "Yellow", FontSize = 20, VerticalOptions = LayoutOptions.Center }); + border2.Content = border2StackLayout; - Frame frame3 = new Frame + Border border3 = new Border { - BorderColor = Colors.Black, + Stroke = Colors.Black, Padding = new Thickness(5) }; - StackLayout frame3StackLayout = new StackLayout + StackLayout border3StackLayout = new StackLayout { Orientation = StackOrientation.Horizontal, Spacing = 15 }; - frame3StackLayout.Add(new BoxView { Color = Colors.Blue, WidthRequest = 40 }); - frame3StackLayout.Add(new Label { Text = "Blue", FontSize = 22, VerticalOptions = LayoutOptions.Center }); - frame3.Content = frame3StackLayout; + border3StackLayout.Add(new BoxView { Color = Colors.Blue, WidthRequest = 40 }); + border3StackLayout.Add(new Label { Text = "Blue", FontSize = 20, VerticalOptions = LayoutOptions.Center }); + border3.Content = border3StackLayout; ... StackLayout stackLayout = new StackLayout { Margin = new Thickness(20) }; stackLayout.Add(new Label { Text = "Primary colors" }); - stackLayout.Add(frame1); - stackLayout.Add(frame2); - stackLayout.Add(frame3); + stackLayout.Add(border1); + stackLayout.Add(border2); + stackLayout.Add(border3); stackLayout.Add(new Label { Text = "Secondary colors" }); - stackLayout.Add(frame4); - stackLayout.Add(frame5); - stackLayout.Add(frame6); + stackLayout.Add(border4); + stackLayout.Add(border5); + stackLayout.Add(border6); Content = stackLayout; } diff --git a/docs/user-interface/layouts/verticalstacklayout.md b/docs/user-interface/layouts/verticalstacklayout.md index 2c0f4048c..2e688fb42 100644 --- a/docs/user-interface/layouts/verticalstacklayout.md +++ b/docs/user-interface/layouts/verticalstacklayout.md @@ -154,8 +154,8 @@ The following XAML shows an example of nesting