Python通过psd-tools解析PSD文件

目录
  • 特点
  • 安装
  • 使用
    • 命令行
    • 操作PSD文件
    • 操作使用层
    • 将数据导出到 PIL
    • 将数据导出到NumPy
  • 更多操作
    • 1.操作一个PSD文件
    • 2.操作一个PSD图层

最近碰到业务需要根据PSD文件实现PSD文件解析图层功能,搜到了Python的一个解析PSD的库。这个库就是psd-toolspsd-tools是一个Python软件包,用于处理Adobe Photoshop PSD文件。以下就是psd-tools的基本介绍。

特点

支持:

  • 读取和写入初级的PSD/PSB文件结构
  • 以NumPy和PIL格式导出原始图层图像

有限的支持:

  • 基于像素的基本图层的构造
  • 填充层效果的构造
  • 矢量面具
  • 编辑一些图层属性,如图层名称
  • 除溶解外的混合模式
  • 绘制贝塞尔曲线

不支持:

  • 编辑图层结构,如添加或删除一个图层
  • 调整层的构造
  • 许多层效果的构造
  • 字体渲染

安装

使用pip来安装该软件包。

pip install psd-tools 

为了用完整的图层图像合成功能,也可以安装NumPy/SciPy:

pip install numpy scipy

使用

简单的例子:

from psd_tools import PSDImage
psd = PSDImage.open('example.psd')
psd.composite().save('example.png')
for layer in psd:
    print(layer)
    layer_image = layer.composite()
    layer_image.save('%s.png' % layer.name)

命令行

该软件包提供命令行工具来处理PSD文件。

psd-tools export <input_file> <output_file> [options]
psd-tools show <input_file> [options]
psd-tools debug <input_file> [options]
psd-tools -h | --help
psd-tools --version

例子:

psd-tools show example.psd  # 显示文件内容
psd-tools export example.psd example.png  # 导出为PNG
psd-tools export example.psd[0] example-0.png  # 将图层导出为PNG

操作PSD文件

psd_tools.api 包提供了用户友好的API来处理PSD文件。

打开一个图像:

from psd_tools import PSDImage
psd = PSDImage.open('my_image.psd')

psd-tools中的大部分数据结构都支持在IPython环境下的打印:

In [1]: PSDImage.open('example.psd')
Out[1]:
PSDImage(mode=RGB size=101x55 depth=8 channels=3)
  [0] PixelLayer('Background' size=101x55)
  [1] PixelLayer('Layer 1' size=85x46)

内部层可以通过迭代器或索引进行访问:

for layer in psd:
    print(layer)
    if layer.is_group():
        for child in layer:
            print(child)

child = psd[0][0]

打开的PSD文件可以保存:

psd.save('output.psd')

操作使用层

在Photoshop中,有各种层的种类。

最基本的图层类型是PixelLayer:

print(layer.name)
layer.kind == 'pixel'

有些图层属性是可编辑的,如图层名称:

layer.name = 'Updated layer 1'

里有内部层:

for layer in group:
    print(layer)
first_layer = group[0]

TypeLayer 是一个带有文本的层:

print(layer.text)

ShapeLayer 绘制一个矢量形状,形状信息存储在vector_mask和origination属性中。其他层也可以有形状信息作为遮罩:

print(layer.vector_mask)
for shape in layer.origination:
    print(shape)

SmartObjectLayer 嵌入或链接一个外部文件,用于非破坏性编辑。文件内容可以通过smart_object属性访问:

import io
if layer.smart_object.filetype in ('jpg', 'png'):
    image = Image.open(io.BytesIO(layer.smart_object.data))

SolidColorFillPatternFill, 和 GradientFill 是填充图层,如果没有相关的遮罩,它们会绘制整个区域。 AdjustmentLayer 的子类表示应用于组成图像的层调整。参见 Adjustment layers.

将数据导出到 PIL

将整个文件导出为 PIL.Image:

image = psd.composite()
image.save('exported.png')

