用 Pillow 阅读的 Gif 有黑色边框

Dra*_*421 5 python gif animated-gif python-imaging-library

我正在使用 Pillow 读取 gif,此代码将 gif 文件中的每一帧输出为 .gif 和 .png,并转换为 RGBA 和不转换为 RGBA。

在显示结果后,我将讨论下面的问题

from PIL import Image

# Download link below
image = Image.open('PepePls.gif')

loop = []
frames = []
durations = []
try:
    while True:
        loop.append(image.info.get('loop', None))
        frame = image.copy()
        frames.append(frame)
        durations.append(image.info.get('duration', None))
        image.seek(image.tell() + 1)
except EOFError:
    pass

for i in range(len(frames)):
    print(f'#{i} loop = {loop[i]} duration = {durations[i]} mode = {frames[i].mode}')
    frames[i].save(f'gif_f{i}.gif')
    frames[i].save(f'png_f{i}.png')
    frames[i].convert(mode='RGBA').save(f'rgba_gif_f{i}.gif')
    frames[i].convert(mode='RGBA').save(f'rgba_png_f{i}.png')
Run Code Online (Sandbox Code Playgroud)

结果,控制台输出然后 explorer.exe 截图:

#0 loop = 0 duration = 70 mode = P
#1 loop = None duration = 70 mode = P
#2 loop = None duration = 70 mode = P
#3 loop = None duration = 70 mode = P
#4 loop = None duration = 70 mode = P
#5 loop = None duration = 70 mode = P
#6 loop = None duration = 70 mode = P
#7 loop = None duration = 70 mode = P
#8 loop = None duration = 70 mode = P
#9 loop = None duration = 70 mode = P
#10 loop = None duration = 70 mode = P
#11 loop = None duration = 70 mode = P
#12 loop = None duration = 70 mode = P
#13 loop = None duration = 70 mode = P
#14 loop = None duration = 70 mode = P
#15 loop = None duration = 70 mode = P
#16 loop = None duration = 70 mode = P
#17 loop = None duration = 70 mode = P
#18 loop = None duration = 70 mode = P
#19 loop = None duration = 70 mode = P
#20 loop = None duration = 70 mode = P
Run Code Online (Sandbox Code Playgroud)

结果帧

问题:

  1. 为什么 rgba gif 失去了透明度?背景颜色从何而来

  2. 为什么到处都是黑边?我从使用 GIMP 中了解到可以组合或替换前一个 gif 帧,这是否解释了我所看到的内容还是其他内容?(在这个主题上,Pillow 是否自己处理合并/替换帧 gif?)

  3. 不确定图像大小发生了什么,explorer.exe 说 68x104,但 Pillow 保存的帧都是 85x112(firefox 在查看输入的 gif 文件时也说 85x112)

在这里找到PepePls.gif

我有一个类似的问题与这一个了

编辑:

经过大量研究和阅读枕头源代码https://github.com/python-pillow/Pillow/blob/master/src/PIL/GifImagePlugin.py后,我很确定黑色边框是一个错误,它必须与每个帧都绘制在不适当颜色的背景上的事实相关联(可能根本没有检查调色板/颜色/透明度索引)

这是我frame = image.copy()在上面的代码中实现的替换

frame = Image.new('RGBA', (85, 112), color=(255,0,0,0))
frame.paste(image.crop(image.dispose_extent), box=(image.dispose_extent[0],image.dispose_extent[1]))
Run Code Online (Sandbox Code Playgroud)

可能想使用image.tile[0][1]而不是,image.dispose_extent但我真的不确定

新结果

这似乎也表明 Pillow 根本不喜欢透明的 gif

编辑2:

我修复了黑色边框并使用以下方法组合/替换框架:

#0 loop = 0 duration = 70 mode = P
#1 loop = None duration = 70 mode = P
#2 loop = None duration = 70 mode = P
#3 loop = None duration = 70 mode = P
#4 loop = None duration = 70 mode = P
#5 loop = None duration = 70 mode = P
#6 loop = None duration = 70 mode = P
#7 loop = None duration = 70 mode = P
#8 loop = None duration = 70 mode = P
#9 loop = None duration = 70 mode = P
#10 loop = None duration = 70 mode = P
#11 loop = None duration = 70 mode = P
#12 loop = None duration = 70 mode = P
#13 loop = None duration = 70 mode = P
#14 loop = None duration = 70 mode = P
#15 loop = None duration = 70 mode = P
#16 loop = None duration = 70 mode = P
#17 loop = None duration = 70 mode = P
#18 loop = None duration = 70 mode = P
#19 loop = None duration = 70 mode = P
#20 loop = None duration = 70 mode = P
Run Code Online (Sandbox Code Playgroud)

PeepoCreepo 镜框几乎完美

我仍然看到的一个问题是,如果放大 f2.png,您会看到眼睛中的一些黑色缺失。这不是 gif 文件的问题,因为 gimp 可以正确打开和显示每一帧:

Gimp 中的 PeepoCreepo gif 帧

编辑3:

在 Github 上打开了一个问题

hid*_*kgb 0

这是 PIL 中的一个已知错误,至少可以追溯到 2018 年 2 月,因此我预计在可预见的将来不会有任何修复。

可能的解决方案之一是使用外部 GIF 库,例如我的