Developing Custom Catalogs

In Shift it is possible to create customized catalogs with operators tailored for specific use cases and softwares.

Writing a Catalog

Custom catalogs are written in Python and include a global variable called catalog. This variable is a Python dictionary containing the information for the catalog, such as a description, version and authors. Additionally, this dictionary will define the list of operator classes included in the catalog.

Catalog Dictionary Example

catalog = {"Description": "This catalog includes fundamental Python type operators.",
          "Version": "1.0.0",
          "Author": "Shift's development team", 
          "Operators":[[String,  []],
                        [Integer, []],
                        [Float,   []],
                        [Bool,    []]]}
NOTE

The Operators list from the catalog is a list of lists containing two elements. The first one corresponds to the operator class, and the second one is a list of compatible hosts for the operator. If no host is specified, the node will be usable in any context. For instance, if the second element is ["maya", "houdini"] the operator will be only available for those DCC's.

There are two ways of adding a custom catalog to Shift:

  • As a User Catalog: Use the Catalog Manager to source the catalog file and add it to Shift. This catalog is saved in the user preferences.
  • As an Environment Catalog: Add the path to the directory containing the catalog file to the SHIFT_CATALOG_PATH environment variable. All files in this path containing a catalog object will be identified as custom catalogs and loaded by Shift.
WARNING

It is strongly suggested to avoid naming custom catalogs after existing Python modules or packages, as this may cause namespace clashes. It is recommended to add a prefix or suffix to custom catalog names for unique identification of the catalog module.

Operator Syntax

All Shift operators inherit from the SOperator Shift's Python class. To write a custom operator, the __init__ and execute methods must be overwritten.

Here is an example of an operator class syntax:

from shift.core.workflow import SOperator
from shift.core.workflow import SPlug
from shift.core.constants import SType
from shift.core.constants import SDirection

class MyOperator(SOperator):
    """ Operator's description.
  
    """
    def __init__(self, code, parent=None):

        super(self.__class__, self).__init__(code, parent)

        inPlug1 = SPlug(code="input1",
                      value=0,
                      type=SType.kInt,
                      direction=SDirection.kIn,
                      parent=self)

        inPlug2 = SPlug(code="input2",
                      value=0,
                      type=SType.kInt,
                      direction=SDirection.kIn,
                      parent=self)

        outPlug = SPlug(code="output",
                      value=0,
                      type=SType.kInt,
                      direction=SDirection.kOut,
                      parent=self)

        self.addPlug(inPlug1)
        self.addPlug(inPlug2)
        self.addPlug(outPlug)

    def execute(self, force=False):

        input1 = self.getPlug("input1", SDirection.kIn).value()
        input2 = self.getPlug("input2", SDirection.kOut).value()

        output = input1 * input2

        self.getPlug("output").setValue(output)
        super(self.__class__, self).execute(force)

The SOperator constructor method takes care of initializing the plug objects and adding them to the operator. Each plug must be initialized specifying:

  • A unique code name, which will correspond to the name displayed in the UI.
  • A value, which will correspond to the value the plug is initialized with.
  • A type, which will define the behaviour of the plug when translating the provided inputs to a Python object. Check the SType class in Shift's API for more information.
  • A direction which can be either SDirection.kIn or SDirection.kOut.
NOTE

The docstring provided for the SOperator class will be use as the operator's description when inspecting a node's information. It is recommended to write a detailed description of the operator's behavior alongside the required inputs and outputs.

The execute method takes care of defining the list of steps to be performed when the execution of the operator is issued inside Shift. In the example above, the execute method picks up the values of the two input plugs and multiplies them together. Finally, it stores the result in the output plug.

Please refer to the Shift API documentation to know more about which plug types are present in Shift, and which additional methods the SPlug and SOperator classes offer.

Forcing Operators to Run on the Main Thread

Certain contexts and applications require specific Python functions to be called from the main thread. Ignoring this requirement can cause unexpected behaviours or crashes. This is the case, for instance, when running maya.cmds or nuke calls. The Shift API offers two quick solutions that can be used to overcome this problem for Maya and Nuke. These solutions allow to force running only the required operator or callable in the main thread.

Python Decorator

The shift.utils.decorators module contains the mainThread decorator, which can be imported into a custom catalog. This decorator can wrap the execute method of an operator and dynamically force the execution logic to run on the main thread if the Shift session is running in Maya or Nuke. Here is an example on how to use it:

from shift.core.workflow import SOperator
from shift.core.workflow import SPlug
from shift.core.constants import SType
from shift.core.constants import SDirection

from shift.utils.decorators import mainThread

class MyOperator(SOperator):
    """ Operator's description.
  
    """
    def __init__(self, code, parent=None):

        super(self.__class__, self).__init__(code, parent)

        inPlug1 = SPlug(code="input1",
                      value=0,
                      type=SType.kInt,
                      direction=SDirection.kIn,
                      parent=self)

        inPlug2 = SPlug(code="input2",
                      value=0,
                      type=SType.kInt,
                      direction=SDirection.kIn,
                      parent=self)

        outPlug = SPlug(code="output",
                      value=0,
                      type=SType.kInt,
                      direction=SDirection.kOut,
                      parent=self)

        self.addPlug(inPlug1)
        self.addPlug(inPlug2)
        self.addPlug(outPlug)

    @mainThread
    def execute(self, force=False):

        input1 = self.getPlug("input1", SDirection.kIn).value()
        input2 = self.getPlug("input2", SDirection.kOut).value()

        [...my dcc code..]

        self.getPlug("output").setValue(output)
        super(self.__class__, self).execute(force)

Wrapper Function

If performance is a must, only a specific callable can be wrapped with the same logic as the execute method decorator. This can be done using the function executeInMainDCCThread from the shift.utils.dcc_threading module. Here is an example of the syntax to use:

from shift.core.workflow import SOperator
from shift.core.workflow import SPlug
from shift.core.constants import SType
from shift.core.constants import SDirection

from shift.utils.dcc_threading import executeInMainDCCThread


def my_dcc_logic(input1, input2, custom_modifier=True):

    [...my dcc code...]

    return output


class MyOperator(SOperator):
    """ Operator's description.
  
    """
    def __init__(self, code, parent=None):

        super(self.__class__, self).__init__(code, parent)

        inPlug1 = SPlug(code="input1",
                      value=0,
                      type=SType.kInt,
                      direction=SDirection.kIn,
                      parent=self)

        inPlug2 = SPlug(code="input2",
                      value=0,
                      type=SType.kInt,
                      direction=SDirection.kIn,
                      parent=self)

        outPlug = SPlug(code="output",
                      value=0,
                      type=SType.kInt,
                      direction=SDirection.kOut,
                      parent=self)

        self.addPlug(inPlug1)
        self.addPlug(inPlug2)
        self.addPlug(outPlug)

    def execute(self, force=False):

        input1 = self.getPlug("input1", SDirection.kIn).value()
        input2 = self.getPlug("input2", SDirection.kOut).value()

        output = executeInMainDCCThread(my_dcc_logic, (input1, input2), {'custom_modifier': False})

        self.getPlug("output").setValue(output)
        super(self.__class__, self).execute(force)
Inbibo Logo

consultancy, automation and digital creatures

65 Compton Street, London, United Kingdom, EC1V 0BN

info@inbibo.co.uk

Information

Products

© 2025 Inbibo LTD. - All rights reserved - Design & Development: def:studio