Houdini for TD - HDA & Operators

Note

This article is part of the serie Houdini for TD

Working with HDA

HDA (Houdini Digital Asset), which used to be name OTL (Orbolt Tool Library), are usually tools built to streamline workflows. They allow to wrap up large network into a single node with an artist-friendly interface.

As TD, HDA are one of the main way to provide tools for the Artist to interact with as they can integrate smoothly in their work.

HDA Creation is covered in many tutorial such as SideFX Introduction to Digital Assets.

Note

I invite you to read this article about HDA Best Practice, which provide general recommendation to improve the HDA's quality.

Special HDA

It is most common to create HDA from a subnetwork and click Create Digital Assets... This mechanic is great for creating tools that are based on Houdini nodes, but other types of HDA can be created.

In the Main menu, go in File > New Asset... , a new dialog will open asking you the same basic questions about a name and label for your HDA and where to save it. But you can notice you have the ability to select a different "Definition" type. The most common type of definition is "Subnetwork", but other most useful one could be Python and Output Driver.

Those HDA behavior can be customized by editing the Code Section when editing them.

You might also notice that those nodes cannot be "Unlocked and edit..."

HDA versioning

Node can only have one active NodeType and NodeType can only have one active HDA Definition.

flowchart LR def_a[HDA Definition A] def_b[HDA Definition B] def_c[HDA Definition C] type_a[NodeType A] type_b[NodeType B] node[Node] def_a & def_b & def_c type_a & type_b def_a --> type_a def_b -.-> type_a def_c --> type_b type_a --> node type_b -.-> node

Following the above graph, there is usually two way of versionning your HDA, with both pro and cons. Either by switching:

  • NodeType
  • HDA Definition

Node Type

NodeType switching is done by calling:

hou.Node.changeNodeType(new_node_type)

A common practice is to use Digital Assets versions and namespace.

When creating a node by just using the basename type, Houdini will use the namespace preference order and latest version to resolve which node type is the most appropriate to create.

You can still provide the full NodeType name if you want a specific version.

The key advantage of this solution is that it allow more granular control as you can update each node individually. So you can have a node at a version A, but the other at a version B. This mechanic can be used for Asset-like behavior.

Note

You can use the following function to easily retrieve the NodeType from a basename when using Houdini Versions and Namespace:

hou.preferredNodeType("Category/basename")

HDA Definition

While in the second case, you can update version by marking the definition included in the hda file as preferred:

hou.HDADefinition.setPreferred(True)

Updating the definition will automaticly update all the instances of that specific node type. This is specificly useful for Pipeline-like HDA.

Embedded HDA

Often overlooked, Embedded HDA is a useful alternative to storing files on disk as it will store your HDA in the hipfile. Storing an HDA as Embedded can be useful in some cases.

If you are planning to share your file with someone who is not the same network as you or won't have access to all your HDA files, embedding the HDA is a really nice and conveniant way to package your scene.

HDA files when saved, depending on your configuration, should create backup file, though, those changes are only stored when the HDA is being written to disk. In development stages, by storing an HDA in your scene, you can use the scene file save mechanic to create clever backup of your HDA development.

Note: When you open a hipfile but the HDA source file is unavailable, Houdini will automaticly create a dummy embedded definition. This way, if you accidently save your scene, Houdini will still be aware of the original NodeType the Node was.

HDA Expanded structure

When you save your HDA, they are by default saved into a binary file which is easy for houdini to open. Though, if the HDA is part of a package it can be really hard to track changes on it. An option is to expand the HDA using the following commands:

# To expand an HDA into a VCS friendly folder
hotl -t expanded.hda original.hda

# To collapse a VCS expanded HDA into a binary HDA
hotl -l expanded.hda collapsed.hda

Once expanded a HDA should be looking like this with more or less files depending on the number of Sections and scripts that you may have defined in your HDA:

  • expanded.hda - This is the main folder under which your HDA is expanded. If it uses the .hda extension, you can load it straight away in Houdini
    • houdini.hdalibrary - Just here to say this is a hdalibrary
    • INDEX_SECTION - Specify some basic information regarding the declared operator such it operator name, operator label, context, number of input and output, etc...
    • Sections.list - List files in this folder and how they should be evaluated by houdini
    • Sop_1hda_name - This is the folder under which you HDADefinition is expanded
      • Contents.dir - (Optional) This folder represent the content inside your HDA
        • Contents.createtimes - When were each children nodes created
        • Contents.houdini_versions - Which version of houdini was used to create each children nodes
        • Contents.mime - This file define most properties of the children nodes such as how they are connected, the values on the parameters, placement in the HDA.
        • Contents.modtimes - When were each children nodes last modified
        • Sections.list - List the files in this folder and how they should be evaluated by houdini
      • CreateScript - Automaticly created script defining how to create the HDA
      • DialogScript - This file define the ParmTemplateGroup on the HDA, with all the information relative to the list of parameters, default values and parameter behavior, etc...
      • ExtraFileOptions - Houdini internal file used to remember where was your cusor when editing files within the HDA Editor dialog
      • Help - (Optional) Help file for your HDA
      • InternalFileOptions
      • PythonCook - (Optional) This is the content to be evaluated when the node is used to programaticly generate contents.
      • PythonModule - (Optional) Whenever you are creating a PythonModule, the content is saved in this file.
      • Sections.list - List the files in this folder and how they should be evaluated by houdini
      • Tools.shelf - A file very similar to a "Shelf Tool", because it is one! This is the tool that is called when you ask to create the node
      • TypePropertyOptions - Various properties about the HDA

