mirror of
				https://github.com/python-pillow/Pillow.git
				synced 2025-10-26 21:51:10 +03:00 
			
		
		
		
	Merge pull request #4230 from dwastberg/new_chops
Added three new channel operations
This commit is contained in:
		
						commit
						c6115c1ceb
					
				|  | @ -38,6 +38,10 @@ def test_sanity(): | |||
|     ImageChops.blend(im, im, 0.5) | ||||
|     ImageChops.composite(im, im, im) | ||||
| 
 | ||||
|     ImageChops.soft_light(im, im) | ||||
|     ImageChops.hard_light(im, im) | ||||
|     ImageChops.overlay(im, im) | ||||
| 
 | ||||
|     ImageChops.offset(im, 10) | ||||
|     ImageChops.offset(im, 10, 20) | ||||
| 
 | ||||
|  | @ -362,6 +366,45 @@ def test_subtract_modulo_no_clip(): | |||
|     assert new.getpixel((50, 50)) == (241, 167, 127) | ||||
| 
 | ||||
| 
 | ||||
| def test_soft_light(): | ||||
|     # Arrange | ||||
|     im1 = Image.open("Tests/images/hopper.png") | ||||
|     im2 = Image.open("Tests/images/hopper-XYZ.png") | ||||
| 
 | ||||
|     # Act | ||||
|     new = ImageChops.soft_light(im1, im2) | ||||
| 
 | ||||
|     # Assert | ||||
|     assert new.getpixel((64, 64)) == (163, 54, 32) | ||||
|     assert new.getpixel((15, 100)) == (1, 1, 3) | ||||
| 
 | ||||
| 
 | ||||
| def test_hard_light(): | ||||
|     # Arrange | ||||
|     im1 = Image.open("Tests/images/hopper.png") | ||||
|     im2 = Image.open("Tests/images/hopper-XYZ.png") | ||||
| 
 | ||||
|     # Act | ||||
|     new = ImageChops.hard_light(im1, im2) | ||||
| 
 | ||||
|     # Assert | ||||
|     assert new.getpixel((64, 64)) == (144, 50, 27) | ||||
|     assert new.getpixel((15, 100)) == (1, 1, 2) | ||||
| 
 | ||||
| 
 | ||||
| def test_overlay(): | ||||
|     # Arrange | ||||
|     im1 = Image.open("Tests/images/hopper.png") | ||||
|     im2 = Image.open("Tests/images/hopper-XYZ.png") | ||||
| 
 | ||||
|     # Act | ||||
|     new = ImageChops.overlay(im1, im2) | ||||
| 
 | ||||
|     # Assert | ||||
|     assert new.getpixel((64, 64)) == (159, 50, 27) | ||||
|     assert new.getpixel((15, 100)) == (1, 1, 2) | ||||
| 
 | ||||
| 
 | ||||
| def test_logical(): | ||||
|     def table(op, a, b): | ||||
|         out = [] | ||||
|  |  | |||
|  | @ -36,6 +36,9 @@ operations in this module). | |||
| .. autofunction:: PIL.ImageChops.logical_or | ||||
| .. autofunction:: PIL.ImageChops.logical_xor | ||||
| .. autofunction:: PIL.ImageChops.multiply | ||||
| .. autofunction:: PIL.ImageChops.soft_light | ||||
| .. autofunction:: PIL.ImageChops.hard_light | ||||
| .. autofunction:: PIL.ImageChops.overlay | ||||
| .. py:method:: PIL.ImageChops.offset(image, xoffset, yoffset=None) | ||||
| 
 | ||||
|     Returns a copy of the image where data has been offset by the given | ||||
|  |  | |||
|  | @ -139,6 +139,42 @@ def screen(image1, image2): | |||
|     return image1._new(image1.im.chop_screen(image2.im)) | ||||
| 
 | ||||
| 
 | ||||
| def soft_light(image1, image2): | ||||
|     """ | ||||
|     Superimposes two images on top of each other using the Soft Light algorithm | ||||
| 
 | ||||
|     :rtype: :py:class:`~PIL.Image.Image` | ||||
|     """ | ||||
| 
 | ||||
|     image1.load() | ||||
|     image2.load() | ||||
|     return image1._new(image1.im.chop_soft_light(image2.im)) | ||||
| 
 | ||||
| 
 | ||||
| def hard_light(image1, image2): | ||||
|     """ | ||||
|     Superimposes two images on top of each other using the Hard Light algorithm | ||||
| 
 | ||||
|     :rtype: :py:class:`~PIL.Image.Image` | ||||
|     """ | ||||
| 
 | ||||
|     image1.load() | ||||
|     image2.load() | ||||
|     return image1._new(image1.im.chop_hard_light(image2.im)) | ||||
| 
 | ||||
| 
 | ||||
| def overlay(image1, image2): | ||||
|     """ | ||||
|     Superimposes two images on top of each other using the Overlay algorithm | ||||
| 
 | ||||
|     :rtype: :py:class:`~PIL.Image.Image` | ||||
|     """ | ||||
| 
 | ||||
|     image1.load() | ||||
|     image2.load() | ||||
|     return image1._new(image1.im.chop_overlay(image2.im)) | ||||
| 
 | ||||
