Skip to content

api

FusionHost

Bases: HostBase, IWorkfileHost, ILoadHost, IPublishHost

Source code in client/ayon_fusion/api/pipeline.py
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
class FusionHost(HostBase, IWorkfileHost, ILoadHost, IPublishHost):
    name = "fusion"

    def __init__(self, **kwargs):
        super().__init__(**kwargs)
        self._forced_current_comp = None

    def install(self):
        """Install fusion-specific functionality of AYON.

        This is where you install menus and register families, data
        and loaders into fusion.

        It is called automatically when installing via
        `ayon_core.pipeline.install_host(ayon_fusion.api)`

        See the Maya equivalent for inspiration on how to implement this.

        """
        # Remove all handlers associated with the root logger object, because
        # that one always logs as "warnings" incorrectly.
        for handler in logging.root.handlers[:]:
            logging.root.removeHandler(handler)

        # Attach default logging handler that prints to active comp
        logger = logging.getLogger()
        formatter = logging.Formatter(fmt="%(message)s\n")
        handler = FusionLogHandler()
        handler.setFormatter(formatter)
        logger.addHandler(handler)
        logger.setLevel(logging.DEBUG)

        pyblish.api.register_host("fusion")
        pyblish.api.register_plugin_path(PUBLISH_PATH)
        log.info("Registering Fusion plug-ins..")

        register_loader_plugin_path(LOAD_PATH)
        register_creator_plugin_path(CREATE_PATH)
        register_inventory_action_path(INVENTORY_PATH)

        # Register events
        register_event_callback("open", on_after_open)
        register_event_callback("workfile.save.before", before_workfile_save)
        register_event_callback("save", on_save)
        register_event_callback("new", on_new)
        register_event_callback("taskChanged", on_task_changed)

    # region workfile io api
    def has_unsaved_changes(self):
        comp = self.get_current_comp()
        return comp.GetAttrs()["COMPB_Modified"]

    def get_workfile_extensions(self):
        return [".comp"]

    def save_workfile(self, dst_path=None):
        comp = self.get_current_comp()
        comp.Save(dst_path)

    def open_workfile(self, filepath):
        # Hack to get fusion, see
        #   ayon_fusion.api.pipeline.get_current_comp()
        fusion = getattr(sys.modules["__main__"], "fusion", None)

        return fusion.LoadComp(filepath)

    def get_current_workfile(self):
        comp = self.get_current_comp()
        current_filepath = comp.GetAttrs()["COMPS_FileName"]
        if not current_filepath:
            return None

        return current_filepath

    def work_root(self, session):
        work_dir = session["AYON_WORKDIR"]
        scene_dir = session.get("AVALON_SCENEDIR")
        if scene_dir:
            return os.path.join(work_dir, scene_dir)
        else:
            return work_dir
    # endregion

    @contextlib.contextmanager
    def maintained_selection(self):
        from .lib import maintained_selection
        return maintained_selection()

    def get_containers(self):
        return ls()

    def update_context_data(self, data, changes):
        comp = self.get_current_comp()
        comp.SetData("openpype", data)

    def get_context_data(self):
        comp = self.get_current_comp()
        return comp.GetData("openpype") or {}

    def get_current_comp(self):
        if self._forced_current_comp is not None:
            return self._forced_current_comp
        return get_current_comp()

    @contextlib.contextmanager
    def current_comp(self, comp):
        """Fusion context manager to temporarily enforce 'current' comp.

        During this context the workfile methods like `get_current_workfile`
        and `save_workfile` will use the specifiec `comp` instead of the actual
        active comp tab in Fusion itself. This is useful during e.g. publishing
        where a user may have initiated a publish from one comp and then as
        the publishing is running they switch to another comp tab.
        """
        old_comp = self._forced_current_comp
        try:
            self._forced_current_comp = comp
            yield
        finally:
            self._forced_current_comp = old_comp

current_comp(comp)

Fusion context manager to temporarily enforce 'current' comp.

During this context the workfile methods like get_current_workfile and save_workfile will use the specifiec comp instead of the actual active comp tab in Fusion itself. This is useful during e.g. publishing where a user may have initiated a publish from one comp and then as the publishing is running they switch to another comp tab.

Source code in client/ayon_fusion/api/pipeline.py
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
@contextlib.contextmanager
def current_comp(self, comp):
    """Fusion context manager to temporarily enforce 'current' comp.

    During this context the workfile methods like `get_current_workfile`
    and `save_workfile` will use the specifiec `comp` instead of the actual
    active comp tab in Fusion itself. This is useful during e.g. publishing
    where a user may have initiated a publish from one comp and then as
    the publishing is running they switch to another comp tab.
    """
    old_comp = self._forced_current_comp
    try:
        self._forced_current_comp = comp
        yield
    finally:
        self._forced_current_comp = old_comp

