mirror of
https://github.com/python-pillow/Pillow.git
synced 2024-12-26 01:46:18 +03:00
Merge pull request #95 from olt/pil-png-transp-backport
Backport PNG/ZIP improvements from PIL repo
This commit is contained in:
commit
d6f597a1ad
25
PIL/Image.py
25
PIL/Image.py
|
@ -143,11 +143,23 @@ FLOYDSTEINBERG = 3 # default
|
||||||
WEB = 0
|
WEB = 0
|
||||||
ADAPTIVE = 1
|
ADAPTIVE = 1
|
||||||
|
|
||||||
|
MEDIANCUT = 0
|
||||||
|
MAXCOVERAGE = 1
|
||||||
|
FASTOCTREE = 2
|
||||||
|
|
||||||
# categories
|
# categories
|
||||||
NORMAL = 0
|
NORMAL = 0
|
||||||
SEQUENCE = 1
|
SEQUENCE = 1
|
||||||
CONTAINER = 2
|
CONTAINER = 2
|
||||||
|
|
||||||
|
if hasattr(core, 'DEFAULT_STRATEGY'):
|
||||||
|
DEFAULT_STRATEGY = core.DEFAULT_STRATEGY
|
||||||
|
FILTERED = core.FILTERED
|
||||||
|
HUFFMAN_ONLY = core.HUFFMAN_ONLY
|
||||||
|
RLE = core.RLE
|
||||||
|
FIXED = core.FIXED
|
||||||
|
|
||||||
|
|
||||||
# --------------------------------------------------------------------
|
# --------------------------------------------------------------------
|
||||||
# Registries
|
# Registries
|
||||||
|
|
||||||
|
@ -616,18 +628,12 @@ class Image:
|
||||||
self.palette.mode = "RGB"
|
self.palette.mode = "RGB"
|
||||||
self.palette.rawmode = None
|
self.palette.rawmode = None
|
||||||
if "transparency" in self.info:
|
if "transparency" in self.info:
|
||||||
|
if isinstance(self.info["transparency"], str):
|
||||||
# XXX Not sure how this ever worked:
|
self.im.putpalettealphas(self.info["transparency"])
|
||||||
# if self.info["transparency_palette"]:
|
|
||||||
# Should probably be:
|
|
||||||
if "transparency_palette" in self.info:
|
|
||||||
# amirite?
|
|
||||||
|
|
||||||
self.im.putpalettealpha(0, 0, self.info["transparency_palette"])
|
|
||||||
else:
|
else:
|
||||||
self.im.putpalettealpha(self.info["transparency"], 0)
|
self.im.putpalettealpha(self.info["transparency"], 0)
|
||||||
|
|
||||||
self.palette.mode = "RGBA"
|
self.palette.mode = "RGBA"
|
||||||
|
|
||||||
if self.im:
|
if self.im:
|
||||||
return self.im.pixel_access(self.readonly)
|
return self.im.pixel_access(self.readonly)
|
||||||
|
|
||||||
|
@ -724,6 +730,7 @@ class Image:
|
||||||
# methods:
|
# methods:
|
||||||
# 0 = median cut
|
# 0 = median cut
|
||||||
# 1 = maximum coverage
|
# 1 = maximum coverage
|
||||||
|
# 2 = fast octree
|
||||||
|
|
||||||
# NOTE: this functionality will be moved to the extended
|
# NOTE: this functionality will be moved to the extended
|
||||||
# quantizer interface in a later version of PIL.
|
# quantizer interface in a later version of PIL.
|
||||||
|
|
|
@ -70,6 +70,8 @@ _MODES = {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
_simple_palette = re.compile(b'^\xff+\x00+$')
|
||||||
|
|
||||||
# --------------------------------------------------------------------
|
# --------------------------------------------------------------------
|
||||||
# Support classes. Suitable for PNG and related formats like MNG etc.
|
# Support classes. Suitable for PNG and related formats like MNG etc.
|
||||||
|
|
||||||
|
@ -251,10 +253,12 @@ class PngStream(ChunkStream):
|
||||||
# transparency
|
# transparency
|
||||||
s = ImageFile._safe_read(self.fp, len)
|
s = ImageFile._safe_read(self.fp, len)
|
||||||
if self.im_mode == "P":
|
if self.im_mode == "P":
|
||||||
|
if _simple_palette.match(s):
|
||||||
i = s.find(b"\0")
|
i = s.find(b"\0")
|
||||||
if i >= 0:
|
if i >= 0:
|
||||||
self.im_info["transparency"] = i
|
self.im_info["transparency"] = i
|
||||||
self.im_info["transparency_palette"] = s
|
else:
|
||||||
|
self.im_info["transparency"] = s
|
||||||
elif self.im_mode == "L":
|
elif self.im_mode == "L":
|
||||||
self.im_info["transparency"] = i16(s)
|
self.im_info["transparency"] = i16(s)
|
||||||
elif self.im_mode == "RGB":
|
elif self.im_mode == "RGB":
|
||||||
|
@ -514,7 +518,10 @@ def _save(im, fp, filename, chunk=putchunk, check=0):
|
||||||
else:
|
else:
|
||||||
dictionary = b""
|
dictionary = b""
|
||||||
|
|
||||||
im.encoderconfig = ("optimize" in im.encoderinfo, dictionary)
|
im.encoderconfig = ("optimize" in im.encoderinfo,
|
||||||
|
im.encoderinfo.get("compress_level", -1),
|
||||||
|
im.encoderinfo.get("compress_type", -1),
|
||||||
|
dictionary)
|
||||||
|
|
||||||
# get the corresponding PNG mode
|
# get the corresponding PNG mode
|
||||||
try:
|
try:
|
||||||
|
@ -538,12 +545,16 @@ def _save(im, fp, filename, chunk=putchunk, check=0):
|
||||||
b'\0') # 12: interlace flag
|
b'\0') # 12: interlace flag
|
||||||
|
|
||||||
if im.mode == "P":
|
if im.mode == "P":
|
||||||
chunk(fp, b"PLTE", im.im.getpalette("RGB"))
|
palette_bytes = (2 ** bits) * 3
|
||||||
|
chunk(fp, b"PLTE", im.im.getpalette("RGB")[:palette_bytes])
|
||||||
|
|
||||||
if "transparency" in im.encoderinfo:
|
if "transparency" in im.encoderinfo:
|
||||||
if im.mode == "P":
|
if im.mode == "P":
|
||||||
transparency = max(0, min(255, im.encoderinfo["transparency"]))
|
transparency = max(0, min(255, im.encoderinfo["transparency"]))
|
||||||
chunk(fp, b"tRNS", b'\xFF' * transparency + b'\0')
|
alpha = b'\xFF' * transparency + b'\0'
|
||||||
|
# limit to actual palette size
|
||||||
|
alpha_bytes = 2**bits
|
||||||
|
chunk(fp, b"tRNS", alpha[:alpha_bytes])
|
||||||
elif im.mode == "L":
|
elif im.mode == "L":
|
||||||
transparency = max(0, min(65535, im.encoderinfo["transparency"]))
|
transparency = max(0, min(65535, im.encoderinfo["transparency"]))
|
||||||
chunk(fp, b"tRNS", o16(transparency))
|
chunk(fp, b"tRNS", o16(transparency))
|
||||||
|
@ -552,6 +563,11 @@ def _save(im, fp, filename, chunk=putchunk, check=0):
|
||||||
chunk(fp, b"tRNS", o16(red) + o16(green) + o16(blue))
|
chunk(fp, b"tRNS", o16(red) + o16(green) + o16(blue))
|
||||||
else:
|
else:
|
||||||
raise IOError("cannot use transparency for this mode")
|
raise IOError("cannot use transparency for this mode")
|
||||||
|
else:
|
||||||
|
if im.mode == "P" and im.im.getpalettemode() == "RGBA":
|
||||||
|
alpha = im.im.getpalette("RGBA", "A")
|
||||||
|
alpha_bytes = 2**bits
|
||||||
|
chunk(fp, b"tRNS", alpha[:alpha_bytes])
|
||||||
|
|
||||||
if 0:
|
if 0:
|
||||||
# FIXME: to be supported some day
|
# FIXME: to be supported some day
|
||||||
|
|
|
@ -117,6 +117,17 @@ def test_interlace():
|
||||||
|
|
||||||
assert_no_exception(lambda: im.load())
|
assert_no_exception(lambda: im.load())
|
||||||
|
|
||||||
|
def test_load_transparent_p():
|
||||||
|
file = "Tests/images/pil123p.png"
|
||||||
|
im = Image.open(file)
|
||||||
|
|
||||||
|
assert_image(im, "P", (162, 150))
|
||||||
|
im = im.convert("RGBA")
|
||||||
|
assert_image(im, "RGBA", (162, 150))
|
||||||
|
|
||||||
|
# image has 124 uniqe qlpha values
|
||||||
|
assert_equal(len(im.split()[3].getcolors()), 124)
|
||||||
|
|
||||||
def test_load_verify():
|
def test_load_verify():
|
||||||
# Check open/load/verify exception (@PIL150)
|
# Check open/load/verify exception (@PIL150)
|
||||||
|
|
||||||
|
|
|
@ -13,3 +13,10 @@ def test_sanity():
|
||||||
im = im.quantize(palette=lena("P"))
|
im = im.quantize(palette=lena("P"))
|
||||||
assert_image(im, "P", im.size)
|
assert_image(im, "P", im.size)
|
||||||
|
|
||||||
|
def test_octree_quantize():
|
||||||
|
im = lena()
|
||||||
|
|
||||||
|
im = im.quantize(100, Image.FASTOCTREE)
|
||||||
|
assert_image(im, "P", im.size)
|
||||||
|
|
||||||
|
assert len(im.getcolors()) == 100
|
59
_imaging.c
59
_imaging.c
|
@ -924,6 +924,17 @@ _getpalette(ImagingObject* self, PyObject* args)
|
||||||
return palette;
|
return palette;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static PyObject*
|
||||||
|
_getpalettemode(ImagingObject* self, PyObject* args)
|
||||||
|
{
|
||||||
|
if (!self->image->palette) {
|
||||||
|
PyErr_SetString(PyExc_ValueError, no_palette);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
return PyString_FromString(self->image->palette->mode);
|
||||||
|
}
|
||||||
|
|
||||||
static inline int
|
static inline int
|
||||||
_getxy(PyObject* xy, int* x, int *y)
|
_getxy(PyObject* xy, int* x, int *y)
|
||||||
{
|
{
|
||||||
|
@ -1415,9 +1426,7 @@ _putpalettealpha(ImagingObject* self, PyObject* args)
|
||||||
{
|
{
|
||||||
int index;
|
int index;
|
||||||
int alpha = 0;
|
int alpha = 0;
|
||||||
char* tpalette = NULL;
|
if (!PyArg_ParseTuple(args, "i|i", &index, &alpha))
|
||||||
int tpaletteSize = 0;
|
|
||||||
if (!PyArg_ParseTuple(args, "i|i"PY_ARG_BYTES_LENGTH, &index, &alpha, &tpalette, &tpaletteSize))
|
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
if (!self->image->palette) {
|
if (!self->image->palette) {
|
||||||
|
@ -1431,14 +1440,34 @@ _putpalettealpha(ImagingObject* self, PyObject* args)
|
||||||
}
|
}
|
||||||
|
|
||||||
strcpy(self->image->palette->mode, "RGBA");
|
strcpy(self->image->palette->mode, "RGBA");
|
||||||
|
|
||||||
if (tpaletteSize > 0) {
|
|
||||||
for (index = 0; index < tpaletteSize; index++) {
|
|
||||||
self->image->palette->palette[index*4+3] = (UINT8) tpalette[index];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
self->image->palette->palette[index*4+3] = (UINT8) alpha;
|
self->image->palette->palette[index*4+3] = (UINT8) alpha;
|
||||||
|
|
||||||
|
Py_INCREF(Py_None);
|
||||||
|
return Py_None;
|
||||||
|
}
|
||||||
|
|
||||||
|
static PyObject*
|
||||||
|
_putpalettealphas(ImagingObject* self, PyObject* args)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
UINT8 *values;
|
||||||
|
int length;
|
||||||
|
if (!PyArg_ParseTuple(args, "s#", &values, &length))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
if (!self->image->palette) {
|
||||||
|
PyErr_SetString(PyExc_ValueError, no_palette);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (length > 256) {
|
||||||
|
PyErr_SetString(PyExc_ValueError, outside_palette);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
strcpy(self->image->palette->mode, "RGBA");
|
||||||
|
for (i=0; i<length; i++) {
|
||||||
|
self->image->palette->palette[i*4+3] = (UINT8) values[i];
|
||||||
}
|
}
|
||||||
|
|
||||||
Py_INCREF(Py_None);
|
Py_INCREF(Py_None);
|
||||||
|
@ -2937,8 +2966,10 @@ static struct PyMethodDef methods[] = {
|
||||||
{"setmode", (PyCFunction)im_setmode, 1},
|
{"setmode", (PyCFunction)im_setmode, 1},
|
||||||
|
|
||||||
{"getpalette", (PyCFunction)_getpalette, 1},
|
{"getpalette", (PyCFunction)_getpalette, 1},
|
||||||
|
{"getpalettemode", (PyCFunction)_getpalettemode, 1},
|
||||||
{"putpalette", (PyCFunction)_putpalette, 1},
|
{"putpalette", (PyCFunction)_putpalette, 1},
|
||||||
{"putpalettealpha", (PyCFunction)_putpalettealpha, 1},
|
{"putpalettealpha", (PyCFunction)_putpalettealpha, 1},
|
||||||
|
{"putpalettealphas", (PyCFunction)_putpalettealphas, 1},
|
||||||
|
|
||||||
#ifdef WITH_IMAGECHOPS
|
#ifdef WITH_IMAGECHOPS
|
||||||
/* Channel operations (ImageChops) */
|
/* Channel operations (ImageChops) */
|
||||||
|
@ -3371,6 +3402,13 @@ setup_module(PyObject* m) {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef HAVE_LIBZ
|
#ifdef HAVE_LIBZ
|
||||||
|
#include "zlib.h"
|
||||||
|
/* zip encoding strategies */
|
||||||
|
PyModule_AddIntConstant(m, "DEFAULT_STRATEGY", Z_DEFAULT_STRATEGY);
|
||||||
|
PyModule_AddIntConstant(m, "FILTERED", Z_FILTERED);
|
||||||
|
PyModule_AddIntConstant(m, "HUFFMAN_ONLY", Z_HUFFMAN_ONLY);
|
||||||
|
PyModule_AddIntConstant(m, "RLE", Z_RLE);
|
||||||
|
PyModule_AddIntConstant(m, "FIXED", Z_FIXED);
|
||||||
{
|
{
|
||||||
extern const char* ImagingZipVersion(void);
|
extern const char* ImagingZipVersion(void);
|
||||||
PyDict_SetItemString(d, "zlib_version", PyUnicode_FromString(ImagingZipVersion()));
|
PyDict_SetItemString(d, "zlib_version", PyUnicode_FromString(ImagingZipVersion()));
|
||||||
|
@ -3409,4 +3447,3 @@ init_imaging(void)
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
|
10
encode.c
10
encode.c
|
@ -445,10 +445,14 @@ PyImaging_ZipEncoderNew(PyObject* self, PyObject* args)
|
||||||
char* mode;
|
char* mode;
|
||||||
char* rawmode;
|
char* rawmode;
|
||||||
int optimize = 0;
|
int optimize = 0;
|
||||||
|
int compress_level = -1;
|
||||||
|
int compress_type = -1;
|
||||||
char* dictionary = NULL;
|
char* dictionary = NULL;
|
||||||
int dictionary_size = 0;
|
int dictionary_size = 0;
|
||||||
if (!PyArg_ParseTuple(args, "ss|i"PY_ARG_BYTES_LENGTH, &mode, &rawmode,
|
if (!PyArg_ParseTuple(args, "ss|iii"PY_ARG_BYTES_LENGTH, &mode, &rawmode,
|
||||||
&optimize, &dictionary, &dictionary_size))
|
&optimize,
|
||||||
|
&compress_level, &compress_type,
|
||||||
|
&dictionary, &dictionary_size))
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
/* Copy to avoid referencing Python's memory, but there's no mechanism to
|
/* Copy to avoid referencing Python's memory, but there's no mechanism to
|
||||||
|
@ -477,6 +481,8 @@ PyImaging_ZipEncoderNew(PyObject* self, PyObject* args)
|
||||||
((ZIPSTATE*)encoder->state.context)->mode = ZIP_PNG_PALETTE;
|
((ZIPSTATE*)encoder->state.context)->mode = ZIP_PNG_PALETTE;
|
||||||
|
|
||||||
((ZIPSTATE*)encoder->state.context)->optimize = optimize;
|
((ZIPSTATE*)encoder->state.context)->optimize = optimize;
|
||||||
|
((ZIPSTATE*)encoder->state.context)->compress_level = compress_level;
|
||||||
|
((ZIPSTATE*)encoder->state.context)->compress_type = compress_type;
|
||||||
((ZIPSTATE*)encoder->state.context)->dictionary = dictionary;
|
((ZIPSTATE*)encoder->state.context)->dictionary = dictionary;
|
||||||
((ZIPSTATE*)encoder->state.context)->dictionary_size = dictionary_size;
|
((ZIPSTATE*)encoder->state.context)->dictionary_size = dictionary_size;
|
||||||
|
|
||||||
|
|
|
@ -26,6 +26,7 @@
|
||||||
#include <time.h>
|
#include <time.h>
|
||||||
|
|
||||||
#include "Quant.h"
|
#include "Quant.h"
|
||||||
|
#include "QuantOctree.h"
|
||||||
|
|
||||||
#include "QuantDefines.h"
|
#include "QuantDefines.h"
|
||||||
#include "QuantHash.h"
|
#include "QuantHash.h"
|
||||||
|
@ -1485,6 +1486,8 @@ ImagingQuantize(Imaging im, int colors, int mode, int kmeans)
|
||||||
int result;
|
int result;
|
||||||
unsigned long* newData;
|
unsigned long* newData;
|
||||||
Imaging imOut;
|
Imaging imOut;
|
||||||
|
int withAlpha = 0;
|
||||||
|
ImagingSectionCookie cookie;
|
||||||
|
|
||||||
if (!im)
|
if (!im)
|
||||||
return ImagingError_ModeError();
|
return ImagingError_ModeError();
|
||||||
|
@ -1494,7 +1497,11 @@ ImagingQuantize(Imaging im, int colors, int mode, int kmeans)
|
||||||
return (Imaging) ImagingError_ValueError("bad number of colors");
|
return (Imaging) ImagingError_ValueError("bad number of colors");
|
||||||
|
|
||||||
if (strcmp(im->mode, "L") != 0 && strcmp(im->mode, "P") != 0 &&
|
if (strcmp(im->mode, "L") != 0 && strcmp(im->mode, "P") != 0 &&
|
||||||
strcmp(im->mode, "RGB"))
|
strcmp(im->mode, "RGB") != 0 && strcmp(im->mode, "RGBA") !=0)
|
||||||
|
return ImagingError_ModeError();
|
||||||
|
|
||||||
|
/* only octree supports RGBA */
|
||||||
|
if (!strcmp(im->mode, "RGBA") && mode != 2)
|
||||||
return ImagingError_ModeError();
|
return ImagingError_ModeError();
|
||||||
|
|
||||||
p = malloc(sizeof(Pixel) * im->xsize * im->ysize);
|
p = malloc(sizeof(Pixel) * im->xsize * im->ysize);
|
||||||
|
@ -1529,7 +1536,7 @@ ImagingQuantize(Imaging im, int colors, int mode, int kmeans)
|
||||||
p[i].c.b = pp[v*4+2];
|
p[i].c.b = pp[v*4+2];
|
||||||
}
|
}
|
||||||
|
|
||||||
} else if (!strcmp(im->mode, "RGB")) {
|
} else if (!strcmp(im->mode, "RGB") || !strcmp(im->mode, "RGBA")) {
|
||||||
/* true colour */
|
/* true colour */
|
||||||
|
|
||||||
for (i = y = 0; y < im->ysize; y++)
|
for (i = y = 0; y < im->ysize; y++)
|
||||||
|
@ -1541,6 +1548,8 @@ ImagingQuantize(Imaging im, int colors, int mode, int kmeans)
|
||||||
return (Imaging) ImagingError_ValueError("internal error");
|
return (Imaging) ImagingError_ValueError("internal error");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ImagingSectionEnter(&cookie);
|
||||||
|
|
||||||
switch (mode) {
|
switch (mode) {
|
||||||
case 0:
|
case 0:
|
||||||
/* median cut */
|
/* median cut */
|
||||||
|
@ -1566,16 +1575,31 @@ ImagingQuantize(Imaging im, int colors, int mode, int kmeans)
|
||||||
kmeans
|
kmeans
|
||||||
);
|
);
|
||||||
break;
|
break;
|
||||||
|
case 2:
|
||||||
|
if (!strcmp(im->mode, "RGBA")) {
|
||||||
|
withAlpha = 1;
|
||||||
|
}
|
||||||
|
result = quantize_octree(
|
||||||
|
p,
|
||||||
|
im->xsize*im->ysize,
|
||||||
|
colors,
|
||||||
|
&palette,
|
||||||
|
&paletteLength,
|
||||||
|
&newData,
|
||||||
|
withAlpha
|
||||||
|
);
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
result = 0;
|
result = 0;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
free(p);
|
free(p);
|
||||||
|
ImagingSectionLeave(&cookie);
|
||||||
|
|
||||||
if (result) {
|
if (result) {
|
||||||
|
|
||||||
imOut = ImagingNew("P", im->xsize, im->ysize);
|
imOut = ImagingNew("P", im->xsize, im->ysize);
|
||||||
|
ImagingSectionEnter(&cookie);
|
||||||
|
|
||||||
for (i = y = 0; y < im->ysize; y++)
|
for (i = y = 0; y < im->ysize; y++)
|
||||||
for (x=0; x < im->xsize; x++)
|
for (x=0; x < im->xsize; x++)
|
||||||
|
@ -1589,8 +1613,12 @@ ImagingQuantize(Imaging im, int colors, int mode, int kmeans)
|
||||||
*pp++ = palette[i].c.r;
|
*pp++ = palette[i].c.r;
|
||||||
*pp++ = palette[i].c.g;
|
*pp++ = palette[i].c.g;
|
||||||
*pp++ = palette[i].c.b;
|
*pp++ = palette[i].c.b;
|
||||||
|
if (withAlpha) {
|
||||||
|
*pp++ = palette[i].c.a;
|
||||||
|
} else {
|
||||||
*pp++ = 255;
|
*pp++ = 255;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
for (; i < 256; i++) {
|
for (; i < 256; i++) {
|
||||||
*pp++ = 0;
|
*pp++ = 0;
|
||||||
*pp++ = 0;
|
*pp++ = 0;
|
||||||
|
@ -1598,7 +1626,12 @@ ImagingQuantize(Imaging im, int colors, int mode, int kmeans)
|
||||||
*pp++ = 255;
|
*pp++ = 255;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (withAlpha) {
|
||||||
|
strcpy(imOut->palette->mode, "RGBA");
|
||||||
|
}
|
||||||
|
|
||||||
free(palette);
|
free(palette);
|
||||||
|
ImagingSectionLeave(&cookie);
|
||||||
|
|
||||||
return imOut;
|
return imOut;
|
||||||
|
|
||||||
|
|
454
libImaging/QuantOctree.c
Normal file
454
libImaging/QuantOctree.c
Normal file
|
@ -0,0 +1,454 @@
|
||||||
|
/* Copyright (c) 2010 Oliver Tonnhofer <olt@bogosoft.com>, Omniscale
|
||||||
|
//
|
||||||
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
// of this software and associated documentation files (the "Software"), to deal
|
||||||
|
// in the Software without restriction, including without limitation the rights
|
||||||
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
// copies of the Software, and to permit persons to whom the Software is
|
||||||
|
// furnished to do so, subject to the following conditions:
|
||||||
|
//
|
||||||
|
// The above copyright notice and this permission notice shall be included in
|
||||||
|
// all copies or substantial portions of the Software.
|
||||||
|
//
|
||||||
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
|
// THE SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
// This file implements a variation of the octree color quantization algorithm.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include "Quant.h"
|
||||||
|
|
||||||
|
typedef struct _ColorBucket{
|
||||||
|
/* contains palette index when used for look up cube */
|
||||||
|
unsigned long count;
|
||||||
|
unsigned long r;
|
||||||
|
unsigned long g;
|
||||||
|
unsigned long b;
|
||||||
|
unsigned long a;
|
||||||
|
} *ColorBucket;
|
||||||
|
|
||||||
|
typedef struct _ColorCube{
|
||||||
|
unsigned int rBits, gBits, bBits, aBits;
|
||||||
|
unsigned int rWidth, gWidth, bWidth, aWidth;
|
||||||
|
unsigned int rOffset, gOffset, bOffset, aOffset;
|
||||||
|
|
||||||
|
long size;
|
||||||
|
ColorBucket buckets;
|
||||||
|
} *ColorCube;
|
||||||
|
|
||||||
|
#define MAX(a, b) (a)>(b) ? (a) : (b)
|
||||||
|
|
||||||
|
static ColorCube
|
||||||
|
new_color_cube(int r, int g, int b, int a) {
|
||||||
|
ColorCube cube;
|
||||||
|
|
||||||
|
cube = malloc(sizeof(struct _ColorCube));
|
||||||
|
if (!cube) return NULL;
|
||||||
|
|
||||||
|
cube->rBits = MAX(r, 0);
|
||||||
|
cube->gBits = MAX(g, 0);
|
||||||
|
cube->bBits = MAX(b, 0);
|
||||||
|
cube->aBits = MAX(a, 0);
|
||||||
|
|
||||||
|
/* the width of the cube for each dimension */
|
||||||
|
cube->rWidth = 1<<cube->rBits;
|
||||||
|
cube->gWidth = 1<<cube->gBits;
|
||||||
|
cube->bWidth = 1<<cube->bBits;
|
||||||
|
cube->aWidth = 1<<cube->aBits;
|
||||||
|
|
||||||
|
/* the offsets of each color */
|
||||||
|
|
||||||
|
cube->rOffset = cube->gBits + cube->bBits + cube->aBits;
|
||||||
|
cube->gOffset = cube->bBits + cube->aBits;
|
||||||
|
cube->bOffset = cube->aBits;
|
||||||
|
cube->aOffset = 0;
|
||||||
|
|
||||||
|
/* the number of color buckets */
|
||||||
|
cube->size = cube->rWidth * cube->gWidth * cube->bWidth * cube->aWidth;
|
||||||
|
cube->buckets = calloc(cube->size, sizeof(struct _ColorBucket));
|
||||||
|
|
||||||
|
if (!cube->buckets) {
|
||||||
|
free(cube);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
return cube;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
free_color_cube(ColorCube cube) {
|
||||||
|
if (cube != NULL) {
|
||||||
|
free(cube->buckets);
|
||||||
|
free(cube);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static long
|
||||||
|
color_bucket_offset_pos(const ColorCube cube,
|
||||||
|
unsigned int r, unsigned int g, unsigned int b, unsigned int a)
|
||||||
|
{
|
||||||
|
return r<<cube->rOffset | g<<cube->gOffset | b<<cube->bOffset | a<<cube->aOffset;
|
||||||
|
}
|
||||||
|
|
||||||
|
static long
|
||||||
|
color_bucket_offset(const ColorCube cube, const Pixel *p) {
|
||||||
|
unsigned int r = p->c.r>>(8-cube->rBits);
|
||||||
|
unsigned int g = p->c.g>>(8-cube->gBits);
|
||||||
|
unsigned int b = p->c.b>>(8-cube->bBits);
|
||||||
|
unsigned int a = p->c.a>>(8-cube->aBits);
|
||||||
|
return color_bucket_offset_pos(cube, r, g, b, a);
|
||||||
|
}
|
||||||
|
|
||||||
|
static ColorBucket
|
||||||
|
color_bucket_from_cube(const ColorCube cube, const Pixel *p) {
|
||||||
|
unsigned int offset = color_bucket_offset(cube, p);
|
||||||
|
return &cube->buckets[offset];
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
add_color_to_color_cube(const ColorCube cube, const Pixel *p) {
|
||||||
|
ColorBucket bucket = color_bucket_from_cube(cube, p);
|
||||||
|
bucket->count += 1;
|
||||||
|
bucket->r += p->c.r;
|
||||||
|
bucket->g += p->c.g;
|
||||||
|
bucket->b += p->c.b;
|
||||||
|
bucket->a += p->c.a;
|
||||||
|
}
|
||||||
|
|
||||||
|
static long
|
||||||
|
count_used_color_buckets(const ColorCube cube) {
|
||||||
|
long usedBuckets = 0;
|
||||||
|
long i;
|
||||||
|
for (i=0; i < cube->size; i++) {
|
||||||
|
if (cube->buckets[i].count > 0) {
|
||||||
|
usedBuckets += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return usedBuckets;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
avg_color_from_color_bucket(const ColorBucket bucket, Pixel *dst) {
|
||||||
|
float count = bucket->count;
|
||||||
|
dst->c.r = (int)(bucket->r / count);
|
||||||
|
dst->c.g = (int)(bucket->g / count);
|
||||||
|
dst->c.b = (int)(bucket->b / count);
|
||||||
|
dst->c.a = (int)(bucket->a / count);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
compare_bucket_count(const ColorBucket a, const ColorBucket b) {
|
||||||
|
return b->count - a->count;
|
||||||
|
}
|
||||||
|
|
||||||
|
static ColorBucket
|
||||||
|
create_sorted_color_palette(const ColorCube cube) {
|
||||||
|
ColorBucket buckets;
|
||||||
|
buckets = malloc(sizeof(struct _ColorBucket)*cube->size);
|
||||||
|
if (!buckets) return NULL;
|
||||||
|
memcpy(buckets, cube->buckets, sizeof(struct _ColorBucket)*cube->size);
|
||||||
|
|
||||||
|
qsort(buckets, cube->size, sizeof(struct _ColorBucket),
|
||||||
|
(int (*)(void const *, void const *))&compare_bucket_count);
|
||||||
|
|
||||||
|
return buckets;
|
||||||
|
}
|
||||||
|
|
||||||
|
void add_bucket_values(ColorBucket src, ColorBucket dst) {
|
||||||
|
dst->count += src->count;
|
||||||
|
dst->r += src->r;
|
||||||
|
dst->g += src->g;
|
||||||
|
dst->b += src->b;
|
||||||
|
dst->a += src->a;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* expand or shrink a given cube to level */
|
||||||
|
static ColorCube copy_color_cube(const ColorCube cube,
|
||||||
|
int rBits, int gBits, int bBits, int aBits)
|
||||||
|
{
|
||||||
|
unsigned int r, g, b, a;
|
||||||
|
long src_pos, dst_pos;
|
||||||
|
unsigned int src_reduce[4] = {0}, dst_reduce[4] = {0};
|
||||||
|
unsigned int width[4];
|
||||||
|
ColorCube result;
|
||||||
|
|
||||||
|
result = new_color_cube(rBits, gBits, bBits, aBits);
|
||||||
|
if (!result) return NULL;
|
||||||
|
|
||||||
|
if (cube->rBits > rBits) {
|
||||||
|
dst_reduce[0] = cube->rBits - result->rBits;
|
||||||
|
width[0] = cube->rWidth;
|
||||||
|
} else {
|
||||||
|
src_reduce[0] = result->rBits - cube->rBits;
|
||||||
|
width[0] = result->rWidth;
|
||||||
|
}
|
||||||
|
if (cube->gBits > gBits) {
|
||||||
|
dst_reduce[1] = cube->gBits - result->gBits;
|
||||||
|
width[1] = cube->gWidth;
|
||||||
|
} else {
|
||||||
|
src_reduce[1] = result->gBits - cube->gBits;
|
||||||
|
width[1] = result->gWidth;
|
||||||
|
}
|
||||||
|
if (cube->bBits > bBits) {
|
||||||
|
dst_reduce[2] = cube->bBits - result->bBits;
|
||||||
|
width[2] = cube->bWidth;
|
||||||
|
} else {
|
||||||
|
src_reduce[2] = result->bBits - cube->bBits;
|
||||||
|
width[2] = result->bWidth;
|
||||||
|
}
|
||||||
|
if (cube->aBits > aBits) {
|
||||||
|
dst_reduce[3] = cube->aBits - result->aBits;
|
||||||
|
width[3] = cube->aWidth;
|
||||||
|
} else {
|
||||||
|
src_reduce[3] = result->aBits - cube->aBits;
|
||||||
|
width[3] = result->aWidth;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (r=0; r<width[0]; r++) {
|
||||||
|
for (g=0; g<width[1]; g++) {
|
||||||
|
for (b=0; b<width[2]; b++) {
|
||||||
|
for (a=0; a<width[3]; a++) {
|
||||||
|
src_pos = color_bucket_offset_pos(cube,
|
||||||
|
r>>src_reduce[0],
|
||||||
|
g>>src_reduce[1],
|
||||||
|
b>>src_reduce[2],
|
||||||
|
a>>src_reduce[3]);
|
||||||
|
dst_pos = color_bucket_offset_pos(result,
|
||||||
|
r>>dst_reduce[0],
|
||||||
|
g>>dst_reduce[1],
|
||||||
|
b>>dst_reduce[2],
|
||||||
|
a>>dst_reduce[3]);
|
||||||
|
add_bucket_values(
|
||||||
|
&cube->buckets[src_pos],
|
||||||
|
&result->buckets[dst_pos]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
subtract_color_buckets(ColorCube cube, ColorBucket buckets, long nBuckets) {
|
||||||
|
ColorBucket minuend, subtrahend;
|
||||||
|
long i;
|
||||||
|
Pixel p;
|
||||||
|
for (i=0; i<nBuckets; i++) {
|
||||||
|
subtrahend = &buckets[i];
|
||||||
|
avg_color_from_color_bucket(subtrahend, &p);
|
||||||
|
minuend = color_bucket_from_cube(cube, &p);
|
||||||
|
minuend->count -= subtrahend->count;
|
||||||
|
minuend->r -= subtrahend->r;
|
||||||
|
minuend->g -= subtrahend->g;
|
||||||
|
minuend->b -= subtrahend->b;
|
||||||
|
minuend->a -= subtrahend->a;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
set_lookup_value(const ColorCube cube, const Pixel *p, long value) {
|
||||||
|
ColorBucket bucket = color_bucket_from_cube(cube, p);
|
||||||
|
bucket->count = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned long
|
||||||
|
lookup_color(const ColorCube cube, const Pixel *p) {
|
||||||
|
ColorBucket bucket = color_bucket_from_cube(cube, p);
|
||||||
|
return bucket->count;
|
||||||
|
}
|
||||||
|
|
||||||
|
void add_lookup_buckets(ColorCube cube, ColorBucket palette, long nColors, long offset) {
|
||||||
|
long i;
|
||||||
|
Pixel p;
|
||||||
|
for (i=offset; i<offset+nColors; i++) {
|
||||||
|
avg_color_from_color_bucket(&palette[i], &p);
|
||||||
|
set_lookup_value(cube, &p, i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ColorBucket
|
||||||
|
combined_palette(ColorBucket bucketsA, long nBucketsA, ColorBucket bucketsB, long nBucketsB) {
|
||||||
|
ColorBucket result;
|
||||||
|
result = malloc(sizeof(struct _ColorBucket)*(nBucketsA+nBucketsB));
|
||||||
|
memcpy(result, bucketsA, sizeof(struct _ColorBucket) * nBucketsA);
|
||||||
|
memcpy(&result[nBucketsA], bucketsB, sizeof(struct _ColorBucket) * nBucketsB);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static Pixel *
|
||||||
|
create_palette_array(const ColorBucket palette, unsigned int paletteLength) {
|
||||||
|
Pixel *paletteArray;
|
||||||
|
unsigned int i;
|
||||||
|
|
||||||
|
paletteArray = malloc(sizeof(Pixel)*paletteLength);
|
||||||
|
if (!paletteArray) return NULL;
|
||||||
|
|
||||||
|
for (i=0; i<paletteLength; i++) {
|
||||||
|
avg_color_from_color_bucket(&palette[i], &paletteArray[i]);
|
||||||
|
}
|
||||||
|
return paletteArray;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
map_image_pixels(const Pixel *pixelData,
|
||||||
|
unsigned long nPixels,
|
||||||
|
const ColorCube lookupCube,
|
||||||
|
unsigned long *pixelArray)
|
||||||
|
{
|
||||||
|
long i;
|
||||||
|
for (i=0; i<nPixels; i++) {
|
||||||
|
pixelArray[i] = lookup_color(lookupCube, &pixelData[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const int CUBE_LEVELS[8] = {4, 4, 4, 0, 2, 2, 2, 0};
|
||||||
|
const int CUBE_LEVELS_ALPHA[8] = {3, 4, 3, 3, 2, 2, 2, 2};
|
||||||
|
|
||||||
|
int quantize_octree(Pixel *pixelData,
|
||||||
|
unsigned long nPixels,
|
||||||
|
unsigned long nQuantPixels,
|
||||||
|
Pixel **palette,
|
||||||
|
unsigned long *paletteLength,
|
||||||
|
unsigned long **quantizedPixels,
|
||||||
|
int withAlpha)
|
||||||
|
{
|
||||||
|
ColorCube fineCube = NULL;
|
||||||
|
ColorCube coarseCube = NULL;
|
||||||
|
ColorCube lookupCube = NULL;
|
||||||
|
ColorCube coarseLookupCube = NULL;
|
||||||
|
ColorBucket paletteBucketsCoarse = NULL;
|
||||||
|
ColorBucket paletteBucketsFine = NULL;
|
||||||
|
ColorBucket paletteBuckets = NULL;
|
||||||
|
unsigned long *qp = NULL;
|
||||||
|
long i;
|
||||||
|
long nCoarseColors, nFineColors, nAlreadySubtracted;
|
||||||
|
const int *cubeBits;
|
||||||
|
|
||||||
|
if (withAlpha) {
|
||||||
|
cubeBits = CUBE_LEVELS_ALPHA;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
cubeBits = CUBE_LEVELS;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Create two color cubes, one fine grained with 8x16x8=1024
|
||||||
|
colors buckets and a coarse with 4x4x4=64 color buckets.
|
||||||
|
The coarse one guarantes that there are color buckets available for
|
||||||
|
the whole color range (assuming nQuantPixels > 64).
|
||||||
|
|
||||||
|
For a quantization to 256 colors all 64 coarse colors will be used
|
||||||
|
plus the 192 most used color buckets from the fine color cube.
|
||||||
|
The average of all colors within one bucket is used as the actual
|
||||||
|
color for that bucket.
|
||||||
|
|
||||||
|
For images with alpha the cubes gets a forth dimension,
|
||||||
|
8x16x8x8 and 4x4x4x4.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* create fine cube */
|
||||||
|
fineCube = new_color_cube(cubeBits[0], cubeBits[1],
|
||||||
|
cubeBits[2], cubeBits[3]);
|
||||||
|
if (!fineCube) goto error;
|
||||||
|
for (i=0; i<nPixels; i++) {
|
||||||
|
add_color_to_color_cube(fineCube, &pixelData[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* create coarse cube */
|
||||||
|
coarseCube = copy_color_cube(fineCube, cubeBits[4], cubeBits[5],
|
||||||
|
cubeBits[6], cubeBits[7]);
|
||||||
|
if (!coarseCube) goto error;
|
||||||
|
nCoarseColors = count_used_color_buckets(coarseCube);
|
||||||
|
|
||||||
|
/* limit to nQuantPixels */
|
||||||
|
if (nCoarseColors > nQuantPixels)
|
||||||
|
nCoarseColors = nQuantPixels;
|
||||||
|
|
||||||
|
/* how many space do we have in our palette for fine colors? */
|
||||||
|
nFineColors = nQuantPixels - nCoarseColors;
|
||||||
|
|
||||||
|
/* create fine color palette */
|
||||||
|
paletteBucketsFine = create_sorted_color_palette(fineCube);
|
||||||
|
if (!paletteBucketsFine) goto error;
|
||||||
|
|
||||||
|
/* remove the used fine colors from the coarse cube */
|
||||||
|
subtract_color_buckets(coarseCube, paletteBucketsFine, nFineColors);
|
||||||
|
|
||||||
|
/* did the substraction cleared one or more coarse bucket? */
|
||||||
|
while (nCoarseColors > count_used_color_buckets(coarseCube)) {
|
||||||
|
/* then we can use the free buckets for fine colors */
|
||||||
|
nAlreadySubtracted = nFineColors;
|
||||||
|
nCoarseColors = count_used_color_buckets(coarseCube);
|
||||||
|
nFineColors = nQuantPixels - nCoarseColors;
|
||||||
|
subtract_color_buckets(coarseCube, &paletteBucketsFine[nAlreadySubtracted],
|
||||||
|
nFineColors-nAlreadySubtracted);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* create our palette buckets with fine and coarse combined */
|
||||||
|
paletteBucketsCoarse = create_sorted_color_palette(coarseCube);
|
||||||
|
if (!paletteBucketsCoarse) goto error;
|
||||||
|
paletteBuckets = combined_palette(paletteBucketsCoarse, nCoarseColors,
|
||||||
|
paletteBucketsFine, nFineColors);
|
||||||
|
|
||||||
|
free(paletteBucketsFine);
|
||||||
|
paletteBucketsFine = NULL;
|
||||||
|
free(paletteBucketsCoarse);
|
||||||
|
paletteBucketsCoarse = NULL;
|
||||||
|
|
||||||
|
/* add all coarse colors to our coarse lookup cube. */
|
||||||
|
coarseLookupCube = new_color_cube(cubeBits[4], cubeBits[5],
|
||||||
|
cubeBits[6], cubeBits[7]);
|
||||||
|
if (!coarseLookupCube) goto error;
|
||||||
|
add_lookup_buckets(coarseLookupCube, paletteBuckets, nCoarseColors, 0);
|
||||||
|
|
||||||
|
/* expand coarse cube (64) to larger fine cube (4k). the value of each
|
||||||
|
coarse bucket is then present in the according 64 fine buckets. */
|
||||||
|
lookupCube = copy_color_cube(coarseLookupCube, cubeBits[0], cubeBits[1],
|
||||||
|
cubeBits[2], cubeBits[3]);
|
||||||
|
if (!lookupCube) goto error;
|
||||||
|
|
||||||
|
/* add fine colors to the lookup cube */
|
||||||
|
add_lookup_buckets(lookupCube, paletteBuckets, nFineColors, nCoarseColors);
|
||||||
|
|
||||||
|
/* create result pixles and map palatte indices */
|
||||||
|
qp = malloc(sizeof(Pixel)*nPixels);
|
||||||
|
if (!qp) goto error;
|
||||||
|
map_image_pixels(pixelData, nPixels, lookupCube, qp);
|
||||||
|
|
||||||
|
/* convert palette buckets to RGB pixel palette */
|
||||||
|
*palette = create_palette_array(paletteBuckets, nQuantPixels);
|
||||||
|
if (!(*palette)) goto error;
|
||||||
|
|
||||||
|
*quantizedPixels = qp;
|
||||||
|
*paletteLength = nQuantPixels;
|
||||||
|
|
||||||
|
free_color_cube(coarseCube);
|
||||||
|
free_color_cube(fineCube);
|
||||||
|
free_color_cube(lookupCube);
|
||||||
|
free_color_cube(coarseLookupCube);
|
||||||
|
free(paletteBuckets);
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
error:
|
||||||
|
/* everything is initialized to NULL
|
||||||
|
so we are safe to call free */
|
||||||
|
free(qp);
|
||||||
|
free_color_cube(lookupCube);
|
||||||
|
free_color_cube(coarseLookupCube);
|
||||||
|
free(paletteBucketsCoarse);
|
||||||
|
free(paletteBucketsFine);
|
||||||
|
free_color_cube(coarseCube);
|
||||||
|
free_color_cube(fineCube);
|
||||||
|
return 0;
|
||||||
|
}
|
12
libImaging/QuantOctree.h
Normal file
12
libImaging/QuantOctree.h
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
#ifndef __QUANT_OCTREE_H__
|
||||||
|
#define __QUANT_OCTREE_H__
|
||||||
|
|
||||||
|
int quantize_octree(Pixel *,
|
||||||
|
unsigned long,
|
||||||
|
unsigned long,
|
||||||
|
Pixel **,
|
||||||
|
unsigned long *,
|
||||||
|
unsigned long **,
|
||||||
|
int);
|
||||||
|
|
||||||
|
#endif
|
|
@ -28,6 +28,11 @@ typedef struct {
|
||||||
/* Optimize (max compression) SLOW!!! */
|
/* Optimize (max compression) SLOW!!! */
|
||||||
int optimize;
|
int optimize;
|
||||||
|
|
||||||
|
/* 0 no compression, 9 best compression, -1 default compression */
|
||||||
|
int compress_level;
|
||||||
|
/* compression strategy Z_XXX */
|
||||||
|
int compress_type;
|
||||||
|
|
||||||
/* Predefined dictionary (experimental) */
|
/* Predefined dictionary (experimental) */
|
||||||
char* dictionary;
|
char* dictionary;
|
||||||
int dictionary_size;
|
int dictionary_size;
|
||||||
|
|
|
@ -26,6 +26,7 @@ ImagingZipEncode(Imaging im, ImagingCodecState state, UINT8* buf, int bytes)
|
||||||
{
|
{
|
||||||
ZIPSTATE* context = (ZIPSTATE*) state->context;
|
ZIPSTATE* context = (ZIPSTATE*) state->context;
|
||||||
int err;
|
int err;
|
||||||
|
int compress_level, compress_type;
|
||||||
UINT8* ptr;
|
UINT8* ptr;
|
||||||
int i, bpp, s, sum;
|
int i, bpp, s, sum;
|
||||||
ImagingSectionCookie cookie;
|
ImagingSectionCookie cookie;
|
||||||
|
@ -73,17 +74,25 @@ ImagingZipEncode(Imaging im, ImagingCodecState state, UINT8* buf, int bytes)
|
||||||
context->z_stream.next_in = 0;
|
context->z_stream.next_in = 0;
|
||||||
context->z_stream.avail_in = 0;
|
context->z_stream.avail_in = 0;
|
||||||
|
|
||||||
|
compress_level = (context->optimize) ? Z_BEST_COMPRESSION
|
||||||
|
: context->compress_level;
|
||||||
|
|
||||||
|
if (context->compress_type == -1) {
|
||||||
|
compress_type = (context->mode == ZIP_PNG) ? Z_FILTERED
|
||||||
|
: Z_DEFAULT_STRATEGY;
|
||||||
|
} else {
|
||||||
|
compress_type = context->compress_type;
|
||||||
|
}
|
||||||
|
|
||||||
err = deflateInit2(&context->z_stream,
|
err = deflateInit2(&context->z_stream,
|
||||||
/* compression level */
|
/* compression level */
|
||||||
(context->optimize) ? Z_BEST_COMPRESSION
|
compress_level,
|
||||||
: Z_DEFAULT_COMPRESSION,
|
|
||||||
/* compression method */
|
/* compression method */
|
||||||
Z_DEFLATED,
|
Z_DEFLATED,
|
||||||
/* compression memory resources */
|
/* compression memory resources */
|
||||||
15, 9,
|
15, 9,
|
||||||
/* compression strategy (image data are filtered)*/
|
/* compression strategy (image data are filtered)*/
|
||||||
(context->mode == ZIP_PNG) ? Z_FILTERED
|
compress_type);
|
||||||
: Z_DEFAULT_STRATEGY);
|
|
||||||
if (err < 0) {
|
if (err < 0) {
|
||||||
state->errcode = IMAGING_CODEC_CONFIG;
|
state->errcode = IMAGING_CODEC_CONFIG;
|
||||||
return -1;
|
return -1;
|
||||||
|
|
|
@ -157,7 +157,7 @@ def testimage():
|
||||||
|
|
||||||
def check_module(feature, module):
|
def check_module(feature, module):
|
||||||
try:
|
try:
|
||||||
__import__(module)
|
__import__("PIL." + module)
|
||||||
except ImportError:
|
except ImportError:
|
||||||
print("***", feature, "support not installed")
|
print("***", feature, "support not installed")
|
||||||
else:
|
else:
|
||||||
|
|
2
setup.py
2
setup.py
|
@ -22,7 +22,7 @@ _LIB_IMAGING = (
|
||||||
"Geometry", "GetBBox", "GifDecode", "GifEncode", "HexDecode",
|
"Geometry", "GetBBox", "GifDecode", "GifEncode", "HexDecode",
|
||||||
"Histo", "JpegDecode", "JpegEncode", "LzwDecode", "Matrix",
|
"Histo", "JpegDecode", "JpegEncode", "LzwDecode", "Matrix",
|
||||||
"ModeFilter", "MspDecode", "Negative", "Offset", "Pack",
|
"ModeFilter", "MspDecode", "Negative", "Offset", "Pack",
|
||||||
"PackDecode", "Palette", "Paste", "Quant", "QuantHash",
|
"PackDecode", "Palette", "Paste", "Quant", "QuantOctree", "QuantHash",
|
||||||
"QuantHeap", "PcdDecode", "PcxDecode", "PcxEncode", "Point",
|
"QuantHeap", "PcdDecode", "PcxDecode", "PcxEncode", "Point",
|
||||||
"RankFilter", "RawDecode", "RawEncode", "Storage", "SunRleDecode",
|
"RankFilter", "RawDecode", "RawEncode", "Storage", "SunRleDecode",
|
||||||
"TgaRleDecode", "Unpack", "UnpackYCC", "UnsharpMask", "XbmDecode",
|
"TgaRleDecode", "Unpack", "UnpackYCC", "UnsharpMask", "XbmDecode",
|
||||||
|
|
Loading…
Reference in New Issue
Block a user