diff --git a/client/ayon_hiero/api/plugin.py b/client/ayon_hiero/api/plugin.py index bc98d2f..06b58ae 100644 --- a/client/ayon_hiero/api/plugin.py +++ b/client/ayon_hiero/api/plugin.py @@ -1,7 +1,7 @@ import os -from pprint import pformat import re import uuid +from copy import deepcopy import hiero @@ -418,8 +418,6 @@ def __init__(self, cls, context, path, **options): # inject folder data to representation dict folder_entity = self.context["folder"] self.data["folderAttributes"] = folder_entity["attrib"] - log.info("__init__ self.data: `{}`".format(pformat(self.data))) - log.info("__init__ options: `{}`".format(pformat(options))) # add active components to class if self.new_sequence: @@ -657,6 +655,10 @@ class PublishClip: Returns: hiero.core.TrackItem: hiero track item object with AYON tag """ + vertical_clip_match = {} + vertical_clip_used = {} + tag_data = {} + types = { "shot": "shot", "folder": "folder", @@ -672,7 +674,7 @@ class PublishClip: rename_default = False hierarchy_default = "{_folder_}/{_sequence_}/{_track_}" clip_name_default = "shot_{_trackIndex_:0>3}_{_clipIndex_:0>4}" - product_name_default = "" + base_product_variant_default = "" review_source_default = None product_type_default = "plate" count_from_default = 10 @@ -694,6 +696,11 @@ class PublishClip: "reviewableSource", } + @classmethod + def restore_all_caches(cls): + cls.vertical_clip_match = {} + cls.vertical_clip_used = {} + def __init__( self, track_item, @@ -702,8 +709,6 @@ def __init__( rename_index=0): self.rename_index = rename_index - self.vertical_clip_match = dict() - self.tag_data = dict() # adding ui inputs if any self.pre_create_data = pre_create_data or {} @@ -726,9 +731,6 @@ def __init__( if data: self.tag_data.update(data) - # add publish attribute to tag data - self.tag_data.update({"publish": True}) - # populate default data before we get other attributes self._populate_track_item_default_data() @@ -799,12 +801,14 @@ def get(key): return self.pre_create_data.get(key) # ui_inputs data or default values if gui was not used - self.rename = self.pre_create_data.get("clipRename", self.rename_default) + self.rename = self.pre_create_data.get( + "clipRename", self.rename_default) self.clip_name = get("clipName") or self.clip_name_default self.hierarchy = get("hierarchy") or self.hierarchy_default self.count_from = get("countFrom") or self.count_from_default self.count_steps = get("countSteps") or self.count_steps_default - self.product_name = get("productName") or self.product_name_default + self.base_product_variant = ( + get("clipVariant") or self.base_product_variant_default) self.product_type = get("productType") or self.product_type_default self.vertical_sync = get("vSyncOn") or self.vertical_sync_default self.driving_layer = get("vSyncTrack") or self.driving_layer_default @@ -818,13 +822,13 @@ def get(key): } # build product name from layer name - if self.product_name == "": - self.product_name = self.track_name + if self.base_product_variant == "": + self.variant = self.track_name + else: + self.variant = self.base_product_variant # create product for publishing - self.product_name = ( - self.product_type + self.product_name.capitalize() - ) + self.product_name = f"{self.product_type}{self.variant.capitalize()}" def _replace_hash_to_expression(self, name, text): """ Replace hash with number in correct padding. """ @@ -841,17 +845,23 @@ def _convert_to_tag_data(self): # define vertical sync attributes hero_track = True self.reviewable_source = "" - if self.vertical_sync: + if ( + self.vertical_sync + and self.track_name != self.driving_layer + ): # check if track name is not in driving layer - if self.track_name not in self.driving_layer: - # if it is not then define vertical sync as None - hero_track = False + # if it is not then define vertical sync as None + hero_track = False # increasing steps by index of rename iteration self.count_steps *= self.rename_index - hierarchy_formatting_data = dict() + hierarchy_formatting_data = {} + hierarchy_data = deepcopy(self.hierarchy_data) _data = self.track_item_default_data.copy() + + # in case we are running creators headless default + # precreate data values are used if self.pre_create_data: # adding tag metadata from ui @@ -883,48 +893,87 @@ def _convert_to_tag_data(self): _data.update({"shot": self.shot_num}) # solve # in test to pythonic expression - for _key, _value in self.hierarchy_data.items(): + for _key, _value in hierarchy_data.items(): if "#" not in _value: continue - self.hierarchy_data[_key] = self._replace_hash_to_expression( - _key, _value - ) + hierarchy_data[_key] = self._replace_hash_to_expression( + _key, _value) # fill up pythonic expresisons in hierarchy data - for _key, _value in self.hierarchy_data.items(): - hierarchy_formatting_data[_key] = _value.format(**_data) + for _key, _value in hierarchy_data.items(): + formatted_value = _value.format(**_data) + hierarchy_formatting_data[_key] = formatted_value + self.tag_data[_key] = formatted_value else: # if no gui mode then just pass default data - hierarchy_formatting_data = self.hierarchy_data + hierarchy_formatting_data = hierarchy_data - tag_hierarchy_data = self._solve_tag_hierarchy_data( + tag_instance_data = self._solve_tag_instance_data( hierarchy_formatting_data ) - tag_hierarchy_data.update({"heroTrack": True}) + tag_instance_data.update({"heroTrack": True}) if hero_track and self.vertical_sync: - self.vertical_clip_match.update({ - (self.clip_in, self.clip_out): tag_hierarchy_data - }) + self.vertical_clip_match.update( + {(self.clip_in, self.clip_out): tag_instance_data} + ) if not hero_track and self.vertical_sync: # driving layer is set as negative match - for (_in, _out), hero_data in self.vertical_clip_match.items(): - hero_data.update({"heroTrack": False}) - if _in == self.clip_in and _out == self.clip_out: - data_product_name = hero_data["productName"] - # add track index in case duplicity of names in hero data - if self.product_name in data_product_name: - hero_data["productName"] = self.product_name + str( - self.track_index) - # in case track name and product name is the same then add - if self.product_name == self.track_name: - hero_data["productName"] = self.product_name - # assign data to return hierarchy data to tag - tag_hierarchy_data = hero_data + for (hero_in, hero_out), hero_data in self.vertical_clip_match.items(): # noqa + """Iterate over all clips in vertical sync match + + If clip frame range is outside of hero clip frame range + then skip this clip and do not add to hierarchical shared + metadata to them. + """ + if self.clip_in < hero_in or self.clip_out > hero_out: + continue + + _distrib_data = deepcopy(hero_data) + _distrib_data["heroTrack"] = False + + # form used clip unique key + data_product_name = hero_data["productName"] + new_clip_name = hero_data["newClipName"] + + # get used names list for duplicity check + used_names_list = self.vertical_clip_used.setdefault( + f"{new_clip_name}{data_product_name}", []) + + clip_product_name = self.product_name + variant = self.variant + + # in case track name and product name is the same then add + if self.variant == self.track_name: + clip_product_name = self.product_name + + # add track index in case duplicity of names in hero data + # INFO: this is for case where hero clip product name + # is the same as current clip product name + if clip_product_name in data_product_name: + clip_product_name = ( + f"{clip_product_name}{self.track_index}") + variant = f"{variant}{self.track_index}" + + # in case track clip product name had been already used + # then add product name with clip index + if clip_product_name in used_names_list: + clip_product_name = ( + f"{clip_product_name}{self.rename_index}") + variant = f"{variant}{self.rename_index}" + + _distrib_data["productName"] = clip_product_name + _distrib_data["variant"] = variant + # assign data to return hierarchy data to tag + tag_instance_data = _distrib_data + + # add used product name to used list to avoid duplicity + used_names_list.append(clip_product_name) + break # add data to return data dict - self.tag_data.update(tag_hierarchy_data) + self.tag_data.update(tag_instance_data) # add uuid to tag data self.tag_data["uuid"] = str(uuid.uuid4()) @@ -954,7 +1003,7 @@ def _convert_to_tag_data(self): else: self.tag_data.pop("reviewableSource", None) - def _solve_tag_hierarchy_data(self, hierarchy_formatting_data): + def _solve_tag_instance_data(self, hierarchy_formatting_data): """ Solve tag data from hierarchy data and templates. """ # fill up clip name and hierarchy keys hierarchy_filled = self.hierarchy.format(**hierarchy_formatting_data) @@ -969,7 +1018,8 @@ def _solve_tag_hierarchy_data(self, hierarchy_formatting_data): "parents": self.parents, "hierarchyData": hierarchy_formatting_data, "productName": self.product_name, - "productType": self.product_type + "productType": self.product_type, + "variant": self.variant, } def _convert_to_entity(self, src_type, template): diff --git a/client/ayon_hiero/plugins/create/create_shot_clip.py b/client/ayon_hiero/plugins/create/create_shot_clip.py index 05f6383..718f9f8 100644 --- a/client/ayon_hiero/plugins/create/create_shot_clip.py +++ b/client/ayon_hiero/plugins/create/create_shot_clip.py @@ -100,8 +100,6 @@ def create(self, instance_data, _): CreatedInstance: The created instance object for the new shot. """ instance_data.update({ - "productName": f"{self.product_type}{instance_data['variant']}", - "productType": self.product_type, "newHierarchyIntegration": True, # Backwards compatible (Deprecated since 24/06/06) "newAssetPublishing": True, @@ -270,12 +268,6 @@ def create(self, instance_data, _): Return: CreatedInstance: The created instance object for the new shot. """ - if instance_data.get("clip_variant") == "": - instance_data["variant"] = instance_data["hierarchyData"]["track"] - - else: - instance_data["variant"] = instance_data["clip_variant"] - return super().create(instance_data, None) @@ -302,6 +294,8 @@ class CreateShotClip(plugin.HieroCreator): """ create_allow_thumbnail = False + shot_instances = {} + def get_pre_create_attr_defs(self): def header_label(text): @@ -436,7 +430,7 @@ def header_label(text): label=header_label("Publish Settings") ), EnumDef( - "clip_variant", + "clipVariant", label="Product Variant", tooltip="Chose variant which will be then used for " "product name, if " @@ -517,7 +511,6 @@ def create(self, subset_name, instance_data, pre_create_data): "timeline in order to export audio." ) - instance_data["clip_variant"] = pre_create_data["clip_variant"] instance_data["task"] = None # sort selected trackItems by @@ -538,20 +531,19 @@ def create(self, subset_name, instance_data, pre_create_data): "io.ayon.creators.hiero.plate": True, "io.ayon.creators.hiero.audio": pre_create_data.get("export_audio", False), } - enabled_creators = tuple(cre for cre, enabled in all_creators.items() if enabled) instances = [] - for idx, track_item in enumerate(sorted_selected_track_items): - - instance_data["clip_index"] = track_item.guid() + _instance_data = copy.deepcopy(instance_data) + _instance_data["clip_index"] = track_item.guid() # convert track item to timeline media pool item publish_clip = plugin.PublishClip( track_item, pre_create_data=pre_create_data, rename_index=idx, - data=instance_data) + data=_instance_data, + ) track_item = publish_clip.convert() if track_item is None: @@ -563,7 +555,7 @@ def create(self, subset_name, instance_data, pre_create_data): "Processing track item data: {} (index: {})".format( track_item, idx) ) - instance_data.update(publish_clip.tag_data) + _instance_data.update(publish_clip.tag_data) # Delete any existing instances previously generated for the clip. prev_tag = lib.get_trackitem_ayon_tag(track_item) @@ -579,11 +571,19 @@ def create(self, subset_name, instance_data, pre_create_data): creator.remove_instances(prev_instances) # Create new product(s) instances. - clip_instances = {} + shot_folder_path = _instance_data["folderPath"] + shot_instances = self.shot_instances.setdefault( + shot_folder_path, {}) shot_creator_id = "io.ayon.creators.hiero.shot" + all_creators["io.ayon.creators.hiero.shot"] = _instance_data.get( + "heroTrack", False) + enabled_creators = tuple( + cre for cre, enabled in all_creators.items() if enabled + ) + clip_instances = {} for creator_id in enabled_creators: creator = self.create_context.creators[creator_id] - sub_instance_data = copy.deepcopy(instance_data) + sub_instance_data = copy.deepcopy(_instance_data) shot_folder_path = sub_instance_data["folderPath"] # Shot creation @@ -591,41 +591,46 @@ def create(self, subset_name, instance_data, pre_create_data): track_item_duration = track_item.duration() workfileFrameStart = \ sub_instance_data["workfileFrameStart"] - sub_instance_data.update({ - "creator_attributes": { - "workfileFrameStart": \ - sub_instance_data["workfileFrameStart"], - "handleStart": sub_instance_data["handleStart"], - "handleEnd": sub_instance_data["handleEnd"], - "frameStart": workfileFrameStart, - "frameEnd": (workfileFrameStart + - track_item_duration), - "clipIn": track_item.timelineIn(), - "clipOut": track_item.timelineOut(), - "clipDuration": track_item_duration, - "sourceIn": track_item.sourceIn(), - "sourceOut": track_item.sourceOut(), - }, - "label": ( - f"{shot_folder_path} shot" - ), - }) - # Plate, Audio - # insert parent instance data to allow - # metadata recollection as publish time. - else: - parenting_data = clip_instances[shot_creator_id] sub_instance_data.update( { - "parent_instance_id": parenting_data["instance_id"], - "label": (f"{shot_folder_path} " f"{creator.product_type}"), + "variant": "main", + "productType": "shot", + "productName": "shotMain", "creator_attributes": { - "parentInstance": parenting_data["label"] + "workfileFrameStart": sub_instance_data[ + "workfileFrameStart" + ], + "handleStart": sub_instance_data["handleStart"], + "handleEnd": sub_instance_data["handleEnd"], + "frameStart": workfileFrameStart, + "frameEnd": (workfileFrameStart + track_item_duration), + "clipIn": track_item.timelineIn(), + "clipOut": track_item.timelineOut(), + "clipDuration": track_item_duration, + "sourceIn": track_item.sourceIn(), + "sourceOut": track_item.sourceOut(), }, + "label": (f"{sub_instance_data['folderPath']} shotMain"), } ) + # Plate, Audio + # insert parent instance data to allow + # metadata recollection as publish time. + else: + parenting_data = shot_instances[shot_creator_id] + sub_instance_data.update({ + "parent_instance_id": parenting_data["instance_id"], + "label": ( + f"{sub_instance_data['folderPath']} " + f"{sub_instance_data['productName']}" + ), + "creator_attributes": { + "parentInstance": parenting_data["label"], + } + }) + # add reviewable source to plate if shot has it if sub_instance_data.get("reviewableSource"): sub_instance_data["creator_attributes"].update({ @@ -637,7 +642,9 @@ def create(self, subset_name, instance_data, pre_create_data): instance = creator.create(sub_instance_data, None) instance.transient_data["track_item"] = track_item self._add_instance_to_context(instance) - clip_instances[creator_id] = instance.data_to_store() + instance_data_to_store = instance.data_to_store() + shot_instances[creator_id] = instance_data_to_store + clip_instances[creator_id] = instance_data_to_store lib.imprint( track_item, @@ -648,6 +655,10 @@ def create(self, subset_name, instance_data, pre_create_data): ) instances.append(instance) + # restore all caches + plugin.PublishClip.restore_all_caches() + self.shot_instances = {} + return instances def _create_and_add_instance(self, data, creator_id, @@ -760,7 +771,9 @@ def _collect_legacy_instance(self, track_item): workfileFrameStart = \ sub_instance_data["workfileFrameStart"] sub_instance_data.update({ - "label": f"{sub_instance_data['folderPath']} shot", + "label": ( + f"{sub_instance_data['folderPath']} " + f"{sub_instance_data['productName']}"), "variant": "main", "creator_attributes": { "workfileFrameStart": workfileFrameStart, @@ -799,17 +812,17 @@ def _collect_legacy_instance(self, track_item): creator = self.create_context.creators[sub_creator_id] sub_instance_data.update( { - "clip_variant": sub_instance_data["variant"], "parent_instance_id": parenting_data["instance_id"], "label": ( f"{sub_instance_data['folderPath']} " - f"{creator.product_type}" + f"{sub_instance_data['productName']}" ), "creator_attributes": { "parentInstance": parenting_data["label"], - "reviewableSource": sub_instance_data["reviewableSource"], + "reviewableSource": sub_instance_data[ + "reviewableSource"], "review": False, - }, + } } )