References to external assets (TextMesh Pro) in asset

I’m creating an asset involving UI text and my scripts contain references to classes in the TMPro namespace.
An important distinction in my case is that my asset doesn’t require TextMesh Pro, it’s only being used for the prettier text, which I would like to be the default.
Is this a bad idea? Should I reference the respective classes in UnityEngine.UI instead?
Or is there a convenient way of auto-detecting the existence of TMPro and changing the scripts accordingly?
Or some simple import-pop-up that shows up when importing the asset, giving you the choice between the two?

It would be nice to avoid two versions of the scripts that diverge, cuz maintenance pains.
The references are literally interchangeable, you could do a Find and Replace between the TMPro and UnityEngine.UI classes.

Any bright ideas? Am I missing something obvious?

It’s not obvious. Use compiler conditions to conditionally compile your TMP code only if TMP is present. TMP used to automatically define a scripting symbol TMP_PRESENT, but it no longer does that. You can define it on your own using PlayerSettings.SetScriptingDefineSymbolsForGroup.

You will probably want to write an editor script with a method that uses UnityEditor.Callbacks.DidReloadScripts. In this method, call PackageManager.Client.List to list what packages are installed. If TMP is installed, set the scripting symbol.

You could also write an import script that checks once if TMP is installed. If it isn’t, ask the user if they want to install it. If so, use PackageManager.Client.Add to install it.

2 Likes

Thanks for the response! I was finally able to sit down with it for a while and your response gave me a great place to start.
I was able to get it working with this:

using UnityEditor.PackageManager;
sealed class CheckForTextMeshPro {
    const string tmpDefine = "TMP_PRESENT";
  
    [UnityEditor.Callbacks.DidReloadScripts]
    static void CheckForTMPPackage(){
        var asyncRequest = Client.List(true);
        while (!asyncRequest.IsCompleted){} //Idle loop
      
        foreach (var package in asyncRequest.Result){
            if (package.displayName == "TextMesh Pro"){
                if (package.status == PackageStatus.Available)
                    DefineUtilities.AddDefine(tmpDefine, DefineUtilities.GetValidBuildTargets());
                else
                    DefineUtilities.RemoveDefine(tmpDefine, DefineUtilities.GetValidBuildTargets());
                return;
            }
        }
    }
}

But after milling it over a bit I figured that this solution wasn’t greatly elegant and that it wouldn’t work with Unity versions older than 2018, so I figured: why not just check for the existence of a namespace?

sealed class CheckForTextMeshPro {
    const string tmpDefine = "TMP_PRESENT";

    [UnityEditor.Callbacks.DidReloadScripts]
    static void CheckForTMPro(){
        var namespaceFound = (from assembly in AppDomain.CurrentDomain.GetAssemblies()
            from type in assembly.GetTypes()
            where type.Namespace == "TMPro"
            select type).Any();

        if (namespaceFound)
            DefineUtilities.AddDefine(tmpDefine, DefineUtilities.GetValidBuildTargets());
        else
            DefineUtilities.RemoveDefine(tmpDefine, DefineUtilities.GetValidBuildTargets());
    }
}

This works like a charm and is significantly faster (4ms vs 250-500ms), which might be relevant since it runs every script reload. Also might be worth doing an “if (EditorApplication.isPlayingOrWillChangePlaymode) return;” at the start of the method since reload also happens when entering play mode.

And let’s of course not forget to include my DefineUtilities class:

public static class DefineUtilities {
    /// <summary>
    /// ScriptingDefineSymbols are separated into a collection by any of these,
    /// but always written back using index 0.
    /// </summary>
    public static char[] separators = { ';', ' ' };

