Embedding Existing JavaScript Components Into Uno-WASM - Part 3

In the previous article, a simple syntax highlighter was used to enhance the display of text in HTML. But it is not enough for most apps: it's often required for JavaScript components to call back into the C# side of application. The easiest way to do that in Uno for WebAssembly is by using DOM Events. Applications using Uno can consume DOM Events and CustomEvents very easily.

Let's create an application illustrating how to use this feature.

Integration of Flatpickr - Callback to app from JavaScript

📝 Flatpickr is a lightweight, self-contained date and time picker. It is an easy way to explore how JavaScript-side code can call back to the managed application using a CustomEvent. In this case, this will be used to report when the picker is opened and a date and time was picked.

0. Before starting

📝 To reproduce the code in this article, you must prepare development environment using Uno's Getting Started article.

1. Create the solution in Visual Studio

📝 This part is very short because it is similar to the previous article (part 2):

  1. Create a Uno Platform App project and name it FlatpickrDemo.
  2. Right-click on the .Wasm project in the Solution Explorer and pick Set as Startup Project.
  3. Update to latest stable version of Uno.* dependencies.
  4. Compile & Run to make sure everything works.

2. Inject Flatpickr from CDN

🎯 This section is using a CDN to get Flatpickr instead of hosting the JavaScript directly in the application. It is not always the best solution as it creates a dependency on the Internet availability. Any change made server-side could break the application.

An easy way to achieve this is to add JavaScript code to load the CSS file directly from the CDN. The JavaScript portion of Flatpickr will be lazy-loaded with the control later.

  1. Create a new JavaScript file flatpickrloader.js in the WasmScripts folder of the .Wasm project:

    (function () {
        const head = document.getElementsByTagName("head")[0];
    
        // Load Flatpickr CSS from CDN
        const link = document.createElement("link");
        link.rel = "stylesheet";
        link.href = "https://cdn.jsdelivr.net/npm/flatpickr/dist/flatpickr.min.css";
        head.appendChild(link);
    })();
    

    This will load the Flatpickr assets directly from CDN.

  2. Set the file as Embedded Resource:

    <ItemGroup>
      <EmbeddedResource Include="WasmCSS\Fonts.css" />
      <EmbeddedResource Include="WasmScripts\AppManifest.js" />
      <EmbeddedResource Include="WasmScripts\flatpickrloader.js" /> <!-- add this line -->
    </ItemGroup>
    

3. Uno controls and XAML

🎯 This section is creating a control used in the XAML. It will activate Flatpickr on the control's <input> element.

  1. Create a FlatpickrView.cs class in the [MyApp] project like this:

    using System;
    using System.Globalization;
    using Windows.UI;
    using Windows.UI.Xaml;
    using Windows.UI.Xaml.Media;
    using Uno.Foundation;
    using Uno.Extensions;
    using Uno.UI.Runtime.WebAssembly;
    
    namespace FlatpickrDemo.Shared
    {
        [HtmlElement("input")] // Flatpickr requires an <input> HTML element
        public class FlatpickrView : FrameworkElement
        {
            // *************************
            // * Dependency Properties *
            // *************************
    
            public static readonly DependencyProperty SelectedDateTimeProperty = DependencyProperty.Register(
                "SelectedDateTime", typeof(DateTimeOffset?), typeof(FlatpickrView), new PropertyMetadata(default(DateTimeOffset?)));
    
            public DateTimeOffset? SelectedDateTime
            {
                get => (DateTimeOffset)GetValue(SelectedDateTimeProperty);
                set => SetValue(SelectedDateTimeProperty, value);
            }
    
            public static readonly DependencyProperty IsPickerOpenedProperty = DependencyProperty.Register(
                "IsPickerOpened", typeof(bool), typeof(FlatpickrView), new PropertyMetadata(false));
    
            public bool IsPickerOpened
            {
                get => (bool)GetValue(IsPickerOpenedProperty);
                set => SetValue(IsPickerOpenedProperty, value);
            }
    
            // ***************
            // * Constructor *
            // ***************
    
            public FlatpickrView()
            {
                // Load Flatpickr using JavaScript
                LoadJavaScript();
            }
    
            // ******************
            // * Initialization *
            // ******************
    
            private void LoadJavaScript()
            {
                // For demo purposes, Flatpickr is loaded directly from CDN.
                // Uno uses AMD module loading, so you must give a callback when the resource is loaded.
                // We can access the corresponding DOM HTML Element by using the "element" variable available in the scope
                var javascript = $@"require([""https://cdn.jsdelivr.net/npm/flatpickr""], f => f(element));";
    
                this.ExecuteJavascript(javascript);
            }
    
            // ******************
            // * Event Handlers *
            // ******************
    
        }
    }
    
  2. Change the MainPage.xaml in the [MyApp] project like this:

    <Page
        x:Class="FlatpickrDemo.MainPage"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:local="using:FlatpickrDemo"
        xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
        xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
        mc:Ignorable="d"
        Background="{ThemeResource ApplicationPageBackgroundThemeBrush}">
    
        <StackPanel Spacing="10" Padding="20">
          <TextBlock FontSize="15">
         Is Picker opened: <Run FontSize="20" FontWeight="Bold" Text="{Binding IsPickerOpened, ElementName=picker}" />
             <LineBreak />Picked Date/Time: <Run FontSize="20" FontWeight="Bold" Text="{Binding SelectedDateTime, ElementName=picker}" />
          </TextBlock>
          <TextBlock FontSize="20">Flatpickr control:</TextBlock>
    
          <local:FlatpickrView Height="20" Width="300"  x:Name="picker" HorizontalAlignment="Left" />
        </StackPanel>
    </Page>
    
  3. After pressing CTRL-F5, after clicking on the <input> rectangle, this should appear:

    Calendar picker 1

