New feature : Drag and drop assets on your custom controls

Hello,

In Unity 2023.1.0a19, we have added the ability to add references to assets in the properties of your custom controls. This effectively means that you can drag references to Textures, Sprites or any kind of asset into the UI Builder Attributes inspector.

Intended usage

To use this feature, create custom controls in C# and declare properties of type UxmlAssetAttributeDescription.
The UI Builder will pick this up to display an object field and save the appropriate path in the UXML file.

In your UXML, the path to the asset is stored in the exact same way as references to UXML or USS files. The path is resolved at import time and captures a direct dependency to the imported asset referenced by this path.

This means that you do not need to rely on Resources folders to locate assets in your custom controls and extensions.

Until UI Toolkit gains the ability to save and author custom data structures, you can use this feature to leverage Unity’s existing capabilities for custom extensions around your UI Toolkit workflow. For example, you can create custom Scriptable Objects and use Unity’s serialization feature to save custom structures, and add custom Inspectors and Drawers to edit the data.

Example

As a quick proof of concept, we can revisit the idea of a custom element to display gradients from the Unity Manual.

First of all, define a custom Scriptable Object type, GradientDefinition that holds a UnityEngine.Gradient property:

using UnityEngine;

namespace UIToolkitExamples
{
    [CreateAssetMenu(fileName = "GradientDefinition", menuName = "GradientDefinition", order = 0)]
    public class GradientDefinition : ScriptableObject
    {
        public Gradient gradient;

        public void Reset()
        {
            gradient = new Gradient();
        }
    }
}

Then, define a custom controls that accepts a GradientDefinition reference, alongside the necessary setup for receiving a reference from UXML/UI Builder.

using UnityEngine;
using UnityEngine.UIElements;

namespace UIToolkitExamples
{
    public class ExampleElementCustomAsset : VisualElement
    {
        // Factory class, required to expose this custom control to UXML
        public new class UxmlFactory : UxmlFactory<ExampleElementCustomAsset, UxmlTraits> { }

        // Traits class
        public new class UxmlTraits : VisualElement.UxmlTraits
        {
            public UxmlAssetAttributeDescription<GradientDefinition> m_Gradient = new() { name = "gradient" };

            public override void Init(VisualElement ve, IUxmlAttributes bag, CreationContext cc)
            {
                base.Init(ve, bag, cc);

                if (m_Gradient.TryGetValueFromBag(bag, cc, out GradientDefinition value))
                {
                    ((ExampleElementCustomAsset)ve).gradient = value;
                }
            }
        }

        private GradientDefinition m_Gradient;

        public GradientDefinition gradient
        {
            get => m_Gradient;
            set
            {
                if (m_Gradient == value)
                    return;
                m_Gradient = value;
                GenerateGradient();
            }
        }

        // Image child element and its texture
        Texture2D m_Texture2D;
        Image m_Image;

        public ExampleElementCustomAsset()
        {
            // Create an Image and a texture for it. Attach Image to self.
            m_Texture2D = new Texture2D(100, 100);
            m_Image = new Image();
            m_Image.image = m_Texture2D;
            Add(m_Image);
        }

        void GenerateGradient()
        {
            if (m_Gradient == null)
                return;

            for (int i = 0; i < m_Texture2D.width; ++i)
            {
                Color color = m_Gradient.gradient.Evaluate(i / (float)m_Texture2D.width);
                for (int j = 0; j < m_Texture2D.height; ++j)
                {
                    m_Texture2D.SetPixel(i, j, color);
                }
            }

            m_Texture2D.Apply();
            m_Image.MarkDirtyRepaint();
        }
    }
}

After creating an instance of GradientDefinition in the project, can edit its gradient property in the regular Inspector (Unity provides a default Editor for the Gradient property).
8597401--1153099--Screen Shot 2022-11-18 at 3.53.27 PM.png

After we put an instance of ExampleElementCustomAsset in the UI Builder, we can edit its gradient property through an object field in its Inspector to reference the instance of GradientDefinition.

Which gets saved in the UXML as follows:

<ui:UXML xmlns:ui="UnityEngine.UIElements" xmlns:uie="UnityEditor.UIElements" editor-extension-mode="False">
    <UIToolkitExamples.ExampleElementCustomAsset gradient="project://database/Assets/MyGradient.asset?fileID=11400000&guid=2b8b0b4d063cf4e5b974fef361238fb6&type=2#MyGradient" />
</ui:UXML>

Behind the scenes

We’ve added this feature because it’s a necessary building block for the data binding workflow that is in development.

The main challenge of the UxmlAssetAttributeDescription implementation is to detect the attributes of custom elements in order to correctly interpret asset paths and provide useful warning and errors when paths are broken.

This means that changes in C# may cause re-imports of UXML file to keep everything consistent (if attributes are added or removed, we need to re-evaluate if attributes are to be interpreted as paths).
You might be might interested to learn that this is partly made possible by the Custom Dependency feature of the Asset Database.

Conclusion

We hope that you will find this feature useful and encourage you to try it out. Feel free to post any question or feedback in this thread. Thanks!

12 Likes

Neat!

How does this serialization handle the case where you move the file-path of the asset that is depended on? Does that mean that moving a file may cause UXML assets to change? What happens if you move the asset, but don’t update the UXML file with the new path?