导出单一图层,包括遮罩和剪裁层:

image = layer.composite()

分别导出图层和蒙版,不需要合成:

image = layer.topil()
mask = layer.mask.topil()

要合成特定的图层,如除文本外的图层,请使用layer_filter选项:

image = psd.composite(
    layer_filter=lambda layer: layer.is_visible() and layer.kind != 'type')

请注意:大多数图层效果和调整层不被支持。合成的结果可能看起来与Photoshop不同。

将数据导出到NumPy

PSDImage或图层可以通过 numpy() 方法导出为NumPy数组:

image = psd.numpy()
layer_image = layer.numpy()

更多操作

1.操作一个PSD文件

可在源码的psd_image.py中看到PSDImage类

1. 打开一个文件

from psd_tools import PSDImage
psd = PSDImage.open('my_image.psd')
#返回一个PSDImage类型的对象

#psd_tools中的大多数数据结构都支持在IPython环境中进行漂亮的打印。

#  In [1]: PSDImage.open('example.psd')
#  Out[1]:
#  PSDImage(mode=RGB size=101x55 depth=8 channels=3)
#    [0] PixelLayer('Background' size=101x55)
#    [1] PixelLayer('Layer 1' size=85x46)

2. psd的属性(可在源码的psd_image.py中看到PSDImage类)

有些无意义的属性也定义了,为了和layer一样可以,如:visible直接返回Ture。

这里列出一些有意义,一般会用到的属性:

psd.width #宽
psd.height #高
psd.size #(width, height) tuple
psd.offset #(left, top) tuple
psd.left #0
psd.right #self.width
psd.top #0
psd.bottom #self.height
psd.viewbox #(left, top, right, bottom) `tuple`

psd.bbox #能包围住所有可见的层的最小的方框(x,y,z,w)
psd.color_mode #颜色模式,如RGB,GRAYSCALE
psd.channels #颜色通道数量
psd.depth #像素深度位数
psd.version #文件版本 psd是1,psb是2.
psd.has_preview #Returns if the document has real merged data. When True, `topil()`returns pre-composed data.
psd.has_thumbnail #是否有缩略图
psd.thumbnail #返回    PIL.Image格式的缩略图

这里列出一些无意义的为了可以和layer一样操作的属性:

psd.is_visible() #True
psd.visible #True
psd.parent #None
psd.name   #'Root'
psd.kind #'psdimage'

print(str(psd.is_group()))#是否是组 psd文件直接传进去也是组
psd文件的层可以遍历:
for layer in psd:
print(layer)
if layer.is_group():
    for child in layer:
        print(child)

child = psd[0][0]
#迭代顺序是从背景到前景,与1.7.x之前的版本相反。使用reverse (list(psd))从前台到后台进行迭代。

3. 保存psd文件

psd.save('output.psd')

4. 用psd文件获取PIL Image.

psd.topil(channel=None, **kwargs)
#channel:0为R,1为G,2为B,-1为A,根据constants.py中ChannelID类。

5. 合并psd文件.

psd.compose(force = False,bbox=None,**kwargs)

6. 用PIL Image生成一个PSDImage对象

from psd_tools import PSDImage
psd = PSDImage.frompilfrompil(image,compression=<Compression.PACK_BITS: 1>)

2.操作一个PSD图层

可在源码的layers.py中看到Layer类

1.Layer的属性(可在源码的layers.py中看到Layer类)

layer.name #层的名字(可写)
layer.kind #层的类别(字符串)

#(group(图层组), pixel(普通图层), shape, type(文本图层), smartobject,or psdimage(psd本身))
#shape绘制矢量形状,形状信息存储在vector_mask和origination属性中。其他图层也可以有形状信息作为蒙版:
#smartobject为非破坏性编辑嵌入或链接外部文件。文件内容可以通过smart_object属性访问。

layer.layer_id #Layer ID.
layer.visible #层本身是否勾选可见(可写)
layer.is_visible() #层是否可见,受父物体影响。(父物体不可见,这个层就算勾选了可见这个也是False)