    public static void AddDefine(string _define, IEnumerable<BuildTargetGroup> _buildTargets){
        foreach (var target in _buildTargets){
            var defines = PlayerSettings.GetScriptingDefineSymbolsForGroup(target).Trim();

            var list = defines.Split(separators)
                .Where(x => !string.IsNullOrEmpty(x))
                .ToList();

            if (list.Contains(_define))
                continue;

            list.Add(_define);
            defines = list.Aggregate((a, b) => a + separators[0] + b);
            PlayerSettings.SetScriptingDefineSymbolsForGroup(target, defines);
        }
    }

    public static void RemoveDefine(string _define, IEnumerable<BuildTargetGroup> _buildTargets){
        foreach (var target in _buildTargets){
            var defines = PlayerSettings.GetScriptingDefineSymbolsForGroup(target).Trim();

            var list = defines.Split(separators)
                .Where(x => !string.IsNullOrEmpty(x))
                .ToList();

            if (!list.Remove(_define)) //If not in list then no changes needed
                continue;

            defines = list.Aggregate((a, b) => a + separators[0] + b);
            PlayerSettings.SetScriptingDefineSymbolsForGroup(target, defines);
        }
    }

    public static IEnumerable<BuildTargetGroup> GetValidBuildTargets(){
        return Enum.GetValues(typeof(BuildTargetGroup))
            .Cast<BuildTargetGroup>()
            .Where(x => x != BuildTargetGroup.Unknown)
            .Where(x => !IsObsolete(x));
    }

    public static bool IsObsolete(BuildTargetGroup group){
        var obsoleteAttributes = typeof(BuildTargetGroup)
            .GetField(group.ToString())
            .GetCustomAttributes(typeof(ObsoleteAttribute), false);

        return obsoleteAttributes != null && obsoleteAttributes.Length > 0;
    }
}

For includes, these cover all the classes:

using System;
using System.Linq;
using System.Collections.Generic;
using UnityEditor;

On that topic, you wouldn’t happen to know the difference between DidReloadScripts and InitializeOnLoadMethod? From my testing I have been unable to tell any difference between the two, and the cryptic info in the docs isn’t helping much.

Been googling for this, but can’t seem to find any info on how to write ‘import’ scripts. Are you referring to some kind of AssetPostprocessor or ScriptedImporter? Because those don’t seem quite like they would be useful for this sort of thing.

In almost all cases, they both get called. I think when you first start the Unity editor, if there are compiler errors then InitializeOnLoad gets called but DidReloadScripts doesn’t. It’s been a while since I played with that, but it would be an easy experiment to run.

Use PackageManager.Client.Add.

Tried it and neither was called.

Oooooh, that’s what you meant. I thought you meant using some kind of script that executes on package import, not a script that does the import :face_with_spiral_eyes:

If I were to construct such a check to only run once, where would be the best place to store the fact that it has been checked? Playerprefs is easy, but might also be wiped, re-triggering the popup. Data in an asset file would work, but I’d prefer to avoid making changes to assets.

I use EditorPrefs. And if it gets wiped, it does the check once, silently notes that no change is needed, and sets EditorPrefs again.

1 Like

Took another look at DidReloadScripts, InitializeOnLoad and InitializeOnLoadMethod and found one difference:
InitializeOnLoad executes before InitializeOnLoadMethod, which in turn executes before DidReloadScripts, and DidReloadScripts can further specify order relative to others with the same attribute through an input.
I.e: [UnityEditor.Callbacks.DidReloadScripts(-1)] executes before [UnityEditor.Callbacks.DidReloadScripts(0)], which in turn executes before [UnityEditor.Callbacks.DidReloadScripts(1)] ( [UnityEditor.Callbacks.DidReloadScripts] defaults to 0), but regardless of the number it always executes after InitializeOnLoadMethod and InitializeOnLoad.

Summarized execution order:
[InitializeOnLoad][InitializeOnLoadMethod][UnityEditor.Callbacks.DidReloadScripts(-1)] → [UnityEditor.Callbacks.DidReloadScripts] → [UnityEditor.Callbacks.DidReloadScripts(1)]

1 Like