mirror of
				https://github.com/python-pillow/Pillow.git
				synced 2025-10-25 13:11:24 +03:00 
			
		
		
		
	Merge remote-tracking branch 'upstream/main' into add-pyproject.toml
# Conflicts: # pyproject.toml
This commit is contained in:
		
						commit
						c068af7630
					
				|  | @ -13,10 +13,6 @@ indent_style = space | ||||||
| 
 | 
 | ||||||
| trim_trailing_whitespace = true | trim_trailing_whitespace = true | ||||||
| 
 | 
 | ||||||
| [*.rst] |  | ||||||
| # Four-space indentation |  | ||||||
| indent_size = 4 |  | ||||||
| 
 |  | ||||||
| [*.yml] | [*.yml] | ||||||
| # Two-space indentation | # Two-space indentation | ||||||
| indent_size = 2 | indent_size = 2 | ||||||
|  |  | ||||||
							
								
								
									
										1
									
								
								.github/CONTRIBUTING.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/CONTRIBUTING.md
									
									
									
									
										vendored
									
									
								
							|  | @ -19,6 +19,7 @@ Please send a pull request to the `main` branch. Please include [documentation]( | ||||||
| - Follow PEP 8. | - Follow PEP 8. | ||||||
| - When committing only documentation changes please include `[ci skip]` in the commit message to avoid running tests on AppVeyor. | - When committing only documentation changes please include `[ci skip]` in the commit message to avoid running tests on AppVeyor. | ||||||
| - Include [release notes](https://github.com/python-pillow/Pillow/tree/main/docs/releasenotes) as needed or appropriate with your bug fixes, feature additions and tests. | - Include [release notes](https://github.com/python-pillow/Pillow/tree/main/docs/releasenotes) as needed or appropriate with your bug fixes, feature additions and tests. | ||||||
|  | - Do not add to the [changelog](https://github.com/python-pillow/Pillow/blob/main/CHANGES.rst) for proposed changes, as that is updated after changes are merged. | ||||||
| 
 | 
 | ||||||
| ## Reporting Issues | ## Reporting Issues | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										2
									
								
								.github/workflows/test-cygwin.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/test-cygwin.yml
									
									
									
									
										vendored
									
									
								
							|  | @ -104,7 +104,7 @@ jobs: | ||||||
|       - name: Build |       - name: Build | ||||||
|         shell: bash.exe -eo pipefail -o igncr "{0}" |         shell: bash.exe -eo pipefail -o igncr "{0}" | ||||||
|         run: | |         run: | | ||||||
|           SETUPTOOLS_USE_DISTUTILS=stdlib .ci/build.sh |           .ci/build.sh | ||||||
| 
 | 
 | ||||||
|       - name: Test |       - name: Test | ||||||
|         run: | |         run: | | ||||||
|  |  | ||||||
							
								
								
									
										1
									
								
								.github/workflows/test-docker.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/test-docker.yml
									
									
									
									
										vendored
									
									
								
							|  | @ -39,6 +39,7 @@ jobs: | ||||||
|           centos-stream-8-amd64, |           centos-stream-8-amd64, | ||||||
|           centos-stream-9-amd64, |           centos-stream-9-amd64, | ||||||
|           debian-11-bullseye-x86, |           debian-11-bullseye-x86, | ||||||
|  |           debian-12-bookworm-x86, | ||||||
|           fedora-37-amd64, |           fedora-37-amd64, | ||||||
|           fedora-38-amd64, |           fedora-38-amd64, | ||||||
|           gentoo, |           gentoo, | ||||||
|  |  | ||||||
							
								
								
									
										2
									
								
								.github/workflows/test-mingw.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/test-mingw.yml
									
									
									
									
										vendored
									
									
								
							|  | @ -80,7 +80,7 @@ jobs: | ||||||
|           pushd depends && ./install_extra_test_images.sh && popd |           pushd depends && ./install_extra_test_images.sh && popd | ||||||
| 
 | 
 | ||||||
|       - name: Build Pillow |       - name: Build Pillow | ||||||
|         run: SETUPTOOLS_USE_DISTUTILS="stdlib" CFLAGS="-coverage" python3 -m pip install --global-option="build_ext" . |         run: SETUPTOOLS_USE_DISTUTILS="stdlib" CFLAGS="-coverage" python3 -m pip install . | ||||||
| 
 | 
 | ||||||
|       - name: Test Pillow |       - name: Test Pillow | ||||||
|         run: | |         run: | | ||||||
|  |  | ||||||
							
								
								
									
										4
									
								
								.github/workflows/test-windows.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/test-windows.yml
									
									
									
									
										vendored
									
									
								
							|  | @ -28,10 +28,10 @@ jobs: | ||||||
|         architecture: ["x86", "x64"] |         architecture: ["x86", "x64"] | ||||||
|         include: |         include: | ||||||
|           # PyPy 7.3.4+ only ships 64-bit binaries for Windows |           # PyPy 7.3.4+ only ships 64-bit binaries for Windows | ||||||
|           - python-version: "pypy3.8" |  | ||||||
|             architecture: "x64" |  | ||||||
|           - python-version: "pypy3.9" |           - python-version: "pypy3.9" | ||||||
|             architecture: "x64" |             architecture: "x64" | ||||||
|  |           - python-version: "pypy3.10" | ||||||
|  |             architecture: "x64" | ||||||
| 
 | 
 | ||||||
|     timeout-minutes: 30 |     timeout-minutes: 30 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										2
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
								
							|  | @ -29,8 +29,8 @@ jobs: | ||||||
|           "ubuntu-latest", |           "ubuntu-latest", | ||||||
|         ] |         ] | ||||||
|         python-version: [ |         python-version: [ | ||||||
|  |           "pypy3.10", | ||||||
|           "pypy3.9", |           "pypy3.9", | ||||||
|           "pypy3.8", |  | ||||||
|           "3.12-dev", |           "3.12-dev", | ||||||
|           "3.11", |           "3.11", | ||||||
|           "3.10", |           "3.10", | ||||||
|  |  | ||||||
|  | @ -4,9 +4,6 @@ repos: | ||||||
|     hooks: |     hooks: | ||||||
|       - id: black |       - id: black | ||||||
|         args: [--target-version=py38] |         args: [--target-version=py38] | ||||||
|         # Only .py files, until https://github.com/psf/black/issues/402 resolved |  | ||||||
|         files: \.py$ |  | ||||||
|         types: [] |  | ||||||
| 
 | 
 | ||||||
|   - repo: https://github.com/PyCQA/isort |   - repo: https://github.com/PyCQA/isort | ||||||
|     rev: 5.12.0 |     rev: 5.12.0 | ||||||
|  |  | ||||||
							
								
								
									
										36
									
								
								CHANGES.rst
									
									
									
									
									
								
							
							
						
						
									
										36
									
								
								CHANGES.rst
									
									
									
									
									
								
							|  | @ -5,6 +5,42 @@ Changelog (Pillow) | ||||||
| 10.0.0 (unreleased) | 10.0.0 (unreleased) | ||||||
| ------------------- | ------------------- | ||||||
| 
 | 
 | ||||||
|  | - Fixed finding dependencies on Cygwin #7175 | ||||||
|  |   [radarhere] | ||||||
|  | 
 | ||||||
|  | - Changed grabclipboard() to use PNG instead of JPG compression on macOS #7219 | ||||||
|  |   [abey79, radarhere] | ||||||
|  | 
 | ||||||
|  | - Added in_place argument to ImageOps.exif_transpose() #7092 | ||||||
|  |   [radarhere] | ||||||
|  | 
 | ||||||
|  | - Fixed calling putpalette() on L and LA images before load() #7187 | ||||||
|  |   [radarhere] | ||||||
|  | 
 | ||||||
|  | - Fixed saving TIFF multiframe images with LONG8 tag types #7078 | ||||||
|  |   [radarhere] | ||||||
|  | 
 | ||||||
|  | - Fixed combining single duration across duplicate APNG frames #7146 | ||||||
|  |   [radarhere] | ||||||
|  | 
 | ||||||
|  | - Remove temporary file when error is raised #7148 | ||||||
|  |   [radarhere] | ||||||
|  | 
 | ||||||
|  | - Do not use temporary file when grabbing clipboard on Linux #7200 | ||||||
|  |   [radarhere] | ||||||
|  | 
 | ||||||
|  | - If the clipboard fails to open on Windows, wait and try again #7141 | ||||||
|  |   [radarhere] | ||||||
|  | 
 | ||||||
|  | - Fixed saving multiple 1 mode frames to GIF #7181 | ||||||
|  |   [radarhere] | ||||||
|  | 
 | ||||||
|  | - Replaced absolute PIL import with relative import #7173 | ||||||
|  |   [radarhere] | ||||||
|  | 
 | ||||||
|  | - Replaced deprecated Py_FileSystemDefaultEncoding for Python >= 3.12 #7192 | ||||||
|  |   [radarhere] | ||||||
|  | 
 | ||||||
| - Improved wl-paste mimetype handling in ImageGrab #7094 | - Improved wl-paste mimetype handling in ImageGrab #7094 | ||||||
|   [rrcgat, radarhere] |   [rrcgat, radarhere] | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -15,6 +15,7 @@ graft src | ||||||
| graft depends | graft depends | ||||||
| graft winbuild | graft winbuild | ||||||
| graft docs | graft docs | ||||||
|  | graft _custom_build | ||||||
| 
 | 
 | ||||||
| # build/src control detritus | # build/src control detritus | ||||||
| exclude .appveyor.yml | exclude .appveyor.yml | ||||||
|  |  | ||||||
							
								
								
									
										9
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										9
									
								
								Makefile
									
									
									
									
									
								
							|  | @ -46,7 +46,6 @@ help: | ||||||
| 	@echo "  docserve           run an HTTP server on the docs directory" | 	@echo "  docserve           run an HTTP server on the docs directory" | ||||||
| 	@echo "  html               make HTML docs" | 	@echo "  html               make HTML docs" | ||||||
| 	@echo "  htmlview           open the index page built by the html target in your browser" | 	@echo "  htmlview           open the index page built by the html target in your browser" | ||||||
| 	@echo "  inplace            make inplace extension" |  | ||||||
| 	@echo "  install            make and install" | 	@echo "  install            make and install" | ||||||
| 	@echo "  install-coverage   make and install with C coverage" | 	@echo "  install-coverage   make and install with C coverage" | ||||||
| 	@echo "  lint               run the lint checks" | 	@echo "  lint               run the lint checks" | ||||||
|  | @ -54,10 +53,6 @@ help: | ||||||
| 	@echo "  release-test       run code and package tests before release" | 	@echo "  release-test       run code and package tests before release" | ||||||
| 	@echo "  test               run tests on installed Pillow" | 	@echo "  test               run tests on installed Pillow" | ||||||
| 
 | 
 | ||||||
| .PHONY: inplace |  | ||||||
| inplace: clean |  | ||||||
| 	python3 -m pip install -e --global-option="build_ext" --global-option="--inplace" . |  | ||||||
| 
 |  | ||||||
| .PHONY: install | .PHONY: install | ||||||
| install: | install: | ||||||
| 	python3 -m pip -v install . | 	python3 -m pip -v install . | ||||||
|  | @ -65,7 +60,7 @@ install: | ||||||
| 
 | 
 | ||||||
| .PHONY: install-coverage | .PHONY: install-coverage | ||||||
| install-coverage: | install-coverage: | ||||||
| 	CFLAGS="-coverage -Werror=implicit-function-declaration" python3 -m pip -v install --global-option="build_ext" . | 	CFLAGS="-coverage -Werror=implicit-function-declaration" python3 -m pip -v install . | ||||||
| 	python3 selftest.py | 	python3 selftest.py | ||||||
| 
 | 
 | ||||||
| .PHONY: debug | .PHONY: debug | ||||||
|  | @ -74,7 +69,7 @@ debug: | ||||||
| # for our stuff, kills optimization, and redirects to dev null so we
 | # for our stuff, kills optimization, and redirects to dev null so we
 | ||||||
| # see any build failures.
 | # see any build failures.
 | ||||||
| 	make clean > /dev/null | 	make clean > /dev/null | ||||||
| 	CFLAGS='-g -O0' python3 -m pip -v install --global-option="build_ext" . > /dev/null | 	CFLAGS='-g -O0' python3 -m pip -v install . > /dev/null | ||||||
| 
 | 
 | ||||||
| .PHONY: release-test | .PHONY: release-test | ||||||
| release-test: | release-test: | ||||||
|  |  | ||||||
|  | @ -27,17 +27,11 @@ def timer(func, label, *args): | ||||||
|     for x in range(iterations): |     for x in range(iterations): | ||||||
|         func(*args) |         func(*args) | ||||||
|         if time.time() - starttime > 10: |         if time.time() - starttime > 10: | ||||||
|             print( |  | ||||||
|                 "{}: breaking at {} iterations, {:.6f} per iteration".format( |  | ||||||
|                     label, x + 1, (time.time() - starttime) / (x + 1.0) |  | ||||||
|                 ) |  | ||||||
|             ) |  | ||||||
|             break |             break | ||||||
|     if x == iterations - 1: |  | ||||||
|     endtime = time.time() |     endtime = time.time() | ||||||
|     print( |     print( | ||||||
|             "{}: {:.4f} s  {:.6f} per iteration".format( |         "{}: completed {} iterations in {:.4f}s, {:.6f}s per iteration".format( | ||||||
|                 label, endtime - starttime, (endtime - starttime) / (x + 1.0) |             label, x + 1, endtime - starttime, (endtime - starttime) / (x + 1.0) | ||||||
|         ) |         ) | ||||||
|     ) |     ) | ||||||
| 
 | 
 | ||||||
|  | @ -45,7 +39,7 @@ def timer(func, label, *args): | ||||||
| def test_direct(): | def test_direct(): | ||||||
|     im = hopper() |     im = hopper() | ||||||
|     im.load() |     im.load() | ||||||
|     # im = Image.new( "RGB", (2000, 2000), (1, 3, 2)) |     # im = Image.new("RGB", (2000, 2000), (1, 3, 2)) | ||||||
|     caccess = im.im.pixel_access(False) |     caccess = im.im.pixel_access(False) | ||||||
|     access = PyAccess.new(im, False) |     access = PyAccess.new(im, False) | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										
											BIN
										
									
								
								Tests/fonts/oom-4da0210eb7081b0bf15bf16cc4c52ce02c1e1bbc.ttf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								Tests/fonts/oom-4da0210eb7081b0bf15bf16cc4c52ce02c1e1bbc.ttf
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								Tests/images/orientation_rectangle.jpg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								Tests/images/orientation_rectangle.jpg
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 669 B | 
|  | @ -447,6 +447,17 @@ def test_apng_save_duration_loop(tmp_path): | ||||||
|         assert im.info.get("duration") == 750 |         assert im.info.get("duration") == 750 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | def test_apng_save_duplicate_duration(tmp_path): | ||||||
|  |     test_file = str(tmp_path / "temp.png") | ||||||
|  |     frame = Image.new("RGB", (1, 1)) | ||||||
|  | 
 | ||||||
|  |     # Test a single duration is correctly combined across duplicate frames | ||||||
|  |     frame.save(test_file, save_all=True, append_images=[frame, frame], duration=500) | ||||||
|  |     with Image.open(test_file) as im: | ||||||
|  |         assert im.n_frames == 1 | ||||||
|  |         assert im.info.get("duration") == 1500 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| def test_apng_save_disposal(tmp_path): | def test_apng_save_disposal(tmp_path): | ||||||
|     test_file = str(tmp_path / "temp.png") |     test_file = str(tmp_path / "temp.png") | ||||||
|     size = (128, 64) |     size = (128, 64) | ||||||
|  |  | ||||||
|  | @ -252,6 +252,19 @@ def test_roundtrip_save_all(tmp_path): | ||||||
|         assert reread.n_frames == 5 |         assert reread.n_frames == 5 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | def test_roundtrip_save_all_1(tmp_path): | ||||||
|  |     out = str(tmp_path / "temp.gif") | ||||||
|  |     im = Image.new("1", (1, 1)) | ||||||
|  |     im2 = Image.new("1", (1, 1), 1) | ||||||
|  |     im.save(out, save_all=True, append_images=[im2]) | ||||||
|  | 
 | ||||||
|  |     with Image.open(out) as reloaded: | ||||||
|  |         assert reloaded.getpixel((0, 0)) == 0 | ||||||
|  | 
 | ||||||
|  |         reloaded.seek(1) | ||||||
|  |         assert reloaded.getpixel((0, 0)) == 255 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| @pytest.mark.parametrize( | @pytest.mark.parametrize( | ||||||
|     "path, mode", |     "path, mode", | ||||||
|     ( |     ( | ||||||
|  |  | ||||||
|  | @ -96,10 +96,17 @@ class TestFileTiff: | ||||||
| 
 | 
 | ||||||
|             assert_image_similar_tofile(im, "Tests/images/pil136.png", 1) |             assert_image_similar_tofile(im, "Tests/images/pil136.png", 1) | ||||||
| 
 | 
 | ||||||
|     def test_bigtiff(self): |     def test_bigtiff(self, tmp_path): | ||||||
|         with Image.open("Tests/images/hopper_bigtiff.tif") as im: |         with Image.open("Tests/images/hopper_bigtiff.tif") as im: | ||||||
|             assert_image_equal_tofile(im, "Tests/images/hopper.tif") |             assert_image_equal_tofile(im, "Tests/images/hopper.tif") | ||||||
| 
 | 
 | ||||||
|  |         with Image.open("Tests/images/hopper_bigtiff.tif") as im: | ||||||
|  |             # multistrip support not yet implemented | ||||||
|  |             del im.tag_v2[273] | ||||||
|  | 
 | ||||||
|  |             outfile = str(tmp_path / "temp.tif") | ||||||
|  |             im.save(outfile, save_all=True, append_images=[im], tiffinfo=im.tag_v2) | ||||||
|  | 
 | ||||||
|     def test_set_legacy_api(self): |     def test_set_legacy_api(self): | ||||||
|         ifd = TiffImagePlugin.ImageFileDirectory_v2() |         ifd = TiffImagePlugin.ImageFileDirectory_v2() | ||||||
|         with pytest.raises(Exception) as e: |         with pytest.raises(Exception) as e: | ||||||
|  |  | ||||||
|  | @ -32,6 +32,14 @@ def test_putpalette(): | ||||||
|     with pytest.raises(ValueError): |     with pytest.raises(ValueError): | ||||||
|         palette("YCbCr") |         palette("YCbCr") | ||||||
| 
 | 
 | ||||||
|  |     with Image.open("Tests/images/hopper_gray.jpg") as im: | ||||||
|  |         assert im.mode == "L" | ||||||
|  |         im.putpalette(list(range(256)) * 3) | ||||||
|  | 
 | ||||||
|  |     with Image.open("Tests/images/la.tga") as im: | ||||||
|  |         assert im.mode == "LA" | ||||||
|  |         im.putpalette(list(range(256)) * 3) | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| def test_imagepalette(): | def test_imagepalette(): | ||||||
|     im = hopper("P") |     im = hopper("P") | ||||||
|  |  | ||||||
|  | @ -463,6 +463,11 @@ def test_default_font(): | ||||||
|     assert_image_equal_tofile(im, "Tests/images/default_font.png") |     assert_image_equal_tofile(im, "Tests/images/default_font.png") | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @pytest.mark.parametrize("mode", (None, "1", "RGBA")) | ||||||
|  | def test_getbbox(font, mode): | ||||||
|  |     assert (0, 4, 12, 16) == font.getbbox("A", mode) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| def test_getbbox_empty(font): | def test_getbbox_empty(font): | ||||||
|     # issue #2614, should not crash. |     # issue #2614, should not crash. | ||||||
|     assert (0, 0, 0, 0) == font.getbbox("") |     assert (0, 0, 0, 0) == font.getbbox("") | ||||||
|  | @ -1037,6 +1042,7 @@ def test_render_mono_size(): | ||||||
|     "test_file", |     "test_file", | ||||||
|     [ |     [ | ||||||
|         "Tests/fonts/oom-e8e927ba6c0d38274a37c1567560eb33baf74627.ttf", |         "Tests/fonts/oom-e8e927ba6c0d38274a37c1567560eb33baf74627.ttf", | ||||||
|  |         "Tests/fonts/oom-4da0210eb7081b0bf15bf16cc4c52ce02c1e1bbc.ttf", | ||||||
|     ], |     ], | ||||||
| ) | ) | ||||||
| def test_oom(test_file): | def test_oom(test_file): | ||||||
|  |  | ||||||
|  | @ -404,6 +404,18 @@ def test_exif_transpose(): | ||||||
|     assert 0x0112 not in transposed_im.getexif() |     assert 0x0112 not in transposed_im.getexif() | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | def test_exif_transpose_in_place(): | ||||||
|  |     with Image.open("Tests/images/orientation_rectangle.jpg") as im: | ||||||
|  |         assert im.size == (2, 1) | ||||||
|  |         assert im.getexif()[0x0112] == 8 | ||||||
|  |         expected = im.rotate(90, expand=True) | ||||||
|  | 
 | ||||||
|  |         ImageOps.exif_transpose(im, in_place=True) | ||||||
|  |         assert im.size == (1, 2) | ||||||
|  |         assert 0x0112 not in im.getexif() | ||||||
|  |         assert_image_equal(im, expected) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| def test_autocontrast_cutoff(): | def test_autocontrast_cutoff(): | ||||||
|     # Test the cutoff argument of autocontrast |     # Test the cutoff argument of autocontrast | ||||||
|     with Image.open("Tests/images/bw_gradient.png") as img: |     with Image.open("Tests/images/bw_gradient.png") as img: | ||||||
|  |  | ||||||
							
								
								
									
										56
									
								
								_custom_build/backend.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										56
									
								
								_custom_build/backend.py
									
									
									
									
									
										Executable file
									
								
							|  | @ -0,0 +1,56 @@ | ||||||
|  | import sys | ||||||
|  | 
 | ||||||
|  | from setuptools.build_meta import *  # noqa: F401, F403 | ||||||
|  | from setuptools.build_meta import build_wheel | ||||||
|  | 
 | ||||||
|  | backend_class = build_wheel.__self__.__class__ | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class _CustomBuildMetaBackend(backend_class): | ||||||
|  |     def run_setup(self, setup_script="setup.py"): | ||||||
|  |         if self.config_settings: | ||||||
|  | 
 | ||||||
|  |             def config_has(key, value): | ||||||
|  |                 settings = self.config_settings.get(key) | ||||||
|  |                 if settings: | ||||||
|  |                     if not isinstance(settings, list): | ||||||
|  |                         settings = [settings] | ||||||
|  |                     return value in settings | ||||||
|  | 
 | ||||||
|  |             flags = [] | ||||||
|  |             for dependency in ( | ||||||
|  |                 "zlib", | ||||||
|  |                 "jpeg", | ||||||
|  |                 "tiff", | ||||||
|  |                 "freetype", | ||||||
|  |                 "raqm", | ||||||
|  |                 "lcms", | ||||||
|  |                 "webp", | ||||||
|  |                 "webpmux", | ||||||
|  |                 "jpeg2000", | ||||||
|  |                 "imagequant", | ||||||
|  |                 "xcb", | ||||||
|  |             ): | ||||||
|  |                 if config_has(dependency, "enable"): | ||||||
|  |                     flags.append("--enable-" + dependency) | ||||||
|  |                 elif config_has(dependency, "disable"): | ||||||
|  |                     flags.append("--disable-" + dependency) | ||||||
|  |             for dependency in ("raqm", "fribidi"): | ||||||
|  |                 if config_has(dependency, "vendor"): | ||||||
|  |                     flags.append("--vendor-" + dependency) | ||||||
|  |             if self.config_settings.get("platform-guessing") == "disable": | ||||||
|  |                 flags.append("--disable-platform-guessing") | ||||||
|  |             if self.config_settings.get("debug") == "true": | ||||||
|  |                 flags.append("--debug") | ||||||
|  |             if flags: | ||||||
|  |                 sys.argv = sys.argv[:1] + ["build_ext"] + flags + sys.argv[1:] | ||||||
|  |         return super().run_setup(setup_script) | ||||||
|  | 
 | ||||||
|  |     def build_wheel( | ||||||
|  |         self, wheel_directory, config_settings=None, metadata_directory=None | ||||||
|  |     ): | ||||||
|  |         self.config_settings = config_settings | ||||||
|  |         return super().build_wheel(wheel_directory, config_settings, metadata_directory) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | build_wheel = _CustomBuildMetaBackend().build_wheel | ||||||
|  | @ -2,7 +2,7 @@ | ||||||
| from livereload.compiler import shell | from livereload.compiler import shell | ||||||
| from livereload.task import Task | from livereload.task import Task | ||||||
| 
 | 
 | ||||||
| Task.add('*.rst', shell('make html')) | Task.add("*.rst", shell("make html")) | ||||||
| Task.add('*/*.rst', shell('make html')) | Task.add("*/*.rst", shell("make html")) | ||||||
| Task.add('Makefile', shell('make html')) | Task.add("Makefile", shell("make html")) | ||||||
| Task.add('conf.py', shell('make html')) | Task.add("conf.py", shell("make html")) | ||||||
|  |  | ||||||
|  | @ -95,9 +95,8 @@ in the upper left corner. Note that the coordinates refer to the implied pixel | ||||||
| corners; the centre of a pixel addressed as (0, 0) actually lies at (0.5, 0.5). | corners; the centre of a pixel addressed as (0, 0) actually lies at (0.5, 0.5). | ||||||
| 
 | 
 | ||||||
| Coordinates are usually passed to the library as 2-tuples (x, y). Rectangles | Coordinates are usually passed to the library as 2-tuples (x, y). Rectangles | ||||||
| are represented as 4-tuples, with the upper left corner given first. For | are represented as 4-tuples, (x1, y1, x2, y2), with the upper left corner given | ||||||
| example, a rectangle covering all of an 800x600 pixel image is written as (0, | first. | ||||||
| 0, 800, 600). |  | ||||||
| 
 | 
 | ||||||
| Palette | Palette | ||||||
| ------- | ------- | ||||||
|  |  | ||||||
|  | @ -1380,6 +1380,12 @@ PSD | ||||||
| 
 | 
 | ||||||
| Pillow identifies and reads PSD files written by Adobe Photoshop 2.5 and 3.0. | Pillow identifies and reads PSD files written by Adobe Photoshop 2.5 and 3.0. | ||||||
| 
 | 
 | ||||||
|  | QOI | ||||||
|  | ^^^ | ||||||
|  | 
 | ||||||
|  | .. versionadded:: 9.5.0 | ||||||
|  | 
 | ||||||
|  | Pillow identifies and reads images in Quite OK Image format. | ||||||
| 
 | 
 | ||||||
| SUN | SUN | ||||||
| ^^^ | ^^^ | ||||||
|  | @ -1562,13 +1568,6 @@ The :py:meth:`~PIL.Image.Image.save` method can take the following keyword argum | ||||||
| 
 | 
 | ||||||
|     .. versionadded:: 5.3.0 |     .. versionadded:: 5.3.0 | ||||||
| 
 | 
 | ||||||
| QOI |  | ||||||
| ^^^ |  | ||||||
| 
 |  | ||||||
| .. versionadded:: 9.5.0 |  | ||||||
| 
 |  | ||||||
| Pillow identifies and reads images in Quite OK Image format. |  | ||||||
| 
 |  | ||||||
| XV Thumbnails | XV Thumbnails | ||||||
| ^^^^^^^^^^^^^ | ^^^^^^^^^^^^^ | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -312,6 +312,11 @@ Many of Pillow's features require external libraries: | ||||||
|             mingw-w64-x86_64-libimagequant \ |             mingw-w64-x86_64-libimagequant \ | ||||||
|             mingw-w64-x86_64-libraqm |             mingw-w64-x86_64-libraqm | ||||||
| 
 | 
 | ||||||
|  |     https://www.msys2.org/docs/python/ states that setuptools >= 60 does not work with | ||||||
|  |     MSYS2. To workaround this, before installing Pillow you must run:: | ||||||
|  | 
 | ||||||
|  |         export SETUPTOOLS_USE_DISTUTILS=stdlib | ||||||
|  | 
 | ||||||
| .. tab:: FreeBSD | .. tab:: FreeBSD | ||||||
| 
 | 
 | ||||||
|     .. Note:: Only FreeBSD 10 and 11 tested |     .. Note:: Only FreeBSD 10 and 11 tested | ||||||
|  | @ -380,40 +385,40 @@ Build Options | ||||||
|   using a setting of 1. By default, it uses 4 CPUs, or if 4 are not |   using a setting of 1. By default, it uses 4 CPUs, or if 4 are not | ||||||
|   available, as many as are present. |   available, as many as are present. | ||||||
| 
 | 
 | ||||||
| * Build flags: ``--disable-zlib``, ``--disable-jpeg``, | * Config settings: ``-C zlib=disable``, ``-C jpeg=disable``, | ||||||
|   ``--disable-tiff``, ``--disable-freetype``, ``--disable-raqm``, |   ``-C tiff=disable``, ``-C freetype=disable``, ``-C raqm=disable``, | ||||||
|   ``--disable-lcms``, ``--disable-webp``, ``--disable-webpmux``, |   ``-C lcms=disable``, ``-C webp=disable``, ``-C webpmux=disable``, | ||||||
|   ``--disable-jpeg2000``, ``--disable-imagequant``, ``--disable-xcb``. |   ``-C jpeg2000=disable``, ``-C imagequant=disable``, ``-C xcb=disable``. | ||||||
|   Disable building the corresponding feature even if the development |   Disable building the corresponding feature even if the development | ||||||
|   libraries are present on the building machine. |   libraries are present on the building machine. | ||||||
| 
 | 
 | ||||||
| * Build flags: ``--enable-zlib``, ``--enable-jpeg``, | * Config settings: ``-C zlib=enable``, ``-C jpeg=enable``, | ||||||
|   ``--enable-tiff``, ``--enable-freetype``, ``--enable-raqm``, |   ``-C tiff=enable``, ``-C freetype=enable``, ``-C raqm=enable``, | ||||||
|   ``--enable-lcms``, ``--enable-webp``, ``--enable-webpmux``, |   ``-C lcms=enable``, ``-C webp=enable``, ``-C webpmux=enable``, | ||||||
|   ``--enable-jpeg2000``, ``--enable-imagequant``, ``--enable-xcb``. |   ``-C jpeg2000=enable``, ``-C imagequant=enable``, ``-C xcb=enable``. | ||||||
|   Require that the corresponding feature is built. The build will raise |   Require that the corresponding feature is built. The build will raise | ||||||
|   an exception if the libraries are not found. Webpmux (WebP metadata) |   an exception if the libraries are not found. Webpmux (WebP metadata) | ||||||
|   relies on WebP support. Tcl and Tk also must be used together. |   relies on WebP support. Tcl and Tk also must be used together. | ||||||
| 
 | 
 | ||||||
| * Build flags: ``--vendor-raqm``, ``--vendor-fribidi``. | * Config settings: ``-C raqm=vendor``, ``-C fribidi=vendor``. | ||||||
|   These flags are used to compile a modified version of libraqm and |   These flags are used to compile a modified version of libraqm and | ||||||
|   a shim that dynamically loads libfribidi at runtime. These are |   a shim that dynamically loads libfribidi at runtime. These are | ||||||
|   used to compile the standard Pillow wheels. Compiling libraqm requires |   used to compile the standard Pillow wheels. Compiling libraqm requires | ||||||
|   a C99-compliant compiler. |   a C99-compliant compiler. | ||||||
| 
 | 
 | ||||||
| * Build flag: ``--disable-platform-guessing``. Skips all of the | * Build flag: ``-C platform-guessing=disable``. Skips all of the | ||||||
|   platform dependent guessing of include and library directories for |   platform dependent guessing of include and library directories for | ||||||
|   automated build systems that configure the proper paths in the |   automated build systems that configure the proper paths in the | ||||||
|   environment variables (e.g. Buildroot). |   environment variables (e.g. Buildroot). | ||||||
| 
 | 
 | ||||||
| * Build flag: ``--debug``. Adds a debugging flag to the include and | * Build flag: ``-C debug=true``. Adds a debugging flag to the include and | ||||||
|   library search process to dump all paths searched for and found to |   library search process to dump all paths searched for and found to | ||||||
|   stdout. |   stdout. | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| Sample usage:: | Sample usage:: | ||||||
| 
 | 
 | ||||||
|     python3 -m pip install --upgrade Pillow --global-option="build_ext" --global-option="--enable-[feature]" |     python3 -m pip install --upgrade Pillow -C [feature]=enable | ||||||
| 
 | 
 | ||||||
| Platform Support | Platform Support | ||||||
| ---------------- | ---------------- | ||||||
|  | @ -448,6 +453,8 @@ These platforms are built and tested for every change. | ||||||
| +----------------------------------+----------------------------+---------------------+ | +----------------------------------+----------------------------+---------------------+ | ||||||
| | Debian 11 Bullseye               | 3.9                        | x86                 | | | Debian 11 Bullseye               | 3.9                        | x86                 | | ||||||
| +----------------------------------+----------------------------+---------------------+ | +----------------------------------+----------------------------+---------------------+ | ||||||
|  | | Debian 12 Bookworm               | 3.11                       | x86                 | | ||||||
|  | +----------------------------------+----------------------------+---------------------+ | ||||||
| | Fedora 37                        | 3.11                       | x86-64              | | | Fedora 37                        | 3.11                       | x86-64              | | ||||||
| +----------------------------------+----------------------------+---------------------+ | +----------------------------------+----------------------------+---------------------+ | ||||||
| | Fedora 38                        | 3.11                       | x86-64              | | | Fedora 38                        | 3.11                       | x86-64              | | ||||||
|  |  | ||||||
|  | @ -328,7 +328,7 @@ Methods | ||||||
| 
 | 
 | ||||||
|         .. versionadded:: 5.3.0 |         .. versionadded:: 5.3.0 | ||||||
| 
 | 
 | ||||||
| .. py:method:: ImageDraw.rounded_rectangle(xy, radius=0, fill=None, outline=None, width=1) | .. py:method:: ImageDraw.rounded_rectangle(xy, radius=0, fill=None, outline=None, width=1, corners=None) | ||||||
| 
 | 
 | ||||||
|     Draws a rounded rectangle. |     Draws a rounded rectangle. | ||||||
| 
 | 
 | ||||||
|  | @ -341,6 +341,7 @@ Methods | ||||||
|     :param width: The line width, in pixels. |     :param width: The line width, in pixels. | ||||||
|     :param corners: A tuple of whether to round each corner, |     :param corners: A tuple of whether to round each corner, | ||||||
|                     ``(top_left, top_right, bottom_right, bottom_left)``. |                     ``(top_left, top_right, bottom_right, bottom_left)``. | ||||||
|  |                     Keyword-only argument. | ||||||
| 
 | 
 | ||||||
|     .. versionadded:: 8.2.0 |     .. versionadded:: 8.2.0 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,5 +1,4 @@ | ||||||
| [build-system] | [build-system] | ||||||
| build-backend = "setuptools.build_meta" | requires = ["setuptools >= 67.8", "wheel"] | ||||||
| requires = [ | build-backend = "backend" | ||||||
|   "setuptools>=67.8", | backend-path = ["_custom_build"] | ||||||
| ] |  | ||||||
|  |  | ||||||
							
								
								
									
										1
									
								
								setup.py
									
									
									
									
									
								
							
							
						
						
									
										1
									
								
								setup.py
									
									
									
									
									
								
							|  | @ -515,6 +515,7 @@ class pil_build_ext(build_ext): | ||||||
| 
 | 
 | ||||||
|         elif sys.platform == "cygwin": |         elif sys.platform == "cygwin": | ||||||
|             # pythonX.Y.dll.a is in the /usr/lib/pythonX.Y/config directory |             # pythonX.Y.dll.a is in the /usr/lib/pythonX.Y/config directory | ||||||
|  |             self.compiler.shared_lib_extension = ".dll.a" | ||||||
|             _add_directory( |             _add_directory( | ||||||
|                 library_dirs, |                 library_dirs, | ||||||
|                 os.path.join( |                 os.path.join( | ||||||
|  |  | ||||||
|  | @ -134,6 +134,13 @@ def Ghostscript(tile, size, fp, scale=1, transparency=False): | ||||||
| 
 | 
 | ||||||
|     if gs_windows_binary is not None: |     if gs_windows_binary is not None: | ||||||
|         if not gs_windows_binary: |         if not gs_windows_binary: | ||||||
|  |             try: | ||||||
|  |                 os.unlink(outfile) | ||||||
|  |                 if infile_temp: | ||||||
|  |                     os.unlink(infile_temp) | ||||||
|  |             except OSError: | ||||||
|  |                 pass | ||||||
|  | 
 | ||||||
|             msg = "Unable to locate Ghostscript on paths" |             msg = "Unable to locate Ghostscript on paths" | ||||||
|             raise OSError(msg) |             raise OSError(msg) | ||||||
|         command[0] = gs_windows_binary |         command[0] = gs_windows_binary | ||||||
|  | @ -354,7 +361,6 @@ class EpsImageFile(ImageFile.ImageFile): | ||||||
|         check_required_header_comments() |         check_required_header_comments() | ||||||
| 
 | 
 | ||||||
|         if not self._size: |         if not self._size: | ||||||
|             self._size = 1, 1  # errors if this isn't set. why (1,1)? |  | ||||||
|             msg = "cannot determine EPS bounding box" |             msg = "cannot determine EPS bounding box" | ||||||
|             raise OSError(msg) |             raise OSError(msg) | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -47,7 +47,7 @@ class GdImageFile(ImageFile.ImageFile): | ||||||
|         # Header |         # Header | ||||||
|         s = self.fp.read(1037) |         s = self.fp.read(1037) | ||||||
| 
 | 
 | ||||||
|         if not i16(s) in [65534, 65535]: |         if i16(s) not in [65534, 65535]: | ||||||
|             msg = "Not a valid GD 2.x .gd file" |             msg = "Not a valid GD 2.x .gd file" | ||||||
|             raise SyntaxError(msg) |             raise SyntaxError(msg) | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -879,7 +879,7 @@ def _get_palette_bytes(im): | ||||||
|     :param im: Image object |     :param im: Image object | ||||||
|     :returns: Bytes, len<=768 suitable for inclusion in gif header |     :returns: Bytes, len<=768 suitable for inclusion in gif header | ||||||
|     """ |     """ | ||||||
|     return im.palette.palette |     return im.palette.palette if im.palette else b"" | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def _get_background(im, info_background): | def _get_background(im, info_background): | ||||||
|  |  | ||||||
|  | @ -22,11 +22,11 @@ import os | ||||||
| import struct | import struct | ||||||
| import sys | import sys | ||||||
| 
 | 
 | ||||||
| from PIL import Image, ImageFile, PngImagePlugin, features | from . import Image, ImageFile, PngImagePlugin, features | ||||||
| 
 | 
 | ||||||
| enable_jpeg2k = features.check_codec("jpg_2000") | enable_jpeg2k = features.check_codec("jpg_2000") | ||||||
| if enable_jpeg2k: | if enable_jpeg2k: | ||||||
|     from PIL import Jpeg2KImagePlugin |     from . import Jpeg2KImagePlugin | ||||||
| 
 | 
 | ||||||
| MAGIC = b"icns" | MAGIC = b"icns" | ||||||
| HEADERSIZE = 8 | HEADERSIZE = 8 | ||||||
|  |  | ||||||
|  | @ -1254,7 +1254,7 @@ class Image: | ||||||
|         if ymargin is None: |         if ymargin is None: | ||||||
|             ymargin = xmargin |             ymargin = xmargin | ||||||
|         self.load() |         self.load() | ||||||
|         return self._new(self.im.expand(xmargin, ymargin, 0)) |         return self._new(self.im.expand(xmargin, ymargin)) | ||||||
| 
 | 
 | ||||||
|     def filter(self, filter): |     def filter(self, filter): | ||||||
|         """ |         """ | ||||||
|  | @ -1433,12 +1433,12 @@ class Image: | ||||||
|             self._exif.load(exif_info) |             self._exif.load(exif_info) | ||||||
| 
 | 
 | ||||||
|         # XMP tags |         # XMP tags | ||||||
|         if 0x0112 not in self._exif: |         if ExifTags.Base.Orientation not in self._exif: | ||||||
|             xmp_tags = self.info.get("XML:com.adobe.xmp") |             xmp_tags = self.info.get("XML:com.adobe.xmp") | ||||||
|             if xmp_tags: |             if xmp_tags: | ||||||
|                 match = re.search(r'tiff:Orientation(="|>)([0-9])', xmp_tags) |                 match = re.search(r'tiff:Orientation(="|>)([0-9])', xmp_tags) | ||||||
|                 if match: |                 if match: | ||||||
|                     self._exif[0x0112] = int(match[2]) |                     self._exif[ExifTags.Base.Orientation] = int(match[2]) | ||||||
| 
 | 
 | ||||||
|         return self._exif |         return self._exif | ||||||
| 
 | 
 | ||||||
|  | @ -1731,7 +1731,7 @@ class Image: | ||||||
|         if not isinstance(dest, (list, tuple)): |         if not isinstance(dest, (list, tuple)): | ||||||
|             msg = "Destination must be a tuple" |             msg = "Destination must be a tuple" | ||||||
|             raise ValueError(msg) |             raise ValueError(msg) | ||||||
|         if not len(source) in (2, 4): |         if len(source) not in (2, 4): | ||||||
|             msg = "Source must be a 2 or 4-tuple" |             msg = "Source must be a 2 or 4-tuple" | ||||||
|             raise ValueError(msg) |             raise ValueError(msg) | ||||||
|         if not len(dest) == 2: |         if not len(dest) == 2: | ||||||
|  | @ -2451,8 +2451,8 @@ class Image: | ||||||
|         The image is first saved to a temporary file. By default, it will be in |         The image is first saved to a temporary file. By default, it will be in | ||||||
|         PNG format. |         PNG format. | ||||||
| 
 | 
 | ||||||
|         On Unix, the image is then opened using the **display**, **eog** or |         On Unix, the image is then opened using the **xdg-open**, **display**, | ||||||
|         **xv** utility, depending on which one can be found. |         **gm**, **eog** or **xv** utility, depending on which one can be found. | ||||||
| 
 | 
 | ||||||
|         On macOS, the image is opened with the native Preview application. |         On macOS, the image is opened with the native Preview application. | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -18,10 +18,10 @@ | ||||||
| import sys | import sys | ||||||
| from enum import IntEnum | from enum import IntEnum | ||||||
| 
 | 
 | ||||||
| from PIL import Image | from . import Image | ||||||
| 
 | 
 | ||||||
| try: | try: | ||||||
|     from PIL import _imagingcms |     from . import _imagingcms | ||||||
| except ImportError as ex: | except ImportError as ex: | ||||||
|     # Allow error import for doc purposes, but error out when accessing |     # Allow error import for doc purposes, but error out when accessing | ||||||
|     # anything in core. |     # anything in core. | ||||||
|  | @ -271,7 +271,7 @@ def get_display_profile(handle=None): | ||||||
|     if sys.platform != "win32": |     if sys.platform != "win32": | ||||||
|         return None |         return None | ||||||
| 
 | 
 | ||||||
|     from PIL import ImageWin |     from . import ImageWin | ||||||
| 
 | 
 | ||||||
|     if isinstance(handle, ImageWin.HDC): |     if isinstance(handle, ImageWin.HDC): | ||||||
|         profile = core.get_display_profile_win32(handle, 1) |         profile = core.get_display_profile_win32(handle, 1) | ||||||
|  |  | ||||||
|  | @ -43,7 +43,8 @@ class Kernel(BuiltinFilter): | ||||||
| 
 | 
 | ||||||
|     :param size: Kernel size, given as (width, height). In the current |     :param size: Kernel size, given as (width, height). In the current | ||||||
|                     version, this must be (3,3) or (5,5). |                     version, this must be (3,3) or (5,5). | ||||||
|     :param kernel: A sequence containing kernel weights. |     :param kernel: A sequence containing kernel weights. The kernel will | ||||||
|  |                    be flipped vertically before being applied to the image. | ||||||
|     :param scale: Scale factor. If given, the result for each pixel is |     :param scale: Scale factor. If given, the result for each pixel is | ||||||
|                     divided by this value. The default is the sum of the |                     divided by this value. The default is the sum of the | ||||||
|                     kernel weights. |                     kernel weights. | ||||||
|  |  | ||||||
|  | @ -26,7 +26,6 @@ | ||||||
| # | # | ||||||
| 
 | 
 | ||||||
| import base64 | import base64 | ||||||
| import math |  | ||||||
| import os | import os | ||||||
| import sys | import sys | ||||||
| import warnings | import warnings | ||||||
|  | @ -226,10 +225,6 @@ class FreeTypeFont: | ||||||
|         path, size, index, encoding, layout_engine = state |         path, size, index, encoding, layout_engine = state | ||||||
|         self.__init__(path, size, index, encoding, layout_engine) |         self.__init__(path, size, index, encoding, layout_engine) | ||||||
| 
 | 
 | ||||||
|     def _multiline_split(self, text): |  | ||||||
|         split_character = "\n" if isinstance(text, str) else b"\n" |  | ||||||
|         return text.split(split_character) |  | ||||||
| 
 |  | ||||||
|     def getname(self): |     def getname(self): | ||||||
|         """ |         """ | ||||||
|         :return: A tuple of the font family (e.g. Helvetica) and the font style |         :return: A tuple of the font family (e.g. Helvetica) and the font style | ||||||
|  | @ -551,28 +546,23 @@ class FreeTypeFont: | ||||||
|                  :py:mod:`PIL.Image.core` interface module, and the text offset, the |                  :py:mod:`PIL.Image.core` interface module, and the text offset, the | ||||||
|                  gap between the starting coordinate and the first marking |                  gap between the starting coordinate and the first marking | ||||||
|         """ |         """ | ||||||
|         size, offset = self.font.getsize( |  | ||||||
|             text, mode, direction, features, language, anchor |  | ||||||
|         ) |  | ||||||
|         if start is None: |         if start is None: | ||||||
|             start = (0, 0) |             start = (0, 0) | ||||||
|         size = tuple(math.ceil(size[i] + stroke_width * 2 + start[i]) for i in range(2)) |         im, size, offset = self.font.render( | ||||||
|         offset = offset[0] - stroke_width, offset[1] - stroke_width |  | ||||||
|         Image._decompression_bomb_check(size) |  | ||||||
|         im = Image.core.fill("RGBA" if mode == "RGBA" else "L", size, 0) |  | ||||||
|         if min(size): |  | ||||||
|             self.font.render( |  | ||||||
|             text, |             text, | ||||||
|                 im.id, |             Image.core.fill, | ||||||
|             mode, |             mode, | ||||||
|             direction, |             direction, | ||||||
|             features, |             features, | ||||||
|             language, |             language, | ||||||
|             stroke_width, |             stroke_width, | ||||||
|  |             anchor, | ||||||
|             ink, |             ink, | ||||||
|             start[0], |             start[0], | ||||||
|             start[1], |             start[1], | ||||||
|  |             Image.MAX_IMAGE_PIXELS, | ||||||
|         ) |         ) | ||||||
|  |         Image._decompression_bomb_check(size) | ||||||
|         return im, offset |         return im, offset | ||||||
| 
 | 
 | ||||||
|     def font_variant( |     def font_variant( | ||||||
|  |  | ||||||
|  | @ -15,6 +15,7 @@ | ||||||
| # See the README file for information on usage and redistribution. | # See the README file for information on usage and redistribution. | ||||||
| # | # | ||||||
| 
 | 
 | ||||||
|  | import io | ||||||
| import os | import os | ||||||
| import shutil | import shutil | ||||||
| import subprocess | import subprocess | ||||||
|  | @ -94,14 +95,14 @@ def grab(bbox=None, include_layered_windows=False, all_screens=False, xdisplay=N | ||||||
| 
 | 
 | ||||||
| def grabclipboard(): | def grabclipboard(): | ||||||
|     if sys.platform == "darwin": |     if sys.platform == "darwin": | ||||||
|         fh, filepath = tempfile.mkstemp(".jpg") |         fh, filepath = tempfile.mkstemp(".png") | ||||||
|         os.close(fh) |         os.close(fh) | ||||||
|         commands = [ |         commands = [ | ||||||
|             'set theFile to (open for access POSIX file "' |             'set theFile to (open for access POSIX file "' | ||||||
|             + filepath |             + filepath | ||||||
|             + '" with write permission)', |             + '" with write permission)', | ||||||
|             "try", |             "try", | ||||||
|             "    write (the clipboard as JPEG picture) to theFile", |             "    write (the clipboard as «class PNGf») to theFile", | ||||||
|             "end try", |             "end try", | ||||||
|             "close access theFile", |             "close access theFile", | ||||||
|         ] |         ] | ||||||
|  | @ -128,8 +129,6 @@ def grabclipboard(): | ||||||
|                 files = data[o:].decode("mbcs").split("\0") |                 files = data[o:].decode("mbcs").split("\0") | ||||||
|             return files[: files.index("")] |             return files[: files.index("")] | ||||||
|         if isinstance(data, bytes): |         if isinstance(data, bytes): | ||||||
|             import io |  | ||||||
| 
 |  | ||||||
|             data = io.BytesIO(data) |             data = io.BytesIO(data) | ||||||
|             if fmt == "png": |             if fmt == "png": | ||||||
|                 from . import PngImagePlugin |                 from . import PngImagePlugin | ||||||
|  | @ -159,13 +158,12 @@ def grabclipboard(): | ||||||
|         else: |         else: | ||||||
|             msg = "wl-paste or xclip is required for ImageGrab.grabclipboard() on Linux" |             msg = "wl-paste or xclip is required for ImageGrab.grabclipboard() on Linux" | ||||||
|             raise NotImplementedError(msg) |             raise NotImplementedError(msg) | ||||||
|         fh, filepath = tempfile.mkstemp() |         p = subprocess.run(args, stdout=subprocess.PIPE, stderr=subprocess.PIPE) | ||||||
|         err = subprocess.run(args, stdout=fh, stderr=subprocess.PIPE).stderr |         err = p.stderr | ||||||
|         os.close(fh) |  | ||||||
|         if err: |         if err: | ||||||
|             msg = f"{args[0]} error: {err.strip().decode()}" |             msg = f"{args[0]} error: {err.strip().decode()}" | ||||||
|             raise ChildProcessError(msg) |             raise ChildProcessError(msg) | ||||||
|         im = Image.open(filepath) |         data = io.BytesIO(p.stdout) | ||||||
|  |         im = Image.open(data) | ||||||
|         im.load() |         im.load() | ||||||
|         os.unlink(filepath) |  | ||||||
|         return im |         return im | ||||||
|  |  | ||||||
|  | @ -21,7 +21,7 @@ import functools | ||||||
| import operator | import operator | ||||||
| import re | import re | ||||||
| 
 | 
 | ||||||
| from . import Image, ImagePalette | from . import ExifTags, Image, ImagePalette | ||||||
| 
 | 
 | ||||||
| # | # | ||||||
| # helpers | # helpers | ||||||
|  | @ -576,19 +576,20 @@ def solarize(image, threshold=128): | ||||||
|     return _lut(image, lut) |     return _lut(image, lut) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def exif_transpose(image): | def exif_transpose(image, *, in_place=False): | ||||||
|     """ |     """ | ||||||
|     If an image has an EXIF Orientation tag, other than 1, return a new image |     If an image has an EXIF Orientation tag, other than 1, transpose the image | ||||||
|     that is transposed accordingly. The new image will have the orientation |     accordingly, and remove the orientation data. | ||||||
|     data removed. |  | ||||||
| 
 |  | ||||||
|     Otherwise, return a copy of the image. |  | ||||||
| 
 | 
 | ||||||
|     :param image: The image to transpose. |     :param image: The image to transpose. | ||||||
|     :return: An image. |     :param in_place: Boolean. Keyword-only argument. | ||||||
|  |         If ``True``, the original image is modified in-place, and ``None`` is returned. | ||||||
|  |         If ``False`` (default), a new :py:class:`~PIL.Image.Image` object is returned | ||||||
|  |         with the transposition applied. If there is no transposition, a copy of the | ||||||
|  |         image will be returned. | ||||||
|     """ |     """ | ||||||
|     exif = image.getexif() |     image_exif = image.getexif() | ||||||
|     orientation = exif.get(0x0112) |     orientation = image_exif.get(ExifTags.Base.Orientation) | ||||||
|     method = { |     method = { | ||||||
|         2: Image.Transpose.FLIP_LEFT_RIGHT, |         2: Image.Transpose.FLIP_LEFT_RIGHT, | ||||||
|         3: Image.Transpose.ROTATE_180, |         3: Image.Transpose.ROTATE_180, | ||||||
|  | @ -600,22 +601,28 @@ def exif_transpose(image): | ||||||
|     }.get(orientation) |     }.get(orientation) | ||||||
|     if method is not None: |     if method is not None: | ||||||
|         transposed_image = image.transpose(method) |         transposed_image = image.transpose(method) | ||||||
|         transposed_exif = transposed_image.getexif() |         if in_place: | ||||||
|         if 0x0112 in transposed_exif: |             image.im = transposed_image.im | ||||||
|             del transposed_exif[0x0112] |             image.pyaccess = None | ||||||
|             if "exif" in transposed_image.info: |             image._size = transposed_image._size | ||||||
|                 transposed_image.info["exif"] = transposed_exif.tobytes() |         exif_image = image if in_place else transposed_image | ||||||
|             elif "Raw profile type exif" in transposed_image.info: | 
 | ||||||
|                 transposed_image.info[ |         exif = exif_image.getexif() | ||||||
|                     "Raw profile type exif" |         if ExifTags.Base.Orientation in exif: | ||||||
|                 ] = transposed_exif.tobytes().hex() |             del exif[ExifTags.Base.Orientation] | ||||||
|             elif "XML:com.adobe.xmp" in transposed_image.info: |             if "exif" in exif_image.info: | ||||||
|  |                 exif_image.info["exif"] = exif.tobytes() | ||||||
|  |             elif "Raw profile type exif" in exif_image.info: | ||||||
|  |                 exif_image.info["Raw profile type exif"] = exif.tobytes().hex() | ||||||
|  |             elif "XML:com.adobe.xmp" in exif_image.info: | ||||||
|                 for pattern in ( |                 for pattern in ( | ||||||
|                     r'tiff:Orientation="([0-9])"', |                     r'tiff:Orientation="([0-9])"', | ||||||
|                     r"<tiff:Orientation>([0-9])</tiff:Orientation>", |                     r"<tiff:Orientation>([0-9])</tiff:Orientation>", | ||||||
|                 ): |                 ): | ||||||
|                     transposed_image.info["XML:com.adobe.xmp"] = re.sub( |                     exif_image.info["XML:com.adobe.xmp"] = re.sub( | ||||||
|                         pattern, "", transposed_image.info["XML:com.adobe.xmp"] |                         pattern, "", exif_image.info["XML:com.adobe.xmp"] | ||||||
|                     ) |                     ) | ||||||
|  |         if not in_place: | ||||||
|             return transposed_image |             return transposed_image | ||||||
|  |     elif not in_place: | ||||||
|         return image.copy() |         return image.copy() | ||||||
|  |  | ||||||
|  | @ -17,7 +17,7 @@ import subprocess | ||||||
| import sys | import sys | ||||||
| from shlex import quote | from shlex import quote | ||||||
| 
 | 
 | ||||||
| from PIL import Image | from . import Image | ||||||
| 
 | 
 | ||||||
| _viewers = [] | _viewers = [] | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -457,6 +457,11 @@ class JpegImageFile(ImageFile.ImageFile): | ||||||
|         if os.path.exists(self.filename): |         if os.path.exists(self.filename): | ||||||
|             subprocess.check_call(["djpeg", "-outfile", path, self.filename]) |             subprocess.check_call(["djpeg", "-outfile", path, self.filename]) | ||||||
|         else: |         else: | ||||||
|  |             try: | ||||||
|  |                 os.unlink(path) | ||||||
|  |             except OSError: | ||||||
|  |                 pass | ||||||
|  | 
 | ||||||
|             msg = "Invalid Filename" |             msg = "Invalid Filename" | ||||||
|             raise ValueError(msg) |             raise ValueError(msg) | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1146,11 +1146,14 @@ def _write_multiple_frames(im, fp, chunk, rawmode, default_image, append_images) | ||||||
|                     and prev_disposal == encoderinfo.get("disposal") |                     and prev_disposal == encoderinfo.get("disposal") | ||||||
|                     and prev_blend == encoderinfo.get("blend") |                     and prev_blend == encoderinfo.get("blend") | ||||||
|                 ): |                 ): | ||||||
|                     if isinstance(duration, (list, tuple)): |                     previous["encoderinfo"]["duration"] += encoderinfo.get( | ||||||
|                         previous["encoderinfo"]["duration"] += encoderinfo["duration"] |                         "duration", duration | ||||||
|  |                     ) | ||||||
|                     continue |                     continue | ||||||
|             else: |             else: | ||||||
|                 bbox = None |                 bbox = None | ||||||
|  |             if "duration" not in encoderinfo: | ||||||
|  |                 encoderinfo["duration"] = duration | ||||||
|             im_frames.append({"im": im_frame, "bbox": bbox, "encoderinfo": encoderinfo}) |             im_frames.append({"im": im_frame, "bbox": bbox, "encoderinfo": encoderinfo}) | ||||||
| 
 | 
 | ||||||
|     # animation control |     # animation control | ||||||
|  | @ -1175,7 +1178,7 @@ def _write_multiple_frames(im, fp, chunk, rawmode, default_image, append_images) | ||||||
|             im_frame = im_frame.crop(bbox) |             im_frame = im_frame.crop(bbox) | ||||||
|         size = im_frame.size |         size = im_frame.size | ||||||
|         encoderinfo = frame_data["encoderinfo"] |         encoderinfo = frame_data["encoderinfo"] | ||||||
|         frame_duration = int(round(encoderinfo.get("duration", duration))) |         frame_duration = int(round(encoderinfo["duration"])) | ||||||
|         frame_disposal = encoderinfo.get("disposal", disposal) |         frame_disposal = encoderinfo.get("disposal", disposal) | ||||||
|         frame_blend = encoderinfo.get("blend", blend) |         frame_blend = encoderinfo.get("blend", blend) | ||||||
|         # frame control |         # frame control | ||||||
|  |  | ||||||
|  | @ -36,7 +36,7 @@ import os | ||||||
| import struct | import struct | ||||||
| import sys | import sys | ||||||
| 
 | 
 | ||||||
| from PIL import Image, ImageFile | from . import Image, ImageFile | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def isInt(f): | def isInt(f): | ||||||
|  | @ -191,7 +191,7 @@ class SpiderImageFile(ImageFile.ImageFile): | ||||||
| 
 | 
 | ||||||
|     # returns a ImageTk.PhotoImage object, after rescaling to 0..255 |     # returns a ImageTk.PhotoImage object, after rescaling to 0..255 | ||||||
|     def tkPhotoImage(self): |     def tkPhotoImage(self): | ||||||
|         from PIL import ImageTk |         from . import ImageTk | ||||||
| 
 | 
 | ||||||
|         return ImageTk.PhotoImage(self.convert2byte(), palette=256) |         return ImageTk.PhotoImage(self.convert2byte(), palette=256) | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -49,7 +49,7 @@ from collections.abc import MutableMapping | ||||||
| from fractions import Fraction | from fractions import Fraction | ||||||
| from numbers import Number, Rational | from numbers import Number, Rational | ||||||
| 
 | 
 | ||||||
| from . import Image, ImageFile, ImageOps, ImagePalette, TiffTags | from . import ExifTags, Image, ImageFile, ImageOps, ImagePalette, TiffTags | ||||||
| from ._binary import i16be as i16 | from ._binary import i16be as i16 | ||||||
| from ._binary import i32be as i32 | from ._binary import i32be as i32 | ||||||
| from ._binary import o8 | from ._binary import o8 | ||||||
|  | @ -1185,7 +1185,7 @@ class TiffImageFile(ImageFile.ImageFile): | ||||||
|         :returns: Photoshop "Image Resource Blocks" in a dictionary. |         :returns: Photoshop "Image Resource Blocks" in a dictionary. | ||||||
|         """ |         """ | ||||||
|         blocks = {} |         blocks = {} | ||||||
|         val = self.tag_v2.get(0x8649) |         val = self.tag_v2.get(ExifTags.Base.ImageResources) | ||||||
|         if val: |         if val: | ||||||
|             while val[:4] == b"8BIM": |             while val[:4] == b"8BIM": | ||||||
|                 id = i16(val[4:6]) |                 id = i16(val[4:6]) | ||||||
|  | @ -1550,7 +1550,7 @@ class TiffImageFile(ImageFile.ImageFile): | ||||||
|             palette = [o8(b // 256) for b in self.tag_v2[COLORMAP]] |             palette = [o8(b // 256) for b in self.tag_v2[COLORMAP]] | ||||||
|             self.palette = ImagePalette.raw("RGB;L", b"".join(palette)) |             self.palette = ImagePalette.raw("RGB;L", b"".join(palette)) | ||||||
| 
 | 
 | ||||||
|         self._tile_orientation = self.tag_v2.get(0x0112) |         self._tile_orientation = self.tag_v2.get(ExifTags.Base.Orientation) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| # | # | ||||||
|  | @ -1894,6 +1894,10 @@ class AppendingTiffWriter: | ||||||
|         8,  # srational |         8,  # srational | ||||||
|         4,  # float |         4,  # float | ||||||
|         8,  # double |         8,  # double | ||||||
|  |         4,  # ifd | ||||||
|  |         2,  # unicode | ||||||
|  |         4,  # complex | ||||||
|  |         8,  # long8 | ||||||
|     ] |     ] | ||||||
| 
 | 
 | ||||||
|     #    StripOffsets = 273 |     #    StripOffsets = 273 | ||||||
|  |  | ||||||
|  | @ -24,7 +24,7 @@ def check_module(feature): | ||||||
|     :returns: ``True`` if available, ``False`` otherwise. |     :returns: ``True`` if available, ``False`` otherwise. | ||||||
|     :raises ValueError: If the module is not defined in this version of Pillow. |     :raises ValueError: If the module is not defined in this version of Pillow. | ||||||
|     """ |     """ | ||||||
|     if not (feature in modules): |     if feature not in modules: | ||||||
|         msg = f"Unknown module {feature}" |         msg = f"Unknown module {feature}" | ||||||
|         raise ValueError(msg) |         raise ValueError(msg) | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1027,12 +1027,11 @@ _crop(ImagingObject *self, PyObject *args) { | ||||||
| static PyObject * | static PyObject * | ||||||
| _expand_image(ImagingObject *self, PyObject *args) { | _expand_image(ImagingObject *self, PyObject *args) { | ||||||
|     int x, y; |     int x, y; | ||||||
|     int mode = 0; |     if (!PyArg_ParseTuple(args, "ii", &x, &y)) { | ||||||
|     if (!PyArg_ParseTuple(args, "ii|i", &x, &y, &mode)) { |  | ||||||
|         return NULL; |         return NULL; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return PyImagingNew(ImagingExpand(self->image, x, y, mode)); |     return PyImagingNew(ImagingExpand(self->image, x, y)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static PyObject * | static PyObject * | ||||||
|  |  | ||||||
							
								
								
									
										263
									
								
								src/_imagingft.c
									
									
									
									
									
								
							
							
						
						
									
										263
									
								
								src/_imagingft.c
									
									
									
									
									
								
							|  | @ -132,6 +132,27 @@ getfont(PyObject *self_, PyObject *args, PyObject *kw) { | ||||||
|         return NULL; |         return NULL; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | #if PY_MAJOR_VERSION > 3 || PY_MINOR_VERSION > 11 | ||||||
|  |     PyConfig config; | ||||||
|  |     PyConfig_InitPythonConfig(&config); | ||||||
|  |     if (!PyArg_ParseTupleAndKeywords( | ||||||
|  |             args, | ||||||
|  |             kw, | ||||||
|  |             "etf|nsy#n", | ||||||
|  |             kwlist, | ||||||
|  |             config.filesystem_encoding, | ||||||
|  |             &filename, | ||||||
|  |             &size, | ||||||
|  |             &index, | ||||||
|  |             &encoding, | ||||||
|  |             &font_bytes, | ||||||
|  |             &font_bytes_size, | ||||||
|  |             &layout_engine)) { | ||||||
|  |         PyConfig_Clear(&config); | ||||||
|  |         return NULL; | ||||||
|  |     } | ||||||
|  |     PyConfig_Clear(&config); | ||||||
|  | #else | ||||||
|     if (!PyArg_ParseTupleAndKeywords( |     if (!PyArg_ParseTupleAndKeywords( | ||||||
|             args, |             args, | ||||||
|             kw, |             kw, | ||||||
|  | @ -147,6 +168,7 @@ getfont(PyObject *self_, PyObject *args, PyObject *kw) { | ||||||
|             &layout_engine)) { |             &layout_engine)) { | ||||||
|         return NULL; |         return NULL; | ||||||
|     } |     } | ||||||
|  | #endif | ||||||
| 
 | 
 | ||||||
|     self = PyObject_New(FontObject, &Font_Type); |     self = PyObject_New(FontObject, &Font_Type); | ||||||
|     if (!self) { |     if (!self) { | ||||||
|  | @ -167,7 +189,7 @@ getfont(PyObject *self_, PyObject *args, PyObject *kw) { | ||||||
|         /* Don't free this before FT_Done_Face */ |         /* Don't free this before FT_Done_Face */ | ||||||
|         self->font_bytes = PyMem_Malloc(font_bytes_size); |         self->font_bytes = PyMem_Malloc(font_bytes_size); | ||||||
|         if (!self->font_bytes) { |         if (!self->font_bytes) { | ||||||
|             error = 65;  // Out of Memory in Freetype.
 |             error = FT_Err_Out_Of_Memory; | ||||||
|         } |         } | ||||||
|         if (!error) { |         if (!error) { | ||||||
|             memcpy(self->font_bytes, font_bytes, (size_t)font_bytes_size); |             memcpy(self->font_bytes, font_bytes, (size_t)font_bytes_size); | ||||||
|  | @ -232,9 +254,7 @@ text_layout_raqm( | ||||||
|     const char *dir, |     const char *dir, | ||||||
|     PyObject *features, |     PyObject *features, | ||||||
|     const char *lang, |     const char *lang, | ||||||
|     GlyphInfo **glyph_info, |     GlyphInfo **glyph_info) { | ||||||
|     int mask, |  | ||||||
|     int color) { |  | ||||||
|     size_t i = 0, count = 0, start = 0; |     size_t i = 0, count = 0, start = 0; | ||||||
|     raqm_t *rq; |     raqm_t *rq; | ||||||
|     raqm_glyph_t *glyphs = NULL; |     raqm_glyph_t *glyphs = NULL; | ||||||
|  | @ -471,7 +491,7 @@ text_layout( | ||||||
| #ifdef HAVE_RAQM | #ifdef HAVE_RAQM | ||||||
|     if (have_raqm && self->layout_engine == LAYOUT_RAQM) { |     if (have_raqm && self->layout_engine == LAYOUT_RAQM) { | ||||||
|         count = text_layout_raqm( |         count = text_layout_raqm( | ||||||
|             string, self, dir, features, lang, glyph_info,  mask, color); |             string, self, dir, features, lang, glyph_info); | ||||||
|     } else |     } else | ||||||
| #endif | #endif | ||||||
|     { |     { | ||||||
|  | @ -529,73 +549,25 @@ font_getlength(FontObject *self, PyObject *args) { | ||||||
|     return PyLong_FromLong(length); |     return PyLong_FromLong(length); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static PyObject * | static int | ||||||
| font_getsize(FontObject *self, PyObject *args) { | bounding_box_and_anchors(FT_Face face, const char *anchor, int horizontal_dir, GlyphInfo *glyph_info, size_t count, int load_flags, int *width, int *height, int *x_offset, int *y_offset) { | ||||||
|     int position; /* pen position along primary axis, in 26.6 precision */ |     int position; /* pen position along primary axis, in 26.6 precision */ | ||||||
|     int advanced; /* pen position along primary axis, in pixels */ |     int advanced; /* pen position along primary axis, in pixels */ | ||||||
|     int px, py;   /* position of current glyph, in pixels */ |     int px, py;   /* position of current glyph, in pixels */ | ||||||
|     int x_min, x_max, y_min, y_max; /* text bounding box, in pixels */ |     int x_min, x_max, y_min, y_max; /* text bounding box, in pixels */ | ||||||
|     int x_anchor, y_anchor;         /* offset of point drawn at (0, 0), in pixels */ |     int x_anchor, y_anchor;         /* offset of point drawn at (0, 0), in pixels */ | ||||||
|     int load_flags;                 /* FreeType load_flags parameter */ |  | ||||||
|     int error; |     int error; | ||||||
|     FT_Face face; |  | ||||||
|     FT_Glyph glyph; |     FT_Glyph glyph; | ||||||
|     FT_BBox bbox;                   /* glyph bounding box */ |     FT_BBox bbox;                   /* glyph bounding box */ | ||||||
|     GlyphInfo *glyph_info = NULL; /* computed text layout */ |     size_t i;                       /* glyph_info index */ | ||||||
|     size_t i, count;              /* glyph_info index and length */ |  | ||||||
|     int horizontal_dir;           /* is primary axis horizontal? */ |  | ||||||
|     int mask = 0;                 /* is FT_LOAD_TARGET_MONO enabled? */ |  | ||||||
|     int color = 0;                /* is FT_LOAD_COLOR enabled? */ |  | ||||||
|     const char *mode = NULL; |  | ||||||
|     const char *dir = NULL; |  | ||||||
|     const char *lang = NULL; |  | ||||||
|     const char *anchor = NULL; |  | ||||||
|     PyObject *features = Py_None; |  | ||||||
|     PyObject *string; |  | ||||||
| 
 |  | ||||||
|     /* calculate size and bearing for a given string */ |  | ||||||
| 
 |  | ||||||
|     if (!PyArg_ParseTuple( |  | ||||||
|             args, "O|zzOzz:getsize", &string, &mode, &dir, &features, &lang, &anchor)) { |  | ||||||
|         return NULL; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     horizontal_dir = dir && strcmp(dir, "ttb") == 0 ? 0 : 1; |  | ||||||
| 
 |  | ||||||
|     mask = mode && strcmp(mode, "1") == 0; |  | ||||||
|     color = mode && strcmp(mode, "RGBA") == 0; |  | ||||||
| 
 |  | ||||||
|     if (anchor == NULL) { |  | ||||||
|         anchor = horizontal_dir ? "la" : "lt"; |  | ||||||
|     } |  | ||||||
|     if (strlen(anchor) != 2) { |  | ||||||
|         goto bad_anchor; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     count = text_layout(string, self, dir, features, lang, &glyph_info, mask, color); |  | ||||||
|     if (PyErr_Occurred()) { |  | ||||||
|         return NULL; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     load_flags = FT_LOAD_DEFAULT; |  | ||||||
|     if (mask) { |  | ||||||
|         load_flags |= FT_LOAD_TARGET_MONO; |  | ||||||
|     } |  | ||||||
|     if (color) { |  | ||||||
|         load_flags |= FT_LOAD_COLOR; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /*
 |     /*
 | ||||||
|      * text bounds are given by: |      * text bounds are given by: | ||||||
|      *   - bounding boxes of individual glyphs |      *   - bounding boxes of individual glyphs | ||||||
|      *   - pen line, i.e. 0 to `advanced` along primary axis |      *   - pen line, i.e. 0 to `advanced` along primary axis | ||||||
|      *     this means point (0, 0) is part of the text bounding box |      *     this means point (0, 0) is part of the text bounding box | ||||||
|      */ |      */ | ||||||
|     face = NULL; |  | ||||||
|     position = x_min = x_max = y_min = y_max = 0; |     position = x_min = x_max = y_min = y_max = 0; | ||||||
|     for (i = 0; i < count; i++) { |     for (i = 0; i < count; i++) { | ||||||
|         face = self->face; |  | ||||||
| 
 |  | ||||||
|         if (horizontal_dir) { |         if (horizontal_dir) { | ||||||
|             px = PIXEL(position + glyph_info[i].x_offset); |             px = PIXEL(position + glyph_info[i].x_offset); | ||||||
|             py = PIXEL(glyph_info[i].y_offset); |             py = PIXEL(glyph_info[i].y_offset); | ||||||
|  | @ -618,12 +590,14 @@ font_getsize(FontObject *self, PyObject *args) { | ||||||
| 
 | 
 | ||||||
|         error = FT_Load_Glyph(face, glyph_info[i].index, load_flags); |         error = FT_Load_Glyph(face, glyph_info[i].index, load_flags); | ||||||
|         if (error) { |         if (error) { | ||||||
|             return geterror(error); |             geterror(error); | ||||||
|  |             return 1; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         error = FT_Get_Glyph(face->glyph, &glyph); |         error = FT_Get_Glyph(face->glyph, &glyph); | ||||||
|         if (error) { |         if (error) { | ||||||
|             return geterror(error); |             geterror(error); | ||||||
|  |             return 1; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         FT_Glyph_Get_CBox(glyph, FT_GLYPH_BBOX_PIXELS, &bbox); |         FT_Glyph_Get_CBox(glyph, FT_GLYPH_BBOX_PIXELS, &bbox); | ||||||
|  | @ -647,13 +621,15 @@ font_getsize(FontObject *self, PyObject *args) { | ||||||
|         FT_Done_Glyph(glyph); |         FT_Done_Glyph(glyph); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (glyph_info) { |     if (anchor == NULL) { | ||||||
|         PyMem_Free(glyph_info); |         anchor = horizontal_dir ? "la" : "lt"; | ||||||
|         glyph_info = NULL; |     } | ||||||
|  |     if (strlen(anchor) != 2) { | ||||||
|  |         goto bad_anchor; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     x_anchor = y_anchor = 0; |     x_anchor = y_anchor = 0; | ||||||
|     if (face) { |     if (count) { | ||||||
|         if (horizontal_dir) { |         if (horizontal_dir) { | ||||||
|             switch (anchor[0]) { |             switch (anchor[0]) { | ||||||
|                 case 'l':  // left
 |                 case 'l':  // left
 | ||||||
|  | @ -671,15 +647,15 @@ font_getsize(FontObject *self, PyObject *args) { | ||||||
|             } |             } | ||||||
|             switch (anchor[1]) { |             switch (anchor[1]) { | ||||||
|                 case 'a':  // ascender
 |                 case 'a':  // ascender
 | ||||||
|                     y_anchor = PIXEL(self->face->size->metrics.ascender); |                     y_anchor = PIXEL(face->size->metrics.ascender); | ||||||
|                     break; |                     break; | ||||||
|                 case 't':  // top
 |                 case 't':  // top
 | ||||||
|                     y_anchor = y_max; |                     y_anchor = y_max; | ||||||
|                     break; |                     break; | ||||||
|                 case 'm':  // middle (ascender + descender) / 2
 |                 case 'm':  // middle (ascender + descender) / 2
 | ||||||
|                     y_anchor = PIXEL( |                     y_anchor = PIXEL( | ||||||
|                         (self->face->size->metrics.ascender + |                         (face->size->metrics.ascender + | ||||||
|                          self->face->size->metrics.descender) / |                          face->size->metrics.descender) / | ||||||
|                         2); |                         2); | ||||||
|                     break; |                     break; | ||||||
|                 case 's':  // horizontal baseline
 |                 case 's':  // horizontal baseline
 | ||||||
|  | @ -689,7 +665,7 @@ font_getsize(FontObject *self, PyObject *args) { | ||||||
|                     y_anchor = y_min; |                     y_anchor = y_min; | ||||||
|                     break; |                     break; | ||||||
|                 case 'd':  // descender
 |                 case 'd':  // descender
 | ||||||
|                     y_anchor = PIXEL(self->face->size->metrics.descender); |                     y_anchor = PIXEL(face->size->metrics.descender); | ||||||
|                     break; |                     break; | ||||||
|                 default: |                 default: | ||||||
|                     goto bad_anchor; |                     goto bad_anchor; | ||||||
|  | @ -729,17 +705,74 @@ font_getsize(FontObject *self, PyObject *args) { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 |     *width = x_max - x_min; | ||||||
|     return Py_BuildValue( |     *height = y_max - y_min; | ||||||
|         "(ii)(ii)", |     *x_offset = -x_anchor + x_min; | ||||||
|         (x_max - x_min), |     *y_offset = -(-y_anchor + y_max); | ||||||
|         (y_max - y_min), |     return 0; | ||||||
|         (-x_anchor + x_min), |  | ||||||
|         -(-y_anchor + y_max)); |  | ||||||
| 
 | 
 | ||||||
| bad_anchor: | bad_anchor: | ||||||
|     PyErr_Format(PyExc_ValueError, "bad anchor specified: %s", anchor); |     PyErr_Format(PyExc_ValueError, "bad anchor specified: %s", anchor); | ||||||
|  |     return 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static PyObject * | ||||||
|  | font_getsize(FontObject *self, PyObject *args) { | ||||||
|  |     int width, height, x_offset, y_offset; | ||||||
|  |     int load_flags;               /* FreeType load_flags parameter */ | ||||||
|  |     int error; | ||||||
|  |     GlyphInfo *glyph_info = NULL; /* computed text layout */ | ||||||
|  |     size_t count;                 /* glyph_info length */ | ||||||
|  |     int horizontal_dir;           /* is primary axis horizontal? */ | ||||||
|  |     int mask = 0;                 /* is FT_LOAD_TARGET_MONO enabled? */ | ||||||
|  |     int color = 0;                /* is FT_LOAD_COLOR enabled? */ | ||||||
|  |     const char *mode = NULL; | ||||||
|  |     const char *dir = NULL; | ||||||
|  |     const char *lang = NULL; | ||||||
|  |     const char *anchor = NULL; | ||||||
|  |     PyObject *features = Py_None; | ||||||
|  |     PyObject *string; | ||||||
|  | 
 | ||||||
|  |     /* calculate size and bearing for a given string */ | ||||||
|  | 
 | ||||||
|  |     if (!PyArg_ParseTuple( | ||||||
|  |             args, "O|zzOzz:getsize", &string, &mode, &dir, &features, &lang, &anchor)) { | ||||||
|         return NULL; |         return NULL; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     horizontal_dir = dir && strcmp(dir, "ttb") == 0 ? 0 : 1; | ||||||
|  | 
 | ||||||
|  |     mask = mode && strcmp(mode, "1") == 0; | ||||||
|  |     color = mode && strcmp(mode, "RGBA") == 0; | ||||||
|  | 
 | ||||||
|  |     count = text_layout(string, self, dir, features, lang, &glyph_info, mask, color); | ||||||
|  |     if (PyErr_Occurred()) { | ||||||
|  |         return NULL; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     load_flags = FT_LOAD_DEFAULT; | ||||||
|  |     if (mask) { | ||||||
|  |         load_flags |= FT_LOAD_TARGET_MONO; | ||||||
|  |     } | ||||||
|  |     if (color) { | ||||||
|  |         load_flags |= FT_LOAD_COLOR; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     error = bounding_box_and_anchors(self->face, anchor, horizontal_dir, glyph_info, count, load_flags, &width, &height, &x_offset, &y_offset); | ||||||
|  |     if (glyph_info) { | ||||||
|  |         PyMem_Free(glyph_info); | ||||||
|  |         glyph_info = NULL; | ||||||
|  |     } | ||||||
|  |     if (error) { | ||||||
|  |         return NULL; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return Py_BuildValue( | ||||||
|  |         "(ii)(ii)", | ||||||
|  |         width, | ||||||
|  |         height, | ||||||
|  |         x_offset, | ||||||
|  |         y_offset); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static PyObject * | static PyObject * | ||||||
|  | @ -763,6 +796,7 @@ font_render(FontObject *self, PyObject *args) { | ||||||
|     unsigned int bitmap_y;          /* glyph bitmap y index */ |     unsigned int bitmap_y;          /* glyph bitmap y index */ | ||||||
|     unsigned char *source;          /* glyph bitmap source buffer */ |     unsigned char *source;          /* glyph bitmap source buffer */ | ||||||
|     unsigned char convert_scale;    /* scale factor for non-8bpp bitmaps */ |     unsigned char convert_scale;    /* scale factor for non-8bpp bitmaps */ | ||||||
|  |     PyObject *image; | ||||||
|     Imaging im; |     Imaging im; | ||||||
|     Py_ssize_t id; |     Py_ssize_t id; | ||||||
|     int mask = 0;  /* is FT_LOAD_TARGET_MONO enabled? */ |     int mask = 0;  /* is FT_LOAD_TARGET_MONO enabled? */ | ||||||
|  | @ -773,27 +807,34 @@ font_render(FontObject *self, PyObject *args) { | ||||||
|     const char *mode = NULL; |     const char *mode = NULL; | ||||||
|     const char *dir = NULL; |     const char *dir = NULL; | ||||||
|     const char *lang = NULL; |     const char *lang = NULL; | ||||||
|  |     const char *anchor = NULL; | ||||||
|     PyObject *features = Py_None; |     PyObject *features = Py_None; | ||||||
|     PyObject *string; |     PyObject *string; | ||||||
|  |     PyObject *fill; | ||||||
|     float x_start = 0; |     float x_start = 0; | ||||||
|     float y_start = 0; |     float y_start = 0; | ||||||
|  |     int width, height, x_offset, y_offset; | ||||||
|  |     int horizontal_dir; /* is primary axis horizontal? */ | ||||||
|  |     PyObject *max_image_pixels = Py_None; | ||||||
| 
 | 
 | ||||||
|     /* render string into given buffer (the buffer *must* have
 |     /* render string into given buffer (the buffer *must* have
 | ||||||
|        the right size, or this will crash) */ |        the right size, or this will crash) */ | ||||||
| 
 | 
 | ||||||
|     if (!PyArg_ParseTuple( |     if (!PyArg_ParseTuple( | ||||||
|             args, |             args, | ||||||
|             "On|zzOziLff:render", |             "OO|zzOzizLffO:render", | ||||||
|             &string, |             &string, | ||||||
|             &id, |             &fill, | ||||||
|             &mode, |             &mode, | ||||||
|             &dir, |             &dir, | ||||||
|             &features, |             &features, | ||||||
|             &lang, |             &lang, | ||||||
|             &stroke_width, |             &stroke_width, | ||||||
|  |             &anchor, | ||||||
|             &foreground_ink_long, |             &foreground_ink_long, | ||||||
|             &x_start, |             &x_start, | ||||||
|             &y_start)) { |             &y_start, | ||||||
|  |             &max_image_pixels)) { | ||||||
|         return NULL; |         return NULL; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -819,14 +860,52 @@ font_render(FontObject *self, PyObject *args) { | ||||||
|     if (PyErr_Occurred()) { |     if (PyErr_Occurred()) { | ||||||
|         return NULL; |         return NULL; | ||||||
|     } |     } | ||||||
|     if (count == 0) { | 
 | ||||||
|         Py_RETURN_NONE; |     load_flags = stroke_width ? FT_LOAD_NO_BITMAP : FT_LOAD_DEFAULT; | ||||||
|  |     if (mask) { | ||||||
|  |         load_flags |= FT_LOAD_TARGET_MONO; | ||||||
|  |     } | ||||||
|  |     if (color) { | ||||||
|  |         load_flags |= FT_LOAD_COLOR; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     horizontal_dir = dir && strcmp(dir, "ttb") == 0 ? 0 : 1; | ||||||
|  | 
 | ||||||
|  |     error = bounding_box_and_anchors(self->face, anchor, horizontal_dir, glyph_info, count, load_flags, &width, &height, &x_offset, &y_offset); | ||||||
|  |     if (error) { | ||||||
|  |         PyMem_Del(glyph_info); | ||||||
|  |         return NULL; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     width += stroke_width * 2 + ceil(x_start); | ||||||
|  |     height += stroke_width * 2 + ceil(y_start); | ||||||
|  |     if (max_image_pixels != Py_None) { | ||||||
|  |         if ((long long)width * height > PyLong_AsLongLong(max_image_pixels) * 2) { | ||||||
|  |             PyMem_Del(glyph_info); | ||||||
|  |             return Py_BuildValue("O(ii)(ii)", Py_None, width, height, 0, 0); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     image = PyObject_CallFunction(fill, "s(ii)", strcmp(mode, "RGBA") == 0 ? "RGBA" : "L", width, height); | ||||||
|  |     if (image == NULL) { | ||||||
|  |         PyMem_Del(glyph_info); | ||||||
|  |         return NULL; | ||||||
|  |     } | ||||||
|  |     id = PyLong_AsSsize_t(PyObject_GetAttrString(image, "id")); | ||||||
|  |     im = (Imaging)id; | ||||||
|  | 
 | ||||||
|  |     x_offset -= stroke_width; | ||||||
|  |     y_offset -= stroke_width; | ||||||
|  |     if (count == 0 || width == 0 || height == 0) { | ||||||
|  |         PyMem_Del(glyph_info); | ||||||
|  |         return Py_BuildValue("O(ii)(ii)", image, width, height, x_offset, y_offset); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (stroke_width) { |     if (stroke_width) { | ||||||
|         error = FT_Stroker_New(library, &stroker); |         error = FT_Stroker_New(library, &stroker); | ||||||
|         if (error) { |         if (error) { | ||||||
|             return geterror(error); |             geterror(error); | ||||||
|  |             goto glyph_error; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         FT_Stroker_Set( |         FT_Stroker_Set( | ||||||
|  | @ -837,15 +916,6 @@ font_render(FontObject *self, PyObject *args) { | ||||||
|             0); |             0); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     im = (Imaging)id; |  | ||||||
|     load_flags = stroke_width ? FT_LOAD_NO_BITMAP : FT_LOAD_DEFAULT; |  | ||||||
|     if (mask) { |  | ||||||
|         load_flags |= FT_LOAD_TARGET_MONO; |  | ||||||
|     } |  | ||||||
|     if (color) { |  | ||||||
|         load_flags |= FT_LOAD_COLOR; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /*
 |     /*
 | ||||||
|      * calculate x_min and y_max |      * calculate x_min and y_max | ||||||
|      * must match font_getsize or there may be clipping! |      * must match font_getsize or there may be clipping! | ||||||
|  | @ -858,7 +928,8 @@ font_render(FontObject *self, PyObject *args) { | ||||||
|         error = |         error = | ||||||
|             FT_Load_Glyph(self->face, glyph_info[i].index, load_flags | FT_LOAD_RENDER); |             FT_Load_Glyph(self->face, glyph_info[i].index, load_flags | FT_LOAD_RENDER); | ||||||
|         if (error) { |         if (error) { | ||||||
|             return geterror(error); |             geterror(error); | ||||||
|  |             goto glyph_error; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         glyph_slot = self->face->glyph; |         glyph_slot = self->face->glyph; | ||||||
|  | @ -889,7 +960,8 @@ font_render(FontObject *self, PyObject *args) { | ||||||
| 
 | 
 | ||||||
|         error = FT_Load_Glyph(self->face, glyph_info[i].index, load_flags); |         error = FT_Load_Glyph(self->face, glyph_info[i].index, load_flags); | ||||||
|         if (error) { |         if (error) { | ||||||
|             return geterror(error); |             geterror(error); | ||||||
|  |             goto glyph_error; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         glyph_slot = self->face->glyph; |         glyph_slot = self->face->glyph; | ||||||
|  | @ -903,7 +975,8 @@ font_render(FontObject *self, PyObject *args) { | ||||||
|                 error = FT_Glyph_To_Bitmap(&glyph, FT_RENDER_MODE_NORMAL, &origin, 1); |                 error = FT_Glyph_To_Bitmap(&glyph, FT_RENDER_MODE_NORMAL, &origin, 1); | ||||||
|             } |             } | ||||||
|             if (error) { |             if (error) { | ||||||
|                 return geterror(error); |                 geterror(error); | ||||||
|  |                 goto glyph_error; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             bitmap_glyph = (FT_BitmapGlyph)glyph; |             bitmap_glyph = (FT_BitmapGlyph)glyph; | ||||||
|  | @ -1042,9 +1115,15 @@ font_render(FontObject *self, PyObject *args) { | ||||||
|     } |     } | ||||||
|     FT_Stroker_Done(stroker); |     FT_Stroker_Done(stroker); | ||||||
|     PyMem_Del(glyph_info); |     PyMem_Del(glyph_info); | ||||||
|     Py_RETURN_NONE; |     return Py_BuildValue("O(ii)(ii)", image, width, height, x_offset, y_offset); | ||||||
| 
 | 
 | ||||||
| glyph_error: | glyph_error: | ||||||
|  |     if (im->destroy) { | ||||||
|  |         im->destroy(im); | ||||||
|  |     } | ||||||
|  |     if (im->image) { | ||||||
|  |         free(im->image); | ||||||
|  |     } | ||||||
|     if (stroker != NULL) { |     if (stroker != NULL) { | ||||||
|         FT_Done_Glyph(glyph); |         FT_Done_Glyph(glyph); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -436,10 +436,16 @@ PyImaging_GrabClipboardWin32(PyObject *self, PyObject *args) { | ||||||
|     UINT formats[] = {CF_DIB, CF_DIBV5, CF_HDROP, RegisterClipboardFormatA("PNG"), 0}; |     UINT formats[] = {CF_DIB, CF_DIBV5, CF_HDROP, RegisterClipboardFormatA("PNG"), 0}; | ||||||
|     LPCSTR format_names[] = {"DIB", "DIB", "file", "png", NULL}; |     LPCSTR format_names[] = {"DIB", "DIB", "file", "png", NULL}; | ||||||
| 
 | 
 | ||||||
|  |     if (!OpenClipboard(NULL)) { | ||||||
|  |         // Maybe the clipboard is temporarily in use by another process.
 | ||||||
|  |         // Wait and try again
 | ||||||
|  |         Sleep(500); | ||||||
|  | 
 | ||||||
|         if (!OpenClipboard(NULL)) { |         if (!OpenClipboard(NULL)) { | ||||||
|             PyErr_SetString(PyExc_OSError, "failed to open clipboard"); |             PyErr_SetString(PyExc_OSError, "failed to open clipboard"); | ||||||
|             return NULL; |             return NULL; | ||||||
|         } |         } | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     // find best format as set by clipboard owner
 |     // find best format as set by clipboard owner
 | ||||||
|     format = 0; |     format = 0; | ||||||
|  |  | ||||||
|  | @ -49,7 +49,7 @@ clip32(float in) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| Imaging | Imaging | ||||||
| ImagingExpand(Imaging imIn, int xmargin, int ymargin, int mode) { | ImagingExpand(Imaging imIn, int xmargin, int ymargin) { | ||||||
|     Imaging imOut; |     Imaging imOut; | ||||||
|     int x, y; |     int x, y; | ||||||
|     ImagingSectionCookie cookie; |     ImagingSectionCookie cookie; | ||||||
|  |  | ||||||
|  | @ -25,7 +25,7 @@ | ||||||
| #endif | #endif | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| #if defined(_WIN32) || defined(__CYGWIN__) | #if defined(_WIN32) || defined(__CYGWIN__) /* WIN */ | ||||||
| 
 | 
 | ||||||
| #define WIN32_LEAN_AND_MEAN | #define WIN32_LEAN_AND_MEAN | ||||||
| #include <Windows.h> | #include <Windows.h> | ||||||
|  | @ -37,15 +37,33 @@ | ||||||
| #undef WIN32 | #undef WIN32 | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| #else | #else /* not WIN */ | ||||||
| /* For System that are not Windows, we'll need to define these. */ | /* For System that are not Windows, we'll need to define these. */ | ||||||
|  | /* We have to define them instead of using typedef because the JPEG lib also
 | ||||||
|  |    defines their own types with the same names, so we need to be able to undef | ||||||
|  |    ours before including the JPEG code. */ | ||||||
|  | 
 | ||||||
|  | #if __STDC_VERSION__ >= 199901L /* C99+ */ | ||||||
|  | 
 | ||||||
|  | #include <stdint.h> | ||||||
|  | 
 | ||||||
|  | #define INT8 int8_t | ||||||
|  | #define UINT8 uint8_t | ||||||
|  | #define INT16 int16_t | ||||||
|  | #define UINT16 uint16_t | ||||||
|  | #define INT32 int32_t | ||||||
|  | #define UINT32 uint32_t | ||||||
|  | 
 | ||||||
|  | #else /* < C99 */ | ||||||
|  | 
 | ||||||
|  | #define INT8 signed char | ||||||
| 
 | 
 | ||||||
| #if SIZEOF_SHORT == 2 | #if SIZEOF_SHORT == 2 | ||||||
| #define INT16 short | #define INT16 short | ||||||
| #elif SIZEOF_INT == 2 | #elif SIZEOF_INT == 2 | ||||||
| #define INT16 int | #define INT16 int | ||||||
| #else | #else | ||||||
| #define INT16 short /* most things works just fine anyway... */ | #error Cannot find required 16-bit integer type | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| #if SIZEOF_SHORT == 4 | #if SIZEOF_SHORT == 4 | ||||||
|  | @ -58,19 +76,13 @@ | ||||||
| #error Cannot find required 32-bit integer type | #error Cannot find required 32-bit integer type | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| #if SIZEOF_LONG == 8 |  | ||||||
| #define INT64 long |  | ||||||
| #elif SIZEOF_LONG_LONG == 8 |  | ||||||
| #define INT64 long |  | ||||||
| #endif |  | ||||||
| 
 |  | ||||||
| #define INT8 signed char |  | ||||||
| #define UINT8 unsigned char | #define UINT8 unsigned char | ||||||
| 
 |  | ||||||
| #define UINT16 unsigned INT16 | #define UINT16 unsigned INT16 | ||||||
| #define UINT32 unsigned INT32 | #define UINT32 unsigned INT32 | ||||||
| 
 | 
 | ||||||
| #endif | #endif /* < C99 */ | ||||||
|  | 
 | ||||||
|  | #endif /* not WIN */ | ||||||
| 
 | 
 | ||||||
| /* assume IEEE; tweak if necessary (patches are welcome) */ | /* assume IEEE; tweak if necessary (patches are welcome) */ | ||||||
| #define FLOAT16 UINT16 | #define FLOAT16 UINT16 | ||||||
|  |  | ||||||
|  | @ -290,7 +290,7 @@ ImagingConvertTransparent(Imaging im, const char *mode, int r, int g, int b); | ||||||
| extern Imaging | extern Imaging | ||||||
| ImagingCrop(Imaging im, int x0, int y0, int x1, int y1); | ImagingCrop(Imaging im, int x0, int y0, int x1, int y1); | ||||||
| extern Imaging | extern Imaging | ||||||
| ImagingExpand(Imaging im, int x, int y, int mode); | ImagingExpand(Imaging im, int x, int y); | ||||||
| extern Imaging | extern Imaging | ||||||
| ImagingFill(Imaging im, const void *ink); | ImagingFill(Imaging im, const void *ink); | ||||||
| extern int | extern int | ||||||
|  |  | ||||||
|  | @ -464,7 +464,7 @@ j2k_encode_entry(Imaging im, ImagingCodecState state) { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (!context->num_resolutions) { |     if (!context->num_resolutions) { | ||||||
|         while (tile_width < (1 << (params.numresolution - 1U)) || tile_height < (1 << (params.numresolution - 1U))) { |         while (tile_width < (1U << (params.numresolution - 1U)) || tile_height < (1U << (params.numresolution - 1U))) { | ||||||
|             params.numresolution -= 1; |             params.numresolution -= 1; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -37,8 +37,6 @@ | ||||||
| #include "Imaging.h" | #include "Imaging.h" | ||||||
| #include <string.h> | #include <string.h> | ||||||
| 
 | 
 | ||||||
| int ImagingNewCount = 0; |  | ||||||
| 
 |  | ||||||
| /* --------------------------------------------------------------------
 | /* --------------------------------------------------------------------
 | ||||||
|  * Standard image object. |  * Standard image object. | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
|  | @ -1552,10 +1552,12 @@ static struct { | ||||||
|     {"P", "P;4L", 4, unpackP4L}, |     {"P", "P;4L", 4, unpackP4L}, | ||||||
|     {"P", "P", 8, copy1}, |     {"P", "P", 8, copy1}, | ||||||
|     {"P", "P;R", 8, unpackLR}, |     {"P", "P;R", 8, unpackLR}, | ||||||
|  |     {"P", "L", 8, copy1}, | ||||||
| 
 | 
 | ||||||
|     /* palette w. alpha */ |     /* palette w. alpha */ | ||||||
|     {"PA", "PA", 16, unpackLA}, |     {"PA", "PA", 16, unpackLA}, | ||||||
|     {"PA", "PA;L", 16, unpackLAL}, |     {"PA", "PA;L", 16, unpackLAL}, | ||||||
|  |     {"PA", "LA", 16, unpackLA}, | ||||||
| 
 | 
 | ||||||
|     /* true colour */ |     /* true colour */ | ||||||
|     {"RGB", "RGB", 24, ImagingUnpackRGB}, |     {"RGB", "RGB", 24, ImagingUnpackRGB}, | ||||||
|  |  | ||||||
							
								
								
									
										2
									
								
								tox.ini
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								tox.ini
									
									
									
									
									
								
							|  | @ -13,7 +13,7 @@ extras = | ||||||
|     tests |     tests | ||||||
| commands = | commands = | ||||||
|     make clean |     make clean | ||||||
|     {envpython} -m pip install --global-option="build_ext" --global-option="--inplace" . |     {envpython} -m pip install . | ||||||
|     {envpython} selftest.py |     {envpython} selftest.py | ||||||
|     {envpython} -m pytest -W always {posargs} |     {envpython} -m pytest -W always {posargs} | ||||||
| allowlist_externals = | allowlist_externals = | ||||||
|  |  | ||||||
|  | @ -152,9 +152,9 @@ deps = { | ||||||
|         "libs": [r"*.lib"], |         "libs": [r"*.lib"], | ||||||
|     }, |     }, | ||||||
|     "xz": { |     "xz": { | ||||||
|         "url": SF_PROJECTS + "/lzmautils/files/xz-5.4.2.tar.gz/download", |         "url": SF_PROJECTS + "/lzmautils/files/xz-5.4.3.tar.gz/download", | ||||||
|         "filename": "xz-5.4.2.tar.gz", |         "filename": "xz-5.4.3.tar.gz", | ||||||
|         "dir": "xz-5.4.2", |         "dir": "xz-5.4.3", | ||||||
|         "license": "COPYING", |         "license": "COPYING", | ||||||
|         "build": [ |         "build": [ | ||||||
|             *cmds_cmake("liblzma", "-DBUILD_SHARED_LIBS:BOOL=OFF"), |             *cmds_cmake("liblzma", "-DBUILD_SHARED_LIBS:BOOL=OFF"), | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue
	
	Block a user