-
-
Notifications
You must be signed in to change notification settings - Fork 2.5k
Expand file tree
/
Copy pathCurImagePlugin.py
More file actions
103 lines (82 loc) · 2.65 KB
/
Copy pathCurImagePlugin.py
File metadata and controls
103 lines (82 loc) · 2.65 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
#
# The Python Imaging Library.
# $Id$
#
# Windows Cursor support for PIL
#
# notes:
# uses BmpImagePlugin.py to read the bitmap data.
#
# history:
# 96-05-27 fl Created
#
# Copyright (c) Secret Labs AB 1997.
# Copyright (c) Fredrik Lundh 1996.
#
# See the README file for information on usage and redistribution.
#
from __future__ import annotations
from . import BmpImagePlugin, Image, ImageOps
from ._binary import i16le as i16
from ._binary import i32le as i32
#
# --------------------------------------------------------------------
def _accept(prefix: bytes) -> bool:
return prefix.startswith(b"\0\0\2\0")
##
# Image plugin for Windows Cursor files.
class CurImageFile(BmpImagePlugin.BmpImageFile):
format = "CUR"
format_description = "Windows Cursor"
def _open(self) -> None:
assert self.fp is not None
offset = self.fp.tell()
# check magic
s = self.fp.read(6)
if not _accept(s):
msg = "not a CUR file"
raise SyntaxError(msg)
# pick the largest cursor in the file
m = b""
for i in range(i16(s, 4)):
s = self.fp.read(16)
if not m:
m = s
elif s[0] > m[0] and s[1] > m[1]:
m = s
if not m:
msg = "No cursors were found"
raise TypeError(msg)
# load as bitmap
self._bitmap(i32(m, 12) + offset)
self._masked = self.mode in ("1", "L")
if self._masked:
self._rawmode = self.mode
self._mode = "LA"
# patch up the bitmap height
self._size = self.width, self.height // 2
if not self._masked:
self.tile = [self.tile[0]._replace(extents=(0, 0) + self.size)]
def load_prepare(self) -> None:
if self._masked:
self._mode = self._rawmode
self._size = self.width, self.height * 2
super().load_prepare()
def load_end(self) -> None:
if not self._masked:
return
self._mode = "LA"
new_height = self.height // 2
and_mask = self.im.crop((0, 0, self.width, new_height))
xor_mask = self.im.crop((0, new_height, self.width, self.height))
self._size = self.width, new_height
self._im = Image.core.fill(self.mode, self.size)
self._im.paste(
xor_mask.convert(self.mode),
(0, 0) + self.size,
ImageOps.invert(Image.Image()._new(and_mask)).im,
)
#
# --------------------------------------------------------------------
Image.register_open(CurImageFile.format, CurImageFile, _accept)
Image.register_extension(CurImageFile.format, ".cur")