layer.opacity #透明度 [0,255](可写)
layer.parent #Parent of this layer.
layer.is_group #是否是个组
layer.blend_mode #混合模式(可写),返回Constants.py中的BlendMode
layer.has_mask #是否有mask
layer.left #左坐标(可写)
layer.top  #顶坐标(可写)
layer.right #右坐标
layer.bottom #底坐标
layer.width #层的宽
layer.height #层的高
layer.offset #(left, top) tuple. (可写)
layer.size #(width, height) tuple.
layer.bbox #(left, top, right, bottom) tuple.
layer.has_pixels() #是否有像素
layer.has_mask() #是否有蒙板
layer.has_vector_mask() #是否有矢量蒙板
layer.mask #层相关的蒙版 return: :py:class:`~psd_tools.api.mask.Mask` or `None`
layer.vector_mask #层相关的矢量蒙版 return: :py:class:`~psd_tools.api.shape.VectorMask` or `None`
layer.has_origination() #是否有实时形状属性
layer.origination #实时形状属性
layer.has_stroke() #是否有比划
layer.stroke #比划
layer.has_clip_layers() #是否有裁剪
layer.clip_layers #裁剪,Clip layers associated with this layer.
layer.has_effects() #是否有效果处理
layer.effects #效果处理 return: :py:class:`~psd_tools.api.effects.Effects`
layer.tagged_blocks #Layer tagged blocks that is a dict-like container of settings.

2. 获得图层的pil图,Get PIL Image of the layer.(返回PIL.Image对象或没像素时返回`None`)

layer.topil(channel=None, **kwargs)
e.g.
from psd_tools.constants import ChannelID
image = layer.topil()
red = layer.topil(ChannelID.CHANNEL_0)
alpha = layer.topil(ChannelID.TRANSPARENCY_MASK)

3. 合并图层和其蒙版(mask, vector mask, and clipping layers)(返回PIL.Image对象或没像素时返回`None`)

layer.compose(bbox=None, **kwargs)

不合并,单独获取:

image = layer.topil()
mask = layer.mask.topil()
from psd_tools import compose
clip_image = compose(layer.clip_layers)