📝 Almost there, still need to call back to the managed code portion of the application.

4. Add a way to call managed code from JavaScript

🎯 This section will use CustomEvent to route Flatpickr's events to managed code.

  1. Register event handlers for 2 custom events: DateChanged and OpenedStateChanged. To achieve this, put this code at the end of the FlatpickrView constructor:

    // Register event handler for custom events from the DOM
    this.RegisterHtmlCustomEventHandler("DateChanged", OnDateChanged, isDetailJson: false);
    this.RegisterHtmlCustomEventHandler("OpenedStateChanged", OnOpenedStateChanged, isDetailJson: false);
    
  2. Add the implementation for the two handlers in the class:

    private void OnDateChanged(object sender, HtmlCustomEventArgs e)
    {
        if(DateTimeOffset.TryParse(e.Detail, DateTimeFormatInfo.InvariantInfo, DateTimeStyles.AssumeLocal, out var dto))
        {
            SelectedDateTime = dto;
        }
    }
    
    private void OnOpenedStateChanged(object sender, HtmlCustomEventArgs e)
    {
        switch(e.Detail)
        {
            case "open":
                IsPickerOpened = true;
                break;
            case "closed":
                IsPickerOpened = false;
                break;
        }
    }
    
  3. Change the initialization of Flatpickr in injected JavaScript to raise events. Change the implementation of the OnLoaded method to this instead:

    private void LoadJavaScript()
    {
        // For demo purposes, Flatpickr is loaded directly from CDN.
        // Uno uses AMD module loading, so you must give a callback when the resource is loaded.
        // We can access the corresponding DOM HTML Element by using the "element" variable available in the scope
        var javascript = $@"require([""https://cdn.jsdelivr.net/npm/flatpickr""], f => {{
            // Route Flatpickr events following Uno's documentation
            // https://platform.uno/docs/articles/wasm-custom-events.html
            const options = {{
                onChange: (dates, str) => element.dispatchEvent(new CustomEvent(""DateChanged"", {{detail: str}})),
                onOpen: () => element.dispatchEvent(new CustomEvent(""OpenedStateChanged"", {{detail: ""open""}})),
                onClose: () => element.dispatchEvent(new CustomEvent(""OpenedStateChanged"", {{detail: ""closed""}}))
             }};
    
             // Instantiate Flatpickr on the element
             f(element, options);
        }});";
    
        this.ExecuteJavascript(javascript);
    }
    
  4. Compile & Run. Here's the result:

    Final result

Troubleshooting

If your JavaScript integration is not behaving properly, you can troubleshoot with hints below.

My JavaScript control does not accept pointer input

In the constructor of your wrapper control, add the following:

// XAML behavior: a non-null background is required on an element to be "visible to pointers".
// Uno reproduces this behavior, so we must set it here even if we're not using the background.
// Not doing this will lead to a `pointer-events: none` CSS style on the control.
Background = new SolidColorBrush(Colors.Transparent);

TextBlock content is not visible in browsers with the dark theme

TextBlock defaults the text color as White correctly but Page background needs to be set correctly.

<Page 
    ...
    Background="{ThemeResource ApplicationPageBackgroundThemeBrush}">

🔬 Going further

This article illustrates how to integrate external assets (JavaScript and css files) and how to leverage JavaScript's CustomEvent in an Uno application.

More steps could be done to make the code production ready:

  • Make the control multi-platform. Many date/time pickers exist on all platforms. It should be easy on other platforms to connect the same control to another great date picker native to the platform - no need to embed a WebView for this on other platforms.
  • Create script files instead of generating dynamic JavaScript. As in previous article, this would have the advantage of improving performance and increase the ability to debug it.
  • Support more Flatpickr features. There are a lot of features in Flatpickr you can leverage to make a perfect versatile control.