Expanded HDA are slower for Houdini to load as it need to compile it. Though, for development, it might be interresting to work with the expanded version as you can edit those files from an external editor, and changes will be reflected in Houdini.

Python Panel instead of Parameters

It is now possible to interact with an operator by using a PythonPanel (Qt UI) instead of the regular Parameter view. For doing so, you first need to create a new Panel.

<xml version="1.0" encoding="UTF-8">
<pythonPanelDocument>
    <interface name="my_ui" label="My UI" icon="MISC_python" shownetworkNavigationBar="true" help_url="">
        <script><![CDATA[
from hutil.Qt import QtWidgets


def onCreateInterface():
    widget = QtWidgets.QLabel("Hello World")
    return widget
        ]]></script>
        <help><![CDATA[]]></help>

        <!-- This is the part where you define all the operators that your UI can be used from. -->
        <showInParametersPane optype="Sop/hda_name"/>

    </interface>
</pythonPanelDocument>

Then when you select your operator, you should have an addional icon on the left of gear parm icon (looking like 2 slider). Click on it to enable/disable the python panel view.

Advanced HDA Functionality

Warning

This section still need to be written. Please come back later.

Managing operators

Over a HDA lifecycle, it's integration with Houdini might require you to define how it integrate with existing tools. For example, you might want to give preference to your newly created caching HDA over the default Houdini filecache.

Here are couple way to help integrate your tools with existing HDA:

Hide an operator

This method can be used to avoid an operator to be use. This can be the case with a deprecated tool. By hidding the operator, it won't be showing as an available operator in the UI. Though, the operator still remain install, which allow previously created node to still be working properly.

To hide an operator, you should use:

hou.NodeType.setHidden(True)

Uninstall an operator

Important

This last method will only work with HDA and can cause some issues in the user scene. Be careful when using it!

By uninstalling the HDA, the operator will fallback to the next available HDA Definition if the uninstalled one was current. In the case no other definition are available, a warning will be displayed to the user.

To fully uninstall an operator, you can use the following script:

# Finally uninstall all definition
for definition in node_type.allAvailableDefinition():
    # Handle the cases where multiple definition might be defined in a file
    for file_definition in hou.hda.definitionsInFile():
        # No problem if current definition is not current
        if not file_definition.isCurrent():
            continue

        # Check if the definition is being used by any node in the scene
        elif file_definition.nodeType().instances():
            raise RuntimeError

    # Finally uninstall the file
    hou.hda.uninstallFile(definition.libraryFilePath())

Create operator aliases

Adding an alias to an HDA can be a good solution when some of your tools are unavailable, but they are still invoked in the scene. Alternativly, aliases can be use to provide easy way for users to create some of your tool.

Aliases can be used in code, but are also visible in the user interface.

Let's take the "box" example in houdini to show how to create and remove an alias:

node_type = hou.nodeType("Sop/box")
node_type.addAlias("cube")

hou.node("/obj").createNode("geo").createNode("cube")

node_type.removeAlias("cube")

HDA & Startup time

When working with Houdini Digital Assets (HDAs), you might consider how they impact loading time.

Expanded vs. Compressed

One of the most significant factors affecting loading time is the format in which HDAs are loaded. Loading HDAs in their expanded format can take up to 3-4 times longer than loading their binary counterparts.

Library File Organization

Another factor to consider is how you organize your HDAs within library files. While it might seem intuitive to store all HDAs in a single library file, the difference in loading time between this approach and storing them in multiple files is negligible. In fact, loading multiple files only takes around 5% more time than loading a single file.

Loading Methods

The method used to load HDAs has a minimal impact on loading time. Whether you load HDAs via environment variables, Houdini packages, or the Python API, the difference in loading time is not noticeable.

Methods Collapse (Single) Expanded (Single) Collapse (Multi) Expanded (Multi)
HOUDINI_PATH        
Houdini Package        
hou.hda.installFile(...)