到此这篇关于Python通过psd-tools解析PSD文件的文章就介绍到这了,更多相关Python PSD 内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Python中itertools库的四个函数介绍

    目录 1. 引言 2. accumulate() 函数 3. compress() 函数 4. groupby() 函数 5. 排列组合操作 6 总结 1. 引言 在Python开发中,​​itertools​​库经常被忽视,实际上该库中抱恨了一些非常棒的函数,特别是用于处于数据流的函数.在本文中,我们将讨论该库中的十分使用的几个函数,并重点介绍什么时候我们应该考虑使用它们.闲话少说,我们直接开始吧! 2. accumulate() 函数 第三方库​​itertools​​提供的函数​​accu

  • 浅析Python自带性能强悍的标准库itertools

    目录 前言 无限迭代 有限迭代 排列组合迭代 前言   可迭代对象就像密闭容器里的水,有货倒不出 itertools是python内置的标准模块,提供了很多简洁又高效的专用功能,使用得当能够极大的简化代码行数,同时所有方法都是实现了生成器函数,这就意味着极大的节省内存. itertools提供的功能主要分为三大块,以最新版本的3.10为例: 对可迭代对象无限迭代,无限输出 对可迭代对象有限迭代 对可迭代对象排列组合 方法如下: 导入包 >>> from iteratortools imp

  • python中setuptools的作用是什么

    什么是setuptools setuptools是Python distutils增强版的集合,它可以帮助我们更简单的创建和分发Python包,尤其是拥有依赖关系的.用户在使用setuptools创建的包时,并不需要已安装setuptools,只要一个启动模块即可. 功能亮点: 利用EasyInstall自动查找.下载.安装.升级依赖包 创建Python Eggs 包含包目录内的数据文件 自动包含包目录内的所有的包,而不用在setup.py中列举 自动包含包内和发布有关的所有相关文件,而不是创建

  • python中itertools模块使用小结

    Python 内置的 itertools 模块包含了一系列用来产生不同类型迭代器的函数或类,这些函数的返回都是一个迭代器,我们可以通过 for 循环来遍历取值,也可以使用 next() 来取值. itertools 模块提供的迭代器函数有以下几种类型: 无限迭代器:生成一个无限序列,比如自然数序列 1, 2, 3, 4, ...: 有限迭代器:接收一个或多个序列(sequence)作为参数,进行组合.分组和过滤等: 组合生成器:序列的排列.组合,求序列的笛卡儿积等. itertools 高效循环

  • Python中itertools模块的使用教程详解

    目录 itertools模块的介绍 无限迭代器(Infinite Iterators) 组合迭代器(Combinatoric Iterators) 有限迭代器(Iterators Terminating on the Shortest Input Sequence) itertools模块的介绍 在Python中,迭代器(Iterator)是常用来做惰性序列的对象,只有当迭代到某个值的时候,才会进行计算得出这个值.因此,迭代器可以用来存储无限大的序列,这样我们就不用把他一次性放在内存中,而只在需

  • 详解Python3.8+PyQt5+pyqt5-tools+Pycharm配置详细教程

    个人使用环境 WIN10x64系统,Python3.8,PyCharm2020.01.03 安装过程 一.安装Python3.8 (自己参考其他教程) 二.安装PyQt5 然后在cmd下输入指令 pip install PyQt5 也可以输入这个指令 pip install PyQt5 -i https://pypi.douban.com/simple (后面是豆瓣的镜像地址,是为了加快下载速度) 提示你更新pip,就按照提示更新(这步骤是可选的,看个人需求) 在cmd下输入 python -m

  • 关于Python中 循环器 itertools的介绍

    目录 1.无穷循环器 2.函数式工具 3.组合工具 4.groupby() 5.其它工具 在for i in iterator结构中,循环器每次返回的对象将赋予给i,直到循环结束.使用iter()内置函数,我们可以将诸如表.字典等容器变为循环器.比如 for i in iter([2, 4, 5, 6]): print(i) 标准库中的itertools包提供了更加灵活的生成循环器的工具.这些工具的输入大都是已有的循环器.另一方面,这些工具完全可以自行使用Python实现,该包只是提供了一种比较

  • python itertools包内置无限迭代器

    目录 1.count()创建自然数序列 2.cycle()创建序列循环 3.repeat()把一个元素有限次重复 4.takewhile()根据条件判断来截取出一个有限的序列 5.chain()迭代对象串联 6.groupby()迭代器元素分组 前言: Python的内建模块itertools提供了非常有用的用于操作迭代对象的函数,itertools提供的几个“无限”迭代器 1.count()创建自然数序列 import itertools   n = 0 natuals = itertools

  • Python itertools.product方法代码实例

    itertools.product:类似于求多个可迭代对象的笛卡尔积. 使用的形式是: itertools.product(*iterables, repeat=1), product(X, repeat=3)等价于product(X, X, X). 1. 直接使用时:分别生成元组,然后合成一个list import itertools aa = itertools.product(['西藏','瀑布','湖水'], ['月色','星空']) bb = list(aa) #按照顺序生成笛卡尔积,

  • python使用imap-tools模块下载邮件附件的示例

    最近在做一些email相关的办公自动化项目,发现一个第三方模块imap-tools不错, 网上没有啥相关介绍,所以记录下来. 环境: python3.8; imap-tools 0.39.0 需要pip 安装一下imap-tools模块 imap-tools模块是python的第三方扩展, 它使用标准库imaplib,并将常见的邮件处理事件封装,邮件处理起来代码短. 下面是个下载邮件附件的示例 from imap_tools import MailBox with MailBox("imap服务

随机推荐