#target photoshop app.bringToFront(); // This script exports Adobe Photoshop layers as individual PNGs. It also // writes a JSON file which can be imported into Spine where the images // will be displayed in the same positions and draw order. // Copyright (c) 2012-2020, Esoteric Software // All rights reserved. // Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: // * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. // * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. // * Neither the name of Esoteric Software nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. var scriptVersion = 7.04; // This is incremented every time the script is modified, so you know if you have the latest. var cs2 = parseInt(app.version) < 10, cID = charIDToTypeID, sID = stringIDToTypeID; var originalDoc, settings, progress, cancel, errors; try { originalDoc = activeDocument; } catch (ignored) {} var defaultSettings = { ignoreHiddenLayers: false, ignoreBackground: true, writeTemplate: false, writeJson: true, trimWhitespace: true, scale: 1, padding: 1, imagesDir: "./images/", jsonPath: "./", }; loadSettings(); function run () { showProgress(); errors = []; // Output dirs. var jsonFile = new File(jsonPath(settings.jsonPath)); jsonFile.parent.create(); var imagesDir = absolutePath(settings.imagesDir); var imagesFolder = new Folder(imagesDir); imagesFolder.create(); var origin = [rulerOrigin("H"), rulerOrigin("V")], xOffSet = origin[0], yOffSet = origin[1]; try { deleteDocumentAncestorsMetadata(); } catch (ignored) {} activeDocument.duplicate(); deselectLayers(); // Uncomment this line to enlarge the canvas so layers are not cropped. //activeDocument.revealAll(); try { convertToRGB(); } catch (ignored) {} if (activeDocument.mode != DocumentMode.RGB) { alert("Please change the image mode to RGB color."); return; } // Output template image. if (settings.writeTemplate) { if (settings.scale != 1) { storeHistory(); scaleImage(settings.scale); } var file = new File(imagesDir + "template.png"); if (file.exists) file.remove(); savePNG(file); if (settings.scale != 1) restoreHistory(); } if (!settings.jsonPath && !settings.imagesDir) return; rasterizeAll(); // Add a history item to prevent layer visibility from changing by the active layer being reset to the top. var topLayer = activeDocument.artLayers.add(); topLayer.name = "Collecting layers..."; // Collect and hide layers. var rootLayers = [], layers = []; var context = { first: hasBackgroundLayer() ? 0 : 1, index: getLayerCount() - 1, total: 0 }; initializeLayers(context, null, rootLayers); showProgress("Collecting layers...", context.total); collectLayers(rootLayers, layers); // Store the bones, slot names, and layers for each skin. var bones = { _root: { name: "root", x: 0, y: 0, children: [] } }; var slots = {}, slotsCount = 0; var skins = { _default: [] }, skinsCount = 0; var skinDuplicates = {}; var totalLayerCount = 0; outer: for (var i = 0, n = layers.length; i < n; i++) { if (cancel) return; var layer = layers[i]; var name = stripTags(layer.name).replace(/.png$/, ""); name = name.replace(/[\\:"*?<>|]/g, "").replace(/^\.+$/, "").replace(/^__drag$/, ""); // Illegal. name = name.replace(/^(con|prn|aux|nul|com[0-9]|lpt[0-9])(\..*)?$/i, ""); // Windows. if (!name || name.length > 255) { error("Layer name is not a valid attachment name:\n\n" + layer.name); continue; } var folderPath = layer.folders(""); if (startsWith(name, "/")) { name = name.substring(1); layer.attachmentName = name; } else layer.attachmentName = folderPath + name; layer.attachmentPath = layer.findTagValue("path:"); if (!layer.attachmentPath) layer.attachmentPath = layer.attachmentName; else if (startsWith(layer.attachmentPath, "/")) layer.attachmentPath = layer.attachmentPath.substring(1); else layer.attachmentPath = folderPath + layer.attachmentPath; var scale = layer.findTagValue("scale:"); if (!scale) scale = 1; layer.scale = parseFloat(scale); if (isNaN(layer.scale)) error("Invalid scale " + scale + ": " + layer.path()); var bone, boneLayer = layer.findTagLayer("bone"); if (boneLayer) { var parent = boneLayer.getParentBone(bones); var boneName = boneLayer.findTagValue("bone"); bone = get(bones, boneName); if (bone) { if (parent != bone.parent) { error("Multiple layers for the \"" + boneName + "\" bone have different parent bones:\n\n" + bone.layer.path() + "\n" + boneLayer.path()); continue; } } else { set(bones, boneName, bone = { name: boneName, parent: parent, children: [], layer: boneLayer }); parent.children.push(bone); } bone.x = layer.left * settings.scale - settings.padding; bone.x += layer.width * settings.scale / 2 + settings.padding; bone.y = layer.bottom * settings.scale + settings.padding; bone.y -= layer.height * settings.scale / 2 + settings.padding; // Make relative to the Photoshop document ruler origin. bone.x -= xOffSet * settings.scale; bone.y -= (activeDocument.height.as("px") - yOffSet) * settings.scale; } var skinName = layer.findTagValue("skin"); if (skinName && skinName.toLowerCase() == "default") { error("The skin name \"default\" is reserved: " + layer.path() + "\nPlease use a different name."); continue; } if (!skinName) skinName = "default"; layer.skinName = skinName; layer.placeholderName = skinName == "default" ? layer.attachmentName : name; layer.slotName = layer.findTagValue("slot") || name; var slot = get(slots, layer.slotName); if (!slot) { slotsCount++; set(slots, layer.slotName, slot = { bone: bone, attachment: layer.wasVisible ? layer.placeholderName : null, placeholders: {}, attachments: false }); } else if (!slot.attachment && layer.wasVisible) slot.attachment = layer.placeholderName; if (layer.blendMode == "linearDodge") slot.blend = "additive"; else if (layer.blendMode == "multiply") slot.blend = "multiply"; else if (layer.blendMode == "screen") slot.blend = "screen"; var placeholders = get(slot.placeholders, skinName); if (!placeholders) set(slot.placeholders, skinName, placeholders = {}); else { var existing = get(placeholders, layer.placeholderName); if (existing) { // Skin has duplicate placeholders. var key = layer.slotName + "|^`" + skinName; remove(skinDuplicates, key, existing); add(skinDuplicates, key, existing); add(skinDuplicates, key, layer); } } set(placeholders, layer.placeholderName, layer); var skinSlots = get(skins, skinName); if (!skinSlots) { set(skins, skinName, skinSlots = {}); skinsCount++; } add(skinSlots, layer.slotName, layer); totalLayerCount++; } // Error if a skin has multiple skin placeholders with the same name. for (var key in skinDuplicates) { if (!skinDuplicates.hasOwnProperty(key)) continue; var layers = skinDuplicates[key]; var message = "Multiple layers for the \"" + layers[0].skinName + "\" skin in the \"" + layers[0].slotName + "\" slot have the same name \"" + layers[0].placeholderName + "\":\n"; for (var i = 0, n = layers.length; i < n; i++) message += "\n" + layers[i].path(); error(message + "\n\nRename or use the [ignore] tag for these layers."); } // Error if a skin placeholder has the same name as a default skin attachment. var slotDuplicates = {}; for (var slotName in slots) { if (!slots.hasOwnProperty(slotName)) continue; var slot = slots[slotName]; var defaultPlaceholders = get(slot.placeholders, "default"); if (!defaultPlaceholders) continue; for (var skinName in slot.placeholders) { if (!slot.placeholders.hasOwnProperty(skinName)) continue; var placeholders = slot.placeholders[skinName]; if (stripName(skinName) == "default") continue; for (var placeholderName in placeholders) { if (!placeholders.hasOwnProperty(placeholderName)) continue; var existing = get(defaultPlaceholders, stripName(placeholderName)); if (existing) { var layer = placeholders[placeholderName]; remove(slotDuplicates, layer.slotName, existing); add(slotDuplicates, layer.slotName, existing); add(slotDuplicates, layer.slotName, layer); } } } } for (var slotName in slotDuplicates) { if (!slotDuplicates.hasOwnProperty(slotName)) continue; var layers = slotDuplicates[slotName]; var message = "Multiple layers for the \"" + layers[0].slotName + "\" slot have the same name \"" + layers[0].placeholderName + "\":\n"; for (var i = 0, n = layers.length; i < n; i++) message += "\n" + layers[i].path(); error(message + "\n\nRename or use the [ignore] tag for these layers."); } var n = errors.length; if (n) { var first = errors[0]; var file; if (n > 1) { try { var all = ""; for (var i = 0; i < n; i++) { if (i > 0) all += "---\n"; all += errors[i].replace(/\n\n/g, "\n") + "\n"; } file = new File(jsonFile.parent + "/errors.txt"); file.parent.create(); file.encoding = "UTF-8"; file.remove(); file.open("w", "TEXT"); file.lineFeed = "\n"; file.write(all); file.close(); if (n == 2) first += "\n\nSee errors.txt for 1 additional error."; else first += "\n\nSee errors.txt for " + (n - 1) + " additional errors."; } catch (e) { if (n == 2) first += "\n\nUnable to write 1 additional error to errors.text.\n"+e; else first += "\n\nUnable to write " + (n - 1) + " additional errors to errors.txt.\n"+e; } } alert(first); if (file) file.execute(); return; } // Add a history item to prevent layer visibility from changing by restoreHistory. topLayer.name = "Processing layers..."; showProgress("Processing layers...", totalLayerCount); // Output skins. var jsonSkins = "", layerCount = 0, writeImages = settings.imagesDir; for (var skinName in skins) { if (!skins.hasOwnProperty(skinName)) continue; var skinSlots = skins[skinName]; skinName = stripName(skinName); var jsonSkin = ""; for (var slotName in skinSlots) { if (!skinSlots.hasOwnProperty(slotName)) continue; var slot = slots[slotName]; var bone = slot.bone; var skinLayers = skinSlots[slotName]; slotName = stripName(slotName); var jsonSlot = ""; for (var i = skinLayers.length - 1; i >= 0; i--) { layerCount++; var layer = skinLayers[i]; layer.setVisible(true); incrProgress(layer.name); if (cancel) return; var attachmentName = layer.attachmentName, attachmentPath = layer.attachmentPath, placeholderName = layer.placeholderName; var scale = layer.scale; if (layer.isGroup) { layer.select(); merge(); layer = new Layer(layer.id, layer.parent); } layer.rasterizeStyles(); var width = layer.width, height = layer.height; if (!width || !height) { layer.deleteLayer(); continue; } slot.attachments = true; if (writeImages) storeHistory(); var x = layer.left, y = layer.top; if (settings.trimWhitespace) { if (writeImages) activeDocument.crop([x - xOffSet, y - yOffSet, layer.right - xOffSet, layer.bottom - yOffSet], 0, width, height); x *= settings.scale; y *= settings.scale; } else { x = 0; y = 0; width = activeDocument.width.as("px") - xOffSet * settings.scale; height = activeDocument.height.as("px") - yOffSet * settings.scale; } width = width * settings.scale + settings.padding * 2; height = height * settings.scale + settings.padding * 2; // Save image. if (writeImages) { scaleImage(settings.scale * scale); if (settings.padding > 0) activeDocument.resizeCanvas(width * scale, height * scale, AnchorPosition.MIDDLECENTER); var file = new File(imagesDir + attachmentPath + ".png"); file.parent.create(); savePNG(file); restoreHistory(); } if (layerCount < totalLayerCount) layer.deleteLayer(); x += Math.round(width) / 2 - settings.padding; y = activeDocument.height.as("px") - (y + Math.round(height) / 2 - settings.padding); width *= scale; height *= scale; // Make relative to the Photoshop document ruler origin. x -= xOffSet * settings.scale; y -= (activeDocument.height.as("px") - yOffSet) * settings.scale; if (bone) { // Make relative to parent bone. x -= bone.x; y -= bone.y; } jsonSlot += "\t\t\t" + quote(placeholderName) + ': { '; if (attachmentName != placeholderName) jsonSlot += '"name": ' + quote(attachmentName) + ', '; if (attachmentName != attachmentPath) jsonSlot += '"path": ' + quote(attachmentPath) + ', '; jsonSlot += '"x": ' + x + ', "y": ' + y + ', "width": ' + Math.round(width) + ', "height": ' + Math.round(height); if (scale != 1) jsonSlot += ', "scaleX": ' + (1 / scale) + ', "scaleY": ' + (1 / scale); jsonSlot += ' },\n'; } if (jsonSlot) jsonSkin += '\t\t' + quote(slotName) + ': {\n' + jsonSlot.substring(0, jsonSlot.length - 2) + '\n\t\t\},\n'; } if (jsonSkin) jsonSkins += '\t"' + skinName + '": {\n' + jsonSkin.substring(0, jsonSkin.length - 2) + '\n\t},\n'; } activeDocument.close(SaveOptions.DONOTSAVECHANGES); // Output skeleton. var json = '{ "skeleton": { "images": "' + imagesDir + '" },\n"bones": [\n'; // Output bones. function outputBone (bone) { var json = bone.parent ? ',\n' : ''; json += '\t{ "name": ' + quote(bone.name); var x = bone.x, y = bone.y; if (bone.parent) { x -= bone.parent.x; y -= bone.parent.y; json += ', "parent": ' + quote(bone.parent.name); } if (x) json += ', "x": ' + x; if (y) json += ', "y": ' + y; json += ' }'; for (var i = 0, n = bone.children.length; i < n; i++) json += outputBone(bone.children[i]); return json; } for (var boneName in bones) { if (cancel) return; if (!bones.hasOwnProperty(boneName)) continue; var bone = bones[boneName]; if (!bone.parent) json += outputBone(bone); } json += '\n],\n"slots": [\n'; // Output slots. var slotIndex = 0; for (var slotName in slots) { if (cancel) return; if (!slots.hasOwnProperty(slotName)) continue; var slot = slots[slotName]; if (!slot.attachments) continue; slotName = stripName(slotName); json += '\t{ "name": ' + quote(slotName) + ', "bone": ' + quote(slot.bone ? slot.bone.name : "root"); if (slot.attachment) json += ', "attachment": ' + quote(slot.attachment); if (slot.blend) json += ', "blend": ' + quote(slot.blend); json += ' }'; slotIndex++; json += slotIndex < slotsCount ? ',\n' : '\n'; } json += '],\n'; if (jsonSkins) json += '"skins": {\n' + jsonSkins.substring(0, jsonSkins.length - 2) + '\n},\n'; json += '"animations": { "animation": {} }\n}'; // Output JSON file. if (settings.writeJson && settings.jsonPath) { if (cancel) return; jsonFile.encoding = "UTF-8"; jsonFile.remove(); jsonFile.open("w", "TEXT"); jsonFile.lineFeed = "\n"; jsonFile.write(json); jsonFile.close(); } } // Settings dialog: function showSettingsDialog () { if (parseInt(app.version) < 9) { alert("Photoshop CS2 or later is required."); return; } if (!originalDoc) { alert("Please open a document before running the PhotoshopToSpine script."); return; } try { decodeURI(activeDocument.path); } catch (e) { alert("Please save the document before running the PhotoshopToSpine script."); return; } // Layout. var dialog, group; try { dialog = new Window("dialog", "PhotoshopToSpine v" + scriptVersion); } catch (e) { throw new Error("\n\nScript is unable to create a Window. Your Photoshop installation may be broken and may need to be reinstalled.\n\n" + e.message); } dialog.alignChildren = "fill"; try { dialog.add("image", undefined, new File(scriptDir() + "logo.png")); } catch (ignored) {} var settingsGroup = dialog.add("panel", undefined, "Settings"); settingsGroup.margins = [10,15,10,10]; settingsGroup.alignChildren = "fill"; var checkboxGroup = settingsGroup.add("group"); checkboxGroup.alignChildren = ["left", ""]; checkboxGroup.orientation = "row"; group = checkboxGroup.add("group"); group.orientation = "column"; group.alignChildren = ["left", ""]; var ignoreHiddenLayersCheckbox = group.add("checkbox", undefined, " Ignore hidden layers"); ignoreHiddenLayersCheckbox.value = settings.ignoreHiddenLayers; var ignoreBackgroundCheckbox = group.add("checkbox", undefined, " Ignore background layer"); ignoreBackgroundCheckbox.value = settings.ignoreBackground; var trimWhitespaceCheckbox = group.add("checkbox", undefined, " Trim whitespace"); trimWhitespaceCheckbox.value = settings.trimWhitespace; group = checkboxGroup.add("group"); group.orientation = "column"; group.alignChildren = ["left", ""]; group.alignment = ["", "top"]; var writeJsonCheckbox = group.add("checkbox", undefined, " Write Spine JSON"); writeJsonCheckbox.value = settings.writeJson; var writeTemplateCheckbox = group.add("checkbox", undefined, " Write template image"); writeTemplateCheckbox.value = settings.writeTemplate; var scaleText, paddingText, scaleSlider, paddingSlider; if (!cs2) { var slidersGroup = settingsGroup.add("group"); group = slidersGroup.add("group"); group.orientation = "column"; group.alignChildren = ["right", ""]; group.add("statictext", undefined, "Scale:"); group.add("statictext", undefined, "Padding:"); group = slidersGroup.add("group"); group.orientation = "column"; scaleText = group.add("edittext", undefined, settings.scale * 100); scaleText.characters = 4; paddingText = group.add("edittext", undefined, settings.padding); paddingText.characters = 4; group = slidersGroup.add("group"); group.orientation = "column"; group.add("statictext", undefined, "%"); group.add("statictext", undefined, "px"); group = slidersGroup.add("group"); group.orientation = "column"; group.alignChildren = ["fill", ""]; group.alignment = ["fill", ""]; scaleSlider = group.add("slider", undefined, settings.scale * 100, 1, 100); paddingSlider = group.add("slider", undefined, settings.padding, 0, 4); } else { group = settingsGroup.add("group"); group.add("statictext", undefined, "Scale:"); scaleText = group.add("edittext", undefined, settings.scale * 100); scaleText.preferredSize.width = 50; scaleSlider = settingsGroup.add("slider", undefined, settings.scale * 100, 1, 100); group = settingsGroup.add("group"); group.add("statictext", undefined, "Padding:"); paddingText = group.add("edittext", undefined, settings.padding); paddingText.preferredSize.width = 50; paddingSlider = settingsGroup.add("slider", undefined, settings.padding, 0, 4); } if (cs2) { ignoreHiddenLayersCheckbox.preferredSize.width = 150; ignoreBackgroundCheckbox.preferredSize.width = 150; trimWhitespaceCheckbox.preferredSize.width = 150; writeJsonCheckbox.preferredSize.width = 150; writeTemplateCheckbox.preferredSize.width = 150; } var outputPathGroup = dialog.add("panel", undefined, "Output Paths"); outputPathGroup.alignChildren = ["fill", ""]; outputPathGroup.margins = [10,15,10,10]; var imagesDirText, imagesDirPreview, jsonPathText, jsonPathPreview; if (!cs2) { var textGroup = outputPathGroup.add("group"); textGroup.orientation = "column"; textGroup.alignChildren = ["fill", ""]; group = textGroup.add("group"); group.add("statictext", undefined, "Images:"); imagesDirText = group.add("edittext", undefined, settings.imagesDir); imagesDirText.alignment = ["fill", ""]; imagesDirPreview = textGroup.add("statictext", undefined, ""); imagesDirPreview.maximumSize.width = 260; group = textGroup.add("group"); var jsonLabel = group.add("statictext", undefined, "JSON:"); jsonLabel.justify = "right"; jsonLabel.minimumSize.width = 41; jsonPathText = group.add("edittext", undefined, settings.jsonPath); jsonPathText.alignment = ["fill", ""]; jsonPathPreview = textGroup.add("statictext", undefined, ""); jsonPathPreview.maximumSize.width = 260; } else { outputPathGroup.add("statictext", undefined, "Images:"); imagesDirText = outputPathGroup.add("edittext", undefined, settings.imagesDir); imagesDirText.alignment = "fill"; outputPathGroup.add("statictext", undefined, "JSON:"); jsonPathText = outputPathGroup.add("edittext", undefined, settings.jsonPath); jsonPathText.alignment = "fill"; } var buttonGroup = dialog.add("group"); var helpButton; if (!cs2) helpButton = buttonGroup.add("button", undefined, "Help"); group = buttonGroup.add("group"); group.alignment = ["fill", ""]; group.alignChildren = ["right", ""]; var runButton = group.add("button", undefined, "OK"); var cancelButton = group.add("button", undefined, "Cancel"); // Tooltips. writeTemplateCheckbox.helpTip = "When checked, a PNG is written for the currently visible layers."; writeJsonCheckbox.helpTip = "When checked, a Spine JSON file is written."; trimWhitespaceCheckbox.helpTip = "When checked, blank pixels around the edges of each image are removed."; scaleSlider.helpTip = "Scales the PNG files. Useful when using higher resolution art in Photoshop than in Spine."; paddingSlider.helpTip = "Blank pixels around the edge of each image. Can avoid aliasing artifacts for opaque pixels along the image edge."; imagesDirText.helpTip = "The folder to write PNGs. Begin with \"./\" to be relative to the PSD file. Blank to disable writing PNGs."; jsonPathText.helpTip = "Output JSON file if ending with \".json\", else the folder to write the JSON file. Begin with \"./\" to be relative to the PSD file. Blank to disable writing a JSON file."; // Events. scaleText.onChanging = function () { scaleSlider.value = scaleText.text; }; scaleSlider.onChanging = function () { scaleText.text = Math.round(scaleSlider.value); }; paddingText.onChanging = function () { paddingSlider.value = paddingText.text; }; paddingSlider.onChanging = function () { paddingText.text = Math.round(paddingSlider.value); }; cancelButton.onClick = function () { cancel = true; dialog.close(); return; }; if (!cs2) helpButton.onClick = showHelpDialog; jsonPathText.onChanging = function () { var text = jsonPathText.text ? jsonPath(jsonPathText.text) : ""; if (!cs2) { jsonPathPreview.text = text; jsonPathPreview.helpTip = text; } else jsonPathText.helpTip = text; }; imagesDirText.onChanging = function () { var text = imagesDirText.text ? absolutePath(imagesDirText.text) : ""; if (!cs2) { imagesDirPreview.text = text; imagesDirPreview.helpTip = text; } else imagesDirText.helpTip = text; }; // Run now. jsonPathText.onChanging(); imagesDirText.onChanging(); function updateSettings () { settings.ignoreHiddenLayers = ignoreHiddenLayersCheckbox.value; settings.ignoreBackground = ignoreBackgroundCheckbox.value; settings.writeTemplate = writeTemplateCheckbox.value; settings.writeJson = writeJsonCheckbox.value; settings.trimWhitespace = trimWhitespaceCheckbox.value; var scaleValue = parseFloat(scaleText.text); if (scaleValue > 0 && scaleValue <= 100) settings.scale = scaleValue / 100; settings.imagesDir = imagesDirText.text; settings.jsonPath = jsonPathText.text; var paddingValue = parseInt(paddingText.text); if (paddingValue >= 0) settings.padding = paddingValue; } runButton.onClick = function () { if (scaleText.text <= 0 || scaleText.text > 100) { alert("Scale must be between > 0 and <= 100."); return; } if (paddingText.text < 0) { alert("Padding must be >= 0."); return; } updateSettings(); saveSettings(); ignoreHiddenLayersCheckbox.enabled = false; ignoreBackgroundCheckbox.enabled = false; writeTemplateCheckbox.enabled = false; writeJsonCheckbox.enabled = false; trimWhitespaceCheckbox.enabled = false; scaleText.enabled = false; scaleSlider.enabled = false; paddingText.enabled = false; paddingSlider.enabled = false; imagesDirText.enabled = false; jsonPathText.enabled = false; if (!cs2) helpButton.enabled = false; runButton.enabled = false; cancelButton.enabled = false; var rulerUnits = app.preferences.rulerUnits; app.preferences.rulerUnits = Units.PIXELS; try { //var start = new Date().getTime(); run(); //alert((new Date().getTime() - start) / 1000 + "s"); } catch (e) { if (e.message == "User cancelled the operation") return; alert("An unexpected error has occurred:\n\n[line " + e.line + "] " + e.message + "\n\nTo debug, run the PhotoshopToSpine script using Adobe ExtendScript " + "with \"Debug > Do not break on guarded exceptions\" unchecked."); debugger; } finally { if (activeDocument != originalDoc) activeDocument.close(SaveOptions.DONOTSAVECHANGES); app.preferences.rulerUnits = rulerUnits; if (progress && progress.dialog) progress.dialog.close(); dialog.close(); } }; dialog.center(); dialog.show(); } function loadSettings () { var options; try { options = app.getCustomOptions(sID("settings")); } catch (ignored) {} settings = {}; for (var key in defaultSettings) { if (!defaultSettings.hasOwnProperty(key)) continue; var typeID = sID(key); if (options && options.hasKey(typeID)) settings[key] = options["get" + getOptionType(defaultSettings[key])](typeID); else settings[key] = defaultSettings[key]; } } function saveSettings () { if (cs2) return; // No putCustomOptions. var desc = new ActionDescriptor(); for (var key in defaultSettings) { if (!defaultSettings.hasOwnProperty(key)) continue; desc["put" + getOptionType(defaultSettings[key])](sID(key), settings[key]); } app.putCustomOptions(sID("settings"), desc, true); } function getOptionType (value) { switch (typeof(value)) { case "boolean": return "Boolean"; case "string": return "String"; case "number": return "Double"; }; throw new Error("Invalid default setting: " + value); } // Help dialog. function showHelpDialog () { var dialog = new Window("dialog", "PhotoshopToSpine - Help"); dialog.alignChildren = ["fill", ""]; dialog.orientation = "column"; dialog.alignment = ["", "top"]; var helpText = dialog.add("statictext", undefined, "" + "This script writes layers as images and creates a JSON file to bring the images into Spine with the same positions and draw order they had in Photoshop.\n" + "\n" + "The Photoshop ruler origin corresponds to 0,0 in Spine.\n" + "\n" + "Tags in square brackets can be used anywhere in layer and group names to customize the output. If \":name\" is omitted, the layer or group name is used.\n" + "\n" + "Group and layer names:\n" + "• [bone] or [bone:name] Layers, slots, and bones are placed under a bone. The bone is created at the center of a visible layer. Bone groups can be nested.\n" + "• [slot] or [slot:name] Layers are placed in a slot.\n" + "• [skin] or [skin:name] Layers are placed in a skin. Skin layer images are output in a subfolder for the skin.\n" + "• [scale:number] Layers are scaled. Their attachments are scaled inversely, so they appear the same size in Spine.\n" + "• [folder] or [folder:name] Layers images are output in a subfolder. Folder groups can be nested.\n" + "• [ignore] Layers, groups, and any child groups will not be output.\n" + "\n" + "Group names:\n" + "• [merge] Layers in the group are merged and a single image is output.\n" + "\n" + "Layer names:\n" + "• The layer name is used for the attachment or skin placeholder name, relative to any parent [skin] or [folder] groups. Can contain / for subfolders.\n" + "• [path:name] Specifies the image file name, if it needs to be different from the attachment name. Can be used on a group with [merge].\n" + "\n" + "If a layer name, folder name, or path name starts with / then parent layers won't affect the name." , {multiline: true}); helpText.preferredSize.width = 325; var closeButton = dialog.add("button", undefined, "Close"); closeButton.alignment = ["center", ""]; closeButton.onClick = function () { dialog.close(); }; dialog.center(); dialog.show(); } // Progress dialog: function showProgress (title, total) { title = title ? "PhotoshopToSpine - " + title : "PhotoshopToSpine"; if (!progress) { var dialog = new Window("palette", title); dialog.alignChildren = "fill"; dialog.orientation = "column"; var message = dialog.add("statictext", undefined, "Initializing..."); var group = dialog.add("group"); var bar = group.add("progressbar"); bar.preferredSize = [300, 16]; bar.maxvalue = total; bar.value = 1; var cancelButton = group.add("button", undefined, "Cancel"); cancelButton.onClick = function () { cancel = true; cancelButton.enabled = false; return; }; dialog.center(); dialog.show(); dialog.active = true; progress = { dialog: dialog, bar: bar, message: message }; } else { progress.dialog.text = title; progress.bar.maxvalue = total; } progress.count = 0; progress.total = total; progress.updateTime = 0; var reset = $.hiresTimer; } function incrProgress (text) { progress.count++; if (progress.count != 1 && progress.count < progress.total) { progress.updateTime += $.hiresTimer; if (progress.updateTime < 500000) return; progress.updateTime = 0; } text = progress.count + " / "+ progress.total + ": " + trim(text); progress.bar.value = progress.count; progress.message.text = text; if (!progress.dialog.active) progress.dialog.active = true; } // PhotoshopToSpine utility: function initializeLayers (context, parent, parentLayers) { while (context.index >= context.first) { if (cancel) return -1; var layer = new Layer(getLayerID(context.index--), parent); if (layer.isGroupEnd) break; context.total++; parentLayers.push(layer); if (layer.isGroup) initializeLayers(context, layer, layer.layers); } } function collectLayers (parentLayers, collect) { outer: for (var i = parentLayers.length - 1; i >= 0; i--) { if (cancel) return; var layer = parentLayers[i]; incrProgress(layer.name); if (settings.ignoreHiddenLayers && !layer.visible) continue; if (settings.ignoreBackground && layer.background) { layer.deleteLayer(); continue; } if (layer.findTagLayer("ignore")) { layer.deleteLayer(); continue; } if (!layer.isGroup && !layer.isNormal()) { layer.rasterize(); // In case rasterizeAll failed. if (!layer.isNormal()) { layer.deleteLayer(); continue; } } // Ensure tags are valid. var re = /\[([^\]]+)\]/g; while (true) { var matches = re.exec(layer.name); if (!matches) break; var tag = matches[1]; if (layer.isGroup) { if (!isValidGroupTag(tag)) { var message = "Invalid group name:\n\n" + layer.name; if (isValidLayerTag(tag)) message += "\n\nThe [" + tag + "] tag is only valid for layers, not for groups."; else message += "\n\nThe [" + tag + "] tag is not a valid tag."; error(message); continue outer; } } else if (tag != "merge" && !isValidLayerTag(tag)) { // Allow merge, the user may have merged manually to save time. var message = "Invalid layer name:\n\n" + layer.name; if (isValidGroupTag(tag)) message += "\n\nThe [" + tag + "] tag is only valid for groups, not for layers."; else message += "\n\nThe [" + tag + "] tag is not a valid tag."; error(message); continue outer; } } layer.wasVisible = layer.visible; layer.setVisible(true); layer.setLocked(false); if (layer.isGroup && layer.findTagLayer("merge")) { collectGroupMerge(layer); if (!layer.layers || layer.layers.length == 0) continue; } else if (layer.layers && layer.layers.length > 0) { collectLayers(layer.layers, collect); continue; } else if (layer.isGroup) continue; layer.setVisible(false); collect.push(layer); } } function collectGroupMerge (parent) { var parentLayers = parent.layers; if (!parentLayers) return; for (var i = parentLayers.length - 1; i >= 0; i--) { var layer = parentLayers[i]; if (settings.ignoreHiddenLayers && !layer.visible) continue; if (layer.findTagLayer("ignore")) { layer.deleteLayer(); continue; } collectGroupMerge(layer); } } function isValidGroupTag (tag) { return isValidLayerTag(tag) || tag == "merge"; } function isValidLayerTag (tag) { switch (tag) { case "bone": case "slot": case "skin": case "folder": case "ignore": return true; } if (startsWith(tag, "bone:")) return true; if (startsWith(tag, "slot:")) return true; if (startsWith(tag, "skin:")) return true; if (startsWith(tag, "folder:")) return true; if (startsWith(tag, "path:")) return true; if (startsWith(tag, "scale:")) return true; return false; } function stripTags (name) { return trim(name.replace(/\[[^\]]+\]/g, "")); } function jsonPath (jsonPath) { if (endsWith(jsonPath, ".json")) { var index = jsonPath.replace(/\\/g, "/").lastIndexOf("/"); if (index != -1) return absolutePath(jsonPath.slice(0, index + 1)) + jsonPath.slice(index + 1); return absolutePath("./") + jsonPath; } var name = decodeURI(originalDoc.name); return absolutePath(jsonPath) + name.substring(0, name.indexOf(".")) + ".json"; } function error (message) { errors.push(message); } // Photoshop utility: function get (object, name) { return object["_" + name]; } function set (object, name, value) { object["_" + name] = value; } function add (object, name, value) { var array = object["_" + name]; if (!array) object["_" + name] = array = []; array[array.length] = value; return array; } function remove (object, name, value) { var array = object["_" + name]; if (!array) return; for (var i = 0, n = array.length; i < n; i++) { if (array[i] == value) { array.splice(i, 1); return; } } } function stripName (name) { return name.substring(1); } function rulerOrigin (axis) { var key = cID("Rlr" + axis); var ref = new ActionReference(); ref.putProperty(cID("Prpr"), key); ref.putEnumerated(cID("Dcmn"), cID("Ordn"), cID("Trgt")); return executeActionGet(ref).getInteger(key) >> 16; } // Seems to not be available when the document has >= 500 layers. function rasterizeAll () { try { executeAction(sID("rasterizeAll"), undefined, DialogModes.NO); } catch (ignored) {} } // Layer must be selected. function newLayerBelow (name) { var ref = new ActionReference(); ref.putClass(cID("Lyr ")); var desc2 = new ActionDescriptor(); desc2.putString(cID("Nm "), name); var desc1 = new ActionDescriptor(); desc1.putReference(cID("null"), ref); desc1.putBoolean(sID("below"), true); desc1.putObject(cID("Usng"), cID("Lyr "), desc2); executeAction(cID("Mk "), desc1, DialogModes.NO); } // Layer must be selected. function merge () { executeAction(cID("Mrg2"), undefined, DialogModes.NO); } function scaleImage (scale) { if (scale == 1) return; var imageSize = activeDocument.width.as("px") * scale; activeDocument.resizeImage(UnitValue(imageSize, "px"), null, null, ResampleMethod.BICUBICSHARPER); } var history; function storeHistory () { history = activeDocument.activeHistoryState; } function restoreHistory () { activeDocument.activeHistoryState = history; } function scriptDir () { var file; if (!cs2) file = $.fileName; else { try { var error = THROW_ERROR; // Force error which provides the script file name. } catch (e) { file = e.fileName; } } return new File(file).parent + "/"; } function absolutePath (path) { path = trim(path); if (!startsWith(path, "./")) { var absolute = decodeURI(new File(path).absoluteURI); if (!startsWith(absolute, decodeURI(new File("child").parent.absoluteURI))) return absolute + "/"; path = "./" + path; } if (path.length == 0) path = decodeURI(activeDocument.path); else if (startsWith(path, "./")) path = decodeURI(activeDocument.path) + path.substring(1); path = (new File(path).fsName).toString(); path = path.replace(/\\/g, "/"); if (path.substring(path.length - 1) != "/") path += "/"; return path; } function bgColor (control, r, g, b) { control.graphics.backgroundColor = control.graphics.newBrush(control.graphics.BrushType.SOLID_COLOR, [r, g, b]); } function deselectLayers () { var ref = new ActionReference(); ref.putEnumerated(cID("Lyr "), cID("Ordn"), cID("Trgt")); var desc = new ActionDescriptor(); desc.putReference(cID("null"), ref); executeAction(sID("selectNoLayers"), desc, DialogModes.NO); } function convertToRGB () { var desc = new ActionDescriptor(); desc.putClass(cID("T "), cID("RGBM")); desc.putBoolean(cID("Mrge"), false); desc.putBoolean(cID("Rstr"), true); executeAction(cID("CnvM"), desc, DialogModes.NO); } function deleteDocumentAncestorsMetadata () { if (ExternalObject.AdobeXMPScript == undefined) ExternalObject.AdobeXMPScript = new ExternalObject("lib:AdobeXMPScript"); app.activeDocument.xmpMetadata.rawData = new XMPMeta().serialize(); } function savePNG (file) { // SaveForWeb changes spaces to dash. Also some users report it writes HTML. //var options = new ExportOptionsSaveForWeb(); //options.format = SaveDocumentType.PNG; //options.PNG8 = false; //options.transparency = true; //options.interlaced = false; //options.includeProfile = false; //activeDocument.exportDocument(file, ExportType.SAVEFORWEB, options); // SaveAs sometimes writes a huge amount of XML in the PNG. Ignore it or use Oxipng to make smaller PNGs. var options = new PNGSaveOptions(); options.compression = 6; activeDocument.saveAs(file, options, true, Extension.LOWERCASE); } function getLayerCount () { var ref = new ActionReference(); ref.putProperty(cID("Prpr"), sID("numberOfLayers")); ref.putEnumerated(cID("Dcmn"), cID("Ordn"), cID("Trgt")); return executeActionGet(ref).getInteger(sID("numberOfLayers")); } function getLayerID (index) { var ref = new ActionReference(); ref.putProperty(cID("Prpr"), sID("layerID")); ref.putIndex(cID("Lyr "), index); return executeActionGet(ref).getInteger(sID("layerID")); } function hasBackgroundLayer () { try { var ref = new ActionReference () ref.putEnumerated(sID("document"), sID("ordinal"), sID("targetEnum")); return executeActionGet(ref).getBoolean(sID("hasBackgroundLayer")); } catch (e) { // CS2. try { return activeDocument.backgroundLayer; } catch (ignored) { } return false; } } function typeToMethod (type) { if (type == "DescValueType.ENUMERATEDTYPE") return "EnumerationValue"; if (type == "DescValueType.OBJECTTYPE") return "ObjectValue"; if (type == "DescValueType.UNITDOUBLE") return "Double"; if (type == "DescValueType.INTEGERTYPE") return "Integer"; if (type == "DescValueType.STRINGTYPE") return "String"; if (type == "DescValueType.BOOLEANTYPE") return "Boolean"; if (type == "DescValueType.LISTTYPE") return "List"; if (type == "DescValueType.REFERENCETYPE") return "Reference"; throw new Error("Unknown type: " + type); } // Example: // var ref = new ActionReference(); // ref.putIdentifier(cID("Lyr "), layer.id); // alert(properties(executeActionGet(ref))); function properties (object, indent) { if (!indent) indent = 0; var text = ""; for (var i = 0, n = object.count; i < n; i++) { var key = object.getKey(i); var type = typeToMethod(object.getType(key)); var value = object["get" + type](key); if (type == "EnumerationValue") value = typeIDToStringID(value); else if (type == "ObjectValue") value = "{\n" + properties(value, indent + 1) + "}"; else if (type == "List") { var items = ""; for (var ii = 0, nn = value.count; ii < nn; ii++) { var itemType = typeToMethod(value.getType(ii)); items += properties(value["get" + itemType](ii), indent + 1); } if (items) items = "\n" + items; value = "[" + items + "]"; } for (var ii = 0; ii < indent; ii++) text += " "; text += typeIDToStringID(key) + ": " + value + " (" + type + ")\n"; } return text; } // Layer class. function Layer (id, parent) { this.id = id; this.parent = parent; this.name = this.get("name", "String"); var type = typeIDToStringID(this.get("layerSection", "EnumerationValue")); this.isGroupEnd = type == "layerSectionEnd"; if (this.isGroupEnd) return; this.isGroup = type == "layerSectionStart"; this.isLayer = type == "layerSectionContent"; this.visible = this.get("visible", "Boolean"); this.background = this.get("background", "Boolean"); this.locked = this.get("layerLocking", "ObjectValue").getBoolean(sID("protectAll")); this.blendMode = typeIDToStringID(this.get("mode", "EnumerationValue")); var bounds = this.get("bounds", "ObjectValue"); this.top = bounds.getDouble(sID("top")); this.left = bounds.getDouble(sID("left")); this.bottom = bounds.getDouble(sID("bottom")); this.right = bounds.getDouble(sID("right")); this.width = this.right - this.left; this.height = this.bottom - this.top; if (this.isGroup) this.layers = []; } Layer.prototype.get = function (name, type) { var property = sID(name); var ref = new ActionReference(); ref.putProperty(cID("Prpr"), property); ref.putIdentifier(cID("Lyr "), this.id); try { return executeActionGet(ref)["get" + type](property); } catch (e) { e.message = "Unable to get layer " + this + " property: " + name + "\n" + e.message; throw e; } }; Layer.prototype.has = function (name) { var property = sID(name); var ref = new ActionReference(); ref.putProperty(cID("Prpr"), property); ref.putIdentifier(cID("Lyr "), this.id); try { return executeActionGet(ref).hasKey(property); } catch (ignored) {} return false; }; Layer.prototype.isNormal = function () { if (cs2) return !this.has("smartObject"); return this.get("layerKind", "Integer") == 1; } Layer.prototype.setVisible = function (visible) { if (this.visible == visible) return; this.visible = visible; var ref = new ActionReference(); ref.putIdentifier(cID("Lyr "), this.id); var desc = new ActionDescriptor(); desc.putReference(cID("null"), ref); executeAction(cID(visible ? "Shw " : "Hd "), desc, DialogModes.NO); }; Layer.prototype.setLocked = function (locked) { if (this.locked == locked) return; this.locked = locked; var desc1 = new ActionDescriptor(); var ref = new ActionReference(); ref.putIdentifier(cID("Lyr "), this.id); desc1.putReference(cID("null"), ref); var desc2 = new ActionDescriptor(); desc2.putBoolean(sID("protectNone"), true); desc1.putObject(sID("layerLocking"), sID("layerLocking"), desc2); executeAction(sID("applyLocking"), desc1, DialogModes.NO); }; Layer.prototype.unlock = function () { this.setLocked(false); if (!this.layers) return; for (var i = this.layers.length - 1; i >= 0; i--) this.layers[i].unlock(); }; Layer.prototype.deleteLayer = function () { this.unlock(); var ref = new ActionReference(); ref.putIdentifier(cID("Lyr "), this.id); var desc = new ActionDescriptor(); desc.putReference(cID("null"), ref); executeAction(cID("Dlt "), desc, DialogModes.NO); }; Layer.prototype.rasterize = function () { var ref = new ActionReference(); ref.putIdentifier(cID("Lyr "), this.id); var desc = new ActionDescriptor(); desc.putReference(cID("null"), ref); executeAction(sID("rasterizeLayer"), desc, DialogModes.NO); }; Layer.prototype.rasterizeStyles = function () { if (!this.has("layerEffects")) return; this.select(); newLayerBelow(this.name); this.select(true); merge(); // Rasterizing styles may not give the desired results in all cases, merge does. //var ref = new ActionReference(); //ref.putProperty(cID("Prpr"), sID("layerEffects")); //ref.putIdentifier(cID("Lyr "), this.id); //if (executeActionGet(ref).hasKey(sID("layerEffects"))) { // var desc = new ActionDescriptor(); // desc.putReference(cID("null"), ref); // desc.putEnumerated(cID("What"), sID("rasterizeItem"), sID("layerStyle")); // executeAction(sID("rasterizeLayer"), desc, DialogModes.NO); //} }; Layer.prototype.select = function (add) { var ref = new ActionReference(); ref.putIdentifier(cID("Lyr "), this.id); var desc = new ActionDescriptor(); desc.putReference(cID("null"), ref); if (add) desc.putEnumerated(sID("selectionModifier"), sID("selectionModifierType"), sID("addToSelection")); desc.putBoolean(cID("MkVs"), false); executeAction(cID("slct"), desc, DialogModes.NO); } Layer.prototype.findTagLayer = function (tag) { var groupTag = isValidGroupTag(tag), layerTag = isValidLayerTag(tag); if (endsWith(tag, ":")) tag = tag.slice(0, -1); var re = new RegExp("\\[" + tag + "(:[^\\]]+)?\\]", "i"); var layer = this; while (layer) { if (((layer.isGroup && groupTag) || (layer.isLayer && layerTag)) && re.exec(layer.name)) return layer; layer = layer.parent; } return null; }; Layer.prototype.findTagValue = function (tag) { var layer = this.findTagLayer(tag); if (!layer) return null; if (endsWith(tag, ":")) tag = tag.slice(0, -1); var matches = new RegExp("\\[" + tag + ":([^\\]]+)\\]", "i").exec(layer.name); if (matches && matches.length) return trim(matches[1]); return stripTags(layer.name); }; Layer.prototype.getParentBone = function (bones) { var parentName = this.parent ? this.parent.findTagValue("bone") : null; if (!parentName) parentName = "root"; var parent = get(bones, parentName); if (!parent) { // Parent bone group with no attachment layers. var parentParent = this.parent.getParentBone(bones); set(bones, parentName, parent = { name: parentName, x: 0, y: 0, parent: parentParent, children: [], layer: this.parent }); parentParent.children.push(parent); } return parent; }; var foldersRE = new RegExp("\\[(folder|skin)(:[^\\]]+)?\\]", "i"); Layer.prototype.folders = function (path) { var layer = this; while (layer) { var matches = foldersRE.exec(layer.name); if (matches) { var folder = layer.findTagValue(matches[1]); if (matches[1] == "skin" && folder == "default") return layer.parent.folders(path); path = folder + "/" + path; if (startsWith(folder, "/")) return path; return layer.parent ? layer.parent.folders(folder + "/" + path) : path; } layer = layer.parent; } return path; }; Layer.prototype.path = function (path) { var layer = this; var path = layer.name; while (true) { layer = layer.parent; if (!layer) return path; path = layer.name + "/" + path; } }; Layer.prototype.toString = function () { return this.name ? this.path() : this.id; }; // JavaScript utility: function joinKeys (object, glue) { if (!glue) glue = ", "; var value = ""; for (var key in object) { if (object.hasOwnProperty(key)) { if (value) value += glue; value += key; } } return value; } function joinValues (object, glue) { if (!glue) glue = ", "; var value = ""; for (var key in object) { if (object.hasOwnProperty(key)) { if (value) value += glue; value += object[key]; } } return value; } function indexOf (array, value) { for (var i = 0, n = array.length; i < n; i++) if (array[i] == value) return i; return -1; } function trim (value) { return value.replace(/^\s+|\s+$/g, ""); } function startsWith (str, prefix) { return str.indexOf(prefix) === 0; } function endsWith (str, suffix) { return str.indexOf(suffix, str.length - suffix.length) !== -1; } function quote (value) { return '"' + value.replace(/"/g, '\\"') + '"'; } showSettingsDialog();