Pillow/src/PIL/ImageShow.py

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

348 lines
9.2 KiB
Python
Raw Normal View History

2010-07-31 06:52:47 +04:00
#
# The Python Imaging Library.
# $Id$
#
# im.show() drivers
#
# History:
# 2008-04-06 fl Created
#
# Copyright (c) Secret Labs AB 2008.
#
# See the README file for information on usage and redistribution.
#
from __future__ import annotations
2024-01-13 06:40:59 +03:00
import abc
2010-07-31 06:52:47 +04:00
import os
import shutil
import subprocess
import sys
2019-09-26 15:12:28 +03:00
from shlex import quote
2024-01-13 06:40:59 +03:00
from typing import Any
2010-07-31 06:52:47 +04:00
from . import Image
2010-07-31 06:52:47 +04:00
_viewers = []
2014-08-26 17:47:10 +04:00
2024-01-13 06:40:59 +03:00
def register(viewer, order: int = 1) -> None:
2020-06-14 13:00:23 +03:00
"""
2022-02-21 12:32:21 +03:00
The :py:func:`register` function is used to register additional viewers::
from PIL import ImageShow
ImageShow.register(MyViewer()) # MyViewer will be used as a last resort
ImageShow.register(MySecondViewer(), 0) # MySecondViewer will be prioritised
ImageShow.register(ImageShow.XVViewer(), 0) # XVViewer will be prioritised
2020-06-14 13:00:23 +03:00
:param viewer: The viewer to be registered.
:param order:
2020-06-20 14:10:10 +03:00
Zero or a negative integer to prepend this viewer to the list,
a positive integer to append it.
2020-06-14 13:00:23 +03:00
"""
2010-07-31 06:52:47 +04:00
try:
if issubclass(viewer, Viewer):
viewer = viewer()
except TypeError:
pass # raised if viewer wasn't a class
if order > 0:
_viewers.append(viewer)
else:
2010-07-31 06:52:47 +04:00
_viewers.insert(0, viewer)
2014-08-26 17:47:10 +04:00
2024-01-13 06:40:59 +03:00
def show(image: Image.Image, title: str | None = None, **options: Any) -> bool:
r"""
2016-05-24 10:36:14 +03:00
Display a given image.
:param image: An image object.
2020-06-14 13:00:23 +03:00
:param title: Optional title. Not all viewers can display the title.
:param \**options: Additional viewer options.
2020-06-14 13:00:23 +03:00
:returns: ``True`` if a suitable viewer was found, ``False`` otherwise.
2016-05-24 10:36:14 +03:00
"""
2010-07-31 06:52:47 +04:00
for viewer in _viewers:
if viewer.show(image, title=title, **options):
return True
return False
2010-07-31 06:52:47 +04:00
2014-08-26 17:47:10 +04:00
class Viewer:
2016-05-24 10:36:14 +03:00
"""Base class for viewers."""
2010-07-31 06:52:47 +04:00
# main api
2024-01-13 06:40:59 +03:00
def show(self, image: Image.Image, **options: Any) -> int:
2020-06-14 13:00:23 +03:00
"""
The main function for displaying an image.
Converts the given image to the target format and displays it.
"""
2010-07-31 06:52:47 +04:00
if not (
image.mode in ("1", "RGBA")
or (self.format == "PNG" and image.mode in ("I;16", "LA"))
):
2010-07-31 06:52:47 +04:00
base = Image.getmodebase(image.mode)
if image.mode != base:
image = image.convert(base)
2010-07-31 06:52:47 +04:00
return self.show_image(image, **options)
2010-07-31 06:52:47 +04:00
# hook methods
2024-01-13 06:40:59 +03:00
format: str | None = None
2020-06-14 13:00:23 +03:00
"""The format to convert the image into."""
2024-01-13 06:40:59 +03:00
options: dict[str, Any] = {}
2020-06-14 13:00:23 +03:00
"""Additional options used to convert the image."""
2010-07-31 06:52:47 +04:00
2024-01-13 06:40:59 +03:00
def get_format(self, image: Image.Image) -> str | None:
2020-06-14 13:00:23 +03:00
"""Return format name, or ``None`` to save as PGM/PPM."""
2010-07-31 06:52:47 +04:00
return self.format
2024-01-13 06:40:59 +03:00
def get_command(self, file: str, **options: Any) -> str:
2020-06-14 13:00:23 +03:00
"""
Returns the command used to display the file.
Not implemented in the base class.
"""
2023-10-19 10:42:41 +03:00
msg = "unavailable in base viewer"
raise NotImplementedError(msg)
2010-07-31 06:52:47 +04:00
2024-01-13 06:40:59 +03:00
def save_image(self, image: Image.Image) -> str:
2020-06-14 13:00:23 +03:00
"""Save to temporary file and return filename."""
2017-05-13 16:08:08 +03:00
return image._dump(format=self.get_format(image), **self.options)
2010-07-31 06:52:47 +04:00
2024-01-13 06:40:59 +03:00
def show_image(self, image: Image.Image, **options: Any) -> int:
2020-06-14 13:00:23 +03:00
"""Display the given image."""
2010-07-31 06:52:47 +04:00
return self.show_file(self.save_image(image), **options)
2024-01-13 06:40:59 +03:00
def show_file(self, path: str, **options: Any) -> int:
2022-01-16 03:38:34 +03:00
"""
Display given file.
"""
2022-12-13 19:32:55 +03:00
os.system(self.get_command(path, **options)) # nosec
2010-07-31 06:52:47 +04:00
return 1
2019-03-21 16:28:20 +03:00
2010-07-31 06:52:47 +04:00
# --------------------------------------------------------------------
2017-11-22 13:13:13 +03:00
2020-06-14 13:00:23 +03:00
class WindowsViewer(Viewer):
"""The default viewer on Windows is the default system application for PNG files."""
2010-07-31 06:52:47 +04:00
2020-06-14 13:00:23 +03:00
format = "PNG"
2022-09-23 13:14:05 +03:00
options = {"compress_level": 1, "save_all": True}
2020-06-14 13:00:23 +03:00
2024-01-13 06:40:59 +03:00
def get_command(self, file: str, **options: Any) -> str:
2020-06-14 13:00:23 +03:00
return (
f'start "Pillow" /WAIT "{file}" '
"&& ping -n 4 127.0.0.1 >NUL "
f'&& del /f "{file}"'
2020-06-14 13:00:23 +03:00
)
2014-08-26 17:47:10 +04:00
def show_file(self, path: str, **options: Any) -> int:
"""
Display given file.
"""
subprocess.Popen(
self.get_command(path, **options),
shell=True,
creationflags=getattr(subprocess, "CREATE_NO_WINDOW"),
) # nosec
return 1
2010-07-31 06:52:47 +04:00
2020-06-14 13:00:23 +03:00
if sys.platform == "win32":
2010-07-31 06:52:47 +04:00
register(WindowsViewer)
2020-06-14 13:00:23 +03:00
class MacViewer(Viewer):
2021-10-02 02:50:27 +03:00
"""The default viewer on macOS using ``Preview.app``."""
2020-06-14 13:00:23 +03:00
format = "PNG"
2022-09-23 13:14:05 +03:00
options = {"compress_level": 1, "save_all": True}
2014-08-26 17:47:10 +04:00
2024-01-13 06:40:59 +03:00
def get_command(self, file: str, **options: Any) -> str:
2020-06-14 13:00:23 +03:00
# on darwin open returns immediately resulting in the temp
# file removal while app is opening
command = "open -a Preview.app"
command = f"({command} {quote(file)}; sleep 20; rm -f {quote(file)})&"
2020-06-14 13:00:23 +03:00
return command
2024-01-13 06:40:59 +03:00
def show_file(self, path: str, **options: Any) -> int:
2022-01-16 03:38:34 +03:00
"""
Display given file.
"""
subprocess.call(["open", "-a", "Preview.app", path])
2022-07-06 01:30:24 +03:00
executable = sys.executable or shutil.which("python3")
2022-07-05 06:42:41 +03:00
if executable:
subprocess.Popen(
[
2022-07-05 06:42:41 +03:00
executable,
"-c",
"import os, sys, time; time.sleep(20); os.remove(sys.argv[1])",
path,
]
)
2020-06-14 13:00:23 +03:00
return 1
2020-06-14 13:00:23 +03:00
if sys.platform == "darwin":
2010-07-31 06:52:47 +04:00
register(MacViewer)
2020-06-14 13:00:23 +03:00
class UnixViewer(Viewer):
format = "PNG"
2022-09-23 13:14:05 +03:00
options = {"compress_level": 1, "save_all": True}
2010-07-31 06:52:47 +04:00
2024-01-13 06:40:59 +03:00
@abc.abstractmethod
def get_command_ex(self, file: str, **options: Any) -> tuple[str, str]:
2024-02-10 14:37:42 +03:00
pass
2024-01-13 06:40:59 +03:00
def get_command(self, file: str, **options: Any) -> str:
2020-06-14 13:00:23 +03:00
command = self.get_command_ex(file, **options)[0]
return f"({command} {quote(file)}"
class XDGViewer(UnixViewer):
"""
The freedesktop.org ``xdg-open`` command.
"""
2024-01-13 06:40:59 +03:00
def get_command_ex(self, file: str, **options: Any) -> tuple[str, str]:
command = executable = "xdg-open"
return command, executable
2024-01-13 06:40:59 +03:00
def show_file(self, path: str, **options: Any) -> int:
2022-01-16 03:38:34 +03:00
"""
Display given file.
"""
subprocess.Popen(["xdg-open", path])
2020-06-14 13:00:23 +03:00
return 1
2010-07-31 06:52:47 +04:00
2020-06-14 13:00:23 +03:00
class DisplayViewer(UnixViewer):
2021-10-23 11:51:46 +03:00
"""
The ImageMagick ``display`` command.
This viewer supports the ``title`` parameter.
"""
2010-07-31 06:52:47 +04:00
2024-01-13 06:40:59 +03:00
def get_command_ex(
self, file: str, title: str | None = None, **options: Any
) -> tuple[str, str]:
2020-06-14 13:00:23 +03:00
command = executable = "display"
2021-10-23 07:53:08 +03:00
if title:
2022-02-09 08:21:47 +03:00
command += f" -title {quote(title)}"
2020-06-14 13:00:23 +03:00
return command, executable
2017-11-22 13:13:13 +03:00
2024-01-13 06:40:59 +03:00
def show_file(self, path: str, **options: Any) -> int:
"""
Display given file.
"""
args = ["display"]
2022-03-15 00:31:17 +03:00
title = options.get("title")
if title:
args += ["-title", title]
args.append(path)
subprocess.Popen(args)
return 1
class GmDisplayViewer(UnixViewer):
"""The GraphicsMagick ``gm display`` command."""
2024-01-13 06:40:59 +03:00
def get_command_ex(self, file: str, **options: Any) -> tuple[str, str]:
executable = "gm"
command = "gm display"
return command, executable
2024-01-13 06:40:59 +03:00
def show_file(self, path: str, **options: Any) -> int:
"""
Display given file.
"""
subprocess.Popen(["gm", "display", path])
return 1
2020-06-14 13:00:23 +03:00
class EogViewer(UnixViewer):
"""The GNOME Image Viewer ``eog`` command."""
2024-01-13 06:40:59 +03:00
def get_command_ex(self, file: str, **options: Any) -> tuple[str, str]:
executable = "eog"
command = "eog -n"
2020-06-14 13:00:23 +03:00
return command, executable
2010-07-31 06:52:47 +04:00
2024-01-13 06:40:59 +03:00
def show_file(self, path: str, **options: Any) -> int:
"""
Display given file.
"""
subprocess.Popen(["eog", "-n", path])
return 1
2010-07-31 06:52:47 +04:00
2020-06-14 13:00:23 +03:00
class XVViewer(UnixViewer):
"""
The X Viewer ``xv`` command.
This viewer supports the ``title`` parameter.
"""
2024-01-13 06:40:59 +03:00
def get_command_ex(
self, file: str, title: str | None = None, **options: Any
) -> tuple[str, str]:
2020-06-14 13:00:23 +03:00
# note: xv is pretty outdated. most modern systems have
# imagemagick's display command instead.
command = executable = "xv"
if title:
command += f" -name {quote(title)}"
2020-06-14 13:00:23 +03:00
return command, executable
2024-01-13 06:40:59 +03:00
def show_file(self, path: str, **options: Any) -> int:
"""
Display given file.
"""
args = ["xv"]
2022-03-15 00:31:17 +03:00
title = options.get("title")
if title:
args += ["-name", title]
args.append(path)
subprocess.Popen(args)
return 1
2020-06-14 13:00:23 +03:00
if sys.platform not in ("win32", "darwin"): # unixoids
if shutil.which("xdg-open"):
register(XDGViewer)
2020-06-14 13:00:23 +03:00
if shutil.which("display"):
register(DisplayViewer)
if shutil.which("gm"):
register(GmDisplayViewer)
2020-06-14 13:00:23 +03:00
if shutil.which("eog"):
register(EogViewer)
if shutil.which("xv"):
2010-07-31 06:52:47 +04:00
register(XVViewer)
2021-03-04 00:55:24 +03:00
class IPythonViewer(Viewer):
2021-03-04 00:56:49 +03:00
"""The viewer for IPython frontends."""
2024-01-13 06:40:59 +03:00
def show_image(self, image: Image.Image, **options: Any) -> int:
2021-03-04 00:56:03 +03:00
ipython_display(image)
2021-03-04 00:55:24 +03:00
return 1
try:
2021-03-04 00:56:03 +03:00
from IPython.display import display as ipython_display
2021-03-04 00:55:24 +03:00
except ImportError:
pass
else:
register(IPythonViewer)
2010-07-31 06:52:47 +04:00
if __name__ == "__main__":
2018-01-06 13:51:45 +03:00
if len(sys.argv) < 2:
2021-05-08 05:37:06 +03:00
print("Syntax: python3 ImageShow.py imagefile [title]")
2018-01-06 13:51:45 +03:00
sys.exit()
2020-02-18 12:49:05 +03:00
with Image.open(sys.argv[1]) as im:
print(show(im, *sys.argv[2:]))