diff --git a/hooks/tk-multi-publish2/basic/collector.py b/hooks/tk-multi-publish2/basic/collector.py index 1c033d1..8748c72 100644 --- a/hooks/tk-multi-publish2/basic/collector.py +++ b/hooks/tk-multi-publish2/basic/collector.py @@ -113,6 +113,8 @@ def process_current_session(self, settings, parent_item): if cmds.ls(geometry=True, noIntermediate=True): self._collect_session_geometry(item) + self._collect_session_cameras(item) + def collect_current_maya_session(self, settings, parent_item): """ Creates an item that represents the current maya session. @@ -241,6 +243,44 @@ def _collect_session_geometry(self, parent_item): geo_item.set_icon_from_path(icon_path) + def _collect_session_cameras(self, parent_item): + """ + Creates items for each camera to be exported. + + :param parent_item: + :return: + """ + + # get the icon path to display for camera items + icon_path = os.path.join( + self.disk_location, + os.pardir, + "icons", + "camera.png" + ) + + for camera_shape in cmds.ls(cameras=True): + + # try to determine the camera display name + try: + camera_name = cmds.listRelatives(camera_shape, parent=True)[0] + except Exception: + # could not determine the name, just use the shape + camera_name = camera_shape + + cam_item = parent_item.create_item( + "maya.session.camera", + "Camera", + camera_name + ) + + cam_item.set_icon_from_path(icon_path) + + # store the camera name so that any attached plugin knows which + # camera this item represents! + cam_item.properties["camera_name"] = camera_name + cam_item.properties["camera_shape"] = camera_shape + def collect_playblasts(self, parent_item, project_root): """ Creates items for quicktime playblasts. diff --git a/hooks/tk-multi-publish2/basic/publish_camera.py b/hooks/tk-multi-publish2/basic/publish_camera.py new file mode 100644 index 0000000..ec62cd1 --- /dev/null +++ b/hooks/tk-multi-publish2/basic/publish_camera.py @@ -0,0 +1,363 @@ +# Copyright (c) 2017 Shotgun Software Inc. +# +# CONFIDENTIAL AND PROPRIETARY +# +# This work is provided "AS IS" and subject to the Shotgun Pipeline Toolkit +# Source Code License included in this distribution package. See LICENSE. +# By accessing, using, copying or modifying this work you indicate your +# agreement to the Shotgun Pipeline Toolkit Source Code License. All rights +# not expressly granted therein are reserved by Shotgun Software Inc. + +import fnmatch +import os + +import maya.cmds as cmds +import maya.mel as mel + +import sgtk + +HookBaseClass = sgtk.get_hook_baseclass() + + +class MayaCameraPublishPlugin(HookBaseClass): + """ + Publish2 plugin for publishing maya cameras to fbx files. + + This hook relies on functionality found in the base file publisher hook in + the publish2 app and should inherit from it in the configuration. The hook + setting for this plugin should look something like this:: + + hook: "{self}/publish_file.py:{engine}/tk-multi-publish2/basic/publish_camera.py" + + """ + + @property + def description(self): + """ + Verbose, multi-line description of what the plugin does. This can + contain simple html for formatting. + """ + + return """ +

This plugin handles publishing of cameras from maya. + A publish template is required to define the destination of the output + file. The FBXExport command is used to create the camera. +

