Skip to main content

Fast exporting of PSDs with [tagged] layers for variants.

Project description

psd-export

  • Fast multi-threaded exporting of PSDs with [tagged] layers for variants.
  • Special named layers can be used to apply filters like mosaic or gaussian blur, or whatever else you can hook in.
  • This tool is primarily meant to be compatible with PSDs exported from PaintTool SAIv2 and Clip Studio Paint.

Why

For my art workflow, I typically make a bunch variation layers and also need to apply mosaics to all of them. As a manual process, exporting every variant can take several minutes, with lots of clicking everywhere to turn layers on and off (potentially missing layers by accident) and then adding mosaics can be another 10 or 20 minutes of extra work. If I find I need to change something in my pictures and export again, I have to repeat this whole ordeal. This script puts this export process on the order of seconds, saving me a lot of time and pain.


Installation

  • Install python: https://www.python.org/downloads/
  • Run pip install psd-export to install the script.
  • Run psd-export to export any PSD files in the current directory.
  • Run psd-export --help for more command line arguments.

Building from source

  • Install a C compiler (like MSVC, GCC, Clang)
  • Install extra dependencies: pip install setuptools wheel cython numpy
  • Install this repository locally: pip install -e .
  • If you modify a Cython file (.pyx), then rebuild it with: python setup.py build_ext --inplace

Building/installing with Nix

  • This project is a Nix flake, so you can run flake commands to interact with the package nix run, nix build, etc.

Setting up the PSD

Automatic Exporting

Primary tags
  • Add tags surrounded by [] to the names of layers that you want as part of an exported picture.
    • A layer name can contain anything else outside of tags: scene [1], which will be ignored.
  • Because whitespace is used to delimit arguments to filters, a tag's name will not include anything after a space, for example:
    • [blur 50] the tag name will be blur, and 50 is an argument to blur.
  • Each set of primary tagged layers will be turned on and off and then exported in turn.
    • That means multiple layers can have the same tag, so you can toggle layers in your foreground and background together, for example:
      • A layer named foreground [1] and a separate layer named background [1]
  • A layer can have multiple tags in the name: scene [1][2]
    • This will export with this layer visible for both tags.
  • A primary tag is not necessary. If no primary tag is provided, then the whole picture is exported as is.
Secondary tags
  • Tags with an @ in the name will be treated as secondary tags.
  • Text before the @ is the tag name, and text after the @ is the exclusion group.
    • The exclusion group can be empty.
    • Valid secondary tag names: [jp@], [jp@text]
  • These tags will be exported in combination with primary tags, for example:
    • If you have layers tagged [1], [thing@] [jp@text], and [en@text], then what will be exported is 1, 1-thing, 1-thing-jp, 1-thing-en, 1-jp, 1-en
    • Because [jp@text] and [en@text] share the same exclusion group text, they will never be enabled together.
    • If there was a second primary tag [2] for example, the whole set of combinations would be exported again but with 2 instead of 1

