Blog post

How to Access Native Code from WebView2 with WinAppSDK/WinUI 3

Illustration: How to Access Native Code from WebView2 with WinAppSDK/WinUI 3

At PSPDFKit, we continuously strive to add support for the latest technologies on the market. As part of this effort, we’ve been researching the Windows App SDK (WinAppSDK), which includes WinUI 3. WinAppSDK is the successor of UWP and will shape the future of Windows app development. This post provides step-by-step instructions for accessing native code using WebView2 and WinAppSDK.

This post will walk through creating a WinAppSDK app to host a local HTML file, which will in turn access the Add function defined in the C++/WinRT project to add two numbers and display the result. We deliberately chose this simple example to both demonstrate how functions are used in programming and keep the blog post focused on the interop between JavaScript and C#.

You’ll create three projects:

  1. AccessNativeCodeInJS — This is the WinAppSDK project you’ll use to host WebView2 for the local HTML.

  2. NativeCode — This project will have the APIs you want to access in JavaScript hosted in WebView2.

  3. NativeAdapter — This is the wrapper of the NativeCode project in which you’ll add the WebView2 WinRT JS Projection tool — wv2winrt — to access native code in JavaScript. wv2winrt generates the files necessary for your JavaScript code to access the native code.

relation-chart

Information

WinUI 3 is the native UI platform component that ships with the Windows App SDK. Hence, WinUI 3 and Windows App SDK are used synonymously in many places.

Setting Up the Projects

These next sections will outline the various steps required to create your projects.

Creating a New WinUI 3 Project — AccessNativeCodeInJS

  1. Open Visual Studio and click Create a new project.

Visual Studio startup window

  1. Search for Blank App, Packaged (WinUI 3 in Desktop) and click Next.

Create new project window

  1. Under the project name, type AccessNativeCodeInJS. Then press Enter.

  2. Once the solution is created, the project tree will look like what’s shown below.

Project structure after step 4

Warning

If the project tree doesn’t look like the image above, ensure you selected the correct project template, as highlighted in second point of this step.

  1. Double-click the project name to access the project file. Then you’ll see the TargetFramework, which you’ll need to note, as you’ll use the same version while creating the other projects. For us, it was net6.0-windows10.0.19041.0, i.e. 10.0.19041.0.

  2. Right-click Assets and select New Item. In the dialog that’s presented, search for HTML Page, rename it to index.html, and click Add.

  3. From the context menu of the index.html file, select Properties and update the Build Action to Content.

  4. Add the WebView2 control to MainWindow.xaml, and create a virtual mapping of the local folder, Assets, using CoreWebView2.SetVirtualHostNameToFolderMapping to host index.html.

<!--MainWindow.xaml-->
<Window
    x:Class="AccessNativeCodeInJS.MainWindow"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:local="using:AccessNativeCodeInJS"
    xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
    xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
    mc:Ignorable="d">
    <Grid>
        <Grid.RowDefinitions>
            <RowDefinition Height="*"/>
        </Grid.RowDefinitions>
        <WebView2 x:Name="webView2"/>
    </Grid>
</Window>
// MainWindow.xaml.cs
using Microsoft.UI.Xaml;
using Microsoft.Web.WebView2.Core;
using System;

namespace AccessNativeCodeInJS
{
    public sealed partial class MainWindow : Window
    {
        public MainWindow()
        {
            this.InitializeComponent();
            webView2.Loaded += WebView2_Loaded;
        }

        private async void WebView2_Loaded(object sender, RoutedEventArgs e)
        {
			    // Ensure the `CoreWebView2` object has been initialized.
            await webView2.EnsureCoreWebView2Async();
            webView2.CoreWebView2.SetVirtualHostNameToFolderMapping(hostName: "pspdfkit",
                                                       folderPath: "Assets",
                                                       accessKind: CoreWebView2HostResourceAccessKind.Allow);

            webView2.Source = new Uri("https://pspdfkit/index.html");
        }
    }
}

Creating a New C++/WinRT Project — NativeCode

  1. In the context menu of the solution, click Add > New Project.

  2. Search for and select Windows Runtime Component (C++/WinRT) and click Next.

