2013-03-14 06:03:38 +04:00
|
|
|
from PIL import Image
|
|
|
|
from PIL import ImageFile
|
2013-03-14 06:42:21 +04:00
|
|
|
from io import BytesIO
|
2013-03-12 18:30:59 +04:00
|
|
|
import _webp
|
|
|
|
|
2013-05-13 20:01:42 +04:00
|
|
|
|
|
|
|
_VALID_WEBP_ENCODERS_BY_MODE = {
|
|
|
|
"RGB": _webp.WebPEncodeRGB,
|
|
|
|
"RGBA": _webp.WebPEncodeRGBA,
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
_VALID_WEBP_DECODERS_BY_MODE = {
|
|
|
|
"RGB": _webp.WebPDecodeRGB,
|
|
|
|
"RGBA": _webp.WebPDecodeRGBA,
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
_STRIDE_MULTIPLIERS_BY_MODE = {
|
|
|
|
"RGB": 3,
|
|
|
|
"RGBA": 4,
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
_VP8_MODES_BY_IDENTIFIER = {
|
|
|
|
b"VP8 ": "RGB",
|
|
|
|
b"VP8X": "RGBA",
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-03-12 18:30:59 +04:00
|
|
|
def _accept(prefix):
|
2013-05-13 20:01:42 +04:00
|
|
|
is_riff_file_format = prefix[:4] == b"RIFF"
|
|
|
|
is_webp_file = prefix[8:12] == b"WEBP"
|
|
|
|
is_valid_vp8_mode = prefix[12:16] in _VP8_MODES_BY_IDENTIFIER
|
|
|
|
|
|
|
|
return is_riff_file_format and is_webp_file and is_valid_vp8_mode
|
|
|
|
|
2013-03-12 18:30:59 +04:00
|
|
|
|
|
|
|
class WebPImageFile(ImageFile.ImageFile):
|
|
|
|
|
|
|
|
format = "WEBP"
|
|
|
|
format_description = "WebP image"
|
|
|
|
|
|
|
|
def _open(self):
|
2013-05-13 20:01:42 +04:00
|
|
|
file_header = self.fp.read(16)
|
|
|
|
vp8_header = file_header[12:16]
|
|
|
|
try:
|
|
|
|
webp_file_mode = _VP8_MODES_BY_IDENTIFIER[vp8_header]
|
|
|
|
except KeyError:
|
|
|
|
raise IOError("Unknown webp file mode")
|
|
|
|
finally:
|
|
|
|
self.fp.seek(0)
|
|
|
|
|
|
|
|
self.mode = webp_file_mode
|
|
|
|
webp_decoder = _VALID_WEBP_DECODERS_BY_MODE[webp_file_mode]
|
|
|
|
|
|
|
|
data, width, height = webp_decoder(self.fp.read())
|
2013-03-12 18:30:59 +04:00
|
|
|
self.size = width, height
|
2013-03-14 06:42:21 +04:00
|
|
|
self.fp = BytesIO(data)
|
2013-05-13 20:01:42 +04:00
|
|
|
self.tile = [("raw", (0, 0) + self.size, 0, webp_file_mode)]
|
|
|
|
|
2013-03-12 18:30:59 +04:00
|
|
|
|
|
|
|
def _save(im, fp, filename):
|
2013-05-13 20:01:42 +04:00
|
|
|
image_mode = im.mode
|
|
|
|
if image_mode not in _VALID_WEBP_ENCODERS_BY_MODE:
|
|
|
|
raise IOError("cannot write mode %s as WEBP" % image_mode)
|
|
|
|
|
|
|
|
webp_encoder = _VALID_WEBP_ENCODERS_BY_MODE[image_mode]
|
|
|
|
|
|
|
|
stride = im.size[0] * _STRIDE_MULTIPLIERS_BY_MODE[image_mode]
|
2013-03-12 18:30:59 +04:00
|
|
|
quality = im.encoderinfo.get("quality", 80)
|
|
|
|
|
2013-05-13 20:01:42 +04:00
|
|
|
data = webp_encoder(
|
|
|
|
im.tobytes(),
|
|
|
|
im.size[0],
|
|
|
|
im.size[1],
|
|
|
|
stride,
|
|
|
|
float(quality),
|
|
|
|
)
|
2013-03-12 18:30:59 +04:00
|
|
|
fp.write(data)
|
|
|
|
|
2013-05-13 20:01:42 +04:00
|
|
|
|
2013-03-12 18:30:59 +04:00
|
|
|
Image.register_open("WEBP", WebPImageFile, _accept)
|
|
|
|
Image.register_save("WEBP", _save)
|
|
|
|
|
|
|
|
Image.register_extension("WEBP", ".webp")
|
|
|
|
Image.register_mime("WEBP", "image/webp")
|