I’m mostly thinking about this because there aren’t any other Unity systems that depend on asset path for referencing. Why not just serialize the guid+fileid?

This is handled the same ways other assets are referenced from UXML and USS. UI Builder stores both the path and the guid+fileID in the same URI. This means that when files are moved, the reference will continue to resolve but a warning will be given in the console that the URI needs to be updated (The UI Builder will override to the correct path the next time it saves the file).

Here is an example of the error given after moving the file:

Assets/ExampleCustomAssetUsage.uxml (3,6): Semantic - The specified URI does not exist in the current project : Asset reference to GUID '2b8b0b4d063cf4e5b974fef361238fb6' was moved from 'Assets/MyGradient.asset' to 'Assets/Common/MyGradient.asset'. Update the URL 'project://database/Assets/MyGradient.asset?fileID=11400000&guid=2b8b0b4d063cf4e5b974fef361238fb6&type=2#MyGradient' to remove this warning.

This is not currently documented, nor do we have great support for making file moves seamless, but any future improvements over the current state will apply to usage of this attribute as well.

Is this planned for release in 2022.x at all, or it it exclusively for 2023.x?

1 Like

No plans to backport currently

Understand this is a feature but if it is at all considered, please take this post as a +1 for 2022 LTS - would be lovely not to use our current by-string hack.

1 Like

I’m assuming stringDefaultValue is read only on purpose, is there a way to set a default asset or address via code?

Another +1 for backporting

If you have no plans on doing the backport yourself then may I ask for some guidance on how we could achieve this on our own.

In general the UI Builder is much less extendable than the normal inspector. It would help so much if we could add custom UI and custom (persistent!) xml attributes. I am hoping for some sort of XMLAttribute class with an “OnInspectorGUI” like method that allows us to draw into a VisualElement container right below the normal “UxmlAttributeDescription” attributes. Together with an overridable serializeToString / deserializeFromString method this would give a lot of flexibility.

I am currently making some assets for the UI Toolkit as I think it has a bright future. Compared to the normal inspector extending the UI Builder is VERY cumbersome (or downright impossible). I already need a lot of reflection code to get even the simplest things done, like query what UI Element is currently selected.

Please let us add to your standard UI. It’s what made the old inspector great and it could make the UI Builder great too. And please backport these at least to 2022. I know the official stance is that UITK is not yet ready for runtime [1], but people are using it already.

Thank you for considering

1 Like

Backporting to 22.3 LTS would be so great ! It would be nice to consider it, as it is such a basic feature to reference asset in Unity. And there is no way to update to a non-LTS version right now …

Edit : UxmlAssetAttributeDescription is available in 22.3 but it is internal …

1 Like

Will this work for runtime? I am creating custom controls wherein I need to change the image of each when I use them in the game UI. Can I do that using this code? If not, how?

Thanks!

I am not entirely sure I understand your question.

UXML assets that receive references to other Unity assets will load correctly during runtime because the paths are resolved when the UXML is imported. As long as the custom element C# class is available in the runtime build, everything will work OK.

If you are asking how to achieve something similar without UXML, then this is nothing different than using a regular property on a C# class, just like it would be with MonoBehaviours.

If you have a list of images (Texture2D) to assign to a list of visual elements, just write C# to iterate through the lists and add the image to the style.backgroundImage property, or add a custom Texture2D property in a custom C# visual element class.

My apologies for not being clearer. I want to create custom controls and instantiate them dynamically at runtime. For example, buttons in a list box wherein the name depends on player level. Can I change the display name of each button in a list box at runtime - that is during gameplay?

Given that UXMLTraits is deprecated and most of this stuff has moved to an attribute style system, how does this work now? I’ve tried everything I can think of with no success. I want to include a property to allow the user to set an image similar to how for example the Button control let’s you set an Icon image.

It’s very simple. This page explains it: https://docs.unity3d.com/2023.3/Documentation/Manual/UIE-custom-tag-name-and-attributes.html

Yeah this is the new system and works perfect for me for simple things like a string or a float but not for textures of other object type things like the gradient example.

EDIT: Ok apparently this does work on Texture. I must have just been super tired last night working past midnight and done something stupid :frowning:

1 Like

I am interested in this as well. Maybe writing a texture attribute entirely from scratch is possible?
By using an asmdef I can put my custom control into a namespace which has internal access to UxmlAssetAttributeDescription and it does show up …9698300--1384352--upload_2024-3-13_11-48-57.png

The reason it wasn’t made public in this version is that, in the event that your code definition for the custom element changes (for example, add or remove an UxmlAssetAttributeDescription to the traits), affected UXML files do not get re-imported. So you may have an out of date import result until the next import of the UXML.

It might not be an issue for some users but it wouldn’t be solid enough for general availability.

1 Like

@antoine-unity

Would it work to write a UxmlTexture2dAttributeDescription : TypedUxmlAttributeDescription?
I tried and it showed up as a string instead not an asset reference onto which I could drop an asset or is the UxmlAssetAttributeDescription the only way to get this (because the whole hack with the asmdef isn’t that flexible usually)?

No it won’t work, TypedUxmlAttributeDescription is only meant to convert from something that can be stored as a raw string in UXML, and UI Builder only handles a specific set of subclasses.

1 Like

Hi Bro, Can you help to sharing about that?