2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# The Python Imaging Library
|
|
|
|
# $Id$
|
|
|
|
#
|
2020-05-03 06:06:25 +03:00
|
|
|
# screen grabber
|
2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# History:
|
|
|
|
# 2001-04-26 fl created
|
|
|
|
# 2001-09-17 fl use builtin driver, if present
|
|
|
|
# 2002-11-19 fl added grabclipboard support
|
|
|
|
#
|
|
|
|
# Copyright (c) 2001-2002 by Secret Labs AB
|
|
|
|
# Copyright (c) 2001-2002 by Fredrik Lundh
|
|
|
|
#
|
|
|
|
# See the README file for information on usage and redistribution.
|
|
|
|
#
|
2023-12-21 14:13:31 +03:00
|
|
|
from __future__ import annotations
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2023-06-05 11:36:41 +03:00
|
|
|
import io
|
2022-06-12 09:00:31 +03:00
|
|
|
import os
|
|
|
|
import shutil
|
|
|
|
import subprocess
|
2014-09-16 17:05:22 +04:00
|
|
|
import sys
|
2022-06-12 09:00:31 +03:00
|
|
|
import tempfile
|
2019-06-11 11:42:05 +03:00
|
|
|
|
2019-07-06 23:40:53 +03:00
|
|
|
from . import Image
|
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2019-09-19 19:57:59 +03:00
|
|
|
def grab(bbox=None, include_layered_windows=False, all_screens=False, xdisplay=None):
|
|
|
|
if xdisplay is None:
|
|
|
|
if sys.platform == "darwin":
|
|
|
|
fh, filepath = tempfile.mkstemp(".png")
|
|
|
|
os.close(fh)
|
2022-03-20 04:59:10 +03:00
|
|
|
args = ["screencapture"]
|
|
|
|
if bbox:
|
|
|
|
left, top, right, bottom = bbox
|
2022-04-26 12:04:08 +03:00
|
|
|
args += ["-R", f"{left},{top},{right-left},{bottom-top}"]
|
2022-03-20 04:59:10 +03:00
|
|
|
subprocess.call(args + ["-x", filepath])
|
2019-09-19 19:57:59 +03:00
|
|
|
im = Image.open(filepath)
|
|
|
|
im.load()
|
|
|
|
os.unlink(filepath)
|
|
|
|
if bbox:
|
2022-03-20 04:59:10 +03:00
|
|
|
im_resized = im.resize((right - left, bottom - top))
|
2019-09-19 19:57:59 +03:00
|
|
|
im.close()
|
2022-03-20 04:59:10 +03:00
|
|
|
return im_resized
|
2019-09-19 19:57:59 +03:00
|
|
|
return im
|
|
|
|
elif sys.platform == "win32":
|
2019-12-09 17:55:15 +03:00
|
|
|
offset, size, data = Image.core.grabscreen_win32(
|
|
|
|
include_layered_windows, all_screens
|
|
|
|
)
|
2019-09-19 19:57:59 +03:00
|
|
|
im = Image.frombytes(
|
|
|
|
"RGB",
|
|
|
|
size,
|
|
|
|
data,
|
|
|
|
# RGB, 32-bit line padding, origin lower left corner
|
|
|
|
"raw",
|
|
|
|
"BGR",
|
|
|
|
(size[0] * 3 + 3) & -4,
|
|
|
|
-1,
|
|
|
|
)
|
|
|
|
if bbox:
|
|
|
|
x0, y0 = offset
|
|
|
|
left, top, right, bottom = bbox
|
|
|
|
im = im.crop((left - x0, top - y0, right - x0, bottom - y0))
|
|
|
|
return im
|
2023-05-09 00:30:11 +03:00
|
|
|
try:
|
|
|
|
if not Image.core.HAVE_XCB:
|
|
|
|
msg = "Pillow was built without XCB support"
|
|
|
|
raise OSError(msg)
|
|
|
|
size, data = Image.core.grabscreen_x11(xdisplay)
|
|
|
|
except OSError:
|
2023-05-10 06:53:55 +03:00
|
|
|
if (
|
|
|
|
xdisplay is None
|
|
|
|
and sys.platform not in ("darwin", "win32")
|
|
|
|
and shutil.which("gnome-screenshot")
|
|
|
|
):
|
2022-06-12 09:00:31 +03:00
|
|
|
fh, filepath = tempfile.mkstemp(".png")
|
|
|
|
os.close(fh)
|
|
|
|
subprocess.call(["gnome-screenshot", "-f", filepath])
|
|
|
|
im = Image.open(filepath)
|
|
|
|
im.load()
|
|
|
|
os.unlink(filepath)
|
|
|
|
if bbox:
|
|
|
|
im_cropped = im.crop(bbox)
|
|
|
|
im.close()
|
|
|
|
return im_cropped
|
|
|
|
return im
|
2023-05-09 00:30:11 +03:00
|
|
|
else:
|
|
|
|
raise
|
2023-05-11 13:08:10 +03:00
|
|
|
else:
|
|
|
|
im = Image.frombytes("RGB", size, data, "raw", "BGRX", size[0] * 4, 1)
|
|
|
|
if bbox:
|
|
|
|
im = im.crop(bbox)
|
|
|
|
return im
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
|
|
|
|
def grabclipboard():
|
2015-08-01 10:44:13 +03:00
|
|
|
if sys.platform == "darwin":
|
2023-06-17 14:01:52 +03:00
|
|
|
fh, filepath = tempfile.mkstemp(".png")
|
2016-04-19 10:39:59 +03:00
|
|
|
os.close(fh)
|
2016-04-18 14:40:52 +03:00
|
|
|
commands = [
|
2018-10-24 22:24:03 +03:00
|
|
|
'set theFile to (open for access POSIX file "'
|
|
|
|
+ filepath
|
|
|
|
+ '" with write permission)',
|
2016-04-18 14:40:52 +03:00
|
|
|
"try",
|
2023-06-17 13:55:58 +03:00
|
|
|
" write (the clipboard as «class PNGf») to theFile",
|
2016-04-18 14:40:52 +03:00
|
|
|
"end try",
|
|
|
|
"close access theFile",
|
|
|
|
]
|
|
|
|
script = ["osascript"]
|
|
|
|
for command in commands:
|
|
|
|
script += ["-e", command]
|
|
|
|
subprocess.call(script)
|
|
|
|
|
|
|
|
im = None
|
2016-04-19 10:39:59 +03:00
|
|
|
if os.stat(filepath).st_size != 0:
|
|
|
|
im = Image.open(filepath)
|
2016-04-18 14:40:52 +03:00
|
|
|
im.load()
|
2016-04-19 10:39:59 +03:00
|
|
|
os.unlink(filepath)
|
2016-04-18 14:40:52 +03:00
|
|
|
return im
|
2019-09-19 19:57:59 +03:00
|
|
|
elif sys.platform == "win32":
|
2020-05-09 10:29:40 +03:00
|
|
|
fmt, data = Image.core.grabclipboard_win32()
|
2020-05-09 11:40:10 +03:00
|
|
|
if fmt == "file": # CF_HDROP
|
|
|
|
import struct
|
|
|
|
|
|
|
|
o = struct.unpack_from("I", data)[0]
|
|
|
|
if data[16] != 0:
|
|
|
|
files = data[o:].decode("utf-16le").split("\0")
|
|
|
|
else:
|
|
|
|
files = data[o:].decode("mbcs").split("\0")
|
2020-05-28 14:07:53 +03:00
|
|
|
return files[: files.index("")]
|
2016-04-18 14:40:52 +03:00
|
|
|
if isinstance(data, bytes):
|
2020-05-09 10:29:40 +03:00
|
|
|
data = io.BytesIO(data)
|
|
|
|
if fmt == "png":
|
|
|
|
from . import PngImagePlugin
|
|
|
|
|
|
|
|
return PngImagePlugin.PngImageFile(data)
|
|
|
|
elif fmt == "DIB":
|
|
|
|
from . import BmpImagePlugin
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2020-05-09 10:29:40 +03:00
|
|
|
return BmpImagePlugin.DibImageFile(data)
|
|
|
|
return None
|
2019-12-09 17:55:15 +03:00
|
|
|
else:
|
2023-08-12 07:51:09 +03:00
|
|
|
if os.getenv("WAYLAND_DISPLAY"):
|
2023-08-12 13:33:36 +03:00
|
|
|
session_type = "wayland"
|
2023-08-12 07:51:09 +03:00
|
|
|
elif os.getenv("DISPLAY"):
|
2023-08-12 13:33:36 +03:00
|
|
|
session_type = "x11"
|
2023-08-11 13:42:09 +03:00
|
|
|
else: # Session type check failed
|
2023-08-12 13:33:36 +03:00
|
|
|
session_type = None
|
2023-08-11 12:47:36 +03:00
|
|
|
|
2023-08-22 03:31:28 +03:00
|
|
|
if shutil.which("wl-paste") and session_type in ("wayland", None):
|
2024-01-22 17:19:59 +03:00
|
|
|
args = ["wl-paste", "-t", "image"]
|
2023-08-22 03:31:28 +03:00
|
|
|
elif shutil.which("xclip") and session_type in ("x11", None):
|
2022-12-07 14:01:37 +03:00
|
|
|
args = ["xclip", "-selection", "clipboard", "-t", "image/png", "-o"]
|
|
|
|
else:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "wl-paste or xclip is required for ImageGrab.grabclipboard() on Linux"
|
|
|
|
raise NotImplementedError(msg)
|
2023-08-11 12:47:36 +03:00
|
|
|
|
2023-09-09 13:07:12 +03:00
|
|
|
p = subprocess.run(args, capture_output=True)
|
2024-01-22 17:19:59 +03:00
|
|
|
if p.returncode != 0:
|
2024-01-27 14:54:01 +03:00
|
|
|
err = p.stderr
|
|
|
|
for silent_error in [
|
2024-01-27 12:15:10 +03:00
|
|
|
# wl-paste, when the clipboard is empty
|
|
|
|
b"Nothing is copied",
|
|
|
|
# wl-paste/debian xclip, when an image isn't available
|
2024-01-28 08:31:03 +03:00
|
|
|
b" not available",
|
2024-01-27 12:15:10 +03:00
|
|
|
# xclip, when an image isn't available
|
2024-01-28 08:31:03 +03:00
|
|
|
b"cannot convert ",
|
2024-01-27 12:15:10 +03:00
|
|
|
# xclip, when the clipboard isn't initialized
|
2024-01-28 08:31:03 +03:00
|
|
|
b"xclip: Error: There is no owner for the ",
|
2024-01-27 14:54:01 +03:00
|
|
|
]:
|
|
|
|
if err in silent_error:
|
|
|
|
return None
|
2024-01-27 14:08:16 +03:00
|
|
|
msg = f"{args[0]} error"
|
|
|
|
if err:
|
|
|
|
msg += f": {err.strip().decode()}"
|
2023-04-24 14:03:27 +03:00
|
|
|
raise ChildProcessError(msg)
|
2024-01-22 17:19:59 +03:00
|
|
|
|
2023-06-05 11:36:41 +03:00
|
|
|
data = io.BytesIO(p.stdout)
|
|
|
|
im = Image.open(data)
|
2022-12-07 13:33:09 +03:00
|
|
|
im.load()
|
|
|
|
return im
|