Copy link to clipboard
Copied
Hello,
I have a question about webp format in photoshop. I want to export many files at once in webp format in Photoshop but I don't find the webp format in photoshop options. Is it normal?
Best regards
WebP was introduced in Ps2022 as a native Save As (a Copy) option, not under Export As/Quick Export. You will either need to set up a batch action to save directly to WebP or to convert from PNG to WebP or use a custom script to save or convert to WebP.
In previous versions, there were 3rd party plugins from Google or other developers.
Here is a batch script to save as WebP.
Features:
* Optionally run an action (edit the code to enable)
* Optionally ask to overwrite existing files (edit the code to enable)
* Set an input and output folder
* Set supported input file types (edit the code)
* Automatically converts non-RGB mode to sRGB space, 8 bpc
* RGB files automatically converted to 8 bpc
* Option to convert RGB mode to sRGB space (edit the code to enable)
* WebP lossy format, 75% quality (larger size), all metadata and PSD
...Copy link to clipboard
Copied
If I do this as a batch action theres no way to set the images for example to be about 50kb upon output is there? The images in my folder that are to be convered range from 90kb to 20mb
By @melajuana
Not with this code.
To hit a target size on drive, the script would need to save, check the size and resave multiple times with decreasing quality levels to try to hit the target. This is certainly possible, however, it's obviously time consuming.
Copy link to clipboard
Copied
So much variables this depends on. First try to make test saves, don't save with extra data options. So no extras, no xnpndata etc. find the correct file size. Than it very much depends per image. The more color data is in your image, the heavier it is. By this I mean. When there are many colors and tints and hues, an image tends to be bigger
Copy link to clipboard
Copied
Your script has been a gift. Thank you @Stephen_A_Marsh .
I have browse almost all the threads about it and didn´t find how could I change the output resolution. Is that possible?
Copy link to clipboard
Copied
Use a script to resize it or use fit command. You can simple record an action for this and than run this script. That's not so hard to make your self. No need for a custom build script for such a small thing
Copy link to clipboard
Copied
Yes, my batch script does offer an option to run a user defined action set/action to provide additional processing without requiring extra code.
Copy link to clipboard
Copied
I've tried, but it seems I can´t make it work. Code is not my thing.
PS stops at the first image without even performing the action.
Copy link to clipboard
Copied
Please clarify which of the following are required:
1) Resize the image, resampling to a desired target size on the longest edge, such as 1920px
2) As #1 but also setting a specific print metadata resolution value such as 300ppi
3) Resize to a target PPI such as 300ppi, without resampling the image pixels
Copy link to clipboard
Copied
@Stephen_A_Marsh Option 2. Thank you.
Copy link to clipboard
Copied
The following changes to the script from page 2 of this thread will fit the image to 1920px on the longest edge, proportionally scaling the short edge and also setting the resolution metadata at 300ppi.
/*
Batch Save As WebP.jsx
https://community.adobe.com/t5/photoshop-ecosystem-discussions/export-many-files-at-once-in-webp-format-photoshop/m-p/13604411
v1.0 - 14th March 2023, Stephen Marsh
v1.1 - 11th January 2024: Added a "fit image" to 1920px step
v1.2 - 10th February 2024: Added an explicit step to change to RGB mode for non-RGB images
*/
#target photoshop
// Optionally run a specified action
//var actionName = "Molten Lead"; // Action to run, change as needed
//var actionSet = "Default Actions"; // Action set to run, change as needed
// Ensure that version 2022 or later is being used
var versionNumber = app.version.split(".");
var versionCheck = parseInt(versionNumber);
// Fail
if (versionCheck < 23) {
alert("You must use Photoshop 2022 or later to save using native WebP format...");
// Pass
} else {
// Set the input and output folders
var inputFolder = Folder.selectDialog("Please select the input folder:");
var outputFolder = Folder.selectDialog("Please select the output folder:");
// Limit the input files, add or remove extensions as required
var fileList = inputFolder.getFiles(/\.(webp|tif|tiff|jpg|jpeg|psd|psb|png)$/i);
fileList.sort();
var savedDisplayDialogs = app.displayDialogs;
app.displayDialogs = DialogModes.NO;
// Set the file processing counter
var fileCounter = 0;
// Process the input files
for (var i = 0; i < fileList.length; i++) {
var doc = open(fileList[i]);
// If the doc isn't in RGB mode
if (activeDocument.mode !== DocumentMode.RGB) {
// Convert to sRGB & 8 bpc
activeDocument.convertProfile("sRGB IEC61966-2.1", Intent.RELATIVECOLORIMETRIC, true, false);
activeDocument.changeMode(ChangeMode.RGB);
activeDocument.bitsPerChannel = BitsPerChannelType.EIGHT;
// Run the optional action
//app.doAction(actionName, actionSet);
// Fit image to 1920px
fitImage(1920, 1920);
// Save as a copy and close
saveWebP("compressionLossy", 75, true, true, true, true);
activeDocument.close(SaveOptions.DONOTSAVECHANGES);
// Increment the file saving counter
fileCounter++;
// If the doc is in RGB mode
} else {
// Convert to sRGB & 8 bpc
//activeDocument.convertProfile("sRGB IEC61966-2.1", Intent.RELATIVECOLORIMETRIC, true, false);
activeDocument.bitsPerChannel = BitsPerChannelType.EIGHT;
// Run the optional action
//app.doAction(actionName, actionSet);
// Fit image to 1920px
fitImage(1920, 1920);
// Save as a copy and close
saveWebP("compressionLossy", 75, true, true, true, true);
activeDocument.close(SaveOptions.DONOTSAVECHANGES);
// Increment the file saving counter
fileCounter++;
}
};
/* NEARESTNEIGHBOR | BILINEAR | BICUBIC | BICUBICSMOOTHER | BICUBICSHARPER | BICUBICAUTOMATIC */
function fitImage(fWidth, fHeight) {
if (activeDocument.height.value > activeDocument.width.value) {
activeDocument.resizeImage(null, UnitValue(fHeight, "px"), 300, ResampleMethod.BICUBIC);
} else {
activeDocument.resizeImage(UnitValue(fWidth, "px"), null, 300, ResampleMethod.BICUBIC);
}
}
app.displayDialogs = savedDisplayDialogs;
alert('Script completed!' + '\n' + fileCounter + ' files saved to:' + '\r' + outputFolder.fsName);
function saveWebP(compType, compValue, xmpData, exifData, psData, asCopy) {
/*
v1.1 - 12th March 2023, Stephen Marsh
https://community.adobe.com/t5/photoshop-ecosystem-discussions/saving-webp-image-by-script/td-p/13642577
*/
// Doc and path save variables
var WebPDocName = activeDocument.name.replace(/\.[^\.]+$/, ''); // Remove file extension
var WebPSavePath = outputFolder + "/" + WebPDocName + ".webp" // Change path as needed
var WebPFile = new File(WebPSavePath); // Create the file object
/*
// Check for existing file object
if (WebPFile.exists) {
// true = 'No' as default active button
if (!confirm("File exists, overwrite: Yes or No?", true))
// throw alert("Script cancelled!");
throw null;
}
*/
function s2t(s) {
return app.stringIDToTypeID(s);
}
var descriptor = new ActionDescriptor();
var descriptor2 = new ActionDescriptor();
// Compression parameters = "compressionLossless" | "compressionLossy"
descriptor2.putEnumerated(s2t("compression"), s2t("WebPCompression"), s2t(compType)); // string variable
var WebPCompIsLossless = false; // set the default flag for compression
if (WebPCompIsLossless == false) {
// 0 (lowest lossy quality) - 100 (highest lossy quality)
descriptor2.putInteger(s2t("quality"), compValue); // number variable
}
// Metadata options
descriptor2.putBoolean(s2t("includeXMPData"), xmpData); // Boolean param moved to function call
descriptor2.putBoolean(s2t("includeEXIFData"), exifData); // Boolean param moved to function call
descriptor2.putBoolean(s2t("includePsExtras"), psData); // Boolean param moved to function call
// WebP format and save path
descriptor.putObject(s2t("as"), s2t("WebPFormat"), descriptor2);
descriptor.putPath(s2t("in"), WebPFile); // Save path variable
// Save As = false | Save As a Copy = true
descriptor.putBoolean(s2t("copy"), asCopy); // Boolean param moved to function call
// The extension
descriptor.putBoolean(s2t("lowerCase"), true);
// Execute the save
executeAction(s2t("save"), descriptor, DialogModes.NO); // Change NO to ALL for dialog
}
}
Copy link to clipboard
Copied
I know. That's what I've been using, but I can´t choose the dpi output just like I set the image size or the compression rate. I'm asking precisely because I need a resolution lower than 300 dpi, images are going to be displayed only in digital devices.
Copy link to clipboard
Copied
I know. That's what I've been using, but I can´t choose the dpi output just like I set the image size or the compression rate. I'm asking precisely because I need a resolution lower than 300 dpi, images are going to be displayed only in digital devices.
By @carmen_maría_9071
The longest edge is 1920px, set by the code here:
// Fit image to 1920px
fitImage(1920, 1920);
This is the size regardless of the PPI print metadata value.
To change the PPI metadata, just change the following 2 instances of 300 to whatever you want, 72 or 96 etc.
/* NEARESTNEIGHBOR | BILINEAR | BICUBIC | BICUBICSMOOTHER | BICUBICSHARPER | BICUBICAUTOMATIC */
function fitImage(fWidth, fHeight) {
if (activeDocument.height.value > activeDocument.width.value) {
activeDocument.resizeImage(null, UnitValue(fHeight, "px"), 300, ResampleMethod.BICUBIC);
} else {
activeDocument.resizeImage(UnitValue(fWidth, "px"), null, 300, ResampleMethod.BICUBIC);
}
}
Keep in mind that the resolution metadata value has no impact on the pixel values of the longest edge.
Copy link to clipboard
Copied
Thank you!
Copy link to clipboard
Copied
I used chatGPT to add a UI with inputs to customize maxsize, resolution and compression quality:
/*
Batch Save As WebP.jsx
https://community.adobe.com/t5/photoshop-ecosystem-discussions/export-many-files-at-once-in-webp-format-photoshop/m-p/13604411
v1.0 - 14th March 2023, Stephen Marsh
v1.1 - 11th January 2024: Added a "fit image" to 1920px step
v1.2 - 10th February 2024: Added an explicit step to change to RGB mode for non-RGB images
v1.3 - 12th September 2024: Added ScriptUI and user input for max image size, resolution and quality
*/
#target photoshop
// Ensure that version 2022 or later is being used
var versionNumber = app.version.split(".");
var versionCheck = parseInt(versionNumber);
// Fail if Photoshop version is less than 2022
if (versionCheck < 23) {
alert("You must use Photoshop 2022 or later to save using native WebP format...");
} else {
// Create the ScriptUI dialog
var dialog = new Window("dialog", "Batch Save As WebP");
// Create a panel to hold the input fields
var mainPanel = dialog.add("panel");
mainPanel.orientation = "column";
mainPanel.alignChildren = "left";
// Input folder selection
var inputGroup = mainPanel.add("group");
inputGroup.orientation = "row";
inputGroup.add("statictext", undefined, "Input Folder:");
var inputFolderInput = inputGroup.add("edittext", undefined, "");
inputFolderInput.characters = 30;
var inputBrowseButton = inputGroup.add("button", undefined, "Browse...");
// Output folder selection
var outputGroup = mainPanel.add("group");
outputGroup.orientation = "row";
outputGroup.add("statictext", undefined, "Output Folder:");
var outputFolderInput = outputGroup.add("edittext", undefined, "");
outputFolderInput.characters = 30;
var outputBrowseButton = outputGroup.add("button", undefined, "Browse...");
// Maximum image size
var sizeGroup = mainPanel.add("group");
sizeGroup.orientation = "row";
sizeGroup.add("statictext", undefined, "Max Image Size (px):");
var maxSizeInput = sizeGroup.add("edittext", undefined, "1920");
maxSizeInput.characters = 10;
// Resolution input
var resolutionGroup = mainPanel.add("group");
resolutionGroup.orientation = "row";
resolutionGroup.add("statictext", undefined, "Resolution (DPI):");
var resolutionInput = resolutionGroup.add("edittext", undefined, "72");
resolutionInput.characters = 10;
// Compression quality
var qualityGroup = mainPanel.add("group");
qualityGroup.orientation = "row";
qualityGroup.add("statictext", undefined, "Quality (0-100):");
var qualityInput = qualityGroup.add("edittext", undefined, "75");
qualityInput.characters = 10;
// OK and Cancel buttons
var buttonGroup = dialog.add("group");
buttonGroup.orientation = "row";
var okButton = buttonGroup.add("button", undefined, "OK");
var cancelButton = buttonGroup.add("button", undefined, "Cancel");
// Folder browsing functionality
inputBrowseButton.onClick = function () {
var inputFolder = Folder.selectDialog("Select the input folder:");
if (inputFolder) {
inputFolderInput.text = inputFolder.fsName;
}
};
outputBrowseButton.onClick = function () {
var outputFolder = Folder.selectDialog("Select the output folder:");
if (outputFolder) {
outputFolderInput.text = outputFolder.fsName;
}
};
// OK button action
okButton.onClick = function () {
var inputFolder = Folder(inputFolderInput.text);
var outputFolder = Folder(outputFolderInput.text);
var maxSize = parseInt(maxSizeInput.text) || 1920;
var resolution = parseInt(resolutionInput.text) || 72;
var quality = parseInt(qualityInput.text) || 75;
// Ensure input/output folders are set
if (!inputFolder.exists || !outputFolder.exists) {
alert("Please select valid input and output folders.");
return;
}
dialog.close(1); // Close the dialog with "OK"
processFiles(inputFolder, outputFolder, maxSize, resolution, quality);
};
// Cancel button action
cancelButton.onClick = function () {
dialog.close(0); // Close the dialog with "Cancel"
};
// Show the dialog
if (dialog.show() == 1) {
// Continue processing if OK is pressed
} else {
// Do nothing if Cancel is pressed
}
// Main function to process the files
function processFiles(inputFolder, outputFolder, maxSize, resolution, quality) {
// Limit the input files, add or remove extensions as required
var fileList = inputFolder.getFiles(/\.(webp|tif|tiff|jpg|jpeg|psd|psb|png)$/i);
fileList.sort();
var savedDisplayDialogs = app.displayDialogs;
app.displayDialogs = DialogModes.NO;
// Set the file processing counter
var fileCounter = 0;
// Process the input files
for (var i = 0; i < fileList.length; i++) {
var doc = open(fileList[i]);
// If the doc isn't in RGB mode
if (activeDocument.mode !== DocumentMode.RGB) {
activeDocument.convertProfile("sRGB IEC61966-2.1", Intent.RELATIVECOLORIMETRIC, true, false);
activeDocument.changeMode(ChangeMode.RGB);
activeDocument.bitsPerChannel = BitsPerChannelType.EIGHT;
}
// Fit image to the specified max size
fitImage(maxSize, maxSize, resolution);
// Save as WebP and close
saveWebP(outputFolder, quality, true, true, true, true);
activeDocument.close(SaveOptions.DONOTSAVECHANGES);
// Increment the file saving counter
fileCounter++;
}
// Function to resize image based on dimensions and resolution
function fitImage(fWidth, fHeight, resolution) {
if (activeDocument.height.value > activeDocument.width.value) {
activeDocument.resizeImage(null, UnitValue(fHeight, "px"), resolution, ResampleMethod.BICUBIC);
} else {
activeDocument.resizeImage(UnitValue(fWidth, "px"), null, resolution, ResampleMethod.BICUBIC);
}
}
app.displayDialogs = savedDisplayDialogs;
alert('Script completed!' + '\n' + fileCounter + ' files saved to:' + '\r' + outputFolder.fsName);
}
// Function to save as WebP
function saveWebP(outputFolder, quality, xmpData, exifData, psData, asCopy) {
var WebPDocName = activeDocument.name.replace(/\.[^\.]+$/, ''); // Remove file extension
var WebPSavePath = outputFolder + "/" + WebPDocName + ".webp"; // Define save path
var WebPFile = new File(WebPSavePath); // Create file object
function s2t(s) {
return app.stringIDToTypeID(s);
}
var descriptor = new ActionDescriptor();
var descriptor2 = new ActionDescriptor();
// Compression parameters = "compressionLossless" | "compressionLossy"
descriptor2.putEnumerated(s2t("compression"), s2t("WebPCompression"), s2t("compressionLossy")); // Lossy compression
descriptor2.putInteger(s2t("quality"), quality); // Set quality (0-100)
// Metadata options
descriptor2.putBoolean(s2t("includeXMPData"), xmpData);
descriptor2.putBoolean(s2t("includeEXIFData"), exifData);
descriptor2.putBoolean(s2t("includePsExtras"), psData);
// WebP format and save path
descriptor.putObject(s2t("as"), s2t("WebPFormat"), descriptor2);
descriptor.putPath(s2t("in"), WebPFile);
// Save As a Copy
descriptor.putBoolean(s2t("copy"), asCopy);
descriptor.putBoolean(s2t("lowerCase"), true);
// Execute save action
executeAction(s2t("save"), descriptor, DialogModes.NO);
}
}
Copy link to clipboard
Copied
I used chatGPT to add a UI with inputs to customize maxsize, resolution and compression quality:
By @filduarte
Thank you for extending my script!
I have never tried adding scriptUI to a script using AI, my expectations are obviously limited...
What did you do, paste in the previous code and add a prompt to add the GUI? Did you have to specify which variables in the existing script needed to have UI controls etc?
Can you share the prompt/s?
Did you have to tidy up the code, or did it work correctly first go?
Copy link to clipboard
Copied
I used several prompts because I made the first prompt before testing the original script and the resulting script ended opening up a prompt to each input field 😂
I pasted the previous code and added this prompt:
The script for photoshop below serves to convert images into webp. Change it to display text fields to set the maximum image size (keeping the default value in 1920px) and resolution (standard at 72).
Then it took 3 more prompts to get to the result I posted:
I tried to make the layout more organized, align the labels and inputs like a table where the first column has the labels and the second has the inputs, but it didn't worked out.
Copy link to clipboard
Copied
Thank you, these LLM’s have certainly come a long way in a short time! I like scripting, but detest building a GUI, so this is great stuff. It's not so much the GUI build, it's also linking up the GUI elements to the underlying code that I dislike.
I just tested this and I was blown away with the results. As I understand the code, I could direct the generative AI to pinpoint certain variables and control how I wanted them, rather than leaving that up to the AI.
Copy link to clipboard
Copied
Yeah I feel you on that. It's so tedious to get it to work. Takes of hours of testing and debugg9ng..
If you than want to expand it using descriptor so it stores the variables, it gets even worse. If you need a preset system, dang that's also tons of work. Yet, somehow I do like this process and it sure does make me feel good when 8 accomplish on the job
Copy link to clipboard
Copied
Is AI cclever" enough to understand which variable is for what and use the inputs as a variable. That's genius!
I always scripture website by Joonas, but it's very tedious work. Especially big scripts with lots of options. Then. Adding presets and storage of the files expands the difficultly and time needed to test and take out errors.
Very cool you were able to do this
Copy link to clipboard
Copied
Can you show a screen grab of the dialog?
Copy link to clipboard
Copied
Can you show a screen grab of the dialog?
By @schroef
Although you directed this request to @filduarte here are some screenshots of some of the prompts that I used. As this was my first time adding a GUI via AI, I wasn't sure if I should be more active or passive in the prompts. Am I overthinking things or not giving enough detail? SO much depends on the prompt, as well as the model.
Copy link to clipboard
Copied
You can change the following entries in your code from "edittext" to "editnumber" to ensure that only numbers are entered into the appropriate GUI fields...
_________
From:
var maxSizeInput = sizeGroup.add("edittext", undefined, "1920");
To:
var maxSizeInput = sizeGroup.add("editnumber", undefined, "1920");
_________
From:
var resolutionInput = resolutionGroup.add("edittext", undefined, "72");
To:
var resolutionInput = resolutionGroup.add("editnumber", undefined, "72");
_________
From:
var qualityInput = qualityGroup.add("edittext", undefined, "75");
To:
var qualityInput = qualityGroup.add("editnumber", undefined, "75");
Copy link to clipboard
Copied
Nice!
Oh, and thank you for making this awesome script! It's saving me a lot of time.
Copy link to clipboard
Copied
You're welcome!
Thank you for opening my eyes to adding scriptUI through AI.
Copy link to clipboard
Copied
You're welcome as well!
One last contribution: this one sets the output folder automatically to a webp subfolder in the input folder if the user don't choose an output folder. It saves a bit more of time 😃
/*
Batch Save As WebP.jsx
https://community.adobe.com/t5/photoshop-ecosystem-discussions/export-many-files-at-once-in-webp-format-photoshop/m-p/13604411
v1.0 - 14th March 2023, Stephen Marsh
v1.1 - 11th January 2024: Added a "fit image" to 1920px step
v1.2 - 10th February 2024: Added an explicit step to change to RGB mode for non-RGB images
v1.3 - 12th September 2024: Added ScriptUI with inputs for input/output folders, max image size, resolution and quality
*/
#target photoshop
// Ensure that version 2022 or later is being used
var versionNumber = app.version.split(".");
var versionCheck = parseInt(versionNumber);
// Fail if Photoshop version is less than 2022
if (versionCheck < 23) {
alert("You must use Photoshop 2022 or later to save using native WebP format...");
} else {
// Create the ScriptUI dialog
var dialog = new Window("dialog", "Batch Save As WebP");
// Create a panel to hold the input fields
var mainPanel = dialog.add("panel");
mainPanel.orientation = "column";
mainPanel.alignChildren = "left";
// Input folder selection
var inputGroup = mainPanel.add("group");
inputGroup.orientation = "row";
inputGroup.add("statictext", undefined, "Input Folder:");
var inputFolderInput = inputGroup.add("edittext", undefined, "");
inputFolderInput.characters = 30;
var inputBrowseButton = inputGroup.add("button", undefined, "Browse...");
// Output folder selection
var outputGroup = mainPanel.add("group");
outputGroup.orientation = "row";
outputGroup.add("statictext", undefined, "Output Folder:");
var outputFolderInput = outputGroup.add("edittext", undefined, "[Input Folder]/webp");
outputFolderInput.characters = 30;
var outputBrowseButton = outputGroup.add("button", undefined, "Browse...");
// Maximum image size
var sizeGroup = mainPanel.add("group");
sizeGroup.orientation = "row";
sizeGroup.add("statictext", undefined, "Max Image Size (px):");
var maxSizeInput = sizeGroup.add("editnumber", undefined, "1920");
maxSizeInput.characters = 10;
// Resolution input
var resolutionGroup = mainPanel.add("group");
resolutionGroup.orientation = "row";
resolutionGroup.add("statictext", undefined, "Resolution (DPI):");
var resolutionInput = resolutionGroup.add("editnumber", undefined, "72");
resolutionInput.characters = 10;
// Compression quality
var qualityGroup = mainPanel.add("group");
qualityGroup.orientation = "row";
qualityGroup.add("statictext", undefined, "Quality (0-100):");
var qualityInput = qualityGroup.add("editnumber", undefined, "75");
qualityInput.characters = 10;
// OK and Cancel buttons
var buttonGroup = dialog.add("group");
buttonGroup.orientation = "row";
var okButton = buttonGroup.add("button", undefined, "OK");
var cancelButton = buttonGroup.add("button", undefined, "Cancel");
// OutputFolderInput Placeholder behavior
outputFolderInput.onSetFocus = function () {
if (outputFolderInput.text === "[Input Folder]/webp") {
outputFolderInput.text = "";
}
};
outputFolderInput.onKillFocus = function () {
if (outputFolderInput.text === "") {
outputFolderInput.text = "[Input Folder]/webp";
}
};
// Folder browsing functionality
inputBrowseButton.onClick = function () {
var inputFolder = Folder.selectDialog("Select the input folder:");
if (inputFolder) {
inputFolderInput.text = inputFolder.fsName;
}
};
outputBrowseButton.onClick = function () {
var outputFolder = Folder.selectDialog("Select the output folder:");
if (outputFolder) {
outputFolderInput.text = outputFolder.fsName;
}
};
// OK button action
okButton.onClick = function () {
var inputFolder = Folder(inputFolderInput.text);
var outputFolder = Folder(inputFolder.fsName + '/webp');
if (outputFolderInput.text !== "[Input Folder]/webp"
&& outputFolderInput.text !== ""
) {
outputFolder = Folder(outputFolderInput.text);
}
var maxSize = parseInt(maxSizeInput.text) || 1920;
var resolution = parseInt(resolutionInput.text) || 72;
var quality = parseInt(qualityInput.text) || 75;
// Ensure input/output folders are set
if (!inputFolder.exists) {
alert("Please select valid input folder.");
return;
}
// Create the output folder if needed
if (!outputFolder.exists) {
outputFolder.create();
}
dialog.close(1); // Close the dialog with "OK"
processFiles(inputFolder, outputFolder, maxSize, resolution, quality);
};
// Cancel button action
cancelButton.onClick = function () {
dialog.close(0); // Close the dialog with "Cancel"
};
// Show the dialog
if (dialog.show() == 1) {
// Continue processing if OK is pressed
} else {
// Do nothing if Cancel is pressed
}
// Main function to process the files
function processFiles(inputFolder, outputFolder, maxSize, resolution, quality) {
// Limit the input files, add or remove extensions as required
var fileList = inputFolder.getFiles(/\.(webp|tif|tiff|jpg|jpeg|psd|psb|png)$/i);
fileList.sort();
var savedDisplayDialogs = app.displayDialogs;
app.displayDialogs = DialogModes.NO;
// Set the file processing counter
var fileCounter = 0;
// Process the input files
for (var i = 0; i < fileList.length; i++) {
var doc = open(fileList[i]);
// If the doc isn't in RGB mode
if (activeDocument.mode !== DocumentMode.RGB) {
activeDocument.convertProfile("sRGB IEC61966-2.1", Intent.RELATIVECOLORIMETRIC, true, false);
activeDocument.changeMode(ChangeMode.RGB);
activeDocument.bitsPerChannel = BitsPerChannelType.EIGHT;
}
// Fit image to the specified max size
fitImage(maxSize, maxSize, resolution);
// Save as WebP and close
saveWebP(outputFolder, quality, true, true, true, true);
activeDocument.close(SaveOptions.DONOTSAVECHANGES);
// Increment the file saving counter
fileCounter++;
}
// Function to resize image based on dimensions and resolution
function fitImage(fWidth, fHeight, resolution) {
if (activeDocument.height.value > activeDocument.width.value) {
activeDocument.resizeImage(null, UnitValue(fHeight, "px"), resolution, ResampleMethod.BICUBIC);
} else {
activeDocument.resizeImage(UnitValue(fWidth, "px"), null, resolution, ResampleMethod.BICUBIC);
}
}
app.displayDialogs = savedDisplayDialogs;
alert('Script completed!' + '\n' + fileCounter + ' files saved to:' + '\r' + outputFolder.fsName);
}
// Function to save as WebP
function saveWebP(outputFolder, quality, xmpData, exifData, psData, asCopy) {
var WebPDocName = activeDocument.name.replace(/\.[^\.]+$/, ''); // Remove file extension
var WebPSavePath = outputFolder + "/" + WebPDocName + ".webp"; // Define save path
var WebPFile = new File(WebPSavePath); // Create file object
function s2t(s) {
return app.stringIDToTypeID(s);
}
var descriptor = new ActionDescriptor();
var descriptor2 = new ActionDescriptor();
// Compression parameters = "compressionLossless" | "compressionLossy"
descriptor2.putEnumerated(s2t("compression"), s2t("WebPCompression"), s2t("compressionLossy")); // Lossy compression
descriptor2.putInteger(s2t("quality"), quality); // Set quality (0-100)
// Metadata options
descriptor2.putBoolean(s2t("includeXMPData"), xmpData);
descriptor2.putBoolean(s2t("includeEXIFData"), exifData);
descriptor2.putBoolean(s2t("includePsExtras"), psData);
// WebP format and save path
descriptor.putObject(s2t("as"), s2t("WebPFormat"), descriptor2);
descriptor.putPath(s2t("in"), WebPFile);
// Save As a Copy
descriptor.putBoolean(s2t("copy"), asCopy);
descriptor.putBoolean(s2t("lowerCase"), true);
// Execute save action
executeAction(s2t("save"), descriptor, DialogModes.NO);
}
}
Copy link to clipboard
Copied
I have been playing around with adding a GUI, I doubt that I will ever be happy with it so this will have to do.
/*
Batch Save As WebP scriptUI GUI v1-5.jsx
https://community.adobe.com/t5/photoshop-ecosystem-discussions/export-many-files-at-once-in-webp-format-photoshop/m-p/13604411
v1.0 - 14th March 2023, Stephen Marsh
v1.1 - 11th January 2024: Added a "fit image" defaulting to 1920px step
v1.2 - 10th February 2024: Added an explicit step to change to RGB mode
v1.3 - 12th September 2024: User "filduarte" Added ScriptUI and user input for max image size, resolution and quality
v1.4 - 13th September 2024: Added various GUI control options. All files saved as 8bpc sRGB, overwriting existing WebP files of the same name without warning
v1.5 - 21st September 2024: Changed the dropdown to use label names rather than compression values as names. Removed the Save a Copy checkbox.
*/
#target photoshop
// Ensure that version 2022 or later is being used
var versionNumber = app.version.split(".");
var versionCheck = parseInt(versionNumber[0]);
if (versionCheck < 23) {
alert("You must use Photoshop 2022 or later to save using native WebP format...");
} else {
// Create the ScriptUI dialog window
var theDialogWin = new Window("dialog", "WebP Batch Processor (v1.5)");
theDialogWin.orientation = "column";
theDialogWin.alignChildren = "left";
var infoText = theDialogWin.add("statictext", undefined, "The WebP files will be converted to sRGB 8 bits/channel");
infoText.alignment = "center";
// Input Folder Section
theDialogWin.add("statictext", undefined, "Select the source image folder to process:");
var inputFolderPanel = theDialogWin.add("panel", undefined, undefined);
inputFolderPanel.orientation = "row";
var inputFolderText = inputFolderPanel.add("edittext", undefined, "");
inputFolderText.characters = 30;
var inputFolderButton = inputFolderPanel.add("button", undefined, "Browse...");
// Process Subfolders Checkbox
var recursiveCheckbox = theDialogWin.add("checkbox", undefined, "Include all subfolders");
recursiveCheckbox.value = false; // Default to not recursive
// Output Folder Section
theDialogWin.add("statictext", undefined, "Select the location to save the WebP files:");
var outputFolderPanel = theDialogWin.add("panel", undefined, undefined);
outputFolderPanel.orientation = "row";
var outputFolderText = outputFolderPanel.add("edittext", undefined, "");
outputFolderText.characters = 30;
var outputFolderButton = outputFolderPanel.add("button", undefined, "Browse...");
// Mirror Directory Structure Checkbox
var mirrorStructureCheckbox = theDialogWin.add("checkbox", undefined, "Retain the source subfolder structure");
mirrorStructureCheckbox.value = false; // Default to not mirroring
mirrorStructureCheckbox.enabled = false; // Disable by default, only enable if recursive is selected
// Enable/Disable "Mirror Structure" based on the recursive option
recursiveCheckbox.onClick = function () {
mirrorStructureCheckbox.enabled = recursiveCheckbox.value;
};
// Compression Type Dropdown (compressionLossy | compressionLossless)
theDialogWin.add("statictext", undefined, "Compression Type:");
var compTypeDropdown = theDialogWin.add("dropdownlist", undefined, ["Lossy", "Lossless"]);
compTypeDropdown.selection = 0; // Default to Lossy
// Compression Quality (compValue) Slider
theDialogWin.add("statictext", undefined, "Compression Quality:");
var compValueSlider = theDialogWin.add("slider", undefined, 75, 0, 100); // Start, Min, Max
compValueSlider.size = [200, 20];
var compValueText = theDialogWin.add("statictext", undefined, "75");
// Update the statictext to reflect the slider's value
compValueSlider.onChanging = function() {
compValueText.text = Math.round(compValueSlider.value).toString();
compValueText.size = [100, 15];
};
// Function to toggle the slider based on compression type
function updateCompressionControls() {
if (compTypeDropdown.selection.text === "Lossless") {
compValueSlider.enabled = false;
compValueText.enabled = false;
} else {
compValueSlider.enabled = true;
compValueText.enabled = true;
}
}
// Initialize control states
updateCompressionControls();
// Disable compression quality slider when "Lossless" is selected
compTypeDropdown.onChange = function() {
updateCompressionControls();
};
// Fit Image Checkbox and Input
var fitImageGroup = theDialogWin.add("group");
fitImageGroup.orientation = "row";
var fitImageCheckbox = fitImageGroup.add("checkbox", undefined, "Fit Image (px):");
fitImageCheckbox.value = false; // Default to inactive
var fitImageInput = fitImageGroup.add("editnumber", undefined, "1920"); // Default to 1920
fitImageInput.characters = 5;
fitImageInput.enabled = fitImageCheckbox.value;
fitImageCheckbox.onClick = function() {
fitImageInput.enabled = fitImageCheckbox.value;
};
// PPI Checkbox and Input
var ppiGroup = theDialogWin.add("group");
ppiGroup.orientation = "row";
var ppiCheckbox = ppiGroup.add("checkbox", undefined, "PPI Value:");
ppiCheckbox.value = false; // Default to inactive
var ppiInput = ppiGroup.add("editnumber", undefined, "300"); // Default to 300
ppiInput.characters = 5;
ppiInput.enabled = ppiCheckbox.value;
ppiCheckbox.onClick = function() {
ppiInput.enabled = ppiCheckbox.value;
};
// XMP Checkbox
var xmpDataCheckbox = theDialogWin.add("checkbox", undefined, "Include XMP Data");
xmpDataCheckbox.value = false; // Default to inactive
// Exif Checkbox
var exifDataCheckbox = theDialogWin.add("checkbox", undefined, "Include EXIF Data");
exifDataCheckbox.value = false; // Default to inactive
// Photoshop Data Checkbox
var psDataCheckbox = theDialogWin.add("checkbox", undefined, "Include Photoshop Data");
psDataCheckbox.value = false; // Default to inactive
// OK and Cancel Buttons
var buttonGroup = theDialogWin.add("group");
buttonGroup.alignment = "right";
var okButton = buttonGroup.add("button", undefined, "OK");
var cancelButton = buttonGroup.add("button", undefined, "Cancel");
// Input Folder Browse Button functionality
inputFolderButton.onClick = function () {
var inputFolder = Folder.selectDialog("Select the input folder:");
if (inputFolder) {
inputFolderText.text = inputFolder.fsName;
}
};
// Output Folder Browse Button functionality
outputFolderButton.onClick = function () {
var outputFolder = Folder.selectDialog("Select the output folder:");
if (outputFolder) {
outputFolderText.text = outputFolder.fsName;
}
};
// Cancel Button functionality
cancelButton.onClick = function () {
theDialogWin.close();
};
// OK Button functionality
okButton.onClick = function () {
theDialogWin.close();
app.refresh();
// Restore the Photoshop panels
app.togglePalettes();
if (inputFolderText.text === "" || outputFolderText.text === "") {
alert("Please select both input and output folders.");
return;
}
// Assign selected folders to variables
var inputFolder = new Folder(inputFolderText.text);
var outputFolder = new Folder(outputFolderText.text);
// Map dropdown selection to original compression type values
var compType = compTypeDropdown.selection.text === "Lossless" ? "compressionLossless" : "compressionLossy";
var compValue = Math.round(compValueSlider.value);
var xmpData = xmpDataCheckbox.value;
var exifData = exifDataCheckbox.value;
var psData = psDataCheckbox.value;
// Gather fitImage and PPI parameters
var fitValue = fitImageCheckbox.value ? parseInt(fitImageInput.text) : null;
var ppi = ppiCheckbox.value ? parseInt(ppiInput.text) : null;
// Process files with the recursive and mirror structure flags
processFiles(inputFolder, outputFolder, compType, compValue, xmpData, exifData, psData, fitValue, ppi, recursiveCheckbox.value, mirrorStructureCheckbox.value);
theDialogWin.close();
};
theDialogWin.show();
}
// Main processing function with optional recursion and mirroring structure
function processFiles(inputFolder, outputFolder, compType, compValue, xmpData, exifData, psData, fitValue, ppi, recursive, mirrorStructure) {
var fileList = getFilesRecursive(inputFolder, recursive);
fileList.sort();
var savedDisplayDialogs = app.displayDialogs;
app.displayDialogs = DialogModes.NO;
// Set the input file processing counter
var inputFileCounter = 0;
// Set the output file processing counter
var fileCounter = 0;
// Process the input files
for (var i = 0; i < fileList.length; i++) {
// Script running notification window - courtesy of William Campbell
// https://www.marspremedia.com/download?asset=adobe-script-tutorial-11.zip
// https://youtu.be/JXPeLi6uPv4?si=Qx0OVNLAOzDrYPB4
var working;
working = new Window("palette");
working.preferredSize = [300, 80];
working.add("statictext");
working.t = working.add("statictext");
working.add("statictext");
working.display = function (message) {
this.t.text = message || "Script running, please wait...";
this.show();
app.refresh();
};
working.display();
open(fileList[i]);
// Create mirrored subfolder structure in output folder if requested
var relativePath = fileList[i].parent.fsName.replace(inputFolder.fsName, "");
var targetFolder = outputFolder;
if (mirrorStructure && relativePath !== "") {
targetFolder = new Folder(outputFolder + "/" + relativePath);
if (!targetFolder.exists) {
targetFolder.create();
}
}
// File processing options
// Bitmap mode input
if (activeDocument.mode == DocumentMode.BITMAP) {
activeDocument.changeMode(ChangeMode.GRAYSCALE);
activeDocument.changeMode(ChangeMode.RGB);
activeDocument.convertProfile("sRGB IEC61966-2.1", Intent.RELATIVECOLORIMETRIC, true, false);
activeDocument.bitsPerChannel = BitsPerChannelType.EIGHT;
// Indexed Color, CMYK or Lab mode input
} else if (activeDocument.mode == DocumentMode.INDEXEDCOLOR || activeDocument.mode == DocumentMode.CMYK || activeDocument.mode == DocumentMode.LAB) {
activeDocument.changeMode(ChangeMode.RGB);
activeDocument.convertProfile("sRGB IEC61966-2.1", Intent.RELATIVECOLORIMETRIC, true, false);
activeDocument.bitsPerChannel = BitsPerChannelType.EIGHT;
} else {
activeDocument.changeMode(ChangeMode.RGB);
activeDocument.convertProfile("sRGB IEC61966-2.1", Intent.RELATIVECOLORIMETRIC, true, false);
activeDocument.bitsPerChannel = BitsPerChannelType.EIGHT;
}
// Run an optional action on each processed file
//app.doAction("Molten Lead", "Default Actions");
// Fit image to specified value and/or change PPI if enabled
if (fitValue !== null || ppi !== null) {
fitImage(fitValue, ppi);
}
// Save as a copy and close
saveWebP(compType, compValue, xmpData, exifData, psData, targetFolder);
activeDocument.close(SaveOptions.DONOTSAVECHANGES);
// Increment the input file counter
inputFileCounter++
// Increment the file processing counter
fileCounter++;
// Ensure Photoshop has focus before closing the running script notification window
app.bringToFront();
working.close();
}
app.displayDialogs = savedDisplayDialogs;
app.beep();
alert('Script completed!' + '\r' + inputFileCounter + ' source files saved as ' + fileCounter + ' WebP files!');
// Restore the Photoshop panels
app.togglePalettes();
}
///// Helper Functions /////
function getFilesRecursive(folder, recursive) {
var fileList = [];
var allFiles = folder.getFiles();
for (var i = 0; i < allFiles.length; i++) {
var file = allFiles[i];
if (file instanceof Folder && recursive) {
fileList = fileList.concat(getFilesRecursive(file, recursive));
} else if (file instanceof File && /\.(webp|tif|tiff|jpg|jpeg|psd|psb|png|tga)$/i.test(file.name)) {
fileList.push(file);
}
}
return fileList;
}
function fitImage(fitValue, ppi) {
// NEARESTNEIGHBOR | BILINEAR | BICUBIC | BICUBICSMOOTHER | BICUBICSHARPER | BICUBICAUTOMATIC
if (fitValue !== null && ppi !== null) {
// Both fit value and PPI are specified
if (activeDocument.height.value > activeDocument.width.value) {
activeDocument.resizeImage(null, UnitValue(fitValue, "px"), ppi, ResampleMethod.BICUBIC);
} else {
activeDocument.resizeImage(UnitValue(fitValue, "px"), null, ppi, ResampleMethod.BICUBIC);
}
} else if (fitValue !== null) {
// Only fit value is specified
if (activeDocument.height.value > activeDocument.width.value) {
activeDocument.resizeImage(null, UnitValue(fitValue, "px"), activeDocument.resolution, ResampleMethod.BICUBIC);
} else {
activeDocument.resizeImage(UnitValue(fitValue, "px"), null, activeDocument.resolution, ResampleMethod.BICUBIC);
}
} else if (ppi !== null) {
// Only PPI is specified
activeDocument.resizeImage(undefined, undefined, ppi, ResampleMethod.NONE);
}
}
function saveWebP(compType, compValue, xmpData, exifData, psData, outputFolder) {
// https://community.adobe.com/t5/photoshop-ecosystem-discussions/saving-webp-image-by-script/td-p/13642577
function s2t(s) {
return app.stringIDToTypeID(s);
}
var WebPDocName = activeDocument.name.replace(/\.[^\.]+$/, ''); // Remove file extension
var WebPSavePath = outputFolder + "/" + WebPDocName + ".webp"; // Save path
var WebPFile = new File(WebPSavePath);
var descriptor = new ActionDescriptor();
var descriptor2 = new ActionDescriptor();
descriptor2.putEnumerated(s2t("compression"), s2t("WebPCompression"), s2t(compType));
if (compType === "compressionLossy") {
descriptor2.putInteger(s2t("quality"), compValue);
}
descriptor2.putBoolean(s2t("includeXMPData"), xmpData);
descriptor2.putBoolean(s2t("includeEXIFData"), exifData);
descriptor2.putBoolean(s2t("includePsExtras"), psData);
descriptor.putObject(s2t("as"), s2t("WebPFormat"), descriptor2);
descriptor.putPath(s2t("in"), WebPFile);
descriptor.putBoolean(s2t("copy"), true); // asCopy
descriptor.putBoolean(s2t("lowerCase"), true);
executeAction(s2t("save"), descriptor, DialogModes.NO);
}