WinRT New project dialog

  1. Name the project NativeCode, and click Create.

  2. Select the same target platform versions as those of AccessNativeCodeInJS — i.e. 10.0.19041.0 — and press Enter.

Target version dialog

  1. Now the solution tree will look like what’s shown below.

Solution tree after adding NativeCode project

  1. Rename Class.idl to PlayWithNumbers.idl. This will automatically rename the related Class.h and Class.cpp files.

  2. To rename the runtime class and add an Add function, replace the contents of PlayWithNumbers.idl and its corresponding header and CPP files with the following code:

// PlayWithNumbers.idl
namespace NativeCode
{
    runtimeclass PlayWithNumbers
    {
        PlayWithNumbers();
        Int32 Add(Int32 number1, Int32 number2);
    }
}
// PlayWithNumbers.h
#pragma once

#include "PlayWithNumbers.g.h"

namespace winrt::NativeCode::implementation
{
    struct PlayWithNumbers : PlayWithNumbersT<PlayWithNumbers>
    {
        PlayWithNumbers() = default;

        int32_t Add(int32_t num1, int32_t num2);
    };
}

namespace winrt::NativeCode::factory_implementation
{
    struct PlayWithNumbers : PlayWithNumbersT<PlayWithNumbers, implementation::PlayWithNumbers>
    {
    };
}
// PlayWithNumbers.cpp
#include "pch.h"
#include "PlayWithNumbers.h"
#include "PlayWithNumbers.g.cpp"

namespace winrt::NativeCode::implementation
{
    int32_t PlayWithNumbers::Add(int32_t num1, int32_t num2)
    {
        return num1 + num2;
    }
}

Creating Another C++/WinRT Project — NativeAdapter

  1. Create another Windows Runtime Component (C++/WinRT) project and name it NativeAdapter. Then, follow the first four steps in the section above.

  2. The updated solution tree will look like what’s shown below.

Solution tree after adding NativeAdapter project

  1. Delete the default class added to the project, Class.idl.

  2. Under the Manage NuGet Packages context menu item, make sure the Package source is set to Nuget.org. If it doesn’t exist, create one using this source.

NuGet package manager source

  1. Browse for the Microsoft.Windows.ImplementationLibrary NuGet package and install the latest version. The Windows Implementation Library (WIL) is a header-only C++ library to make using Component Object Model (COM) coding for Windows easier. It provides readable, type-safe C++ interfaces for Windows COM coding patterns.

WIL NuGet Package

  1. Browse for the Microsoft.Web.WebView2 NuGet package and install the latest version of it as well.

WebView2 NuGet Package

  1. Right-click References. Then, click Add Reference… to add a reference to the NativeCode project.

  2. From the context menu of NativeAdapter, select Properties > Common Properties. Click Include filters and add NativeCode, which is the namespace for accessing APIs in JavaScript. Then, update the properties so that the dialog looks like what’s shown below.

You can use the command-line help to find more information about the parameters of wv2winrt.exe.

Native Adapter Property Pages dialog

Information

The $(WebView2DispatchAdapterIncludeFilters) part in the Include filters part of the image above is generated automatically.

  1. Build the NativeAdapter project to generate the required files used in the AccessNativeCodeInJS project.

Adding a NativeAdapter Reference to AccessNativeCodeInJS

  1. To make the WinAppSDK project understand the C++/WinRT reference, add the Microsoft.Winows.CsWinRT NuGet package. To do this, from the context menu of the AccessNativeCodeInJS project, click Manage NuGet Packages…, which installs CSWinRT.

CSWinRT NuGet Package

  1. Under AccessNativeCodeInJS, in the context menu of Dependencies, click Add Project Reference… to add a reference to NativeAdapter.

CSWinRT NuGet Package

  1. Double-click the AccessNativeCodeInJS project to access its project file. Add the following lines before the closing project tag, </Project>, and after all the other PropertyGroup tags in the file:

<PropertyGroup>
    <CsWinRTIncludes>NativeAdapter</CsWinRTIncludes>