install()

Install fusion-specific functionality of AYON.

This is where you install menus and register families, data and loaders into fusion.

It is called automatically when installing via ayon_core.pipeline.install_host(ayon_fusion.api)

See the Maya equivalent for inspiration on how to implement this.

Source code in client/ayon_fusion/api/pipeline.py
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
def install(self):
    """Install fusion-specific functionality of AYON.

    This is where you install menus and register families, data
    and loaders into fusion.

    It is called automatically when installing via
    `ayon_core.pipeline.install_host(ayon_fusion.api)`

    See the Maya equivalent for inspiration on how to implement this.

    """
    # Remove all handlers associated with the root logger object, because
    # that one always logs as "warnings" incorrectly.
    for handler in logging.root.handlers[:]:
        logging.root.removeHandler(handler)

    # Attach default logging handler that prints to active comp
    logger = logging.getLogger()
    formatter = logging.Formatter(fmt="%(message)s\n")
    handler = FusionLogHandler()
    handler.setFormatter(formatter)
    logger.addHandler(handler)
    logger.setLevel(logging.DEBUG)

    pyblish.api.register_host("fusion")
    pyblish.api.register_plugin_path(PUBLISH_PATH)
    log.info("Registering Fusion plug-ins..")

    register_loader_plugin_path(LOAD_PATH)
    register_creator_plugin_path(CREATE_PATH)
    register_inventory_action_path(INVENTORY_PATH)

    # Register events
    register_event_callback("open", on_after_open)
    register_event_callback("workfile.save.before", before_workfile_save)
    register_event_callback("save", on_save)
    register_event_callback("new", on_new)
    register_event_callback("taskChanged", on_task_changed)

comp_lock_and_undo_chunk(comp, undo_queue_name='Script CMD', keep_undo=True)

Lock comp and open an undo chunk during the context

Source code in client/ayon_fusion/api/lib.py
301
302
303
304
305
306
307
308
309
310
311
312
313
314
@contextlib.contextmanager
def comp_lock_and_undo_chunk(
    comp,
    undo_queue_name="Script CMD",
    keep_undo=True,
):
    """Lock comp and open an undo chunk during the context"""
    try:
        comp.Lock()
        comp.StartUndo(undo_queue_name)
        yield
    finally:
        comp.Unlock()
        comp.EndUndo(keep_undo)

get_bmd_library()

Get bmd library

Source code in client/ayon_fusion/api/lib.py
287
288
289
290
def get_bmd_library():
    """Get bmd library"""
    bmd = getattr(sys.modules["__main__"], "bmd", None)
    return bmd

get_current_comp()

Get current comp in this session

Source code in client/ayon_fusion/api/lib.py
293
294
295
296
297
298
def get_current_comp():
    """Get current comp in this session"""
    fusion = get_fusion_module()
    if fusion is not None:
        comp = fusion.CurrentComp
        return comp

imprint_container(tool, name, namespace, context, loader=None)

Imprint a Loader with metadata

Containerisation enables a tracking of version, author and origin for loaded assets.

Parameters:

Name Type Description Default
tool object

The node in Fusion to imprint as container, usually a Loader.

required
name str

Name of resulting assembly

required
namespace str

Namespace under which to host container

required
context dict

Asset information

required
loader str

Name of loader used to produce this container.

None

Returns:

Type Description

None

Source code in client/ayon_fusion/api/pipeline.py
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
def imprint_container(tool,
                      name,
                      namespace,
                      context,
                      loader=None):
    """Imprint a Loader with metadata

    Containerisation enables a tracking of version, author and origin
    for loaded assets.

    Arguments:
        tool (object): The node in Fusion to imprint as container, usually a
            Loader.
        name (str): Name of resulting assembly
        namespace (str): Namespace under which to host container
        context (dict): Asset information
        loader (str, optional): Name of loader used to produce this container.

    Returns:
        None

    """

    data = [
        ("schema", "openpype:container-2.0"),
        ("id", AVALON_CONTAINER_ID),
        ("name", str(name)),
        ("namespace", str(namespace)),
        ("loader", str(loader)),
        ("representation", context["representation"]["id"]),
        ("project_name", context["project"]["name"]),
    ]

    for key, value in data:
        tool.SetData("avalon.{}".format(key), value)

