svg-ultralight 0.34.0__py3-none-any.whl → 0.36.0__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of svg-ultralight might be problematic. Click here for more details.
- svg_ultralight/__init__.py +7 -5
- svg_ultralight/bounding_boxes/bound_helpers.py +38 -2
- svg_ultralight/constructors/__init__.py +1 -1
- svg_ultralight/constructors/new_element.py +2 -1
- svg_ultralight/image_ops.py +134 -0
- svg_ultralight/query.py +0 -1
- svg_ultralight/strings/__init__.py +1 -1
- {svg_ultralight-0.34.0.dist-info → svg_ultralight-0.36.0.dist-info}/METADATA +3 -1
- {svg_ultralight-0.34.0.dist-info → svg_ultralight-0.36.0.dist-info}/RECORD +11 -10
- {svg_ultralight-0.34.0.dist-info → svg_ultralight-0.36.0.dist-info}/WHEEL +0 -0
- {svg_ultralight-0.34.0.dist-info → svg_ultralight-0.36.0.dist-info}/top_level.txt +0 -0
svg_ultralight/__init__.py
CHANGED
|
@@ -5,13 +5,14 @@
|
|
|
5
5
|
"""
|
|
6
6
|
|
|
7
7
|
from svg_ultralight.bounding_boxes.bound_helpers import (
|
|
8
|
+
bbox_dict,
|
|
9
|
+
cut_bbox,
|
|
10
|
+
new_bbox_rect,
|
|
8
11
|
new_bbox_union,
|
|
9
12
|
new_bound_union,
|
|
10
13
|
new_element_union,
|
|
11
|
-
cut_bbox,
|
|
12
14
|
pad_bbox,
|
|
13
|
-
|
|
14
|
-
new_bbox_rect,
|
|
15
|
+
parse_bound_element,
|
|
15
16
|
)
|
|
16
17
|
from svg_ultralight.bounding_boxes.supports_bounds import SupportsBounds
|
|
17
18
|
from svg_ultralight.bounding_boxes.type_bound_collection import BoundCollection
|
|
@@ -35,10 +36,10 @@ from svg_ultralight.main import new_svg_root, write_svg
|
|
|
35
36
|
from svg_ultralight.metadata import new_metadata
|
|
36
37
|
from svg_ultralight.nsmap import NSMAP, new_qname
|
|
37
38
|
from svg_ultralight.query import (
|
|
39
|
+
clear_svg_ultralight_cache,
|
|
38
40
|
get_bounding_box,
|
|
39
41
|
get_bounding_boxes,
|
|
40
42
|
pad_text,
|
|
41
|
-
clear_svg_ultralight_cache,
|
|
42
43
|
)
|
|
43
44
|
from svg_ultralight.root_elements import new_svg_root_around_bounds
|
|
44
45
|
from svg_ultralight.string_conversion import (
|
|
@@ -55,10 +56,10 @@ from svg_ultralight.transformations import (
|
|
|
55
56
|
)
|
|
56
57
|
|
|
57
58
|
__all__ = [
|
|
59
|
+
"NSMAP",
|
|
58
60
|
"BoundCollection",
|
|
59
61
|
"BoundElement",
|
|
60
62
|
"BoundingBox",
|
|
61
|
-
"NSMAP",
|
|
62
63
|
"PaddedText",
|
|
63
64
|
"SupportsBounds",
|
|
64
65
|
"bbox_dict",
|
|
@@ -86,6 +87,7 @@ __all__ = [
|
|
|
86
87
|
"new_svg_root_around_bounds",
|
|
87
88
|
"pad_bbox",
|
|
88
89
|
"pad_text",
|
|
90
|
+
"parse_bound_element",
|
|
89
91
|
"transform_element",
|
|
90
92
|
"update_element",
|
|
91
93
|
"write_pdf",
|
|
@@ -16,7 +16,10 @@ from svg_ultralight.bounding_boxes.type_padded_text import PaddedText
|
|
|
16
16
|
from svg_ultralight.constructors import new_element
|
|
17
17
|
|
|
18
18
|
if TYPE_CHECKING:
|
|
19
|
+
import os
|
|
20
|
+
|
|
19
21
|
from svg_ultralight.bounding_boxes.supports_bounds import SupportsBounds
|
|
22
|
+
from lxml import etree
|
|
20
23
|
|
|
21
24
|
_Matrix = tuple[float, float, float, float, float, float]
|
|
22
25
|
|
|
@@ -149,8 +152,10 @@ def pad_bbox(bbox: SupportsBounds, pad: float | tuple[float, ...]) -> BoundingBo
|
|
|
149
152
|
len = 4 : 0, 1, 2, 3
|
|
150
153
|
:return: a new bounding box with padding applied.
|
|
151
154
|
"""
|
|
152
|
-
|
|
153
|
-
return cut_bbox(
|
|
155
|
+
top, right, bottom, left = _expand_pad(pad)
|
|
156
|
+
return cut_bbox(
|
|
157
|
+
bbox, x=bbox.x - left, y=bbox.y - top, x2=bbox.x2 + right, y2=bbox.y2 + bottom
|
|
158
|
+
)
|
|
154
159
|
|
|
155
160
|
|
|
156
161
|
def bbox_dict(bbox: SupportsBounds) -> dict[str, float]:
|
|
@@ -169,3 +174,34 @@ def new_bbox_rect(bbox: BoundingBox, **kwargs: float | str) -> EtreeElement:
|
|
|
169
174
|
:param kwargs: additional attributes for the rect element.
|
|
170
175
|
"""
|
|
171
176
|
return new_element("rect", **bbox_dict(bbox), **kwargs)
|
|
177
|
+
|
|
178
|
+
|
|
179
|
+
def _get_view_box(elem: EtreeElement) -> tuple[float, float, float, float]:
|
|
180
|
+
"""Return the view box of an element as a tuple of floats.
|
|
181
|
+
|
|
182
|
+
:param elem: the element from which to extract the view box.
|
|
183
|
+
:return: a tuple of floats representing the view box.
|
|
184
|
+
|
|
185
|
+
This will work on svg files created by this library and some others. Not all svg
|
|
186
|
+
files have a viewBox attribute.
|
|
187
|
+
"""
|
|
188
|
+
view_box = elem.get("viewBox")
|
|
189
|
+
if view_box is None:
|
|
190
|
+
msg = "Element does not have a viewBox attribute."
|
|
191
|
+
raise ValueError(msg)
|
|
192
|
+
x, y, width, height = map(float, view_box.split())
|
|
193
|
+
return x, y, width, height
|
|
194
|
+
|
|
195
|
+
|
|
196
|
+
def parse_bound_element(svg_fil: str | os.PathLike[str]) -> BoundElement:
|
|
197
|
+
"""Import an element as a BoundElement.
|
|
198
|
+
|
|
199
|
+
:param elem: the element to import.
|
|
200
|
+
:return: a BoundElement instance.
|
|
201
|
+
"""
|
|
202
|
+
tree = etree.parse(svg_fil)
|
|
203
|
+
root = tree.getroot()
|
|
204
|
+
elem = new_element("g")
|
|
205
|
+
elem.extend(list(root))
|
|
206
|
+
bbox = BoundingBox(*_get_view_box(root))
|
|
207
|
+
return BoundElement(elem, bbox)
|
|
@@ -20,7 +20,8 @@ from lxml import etree
|
|
|
20
20
|
from svg_ultralight.string_conversion import set_attributes
|
|
21
21
|
|
|
22
22
|
if TYPE_CHECKING:
|
|
23
|
-
from lxml.etree import QName
|
|
23
|
+
from lxml.etree import QName
|
|
24
|
+
from lxml.etree import _Element as EtreeElement # type: ignore
|
|
24
25
|
|
|
25
26
|
|
|
26
27
|
def new_element(tag: str | QName, **attributes: str | float) -> EtreeElement:
|
|
@@ -0,0 +1,134 @@
|
|
|
1
|
+
"""Crop an image before converting to binary and including in the svg file.
|
|
2
|
+
|
|
3
|
+
This optional module requires the Pillow library. Create an svg image element with a
|
|
4
|
+
rasterized image positioned inside a bounding box.
|
|
5
|
+
|
|
6
|
+
:author: Shay Hill
|
|
7
|
+
:created: 2024-11-20
|
|
8
|
+
"""
|
|
9
|
+
|
|
10
|
+
from __future__ import annotations
|
|
11
|
+
|
|
12
|
+
from typing import TYPE_CHECKING
|
|
13
|
+
|
|
14
|
+
from paragraphs import par
|
|
15
|
+
|
|
16
|
+
try:
|
|
17
|
+
from PIL import Image
|
|
18
|
+
|
|
19
|
+
if TYPE_CHECKING:
|
|
20
|
+
from PIL.Image import Image as ImageType
|
|
21
|
+
except ImportError as err:
|
|
22
|
+
msg = par(
|
|
23
|
+
"""PIL is not installed. Install it using 'pip install Pillow' to use
|
|
24
|
+
svg_ultralight.image_ops module."""
|
|
25
|
+
)
|
|
26
|
+
raise ImportError(msg) from err
|
|
27
|
+
|
|
28
|
+
import base64
|
|
29
|
+
import io
|
|
30
|
+
|
|
31
|
+
from lxml import etree
|
|
32
|
+
|
|
33
|
+
from svg_ultralight import NSMAP
|
|
34
|
+
from svg_ultralight.bounding_boxes.bound_helpers import bbox_dict
|
|
35
|
+
from svg_ultralight.constructors import new_element
|
|
36
|
+
|
|
37
|
+
if TYPE_CHECKING:
|
|
38
|
+
from pathlib import Path
|
|
39
|
+
|
|
40
|
+
from lxml.etree import (
|
|
41
|
+
_Element as EtreeElement, # pyright: ignore [reportPrivateUsage]
|
|
42
|
+
)
|
|
43
|
+
|
|
44
|
+
from svg_ultralight.bounding_boxes.type_bounding_box import BoundingBox
|
|
45
|
+
|
|
46
|
+
|
|
47
|
+
def _symmetric_crop(
|
|
48
|
+
image: ImageType, center: tuple[float, float] | None = None
|
|
49
|
+
) -> ImageType:
|
|
50
|
+
"""Crop an image symmetrically around a center point.
|
|
51
|
+
|
|
52
|
+
:param image: PIL.Image instance
|
|
53
|
+
:param center: optional center point for cropping. Proportions of image with and
|
|
54
|
+
image height, so the default value, (0.5, 0.5), is the true center of the
|
|
55
|
+
image. (0.4, 0.5) would crop 20% off the right side of the image.
|
|
56
|
+
:return: PIL.Image instance
|
|
57
|
+
"""
|
|
58
|
+
if center is None:
|
|
59
|
+
return image
|
|
60
|
+
|
|
61
|
+
if not all(0 < x < 1 for x in center):
|
|
62
|
+
msg = "Center must be between (0, 0) and (1, 1)"
|
|
63
|
+
raise ValueError(msg)
|
|
64
|
+
|
|
65
|
+
xd, yd = (min(x, 1 - x) for x in center)
|
|
66
|
+
left, right = sorted(x * image.width for x in (center[0] - xd, center[0] + xd))
|
|
67
|
+
top, bottom = sorted(x * image.height for x in (center[1] - yd, center[1] + yd))
|
|
68
|
+
|
|
69
|
+
return image.crop((left, top, right, bottom))
|
|
70
|
+
|
|
71
|
+
|
|
72
|
+
def _crop_image_to_bbox_ratio(
|
|
73
|
+
image: ImageType, bbox: BoundingBox, center: tuple[float, float] | None = None
|
|
74
|
+
) -> ImageType:
|
|
75
|
+
"""Crop an image to the ratio of a bounding box.
|
|
76
|
+
|
|
77
|
+
:param image: PIL.Image instance
|
|
78
|
+
:param bbox: BoundingBox instance
|
|
79
|
+
:param center: optional center point for cropping. Proportions of image with and
|
|
80
|
+
image height, so the default value, (0.5, 0.5), is the true center of the
|
|
81
|
+
image. (0.4, 0.5) would crop 20% off the right side of the image.
|
|
82
|
+
:return: PIL.Image instance
|
|
83
|
+
|
|
84
|
+
This crops the image to the specified ratio. It's not a resize, so it will cut
|
|
85
|
+
off the top and bottom or the sides of the image to fit the ratio.
|
|
86
|
+
"""
|
|
87
|
+
image = _symmetric_crop(image, center)
|
|
88
|
+
width, height = image.size
|
|
89
|
+
|
|
90
|
+
ratio = bbox.width / bbox.height
|
|
91
|
+
if width / height > ratio:
|
|
92
|
+
new_width = height * ratio
|
|
93
|
+
left = (width - new_width) / 2
|
|
94
|
+
right = width - left
|
|
95
|
+
return image.crop((left, 0, right, height))
|
|
96
|
+
new_height = width / ratio
|
|
97
|
+
top = (height - new_height) / 2
|
|
98
|
+
bottom = height - top
|
|
99
|
+
return image.crop((0, top, width, bottom))
|
|
100
|
+
|
|
101
|
+
|
|
102
|
+
def _get_svg_embedded_image_str(image: ImageType) -> str:
|
|
103
|
+
"""Return the string you'll need to embed an image in an svg.
|
|
104
|
+
|
|
105
|
+
:param image: PIL.Image instance
|
|
106
|
+
:return: argument for xlink:href
|
|
107
|
+
"""
|
|
108
|
+
in_mem_file = io.BytesIO()
|
|
109
|
+
image.save(in_mem_file, format="PNG")
|
|
110
|
+
_ = in_mem_file.seek(0)
|
|
111
|
+
img_bytes = in_mem_file.read()
|
|
112
|
+
base64_encoded_result_bytes = base64.b64encode(img_bytes)
|
|
113
|
+
base64_encoded_result_str = base64_encoded_result_bytes.decode("ascii")
|
|
114
|
+
return "data:image/png;base64," + base64_encoded_result_str
|
|
115
|
+
|
|
116
|
+
|
|
117
|
+
def new_image_elem_in_bbox(
|
|
118
|
+
filename: Path | str, bbox: BoundingBox, center: tuple[float, float] | None
|
|
119
|
+
) -> EtreeElement:
|
|
120
|
+
"""Create a new svg image element inside a bounding box.
|
|
121
|
+
|
|
122
|
+
:param filename: filename of source image
|
|
123
|
+
:param bbox: bounding box for the image
|
|
124
|
+
:param center: center point for cropping. Proportions of image width and image
|
|
125
|
+
height, so the default value, (0.5, 0.5), is the true center of the image.
|
|
126
|
+
(0.4, 0.5) would crop 20% off the right side of the image.
|
|
127
|
+
:return: an etree image element with the cropped image embedded
|
|
128
|
+
"""
|
|
129
|
+
image = _crop_image_to_bbox_ratio(Image.open(filename), bbox, center)
|
|
130
|
+
svg_image = new_element("image", **bbox_dict(bbox))
|
|
131
|
+
svg_image.set(
|
|
132
|
+
etree.QName(NSMAP["xlink"], "href"), _get_svg_embedded_image_str(image)
|
|
133
|
+
)
|
|
134
|
+
return svg_image
|
svg_ultralight/query.py
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
Metadata-Version: 2.1
|
|
2
2
|
Name: svg-ultralight
|
|
3
|
-
Version: 0.
|
|
3
|
+
Version: 0.36.0
|
|
4
4
|
Summary: a sensible way to create svg files with Python
|
|
5
5
|
Author-email: Shay Hill <shay_public@hotmail.com>
|
|
6
6
|
License: MIT
|
|
@@ -15,6 +15,8 @@ Requires-Dist: pytest; extra == "dev"
|
|
|
15
15
|
Requires-Dist: commitizen; extra == "dev"
|
|
16
16
|
Requires-Dist: pre-commit; extra == "dev"
|
|
17
17
|
Requires-Dist: tox; extra == "dev"
|
|
18
|
+
Provides-Extra: images
|
|
19
|
+
Requires-Dist: pillow; extra == "images"
|
|
18
20
|
|
|
19
21
|
# svg_ultralight
|
|
20
22
|
|
|
@@ -1,28 +1,29 @@
|
|
|
1
|
-
svg_ultralight/__init__.py,sha256=
|
|
1
|
+
svg_ultralight/__init__.py,sha256=wUc79mKsG6lGZ1xaYijyJ4Sm9lG5-5XgRArVsCI0niY,2554
|
|
2
2
|
svg_ultralight/animate.py,sha256=JSrBm-59BcNXDF0cGgl4-C89eBunjevZnwZxIWt48TU,1112
|
|
3
|
+
svg_ultralight/image_ops.py,sha256=PXN_p5GX91UTvhnwwU-bPuj6WzM9wCx1SqfzR5icNnQ,4686
|
|
3
4
|
svg_ultralight/inkscape.py,sha256=M8yTxXOu4NlXnhsMycvEJiIDpnDeiZ_bZakJBM38ZoU,9152
|
|
4
5
|
svg_ultralight/layout.py,sha256=FgR45FsHax4xDjGkk9HEVW4OcwhtM8aqw2JUdZs_m7Q,12326
|
|
5
6
|
svg_ultralight/main.py,sha256=6oNkZfD27UMdP-oYqp5agS_IGcYb8NkUZwM9Zdyb3SA,7287
|
|
6
7
|
svg_ultralight/metadata.py,sha256=xR3ObM0QV7OQ90IKvfigR5B6e0JW6GGVGvTlL5NswWI,4211
|
|
7
8
|
svg_ultralight/nsmap.py,sha256=y63upO78Rr-JJT56RWWZuyrsILh6HPoY4GhbYnK1A0g,1244
|
|
8
9
|
svg_ultralight/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
|
9
|
-
svg_ultralight/query.py,sha256=
|
|
10
|
+
svg_ultralight/query.py,sha256=SufFumN11VTWloqjMpSuQ5eghAYx4ABQRIMXRJGs70o,9428
|
|
10
11
|
svg_ultralight/root_elements.py,sha256=pt9J6mPrnoTAZVF6vKTZoM_o947I8UCj6MbGcD2JUCk,2869
|
|
11
12
|
svg_ultralight/string_conversion.py,sha256=WEmpf75RJmJ2lfJluagAz2wPsz6wM8XvTEwkq4U0vEc,7353
|
|
12
13
|
svg_ultralight/transformations.py,sha256=7-6NNh6xZ45mM_933fMuQfRGpI7q9Qrt5qse9e6FUek,4036
|
|
13
14
|
svg_ultralight/unit_conversion.py,sha256=g07nhzXdjPvGcJmkhLdFbeDLrSmbI8uFoVgPo7G62Bg,9258
|
|
14
15
|
svg_ultralight/bounding_boxes/__init__.py,sha256=qUEn3r4s-1QNHaguhWhhaNfdP4tl_B6YEqxtiTFuzhQ,78
|
|
15
|
-
svg_ultralight/bounding_boxes/bound_helpers.py,sha256=
|
|
16
|
+
svg_ultralight/bounding_boxes/bound_helpers.py,sha256=V0BlBHDZ9xksVG5dGLPfcyRwPP3jSj6ODdtIpqT-FcI,7273
|
|
16
17
|
svg_ultralight/bounding_boxes/supports_bounds.py,sha256=fbHV6mGdeIVV3lS15vBKSrHiHKR7DMg4K5X__4LLmCE,4569
|
|
17
18
|
svg_ultralight/bounding_boxes/type_bound_collection.py,sha256=b89TM2UsdaeApyjTQeMbx_FG_WcCiLAImEiHiZ6EWPI,2600
|
|
18
19
|
svg_ultralight/bounding_boxes/type_bound_element.py,sha256=9RdxH8osOlAvPdWR0Ww9NsasHLPYFDs-MbydoV48x4E,2239
|
|
19
20
|
svg_ultralight/bounding_boxes/type_bounding_box.py,sha256=cDrMp6uwaA--KJIQS2puG10qh8n3TBmiscg-cfk1f3w,16149
|
|
20
21
|
svg_ultralight/bounding_boxes/type_padded_text.py,sha256=QA6PfeO_sQYc5pEXuyfyQ3lRUcZAc4B2BthWXpdt3qQ,14848
|
|
21
|
-
svg_ultralight/constructors/__init__.py,sha256=
|
|
22
|
-
svg_ultralight/constructors/new_element.py,sha256=
|
|
23
|
-
svg_ultralight/strings/__init__.py,sha256=
|
|
22
|
+
svg_ultralight/constructors/__init__.py,sha256=XLOInLhzMERWNnFAs-itMs-OZrBOpvQthZJ2T5duqBE,327
|
|
23
|
+
svg_ultralight/constructors/new_element.py,sha256=8nqmOEgt3j-aOVeRaMLFHqrwKg2Dm5w0AfuK9MP4ak8,3433
|
|
24
|
+
svg_ultralight/strings/__init__.py,sha256=BMGhF1pulscIgkiYvZLr6kPRR0L4lW0jUNFxkul4_EM,295
|
|
24
25
|
svg_ultralight/strings/svg_strings.py,sha256=RYKMxOHq9abbZyGcFqsElBGLrBX-EjjNxln3s_ibi30,1296
|
|
25
|
-
svg_ultralight-0.
|
|
26
|
-
svg_ultralight-0.
|
|
27
|
-
svg_ultralight-0.
|
|
28
|
-
svg_ultralight-0.
|
|
26
|
+
svg_ultralight-0.36.0.dist-info/METADATA,sha256=uqmORGYy_BuIUjEp81PgzPeqEZMLNbZkY0_94x5NT0Y,8933
|
|
27
|
+
svg_ultralight-0.36.0.dist-info/WHEEL,sha256=PZUExdf71Ui_so67QXpySuHtCi3-J3wvF4ORK6k_S8U,91
|
|
28
|
+
svg_ultralight-0.36.0.dist-info/top_level.txt,sha256=se-6yqM_0Yg5orJKvKWdjQZ4iR4G_EjhL7oRgju-fdY,15
|
|
29
|
+
svg_ultralight-0.36.0.dist-info/RECORD,,
|
|
File without changes
|
|
File without changes
|