</PropertyGroup>

Calling APIs from NativeCode in JavaScript Hosted in AccessNativeCodeInJS

  1. Replace the WebView2_Loaded function in MainWindow.xaml.cs with the code below.

Here, you create a DispatcherAdapter object and add it to WebView2 to inject a native object into web:

// In MainWindow.xaml.cs
private async void WebView2_Loaded(object sender, RoutedEventArgs e)
{
    await webView2.EnsureCoreWebView2Async();

		// Add the following two lines.
    var dispatchAdapter = new NativeAdapter.DispatchAdapter();
    webView2.CoreWebView2.AddHostObjectToScript("NativeCode", dispatchAdapter.WrapNamedObject("NativeCode", dispatchAdapter));
		// End of added lines.

    webView2.CoreWebView2.SetVirtualHostNameToFolderMapping(hostName: "pspdfkit",
                                               folderPath: "Assets",
                                               accessKind: CoreWebView2HostResourceAccessKind.Allow);
    webView2.Source = new Uri("https://pspdfkit/index.html");
}
  1. In the index.html file:

    a. Create a user interface to accept two numbers and display the result.

    b. Add a script tag at the end of the body to access the DispatcherAdapter object and create an object of the PlayWithNumbers class. In the add method, call playWithNumbers.add from NativeCode to add the two numbers and display the result.

    The resulting HTML file will look like this:

    <!DOCTYPE html>
    
    <html lang="en" xmlns="http://www.w3.org/1999/xhtml">
    	<head>
    		<meta charset="utf-8" />
    		<title>Access Native Code in JS Example</title>
    	</head>
    	<body>
    		<h1>Access Native Code in JS Example</h1>
    		<input
    			type="number"
    			id="Number1"
    			value="1"
    			style="width:2rem"
    		/>
    		<label>+</label>
    		<input
    			type="number"
    			id="Number2"
    			value="4"
    			style="width:2rem"
    		/>
    		<button onclick="add()">=</button>
    		<label id="result">5</label>
    
    		<script>
    			const nativeCode =
    				chrome.webview.hostObjects.sync.NativeCode;
    			const playWithNumbers = new nativeCode.PlayWithNumbers();
    
    			function add() {
    				let num1 = document.getElementById('Number1')
    					.valueAsNumber;
    				let num2 = document.getElementById('Number2')
    					.valueAsNumber;
    				document.getElementById(
    					'result',
    				).innerHTML = playWithNumbers.add(num1, num2);
    			}
    		</script>
    	</body>
    </html>
Warning

Note that Add from PlayWithNumbers has an uppercase letter A, whereas when calling it in JavaScript, you use a lowercase letter a. This is because you set Use JavaScript case to Yes in the WebView2 Properties of NativeAdapter.

Building and Running the Application

Press Control-F5 to build and run the application without the debugger attached, or just press F5 to run the application with the debugger attached.

Application screenshot

Warning

The debugger is attached to the native app and can only debug the application code, and not the JavaScript code inside WebView2. To debug the code from WebView2, refer to the Microsoft documentation.

Conclusion

This blog post demonstrated how to access native code from WebView2 using the Windows App SDK (WinAppSDK) and WinUI 3. By following the step-by-step instructions provided, you created a WinAppSDK app that hosts a local HTML file and accesses native code defined in a C++/WinRT project.

You’re now able to successfully integrate WebView2 and WinAppSDK to leverage the power of native code in your JavaScript-based Windows applications. In addition to calling JavaScript from native Windows code, it’s also possible to call native code from JavaScript. This allows full utilization of the performance of native code and access to the capabilities of Windows. Microsoft makes this even easier by providing the wv2winrt tool, which generates the necessary code to enable this interoperability.

References

  1. Windows Implementation Library

  2. C#/WinRT

  3. Call native-side WinRT code from web-side code

Author
Shantanu Methikar
Shantanu Methikar .NET Engineer

Shantanu likes to explore and experiment — be it with technology, traveling, food, or almost anything else.

Free trial Ready to get started?
Free trial