+ """ + + @property + def settings(self): + """ + Dictionary defining the settings that this plugin expects to receive + through the settings parameter in the accept, validate, publish and + finalize methods. + + A dictionary on the following form:: + + { + "Settings Name": { + "type": "settings_type", + "default": "default_value", + "description": "One line description of the setting" + } + + The type string should be one of the data types that toolkit accepts as + part of its environment configuration. + """ + # inherit the settings from the base publish plugin + base_settings = super(MayaCameraPublishPlugin, self).settings or {} + + # settings specific to this class + maya_camera_publish_settings = { + "Publish Template": { + "type": "template", + "default": None, + "description": "Template path for published camera. Should" + "correspond to a template defined in " + "templates.yml.", + }, + "Cameras": { + "type": "list", + "default": ["camera*"], + "description": "Glob-style list of camera names to publish. " + "Example: ['camMain', 'camAux*']." + } + } + + # update the base settings + base_settings.update(maya_camera_publish_settings) + + return base_settings + + @property + def item_filters(self): + """ + List of item types that this plugin is interested in. + + Only items matching entries in this list will be presented to the + accept() method. Strings can contain glob patters such as *, for example + ["maya.*", "file.maya"] + """ + return ["maya.session.camera"] + + def accept(self, settings, item): + """ + Method called by the publisher to determine if an item is of any + interest to this plugin. Only items matching the filters defined via the + item_filters property will be presented to this method. + + A publish task will be generated for each item accepted here. Returns a + dictionary with the following booleans: + + - accepted: Indicates if the plugin is interested in this value at + all. Required. + - enabled: If True, the plugin will be enabled in the UI, otherwise + it will be disabled. Optional, True by default. + - visible: If True, the plugin will be visible in the UI, otherwise + it will be hidden. Optional, True by default. + - checked: If True, the plugin will be checked in the UI, otherwise + it will be unchecked. Optional, True by default. + + :param settings: Dictionary of Settings. The keys are strings, matching + the keys returned in the settings property. The values are `Setting` + instances. + :param item: Item to process + + :returns: dictionary with boolean keys accepted, required and enabled + """ + + publisher = self.parent + template_name = settings["Publish Template"].value + + # validate the camera name first + cam_name = item.properties.get("camera_name") + cam_shape = item.properties.get("camera_shape") + + if cam_name and cam_shape: + if not self._cam_name_matches_settings(cam_name, settings): + self.logger.debug( + "Camera name %s does not match any of the configured " + "patterns for camera names to publish. Not accepting " + "session camera item." % (cam_name,) + ) + return {"accepted": False} + else: + self.logger.debug( + "Camera name or shape was set on the item properties. Not " + "accepting session camera item." + ) + return {"accepted": False} + + # ensure a camera file template is available on the parent item + work_template = item.parent.properties.get("work_template") + if not work_template: + self.logger.debug( + "A work template is required for the session item in order to " + "publish a camera. Not accepting session camera item." + ) + return {"accepted": False} + + # ensure the publish template is defined and valid and that we also have + publish_template = publisher.get_template_by_name(template_name) + if publish_template: + item.properties["publish_template"] = publish_template + # because a publish template is configured, disable context change. + # This is a temporary measure until the publisher handles context + # switching natively. + item.context_change_allowed = False + else: + self.logger.debug( + "The valid publish template could not be determined for the " + "session camera item. Not accepting the item." + ) + return {"accepted": False} + + # check that the FBXExport command is available! + if not mel.eval("exists \"FBXExport\""): + self.logger.debug( + "Item not accepted because fbx export command 'FBXExport' " + "is not available. Perhaps the plugin is not enabled?" + ) + return {"accepted": False} + + # all good! + return { + "accepted": True, + "checked": True + } + + def validate(self, settings, item): + """ + Validates the given item to check that it is ok to publish. Returns a + boolean to indicate validity. + + :param settings: Dictionary of Settings. The keys are strings, matching + the keys returned in the settings property. The values are `Setting` + instances. + :param item: Item to process + :returns: True if item is valid, False otherwise. + """ + + path = _session_path() + + # ---- ensure the session has been saved + + if not path: + # the session still requires saving. provide a save button. + # validation fails. + error_msg = "The Maya session has not been saved." + self.logger.error( + error_msg, + extra=_get_save_as_action() + ) + raise Exception(error_msg) + + # get the normalized path + path = sgtk.util.ShotgunPath.normalize(path) + + cam_name = item.properties["camera_name"] + + # check that the camera still exists in the file + if not cmds.ls(cam_name): + error_msg = ( + "Validation failed because the collected camera (%s) is no " + "longer in the scene. You can uncheck this plugin or create " + "a camera with this name to export to avoid this error." % + (cam_name,) + ) + self.logger.error(error_msg) + raise Exception(error_msg) + + # get the configured work file template + work_template = item.parent.properties.get("work_template") + publish_template = item.properties.get("publish_template") + + # get the current scene path and extract fields from it using the work + # template: + work_fields = work_template.get_fields(path) + + # include the camera name in the fields + work_fields["camera_name"] = cam_name + + # ensure the fields work for the publish template + missing_keys = publish_template.missing_keys(work_fields) + if missing_keys: + error_msg = "Work file '%s' missing keys required for the " \ + "publish template: %s" % (path, missing_keys) + self.logger.error(error_msg) + raise Exception(error_msg) + + # create the publish path by applying the fields. store it in the item's + # properties. This is the path we'll create and then publish in the base + # publish plugin. Also set the publish_path to be explicit. + publish_path = publish_template.apply_fields(work_fields) + item.properties["path"] = publish_path + item.properties["publish_path"] = publish_path + + # use the work file's version number when publishing + if "version" in work_fields: + item.properties["publish_version"] = work_fields["version"] + + # run the base class validation + return super(MayaCameraPublishPlugin, self).validate(settings, item) + + def publish(self, settings, item): + """ + Executes the publish logic for the given item and settings. + + :param settings: Dictionary of Settings. The keys are strings, matching + the keys returned in the settings property. The values are `Setting` + instances. + :param item: Item to process + """ + + # keep track of everything currently selected. we will restore at the + # end of the publish method + cur_selection = cmds.ls(selection=True) + + # the camera to publish + cam_shape = item.properties["camera_shape"] + + # make sure it is selected + cmds.select(cam_shape) + + # get the path to create and publish + publish_path = item.properties["publish_path"] + + # ensure the publish folder exists: + publish_folder = os.path.dirname(publish_path) + self.parent.ensure_folder_exists(publish_folder) + + alembic_cmd = 'FBXExport -f "%s" -s' % (publish_path,) + + # ...and execute it: + try: + self.logger.debug("Executing command: %s" % alembic_cmd) + mel.eval(alembic_cmd) + except Exception, e: + self.logger.error("Failed to export camera: %s" % e) + return + + # Now that the path has been generated, hand it off to the + super(MayaCameraPublishPlugin, self).publish(settings, item) + + # restore selection + cmds.select(cur_selection) + + def _cam_name_matches_settings(self, cam_name, settings): + """ + Returns True if the supplied camera name matches any of the configured + camera name patterns. + """ + + # loop through each pattern specified and see if the supplied camera + # name matches the pattern + cam_patterns = settings["Cameras"].value + + # if no patterns specified, then no constraints on camera name + if not cam_patterns: + return True + + for camera_pattern in cam_patterns: + if fnmatch.fnmatch(cam_name, camera_pattern): + return True + + return False + + +def _session_path(): + """ + Return the path to the current session + :return: + """ + path = cmds.file(query=True, sn=True) + + if isinstance(path, unicode): + path = path.encode("utf-8") + + return path + + +def _get_save_as_action(): + """ + Simple helper for returning a log action dict for saving the session + """ + + engine = sgtk.platform.current_engine() + + # default save callback + callback = cmds.SaveScene + + # if workfiles2 is configured, use that for file save + if "tk-multi-workfiles2" in engine.apps: + app = engine.apps["tk-multi-workfiles2"] + if hasattr(app, "show_file_save_dlg"): + callback = app.show_file_save_dlg + + return { + "action_button": { + "label": "Save As...", + "tooltip": "Save the current session", + "callback": callback + } + } diff --git a/hooks/tk-multi-publish2/icons/camera.png b/hooks/tk-multi-publish2/icons/camera.png new file mode 100644 index 0000000..e9f8472 Binary files /dev/null and b/hooks/tk-multi-publish2/icons/camera.png differ