ls()

List containers from active Fusion scene

This is the host-equivalent of api.ls(), but instead of listing assets on disk, it lists assets already loaded in Fusion; once loaded they are called 'containers'

Yields:

Name Type Description
dict

container

Source code in client/ayon_fusion/api/pipeline.py
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
def ls():
    """List containers from active Fusion scene

    This is the host-equivalent of api.ls(), but instead of listing
    assets on disk, it lists assets already loaded in Fusion; once loaded
    they are called 'containers'

    Yields:
        dict: container

    """
    host = registered_host()
    comp = host.get_current_comp()
    tools = comp.GetToolList(False).values()

    for tool in tools:
        container = parse_container(tool)
        if container:
            yield container

maintained_selection(comp=None)

Reset comp selection from before the context after the context

Source code in client/ayon_fusion/api/lib.py
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
@contextlib.contextmanager
def maintained_selection(comp=None):
    """Reset comp selection from before the context after the context"""
    if comp is None:
        comp = get_current_comp()

    previous_selection = comp.GetToolList(True).values()
    try:
        yield
    finally:
        flow = comp.CurrentFrame.FlowView
        flow.Select()  # No args equals clearing selection
        if previous_selection:
            for tool in previous_selection:
                flow.Select(tool, True)

parse_container(tool)

Returns imprinted container data of a tool

This reads the imprinted data from imprint_container.

Source code in client/ayon_fusion/api/pipeline.py
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
def parse_container(tool):
    """Returns imprinted container data of a tool

    This reads the imprinted data from `imprint_container`.

    """

    data = tool.GetData('avalon')
    if not isinstance(data, dict):
        return

    # If not all required data return the empty container
    required = ['schema', 'id', 'name',
                'namespace', 'loader', 'representation']
    if not all(key in data for key in required):
        return

    container = {key: data[key] for key in required}

    # Add optional keys, like `project_name`
    optional = ["project_name"]
    for key in optional:
        if key in data:
            container[key] = data[key]

    # Store the tool's name
    container["objectName"] = tool.Name

    # Store reference to the tool object
    container["_tool"] = tool

    return container

set_current_context_framerange(task_entity=None)

Set Comp's frame range based on current task.

Source code in client/ayon_fusion/api/lib.py
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
def set_current_context_framerange(task_entity=None):
    """Set Comp's frame range based on current task."""
    if task_entity is None:
        task_entity = get_current_task_entity(
            fields={"attrib.frameStart",
                    "attrib.frameEnd",
                    "attrib.handleStart",
                    "attrib.handleEnd"})

    task_attributes = task_entity["attrib"]
    start = task_attributes["frameStart"]
    end = task_attributes["frameEnd"]
    handle_start = task_attributes["handleStart"]
    handle_end = task_attributes["handleEnd"]
    update_frame_range(start, end, set_render_range=True,
                       handle_start=handle_start,
                       handle_end=handle_end)

update_frame_range(start, end, comp=None, set_render_range=True, handle_start=0, handle_end=0)

Set Fusion comp's start and end frame range

Parameters:

Name Type Description Default
start (float, int)

start frame

required
end (float, int)

end frame

required
comp (object, Optional)

comp object from fusion

None
set_render_range (bool, Optional)

When True this will also set the composition's render start and end frame.

True
handle_start (float, int, Optional)

frame handles before start frame

0
handle_end (float, int, Optional)

frame handles after end frame

0

Returns:

Type Description

None

Source code in client/ayon_fusion/api/lib.py
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
def update_frame_range(start, end, comp=None, set_render_range=True,
                       handle_start=0, handle_end=0):
    """Set Fusion comp's start and end frame range

    Args:
        start (float, int): start frame
        end (float, int): end frame
        comp (object, Optional): comp object from fusion
        set_render_range (bool, Optional): When True this will also set the
            composition's render start and end frame.
        handle_start (float, int, Optional): frame handles before start frame
        handle_end (float, int, Optional): frame handles after end frame

    Returns:
        None

    """

    if not comp:
        comp = get_current_comp()

    # Convert any potential none type to zero
    handle_start = handle_start or 0
    handle_end = handle_end or 0

    attrs = {
        "COMPN_GlobalStart": start - handle_start,
        "COMPN_GlobalEnd": end + handle_end
    }

    # set frame range
    if set_render_range:
        attrs.update({
            "COMPN_RenderStart": start,
            "COMPN_RenderEnd": end
        })

    with comp_lock_and_undo_chunk(comp):
        comp.SetAttrs(attrs)