| 
 | ||||
| def add(image1, image2, scale=1.0, offset=0): | ||||
|     """ | ||||
|     Adds two images, dividing the result by scale and adding the | ||||
|  |  | |||
|  | @ -2406,6 +2406,38 @@ _chop_subtract_modulo(ImagingObject* self, PyObject* args) | |||
|     return PyImagingNew(ImagingChopSubtractModulo(self->image, imagep->image)); | ||||
| } | ||||
| 
 | ||||
| static PyObject* | ||||
| _chop_soft_light(ImagingObject* self, PyObject* args) | ||||
| { | ||||
|     ImagingObject* imagep; | ||||
| 
 | ||||
|     if (!PyArg_ParseTuple(args, "O!", &Imaging_Type, &imagep)) | ||||
|         return NULL; | ||||
| 
 | ||||
|     return PyImagingNew(ImagingChopSoftLight(self->image, imagep->image)); | ||||
| } | ||||
| 
 | ||||
| static PyObject* | ||||
| _chop_hard_light(ImagingObject* self, PyObject* args) | ||||
| { | ||||
|     ImagingObject* imagep; | ||||
| 
 | ||||
|     if (!PyArg_ParseTuple(args, "O!", &Imaging_Type, &imagep)) | ||||
|         return NULL; | ||||
| 
 | ||||
|     return PyImagingNew(ImagingChopHardLight(self->image, imagep->image)); | ||||
| } | ||||
| 
 | ||||
| static PyObject* | ||||
| _chop_overlay(ImagingObject* self, PyObject* args) | ||||
| { | ||||
|     ImagingObject* imagep; | ||||
| 
 | ||||
|     if (!PyArg_ParseTuple(args, "O!", &Imaging_Type, &imagep)) | ||||
|         return NULL; | ||||
| 
 | ||||
|     return PyImagingNew(ImagingOverlay(self->image, imagep->image)); | ||||
| } | ||||
| #endif | ||||
| 
 | ||||
| 
 | ||||
|  | @ -3325,6 +3357,10 @@ static struct PyMethodDef methods[] = { | |||
|     {"chop_and", (PyCFunction)_chop_and, 1}, | ||||
|     {"chop_or", (PyCFunction)_chop_or, 1}, | ||||
|     {"chop_xor", (PyCFunction)_chop_xor, 1}, | ||||
|     {"chop_soft_light", (PyCFunction)_chop_soft_light, 1}, | ||||
|     {"chop_hard_light", (PyCFunction)_chop_hard_light, 1}, | ||||
|     {"chop_overlay", (PyCFunction)_chop_overlay, 1}, | ||||
| 
 | ||||
| #endif | ||||
| 
 | ||||
| #ifdef WITH_UNSHARPMASK | ||||
|  |  | |||
|  | @ -146,3 +146,27 @@ ImagingChopSubtractModulo(Imaging imIn1, Imaging imIn2) | |||
| { | ||||
|     CHOP2(in1[x] - in2[x], NULL); | ||||
| } | ||||
| 
 | ||||
| Imaging | ||||
| ImagingChopSoftLight(Imaging imIn1, Imaging imIn2) | ||||
| { | ||||
|     CHOP2( (((255-in1[x]) * (in1[x]*in2[x]) ) / 65536) + | ||||
|          (in1[x] * ( 255 - ( (255 - in1[x]) * (255 - in2[x] ) / 255) )) / 255 | ||||
|          , NULL ); | ||||
| } | ||||
| 
 | ||||
| Imaging | ||||
| ImagingChopHardLight(Imaging imIn1, Imaging imIn2) | ||||
| { | ||||
|     CHOP2( (in2[x]<128) ? ( (in1[x]*in2[x])/127) | ||||
|                         : 255 - ( ((255-in2[x]) * (255-in1[x])) / 127) | ||||
|          , NULL); | ||||
| } | ||||
| 
 | ||||
| Imaging | ||||
| ImagingOverlay(Imaging imIn1, Imaging imIn2) | ||||
| { | ||||
|     CHOP2( (in1[x]<128) ? ( (in1[x]*in2[x])/127) | ||||
|                         : 255 - ( ((255-in1[x]) * (255-in2[x])) / 127) | ||||
|          , NULL); | ||||
| } | ||||
|  |  | |||
|  | @ -339,6 +339,9 @@ extern Imaging ImagingChopSubtract( | |||
|     Imaging imIn1, Imaging imIn2, float scale, int offset); | ||||
| extern Imaging ImagingChopAddModulo(Imaging imIn1, Imaging imIn2); | ||||
| extern Imaging ImagingChopSubtractModulo(Imaging imIn1, Imaging imIn2); | ||||
| extern Imaging ImagingChopSoftLight(Imaging imIn1, Imaging imIn2); | ||||
| extern Imaging ImagingChopHardLight(Imaging imIn1, Imaging imIn2); | ||||
| extern Imaging ImagingOverlay(Imaging imIn1, Imaging imIn2); | ||||
| 
 | ||||
| /* "1" images only */ | ||||
| extern Imaging ImagingChopAnd(Imaging imIn1, Imaging imIn2); | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue
	
	Block a user