Theme resource internals for contributors
This article looks at the way theme resources are handled under the hood, including the way that resource assignations are resolved, and the way that assigned resources are updated, for example when the active theme changes in the running application (from light to dark or vice versa). It's aimed at Uno Platform contributors looking to better understand the internals of the feature.
From hereon, an understanding of the contract and consumer-visible behavior of XAML resources and ThemeResource markup extensions is assumed. Review those links if you need a refresher on how the feature is expected to behave.
Resource resolution
When a theme resource or static resource assignation is resolved, the framework looks through all ResourceDictionaries that are in scope from the point of assignation, in a defined order, for a resource with a matching key.
Priority of ResourceDictionaries
Scope and priority are determined by the following rules:
- if the
{ThemeResource}
assignation is itself inside a dictionary, other than inside a template, then the containing dictionary is checked first. - if the assignation is on a
frameworkElement
or the logical child of aframeworkElement
, and theframeworkElement
is in the visual tree, then theResources
of theframeworkElement
and its visual ancestors are checked in order, outwards from theframeworkElement
. This typically occurs when the element is first loaded into the visual tree, or when the active theme changes (see below). - the
Application.Resources
dictionary is checked next. - if the XAML containing the assignation comes from an external assembly (ie a class library), then next the dictionaries within that assembly are checked. (Note that Uno's behavior departs somewhat from UWP here, because only the
Themes/Generic.xaml
dictionary from that assembly should be checked - see issue #4424.) - finally, 'system level' resources are checked. This simply means the resources that are defined within the Uno.UI assembly.
Searching within a ResourceDictionary
Resolving a resource within a ResourceDictionary
is a somewhat complex operation in its own right, because a ResourceDictionary
may contain nested dictionaries within its MergedDictionaries
and ThemeDictionaries
collection properties, those dictionaries may in turn contain dictionaries, and so on recursively. This can be thought of as a 'resource tree' that is traversed in order to find a match for a given resource key.
The matching resource is determined according to the following priority:
- the key-value pairs directly added to the
ResourceDictionary
are considered first. - ResourceDictionaries in the
MergedDictionaries
collection are searched, in reverse order, in other words the last dictionary added has first priority. - The
ResourceDictionary
inThemeDictionaries
whose key matches the active theme will be searched. This will either be the dictionary whose key exactly matches the active theme (eg "Dark" key when dark theme is active), or the "Default"-keyed dictionary if it exists and no exact match is found. Note that if an exact-match dictionary is found, the "Default" dictionary will be ignored, even if the resource key is not found within the exact-match dictionary. - finally, if the search was initiated from consumer code, eg via indexing or the public
TryGetValue()
method, then system level resources will be checked, as defined above.
Searching for a resource is a very frequently performed operation and a bottleneck during page loading, and as such is heavily optimized within ResourceDictionary
, via caching, dedicated internal resource keys, and other mechanisms. It may also be optimized 'outside' ResourceDictionary
by using a tool to create a single flat merged dictionary out of multiple top-level ControlName.xaml files, which is faster to search (O(1)
) than many nested MergedDictionaries.
Resource updates
The resource update machinery can be broadly split into two parts: registering resource assignations for updates; and propagating updates to assigned values when the active theme changes.
Registering assignations for updates
ThemeResource assignments can only be created in XAML - WinUI provides no way to create them in code. Thus, in Uno, all assignations come ultimately from either XamlFileGenerator or XamlReader/XamlObjectBuilder.
There are 3 cases for ThemeResource assignations that are handled differently:
- Assigning an ordinary dependency property to a ThemeResource reference.
- Assigning an ordinary non-dependency property to a ThemeResource reference.
- Assigning the Value property of a Setter to a ThemeResource reference.
In case 1., we create a ResourceBinding
, an internal Uno-only type that inherits from BindingBase
, that binds the property to the ThemeResource, allowing its value to be subsequently reevaluated when the owner enters the live visual tree and also when the active theme changes. This is done through the ResourceResolver
static class. As much as possible, XAML-generated code should use helper intermediaries that will not change when the underlying implementation evolves. XAML-generated code compiled with older versions of Uno may run against newer Uno versions, in the case of 3rd-party libraries, and we don't want to introduce breaking changes that could be avoided.
In case 2. we can't create a binding because we don't have a dependency property to bind to. The best we can do in this case is to do a one-time resolution of the value.
In case 3., the details of the ThemeResource reference are captured on the Setter
(via the ApplyThemeResourceUpdateValues()
extension method). When the setter is later applied, either within a Style
or within a VisualState
, then the ThemeResource assignation will be applied according to either case 1. or case 2. depending on whether the target property is a DependencyProperty or not.
In cases 1. and 3., at the time of registration we capture the 'context'. This is an instance of XamlParseContext
, which captures the assembly in which the ThemeResource reference is declared, and is used to determine the scope which should be used when resolving it.
Propagating updates on theme change
A theme change update is propagated whenever the active theme changes. This may be because a change in the system theme was detected (eg the user enabled dark mode in their system settings, or perhaps it changed automatically based on the time of day), or because the RequestedTheme
property on the root element of the visual tree was programmatically changed (eg to support an in-app light/dark mode toggle).
In either case, the Application.OnResourcesChanged()
method will be called. It will:
- Update all items in the tree of ResourceDictionaries rooted by
Application.Resources
. - Update all items in system resources, ie ResourceDictionaries defined within the Uno.UI assembly.
- Update all items within the active visual tree.
Cases 1. and 2. are functionally identical, operating on different root dictionaries.
Note that changing the active theme is something that we expect to happen relatively rarely during any given app session, and usually due to explicit user input. The relevant code nonetheless sits alongside and interacts with code paths that are very heavily used during any app session: loading controls, applying bindings, resolving resources etc, which all have an important impact on perceived performance. As much as possible, the theme change machinery has been and should be architected to place its 'performance burden' upon the update itself, rather than the code supporting the update which is called frequently during ordinary activities of the app.
Updating resource bindings
Resource bindings are managed by DependencyObjectStore
. It checks if any properties of its parent instance are resource-bound, by checking if it has a non-empty resource bindings list.
If so, for each binding, it tries to re-resolve the correct theme value from the ResourceDictionaries currently in its scope. If it can, it will then set the value, with the correct precedence as captured when the binding was created.
The DependencyObjectStore
will also propagate the resource binding update to any 'logical children' that are also DependencyObjects, but not FrameworkElements. (Child FrameworkElements will be updated during the traversal of the visual tree.) For performance reasons, the DependencyObjectStore
does not explicitly maintain a list of logical children. Instead, the InnerUpdateChildResourceBindings()
method looks at all current property values and pushes the update to those that are either DependencyObjects or collections of DependencyObjects. In case this is inadequate, it also tries to cast to the IAdditionalChildrenProvider
internal interface, which allows a type to explicitly specify its logical child dependency objects.
Update other FrameworkElement properties
The FrameworkElement.UpdateThemeBindings()
method is virtual. In addition to updating resource bindings, some controls override it to perform specific updates required when the theme changes. For example, TextBlock
and Control
set the default-precedence value of Foreground to the appropriate one for the current theme.
Updating 'theme-change-aware' entities
Some non-FrameworkElement types may also need to react to the theme change. This is supported by the IThemeChangeAware
internal interface, whose OnThemeChanged()
method is invoked after resource bindings are applied. For example, Storyboard
implements this interface in order to re-apply long-running animations with the correct updated values.
Hot reload updates
The ResourceBinding machinery is reused for supporting XAML Hot Reload on standalone ResourceDictionaries. When an application is run in debug with XAML Hot Reload enabled, ResourceBindings are created for all resources assignations, including StaticResource assignations. When a standalone ResourceDictionary XAML file is edited, the file is re-parsed and then an update is triggered in much the same way as if the active theme had changed, so that any modified resources will be reapplied.
To support these different use cases whilst preserving correct semantics, resource bindings have a ResourceUpdateReason
flag associated them. This ensures that static resources are not updated when they shouldn't be while hot reload is operating.