In previous articles, we’ve covered how the Uno Platform takes a visual tree defined in the XAML markup language and creates it on iOS, Android, and WebAssembly. In this article I want to dive into a key intermediate step: how the XAML is parsed and mapped to generated C# code. In part 2, we’ll look at a few other ways in which Uno leverages code generation to make the wheels turn.
Parsing XAML
XAML stands for eXtensible Application Markup Language. It’s an XML-based syntax. Although it can be used to describe pretty much anything, it’s geared toward describing the structure of an application. It was first used in Windows Presentation Foundation, and since then has been used (with minor syntactic differences) in a number of contexts, including Silverlight (RIP), WinRT, Xamarin Forms, and the Universal Windows Platform.
XAML hits a sweet spot: it’s human readable, expressive, and sufficiently structured to support design tools like Blend and XAML Designer. One nice feature is that it can be seamlessly intermingled with C# via code-behind files.
To tap into that power, Uno needs to be able to read a XAML file and convert it into compileable C# code.
The first step is to parse the file into a Xaml object tree, using the System.Xamlnamespace.
Next, the Xaml object tree is processed into a C# class definition. The bulk of the heavy lifting is done in the mammoth XamlFileGenerator class. We lean on the Microsoft.CodeAnalysis API here (aka ‘Roslyn’) to match types used in XAML to types defined in the app assembly or its dependencies. The output is saved to a generated file.
All this happens behind the scenes whenever you build an Uno head.
The end result
Let’s look at a concrete example. This is part of the markup we put in MainPage.xaml
in a previous article:
<StackPanel> <TextBlock x:Name="ClickTextBlock" Text="Button wasn't clicked yet" /> <Button Content="Click me" Click="Button_Click" /> </StackPanel>
The generated output can be found in the ‘obj’ folder under the project head.
The code file generated from MainPage.xaml.
This is a part of the generated code, showing the output for the XAML snippet above:
public sealed partial class MainPage : Windows.UI.Xaml.Controls.Page { private void InitializeComponent() { var nameScope = new global::Windows.UI.Xaml.NameScope(); NameScope.SetNameScope(this, nameScope); // Source ..\..\..\..\..\..\UnoExtTestbed.Shared\MainPage.xaml (Line 1:2) Content = new global::Windows.UI.Xaml.Controls.StackPanel { // Source ..\..\..\..\..\..\UnoExtTestbed.Shared\MainPage.xaml (Line 8:3) Children = { new global::Windows.UI.Xaml.Controls.TextBlock { Name = "ClickTextBlock", Text = "Button wasn't clicked yet", // Source ..\..\..\..\..\..\UnoExtTestbed.Shared\MainPage.xaml (Line 9:4) } .MainPage_05a03e67d6a6581089c5d12899278d61_XamlApply((MainPage_05a03e67d6a6581089c5d12899278d61XamlApplyExtensions.XamlApplyHandler0)(c0 => { nameScope.RegisterName("ClickTextBlock", c0); this.ClickTextBlock = c0; } )) , new global::Windows.UI.Xaml.Controls.Button { Content = @"Click me", // Source ..\..\..\..\..\..\UnoExtTestbed.Shared\MainPage.xaml (Line 11:4) } .MainPage_05a03e67d6a6581089c5d12899278d61_XamlApply((MainPage_05a03e67d6a6581089c5d12899278d61XamlApplyExtensions.XamlApplyHandler1)(c1 => { c1.Click += Button_Click; } )) , } }
What a mouthful! Generated code tends to be optimized for machine readability, rather than for human readability. For example, the explicit casts to XamlApplyHandler
were added when we found that the compiler was having to do a lot of computationally-expensive type inference. That one change doubled the compilation speed for large XAML files.
In the normal course of Uno development you don’t need to look at code generated from XAML or even know that it’s there, though occasionally it’s useful to take a peek when debugging.
Recall that the Button_Click
event is defined in the code-behind for MainPage. As far as the compiler is concerned, both the authored and generated files are just partial definitions of the same class.
One last fun fact: when building on MacOS using Visual Studio for Mac(which Uno tentatively supports in preview), Uno uses an internal port in place of the System.Xaml namespace which isn’t available on Mac. The same code backs Uno’s support for runtime Xaml interpretation via Windows.UI.Xaml.Markup.XamlReader, which is used in the interactive modeof the Uno Gallery app and Uno.Playground website.
The post Talkin’ ‘bout my generation: How the Uno Platform generates code, part 1 — Under The hood appeared first on Uno Platform.
Top comments (0)