Image filters

  • Tags with special names will be treated as filters on the colors below them.
  • Filters can double as export tags too.
  • If you want a filter to not be treated as an export tag, you can preceed it with # to set the ignore flag, for example [#censor]
  • Filters can have arguments as well to control their behavior, separated by spaces, for example [#censor 50]
  • If you want the filter to apply to layers outside of the group it is in, then the group blend mode should be set to pass-through, otherwise it may blend with transparent black pixels if the filter is over a transparent part of a group. The blur and motion blur filters apply to alpha as well, so they should behave as expected in isolated groups.
  • If multiple filters are enabled in one layer, they will be applied from left to right on top of each result, example:
    • [#censor][#blur] will apply a mosaic, and then a blur on top of that mosaic.
  • Blend modes and clipping layers applied directly to filter layers are ignored.
Available default filters:
  • [censor mosaic_factor apply_to_alpha]
    • If the mosaic_factor argument is omitted, then it is defaulted to 100, which means 1/100th the smallest dimension, (or 4 pixels, whichever is larger) in order to be Pixiv compliant.
    • apply_to_alpha defaults to False. Any value is treated as True.
    • Typically you will want this filter to be a secondary tag, for example: [censor@], so you can have censored and uncensored outputs.
  • [blur size]
    • The size argument defaults to 50 if omitted.
    • This filter is best used to create a non-destructive blur, such as for a background layer. You can fill an entire layer and set it to [#blur 8] for example.
  • [motion-blur angle size]
    • angle is in degrees, starting from horizontal to the right; Default 0.
    • size defaults to 50.
    • Best used for non-destructive blur: [#motion-blur 45 20]
Adding a new filter:

In your own script:

# my-export.py
from psd_export import (export, filters, util, blendfuncs)
import numpy as np

my_arg1_default = 1.0

# Register the filter with this decorator:
@filters.filter('my-filter')
# Only positional arguments work right now. The result of this function replaces the destination color and alpha.
def some_filter(color_dst, color_src, alpha_dst, alpha_src, arg1=None, arg2=100, *_):
    # Cast arguments to your desired types, as they will come in as strings.
    if arg1 is None:
        arg1 = my_arg1_default
    arg1 = float(arg1)
    # Manipulate color and alpha numpy arrays, in-place if you want.
    color = np.subtract(arg1, color, out=color)
    color = blendfuncs.lerp(color_dst, color, alpha_src)
    # Always return the same shaped arrays as a tuple:
    return color, alpha

if __name__ == '__main__':
    # Add your own command line arguments if needed.
    export.arg_parser.add_argument('--my-arg1', default=my_arg1_default, type=float,
        help='Set the arg1 default parameter.')
    args = export.arg_parser.parse_args()
    my_arg1_default = args.my_arg1

    export.main()

Apply it to a layer in your PSD, for example: [my-filter 20.4] or [#my-filter], etc.


Examples

In the layers below, there are 2 primary tags and 3 secondary tags (with two unique exclusion groups):

Primary tags: 1, 2

Secondary tags: jp, en, censor

Exclusion groups: text, <empty>

Groups with primary tags will be exported again even if the secondary tag does not exist under that group! This keeps the exported folders uniformly sized for publishing, so different folders may appear to have duplicate outputs.

Example layer configuration in SAI:

image

Example output from script, showing every valid combination:

image

Folder after exporting everything:

image

Example of a layer with the tag [censor@] (the layer does not need to be set to visible before exporting):

image

After exporting:

image


Blendmode status:

Blendmode Status
Normal Pass
Multiply Pass
Screen Pass
Overlay Pass
Linear Burn (Shade) Pass
Linear Dodge (Shine) Pass
Linear Light (Shade/Shine) Pass
Color Burn (Burn) Pass
Color Dodge (Dodge) Pass
Vivid Light (Burn/Dodge) Pass
Soft Light Pass
Hard Light Pass
Pin Light Pass
Hard Mix Small precision error for near-black colors, can look slightly different from SAI
Darken Pass
Lighten Pass
Darken Color Pass
Lighten Color Pass
Difference Pass
Exclude Pass
Subtract Pass
Divide Pass
Hue Pass
Saturation Pass
Color Pass
Luminosity Pass
[TS] Linear Burn (Shade) Pass
[TS] Linear Dodge (Shine) Pass
[TS] Linear Light (Shade/Shine) Pass
[TS] Color Burn (Burn) Small precision error for near-black colors, can look slightly different from SAI
[TS] Color Dodge (Dodge) Pass
[TS] Vivid Light (Burn/Dodge) Pass
[TS] Hard Mix Small precision error for near-black colors, can look slightly different from SAI
[TS] Difference Pass

Missing PSD features:

  • Other things that are not implemented (also not implemented in psd-tools):
    • Adjustment layers (gradient map, color balance, etc.)
    • Layer effects (shadow, glow, overlay, strokes, etc.)
    • Font rendering
    • Probably some other things I'm unaware of.

TODO:

  • Fix blend modes that don't quite work properly. This is low priority because I hardly use these modes myself or merge the results when painting.
  • Binary package export

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

psd_export-1.1.1.tar.gz (389.2 kB view details)

Uploaded Source

Built Distributions

If you're not sure about the file name format, learn more about wheel file names.

psd_export-1.1.1-cp313-cp313-win_amd64.whl (546.7 kB view details)

Uploaded CPython 3.13Windows x86-64

psd_export-1.1.1-cp313-cp313-win32.whl (519.6 kB view details)

Uploaded CPython 3.13Windows x86

psd_export-1.1.1-cp313-cp313-musllinux_1_2_x86_64.whl (1.6 MB view details)

Uploaded CPython 3.13musllinux: musl 1.2+ x86-64

psd_export-1.1.1-cp313-cp313-musllinux_1_2_aarch64.whl (1.6 MB view details)

Uploaded CPython 3.13musllinux: musl 1.2+ ARM64

psd_export-1.1.1-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl (1.6 MB view details)

Uploaded CPython 3.13manylinux: glibc 2.17+ x86-64manylinux: glibc 2.28+ x86-64

psd_export-1.1.1-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl (1.6 MB view details)

Uploaded CPython 3.13manylinux: glibc 2.17+ ARM64manylinux: glibc 2.28+ ARM64

psd_export-1.1.1-cp313-cp313-macosx_11_0_arm64.whl (559.7 kB view details)

Uploaded CPython 3.13macOS 11.0+ ARM64

psd_export-1.1.1-cp313-cp313-macosx_10_13_x86_64.whl (567.9 kB view details)

Uploaded CPython 3.13macOS 10.13+ x86-64

File details

Details for the file psd_export-1.1.1.tar.gz.

File metadata

  • Download URL: psd_export-1.1.1.tar.gz
  • Upload date:
  • Size: 389.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.9.25

File hashes

Hashes for psd_export-1.1.1.tar.gz
Algorithm Hash digest
SHA256 5d18107c623c631dea8f34010608c70f7aca1a92e9382f84ae69ef9a51e6fb68
MD5 0b5b1d40e9792c5bc3ae64ec6d1e03a8
BLAKE2b-256 3669614c643b3e351344db1ed4413f261c398d026a4ff828f07d6cdb28093f05

See more details on using hashes here.

File details

Details for the file psd_export-1.1.1-cp313-cp313-win_amd64.whl.

File metadata

  • Download URL: psd_export-1.1.1-cp313-cp313-win_amd64.whl
  • Upload date:
  • Size: 546.7 kB
  • Tags: CPython 3.13, Windows x86-64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.9.25

File hashes

Hashes for psd_export-1.1.1-cp313-cp313-win_amd64.whl
Algorithm Hash digest
SHA256 5485a6c4273ace4a91a2976d5ce8186f44d196b723a57c99ac2f82102177fa1b
MD5 0fc1dac572cd073028802a05b1fbf2fc
BLAKE2b-256 b452e9d9a4ed1a9a6ce732003734a008d92be94b627a679f692bb23a630d85e3

See more details on using hashes here.

File details

Details for the file psd_export-1.1.1-cp313-cp313-win32.whl.

File metadata

  • Download URL: psd_export-1.1.1-cp313-cp313-win32.whl
  • Upload date:
  • Size: 519.6 kB
  • Tags: CPython 3.13, Windows x86
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.9.25

File hashes

Hashes for psd_export-1.1.1-cp313-cp313-win32.whl
Algorithm Hash digest
SHA256 4959d7ceed62474218ab2a38af040b9e1b05573204c07a8d035f216815357e89
MD5 1f0f3d0037650df7faf0854b473a116b
BLAKE2b-256 aa679186c9356630a0cd6b1c94b8eee7f8963d690f367800727fdb6af46288e0

See more details on using hashes here.

File details

Details for the file psd_export-1.1.1-cp313-cp313-musllinux_1_2_x86_64.whl.

File metadata

File hashes

Hashes for psd_export-1.1.1-cp313-cp313-musllinux_1_2_x86_64.whl
Algorithm Hash digest
SHA256 63fbe10cab096879e104ddb0a189e845cc1a99b52c11f751770d4ec2752ba4ac
MD5 e1e4b294e8cc07ef196b42f4c89b0297
BLAKE2b-256 300225fc04bfdba8dae201547726a681a87f02d17b57cc64f1e6574a480d7587

See more details on using hashes here.

File details

Details for the file psd_export-1.1.1-cp313-cp313-musllinux_1_2_aarch64.whl.

File metadata

File hashes

Hashes for psd_export-1.1.1-cp313-cp313-musllinux_1_2_aarch64.whl
Algorithm Hash digest
SHA256 cc58f643841c57d3c95c022763900d3e138d5017b701329c6721d574056adff4
MD5 015826e4928377cf2ace50194a9c61f1
BLAKE2b-256 58c97d31e957b9e74ef26ad7364814141588fd073e228e7fd928db121e550e7b

See more details on using hashes here.

File details

Details for the file psd_export-1.1.1-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl.

File metadata

File hashes

Hashes for psd_export-1.1.1-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl
Algorithm Hash digest
SHA256 6471a489fba14f9ac416e44439b67ab6af16020142d2500513ea1c61f27d3aa4
MD5 c9dd26c1a05681beef643ff6fb98f2bf
BLAKE2b-256 c7d944622ca05395d3f8b3ebadc53792bad93356624c8dc55e92d816ac79657f

See more details on using hashes here.

File details

Details for the file psd_export-1.1.1-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl.

File metadata

File hashes

Hashes for psd_export-1.1.1-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl
Algorithm Hash digest
SHA256 fd83bc539fe03c302a6bf81117ca8be1317536a5880714b8c8cff8c4b5bed588
MD5 5fd9ddbba59c6470bb04ef8b7cec3b2b
BLAKE2b-256 b55b7acd4d3eedde75da33ac4721abe4f84c384d835369f2469ba2d3dc7d5ec3

See more details on using hashes here.

File details

Details for the file psd_export-1.1.1-cp313-cp313-macosx_11_0_arm64.whl.

File metadata

File hashes

Hashes for psd_export-1.1.1-cp313-cp313-macosx_11_0_arm64.whl
Algorithm Hash digest
SHA256 bc2ad320d74ddc9135c888dd677db4e72beeb072930c6b8d07d595b770f9648e
MD5 c14d02130e618b356c70a59d1b0786d6
BLAKE2b-256 34cf8a42169c8cdb746ca96b7ad03e905ef85c291cdbf6b90fbb312349098ecc

See more details on using hashes here.

File details

Details for the file psd_export-1.1.1-cp313-cp313-macosx_10_13_x86_64.whl.

File metadata

File hashes

Hashes for psd_export-1.1.1-cp313-cp313-macosx_10_13_x86_64.whl
Algorithm Hash digest
SHA256 73445897b6ea56ce383731fb848bb987f4f7a1a075061ca3d49dd0b05d4c5284
MD5 bae3d41c83fde75a49facac850c2537d
BLAKE2b-256 9cc710f23a38344bfe32045f66251eacc1b0cbc23902edf737acb4d4243bf091

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page