2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# The Python Imaging Library.
|
|
|
|
# $Id$
|
|
|
|
#
|
|
|
|
# GIF file handling
|
|
|
|
#
|
|
|
|
# History:
|
|
|
|
# 1995-09-01 fl Created
|
|
|
|
# 1996-12-14 fl Added interlace support
|
|
|
|
# 1996-12-30 fl Added animation support
|
|
|
|
# 1997-01-05 fl Added write support, fixed local colour map bug
|
|
|
|
# 1997-02-23 fl Make sure to load raster data in getdata()
|
|
|
|
# 1997-07-05 fl Support external decoder (0.4)
|
|
|
|
# 1998-07-09 fl Handle all modes when saving (0.5)
|
|
|
|
# 1998-07-15 fl Renamed offset attribute to avoid name clash
|
|
|
|
# 2001-04-16 fl Added rewind support (seek to frame 0) (0.6)
|
|
|
|
# 2001-04-17 fl Added palette optimization (0.7)
|
|
|
|
# 2002-06-06 fl Added transparency support for save (0.8)
|
|
|
|
# 2004-02-24 fl Disable interlacing for small images
|
|
|
|
#
|
|
|
|
# Copyright (c) 1997-2004 by Secret Labs AB
|
|
|
|
# Copyright (c) 1995-2004 by Fredrik Lundh
|
|
|
|
#
|
|
|
|
# See the README file for information on usage and redistribution.
|
|
|
|
#
|
2023-12-21 14:13:31 +03:00
|
|
|
from __future__ import annotations
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2017-03-07 01:46:55 +03:00
|
|
|
import itertools
|
2020-02-08 00:34:53 +03:00
|
|
|
import math
|
2019-10-07 16:28:36 +03:00
|
|
|
import os
|
|
|
|
import subprocess
|
2024-06-10 09:08:06 +03:00
|
|
|
import sys
|
2022-03-20 08:28:31 +03:00
|
|
|
from enum import IntEnum
|
2024-05-22 12:43:00 +03:00
|
|
|
from functools import cached_property
|
2024-06-10 09:08:06 +03:00
|
|
|
from typing import IO, TYPE_CHECKING, Any, List, Literal, NamedTuple, Union
|
2017-03-07 01:46:55 +03:00
|
|
|
|
2023-11-25 11:16:32 +03:00
|
|
|
from . import (
|
|
|
|
Image,
|
|
|
|
ImageChops,
|
|
|
|
ImageFile,
|
|
|
|
ImageMath,
|
|
|
|
ImageOps,
|
|
|
|
ImagePalette,
|
|
|
|
ImageSequence,
|
|
|
|
)
|
2020-09-01 20:16:46 +03:00
|
|
|
from ._binary import i16le as i16
|
|
|
|
from ._binary import o8
|
|
|
|
from ._binary import o16le as o16
|
2019-07-06 23:40:53 +03:00
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
if TYPE_CHECKING:
|
|
|
|
from . import _imaging
|
|
|
|
|
2022-03-20 08:28:31 +03:00
|
|
|
|
2022-03-29 13:26:29 +03:00
|
|
|
class LoadingStrategy(IntEnum):
|
2022-03-28 14:02:26 +03:00
|
|
|
""".. versionadded:: 9.1.0"""
|
|
|
|
|
2022-03-29 13:26:29 +03:00
|
|
|
RGB_AFTER_FIRST = 0
|
|
|
|
RGB_AFTER_DIFFERENT_PALETTE_ONLY = 1
|
|
|
|
RGB_ALWAYS = 2
|
2022-03-20 08:28:31 +03:00
|
|
|
|
|
|
|
|
2022-03-28 14:02:26 +03:00
|
|
|
#: .. versionadded:: 9.1.0
|
2022-03-29 13:26:29 +03:00
|
|
|
LOADING_STRATEGY = LoadingStrategy.RGB_AFTER_FIRST
|
2022-03-20 08:28:31 +03:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
# --------------------------------------------------------------------
|
|
|
|
# Identify/read GIF files
|
|
|
|
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2024-04-06 05:58:53 +03:00
|
|
|
def _accept(prefix: bytes) -> bool:
|
py3k: The big push
There are two main issues fixed with this commit:
* bytes vs. str: All file, image, and palette data are now handled as
bytes. A new _binary module consolidates the hacks needed to do this
across Python versions. tostring/fromstring methods have been renamed to
tobytes/frombytes, but the Python 2.6/2.7 versions alias them to the old
names for compatibility. Users should move to tobytes/frombytes.
One other potentially-breaking change is that text data in image files
(such as tags, comments) are now explicitly handled with a specific
character encoding in mind. This works well with the Unicode str in
Python 3, but may trip up old code expecting a straight byte-for-byte
translation to a Python string. This also required a change to Gohlke's
tags tests (in Tests/test_file_png.py) to expect Unicode strings from
the code.
* True div vs. floor div: Many division operations used the "/" operator
to do floor division, which is now the "//" operator in Python 3. These
were fixed.
As of this commit, on the first pass, I have one failing test (improper
handling of a slice object in a C module, test_imagepath.py) in Python 3,
and three that that I haven't tried running yet (test_imagegl,
test_imagegrab, and test_imageqt). I also haven't tested anything on
Windows. All but the three skipped tests run flawlessly against Pythons
2.6 and 2.7.
2012-10-21 01:01:53 +04:00
|
|
|
return prefix[:6] in [b"GIF87a", b"GIF89a"]
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
##
|
|
|
|
# Image plugin for GIF images. This plugin supports both GIF87 and
|
|
|
|
# GIF89 images.
|
|
|
|
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
class GifImageFile(ImageFile.ImageFile):
|
|
|
|
format = "GIF"
|
|
|
|
format_description = "Compuserve GIF"
|
2017-03-15 02:16:38 +03:00
|
|
|
_close_exclusive_fp_after_loading = False
|
2017-09-30 06:32:43 +03:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
global_palette = None
|
|
|
|
|
2024-05-04 13:51:54 +03:00
|
|
|
def data(self) -> bytes | None:
|
2010-07-31 06:52:47 +04:00
|
|
|
s = self.fp.read(1)
|
2020-05-08 19:48:02 +03:00
|
|
|
if s and s[0]:
|
|
|
|
return self.fp.read(s[0])
|
2010-07-31 06:52:47 +04:00
|
|
|
return None
|
|
|
|
|
2024-05-13 11:47:51 +03:00
|
|
|
def _is_palette_needed(self, p: bytes) -> bool:
|
2022-03-22 14:07:37 +03:00
|
|
|
for i in range(0, len(p), 3):
|
|
|
|
if not (i // 3 == p[i] == p[i + 1] == p[i + 2]):
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
2024-05-04 13:51:54 +03:00
|
|
|
def _open(self) -> None:
|
2010-07-31 06:52:47 +04:00
|
|
|
# Screen
|
|
|
|
s = self.fp.read(13)
|
2020-06-23 10:41:13 +03:00
|
|
|
if not _accept(s):
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "not a GIF file"
|
|
|
|
raise SyntaxError(msg)
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
self.info["version"] = s[:6]
|
2020-05-08 21:11:58 +03:00
|
|
|
self._size = i16(s, 6), i16(s, 8)
|
2010-07-31 06:52:47 +04:00
|
|
|
self.tile = []
|
2020-05-08 19:48:02 +03:00
|
|
|
flags = s[10]
|
2010-07-31 06:52:47 +04:00
|
|
|
bits = (flags & 7) + 1
|
|
|
|
|
|
|
|
if flags & 128:
|
|
|
|
# get global palette
|
2020-05-08 19:48:02 +03:00
|
|
|
self.info["background"] = s[11]
|
2010-07-31 06:52:47 +04:00
|
|
|
# check if palette contains colour indices
|
|
|
|
p = self.fp.read(3 << bits)
|
2022-03-22 14:07:37 +03:00
|
|
|
if self._is_palette_needed(p):
|
|
|
|
p = ImagePalette.raw("RGB", p)
|
|
|
|
self.global_palette = self.palette = p
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2022-04-13 02:54:17 +03:00
|
|
|
self._fp = self.fp # FIXME: hack
|
2010-07-31 06:52:47 +04:00
|
|
|
self.__rewind = self.fp.tell()
|
2024-05-22 12:43:00 +03:00
|
|
|
self._n_frames: int | None = None
|
2015-04-15 03:43:05 +03:00
|
|
|
self._seek(0) # get ready to read first frame
|
|
|
|
|
|
|
|
@property
|
2024-06-10 09:08:06 +03:00
|
|
|
def n_frames(self) -> int:
|
2015-04-15 03:43:05 +03:00
|
|
|
if self._n_frames is None:
|
|
|
|
current = self.tell()
|
|
|
|
try:
|
|
|
|
while True:
|
2022-02-21 10:34:05 +03:00
|
|
|
self._seek(self.tell() + 1, False)
|
2015-04-15 03:43:05 +03:00
|
|
|
except EOFError:
|
|
|
|
self._n_frames = self.tell() + 1
|
|
|
|
self.seek(current)
|
|
|
|
return self._n_frames
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2024-05-22 12:43:00 +03:00
|
|
|
@cached_property
|
|
|
|
def is_animated(self) -> bool:
|
|
|
|
if self._n_frames is not None:
|
|
|
|
return self._n_frames != 1
|
|
|
|
|
|
|
|
current = self.tell()
|
|
|
|
if current:
|
|
|
|
return True
|
|
|
|
|
|
|
|
try:
|
|
|
|
self._seek(1, False)
|
|
|
|
is_animated = True
|
|
|
|
except EOFError:
|
|
|
|
is_animated = False
|
|
|
|
|
|
|
|
self.seek(current)
|
|
|
|
return is_animated
|
2015-06-30 06:25:00 +03:00
|
|
|
|
2024-05-04 13:51:54 +03:00
|
|
|
def seek(self, frame: int) -> None:
|
2017-09-30 06:32:43 +03:00
|
|
|
if not self._seek_check(frame):
|
2015-04-15 03:43:05 +03:00
|
|
|
return
|
|
|
|
if frame < self.__frame:
|
2021-11-29 09:49:06 +03:00
|
|
|
self.im = None
|
2015-04-15 03:43:05 +03:00
|
|
|
self._seek(0)
|
2015-06-18 17:49:18 +03:00
|
|
|
|
|
|
|
last_frame = self.__frame
|
2015-04-15 03:43:05 +03:00
|
|
|
for f in range(self.__frame + 1, frame + 1):
|
2015-06-18 17:49:18 +03:00
|
|
|
try:
|
|
|
|
self._seek(f)
|
2020-06-21 13:13:35 +03:00
|
|
|
except EOFError as e:
|
2015-06-18 17:49:18 +03:00
|
|
|
self.seek(last_frame)
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "no more images in GIF file"
|
|
|
|
raise EOFError(msg) from e
|
2015-04-15 03:43:05 +03:00
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
def _seek(self, frame: int, update_image: bool = True) -> None:
|
2010-07-31 06:52:47 +04:00
|
|
|
if frame == 0:
|
|
|
|
# rewind
|
|
|
|
self.__offset = 0
|
2024-06-10 09:08:06 +03:00
|
|
|
self.dispose: _imaging.ImagingCore | None = None
|
2010-07-31 06:52:47 +04:00
|
|
|
self.__frame = -1
|
2022-04-13 02:54:17 +03:00
|
|
|
self._fp.seek(self.__rewind)
|
2014-07-03 20:48:12 +04:00
|
|
|
self.disposal_method = 0
|
2022-05-22 08:30:16 +03:00
|
|
|
if "comment" in self.info:
|
|
|
|
del self.info["comment"]
|
2014-07-07 22:46:54 +04:00
|
|
|
else:
|
|
|
|
# ensure that the previous frame was loaded
|
2022-02-21 10:34:05 +03:00
|
|
|
if self.tile and update_image:
|
2014-07-07 22:46:54 +04:00
|
|
|
self.load()
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
if frame != self.__frame + 1:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = f"cannot seek to frame {frame}"
|
|
|
|
raise ValueError(msg)
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2022-04-13 02:54:17 +03:00
|
|
|
self.fp = self._fp
|
2010-07-31 06:52:47 +04:00
|
|
|
if self.__offset:
|
|
|
|
# backup to last frame
|
|
|
|
self.fp.seek(self.__offset)
|
|
|
|
while self.data():
|
|
|
|
pass
|
|
|
|
self.__offset = 0
|
|
|
|
|
2022-02-21 09:35:50 +03:00
|
|
|
s = self.fp.read(1)
|
|
|
|
if not s or s == b";":
|
2023-10-19 10:42:41 +03:00
|
|
|
msg = "no more images in GIF file"
|
|
|
|
raise EOFError(msg)
|
2022-02-21 09:35:50 +03:00
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
palette: ImagePalette.ImagePalette | Literal[False] | None = None
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
info: dict[str, Any] = {}
|
2021-03-16 16:24:57 +03:00
|
|
|
frame_transparency = None
|
|
|
|
interlace = None
|
2022-03-22 12:28:49 +03:00
|
|
|
frame_dispose_extent = None
|
2012-10-17 07:39:56 +04:00
|
|
|
while True:
|
2022-02-21 09:35:50 +03:00
|
|
|
if not s:
|
|
|
|
s = self.fp.read(1)
|
py3k: The big push
There are two main issues fixed with this commit:
* bytes vs. str: All file, image, and palette data are now handled as
bytes. A new _binary module consolidates the hacks needed to do this
across Python versions. tostring/fromstring methods have been renamed to
tobytes/frombytes, but the Python 2.6/2.7 versions alias them to the old
names for compatibility. Users should move to tobytes/frombytes.
One other potentially-breaking change is that text data in image files
(such as tags, comments) are now explicitly handled with a specific
character encoding in mind. This works well with the Unicode str in
Python 3, but may trip up old code expecting a straight byte-for-byte
translation to a Python string. This also required a change to Gohlke's
tags tests (in Tests/test_file_png.py) to expect Unicode strings from
the code.
* True div vs. floor div: Many division operations used the "/" operator
to do floor division, which is now the "//" operator in Python 3. These
were fixed.
As of this commit, on the first pass, I have one failing test (improper
handling of a slice object in a C module, test_imagepath.py) in Python 3,
and three that that I haven't tried running yet (test_imagegl,
test_imagegrab, and test_imageqt). I also haven't tested anything on
Windows. All but the three skipped tests run flawlessly against Pythons
2.6 and 2.7.
2012-10-21 01:01:53 +04:00
|
|
|
if not s or s == b";":
|
2010-07-31 06:52:47 +04:00
|
|
|
break
|
|
|
|
|
py3k: The big push
There are two main issues fixed with this commit:
* bytes vs. str: All file, image, and palette data are now handled as
bytes. A new _binary module consolidates the hacks needed to do this
across Python versions. tostring/fromstring methods have been renamed to
tobytes/frombytes, but the Python 2.6/2.7 versions alias them to the old
names for compatibility. Users should move to tobytes/frombytes.
One other potentially-breaking change is that text data in image files
(such as tags, comments) are now explicitly handled with a specific
character encoding in mind. This works well with the Unicode str in
Python 3, but may trip up old code expecting a straight byte-for-byte
translation to a Python string. This also required a change to Gohlke's
tags tests (in Tests/test_file_png.py) to expect Unicode strings from
the code.
* True div vs. floor div: Many division operations used the "/" operator
to do floor division, which is now the "//" operator in Python 3. These
were fixed.
As of this commit, on the first pass, I have one failing test (improper
handling of a slice object in a C module, test_imagepath.py) in Python 3,
and three that that I haven't tried running yet (test_imagegl,
test_imagegrab, and test_imageqt). I also haven't tested anything on
Windows. All but the three skipped tests run flawlessly against Pythons
2.6 and 2.7.
2012-10-21 01:01:53 +04:00
|
|
|
elif s == b"!":
|
2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# extensions
|
|
|
|
#
|
|
|
|
s = self.fp.read(1)
|
|
|
|
block = self.data()
|
2024-06-10 09:08:06 +03:00
|
|
|
if s[0] == 249 and block is not None:
|
2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# graphic control extension
|
|
|
|
#
|
2020-05-08 19:48:02 +03:00
|
|
|
flags = block[0]
|
2010-07-31 06:52:47 +04:00
|
|
|
if flags & 1:
|
2021-03-16 16:24:57 +03:00
|
|
|
frame_transparency = block[3]
|
2020-05-08 21:11:58 +03:00
|
|
|
info["duration"] = i16(block, 1) * 10
|
2014-07-02 23:27:52 +04:00
|
|
|
|
|
|
|
# disposal method - find the value of bits 4 - 6
|
2014-07-03 21:01:18 +04:00
|
|
|
dispose_bits = 0b00011100 & flags
|
|
|
|
dispose_bits = dispose_bits >> 2
|
|
|
|
if dispose_bits:
|
|
|
|
# only set the dispose if it is not
|
|
|
|
# unspecified. I'm not sure if this is
|
|
|
|
# correct, but it seems to prevent the last
|
|
|
|
# frame from looking odd for some animations
|
|
|
|
self.disposal_method = dispose_bits
|
2020-05-08 19:48:02 +03:00
|
|
|
elif s[0] == 254:
|
2016-05-07 06:57:40 +03:00
|
|
|
#
|
|
|
|
# comment extension
|
|
|
|
#
|
2022-05-13 20:38:39 +03:00
|
|
|
comment = b""
|
2022-05-22 07:11:11 +03:00
|
|
|
|
2022-05-22 08:30:16 +03:00
|
|
|
# Read this comment block
|
2018-11-27 13:05:41 +03:00
|
|
|
while block:
|
2022-05-13 20:38:39 +03:00
|
|
|
comment += block
|
2018-11-27 13:05:41 +03:00
|
|
|
block = self.data()
|
2022-05-13 20:38:39 +03:00
|
|
|
|
|
|
|
if "comment" in info:
|
2022-05-22 07:11:11 +03:00
|
|
|
# If multiple comment blocks in frame, separate with \n
|
|
|
|
info["comment"] += b"\n" + comment
|
2022-05-13 20:38:39 +03:00
|
|
|
else:
|
|
|
|
info["comment"] = comment
|
2022-02-21 09:35:50 +03:00
|
|
|
s = None
|
2018-11-27 13:05:41 +03:00
|
|
|
continue
|
2024-06-10 09:08:06 +03:00
|
|
|
elif s[0] == 255 and frame == 0 and block is not None:
|
2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# application extension
|
|
|
|
#
|
2018-09-01 02:28:22 +03:00
|
|
|
info["extension"] = block, self.fp.tell()
|
py3k: The big push
There are two main issues fixed with this commit:
* bytes vs. str: All file, image, and palette data are now handled as
bytes. A new _binary module consolidates the hacks needed to do this
across Python versions. tostring/fromstring methods have been renamed to
tobytes/frombytes, but the Python 2.6/2.7 versions alias them to the old
names for compatibility. Users should move to tobytes/frombytes.
One other potentially-breaking change is that text data in image files
(such as tags, comments) are now explicitly handled with a specific
character encoding in mind. This works well with the Unicode str in
Python 3, but may trip up old code expecting a straight byte-for-byte
translation to a Python string. This also required a change to Gohlke's
tags tests (in Tests/test_file_png.py) to expect Unicode strings from
the code.
* True div vs. floor div: Many division operations used the "/" operator
to do floor division, which is now the "//" operator in Python 3. These
were fixed.
As of this commit, on the first pass, I have one failing test (improper
handling of a slice object in a C module, test_imagepath.py) in Python 3,
and three that that I haven't tried running yet (test_imagegl,
test_imagegrab, and test_imageqt). I also haven't tested anything on
Windows. All but the three skipped tests run flawlessly against Pythons
2.6 and 2.7.
2012-10-21 01:01:53 +04:00
|
|
|
if block[:11] == b"NETSCAPE2.0":
|
2010-07-31 06:52:47 +04:00
|
|
|
block = self.data()
|
2024-06-10 09:08:06 +03:00
|
|
|
if block and len(block) >= 3 and block[0] == 1:
|
2022-04-15 09:44:23 +03:00
|
|
|
self.info["loop"] = i16(block, 1)
|
2010-07-31 06:52:47 +04:00
|
|
|
while self.data():
|
|
|
|
pass
|
|
|
|
|
py3k: The big push
There are two main issues fixed with this commit:
* bytes vs. str: All file, image, and palette data are now handled as
bytes. A new _binary module consolidates the hacks needed to do this
across Python versions. tostring/fromstring methods have been renamed to
tobytes/frombytes, but the Python 2.6/2.7 versions alias them to the old
names for compatibility. Users should move to tobytes/frombytes.
One other potentially-breaking change is that text data in image files
(such as tags, comments) are now explicitly handled with a specific
character encoding in mind. This works well with the Unicode str in
Python 3, but may trip up old code expecting a straight byte-for-byte
translation to a Python string. This also required a change to Gohlke's
tags tests (in Tests/test_file_png.py) to expect Unicode strings from
the code.
* True div vs. floor div: Many division operations used the "/" operator
to do floor division, which is now the "//" operator in Python 3. These
were fixed.
As of this commit, on the first pass, I have one failing test (improper
handling of a slice object in a C module, test_imagepath.py) in Python 3,
and three that that I haven't tried running yet (test_imagegl,
test_imagegrab, and test_imageqt). I also haven't tested anything on
Windows. All but the three skipped tests run flawlessly against Pythons
2.6 and 2.7.
2012-10-21 01:01:53 +04:00
|
|
|
elif s == b",":
|
2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# local image
|
|
|
|
#
|
|
|
|
s = self.fp.read(9)
|
|
|
|
|
|
|
|
# extent
|
2020-05-08 21:11:58 +03:00
|
|
|
x0, y0 = i16(s, 0), i16(s, 2)
|
|
|
|
x1, y1 = x0 + i16(s, 4), y0 + i16(s, 6)
|
2022-03-21 15:19:26 +03:00
|
|
|
if (x1 > self.size[0] or y1 > self.size[1]) and update_image:
|
2019-05-02 12:46:17 +03:00
|
|
|
self._size = max(x1, self.size[0]), max(y1, self.size[1])
|
2022-06-30 05:47:35 +03:00
|
|
|
Image._decompression_bomb_check(self._size)
|
2022-03-22 12:28:49 +03:00
|
|
|
frame_dispose_extent = x0, y0, x1, y1
|
2020-05-08 19:48:02 +03:00
|
|
|
flags = s[8]
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
interlace = (flags & 64) != 0
|
|
|
|
|
|
|
|
if flags & 128:
|
|
|
|
bits = (flags & 7) + 1
|
2022-03-22 14:07:37 +03:00
|
|
|
p = self.fp.read(3 << bits)
|
|
|
|
if self._is_palette_needed(p):
|
|
|
|
palette = ImagePalette.raw("RGB", p)
|
2022-10-06 00:46:31 +03:00
|
|
|
else:
|
|
|
|
palette = False
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
# image data
|
2020-05-08 19:48:02 +03:00
|
|
|
bits = self.fp.read(1)[0]
|
2010-07-31 06:52:47 +04:00
|
|
|
self.__offset = self.fp.tell()
|
|
|
|
break
|
2022-02-21 09:35:50 +03:00
|
|
|
s = None
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2022-02-21 10:34:05 +03:00
|
|
|
if interlace is None:
|
2023-10-19 10:42:41 +03:00
|
|
|
msg = "image not found in GIF frame"
|
|
|
|
raise EOFError(msg)
|
2022-04-16 10:07:39 +03:00
|
|
|
|
|
|
|
self.__frame = frame
|
2022-02-21 10:34:05 +03:00
|
|
|
if not update_image:
|
|
|
|
return
|
|
|
|
|
2022-07-21 02:05:14 +03:00
|
|
|
self.tile = []
|
|
|
|
|
2022-03-22 12:28:49 +03:00
|
|
|
if self.dispose:
|
|
|
|
self.im.paste(self.dispose, self.dispose_extent)
|
|
|
|
|
2022-10-06 00:46:31 +03:00
|
|
|
self._frame_palette = palette if palette is not None else self.global_palette
|
2022-09-17 10:56:36 +03:00
|
|
|
self._frame_transparency = frame_transparency
|
2022-03-22 12:28:49 +03:00
|
|
|
if frame == 0:
|
|
|
|
if self._frame_palette:
|
2022-09-17 10:56:36 +03:00
|
|
|
if LOADING_STRATEGY == LoadingStrategy.RGB_ALWAYS:
|
2023-07-29 02:28:18 +03:00
|
|
|
self._mode = "RGBA" if frame_transparency is not None else "RGB"
|
2022-09-17 10:56:36 +03:00
|
|
|
else:
|
2023-07-29 02:28:18 +03:00
|
|
|
self._mode = "P"
|
2022-03-22 12:28:49 +03:00
|
|
|
else:
|
2023-07-29 02:28:18 +03:00
|
|
|
self._mode = "L"
|
2022-03-22 12:28:49 +03:00
|
|
|
|
|
|
|
if not palette and self.global_palette:
|
|
|
|
from copy import copy
|
|
|
|
|
|
|
|
palette = copy(self.global_palette)
|
|
|
|
self.palette = palette
|
|
|
|
else:
|
|
|
|
if self.mode == "P":
|
2022-03-29 13:26:29 +03:00
|
|
|
if (
|
|
|
|
LOADING_STRATEGY != LoadingStrategy.RGB_AFTER_DIFFERENT_PALETTE_ONLY
|
|
|
|
or palette
|
|
|
|
):
|
2022-03-22 12:28:49 +03:00
|
|
|
self.pyaccess = None
|
|
|
|
if "transparency" in self.info:
|
|
|
|
self.im.putpalettealpha(self.info["transparency"], 0)
|
|
|
|
self.im = self.im.convert("RGBA", Image.Dither.FLOYDSTEINBERG)
|
2023-07-29 02:28:18 +03:00
|
|
|
self._mode = "RGBA"
|
2022-03-22 12:28:49 +03:00
|
|
|
del self.info["transparency"]
|
|
|
|
else:
|
2023-07-29 02:28:18 +03:00
|
|
|
self._mode = "RGB"
|
2022-03-22 12:28:49 +03:00
|
|
|
self.im = self.im.convert("RGB", Image.Dither.FLOYDSTEINBERG)
|
2021-11-29 09:49:06 +03:00
|
|
|
|
2024-06-05 01:29:28 +03:00
|
|
|
def _rgb(color: int) -> tuple[int, int, int]:
|
2022-03-22 12:28:49 +03:00
|
|
|
if self._frame_palette:
|
2023-11-02 08:05:13 +03:00
|
|
|
if color * 3 + 3 > len(self._frame_palette.palette):
|
|
|
|
color = 0
|
2024-06-05 01:29:28 +03:00
|
|
|
return tuple(self._frame_palette.palette[color * 3 : color * 3 + 3])
|
2021-11-29 09:49:06 +03:00
|
|
|
else:
|
2024-06-05 01:29:28 +03:00
|
|
|
return (color, color, color)
|
2021-11-29 09:49:06 +03:00
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
self.dispose = None
|
2022-03-22 12:28:49 +03:00
|
|
|
self.dispose_extent = frame_dispose_extent
|
2024-06-10 09:08:06 +03:00
|
|
|
if self.dispose_extent and self.disposal_method >= 2:
|
|
|
|
try:
|
|
|
|
if self.disposal_method == 2:
|
|
|
|
# replace with background colour
|
|
|
|
|
2021-03-14 05:31:16 +03:00
|
|
|
# only dispose the extent in this frame
|
2021-04-15 12:01:12 +03:00
|
|
|
x0, y0, x1, y1 = self.dispose_extent
|
|
|
|
dispose_size = (x1 - x0, y1 - y0)
|
|
|
|
|
|
|
|
Image._decompression_bomb_check(dispose_size)
|
2024-06-10 09:08:06 +03:00
|
|
|
|
|
|
|
# by convention, attempt to use transparency first
|
2022-03-22 12:28:49 +03:00
|
|
|
dispose_mode = "P"
|
2024-06-10 09:08:06 +03:00
|
|
|
color = self.info.get("transparency", frame_transparency)
|
|
|
|
if color is not None:
|
|
|
|
if self.mode in ("RGB", "RGBA"):
|
|
|
|
dispose_mode = "RGBA"
|
|
|
|
color = _rgb(color) + (0,)
|
|
|
|
else:
|
|
|
|
color = self.info.get("background", 0)
|
|
|
|
if self.mode in ("RGB", "RGBA"):
|
|
|
|
dispose_mode = "RGB"
|
|
|
|
color = _rgb(color)
|
2022-03-22 12:28:49 +03:00
|
|
|
self.dispose = Image.core.fill(dispose_mode, dispose_size, color)
|
2024-06-10 09:08:06 +03:00
|
|
|
else:
|
|
|
|
# replace with previous contents
|
|
|
|
if self.im is not None:
|
|
|
|
# only dispose the extent in this frame
|
|
|
|
self.dispose = self._crop(self.im, self.dispose_extent)
|
|
|
|
elif frame_transparency is not None:
|
|
|
|
x0, y0, x1, y1 = self.dispose_extent
|
|
|
|
dispose_size = (x1 - x0, y1 - y0)
|
|
|
|
|
|
|
|
Image._decompression_bomb_check(dispose_size)
|
|
|
|
dispose_mode = "P"
|
|
|
|
color = frame_transparency
|
|
|
|
if self.mode in ("RGB", "RGBA"):
|
|
|
|
dispose_mode = "RGBA"
|
|
|
|
color = _rgb(frame_transparency) + (0,)
|
|
|
|
self.dispose = Image.core.fill(
|
|
|
|
dispose_mode, dispose_size, color
|
|
|
|
)
|
|
|
|
except AttributeError:
|
|
|
|
pass
|
2014-07-02 23:27:52 +04:00
|
|
|
|
2021-03-16 16:24:57 +03:00
|
|
|
if interlace is not None:
|
2022-03-22 12:28:49 +03:00
|
|
|
transparency = -1
|
|
|
|
if frame_transparency is not None:
|
|
|
|
if frame == 0:
|
2022-09-17 10:56:36 +03:00
|
|
|
if LOADING_STRATEGY != LoadingStrategy.RGB_ALWAYS:
|
|
|
|
self.info["transparency"] = frame_transparency
|
2022-03-22 14:07:37 +03:00
|
|
|
elif self.mode not in ("RGB", "RGBA"):
|
2022-03-22 12:28:49 +03:00
|
|
|
transparency = frame_transparency
|
2021-03-16 16:24:57 +03:00
|
|
|
self.tile = [
|
|
|
|
(
|
|
|
|
"gif",
|
|
|
|
(x0, y0, x1, y1),
|
|
|
|
self.__offset,
|
2022-03-22 12:28:49 +03:00
|
|
|
(bits, interlace, transparency),
|
2021-03-16 16:24:57 +03:00
|
|
|
)
|
|
|
|
]
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2022-05-22 08:30:16 +03:00
|
|
|
if info.get("comment"):
|
|
|
|
self.info["comment"] = info["comment"]
|
2022-04-15 09:44:23 +03:00
|
|
|
for k in ["duration", "extension"]:
|
2018-09-01 02:28:22 +03:00
|
|
|
if k in info:
|
|
|
|
self.info[k] = info[k]
|
|
|
|
elif k in self.info:
|
|
|
|
del self.info[k]
|
|
|
|
|
2024-05-04 13:51:54 +03:00
|
|
|
def load_prepare(self) -> None:
|
2022-03-22 12:28:49 +03:00
|
|
|
temp_mode = "P" if self._frame_palette else "L"
|
|
|
|
self._prev_im = None
|
2021-11-29 09:49:06 +03:00
|
|
|
if self.__frame == 0:
|
2022-09-17 10:56:36 +03:00
|
|
|
if self._frame_transparency is not None:
|
2021-11-29 09:49:06 +03:00
|
|
|
self.im = Image.core.fill(
|
2022-09-17 10:56:36 +03:00
|
|
|
temp_mode, self.size, self._frame_transparency
|
2021-11-29 09:49:06 +03:00
|
|
|
)
|
2022-03-22 14:07:37 +03:00
|
|
|
elif self.mode in ("RGB", "RGBA"):
|
2021-11-29 09:49:06 +03:00
|
|
|
self._prev_im = self.im
|
|
|
|
if self._frame_palette:
|
|
|
|
self.im = Image.core.fill("P", self.size, self._frame_transparency or 0)
|
|
|
|
self.im.putpalette(*self._frame_palette.getdata())
|
|
|
|
else:
|
|
|
|
self.im = None
|
2023-07-29 02:28:18 +03:00
|
|
|
self._mode = temp_mode
|
2022-03-20 08:28:31 +03:00
|
|
|
self._frame_palette = None
|
2021-06-25 14:54:21 +03:00
|
|
|
|
2021-10-15 13:10:22 +03:00
|
|
|
super().load_prepare()
|
2021-06-25 14:54:21 +03:00
|
|
|
|
2024-05-04 13:51:54 +03:00
|
|
|
def load_end(self) -> None:
|
2021-11-29 09:49:06 +03:00
|
|
|
if self.__frame == 0:
|
2022-03-29 13:26:29 +03:00
|
|
|
if self.mode == "P" and LOADING_STRATEGY == LoadingStrategy.RGB_ALWAYS:
|
2022-09-17 10:56:36 +03:00
|
|
|
if self._frame_transparency is not None:
|
|
|
|
self.im.putpalettealpha(self._frame_transparency, 0)
|
2023-07-29 02:28:18 +03:00
|
|
|
self._mode = "RGBA"
|
2022-09-17 10:56:36 +03:00
|
|
|
else:
|
2023-07-29 02:28:18 +03:00
|
|
|
self._mode = "RGB"
|
2022-09-17 10:56:36 +03:00
|
|
|
self.im = self.im.convert(self.mode, Image.Dither.FLOYDSTEINBERG)
|
2021-11-29 09:49:06 +03:00
|
|
|
return
|
2022-09-13 17:05:23 +03:00
|
|
|
if not self._prev_im:
|
|
|
|
return
|
|
|
|
if self._frame_transparency is not None:
|
|
|
|
self.im.putpalettealpha(self._frame_transparency, 0)
|
|
|
|
frame_im = self.im.convert("RGBA")
|
2021-11-29 09:49:06 +03:00
|
|
|
else:
|
2022-09-13 17:05:23 +03:00
|
|
|
frame_im = self.im.convert("RGB")
|
2024-06-18 15:44:17 +03:00
|
|
|
|
|
|
|
assert self.dispose_extent is not None
|
2021-11-29 09:49:06 +03:00
|
|
|
frame_im = self._crop(frame_im, self.dispose_extent)
|
|
|
|
|
|
|
|
self.im = self._prev_im
|
2023-07-29 02:28:18 +03:00
|
|
|
self._mode = self.im.mode
|
2022-03-22 14:07:37 +03:00
|
|
|
if frame_im.mode == "RGBA":
|
2021-11-29 09:49:06 +03:00
|
|
|
self.im.paste(frame_im, self.dispose_extent, frame_im)
|
|
|
|
else:
|
|
|
|
self.im.paste(frame_im, self.dispose_extent)
|
|
|
|
|
2024-05-04 13:51:54 +03:00
|
|
|
def tell(self) -> int:
|
2010-07-31 06:52:47 +04:00
|
|
|
return self.__frame
|
|
|
|
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
# --------------------------------------------------------------------
|
|
|
|
# Write GIF files
|
|
|
|
|
2018-03-03 12:54:00 +03:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
RAWMODE = {"1": "L", "L": "L", "P": "P"}
|
|
|
|
|
2015-07-01 02:18:05 +03:00
|
|
|
|
2024-05-13 11:47:51 +03:00
|
|
|
def _normalize_mode(im: Image.Image) -> Image.Image:
|
2017-03-03 18:51:07 +03:00
|
|
|
"""
|
|
|
|
Takes an image (or frame), returns an image in a mode that is appropriate
|
|
|
|
for saving in a Gif.
|
|
|
|
|
|
|
|
It may return the original image, or it may return an image converted to
|
|
|
|
palette or 'L' mode.
|
|
|
|
|
|
|
|
:param im: Image object
|
|
|
|
:returns: Image object
|
|
|
|
"""
|
2017-02-23 17:58:37 +03:00
|
|
|
if im.mode in RAWMODE:
|
2017-02-23 18:37:14 +03:00
|
|
|
im.load()
|
2017-02-23 17:58:37 +03:00
|
|
|
return im
|
2015-07-01 02:18:05 +03:00
|
|
|
if Image.getmodebase(im.mode) == "RGB":
|
2022-03-12 07:54:08 +03:00
|
|
|
im = im.convert("P", palette=Image.Palette.ADAPTIVE)
|
2022-03-12 07:14:36 +03:00
|
|
|
if im.palette.mode == "RGBA":
|
2023-01-07 21:36:17 +03:00
|
|
|
for rgba in im.palette.colors:
|
2022-03-12 07:14:36 +03:00
|
|
|
if rgba[3] == 0:
|
|
|
|
im.info["transparency"] = im.palette.colors[rgba]
|
|
|
|
break
|
|
|
|
return im
|
2015-07-01 02:18:05 +03:00
|
|
|
return im.convert("L")
|
|
|
|
|
2017-04-20 14:14:23 +03:00
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
_Palette = Union[bytes, bytearray, List[int], ImagePalette.ImagePalette]
|
|
|
|
|
|
|
|
|
|
|
|
def _normalize_palette(
|
|
|
|
im: Image.Image, palette: _Palette | None, info: dict[str, Any]
|
|
|
|
) -> Image.Image:
|
2017-03-03 18:51:31 +03:00
|
|
|
"""
|
|
|
|
Normalizes the palette for image.
|
|
|
|
- Sets the palette to the incoming palette, if provided.
|
|
|
|
- Ensures that there's a palette for L mode images
|
|
|
|
- Optimizes the palette if necessary/desired.
|
|
|
|
|
|
|
|
:param im: Image object
|
|
|
|
:param palette: bytes object containing the source palette, or ....
|
2017-04-20 14:14:23 +03:00
|
|
|
:param info: encoderinfo
|
2017-03-03 18:51:31 +03:00
|
|
|
:returns: Image object
|
|
|
|
"""
|
2017-03-07 01:46:55 +03:00
|
|
|
source_palette = None
|
|
|
|
if palette:
|
|
|
|
# a bytes palette
|
|
|
|
if isinstance(palette, (bytes, bytearray, list)):
|
|
|
|
source_palette = bytearray(palette[:768])
|
|
|
|
if isinstance(palette, ImagePalette.ImagePalette):
|
2021-07-09 17:00:50 +03:00
|
|
|
source_palette = bytearray(palette.palette)
|
2017-04-20 14:14:23 +03:00
|
|
|
|
2017-03-03 18:51:31 +03:00
|
|
|
if im.mode == "P":
|
2017-03-07 01:46:55 +03:00
|
|
|
if not source_palette:
|
2017-03-03 18:51:31 +03:00
|
|
|
source_palette = im.im.getpalette("RGB")[:768]
|
|
|
|
else: # L-mode
|
2017-03-07 01:46:55 +03:00
|
|
|
if not source_palette:
|
2017-03-03 18:51:31 +03:00
|
|
|
source_palette = bytearray(i // 3 for i in range(768))
|
|
|
|
im.palette = ImagePalette.ImagePalette("RGB", palette=source_palette)
|
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
used_palette_colors: list[int] | None
|
2021-07-11 15:52:32 +03:00
|
|
|
if palette:
|
|
|
|
used_palette_colors = []
|
2024-06-10 09:08:06 +03:00
|
|
|
assert source_palette is not None
|
2021-07-11 15:52:32 +03:00
|
|
|
for i in range(0, len(source_palette), 3):
|
|
|
|
source_color = tuple(source_palette[i : i + 3])
|
2022-08-30 01:08:01 +03:00
|
|
|
index = im.palette.colors.get(source_color)
|
2022-08-29 16:20:31 +03:00
|
|
|
if index in used_palette_colors:
|
2021-07-11 15:52:32 +03:00
|
|
|
index = None
|
|
|
|
used_palette_colors.append(index)
|
|
|
|
for i, index in enumerate(used_palette_colors):
|
|
|
|
if index is None:
|
|
|
|
for j in range(len(used_palette_colors)):
|
|
|
|
if j not in used_palette_colors:
|
|
|
|
used_palette_colors[i] = j
|
|
|
|
break
|
|
|
|
im = im.remap_palette(used_palette_colors)
|
|
|
|
else:
|
|
|
|
used_palette_colors = _get_optimize(im, info)
|
|
|
|
if used_palette_colors is not None:
|
2023-11-25 11:16:32 +03:00
|
|
|
im = im.remap_palette(used_palette_colors, source_palette)
|
|
|
|
if "transparency" in info:
|
|
|
|
try:
|
|
|
|
info["transparency"] = used_palette_colors.index(
|
|
|
|
info["transparency"]
|
|
|
|
)
|
|
|
|
except ValueError:
|
|
|
|
del info["transparency"]
|
|
|
|
return im
|
2017-03-03 18:51:31 +03:00
|
|
|
|
|
|
|
im.palette.palette = source_palette
|
|
|
|
return im
|
|
|
|
|
2017-04-20 14:14:23 +03:00
|
|
|
|
2024-06-09 00:12:05 +03:00
|
|
|
def _write_single_frame(
|
|
|
|
im: Image.Image,
|
|
|
|
fp: IO[bytes],
|
2024-06-10 09:08:06 +03:00
|
|
|
palette: _Palette | None,
|
2024-06-09 00:12:05 +03:00
|
|
|
) -> None:
|
2022-03-12 07:14:36 +03:00
|
|
|
im_out = _normalize_mode(im)
|
2018-06-16 12:47:57 +03:00
|
|
|
for k, v in im_out.info.items():
|
|
|
|
im.encoderinfo.setdefault(k, v)
|
2017-02-23 17:14:04 +03:00
|
|
|
im_out = _normalize_palette(im_out, palette, im.encoderinfo)
|
|
|
|
|
2017-02-23 18:41:42 +03:00
|
|
|
for s in _get_global_header(im_out, im.encoderinfo):
|
2017-02-18 13:23:42 +03:00
|
|
|
fp.write(s)
|
|
|
|
|
|
|
|
# local image header
|
|
|
|
flags = 0
|
|
|
|
if get_interlace(im):
|
|
|
|
flags = flags | 64
|
2017-02-25 13:24:53 +03:00
|
|
|
_write_local_header(fp, im, (0, 0), flags)
|
2017-02-18 13:23:42 +03:00
|
|
|
|
|
|
|
im_out.encoderconfig = (8, get_interlace(im))
|
|
|
|
ImageFile._save(im_out, fp, [("gif", (0, 0) + im.size, 0, RAWMODE[im_out.mode])])
|
|
|
|
|
|
|
|
fp.write(b"\0") # end of image data
|
|
|
|
|
2017-04-20 14:14:23 +03:00
|
|
|
|
2024-06-09 00:12:05 +03:00
|
|
|
def _getbbox(
|
|
|
|
base_im: Image.Image, im_frame: Image.Image
|
2024-06-10 09:08:06 +03:00
|
|
|
) -> tuple[Image.Image, tuple[int, int, int, int] | None]:
|
2023-11-25 11:16:32 +03:00
|
|
|
if _get_palette_bytes(im_frame) != _get_palette_bytes(base_im):
|
|
|
|
im_frame = im_frame.convert("RGBA")
|
|
|
|
base_im = base_im.convert("RGBA")
|
|
|
|
delta = ImageChops.subtract_modulo(im_frame, base_im)
|
|
|
|
return delta, delta.getbbox(alpha_only=False)
|
2022-12-08 03:35:48 +03:00
|
|
|
|
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
class _Frame(NamedTuple):
|
|
|
|
im: Image.Image
|
|
|
|
bbox: tuple[int, int, int, int] | None
|
|
|
|
encoderinfo: dict[str, Any]
|
|
|
|
|
|
|
|
|
|
|
|
def _write_multiple_frames(
|
|
|
|
im: Image.Image, fp: IO[bytes], palette: _Palette | None
|
|
|
|
) -> bool:
|
2022-05-03 13:07:47 +03:00
|
|
|
duration = im.encoderinfo.get("duration")
|
2018-06-16 12:47:57 +03:00
|
|
|
disposal = im.encoderinfo.get("disposal", im.info.get("disposal"))
|
2017-02-18 13:23:42 +03:00
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
im_frames: list[_Frame] = []
|
|
|
|
previous_im: Image.Image | None = None
|
2017-02-18 13:23:42 +03:00
|
|
|
frame_count = 0
|
2019-06-29 16:06:45 +03:00
|
|
|
background_im = None
|
|
|
|
for imSequence in itertools.chain([im], im.encoderinfo.get("append_images", [])):
|
2017-02-18 13:23:42 +03:00
|
|
|
for im_frame in ImageSequence.Iterator(imSequence):
|
2017-02-23 18:37:14 +03:00
|
|
|
# a copy is required here since seek can still mutate the image
|
|
|
|
im_frame = _normalize_mode(im_frame.copy())
|
2018-06-16 12:47:57 +03:00
|
|
|
if frame_count == 0:
|
|
|
|
for k, v in im_frame.info.items():
|
2022-04-02 13:04:22 +03:00
|
|
|
if k == "transparency":
|
|
|
|
continue
|
2018-06-16 12:47:57 +03:00
|
|
|
im.encoderinfo.setdefault(k, v)
|
2017-02-23 18:38:46 +03:00
|
|
|
|
2017-02-18 13:23:42 +03:00
|
|
|
encoderinfo = im.encoderinfo.copy()
|
2022-04-02 13:04:22 +03:00
|
|
|
if "transparency" in im_frame.info:
|
|
|
|
encoderinfo.setdefault("transparency", im_frame.info["transparency"])
|
2023-11-25 11:16:32 +03:00
|
|
|
im_frame = _normalize_palette(im_frame, palette, encoderinfo)
|
2017-02-18 13:23:42 +03:00
|
|
|
if isinstance(duration, (list, tuple)):
|
|
|
|
encoderinfo["duration"] = duration[frame_count]
|
2022-05-03 13:07:47 +03:00
|
|
|
elif duration is None and "duration" in im_frame.info:
|
|
|
|
encoderinfo["duration"] = im_frame.info["duration"]
|
2017-12-22 01:26:58 +03:00
|
|
|
if isinstance(disposal, (list, tuple)):
|
|
|
|
encoderinfo["disposal"] = disposal[frame_count]
|
2017-02-18 13:23:42 +03:00
|
|
|
frame_count += 1
|
|
|
|
|
2023-11-25 11:16:32 +03:00
|
|
|
diff_frame = None
|
2024-06-10 09:08:06 +03:00
|
|
|
if im_frames and previous_im:
|
2017-02-18 13:23:42 +03:00
|
|
|
# delta frame
|
2023-11-25 11:16:32 +03:00
|
|
|
delta, bbox = _getbbox(previous_im, im_frame)
|
2022-12-08 03:35:48 +03:00
|
|
|
if not bbox:
|
|
|
|
# This frame is identical to the previous frame
|
2022-12-22 00:32:27 +03:00
|
|
|
if encoderinfo.get("duration"):
|
2024-06-10 09:08:06 +03:00
|
|
|
im_frames[-1].encoderinfo["duration"] += encoderinfo["duration"]
|
2022-12-08 03:35:48 +03:00
|
|
|
continue
|
2024-06-10 09:08:06 +03:00
|
|
|
if im_frames[-1].encoderinfo.get("disposal") == 2:
|
2019-06-29 16:06:45 +03:00
|
|
|
if background_im is None:
|
2021-06-28 15:12:38 +03:00
|
|
|
color = im.encoderinfo.get(
|
|
|
|
"transparency", im.info.get("transparency", (0, 0, 0))
|
2019-06-29 16:06:45 +03:00
|
|
|
)
|
2021-06-28 15:12:38 +03:00
|
|
|
background = _get_background(im_frame, color)
|
2019-06-29 16:06:45 +03:00
|
|
|
background_im = Image.new("P", im_frame.size, background)
|
2024-06-10 09:08:06 +03:00
|
|
|
background_im.putpalette(im_frames[0].im.palette)
|
2024-02-10 13:33:32 +03:00
|
|
|
bbox = _getbbox(background_im, im_frame)[1]
|
|
|
|
elif encoderinfo.get("optimize") and im_frame.mode != "1":
|
2023-11-25 11:16:32 +03:00
|
|
|
if "transparency" not in encoderinfo:
|
|
|
|
try:
|
2024-02-05 20:18:49 +03:00
|
|
|
encoderinfo["transparency"] = (
|
|
|
|
im_frame.palette._new_color_index(im_frame)
|
|
|
|
)
|
2023-11-25 11:16:32 +03:00
|
|
|
except ValueError:
|
|
|
|
pass
|
|
|
|
if "transparency" in encoderinfo:
|
|
|
|
# When the delta is zero, fill the image with transparency
|
|
|
|
diff_frame = im_frame.copy()
|
2024-02-05 11:16:15 +03:00
|
|
|
fill = Image.new("P", delta.size, encoderinfo["transparency"])
|
2023-11-25 11:16:32 +03:00
|
|
|
if delta.mode == "RGBA":
|
|
|
|
r, g, b, a = delta.split()
|
2024-03-25 12:38:52 +03:00
|
|
|
mask = ImageMath.lambda_eval(
|
|
|
|
lambda args: args["convert"](
|
|
|
|
args["max"](
|
|
|
|
args["max"](
|
|
|
|
args["max"](args["r"], args["g"]), args["b"]
|
|
|
|
),
|
|
|
|
args["a"],
|
|
|
|
)
|
|
|
|
* 255,
|
|
|
|
"1",
|
|
|
|
),
|
2023-11-25 11:16:32 +03:00
|
|
|
r=r,
|
|
|
|
g=g,
|
|
|
|
b=b,
|
|
|
|
a=a,
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
if delta.mode == "P":
|
|
|
|
# Convert to L without considering palette
|
|
|
|
delta_l = Image.new("L", delta.size)
|
|
|
|
delta_l.putdata(delta.getdata())
|
|
|
|
delta = delta_l
|
2024-03-25 12:38:52 +03:00
|
|
|
mask = ImageMath.lambda_eval(
|
|
|
|
lambda args: args["convert"](args["im"] * 255, "1"),
|
|
|
|
im=delta,
|
|
|
|
)
|
2023-11-25 11:16:32 +03:00
|
|
|
diff_frame.paste(fill, mask=ImageOps.invert(mask))
|
2017-02-18 13:23:42 +03:00
|
|
|
else:
|
|
|
|
bbox = None
|
2023-11-25 11:16:32 +03:00
|
|
|
previous_im = im_frame
|
2024-06-10 09:08:06 +03:00
|
|
|
im_frames.append(_Frame(diff_frame or im_frame, bbox, encoderinfo))
|
2017-02-23 18:38:46 +03:00
|
|
|
|
2023-11-03 13:09:16 +03:00
|
|
|
if len(im_frames) == 1:
|
|
|
|
if "duration" in im.encoderinfo:
|
|
|
|
# Since multiple frames will not be written, use the combined duration
|
2024-06-10 09:08:06 +03:00
|
|
|
im.encoderinfo["duration"] = im_frames[0].encoderinfo["duration"]
|
|
|
|
return False
|
2023-11-03 13:09:16 +03:00
|
|
|
|
|
|
|
for frame_data in im_frames:
|
2024-06-10 09:08:06 +03:00
|
|
|
im_frame = frame_data.im
|
|
|
|
if not frame_data.bbox:
|
2023-11-03 13:09:16 +03:00
|
|
|
# global header
|
2024-06-10 09:08:06 +03:00
|
|
|
for s in _get_global_header(im_frame, frame_data.encoderinfo):
|
2023-11-03 13:09:16 +03:00
|
|
|
fp.write(s)
|
|
|
|
offset = (0, 0)
|
|
|
|
else:
|
|
|
|
# compress difference
|
|
|
|
if not palette:
|
2024-06-10 09:08:06 +03:00
|
|
|
frame_data.encoderinfo["include_color_table"] = True
|
2023-11-03 13:09:16 +03:00
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
im_frame = im_frame.crop(frame_data.bbox)
|
|
|
|
offset = frame_data.bbox[:2]
|
|
|
|
_write_frame_data(fp, im_frame, offset, frame_data.encoderinfo)
|
2023-11-03 13:09:16 +03:00
|
|
|
return True
|
2015-07-01 02:18:05 +03:00
|
|
|
|
2017-04-20 14:14:23 +03:00
|
|
|
|
2024-06-10 07:15:28 +03:00
|
|
|
def _save_all(im: Image.Image, fp: IO[bytes], filename: str | bytes) -> None:
|
2015-06-30 11:02:48 +03:00
|
|
|
_save(im, fp, filename, save_all=True)
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2015-07-01 02:18:05 +03:00
|
|
|
|
2024-06-05 01:29:28 +03:00
|
|
|
def _save(
|
2024-06-10 07:15:28 +03:00
|
|
|
im: Image.Image, fp: IO[bytes], filename: str | bytes, save_all: bool = False
|
2024-06-05 01:29:28 +03:00
|
|
|
) -> None:
|
2010-07-31 06:52:47 +04:00
|
|
|
# header
|
2018-06-16 12:47:57 +03:00
|
|
|
if "palette" in im.encoderinfo or "palette" in im.info:
|
|
|
|
palette = im.encoderinfo.get("palette", im.info.get("palette"))
|
|
|
|
else:
|
2013-01-10 23:34:58 +04:00
|
|
|
palette = None
|
2023-11-25 09:36:19 +03:00
|
|
|
im.encoderinfo.setdefault("optimize", True)
|
2013-05-23 17:45:11 +04:00
|
|
|
|
2017-02-18 13:23:42 +03:00
|
|
|
if not save_all or not _write_multiple_frames(im, fp, palette):
|
|
|
|
_write_single_frame(im, fp, palette)
|
2015-04-03 15:22:13 +03:00
|
|
|
|
|
|
|
fp.write(b";") # end of file
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2015-12-09 07:47:53 +03:00
|
|
|
if hasattr(fp, "flush"):
|
2015-04-03 15:22:13 +03:00
|
|
|
fp.flush()
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2015-04-23 16:40:42 +03:00
|
|
|
|
2024-06-05 01:29:28 +03:00
|
|
|
def get_interlace(im: Image.Image) -> int:
|
2017-02-25 05:49:05 +03:00
|
|
|
interlace = im.encoderinfo.get("interlace", 1)
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
# workaround for @PIL153
|
|
|
|
if min(im.size) < 16:
|
|
|
|
interlace = 0
|
|
|
|
|
2015-04-03 15:22:13 +03:00
|
|
|
return interlace
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
def _write_local_header(
|
|
|
|
fp: IO[bytes], im: Image.Image, offset: tuple[int, int], flags: int
|
|
|
|
) -> None:
|
2010-07-31 06:52:47 +04:00
|
|
|
try:
|
2023-11-25 11:16:32 +03:00
|
|
|
transparency = im.encoderinfo["transparency"]
|
|
|
|
except KeyError:
|
|
|
|
transparency = None
|
2013-07-01 02:42:19 +04:00
|
|
|
|
2015-04-04 02:32:17 +03:00
|
|
|
if "duration" in im.encoderinfo:
|
2015-04-04 03:45:30 +03:00
|
|
|
duration = int(im.encoderinfo["duration"] / 10)
|
2015-04-03 15:22:13 +03:00
|
|
|
else:
|
|
|
|
duration = 0
|
2017-01-30 22:34:48 +03:00
|
|
|
|
2017-01-31 10:32:01 +03:00
|
|
|
disposal = int(im.encoderinfo.get("disposal", 0))
|
2017-01-30 22:34:48 +03:00
|
|
|
|
2023-11-25 11:16:32 +03:00
|
|
|
if transparency is not None or duration != 0 or disposal:
|
|
|
|
packed_flag = 1 if transparency is not None else 0
|
2017-01-30 22:34:48 +03:00
|
|
|
packed_flag |= disposal << 2
|
2015-04-23 16:40:42 +03:00
|
|
|
|
2015-04-03 15:22:13 +03:00
|
|
|
fp.write(
|
|
|
|
b"!"
|
2015-04-23 16:40:42 +03:00
|
|
|
+ o8(249) # extension intro
|
|
|
|
+ o8(4) # length
|
2017-01-30 22:34:48 +03:00
|
|
|
+ o8(packed_flag) # packed fields
|
2015-04-23 16:40:42 +03:00
|
|
|
+ o16(duration) # duration
|
2023-11-25 11:16:32 +03:00
|
|
|
+ o8(transparency or 0) # transparency index
|
2015-04-03 15:22:13 +03:00
|
|
|
+ o8(0)
|
2019-03-21 16:28:20 +03:00
|
|
|
)
|
|
|
|
|
2016-09-11 04:57:45 +03:00
|
|
|
include_color_table = im.encoderinfo.get("include_color_table")
|
|
|
|
if include_color_table:
|
2017-02-23 18:38:46 +03:00
|
|
|
palette_bytes = _get_palette_bytes(im)
|
2016-09-11 04:57:45 +03:00
|
|
|
color_table_size = _get_color_table_size(palette_bytes)
|
|
|
|
if color_table_size:
|
|
|
|
flags = flags | 128 # local color table flag
|
|
|
|
flags = flags | color_table_size
|
|
|
|
|
py3k: The big push
There are two main issues fixed with this commit:
* bytes vs. str: All file, image, and palette data are now handled as
bytes. A new _binary module consolidates the hacks needed to do this
across Python versions. tostring/fromstring methods have been renamed to
tobytes/frombytes, but the Python 2.6/2.7 versions alias them to the old
names for compatibility. Users should move to tobytes/frombytes.
One other potentially-breaking change is that text data in image files
(such as tags, comments) are now explicitly handled with a specific
character encoding in mind. This works well with the Unicode str in
Python 3, but may trip up old code expecting a straight byte-for-byte
translation to a Python string. This also required a change to Gohlke's
tags tests (in Tests/test_file_png.py) to expect Unicode strings from
the code.
* True div vs. floor div: Many division operations used the "/" operator
to do floor division, which is now the "//" operator in Python 3. These
were fixed.
As of this commit, on the first pass, I have one failing test (improper
handling of a slice object in a C module, test_imagepath.py) in Python 3,
and three that that I haven't tried running yet (test_imagegl,
test_imagegrab, and test_imageqt). I also haven't tested anything on
Windows. All but the three skipped tests run flawlessly against Pythons
2.6 and 2.7.
2012-10-21 01:01:53 +04:00
|
|
|
fp.write(
|
|
|
|
b","
|
2015-04-23 16:40:42 +03:00
|
|
|
+ o16(offset[0]) # offset
|
2015-04-03 15:22:13 +03:00
|
|
|
+ o16(offset[1])
|
2015-04-23 16:40:42 +03:00
|
|
|
+ o16(im.size[0]) # size
|
2010-07-31 06:52:47 +04:00
|
|
|
+ o16(im.size[1])
|
2016-09-11 04:57:45 +03:00
|
|
|
+ o8(flags) # flags
|
2019-03-21 16:28:20 +03:00
|
|
|
)
|
2016-09-11 04:57:45 +03:00
|
|
|
if include_color_table and color_table_size:
|
|
|
|
fp.write(_get_header_palette(palette_bytes))
|
|
|
|
fp.write(o8(8)) # bits
|
2013-05-23 17:45:11 +04:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2024-06-10 07:15:28 +03:00
|
|
|
def _save_netpbm(im: Image.Image, fp: IO[bytes], filename: str | bytes) -> None:
|
2017-01-26 05:51:20 +03:00
|
|
|
# Unused by default.
|
|
|
|
# To use, uncomment the register_save call at the end of the file.
|
2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# If you need real GIF compression and/or RGB quantization, you
|
|
|
|
# can use the external NETPBM/PBMPLUS utilities. See comments
|
|
|
|
# below for information on how to enable this.
|
2019-03-03 05:02:00 +03:00
|
|
|
tempfile = im._dump()
|
2014-06-30 01:24:32 +04:00
|
|
|
|
2019-12-25 07:22:54 +03:00
|
|
|
try:
|
|
|
|
with open(filename, "wb") as f:
|
|
|
|
if im.mode != "RGB":
|
|
|
|
subprocess.check_call(
|
|
|
|
["ppmtogif", tempfile], stdout=f, stderr=subprocess.DEVNULL
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
# Pipe ppmquant output into ppmtogif
|
|
|
|
# "ppmquant 256 %s | ppmtogif > %s" % (tempfile, filename)
|
|
|
|
quant_cmd = ["ppmquant", "256", tempfile]
|
|
|
|
togif_cmd = ["ppmtogif"]
|
|
|
|
quant_proc = subprocess.Popen(
|
|
|
|
quant_cmd, stdout=subprocess.PIPE, stderr=subprocess.DEVNULL
|
|
|
|
)
|
|
|
|
togif_proc = subprocess.Popen(
|
|
|
|
togif_cmd,
|
|
|
|
stdin=quant_proc.stdout,
|
|
|
|
stdout=f,
|
|
|
|
stderr=subprocess.DEVNULL,
|
|
|
|
)
|
2014-06-27 19:02:36 +04:00
|
|
|
|
2019-12-25 07:22:54 +03:00
|
|
|
# Allow ppmquant to receive SIGPIPE if ppmtogif exits
|
2024-06-09 00:12:05 +03:00
|
|
|
assert quant_proc.stdout is not None
|
2019-12-25 07:22:54 +03:00
|
|
|
quant_proc.stdout.close()
|
2014-06-27 19:02:36 +04:00
|
|
|
|
2019-12-25 07:22:54 +03:00
|
|
|
retcode = quant_proc.wait()
|
|
|
|
if retcode:
|
|
|
|
raise subprocess.CalledProcessError(retcode, quant_cmd)
|
2014-06-27 19:02:36 +04:00
|
|
|
|
2019-12-25 07:22:54 +03:00
|
|
|
retcode = togif_proc.wait()
|
|
|
|
if retcode:
|
|
|
|
raise subprocess.CalledProcessError(retcode, togif_cmd)
|
|
|
|
finally:
|
|
|
|
try:
|
|
|
|
os.unlink(tempfile)
|
|
|
|
except OSError:
|
|
|
|
pass
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
|
2016-11-05 05:32:02 +03:00
|
|
|
# Force optimization so that we can test performance against
|
|
|
|
# cases where it took lots of memory and time previously.
|
|
|
|
_FORCE_OPTIMIZE = False
|
|
|
|
|
2017-04-20 14:14:23 +03:00
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
def _get_optimize(im: Image.Image, info: dict[str, Any]) -> list[int] | None:
|
2017-02-23 15:25:04 +03:00
|
|
|
"""
|
|
|
|
Palette optimization is a potentially expensive operation.
|
2017-02-23 18:38:46 +03:00
|
|
|
|
2017-02-23 15:25:04 +03:00
|
|
|
This function determines if the palette should be optimized using
|
|
|
|
some heuristics, then returns the list of palette entries in use.
|
2017-02-23 18:38:46 +03:00
|
|
|
|
2017-02-23 15:25:04 +03:00
|
|
|
:param im: Image object
|
2017-02-23 18:38:46 +03:00
|
|
|
:param info: encoderinfo
|
2017-02-23 15:25:04 +03:00
|
|
|
:returns: list of indexes of palette entries in use, or None
|
|
|
|
"""
|
2023-11-25 11:16:32 +03:00
|
|
|
if im.mode in ("P", "L") and info and info.get("optimize"):
|
2016-11-05 05:32:02 +03:00
|
|
|
# Potentially expensive operation.
|
2015-04-23 17:44:27 +03:00
|
|
|
|
2016-11-05 05:32:02 +03:00
|
|
|
# The palette saves 3 bytes per color not used, but palette
|
|
|
|
# lengths are restricted to 3*(2**N) bytes. Max saving would
|
|
|
|
# be 768 -> 6 bytes if we went all the way down to 2 colors.
|
|
|
|
# * If we're over 128 colors, we can't save any space.
|
|
|
|
# * If there aren't any holes, it's not worth collapsing.
|
|
|
|
# * If we have a 'large' image, the palette is in the noise.
|
|
|
|
|
|
|
|
# create the new palette if not every color is used
|
2017-01-26 07:27:43 +03:00
|
|
|
optimise = _FORCE_OPTIMIZE or im.mode == "L"
|
|
|
|
if optimise or im.width * im.height < 512 * 512:
|
2017-01-26 06:17:20 +03:00
|
|
|
# check which colors are used
|
|
|
|
used_palette_colors = []
|
|
|
|
for i, count in enumerate(im.histogram()):
|
|
|
|
if count:
|
|
|
|
used_palette_colors.append(i)
|
|
|
|
|
2022-06-19 09:47:50 +03:00
|
|
|
if optimise or max(used_palette_colors) >= len(used_palette_colors):
|
|
|
|
return used_palette_colors
|
|
|
|
|
|
|
|
num_palette_colors = len(im.palette.palette) // Image.getmodebands(
|
|
|
|
im.palette.mode
|
2022-06-19 03:20:25 +03:00
|
|
|
)
|
2022-06-26 13:58:36 +03:00
|
|
|
current_palette_size = 1 << (num_palette_colors - 1).bit_length()
|
|
|
|
if (
|
|
|
|
# check that the palette would become smaller when saved
|
|
|
|
len(used_palette_colors) <= current_palette_size // 2
|
|
|
|
# check that the palette is not already the smallest possible size
|
|
|
|
and current_palette_size > 2
|
|
|
|
):
|
2017-01-26 07:27:43 +03:00
|
|
|
return used_palette_colors
|
2024-06-10 09:08:06 +03:00
|
|
|
return None
|
2015-04-23 17:44:27 +03:00
|
|
|
|
2017-04-20 14:14:23 +03:00
|
|
|
|
2024-05-13 11:47:51 +03:00
|
|
|
def _get_color_table_size(palette_bytes: bytes) -> int:
|
2016-09-11 04:57:45 +03:00
|
|
|
# calculate the palette size for the header
|
2019-06-29 12:24:12 +03:00
|
|
|
if not palette_bytes:
|
|
|
|
return 0
|
|
|
|
elif len(palette_bytes) < 9:
|
|
|
|
return 1
|
|
|
|
else:
|
2020-02-08 00:34:53 +03:00
|
|
|
return math.ceil(math.log(len(palette_bytes) // 3, 2)) - 1
|
2015-04-23 17:44:27 +03:00
|
|
|
|
2017-04-20 14:14:23 +03:00
|
|
|
|
2024-05-13 11:47:51 +03:00
|
|
|
def _get_header_palette(palette_bytes: bytes) -> bytes:
|
2017-02-23 18:58:30 +03:00
|
|
|
"""
|
|
|
|
Returns the palette, null padded to the next power of 2 (*3) bytes
|
|
|
|
suitable for direct inclusion in the GIF header
|
|
|
|
|
|
|
|
:param palette_bytes: Unpadded palette bytes, in RGBRGB form
|
|
|
|
:returns: Null padded palette
|
|
|
|
"""
|
2016-09-11 04:57:45 +03:00
|
|
|
color_table_size = _get_color_table_size(palette_bytes)
|
2015-08-21 15:10:13 +03:00
|
|
|
|
2016-09-11 04:57:45 +03:00
|
|
|
# add the missing amount of bytes
|
|
|
|
# the palette has to be 2<<n in size
|
|
|
|
actual_target_size_diff = (2 << color_table_size) - len(palette_bytes) // 3
|
|
|
|
if actual_target_size_diff > 0:
|
|
|
|
palette_bytes += o8(0) * 3 * actual_target_size_diff
|
|
|
|
return palette_bytes
|
2013-05-23 17:45:11 +04:00
|
|
|
|
2017-04-20 14:14:23 +03:00
|
|
|
|
2024-05-13 11:47:51 +03:00
|
|
|
def _get_palette_bytes(im: Image.Image) -> bytes:
|
2017-04-20 14:14:23 +03:00
|
|
|
"""
|
|
|
|
Gets the palette for inclusion in the gif header
|
|
|
|
|
|
|
|
:param im: Image object
|
|
|
|
:returns: Bytes, len<=768 suitable for inclusion in gif header
|
|
|
|
"""
|
2023-05-24 01:55:14 +03:00
|
|
|
return im.palette.palette if im.palette else b""
|
2017-02-23 17:14:04 +03:00
|
|
|
|
2016-09-11 04:57:45 +03:00
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
def _get_background(
|
|
|
|
im: Image.Image,
|
|
|
|
info_background: int | tuple[int, int, int] | tuple[int, int, int, int] | None,
|
|
|
|
) -> int:
|
2019-06-29 16:06:45 +03:00
|
|
|
background = 0
|
2022-04-10 21:24:15 +03:00
|
|
|
if info_background:
|
2022-12-09 02:45:09 +03:00
|
|
|
if isinstance(info_background, tuple):
|
2019-06-29 16:06:45 +03:00
|
|
|
# WebPImagePlugin stores an RGBA value in info["background"]
|
|
|
|
# So it must be converted to the same format as GifImagePlugin's
|
|
|
|
# info["background"] - a global color table index
|
2021-06-23 12:28:46 +03:00
|
|
|
try:
|
2022-12-09 02:45:09 +03:00
|
|
|
background = im.palette.getcolor(info_background, im)
|
2021-06-23 12:28:46 +03:00
|
|
|
except ValueError as e:
|
2022-12-09 02:45:09 +03:00
|
|
|
if str(e) not in (
|
2021-06-23 12:28:46 +03:00
|
|
|
# If all 256 colors are in use,
|
|
|
|
# then there is no need for the background color
|
2022-12-09 02:45:09 +03:00
|
|
|
"cannot allocate more than 256 colors",
|
|
|
|
# Ignore non-opaque WebP background
|
|
|
|
"cannot add non-opaque RGBA color to RGB palette",
|
|
|
|
):
|
2021-06-23 12:28:46 +03:00
|
|
|
raise
|
2022-12-09 02:45:09 +03:00
|
|
|
else:
|
|
|
|
background = info_background
|
2019-06-29 16:06:45 +03:00
|
|
|
return background
|
|
|
|
|
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
def _get_global_header(im: Image.Image, info: dict[str, Any]) -> list[bytes]:
|
2016-09-11 04:57:45 +03:00
|
|
|
"""Return a list of strings representing a GIF header"""
|
|
|
|
|
|
|
|
# Header Block
|
2021-07-12 14:00:36 +03:00
|
|
|
# https://www.matthewflickinger.com/lab/whatsinagif/bits_and_bytes.asp
|
2016-09-11 04:57:45 +03:00
|
|
|
|
|
|
|
version = b"87a"
|
2022-05-19 13:59:16 +03:00
|
|
|
if im.info.get("version") == b"89a" or (
|
|
|
|
info
|
|
|
|
and (
|
|
|
|
"transparency" in info
|
2023-08-09 03:31:34 +03:00
|
|
|
or info.get("loop") is not None
|
2022-05-19 13:59:16 +03:00
|
|
|
or info.get("duration")
|
|
|
|
or info.get("comment")
|
|
|
|
)
|
|
|
|
):
|
|
|
|
version = b"89a"
|
2016-09-11 04:57:45 +03:00
|
|
|
|
2019-06-29 16:06:45 +03:00
|
|
|
background = _get_background(im, info.get("background"))
|
2018-10-02 13:52:07 +03:00
|
|
|
|
2017-02-23 18:38:46 +03:00
|
|
|
palette_bytes = _get_palette_bytes(im)
|
2017-02-18 11:02:52 +03:00
|
|
|
color_table_size = _get_color_table_size(palette_bytes)
|
|
|
|
|
2022-05-13 21:45:01 +03:00
|
|
|
header = [
|
2017-02-18 11:02:52 +03:00
|
|
|
b"GIF" # signature
|
|
|
|
+ version # version
|
|
|
|
+ o16(im.size[0]) # canvas width
|
2017-03-07 12:52:31 +03:00
|
|
|
+ o16(im.size[1]), # canvas height
|
2017-02-18 11:02:52 +03:00
|
|
|
# Logical Screen Descriptor
|
|
|
|
# size of global color table + global color table flag
|
2017-03-07 12:52:31 +03:00
|
|
|
o8(color_table_size + 128), # packed fields
|
2017-02-18 11:02:52 +03:00
|
|
|
# background + reserved/aspect
|
2017-04-20 14:14:23 +03:00
|
|
|
o8(background) + o8(0),
|
2017-02-18 11:02:52 +03:00
|
|
|
# Global Color Table
|
|
|
|
_get_header_palette(palette_bytes),
|
2016-09-11 04:57:45 +03:00
|
|
|
]
|
2023-08-09 03:31:34 +03:00
|
|
|
if info.get("loop") is not None:
|
2022-04-15 09:46:33 +03:00
|
|
|
header.append(
|
|
|
|
b"!"
|
|
|
|
+ o8(255) # extension intro
|
|
|
|
+ o8(11)
|
|
|
|
+ b"NETSCAPE2.0"
|
|
|
|
+ o8(3)
|
|
|
|
+ o8(1)
|
|
|
|
+ o16(info["loop"]) # number of loops
|
|
|
|
+ o8(0)
|
|
|
|
)
|
2022-05-22 08:30:16 +03:00
|
|
|
if info.get("comment"):
|
|
|
|
comment_block = b"!" + o8(254) # extension intro
|
2016-09-11 04:57:45 +03:00
|
|
|
|
2022-05-13 21:45:01 +03:00
|
|
|
comment = info["comment"]
|
|
|
|
if isinstance(comment, str):
|
|
|
|
comment = comment.encode()
|
|
|
|
for i in range(0, len(comment), 255):
|
|
|
|
subblock = comment[i : i + 255]
|
2022-05-22 08:30:16 +03:00
|
|
|
comment_block += o8(len(subblock)) + subblock
|
2022-05-13 21:45:01 +03:00
|
|
|
|
2022-05-22 08:30:16 +03:00
|
|
|
comment_block += o8(0)
|
|
|
|
header.append(comment_block)
|
2022-05-13 21:45:01 +03:00
|
|
|
return header
|
|
|
|
|
2017-04-20 14:14:23 +03:00
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
def _write_frame_data(
|
|
|
|
fp: IO[bytes],
|
|
|
|
im_frame: Image.Image,
|
|
|
|
offset: tuple[int, int],
|
|
|
|
params: dict[str, Any],
|
|
|
|
) -> None:
|
2017-02-25 13:29:54 +03:00
|
|
|
try:
|
|
|
|
im_frame.encoderinfo = params
|
|
|
|
|
|
|
|
# local image header
|
|
|
|
_write_local_header(fp, im_frame, offset, 0)
|
|
|
|
|
|
|
|
ImageFile._save(
|
|
|
|
im_frame, fp, [("gif", (0, 0) + im_frame.size, 0, RAWMODE[im_frame.mode])]
|
|
|
|
)
|
|
|
|
|
|
|
|
fp.write(b"\0") # end of image data
|
|
|
|
finally:
|
|
|
|
del im_frame.encoderinfo
|
|
|
|
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2017-02-25 13:29:54 +03:00
|
|
|
# --------------------------------------------------------------------
|
|
|
|
# Legacy GIF utilities
|
|
|
|
|
2017-04-20 14:14:23 +03:00
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
def getheader(
|
|
|
|
im: Image.Image, palette: _Palette | None = None, info: dict[str, Any] | None = None
|
|
|
|
) -> tuple[list[bytes], list[int] | None]:
|
2017-02-23 18:58:30 +03:00
|
|
|
"""
|
|
|
|
Legacy Method to get Gif data from image.
|
|
|
|
|
|
|
|
Warning:: May modify image data.
|
|
|
|
|
|
|
|
:param im: Image object
|
|
|
|
:param palette: bytes object containing the source palette, or ....
|
2017-04-20 14:14:23 +03:00
|
|
|
:param info: encoderinfo
|
2017-02-23 18:58:30 +03:00
|
|
|
:returns: tuple of(list of header items, optimized palette)
|
|
|
|
|
|
|
|
"""
|
2017-03-07 12:32:01 +03:00
|
|
|
if info is None:
|
|
|
|
info = {}
|
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
used_palette_colors = _get_optimize(im, info)
|
|
|
|
|
2017-04-20 14:14:23 +03:00
|
|
|
if "background" not in info and "background" in im.info:
|
2017-02-18 11:02:52 +03:00
|
|
|
info["background"] = im.info["background"]
|
|
|
|
|
2017-02-23 18:59:06 +03:00
|
|
|
im_mod = _normalize_palette(im, palette, info)
|
|
|
|
im.palette = im_mod.palette
|
|
|
|
im.im = im_mod.im
|
2017-02-23 18:41:42 +03:00
|
|
|
header = _get_global_header(im, info)
|
2013-05-23 17:45:11 +04:00
|
|
|
|
2015-04-01 16:47:01 +03:00
|
|
|
return header, used_palette_colors
|
2013-05-23 17:45:11 +04:00
|
|
|
|
2017-04-20 14:14:23 +03:00
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
def getdata(
|
|
|
|
im: Image.Image, offset: tuple[int, int] = (0, 0), **params: Any
|
|
|
|
) -> list[bytes]:
|
2017-02-25 05:50:49 +03:00
|
|
|
"""
|
|
|
|
Legacy Method
|
|
|
|
|
|
|
|
Return a list of strings representing this image.
|
|
|
|
The first string is a local image header, the rest contains
|
|
|
|
encoded image data.
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2022-04-10 20:22:47 +03:00
|
|
|
To specify duration, add the time in milliseconds,
|
|
|
|
e.g. ``getdata(im_frame, duration=1000)``
|
|
|
|
|
2017-02-25 13:29:54 +03:00
|
|
|
:param im: Image object
|
2022-04-10 20:22:47 +03:00
|
|
|
:param offset: Tuple of (x, y) pixels. Defaults to (0, 0)
|
|
|
|
:param \\**params: e.g. duration or other encoder info parameters
|
|
|
|
:returns: List of bytes containing GIF encoded frame data
|
2017-02-25 13:29:54 +03:00
|
|
|
|
2017-02-25 05:50:49 +03:00
|
|
|
"""
|
2024-06-10 09:08:06 +03:00
|
|
|
from io import BytesIO
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
class Collector(BytesIO):
|
2010-07-31 06:52:47 +04:00
|
|
|
data = []
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2024-06-10 09:08:06 +03:00
|
|
|
if sys.version_info >= (3, 12):
|
|
|
|
from collections.abc import Buffer
|
|
|
|
|
|
|
|
def write(self, data: Buffer) -> int:
|
|
|
|
self.data.append(data)
|
|
|
|
return len(data)
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
|
|
|
def write(self, data: Any) -> int:
|
|
|
|
self.data.append(data)
|
|
|
|
return len(data)
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
im.load() # make sure raster data is available
|
|
|
|
|
2015-04-01 16:47:01 +03:00
|
|
|
fp = Collector()
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2017-02-18 11:09:10 +03:00
|
|
|
_write_frame_data(fp, im, offset, params)
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
return fp.data
|
|
|
|
|
|
|
|
|
|
|
|
# --------------------------------------------------------------------
|
|
|
|
# Registry
|
|
|
|
|
|
|
|
Image.register_open(GifImageFile.format, GifImageFile, _accept)
|
|
|
|
Image.register_save(GifImageFile.format, _save)
|
2015-06-30 11:02:48 +03:00
|
|
|
Image.register_save_all(GifImageFile.format, _save_all)
|
2010-07-31 06:52:47 +04:00
|
|
|
Image.register_extension(GifImageFile.format, ".gif")
|
|
|
|
Image.register_mime(GifImageFile.format, "image/gif")
|
|
|
|
|
|
|
|
#
|
|
|
|
# Uncomment the following line if you wish to use NETPBM/PBMPLUS
|
|
|
|
# instead of the built-in "uncompressed" GIF encoder
|
|
|
|
|
|
|
|
# Image.register_save(GifImageFile.format, _save_netpbm)
|