Module fpdf.drawing
Vector drawing: managing colors, graphics states, paths, transforms…
The contents of this module are internal to fpdf2, and not part of the public API. They may change at any time without prior warning or any deprecation period, in non-backward-compatible ways.
Usage documentation at: https://py-pdf.github.io/fpdf2/Drawing.html
Functions
def render_pdf_primitive(primitive)
-
Expand source code Browse git
def render_pdf_primitive(primitive): """ Render a Python value as a PDF primitive type. Container types (tuples/lists and dicts) are rendered recursively. This supports values of the type Name, str, bytes, numbers, booleans, list/tuple, and dict. Any custom type can be passed in as long as it provides a `serialize` method that takes no arguments and returns a string. The primitive object is returned directly if it is an instance of the `Raw` class. Otherwise, The existence of the `serialize` method is checked before any other type checking is performed, so, for example, a `dict` subclass with a `serialize` method would be converted using its `pdf_repr` method rather than the built-in `dict` conversion process. Args: primitive: the primitive value to convert to its PDF representation. Returns: Raw-wrapped str of the PDF representation. Raises: ValueError: if a dictionary key is not a Name. TypeError: if `primitive` does not have a known conversion to a PDF representation. """ if isinstance(primitive, Raw): return primitive if callable(getattr(primitive, "serialize", None)): output = primitive.serialize() elif primitive is None: output = "null" elif isinstance(primitive, str): output = f"({escape_parens(primitive)})" elif isinstance(primitive, bytes): output = f"<{primitive.hex()}>" elif isinstance(primitive, bool): # has to come before number check output = ["false", "true"][primitive] elif isinstance(primitive, NumberClass): output = number_to_str(primitive) elif isinstance(primitive, (list, tuple)): output = "[" + " ".join(render_pdf_primitive(val) for val in primitive) + "]" elif isinstance(primitive, dict): item_list = [] for key, val in primitive.items(): if not isinstance(key, Name): raise ValueError("dict keys must be Names") item_list.append( render_pdf_primitive(key) + " " + render_pdf_primitive(val) ) output = "<< " + "\n".join(item_list) + " >>" else: raise TypeError(f"cannot produce PDF representation for value {primitive!r}") return Raw(output)
Render a Python value as a PDF primitive type.
Container types (tuples/lists and dicts) are rendered recursively. This supports values of the type Name, str, bytes, numbers, booleans, list/tuple, and dict.
Any custom type can be passed in as long as it provides a
serialize
method that takes no arguments and returns a string. The primitive object is returned directly if it is an instance of theRaw
class. Otherwise, The existence of theserialize
method is checked before any other type checking is performed, so, for example, adict
subclass with aserialize
method would be converted using itspdf_repr
method rather than the built-indict
conversion process.Args
primitive
- the primitive value to convert to its PDF representation.
Returns
Raw-wrapped str of the PDF representation.
Raises
ValueError
- if a dictionary key is not a Name.
TypeError
- if
primitive
does not have a known conversion to a PDF representation.
Classes
class Arc (radii: Point,
rotation: int | float | decimal.Decimal,
large: bool,
sweep: bool,
end: Point)-
Expand source code Browse git
class Arc(NamedTuple): """ An elliptical arc path element. The arc is drawn from the end of the current path element to its specified end point using a number of parameters to determine how it is constructed. See: `PaintedPath.arc_to` """ radii: Point """ The x- and y-radii of the arc. If `radii.x == radii.y` the arc will be circular. """ rotation: Number """The rotation of the arc's major/minor axes relative to the coordinate frame.""" large: bool """If True, sweep the arc over an angle greater than or equal to 180 degrees.""" sweep: bool """If True, the arc is swept in the positive angular direction.""" end: Point """The end point of the arc.""" @staticmethod @force_nodocument def subdivide_sweep( sweep_angle: float, ) -> Generator[tuple[Point, Point, Point], None, None]: """ A generator that subdivides a swept angle into segments no larger than a quarter turn. Any sweep that is larger than a quarter turn is subdivided into as many equally sized segments as necessary to prevent any individual segment from being larger than a quarter turn. This is used for approximating a circular curve segment using cubic Bézier curves. This computes the parameters used for the Bézier approximation up front, as well as the transform necessary to place the segment in the correct position. Args: sweep_angle (float): the angle to subdivide. Yields: A tuple of (ctrl1, ctrl2, end) representing the control and end points of the cubic Bézier curve approximating the segment as a unit circle centered at the origin. """ sweep_angle = abs(sweep_angle) sweep_left = sweep_angle quarterturn = math.pi / 2 chunks = math.ceil(sweep_angle / quarterturn) sweep_segment = sweep_angle / chunks cos_t = math.cos(sweep_segment) sin_t = math.sin(sweep_segment) kappa = 4 / 3 * math.tan(sweep_segment / 4) ctrl1 = Point(1, kappa) ctrl2 = Point(cos_t + kappa * sin_t, sin_t - kappa * cos_t) end = Point(cos_t, sin_t) for _ in range(chunks): offset = sweep_angle - sweep_left transform = Transform.rotation(offset) yield ctrl1 @ transform, ctrl2 @ transform, end @ transform sweep_left -= sweep_segment def _approximate_arc(self, last_item: Renderable) -> list[BezierCurve]: """ Approximate this arc with a sequence of `BezierCurve`. Args: last_item: the previous path element (used for its end point) Returns: a list of `BezierCurve`. """ radii = self.radii reverse = Transform.rotation(-self.rotation) forward = Transform.rotation(self.rotation) prime = ((last_item.end_point - self.end) * 0.5) @ reverse lam_da = (prime.x / radii.x) ** 2 + (prime.y / radii.y) ** 2 if lam_da > 1: radii = Point(x=(lam_da**0.5) * radii.x, y=(lam_da**0.5) * radii.y) sign = (self.large != self.sweep) - (self.large == self.sweep) rxry2 = (radii.x * radii.y) ** 2 rxpy2 = (radii.x * prime.y) ** 2 rypx2 = (radii.y * prime.x) ** 2 centerprime = ( sign * math.sqrt(round(rxry2 - rxpy2 - rypx2, 8) / (rxpy2 + rypx2)) * Point( x=radii.x * prime.y / radii.y, y=-radii.y * prime.x / radii.x, ) ) center = (centerprime @ forward) + ((last_item.end_point + self.end) * 0.5) arcstart = Point( x=(prime.x - centerprime.x) / radii.x, y=(prime.y - centerprime.y) / radii.y, ) arcend = Point( x=(-prime.x - centerprime.x) / radii.x, y=(-prime.y - centerprime.y) / radii.y, ) theta = Point(1, 0).angle(arcstart) deltatheta = arcstart.angle(arcend) if (self.sweep is False) and (deltatheta > 0): deltatheta -= math.tau elif (self.sweep is True) and (deltatheta < 0): deltatheta += math.tau sweep_sign = (deltatheta >= 0) - (deltatheta < 0) final_tf = ( Transform.scaling(x=1, y=sweep_sign) # flip negative sweeps .rotate(theta) # rotate start of arc to correct position .scale(radii.x, radii.y) # scale unit circle into the final ellipse shape .rotate(self.rotation) # rotate the ellipse the specified angle .translate(center.x, center.y) # translate to the final coordinates ) curves = [] for ctrl1, ctrl2, end in self.subdivide_sweep(deltatheta): curves.append( BezierCurve(ctrl1 @ final_tf, ctrl2 @ final_tf, end @ final_tf) ) return curves def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this arc by approximating it with a series of Bezier curves and aggregating their bounding boxes. """ bbox = BoundingBox.empty() prev = Move(start) for curve in self._approximate_arc(prev): segment_bbox, _ = curve.bounding_box(prev.end_point) bbox = bbox.merge(segment_bbox) prev = curve return bbox, self.end @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `BezierCurve`. """ curves = self._approximate_arc(last_item) if not curves: return "", last_item, initial_point return ( " ".join( curve.render(resource_registry, style, prev, initial_point)[0] for prev, curve in zip([last_item, *curves[:-1]], curves) ), curves[-1], initial_point, ) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Arc.render`. """ curves = self._approximate_arc(last_item) debug_stream.write(f"{self} resolved to:\n") if not curves: debug_stream.write(pfx + " └─ nothing\n") return "", last_item, initial_point previous = [last_item] for curve in curves[:-1]: previous.append(curve) debug_stream.write(pfx + f" ├─ {curve}\n") debug_stream.write(pfx + f" └─ {curves[-1]}\n") return ( " ".join( curve.render(resource_registry, style, prev, initial_point)[0] for prev, curve in zip(previous, curves) ), curves[-1], initial_point, )
An elliptical arc path element.
The arc is drawn from the end of the current path element to its specified end point using a number of parameters to determine how it is constructed.
See:
PaintedPath.arc_to()
Ancestors
- builtins.tuple
Static methods
def subdivide_sweep(sweep_angle: float) ‑> Generator[tuple[Point, Point, Point], None, None]
-
Expand source code Browse git
@staticmethod @force_nodocument def subdivide_sweep( sweep_angle: float, ) -> Generator[tuple[Point, Point, Point], None, None]: """ A generator that subdivides a swept angle into segments no larger than a quarter turn. Any sweep that is larger than a quarter turn is subdivided into as many equally sized segments as necessary to prevent any individual segment from being larger than a quarter turn. This is used for approximating a circular curve segment using cubic Bézier curves. This computes the parameters used for the Bézier approximation up front, as well as the transform necessary to place the segment in the correct position. Args: sweep_angle (float): the angle to subdivide. Yields: A tuple of (ctrl1, ctrl2, end) representing the control and end points of the cubic Bézier curve approximating the segment as a unit circle centered at the origin. """ sweep_angle = abs(sweep_angle) sweep_left = sweep_angle quarterturn = math.pi / 2 chunks = math.ceil(sweep_angle / quarterturn) sweep_segment = sweep_angle / chunks cos_t = math.cos(sweep_segment) sin_t = math.sin(sweep_segment) kappa = 4 / 3 * math.tan(sweep_segment / 4) ctrl1 = Point(1, kappa) ctrl2 = Point(cos_t + kappa * sin_t, sin_t - kappa * cos_t) end = Point(cos_t, sin_t) for _ in range(chunks): offset = sweep_angle - sweep_left transform = Transform.rotation(offset) yield ctrl1 @ transform, ctrl2 @ transform, end @ transform sweep_left -= sweep_segment
A generator that subdivides a swept angle into segments no larger than a quarter turn.
Any sweep that is larger than a quarter turn is subdivided into as many equally sized segments as necessary to prevent any individual segment from being larger than a quarter turn.
This is used for approximating a circular curve segment using cubic Bézier curves. This computes the parameters used for the Bézier approximation up front, as well as the transform necessary to place the segment in the correct position.
Args
sweep_angle
:float
- the angle to subdivide.
Yields
A tuple of (ctrl1, ctrl2, end) representing the control and end points of the cubic Bézier curve approximating the segment as a unit circle centered at the origin.
Instance variables
var end : Point
-
Expand source code Browse git
class Arc(NamedTuple): """ An elliptical arc path element. The arc is drawn from the end of the current path element to its specified end point using a number of parameters to determine how it is constructed. See: `PaintedPath.arc_to` """ radii: Point """ The x- and y-radii of the arc. If `radii.x == radii.y` the arc will be circular. """ rotation: Number """The rotation of the arc's major/minor axes relative to the coordinate frame.""" large: bool """If True, sweep the arc over an angle greater than or equal to 180 degrees.""" sweep: bool """If True, the arc is swept in the positive angular direction.""" end: Point """The end point of the arc.""" @staticmethod @force_nodocument def subdivide_sweep( sweep_angle: float, ) -> Generator[tuple[Point, Point, Point], None, None]: """ A generator that subdivides a swept angle into segments no larger than a quarter turn. Any sweep that is larger than a quarter turn is subdivided into as many equally sized segments as necessary to prevent any individual segment from being larger than a quarter turn. This is used for approximating a circular curve segment using cubic Bézier curves. This computes the parameters used for the Bézier approximation up front, as well as the transform necessary to place the segment in the correct position. Args: sweep_angle (float): the angle to subdivide. Yields: A tuple of (ctrl1, ctrl2, end) representing the control and end points of the cubic Bézier curve approximating the segment as a unit circle centered at the origin. """ sweep_angle = abs(sweep_angle) sweep_left = sweep_angle quarterturn = math.pi / 2 chunks = math.ceil(sweep_angle / quarterturn) sweep_segment = sweep_angle / chunks cos_t = math.cos(sweep_segment) sin_t = math.sin(sweep_segment) kappa = 4 / 3 * math.tan(sweep_segment / 4) ctrl1 = Point(1, kappa) ctrl2 = Point(cos_t + kappa * sin_t, sin_t - kappa * cos_t) end = Point(cos_t, sin_t) for _ in range(chunks): offset = sweep_angle - sweep_left transform = Transform.rotation(offset) yield ctrl1 @ transform, ctrl2 @ transform, end @ transform sweep_left -= sweep_segment def _approximate_arc(self, last_item: Renderable) -> list[BezierCurve]: """ Approximate this arc with a sequence of `BezierCurve`. Args: last_item: the previous path element (used for its end point) Returns: a list of `BezierCurve`. """ radii = self.radii reverse = Transform.rotation(-self.rotation) forward = Transform.rotation(self.rotation) prime = ((last_item.end_point - self.end) * 0.5) @ reverse lam_da = (prime.x / radii.x) ** 2 + (prime.y / radii.y) ** 2 if lam_da > 1: radii = Point(x=(lam_da**0.5) * radii.x, y=(lam_da**0.5) * radii.y) sign = (self.large != self.sweep) - (self.large == self.sweep) rxry2 = (radii.x * radii.y) ** 2 rxpy2 = (radii.x * prime.y) ** 2 rypx2 = (radii.y * prime.x) ** 2 centerprime = ( sign * math.sqrt(round(rxry2 - rxpy2 - rypx2, 8) / (rxpy2 + rypx2)) * Point( x=radii.x * prime.y / radii.y, y=-radii.y * prime.x / radii.x, ) ) center = (centerprime @ forward) + ((last_item.end_point + self.end) * 0.5) arcstart = Point( x=(prime.x - centerprime.x) / radii.x, y=(prime.y - centerprime.y) / radii.y, ) arcend = Point( x=(-prime.x - centerprime.x) / radii.x, y=(-prime.y - centerprime.y) / radii.y, ) theta = Point(1, 0).angle(arcstart) deltatheta = arcstart.angle(arcend) if (self.sweep is False) and (deltatheta > 0): deltatheta -= math.tau elif (self.sweep is True) and (deltatheta < 0): deltatheta += math.tau sweep_sign = (deltatheta >= 0) - (deltatheta < 0) final_tf = ( Transform.scaling(x=1, y=sweep_sign) # flip negative sweeps .rotate(theta) # rotate start of arc to correct position .scale(radii.x, radii.y) # scale unit circle into the final ellipse shape .rotate(self.rotation) # rotate the ellipse the specified angle .translate(center.x, center.y) # translate to the final coordinates ) curves = [] for ctrl1, ctrl2, end in self.subdivide_sweep(deltatheta): curves.append( BezierCurve(ctrl1 @ final_tf, ctrl2 @ final_tf, end @ final_tf) ) return curves def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this arc by approximating it with a series of Bezier curves and aggregating their bounding boxes. """ bbox = BoundingBox.empty() prev = Move(start) for curve in self._approximate_arc(prev): segment_bbox, _ = curve.bounding_box(prev.end_point) bbox = bbox.merge(segment_bbox) prev = curve return bbox, self.end @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `BezierCurve`. """ curves = self._approximate_arc(last_item) if not curves: return "", last_item, initial_point return ( " ".join( curve.render(resource_registry, style, prev, initial_point)[0] for prev, curve in zip([last_item, *curves[:-1]], curves) ), curves[-1], initial_point, ) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Arc.render`. """ curves = self._approximate_arc(last_item) debug_stream.write(f"{self} resolved to:\n") if not curves: debug_stream.write(pfx + " └─ nothing\n") return "", last_item, initial_point previous = [last_item] for curve in curves[:-1]: previous.append(curve) debug_stream.write(pfx + f" ├─ {curve}\n") debug_stream.write(pfx + f" └─ {curves[-1]}\n") return ( " ".join( curve.render(resource_registry, style, prev, initial_point)[0] for prev, curve in zip(previous, curves) ), curves[-1], initial_point, )
The end point of the arc.
var large : bool
-
Expand source code Browse git
class Arc(NamedTuple): """ An elliptical arc path element. The arc is drawn from the end of the current path element to its specified end point using a number of parameters to determine how it is constructed. See: `PaintedPath.arc_to` """ radii: Point """ The x- and y-radii of the arc. If `radii.x == radii.y` the arc will be circular. """ rotation: Number """The rotation of the arc's major/minor axes relative to the coordinate frame.""" large: bool """If True, sweep the arc over an angle greater than or equal to 180 degrees.""" sweep: bool """If True, the arc is swept in the positive angular direction.""" end: Point """The end point of the arc.""" @staticmethod @force_nodocument def subdivide_sweep( sweep_angle: float, ) -> Generator[tuple[Point, Point, Point], None, None]: """ A generator that subdivides a swept angle into segments no larger than a quarter turn. Any sweep that is larger than a quarter turn is subdivided into as many equally sized segments as necessary to prevent any individual segment from being larger than a quarter turn. This is used for approximating a circular curve segment using cubic Bézier curves. This computes the parameters used for the Bézier approximation up front, as well as the transform necessary to place the segment in the correct position. Args: sweep_angle (float): the angle to subdivide. Yields: A tuple of (ctrl1, ctrl2, end) representing the control and end points of the cubic Bézier curve approximating the segment as a unit circle centered at the origin. """ sweep_angle = abs(sweep_angle) sweep_left = sweep_angle quarterturn = math.pi / 2 chunks = math.ceil(sweep_angle / quarterturn) sweep_segment = sweep_angle / chunks cos_t = math.cos(sweep_segment) sin_t = math.sin(sweep_segment) kappa = 4 / 3 * math.tan(sweep_segment / 4) ctrl1 = Point(1, kappa) ctrl2 = Point(cos_t + kappa * sin_t, sin_t - kappa * cos_t) end = Point(cos_t, sin_t) for _ in range(chunks): offset = sweep_angle - sweep_left transform = Transform.rotation(offset) yield ctrl1 @ transform, ctrl2 @ transform, end @ transform sweep_left -= sweep_segment def _approximate_arc(self, last_item: Renderable) -> list[BezierCurve]: """ Approximate this arc with a sequence of `BezierCurve`. Args: last_item: the previous path element (used for its end point) Returns: a list of `BezierCurve`. """ radii = self.radii reverse = Transform.rotation(-self.rotation) forward = Transform.rotation(self.rotation) prime = ((last_item.end_point - self.end) * 0.5) @ reverse lam_da = (prime.x / radii.x) ** 2 + (prime.y / radii.y) ** 2 if lam_da > 1: radii = Point(x=(lam_da**0.5) * radii.x, y=(lam_da**0.5) * radii.y) sign = (self.large != self.sweep) - (self.large == self.sweep) rxry2 = (radii.x * radii.y) ** 2 rxpy2 = (radii.x * prime.y) ** 2 rypx2 = (radii.y * prime.x) ** 2 centerprime = ( sign * math.sqrt(round(rxry2 - rxpy2 - rypx2, 8) / (rxpy2 + rypx2)) * Point( x=radii.x * prime.y / radii.y, y=-radii.y * prime.x / radii.x, ) ) center = (centerprime @ forward) + ((last_item.end_point + self.end) * 0.5) arcstart = Point( x=(prime.x - centerprime.x) / radii.x, y=(prime.y - centerprime.y) / radii.y, ) arcend = Point( x=(-prime.x - centerprime.x) / radii.x, y=(-prime.y - centerprime.y) / radii.y, ) theta = Point(1, 0).angle(arcstart) deltatheta = arcstart.angle(arcend) if (self.sweep is False) and (deltatheta > 0): deltatheta -= math.tau elif (self.sweep is True) and (deltatheta < 0): deltatheta += math.tau sweep_sign = (deltatheta >= 0) - (deltatheta < 0) final_tf = ( Transform.scaling(x=1, y=sweep_sign) # flip negative sweeps .rotate(theta) # rotate start of arc to correct position .scale(radii.x, radii.y) # scale unit circle into the final ellipse shape .rotate(self.rotation) # rotate the ellipse the specified angle .translate(center.x, center.y) # translate to the final coordinates ) curves = [] for ctrl1, ctrl2, end in self.subdivide_sweep(deltatheta): curves.append( BezierCurve(ctrl1 @ final_tf, ctrl2 @ final_tf, end @ final_tf) ) return curves def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this arc by approximating it with a series of Bezier curves and aggregating their bounding boxes. """ bbox = BoundingBox.empty() prev = Move(start) for curve in self._approximate_arc(prev): segment_bbox, _ = curve.bounding_box(prev.end_point) bbox = bbox.merge(segment_bbox) prev = curve return bbox, self.end @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `BezierCurve`. """ curves = self._approximate_arc(last_item) if not curves: return "", last_item, initial_point return ( " ".join( curve.render(resource_registry, style, prev, initial_point)[0] for prev, curve in zip([last_item, *curves[:-1]], curves) ), curves[-1], initial_point, ) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Arc.render`. """ curves = self._approximate_arc(last_item) debug_stream.write(f"{self} resolved to:\n") if not curves: debug_stream.write(pfx + " └─ nothing\n") return "", last_item, initial_point previous = [last_item] for curve in curves[:-1]: previous.append(curve) debug_stream.write(pfx + f" ├─ {curve}\n") debug_stream.write(pfx + f" └─ {curves[-1]}\n") return ( " ".join( curve.render(resource_registry, style, prev, initial_point)[0] for prev, curve in zip(previous, curves) ), curves[-1], initial_point, )
If True, sweep the arc over an angle greater than or equal to 180 degrees.
var radii : Point
-
Expand source code Browse git
class Arc(NamedTuple): """ An elliptical arc path element. The arc is drawn from the end of the current path element to its specified end point using a number of parameters to determine how it is constructed. See: `PaintedPath.arc_to` """ radii: Point """ The x- and y-radii of the arc. If `radii.x == radii.y` the arc will be circular. """ rotation: Number """The rotation of the arc's major/minor axes relative to the coordinate frame.""" large: bool """If True, sweep the arc over an angle greater than or equal to 180 degrees.""" sweep: bool """If True, the arc is swept in the positive angular direction.""" end: Point """The end point of the arc.""" @staticmethod @force_nodocument def subdivide_sweep( sweep_angle: float, ) -> Generator[tuple[Point, Point, Point], None, None]: """ A generator that subdivides a swept angle into segments no larger than a quarter turn. Any sweep that is larger than a quarter turn is subdivided into as many equally sized segments as necessary to prevent any individual segment from being larger than a quarter turn. This is used for approximating a circular curve segment using cubic Bézier curves. This computes the parameters used for the Bézier approximation up front, as well as the transform necessary to place the segment in the correct position. Args: sweep_angle (float): the angle to subdivide. Yields: A tuple of (ctrl1, ctrl2, end) representing the control and end points of the cubic Bézier curve approximating the segment as a unit circle centered at the origin. """ sweep_angle = abs(sweep_angle) sweep_left = sweep_angle quarterturn = math.pi / 2 chunks = math.ceil(sweep_angle / quarterturn) sweep_segment = sweep_angle / chunks cos_t = math.cos(sweep_segment) sin_t = math.sin(sweep_segment) kappa = 4 / 3 * math.tan(sweep_segment / 4) ctrl1 = Point(1, kappa) ctrl2 = Point(cos_t + kappa * sin_t, sin_t - kappa * cos_t) end = Point(cos_t, sin_t) for _ in range(chunks): offset = sweep_angle - sweep_left transform = Transform.rotation(offset) yield ctrl1 @ transform, ctrl2 @ transform, end @ transform sweep_left -= sweep_segment def _approximate_arc(self, last_item: Renderable) -> list[BezierCurve]: """ Approximate this arc with a sequence of `BezierCurve`. Args: last_item: the previous path element (used for its end point) Returns: a list of `BezierCurve`. """ radii = self.radii reverse = Transform.rotation(-self.rotation) forward = Transform.rotation(self.rotation) prime = ((last_item.end_point - self.end) * 0.5) @ reverse lam_da = (prime.x / radii.x) ** 2 + (prime.y / radii.y) ** 2 if lam_da > 1: radii = Point(x=(lam_da**0.5) * radii.x, y=(lam_da**0.5) * radii.y) sign = (self.large != self.sweep) - (self.large == self.sweep) rxry2 = (radii.x * radii.y) ** 2 rxpy2 = (radii.x * prime.y) ** 2 rypx2 = (radii.y * prime.x) ** 2 centerprime = ( sign * math.sqrt(round(rxry2 - rxpy2 - rypx2, 8) / (rxpy2 + rypx2)) * Point( x=radii.x * prime.y / radii.y, y=-radii.y * prime.x / radii.x, ) ) center = (centerprime @ forward) + ((last_item.end_point + self.end) * 0.5) arcstart = Point( x=(prime.x - centerprime.x) / radii.x, y=(prime.y - centerprime.y) / radii.y, ) arcend = Point( x=(-prime.x - centerprime.x) / radii.x, y=(-prime.y - centerprime.y) / radii.y, ) theta = Point(1, 0).angle(arcstart) deltatheta = arcstart.angle(arcend) if (self.sweep is False) and (deltatheta > 0): deltatheta -= math.tau elif (self.sweep is True) and (deltatheta < 0): deltatheta += math.tau sweep_sign = (deltatheta >= 0) - (deltatheta < 0) final_tf = ( Transform.scaling(x=1, y=sweep_sign) # flip negative sweeps .rotate(theta) # rotate start of arc to correct position .scale(radii.x, radii.y) # scale unit circle into the final ellipse shape .rotate(self.rotation) # rotate the ellipse the specified angle .translate(center.x, center.y) # translate to the final coordinates ) curves = [] for ctrl1, ctrl2, end in self.subdivide_sweep(deltatheta): curves.append( BezierCurve(ctrl1 @ final_tf, ctrl2 @ final_tf, end @ final_tf) ) return curves def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this arc by approximating it with a series of Bezier curves and aggregating their bounding boxes. """ bbox = BoundingBox.empty() prev = Move(start) for curve in self._approximate_arc(prev): segment_bbox, _ = curve.bounding_box(prev.end_point) bbox = bbox.merge(segment_bbox) prev = curve return bbox, self.end @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `BezierCurve`. """ curves = self._approximate_arc(last_item) if not curves: return "", last_item, initial_point return ( " ".join( curve.render(resource_registry, style, prev, initial_point)[0] for prev, curve in zip([last_item, *curves[:-1]], curves) ), curves[-1], initial_point, ) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Arc.render`. """ curves = self._approximate_arc(last_item) debug_stream.write(f"{self} resolved to:\n") if not curves: debug_stream.write(pfx + " └─ nothing\n") return "", last_item, initial_point previous = [last_item] for curve in curves[:-1]: previous.append(curve) debug_stream.write(pfx + f" ├─ {curve}\n") debug_stream.write(pfx + f" └─ {curves[-1]}\n") return ( " ".join( curve.render(resource_registry, style, prev, initial_point)[0] for prev, curve in zip(previous, curves) ), curves[-1], initial_point, )
The x- and y-radii of the arc. If
radii.x == radii.y
the arc will be circular. var rotation : int | float | decimal.Decimal
-
Expand source code Browse git
class Arc(NamedTuple): """ An elliptical arc path element. The arc is drawn from the end of the current path element to its specified end point using a number of parameters to determine how it is constructed. See: `PaintedPath.arc_to` """ radii: Point """ The x- and y-radii of the arc. If `radii.x == radii.y` the arc will be circular. """ rotation: Number """The rotation of the arc's major/minor axes relative to the coordinate frame.""" large: bool """If True, sweep the arc over an angle greater than or equal to 180 degrees.""" sweep: bool """If True, the arc is swept in the positive angular direction.""" end: Point """The end point of the arc.""" @staticmethod @force_nodocument def subdivide_sweep( sweep_angle: float, ) -> Generator[tuple[Point, Point, Point], None, None]: """ A generator that subdivides a swept angle into segments no larger than a quarter turn. Any sweep that is larger than a quarter turn is subdivided into as many equally sized segments as necessary to prevent any individual segment from being larger than a quarter turn. This is used for approximating a circular curve segment using cubic Bézier curves. This computes the parameters used for the Bézier approximation up front, as well as the transform necessary to place the segment in the correct position. Args: sweep_angle (float): the angle to subdivide. Yields: A tuple of (ctrl1, ctrl2, end) representing the control and end points of the cubic Bézier curve approximating the segment as a unit circle centered at the origin. """ sweep_angle = abs(sweep_angle) sweep_left = sweep_angle quarterturn = math.pi / 2 chunks = math.ceil(sweep_angle / quarterturn) sweep_segment = sweep_angle / chunks cos_t = math.cos(sweep_segment) sin_t = math.sin(sweep_segment) kappa = 4 / 3 * math.tan(sweep_segment / 4) ctrl1 = Point(1, kappa) ctrl2 = Point(cos_t + kappa * sin_t, sin_t - kappa * cos_t) end = Point(cos_t, sin_t) for _ in range(chunks): offset = sweep_angle - sweep_left transform = Transform.rotation(offset) yield ctrl1 @ transform, ctrl2 @ transform, end @ transform sweep_left -= sweep_segment def _approximate_arc(self, last_item: Renderable) -> list[BezierCurve]: """ Approximate this arc with a sequence of `BezierCurve`. Args: last_item: the previous path element (used for its end point) Returns: a list of `BezierCurve`. """ radii = self.radii reverse = Transform.rotation(-self.rotation) forward = Transform.rotation(self.rotation) prime = ((last_item.end_point - self.end) * 0.5) @ reverse lam_da = (prime.x / radii.x) ** 2 + (prime.y / radii.y) ** 2 if lam_da > 1: radii = Point(x=(lam_da**0.5) * radii.x, y=(lam_da**0.5) * radii.y) sign = (self.large != self.sweep) - (self.large == self.sweep) rxry2 = (radii.x * radii.y) ** 2 rxpy2 = (radii.x * prime.y) ** 2 rypx2 = (radii.y * prime.x) ** 2 centerprime = ( sign * math.sqrt(round(rxry2 - rxpy2 - rypx2, 8) / (rxpy2 + rypx2)) * Point( x=radii.x * prime.y / radii.y, y=-radii.y * prime.x / radii.x, ) ) center = (centerprime @ forward) + ((last_item.end_point + self.end) * 0.5) arcstart = Point( x=(prime.x - centerprime.x) / radii.x, y=(prime.y - centerprime.y) / radii.y, ) arcend = Point( x=(-prime.x - centerprime.x) / radii.x, y=(-prime.y - centerprime.y) / radii.y, ) theta = Point(1, 0).angle(arcstart) deltatheta = arcstart.angle(arcend) if (self.sweep is False) and (deltatheta > 0): deltatheta -= math.tau elif (self.sweep is True) and (deltatheta < 0): deltatheta += math.tau sweep_sign = (deltatheta >= 0) - (deltatheta < 0) final_tf = ( Transform.scaling(x=1, y=sweep_sign) # flip negative sweeps .rotate(theta) # rotate start of arc to correct position .scale(radii.x, radii.y) # scale unit circle into the final ellipse shape .rotate(self.rotation) # rotate the ellipse the specified angle .translate(center.x, center.y) # translate to the final coordinates ) curves = [] for ctrl1, ctrl2, end in self.subdivide_sweep(deltatheta): curves.append( BezierCurve(ctrl1 @ final_tf, ctrl2 @ final_tf, end @ final_tf) ) return curves def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this arc by approximating it with a series of Bezier curves and aggregating their bounding boxes. """ bbox = BoundingBox.empty() prev = Move(start) for curve in self._approximate_arc(prev): segment_bbox, _ = curve.bounding_box(prev.end_point) bbox = bbox.merge(segment_bbox) prev = curve return bbox, self.end @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `BezierCurve`. """ curves = self._approximate_arc(last_item) if not curves: return "", last_item, initial_point return ( " ".join( curve.render(resource_registry, style, prev, initial_point)[0] for prev, curve in zip([last_item, *curves[:-1]], curves) ), curves[-1], initial_point, ) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Arc.render`. """ curves = self._approximate_arc(last_item) debug_stream.write(f"{self} resolved to:\n") if not curves: debug_stream.write(pfx + " └─ nothing\n") return "", last_item, initial_point previous = [last_item] for curve in curves[:-1]: previous.append(curve) debug_stream.write(pfx + f" ├─ {curve}\n") debug_stream.write(pfx + f" └─ {curves[-1]}\n") return ( " ".join( curve.render(resource_registry, style, prev, initial_point)[0] for prev, curve in zip(previous, curves) ), curves[-1], initial_point, )
The rotation of the arc's major/minor axes relative to the coordinate frame.
var sweep : bool
-
Expand source code Browse git
class Arc(NamedTuple): """ An elliptical arc path element. The arc is drawn from the end of the current path element to its specified end point using a number of parameters to determine how it is constructed. See: `PaintedPath.arc_to` """ radii: Point """ The x- and y-radii of the arc. If `radii.x == radii.y` the arc will be circular. """ rotation: Number """The rotation of the arc's major/minor axes relative to the coordinate frame.""" large: bool """If True, sweep the arc over an angle greater than or equal to 180 degrees.""" sweep: bool """If True, the arc is swept in the positive angular direction.""" end: Point """The end point of the arc.""" @staticmethod @force_nodocument def subdivide_sweep( sweep_angle: float, ) -> Generator[tuple[Point, Point, Point], None, None]: """ A generator that subdivides a swept angle into segments no larger than a quarter turn. Any sweep that is larger than a quarter turn is subdivided into as many equally sized segments as necessary to prevent any individual segment from being larger than a quarter turn. This is used for approximating a circular curve segment using cubic Bézier curves. This computes the parameters used for the Bézier approximation up front, as well as the transform necessary to place the segment in the correct position. Args: sweep_angle (float): the angle to subdivide. Yields: A tuple of (ctrl1, ctrl2, end) representing the control and end points of the cubic Bézier curve approximating the segment as a unit circle centered at the origin. """ sweep_angle = abs(sweep_angle) sweep_left = sweep_angle quarterturn = math.pi / 2 chunks = math.ceil(sweep_angle / quarterturn) sweep_segment = sweep_angle / chunks cos_t = math.cos(sweep_segment) sin_t = math.sin(sweep_segment) kappa = 4 / 3 * math.tan(sweep_segment / 4) ctrl1 = Point(1, kappa) ctrl2 = Point(cos_t + kappa * sin_t, sin_t - kappa * cos_t) end = Point(cos_t, sin_t) for _ in range(chunks): offset = sweep_angle - sweep_left transform = Transform.rotation(offset) yield ctrl1 @ transform, ctrl2 @ transform, end @ transform sweep_left -= sweep_segment def _approximate_arc(self, last_item: Renderable) -> list[BezierCurve]: """ Approximate this arc with a sequence of `BezierCurve`. Args: last_item: the previous path element (used for its end point) Returns: a list of `BezierCurve`. """ radii = self.radii reverse = Transform.rotation(-self.rotation) forward = Transform.rotation(self.rotation) prime = ((last_item.end_point - self.end) * 0.5) @ reverse lam_da = (prime.x / radii.x) ** 2 + (prime.y / radii.y) ** 2 if lam_da > 1: radii = Point(x=(lam_da**0.5) * radii.x, y=(lam_da**0.5) * radii.y) sign = (self.large != self.sweep) - (self.large == self.sweep) rxry2 = (radii.x * radii.y) ** 2 rxpy2 = (radii.x * prime.y) ** 2 rypx2 = (radii.y * prime.x) ** 2 centerprime = ( sign * math.sqrt(round(rxry2 - rxpy2 - rypx2, 8) / (rxpy2 + rypx2)) * Point( x=radii.x * prime.y / radii.y, y=-radii.y * prime.x / radii.x, ) ) center = (centerprime @ forward) + ((last_item.end_point + self.end) * 0.5) arcstart = Point( x=(prime.x - centerprime.x) / radii.x, y=(prime.y - centerprime.y) / radii.y, ) arcend = Point( x=(-prime.x - centerprime.x) / radii.x, y=(-prime.y - centerprime.y) / radii.y, ) theta = Point(1, 0).angle(arcstart) deltatheta = arcstart.angle(arcend) if (self.sweep is False) and (deltatheta > 0): deltatheta -= math.tau elif (self.sweep is True) and (deltatheta < 0): deltatheta += math.tau sweep_sign = (deltatheta >= 0) - (deltatheta < 0) final_tf = ( Transform.scaling(x=1, y=sweep_sign) # flip negative sweeps .rotate(theta) # rotate start of arc to correct position .scale(radii.x, radii.y) # scale unit circle into the final ellipse shape .rotate(self.rotation) # rotate the ellipse the specified angle .translate(center.x, center.y) # translate to the final coordinates ) curves = [] for ctrl1, ctrl2, end in self.subdivide_sweep(deltatheta): curves.append( BezierCurve(ctrl1 @ final_tf, ctrl2 @ final_tf, end @ final_tf) ) return curves def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this arc by approximating it with a series of Bezier curves and aggregating their bounding boxes. """ bbox = BoundingBox.empty() prev = Move(start) for curve in self._approximate_arc(prev): segment_bbox, _ = curve.bounding_box(prev.end_point) bbox = bbox.merge(segment_bbox) prev = curve return bbox, self.end @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `BezierCurve`. """ curves = self._approximate_arc(last_item) if not curves: return "", last_item, initial_point return ( " ".join( curve.render(resource_registry, style, prev, initial_point)[0] for prev, curve in zip([last_item, *curves[:-1]], curves) ), curves[-1], initial_point, ) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Arc.render`. """ curves = self._approximate_arc(last_item) debug_stream.write(f"{self} resolved to:\n") if not curves: debug_stream.write(pfx + " └─ nothing\n") return "", last_item, initial_point previous = [last_item] for curve in curves[:-1]: previous.append(curve) debug_stream.write(pfx + f" ├─ {curve}\n") debug_stream.write(pfx + f" └─ {curves[-1]}\n") return ( " ".join( curve.render(resource_registry, style, prev, initial_point)[0] for prev, curve in zip(previous, curves) ), curves[-1], initial_point, )
If True, the arc is swept in the positive angular direction.
Methods
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this arc by approximating it with a series of Bezier curves and aggregating their bounding boxes. """ bbox = BoundingBox.empty() prev = Move(start) for curve in self._approximate_arc(prev): segment_bbox, _ = curve.bounding_box(prev.end_point) bbox = bbox.merge(segment_bbox) prev = curve return bbox, self.end
Compute the bounding box of this arc by approximating it with a series of Bezier curves and aggregating their bounding boxes.
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
@force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `BezierCurve`. """ curves = self._approximate_arc(last_item) if not curves: return "", last_item, initial_point return ( " ".join( curve.render(resource_registry, style, prev, initial_point)[0] for prev, curve in zip([last_item, *curves[:-1]], curves) ), curves[-1], initial_point, )
Render this path element to its PDF representation.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
Returns
a tuple of
(str, new_last_item)
, wherenew_last_item
is a resolvedBezierCurve
. def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
@force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Arc.render`. """ curves = self._approximate_arc(last_item) debug_stream.write(f"{self} resolved to:\n") if not curves: debug_stream.write(pfx + " └─ nothing\n") return "", last_item, initial_point previous = [last_item] for curve in curves[:-1]: previous.append(curve) debug_stream.write(pfx + f" ├─ {curve}\n") debug_stream.write(pfx + f" └─ {curves[-1]}\n") return ( " ".join( curve.render(resource_registry, style, prev, initial_point)[0] for prev, curve in zip(previous, curves) ), curves[-1], initial_point, )
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
Arc.render()
.
class BezierCurve (c1: Point,
c2: Point,
end: Point)-
Expand source code Browse git
class BezierCurve(NamedTuple): """ A cubic Bézier curve path element. This draws a Bézier curve parameterized by the end point of the previous path element, two off-curve control points, and an end point. See: `PaintedPath.curve_to` """ c1: Point """The curve's first control point.""" c2: Point """The curve's second control point.""" end: Point """The curve's end point.""" @property def end_point(self) -> Point: """The end point of this path element.""" return self.end def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this cubic Bézier curve.""" # Evaluate all candidate t values (endpoints + extrema) for both axes px = [start.x, self.c1.x, self.c2.x, self.end.x] py = [start.y, self.c1.y, self.c2.y, self.end.y] tx = [0, 1] + _cubic_bezier_critical_ts_1d(*px) ty = [0, 1] + _cubic_bezier_critical_ts_1d(*py) xs = [_eval_cubic_bezier_1d(t, *px) for t in tx] ys = [_eval_cubic_bezier_1d(t, *py) for t in ty] bbox = BoundingBox(min(xs), min(ys), max(xs), max(ys)) return bbox, self.end @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is `self` """ # pylint: disable=unused-argument return _render_curve(self.c1, self.c2, self.end), self, initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `BezierCurve.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(str(self) + "\n") return rendered, resolved, initial_point
A cubic Bézier curve path element.
This draws a Bézier curve parameterized by the end point of the previous path element, two off-curve control points, and an end point.
Ancestors
- builtins.tuple
Instance variables
var c1 : Point
-
Expand source code Browse git
class BezierCurve(NamedTuple): """ A cubic Bézier curve path element. This draws a Bézier curve parameterized by the end point of the previous path element, two off-curve control points, and an end point. See: `PaintedPath.curve_to` """ c1: Point """The curve's first control point.""" c2: Point """The curve's second control point.""" end: Point """The curve's end point.""" @property def end_point(self) -> Point: """The end point of this path element.""" return self.end def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this cubic Bézier curve.""" # Evaluate all candidate t values (endpoints + extrema) for both axes px = [start.x, self.c1.x, self.c2.x, self.end.x] py = [start.y, self.c1.y, self.c2.y, self.end.y] tx = [0, 1] + _cubic_bezier_critical_ts_1d(*px) ty = [0, 1] + _cubic_bezier_critical_ts_1d(*py) xs = [_eval_cubic_bezier_1d(t, *px) for t in tx] ys = [_eval_cubic_bezier_1d(t, *py) for t in ty] bbox = BoundingBox(min(xs), min(ys), max(xs), max(ys)) return bbox, self.end @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is `self` """ # pylint: disable=unused-argument return _render_curve(self.c1, self.c2, self.end), self, initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `BezierCurve.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(str(self) + "\n") return rendered, resolved, initial_point
The curve's first control point.
var c2 : Point
-
Expand source code Browse git
class BezierCurve(NamedTuple): """ A cubic Bézier curve path element. This draws a Bézier curve parameterized by the end point of the previous path element, two off-curve control points, and an end point. See: `PaintedPath.curve_to` """ c1: Point """The curve's first control point.""" c2: Point """The curve's second control point.""" end: Point """The curve's end point.""" @property def end_point(self) -> Point: """The end point of this path element.""" return self.end def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this cubic Bézier curve.""" # Evaluate all candidate t values (endpoints + extrema) for both axes px = [start.x, self.c1.x, self.c2.x, self.end.x] py = [start.y, self.c1.y, self.c2.y, self.end.y] tx = [0, 1] + _cubic_bezier_critical_ts_1d(*px) ty = [0, 1] + _cubic_bezier_critical_ts_1d(*py) xs = [_eval_cubic_bezier_1d(t, *px) for t in tx] ys = [_eval_cubic_bezier_1d(t, *py) for t in ty] bbox = BoundingBox(min(xs), min(ys), max(xs), max(ys)) return bbox, self.end @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is `self` """ # pylint: disable=unused-argument return _render_curve(self.c1, self.c2, self.end), self, initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `BezierCurve.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(str(self) + "\n") return rendered, resolved, initial_point
The curve's second control point.
var end : Point
-
Expand source code Browse git
class BezierCurve(NamedTuple): """ A cubic Bézier curve path element. This draws a Bézier curve parameterized by the end point of the previous path element, two off-curve control points, and an end point. See: `PaintedPath.curve_to` """ c1: Point """The curve's first control point.""" c2: Point """The curve's second control point.""" end: Point """The curve's end point.""" @property def end_point(self) -> Point: """The end point of this path element.""" return self.end def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this cubic Bézier curve.""" # Evaluate all candidate t values (endpoints + extrema) for both axes px = [start.x, self.c1.x, self.c2.x, self.end.x] py = [start.y, self.c1.y, self.c2.y, self.end.y] tx = [0, 1] + _cubic_bezier_critical_ts_1d(*px) ty = [0, 1] + _cubic_bezier_critical_ts_1d(*py) xs = [_eval_cubic_bezier_1d(t, *px) for t in tx] ys = [_eval_cubic_bezier_1d(t, *py) for t in ty] bbox = BoundingBox(min(xs), min(ys), max(xs), max(ys)) return bbox, self.end @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is `self` """ # pylint: disable=unused-argument return _render_curve(self.c1, self.c2, self.end), self, initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `BezierCurve.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(str(self) + "\n") return rendered, resolved, initial_point
The curve's end point.
prop end_point : Point
-
Expand source code Browse git
@property def end_point(self) -> Point: """The end point of this path element.""" return self.end
The end point of this path element.
Methods
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this cubic Bézier curve.""" # Evaluate all candidate t values (endpoints + extrema) for both axes px = [start.x, self.c1.x, self.c2.x, self.end.x] py = [start.y, self.c1.y, self.c2.y, self.end.y] tx = [0, 1] + _cubic_bezier_critical_ts_1d(*px) ty = [0, 1] + _cubic_bezier_critical_ts_1d(*py) xs = [_eval_cubic_bezier_1d(t, *px) for t in tx] ys = [_eval_cubic_bezier_1d(t, *py) for t in ty] bbox = BoundingBox(min(xs), min(ys), max(xs), max(ys)) return bbox, self.end
Compute the bounding box of this cubic Bézier curve.
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
@force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is `self` """ # pylint: disable=unused-argument return _render_curve(self.c1, self.c2, self.end), self, initial_point
Render this path element to its PDF representation.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
Returns
a tuple of
(str, new_last_item)
, wherenew_last_item
isself
def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
@force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `BezierCurve.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(str(self) + "\n") return rendered, resolved, initial_point
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
BezierCurve.render()
.
class BoundingBox (x0: float, y0: float, x1: float, y1: float)
-
Expand source code Browse git
@dataclass(frozen=True, eq=False) class BoundingBox: """Represents a bounding box, with utility methods for creating and manipulating them.""" x0: float y0: float x1: float y1: float @classmethod def empty(cls) -> "BoundingBox": """ Return an 'empty' bounding box with extreme values that collapse on merge. """ return cls(float("inf"), float("inf"), float("-inf"), float("-inf")) def is_valid(self) -> bool: """Return True if the bounding box is not empty.""" return self.x0 <= self.x1 and self.y0 <= self.y1 @classmethod def from_points(cls, points: list[Point]) -> "BoundingBox": """Given a list of points, create a bounding box that encloses them all.""" xs = [float(p.x) for p in points] ys = [float(p.y) for p in points] return cls(min(xs), min(ys), max(xs), max(ys)) def merge(self, other: "BoundingBox") -> "BoundingBox": """Expand this bounding box to include another one.""" if not self.is_valid(): return other if not other.is_valid(): return self return BoundingBox( min(self.x0, other.x0), min(self.y0, other.y0), max(self.x1, other.x1), max(self.y1, other.y1), ) def transformed(self, tf: Transform) -> "BoundingBox": """ Return a new bounding box resulting from applying a transform to this one. """ corners = [ Point(self.x0, self.y0), Point(self.x1, self.y0), Point(self.x0, self.y1), Point(self.x1, self.y1), ] transformed_points = [pt @ tf for pt in corners] return BoundingBox.from_points(transformed_points) def expanded(self, dx: float, dy: Optional[float] = None) -> "BoundingBox": """Return a new bounding box expanded by the given amounts in each direction.""" if dy is None: dy = dx return BoundingBox(self.x0 - dx, self.y0 - dy, self.x1 + dx, self.y1 + dy) def expanded_to_stroke( self, style: "GraphicsStyle", row_norms: tuple[float, float] = (1.0, 1.0) ) -> "BoundingBox": """Expand this bbox to include stroke coverage, given a graphics style.""" # 1) Is there any stroke to consider? if not style.resolve_paint_rule() in ( PathPaintRule.STROKE, PathPaintRule.STROKE_FILL_NONZERO, PathPaintRule.STROKE_FILL_EVENODD, ): return self # If stroke opacity resolves to 0, no visible stroke => no expansion so = getattr(style, "stroke_opacity", GraphicsStyle.INHERIT) if (so is not GraphicsStyle.INHERIT) and (so is not None): try: if float(so) <= 0.0: return self # no visible stroke, don't expand except (TypeError, ValueError): pass # 2) Effective stroke width (PDF default is 1 if unset/inherit) w = ( 1.00 if style.stroke_width is None or style.stroke_width is GraphicsStyle.INHERIT else style.stroke_width ) w = float(w) if w == 0.0: return self r = 0.5 * w # 3) Row norms from CTM to scale the half-stroke in X/Y nx, ny = row_norms return self.expanded(r * nx, r * ny) def to_tuple(self) -> tuple[float, float, float, float]: """Convert bounding box to a 4-tuple.""" return (self.x0, self.y0, self.x1, self.y1) @property def width(self) -> float: """Return the width of the bounding box.""" return self.x1 - self.x0 @property def height(self) -> float: """Return the height of the bounding box.""" return self.y1 - self.y0 def __str__(self) -> str: return f"BoundingBox({self.x0}, {self.y0}, {self.x1}, {self.y1})" def __eq__(self, other: object) -> bool: if not isinstance(other, BoundingBox): return False tolerance = 1e-6 return ( abs(self.x0 - other.x0) < tolerance and abs(self.y0 - other.y0) < tolerance and abs(self.x1 - other.x1) < tolerance and abs(self.y1 - other.y1) < tolerance ) def __hash__(self) -> int: # Round to match the tolerance used in __eq__ return hash( ( round(self.x0, 6), round(self.y0, 6), round(self.x1, 6), round(self.y1, 6), ) )
Represents a bounding box, with utility methods for creating and manipulating them.
Static methods
def empty() ‑> BoundingBox
-
Return an 'empty' bounding box with extreme values that collapse on merge.
def from_points(points: list[Point]) ‑> BoundingBox
-
Given a list of points, create a bounding box that encloses them all.
Instance variables
prop height : float
-
Expand source code Browse git
@property def height(self) -> float: """Return the height of the bounding box.""" return self.y1 - self.y0
Return the height of the bounding box.
prop width : float
-
Expand source code Browse git
@property def width(self) -> float: """Return the width of the bounding box.""" return self.x1 - self.x0
Return the width of the bounding box.
var x0 : float
-
The type of the None singleton.
var x1 : float
-
The type of the None singleton.
var y0 : float
-
The type of the None singleton.
var y1 : float
-
The type of the None singleton.
Methods
def expanded(self, dx: float, dy: float | None = None) ‑> BoundingBox
-
Expand source code Browse git
def expanded(self, dx: float, dy: Optional[float] = None) -> "BoundingBox": """Return a new bounding box expanded by the given amounts in each direction.""" if dy is None: dy = dx return BoundingBox(self.x0 - dx, self.y0 - dy, self.x1 + dx, self.y1 + dy)
Return a new bounding box expanded by the given amounts in each direction.
def expanded_to_stroke(self,
style: GraphicsStyle,
row_norms: tuple[float, float] = (1.0, 1.0)) ‑> BoundingBox-
Expand source code Browse git
def expanded_to_stroke( self, style: "GraphicsStyle", row_norms: tuple[float, float] = (1.0, 1.0) ) -> "BoundingBox": """Expand this bbox to include stroke coverage, given a graphics style.""" # 1) Is there any stroke to consider? if not style.resolve_paint_rule() in ( PathPaintRule.STROKE, PathPaintRule.STROKE_FILL_NONZERO, PathPaintRule.STROKE_FILL_EVENODD, ): return self # If stroke opacity resolves to 0, no visible stroke => no expansion so = getattr(style, "stroke_opacity", GraphicsStyle.INHERIT) if (so is not GraphicsStyle.INHERIT) and (so is not None): try: if float(so) <= 0.0: return self # no visible stroke, don't expand except (TypeError, ValueError): pass # 2) Effective stroke width (PDF default is 1 if unset/inherit) w = ( 1.00 if style.stroke_width is None or style.stroke_width is GraphicsStyle.INHERIT else style.stroke_width ) w = float(w) if w == 0.0: return self r = 0.5 * w # 3) Row norms from CTM to scale the half-stroke in X/Y nx, ny = row_norms return self.expanded(r * nx, r * ny)
Expand this bbox to include stroke coverage, given a graphics style.
def is_valid(self) ‑> bool
-
Expand source code Browse git
def is_valid(self) -> bool: """Return True if the bounding box is not empty.""" return self.x0 <= self.x1 and self.y0 <= self.y1
Return True if the bounding box is not empty.
def merge(self,
other: BoundingBox) ‑> BoundingBox-
Expand source code Browse git
def merge(self, other: "BoundingBox") -> "BoundingBox": """Expand this bounding box to include another one.""" if not self.is_valid(): return other if not other.is_valid(): return self return BoundingBox( min(self.x0, other.x0), min(self.y0, other.y0), max(self.x1, other.x1), max(self.y1, other.y1), )
Expand this bounding box to include another one.
def to_tuple(self) ‑> tuple[float, float, float, float]
-
Expand source code Browse git
def to_tuple(self) -> tuple[float, float, float, float]: """Convert bounding box to a 4-tuple.""" return (self.x0, self.y0, self.x1, self.y1)
Convert bounding box to a 4-tuple.
def transformed(self,
tf: Transform) ‑> BoundingBox-
Expand source code Browse git
def transformed(self, tf: Transform) -> "BoundingBox": """ Return a new bounding box resulting from applying a transform to this one. """ corners = [ Point(self.x0, self.y0), Point(self.x1, self.y0), Point(self.x0, self.y1), Point(self.x1, self.y1), ] transformed_points = [pt @ tf for pt in corners] return BoundingBox.from_points(transformed_points)
Return a new bounding box resulting from applying a transform to this one.
class ClippingPath (x=0, y=0)
-
Expand source code Browse git
class ClippingPath(PaintedPath): """ The PaintedPath API but to be used to create clipping paths. .. warning:: Unless you really know what you're doing, changing attributes of the clipping path style is likely to produce unexpected results. This is because the clipping path styles override implicit style inheritance of the `PaintedPath` it applies to. For example, `clippath.style.stroke_width = 2` can unexpectedly override `paintpath.style.stroke_width = GraphicsStyle.INHERIT` and cause the painted path to be rendered with a stroke of 2 instead of what it would have normally inherited. Because a `ClippingPath` can be painted like a normal `PaintedPath`, it would be overly restrictive to remove the ability to style it, so instead this warning is here. """ __slots__ = () # no new attributes; preserve slotted layout from PaintedPath # because clipping paths can be painted, we inherit from PaintedPath. However, when # setting the styling on the clipping path, those values will also be applied to # the PaintedPath the ClippingPath is applied to unless they are explicitly set for # that painted path. This is not ideal, but there's no way to really fix it from # the PDF rendering model, and trying to track the appropriate state/defaults seems # similarly error prone. # In general, the expectation is that painted clipping paths are likely to be very # uncommon, so it's an edge case that isn't worth worrying too much about. def __init__(self, x=0, y=0): super().__init__(x=x, y=y) self.paint_rule = PathPaintRule.DONT_PAINT def render( self, resource_registry, style, last_item, initial_point, debug_stream=None, pfx=None, ): # painting the clipping path outside of its root graphics context allows it to # be transformed without affecting the transform of the graphics context of the # path it is being used to clip. This is because, unlike all of the other style # settings, transformations immediately affect the points following them, # rather than only affecting them at painting time. stroke settings and color # settings are applied only at paint time. if debug_stream: debug_stream.write("<ClippingPath> ") ( render_list, last_item, initial_point, ) = self._root_graphics_context.build_render_list( resource_registry, style, last_item, initial_point, debug_stream, pfx, _push_stack=False, ) merged_style = GraphicsStyle.merge(style, self.style) # we should never get a collision error here intersection_rule = merged_style.intersection_rule if intersection_rule is merged_style.INHERIT: intersection_rule = ClippingPathIntersectionRule.NONZERO else: intersection_rule = ClippingPathIntersectionRule[ intersection_rule.name # pylint: disable=no-member, useless-suppression ] paint_rule = merged_style.resolve_paint_rule() render_list.append(intersection_rule.value) render_list.append(paint_rule.value) return " ".join(render_list), last_item, initial_point def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `ClippingPath.render`. """ return self.render( resource_registry, style, last_item, initial_point, debug_stream, pfx )
The PaintedPath API but to be used to create clipping paths.
Warning
Unless you really know what you're doing, changing attributes of the clipping path style is likely to produce unexpected results. This is because the clipping path styles override implicit style inheritance of the
PaintedPath
it applies to.For example,
clippath.style.stroke_width = 2
can unexpectedly overridepaintpath.style.stroke_width = GraphicsStyle.INHERIT
and cause the painted path to be rendered with a stroke of 2 instead of what it would have normally inherited. Because aClippingPath
can be painted like a normalPaintedPath
, it would be overly restrictive to remove the ability to style it, so instead this warning is here.Ancestors
Instance variables
prop auto_close : bool
-
Inherited from:
PaintedPath
.auto_close
If true, the path should automatically close itself before painting.
prop clipping_path
-
Inherited from:
PaintedPath
.clipping_path
Set the clipping path for this path.
prop paint_rule : PathPaintRule
-
Inherited from:
PaintedPath
.paint_rule
Manually specify the
PathPaintRule
to use for rendering the path. prop style : GraphicsStyle
-
Inherited from:
PaintedPath
.style
The
GraphicsStyle
applied to all elements of this path. prop transform : Transform | None
-
Inherited from:
PaintedPath
.transform
The
Transform
that applies to all of the elements of this path.
Methods
def add_path_element(self, item)
-
Inherited from:
PaintedPath
.add_path_element
Add the given element as a path item of this path …
def arc_relative(self, rx, ry, rotation, large_arc, positive_sweep, dx, dy)
-
Inherited from:
PaintedPath
.arc_relative
Append an elliptical arc from the end of the previous path point to an offset point …
def arc_to(self, rx, ry, rotation, large_arc, positive_sweep, x, y)
-
Inherited from:
PaintedPath
.arc_to
Append an elliptical arc from the end of the previous path point to the specified end point …
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Inherited from:
PaintedPath
.bounding_box
Compute the bounding box of this painted path, including nested contexts and transformations.
def circle(self, cx, cy, r)
-
Inherited from:
PaintedPath
.circle
Append a circle as a closed subpath to the current path …
def close(self)
-
Inherited from:
PaintedPath
.close
Explicitly close the current (sub)path.
def curve_relative(self, dx1, dy1, dx2, dy2, dx3, dy3)
-
Inherited from:
PaintedPath
.curve_relative
Append a cubic Bézier curve whose points are expressed relative to the end point of the previous path element …
def curve_to(self, x1, y1, x2, y2, x3, y3)
-
Inherited from:
PaintedPath
.curve_to
Append a cubic Bézier curve to this path …
def ellipse(self, cx, cy, rx, ry)
-
Inherited from:
PaintedPath
.ellipse
Append an ellipse as a closed subpath to the current path …
def horizontal_line_relative(self, dx)
-
Inherited from:
PaintedPath
.horizontal_line_relative
Append a straight horizontal line to the given offset from the previous path element. The ordinate is retrieved from the end point of the previous …
def horizontal_line_to(self, x)
-
Inherited from:
PaintedPath
.horizontal_line_to
Append a straight horizontal line to the given abscissa. The ordinate is retrieved from the end point of the previous path element …
def line_relative(self, dx, dy)
-
Inherited from:
PaintedPath
.line_relative
Append a straight line whose end is computed as an offset from the end of the previous path element …
def line_to(self, x, y)
-
Inherited from:
PaintedPath
.line_to
Append a straight line to this path …
def move_relative(self, x, y)
-
Inherited from:
PaintedPath
.move_relative
Start a new subpath or move the path start point relative to the previous point …
def move_to(self, x, y)
-
Inherited from:
PaintedPath
.move_to
Start a new subpath or move the path starting point …
def quadratic_curve_relative(self, dx1, dy1, dx2, dy2)
-
Inherited from:
PaintedPath
.quadratic_curve_relative
Append a cubic Bézier curve mimicking the specified quadratic Bézier curve …
def quadratic_curve_to(self, x1, y1, x2, y2)
-
Inherited from:
PaintedPath
.quadratic_curve_to
Append a cubic Bézier curve mimicking the specified quadratic Bézier curve …
def rectangle(self, x, y, w, h, rx=0, ry=0)
-
Inherited from:
PaintedPath
.rectangle
Append a rectangle as a closed subpath to the current path …
def render(self, resource_registry, style, last_item, initial_point, debug_stream=None, pfx=None)
-
Expand source code Browse git
def render( self, resource_registry, style, last_item, initial_point, debug_stream=None, pfx=None, ): # painting the clipping path outside of its root graphics context allows it to # be transformed without affecting the transform of the graphics context of the # path it is being used to clip. This is because, unlike all of the other style # settings, transformations immediately affect the points following them, # rather than only affecting them at painting time. stroke settings and color # settings are applied only at paint time. if debug_stream: debug_stream.write("<ClippingPath> ") ( render_list, last_item, initial_point, ) = self._root_graphics_context.build_render_list( resource_registry, style, last_item, initial_point, debug_stream, pfx, _push_stack=False, ) merged_style = GraphicsStyle.merge(style, self.style) # we should never get a collision error here intersection_rule = merged_style.intersection_rule if intersection_rule is merged_style.INHERIT: intersection_rule = ClippingPathIntersectionRule.NONZERO else: intersection_rule = ClippingPathIntersectionRule[ intersection_rule.name # pylint: disable=no-member, useless-suppression ] paint_rule = merged_style.resolve_paint_rule() render_list.append(intersection_rule.value) render_list.append(paint_rule.value) return " ".join(render_list), last_item, initial_point
def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `ClippingPath.render`. """ return self.render( resource_registry, style, last_item, initial_point, debug_stream, pfx )
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
ClippingPath.render()
. def transform_group(self, transform)
-
Inherited from:
PaintedPath
.transform_group
Apply the provided
Transform
to all points added within this context. def vertical_line_relative(self, dy)
-
Inherited from:
PaintedPath
.vertical_line_relative
Append a straight vertical line to the given offset from the previous path element. The abscissa is retrieved from the end point of the previous path …
def vertical_line_to(self, y)
-
Inherited from:
PaintedPath
.vertical_line_to
Append a straight vertical line to the given ordinate. The abscissa is retrieved from the end point of the previous path element …
class Close
-
Expand source code Browse git
class Close(NamedTuple): """ A path close element. Instructs the renderer to draw a straight line from the end of the last path element to the start of the current path. See: `PaintedPath.close` """ # pylint: disable=no-self-use def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Return an empty bounding box; Close does not affect the geometry.""" return BoundingBox.empty(), start # pylint: disable=no-self-use @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is whatever the old last_item was. """ # pylint: disable=unused-argument return "h", Move(initial_point), initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Close.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(str(self) + "\n") return rendered, resolved, initial_point
A path close element.
Instructs the renderer to draw a straight line from the end of the last path element to the start of the current path.
See:
PaintedPath.close()
Ancestors
- builtins.tuple
Methods
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Return an empty bounding box; Close does not affect the geometry.""" return BoundingBox.empty(), start
Return an empty bounding box; Close does not affect the geometry.
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
@force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is whatever the old last_item was. """ # pylint: disable=unused-argument return "h", Move(initial_point), initial_point
Render this path element to its PDF representation.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
Returns
a tuple of
(str, new_last_item)
, wherenew_last_item
is whatever the old last_item was. def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
@force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Close.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(str(self) + "\n") return rendered, resolved, initial_point
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
Close.render()
.
class DrawingContext
-
Expand source code Browse git
class DrawingContext: """ Base context for a drawing in a PDF This context is not stylable and is mainly responsible for transforming path drawing coordinates into user coordinates (i.e. it ensures that the output drawing is correctly scaled). """ __slots__ = ("_subitems",) def __init__(self): self._subitems: list[Union[GraphicsContext, PaintedPath, PaintComposite]] = [] def add_item( self, item: Union["GraphicsContext", "PaintedPath", "PaintComposite"], _copy: bool = True, ) -> None: """ Append an item to this drawing context Args: item (GraphicsContext, PaintedPath): the item to be appended. _copy (bool): if true (the default), the item will be copied before being appended. This prevents modifications to a referenced object from "retroactively" altering its style/shape and should be disabled with caution. """ if not isinstance(item, (GraphicsContext, PaintedPath, PaintComposite)): raise TypeError(f"{item} doesn't belong in a DrawingContext") if _copy: item = deepcopy(item) self._subitems.append(item) @staticmethod def _setup_render_prereqs( style, first_point: Point, scale: float, height: float ) -> tuple[list[str], GraphicsStyle, Renderable]: style.auto_close = True style.paint_rule = PathPaintRule.AUTO style.intersection_rule = IntersectionRule.NONZERO last_item = Move(first_point) scale, last_item = ( Transform.scaling(x=1, y=-1) .about(x=0, y=height / 2) .scale(scale) .render(last_item) ) render_list = ["q", scale] return render_list, style, last_item def render( self, resource_registry: "ResourceCatalog", first_point: Point, scale: float, height: float, starting_style: GraphicsStyle, ) -> str: """ Render the drawing context to PDF format. Args: resource_registry (ResourceCatalog): the parent document's graphics state registry. first_point (Point): the starting point to use if the first path element is a relative element. scale (Number): the scale factor to convert from PDF pt units into the document's semantic units (e.g. mm or in). height (Number): the page height. This is used to remap the coordinates to be from the top-left corner of the page (matching fpdf's behavior) instead of the PDF native behavior of bottom-left. starting_style (GraphicsStyle): the base style for this drawing context, derived from the document's current style defaults. Returns: A string composed of the PDF representation of all the paths and groups in this context (an empty string is returned if there are no paths or groups) """ if not self._subitems: return "" render_list, style, last_item = self._setup_render_prereqs( starting_style, first_point, scale, height ) for item in self._subitems: rendered, last_item, first_point = item.render( resource_registry, style, last_item, first_point ) if rendered: render_list.append(rendered) # there was nothing to render: the only items are the start group and scale # transform. if len(render_list) == 2: return "" if ( style.soft_mask and style.soft_mask is not GraphicsStyle.INHERIT and style.soft_mask.object_id == 0 ): style.soft_mask.object_id = resource_registry.register_soft_mask( style.soft_mask ) style_dict_name = resource_registry.register_graphics_style(style) if style_dict_name is not None: render_list.insert(2, f"{render_pdf_primitive(style_dict_name)} gs") render_list.insert( 3, render_pdf_primitive(style.stroke_dash_pattern) + f" {number_to_str(style.stroke_dash_phase)} d", ) render_list.append("Q") return " ".join(render_list) def render_debug( self, resource_registry, first_point, scale, height, starting_style, debug_stream, ): """ Render the drawing context to PDF format. Args: resource_registry (ResourceCatalog): the parent document's graphics state registry. first_point (Point): the starting point to use if the first path element is a relative element. scale (Number): the scale factor to convert from PDF pt units into the document's semantic units (e.g. mm or in). height (Number): the page height. This is used to remap the coordinates to be from the top-left corner of the page (matching fpdf's behavior) instead of the PDF native behavior of bottom-left. starting_style (GraphicsStyle): the base style for this drawing context, derived from the document's current style defaults. debug_stream (TextIO): a text stream to which a debug representation of the drawing structure will be written. Returns: A string composed of the PDF representation of all the paths and groups in this context (an empty string is returned if there are no paths or groups) """ render_list, style, last_item = self._setup_render_prereqs( starting_style, first_point, scale, height ) debug_stream.write("ROOT\n") for child in self._subitems[:-1]: debug_stream.write(" ├─ ") rendered, last_item = child.render_debug( resource_registry, style, last_item, debug_stream, " │ " ) if rendered: render_list.append(rendered) if self._subitems: debug_stream.write(" └─ ") rendered, last_item, first_point = self._subitems[-1].render_debug( resource_registry, style, last_item, first_point, debug_stream, " " ) if rendered: render_list.append(rendered) # there was nothing to render: the only items are the start group and scale # transform. if len(render_list) == 2: return "" if ( style.soft_mask and style.soft_mask is not GraphicsStyle.INHERIT and style.soft_mask.object_id == 0 ): style.soft_mask.object_id = resource_registry.register_soft_mask( style.soft_mask ) style_dict_name = resource_registry.register_graphics_style(style) if style_dict_name is not None: render_list.insert(2, f"{render_pdf_primitive(style_dict_name)} gs") render_list.insert( 3, render_pdf_primitive(style.stroke_dash_pattern) + f" {number_to_str(style.stroke_dash_phase)} d", ) render_list.append("Q") return " ".join(render_list) return ""
Base context for a drawing in a PDF
This context is not stylable and is mainly responsible for transforming path drawing coordinates into user coordinates (i.e. it ensures that the output drawing is correctly scaled).
Methods
def add_item(self,
item: ForwardRef('GraphicsContext') | ForwardRef('PaintedPath') | ForwardRef('PaintComposite')) ‑> None-
Expand source code Browse git
def add_item( self, item: Union["GraphicsContext", "PaintedPath", "PaintComposite"], _copy: bool = True, ) -> None: """ Append an item to this drawing context Args: item (GraphicsContext, PaintedPath): the item to be appended. _copy (bool): if true (the default), the item will be copied before being appended. This prevents modifications to a referenced object from "retroactively" altering its style/shape and should be disabled with caution. """ if not isinstance(item, (GraphicsContext, PaintedPath, PaintComposite)): raise TypeError(f"{item} doesn't belong in a DrawingContext") if _copy: item = deepcopy(item) self._subitems.append(item)
Append an item to this drawing context
Args
item
:GraphicsContext, PaintedPath
- the item to be appended.
_copy
:bool
- if true (the default), the item will be copied before being appended. This prevents modifications to a referenced object from "retroactively" altering its style/shape and should be disabled with caution.
def render(self,
resource_registry: ResourceCatalog,
first_point: Point,
scale: float,
height: float,
starting_style: GraphicsStyle) ‑> str-
Expand source code Browse git
def render( self, resource_registry: "ResourceCatalog", first_point: Point, scale: float, height: float, starting_style: GraphicsStyle, ) -> str: """ Render the drawing context to PDF format. Args: resource_registry (ResourceCatalog): the parent document's graphics state registry. first_point (Point): the starting point to use if the first path element is a relative element. scale (Number): the scale factor to convert from PDF pt units into the document's semantic units (e.g. mm or in). height (Number): the page height. This is used to remap the coordinates to be from the top-left corner of the page (matching fpdf's behavior) instead of the PDF native behavior of bottom-left. starting_style (GraphicsStyle): the base style for this drawing context, derived from the document's current style defaults. Returns: A string composed of the PDF representation of all the paths and groups in this context (an empty string is returned if there are no paths or groups) """ if not self._subitems: return "" render_list, style, last_item = self._setup_render_prereqs( starting_style, first_point, scale, height ) for item in self._subitems: rendered, last_item, first_point = item.render( resource_registry, style, last_item, first_point ) if rendered: render_list.append(rendered) # there was nothing to render: the only items are the start group and scale # transform. if len(render_list) == 2: return "" if ( style.soft_mask and style.soft_mask is not GraphicsStyle.INHERIT and style.soft_mask.object_id == 0 ): style.soft_mask.object_id = resource_registry.register_soft_mask( style.soft_mask ) style_dict_name = resource_registry.register_graphics_style(style) if style_dict_name is not None: render_list.insert(2, f"{render_pdf_primitive(style_dict_name)} gs") render_list.insert( 3, render_pdf_primitive(style.stroke_dash_pattern) + f" {number_to_str(style.stroke_dash_phase)} d", ) render_list.append("Q") return " ".join(render_list)
Render the drawing context to PDF format.
Args
resource_registry
:ResourceCatalog
- the parent document's graphics state registry.
first_point
:Point
- the starting point to use if the first path element is a relative element.
scale
:Number
- the scale factor to convert from PDF pt units into the document's semantic units (e.g. mm or in).
height
:Number
- the page height. This is used to remap the coordinates to be from the top-left corner of the page (matching fpdf's behavior) instead of the PDF native behavior of bottom-left.
starting_style
:GraphicsStyle
- the base style for this drawing context, derived from the document's current style defaults.
Returns
A string composed of the PDF representation of all the paths and groups in this context (an empty string is returned if there are no paths or groups)
def render_debug(self, resource_registry, first_point, scale, height, starting_style, debug_stream)
-
Expand source code Browse git
def render_debug( self, resource_registry, first_point, scale, height, starting_style, debug_stream, ): """ Render the drawing context to PDF format. Args: resource_registry (ResourceCatalog): the parent document's graphics state registry. first_point (Point): the starting point to use if the first path element is a relative element. scale (Number): the scale factor to convert from PDF pt units into the document's semantic units (e.g. mm or in). height (Number): the page height. This is used to remap the coordinates to be from the top-left corner of the page (matching fpdf's behavior) instead of the PDF native behavior of bottom-left. starting_style (GraphicsStyle): the base style for this drawing context, derived from the document's current style defaults. debug_stream (TextIO): a text stream to which a debug representation of the drawing structure will be written. Returns: A string composed of the PDF representation of all the paths and groups in this context (an empty string is returned if there are no paths or groups) """ render_list, style, last_item = self._setup_render_prereqs( starting_style, first_point, scale, height ) debug_stream.write("ROOT\n") for child in self._subitems[:-1]: debug_stream.write(" ├─ ") rendered, last_item = child.render_debug( resource_registry, style, last_item, debug_stream, " │ " ) if rendered: render_list.append(rendered) if self._subitems: debug_stream.write(" └─ ") rendered, last_item, first_point = self._subitems[-1].render_debug( resource_registry, style, last_item, first_point, debug_stream, " " ) if rendered: render_list.append(rendered) # there was nothing to render: the only items are the start group and scale # transform. if len(render_list) == 2: return "" if ( style.soft_mask and style.soft_mask is not GraphicsStyle.INHERIT and style.soft_mask.object_id == 0 ): style.soft_mask.object_id = resource_registry.register_soft_mask( style.soft_mask ) style_dict_name = resource_registry.register_graphics_style(style) if style_dict_name is not None: render_list.insert(2, f"{render_pdf_primitive(style_dict_name)} gs") render_list.insert( 3, render_pdf_primitive(style.stroke_dash_pattern) + f" {number_to_str(style.stroke_dash_phase)} d", ) render_list.append("Q") return " ".join(render_list) return ""
Render the drawing context to PDF format.
Args
resource_registry
:ResourceCatalog
- the parent document's graphics state registry.
first_point
:Point
- the starting point to use if the first path element is a relative element.
scale
:Number
- the scale factor to convert from PDF pt units into the document's semantic units (e.g. mm or in).
height
:Number
- the page height. This is used to remap the coordinates to be from the top-left corner of the page (matching fpdf's behavior) instead of the PDF native behavior of bottom-left.
starting_style
:GraphicsStyle
- the base style for this drawing context, derived from the document's current style defaults.
debug_stream
:TextIO
- a text stream to which a debug representation of the drawing structure will be written.
Returns
A string composed of the PDF representation of all the paths and groups in this context (an empty string is returned if there are no paths or groups)
class Ellipse (radii: Point,
center: Point)-
Expand source code Browse git
class Ellipse(NamedTuple): """ An ellipse. See: `PaintedPath.ellipse` """ radii: Point """The x- and y-radii of the ellipse""" center: Point """The abscissa and ordinate of the center of the ellipse""" def _decompose(self) -> list[Renderable]: items = [] rx = abs(self.radii.x) ry = abs(self.radii.y) cx, cy = self.center arc_rad = Point(rx, ry) # this isn't the most efficient way to do this, computationally, but it's # internally consistent. if (rx != 0) and (ry != 0): items.append(Move(Point(cx + rx, cy))) items.append(Arc(arc_rad, 0, False, True, Point(cx, cy + ry))) items.append(Arc(arc_rad, 0, False, True, Point(cx - rx, cy))) items.append(Arc(arc_rad, 0, False, True, Point(cx, cy - ry))) items.append(Arc(arc_rad, 0, False, True, Point(cx + rx, cy))) items.append(Close()) return items def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this ellipse by decomposing it and merging the bounding boxes of its components. """ bbox = BoundingBox.empty() current_point = start for item in self._decompose(): b, current_point = item.bounding_box(current_point) bbox = bbox.merge(b) return bbox, self.center @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `Move` to the center of the ellipse. """ components = self._decompose() if not components: return "", last_item, initial_point render_list = [] for item in components: rendered, last_item, initial_point = item.render( resource_registry, style, last_item, initial_point ) render_list.append(rendered) return " ".join(render_list), Move(self.center), initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Ellipse.render`. """ components = self._decompose() debug_stream.write(f"{self} resolved to:\n") if not components: debug_stream.write(pfx + " └─ nothing\n") return "", last_item, initial_point render_list = [] for item in components[:-1]: rendered, last_item, initial_point = item.render( resource_registry, style, last_item, initial_point ) debug_stream.write(pfx + f" ├─ {item}\n") render_list.append(rendered) rendered, last_item, initial_point = components[-1].render( resource_registry, style, last_item, initial_point ) debug_stream.write(pfx + f" └─ {components[-1]}\n") render_list.append(rendered) return " ".join(render_list), Move(self.center), initial_point
An ellipse.
Ancestors
- builtins.tuple
Instance variables
var center : Point
-
Expand source code Browse git
class Ellipse(NamedTuple): """ An ellipse. See: `PaintedPath.ellipse` """ radii: Point """The x- and y-radii of the ellipse""" center: Point """The abscissa and ordinate of the center of the ellipse""" def _decompose(self) -> list[Renderable]: items = [] rx = abs(self.radii.x) ry = abs(self.radii.y) cx, cy = self.center arc_rad = Point(rx, ry) # this isn't the most efficient way to do this, computationally, but it's # internally consistent. if (rx != 0) and (ry != 0): items.append(Move(Point(cx + rx, cy))) items.append(Arc(arc_rad, 0, False, True, Point(cx, cy + ry))) items.append(Arc(arc_rad, 0, False, True, Point(cx - rx, cy))) items.append(Arc(arc_rad, 0, False, True, Point(cx, cy - ry))) items.append(Arc(arc_rad, 0, False, True, Point(cx + rx, cy))) items.append(Close()) return items def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this ellipse by decomposing it and merging the bounding boxes of its components. """ bbox = BoundingBox.empty() current_point = start for item in self._decompose(): b, current_point = item.bounding_box(current_point) bbox = bbox.merge(b) return bbox, self.center @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `Move` to the center of the ellipse. """ components = self._decompose() if not components: return "", last_item, initial_point render_list = [] for item in components: rendered, last_item, initial_point = item.render( resource_registry, style, last_item, initial_point ) render_list.append(rendered) return " ".join(render_list), Move(self.center), initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Ellipse.render`. """ components = self._decompose() debug_stream.write(f"{self} resolved to:\n") if not components: debug_stream.write(pfx + " └─ nothing\n") return "", last_item, initial_point render_list = [] for item in components[:-1]: rendered, last_item, initial_point = item.render( resource_registry, style, last_item, initial_point ) debug_stream.write(pfx + f" ├─ {item}\n") render_list.append(rendered) rendered, last_item, initial_point = components[-1].render( resource_registry, style, last_item, initial_point ) debug_stream.write(pfx + f" └─ {components[-1]}\n") render_list.append(rendered) return " ".join(render_list), Move(self.center), initial_point
The abscissa and ordinate of the center of the ellipse
var radii : Point
-
Expand source code Browse git
class Ellipse(NamedTuple): """ An ellipse. See: `PaintedPath.ellipse` """ radii: Point """The x- and y-radii of the ellipse""" center: Point """The abscissa and ordinate of the center of the ellipse""" def _decompose(self) -> list[Renderable]: items = [] rx = abs(self.radii.x) ry = abs(self.radii.y) cx, cy = self.center arc_rad = Point(rx, ry) # this isn't the most efficient way to do this, computationally, but it's # internally consistent. if (rx != 0) and (ry != 0): items.append(Move(Point(cx + rx, cy))) items.append(Arc(arc_rad, 0, False, True, Point(cx, cy + ry))) items.append(Arc(arc_rad, 0, False, True, Point(cx - rx, cy))) items.append(Arc(arc_rad, 0, False, True, Point(cx, cy - ry))) items.append(Arc(arc_rad, 0, False, True, Point(cx + rx, cy))) items.append(Close()) return items def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this ellipse by decomposing it and merging the bounding boxes of its components. """ bbox = BoundingBox.empty() current_point = start for item in self._decompose(): b, current_point = item.bounding_box(current_point) bbox = bbox.merge(b) return bbox, self.center @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `Move` to the center of the ellipse. """ components = self._decompose() if not components: return "", last_item, initial_point render_list = [] for item in components: rendered, last_item, initial_point = item.render( resource_registry, style, last_item, initial_point ) render_list.append(rendered) return " ".join(render_list), Move(self.center), initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Ellipse.render`. """ components = self._decompose() debug_stream.write(f"{self} resolved to:\n") if not components: debug_stream.write(pfx + " └─ nothing\n") return "", last_item, initial_point render_list = [] for item in components[:-1]: rendered, last_item, initial_point = item.render( resource_registry, style, last_item, initial_point ) debug_stream.write(pfx + f" ├─ {item}\n") render_list.append(rendered) rendered, last_item, initial_point = components[-1].render( resource_registry, style, last_item, initial_point ) debug_stream.write(pfx + f" └─ {components[-1]}\n") render_list.append(rendered) return " ".join(render_list), Move(self.center), initial_point
The x- and y-radii of the ellipse
Methods
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this ellipse by decomposing it and merging the bounding boxes of its components. """ bbox = BoundingBox.empty() current_point = start for item in self._decompose(): b, current_point = item.bounding_box(current_point) bbox = bbox.merge(b) return bbox, self.center
Compute the bounding box of this ellipse by decomposing it and merging the bounding boxes of its components.
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
@force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `Move` to the center of the ellipse. """ components = self._decompose() if not components: return "", last_item, initial_point render_list = [] for item in components: rendered, last_item, initial_point = item.render( resource_registry, style, last_item, initial_point ) render_list.append(rendered) return " ".join(render_list), Move(self.center), initial_point
Render this path element to its PDF representation.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
Returns
a tuple of
(str, new_last_item)
, wherenew_last_item
is a resolvedMove
to the center of the ellipse. def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
@force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Ellipse.render`. """ components = self._decompose() debug_stream.write(f"{self} resolved to:\n") if not components: debug_stream.write(pfx + " └─ nothing\n") return "", last_item, initial_point render_list = [] for item in components[:-1]: rendered, last_item, initial_point = item.render( resource_registry, style, last_item, initial_point ) debug_stream.write(pfx + f" ├─ {item}\n") render_list.append(rendered) rendered, last_item, initial_point = components[-1].render( resource_registry, style, last_item, initial_point ) debug_stream.write(pfx + f" └─ {components[-1]}\n") render_list.append(rendered) return " ".join(render_list), Move(self.center), initial_point
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
Ellipse.render()
.
class GradientPaint (gradient: Gradient,
units: GradientUnits | str = GradientUnits.USER_SPACE_ON_USE,
gradient_transform: ForwardRef('Transform') | None = None)-
Expand source code Browse git
class GradientPaint: """Fill/stroke paint using a gradient""" __slots__ = ("gradient", "units", "gradient_transform") def __init__( self, gradient: "Gradient", units: Union[GradientUnits, str] = GradientUnits.USER_SPACE_ON_USE, gradient_transform: Optional["Transform"] = None, ): self.gradient = gradient self.units = GradientUnits.coerce(units) self.gradient_transform = gradient_transform or Transform.identity() def _matrix_for(self, bbox: Optional["BoundingBox"]) -> "Transform": """Return the final /Matrix for this gradient, given an optional bbox.""" if self.units == GradientUnits.OBJECT_BOUNDING_BOX: if bbox is None: raise RuntimeError( "GradientPaint requires bbox for objectBoundingBox units" ) # Map [0,1]x[0,1] object space, then apply gradient_transform matrix_bbox = Transform( a=bbox.width, b=0, c=0, d=bbox.height, e=bbox.x0, f=bbox.y0 ) return self.gradient_transform @ matrix_bbox # userSpaceOnUse: only the provided gradient_transform return self.gradient_transform def _register_pattern(self, resource_catalog, matrix: "Transform") -> str: """Create a Pattern with the given matrix, register shading+pattern, return pattern name.""" resource_catalog.add(PDFResourceType.SHADING, self.gradient, None) pattern = Pattern(self.gradient).set_matrix(matrix) return resource_catalog.add(PDFResourceType.PATTERN, pattern, None) def emit_fill(self, resource_catalog, bbox: Optional["BoundingBox"] = None) -> str: matrix = self._matrix_for(bbox) pname = self._register_pattern(resource_catalog, matrix) return f"/Pattern cs /{pname} scn" def emit_stroke( self, resource_catalog, bbox: Optional["BoundingBox"] = None ) -> str: matrix = self._matrix_for(bbox) pname = self._register_pattern(resource_catalog, matrix) return f"/Pattern CS /{pname} SCN"
Fill/stroke paint using a gradient
Instance variables
var gradient
-
Expand source code Browse git
class GradientPaint: """Fill/stroke paint using a gradient""" __slots__ = ("gradient", "units", "gradient_transform") def __init__( self, gradient: "Gradient", units: Union[GradientUnits, str] = GradientUnits.USER_SPACE_ON_USE, gradient_transform: Optional["Transform"] = None, ): self.gradient = gradient self.units = GradientUnits.coerce(units) self.gradient_transform = gradient_transform or Transform.identity() def _matrix_for(self, bbox: Optional["BoundingBox"]) -> "Transform": """Return the final /Matrix for this gradient, given an optional bbox.""" if self.units == GradientUnits.OBJECT_BOUNDING_BOX: if bbox is None: raise RuntimeError( "GradientPaint requires bbox for objectBoundingBox units" ) # Map [0,1]x[0,1] object space, then apply gradient_transform matrix_bbox = Transform( a=bbox.width, b=0, c=0, d=bbox.height, e=bbox.x0, f=bbox.y0 ) return self.gradient_transform @ matrix_bbox # userSpaceOnUse: only the provided gradient_transform return self.gradient_transform def _register_pattern(self, resource_catalog, matrix: "Transform") -> str: """Create a Pattern with the given matrix, register shading+pattern, return pattern name.""" resource_catalog.add(PDFResourceType.SHADING, self.gradient, None) pattern = Pattern(self.gradient).set_matrix(matrix) return resource_catalog.add(PDFResourceType.PATTERN, pattern, None) def emit_fill(self, resource_catalog, bbox: Optional["BoundingBox"] = None) -> str: matrix = self._matrix_for(bbox) pname = self._register_pattern(resource_catalog, matrix) return f"/Pattern cs /{pname} scn" def emit_stroke( self, resource_catalog, bbox: Optional["BoundingBox"] = None ) -> str: matrix = self._matrix_for(bbox) pname = self._register_pattern(resource_catalog, matrix) return f"/Pattern CS /{pname} SCN"
var gradient_transform
-
Expand source code Browse git
class GradientPaint: """Fill/stroke paint using a gradient""" __slots__ = ("gradient", "units", "gradient_transform") def __init__( self, gradient: "Gradient", units: Union[GradientUnits, str] = GradientUnits.USER_SPACE_ON_USE, gradient_transform: Optional["Transform"] = None, ): self.gradient = gradient self.units = GradientUnits.coerce(units) self.gradient_transform = gradient_transform or Transform.identity() def _matrix_for(self, bbox: Optional["BoundingBox"]) -> "Transform": """Return the final /Matrix for this gradient, given an optional bbox.""" if self.units == GradientUnits.OBJECT_BOUNDING_BOX: if bbox is None: raise RuntimeError( "GradientPaint requires bbox for objectBoundingBox units" ) # Map [0,1]x[0,1] object space, then apply gradient_transform matrix_bbox = Transform( a=bbox.width, b=0, c=0, d=bbox.height, e=bbox.x0, f=bbox.y0 ) return self.gradient_transform @ matrix_bbox # userSpaceOnUse: only the provided gradient_transform return self.gradient_transform def _register_pattern(self, resource_catalog, matrix: "Transform") -> str: """Create a Pattern with the given matrix, register shading+pattern, return pattern name.""" resource_catalog.add(PDFResourceType.SHADING, self.gradient, None) pattern = Pattern(self.gradient).set_matrix(matrix) return resource_catalog.add(PDFResourceType.PATTERN, pattern, None) def emit_fill(self, resource_catalog, bbox: Optional["BoundingBox"] = None) -> str: matrix = self._matrix_for(bbox) pname = self._register_pattern(resource_catalog, matrix) return f"/Pattern cs /{pname} scn" def emit_stroke( self, resource_catalog, bbox: Optional["BoundingBox"] = None ) -> str: matrix = self._matrix_for(bbox) pname = self._register_pattern(resource_catalog, matrix) return f"/Pattern CS /{pname} SCN"
var units
-
Expand source code Browse git
class GradientPaint: """Fill/stroke paint using a gradient""" __slots__ = ("gradient", "units", "gradient_transform") def __init__( self, gradient: "Gradient", units: Union[GradientUnits, str] = GradientUnits.USER_SPACE_ON_USE, gradient_transform: Optional["Transform"] = None, ): self.gradient = gradient self.units = GradientUnits.coerce(units) self.gradient_transform = gradient_transform or Transform.identity() def _matrix_for(self, bbox: Optional["BoundingBox"]) -> "Transform": """Return the final /Matrix for this gradient, given an optional bbox.""" if self.units == GradientUnits.OBJECT_BOUNDING_BOX: if bbox is None: raise RuntimeError( "GradientPaint requires bbox for objectBoundingBox units" ) # Map [0,1]x[0,1] object space, then apply gradient_transform matrix_bbox = Transform( a=bbox.width, b=0, c=0, d=bbox.height, e=bbox.x0, f=bbox.y0 ) return self.gradient_transform @ matrix_bbox # userSpaceOnUse: only the provided gradient_transform return self.gradient_transform def _register_pattern(self, resource_catalog, matrix: "Transform") -> str: """Create a Pattern with the given matrix, register shading+pattern, return pattern name.""" resource_catalog.add(PDFResourceType.SHADING, self.gradient, None) pattern = Pattern(self.gradient).set_matrix(matrix) return resource_catalog.add(PDFResourceType.PATTERN, pattern, None) def emit_fill(self, resource_catalog, bbox: Optional["BoundingBox"] = None) -> str: matrix = self._matrix_for(bbox) pname = self._register_pattern(resource_catalog, matrix) return f"/Pattern cs /{pname} scn" def emit_stroke( self, resource_catalog, bbox: Optional["BoundingBox"] = None ) -> str: matrix = self._matrix_for(bbox) pname = self._register_pattern(resource_catalog, matrix) return f"/Pattern CS /{pname} SCN"
Methods
def emit_fill(self,
resource_catalog,
bbox: ForwardRef('BoundingBox') | None = None) ‑> str-
Expand source code Browse git
def emit_fill(self, resource_catalog, bbox: Optional["BoundingBox"] = None) -> str: matrix = self._matrix_for(bbox) pname = self._register_pattern(resource_catalog, matrix) return f"/Pattern cs /{pname} scn"
def emit_stroke(self,
resource_catalog,
bbox: ForwardRef('BoundingBox') | None = None) ‑> str-
Expand source code Browse git
def emit_stroke( self, resource_catalog, bbox: Optional["BoundingBox"] = None ) -> str: matrix = self._matrix_for(bbox) pname = self._register_pattern(resource_catalog, matrix) return f"/Pattern CS /{pname} SCN"
class GraphicsContext
-
Expand source code Browse git
class GraphicsContext: """ Page-level container that collects drawable items and renders them into a PDF content stream. Converts model coordinates to PDF user space by applying the provided `scale` and a vertical flip so (0, 0) is the top-left of the page. Wraps output in a saved graphics state (`q … Q`) and registers any required resources (graphics state dictionaries, soft masks, dash pattern). Child items are typically `GraphicsContext`, `PaintedPath`, or `PaintComposite` objects added via `add_item()`. By default, items are deep-copied on insert to avoid later mutations affecting the emitted stream. """ __slots__ = ("style", "path_items", "_transform", "_clipping_path") def __init__(self): self.style: GraphicsStyle = GraphicsStyle() self.path_items: list[Renderable] = [] self._transform: Optional[Transform] = None self._clipping_path: Optional[ClippingPath] = None def __deepcopy__(self, memo): copied = self.__class__() copied.style = deepcopy(self.style, memo) copied.path_items = deepcopy(self.path_items, memo) copied._transform = deepcopy(self.transform, memo) copied._clipping_path = deepcopy(self.clipping_path, memo) return copied @property def transform(self) -> Optional[Transform]: return self._transform @transform.setter def transform(self, tf: Transform) -> None: self._transform = tf @property def clipping_path(self) -> Optional[ClippingPath]: """The `ClippingPath` for this graphics context.""" return self._clipping_path @clipping_path.setter def clipping_path(self, new_clipath: ClippingPath) -> None: self._clipping_path = new_clipath def add_item(self, item: Renderable, _copy: bool = True) -> None: """ Add a path element to this graphics context. Args: item: the path element to add. May be a primitive element or another `GraphicsContext` or a `PaintedPath`. _copy (bool): if true (the default), the item will be copied before being appended. This prevents modifications to a referenced object from "retroactively" altering its style/shape and should be disabled with caution. """ if _copy: item = deepcopy(item) self.path_items.append(item) def remove_last_item(self) -> None: del self.path_items[-1] def merge(self, other_context: "GraphicsContext") -> None: """Copy another `GraphicsContext`'s path items into this one.""" self.path_items.extend(other_context.path_items) @force_nodocument def build_render_list( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, debug_stream=None, pfx: Optional[str] = None, _push_stack: bool = True, ): """ Build a list composed of all all the individual elements rendered. This is used by `PaintedPath` and `ClippingPath` to reuse the `GraphicsContext` rendering process while still being able to inject some path specific items (e.g. the painting directive) before the render is collapsed into a single string. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). _push_stack (bool): if True, wrap the resulting render list in a push/pop graphics stack directive pair. Returns: `tuple[list[str], last_item]` where `last_item` is the past path element in this `GraphicsContext` """ render_list = [] if self.path_items: if debug_stream is not None: debug_stream.write(f"{self.__class__.__name__}") merged_style = style.__class__.merge(style, self.style) if debug_stream is not None: if self._transform: debug_stream.write(f"({self._transform})") styles_dbg = [] for attr in merged_style.MERGE_PROPERTIES: val = getattr(merged_style, attr) if val is not merged_style.INHERIT: if getattr(self.style, attr) is merged_style.INHERIT: inherited = " (inherited)" else: inherited = "" styles_dbg.append(f"{attr}: {val}{inherited}") if styles_dbg: debug_stream.write(" {\n") for style_dbg_line in styles_dbg: debug_stream.write(pfx + " ") debug_stream.write(style_dbg_line) debug_stream.write("\n") debug_stream.write(pfx + "}┐\n") else: debug_stream.write("\n") NO_EMIT_SET = (None, merged_style.INHERIT) emit_style = self.style if merged_style.allow_transparency != self.style.allow_transparency: emit_style = deepcopy(self.style) emit_style.allow_transparency = merged_style.allow_transparency # in order to decouple the dash pattern and the dash phase at the API layer, # we have to perform additional logic here to recombine them. We can rely # on these being serializable because we always get a sane style on the # drawing context. dash_pattern = merged_style.stroke_dash_pattern dash_phase = merged_style.stroke_dash_phase emit_dash = None if ( dash_pattern != style.stroke_dash_pattern or dash_phase != style.stroke_dash_phase ): if emit_style is self.style: emit_style = deepcopy(emit_style) emit_style.stroke_dash_pattern = dash_pattern emit_style.stroke_dash_phase = dash_phase emit_dash = (dash_pattern, dash_phase) if ( emit_style.soft_mask and emit_style.soft_mask is not GraphicsStyle.INHERIT and emit_style.soft_mask.object_id == 0 ): emit_style.soft_mask.object_id = resource_registry.register_soft_mask( emit_style.soft_mask ) style_dict_name = resource_registry.register_graphics_style(emit_style) if style_dict_name is not None: render_list.append(f"{render_pdf_primitive(style_dict_name)} gs") # we can't set color in the graphics state context dictionary, so we have to # manually inherit it and emit it here. fill_color = self.style.fill_color stroke_color = self.style.stroke_color if fill_color not in NO_EMIT_SET: if isinstance(fill_color, GradientPaint): render_list.append( fill_color.emit_fill( resource_registry, self.bounding_box( initial_point, style=self.style, expand_for_stroke=False )[0], ) ) else: render_list.append(fill_color.serialize().lower()) if stroke_color not in NO_EMIT_SET: if isinstance(stroke_color, GradientPaint): render_list.append( stroke_color.emit_stroke( resource_registry, self.bounding_box( initial_point, style=self.style, expand_for_stroke=False )[0], ) ) else: render_list.append(stroke_color.serialize().upper()) if emit_dash is not None: render_list.append( render_pdf_primitive(emit_dash[0]) + f" {number_to_str(emit_dash[1])} d" ) if debug_stream: if self.clipping_path is not None: debug_stream.write(pfx + " ├─ ") rendered_cpath, _, __ = self.clipping_path.render_debug( resource_registry, merged_style, last_item, initial_point, debug_stream, pfx + " │ ", ) if rendered_cpath: render_list.append(rendered_cpath) for item in self.path_items[:-1]: debug_stream.write(pfx + " ├─ ") rendered, last_item, initial_point = item.render_debug( resource_registry, merged_style, last_item, initial_point, debug_stream, pfx + " │ ", ) if rendered: render_list.append(rendered) debug_stream.write(pfx + " └─ ") rendered, last_item, initial_point = self.path_items[-1].render_debug( resource_registry, merged_style, last_item, initial_point, debug_stream, pfx + " ", ) if rendered: render_list.append(rendered) else: if self.clipping_path is not None: rendered_cpath, _, __ = self.clipping_path.render( resource_registry, merged_style, last_item, initial_point ) if rendered_cpath: render_list.append(rendered_cpath) for item in self.path_items: rendered, last_item, initial_point = item.render( resource_registry, merged_style, last_item, initial_point ) if rendered: render_list.append(rendered) # insert transform before points if self.transform is not None: render_list.insert(0, self.transform.render(last_item)[0]) if _push_stack: render_list.insert(0, "q") render_list.append("Q") return render_list, last_item, initial_point def bounding_box( self, start: Point, style: Optional[GraphicsStyle] = None, expand_for_stroke=True, ) -> tuple[BoundingBox, Point]: """ Compute bbox of all path items. We: 1) recurse with accumulated CTM, 2) merge child bboxes already transformed to this level, 3) at the end, expand once for stroke using the worst-case CTM row norms. """ I = Transform.identity() def walk( ctx: "GraphicsContext", current_point: Point, ambient_style: Optional[GraphicsStyle], accum_tf: Transform, ) -> tuple[BoundingBox, Point, float, float]: bbox = BoundingBox.empty() tf = accum_tf @ (ctx.transform or I) merged_style = ( ambient_style.__class__.merge(ambient_style, ctx.style) if ambient_style else ctx.style ) max_nx, max_ny = tf.row_norms() for item in ctx.path_items: if isinstance(item, GraphicsContext): child_bbox, end_point, cnx, cny = walk( item, current_point, merged_style, tf ) bbox = bbox.merge(child_bbox) # child bbox already in this space current_point = end_point max_nx = max(max_nx, cnx) max_ny = max(max_ny, cny) elif hasattr(item, "bounding_box"): item_bbox, end_point = item.bounding_box(current_point) bbox = bbox.merge(item_bbox.transformed(tf)) current_point = end_point return bbox, current_point, max_nx, max_ny # 1) geometric + collect CTM scales geom_bbox, end_pt, nx, ny = walk(self, start, style, I) final_bbox = geom_bbox if expand_for_stroke: # 2) expand once for stroke with the effective style at *this* level effective_style = ( style.__class__.merge(style, self.style) if style else self.style ) final_bbox = geom_bbox.expanded_to_stroke( effective_style, row_norms=(nx, ny) ) return final_bbox, end_pt def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, debug_stream=None, pfx=None, _push_stack=True, ) -> tuple[str, Renderable, Point]: render_list, last_item, initial_point = self.build_render_list( resource_registry, style, last_item, initial_point, debug_stream, pfx, _push_stack=_push_stack, ) return " ".join(render_list), last_item, initial_point def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx, _push_stack=True, ): return self.render( resource_registry, style, last_item, initial_point, debug_stream, pfx, _push_stack=_push_stack, )
Page-level container that collects drawable items and renders them into a PDF content stream.
Converts model coordinates to PDF user space by applying the provided
scale
and a vertical flip so (0, 0) is the top-left of the page.Wraps output in a saved graphics state (
q … Q
) and registers any required resources (graphics state dictionaries, soft masks, dash pattern).Child items are typically
GraphicsContext
,PaintedPath
, orPaintComposite
objects added viaadd_item()
. By default, items are deep-copied on insert to avoid later mutations affecting the emitted stream.Instance variables
prop clipping_path : ClippingPath | None
-
Expand source code Browse git
@property def clipping_path(self) -> Optional[ClippingPath]: """The `ClippingPath` for this graphics context.""" return self._clipping_path
The
ClippingPath
for this graphics context. var path_items
-
Expand source code Browse git
class GraphicsContext: """ Page-level container that collects drawable items and renders them into a PDF content stream. Converts model coordinates to PDF user space by applying the provided `scale` and a vertical flip so (0, 0) is the top-left of the page. Wraps output in a saved graphics state (`q … Q`) and registers any required resources (graphics state dictionaries, soft masks, dash pattern). Child items are typically `GraphicsContext`, `PaintedPath`, or `PaintComposite` objects added via `add_item()`. By default, items are deep-copied on insert to avoid later mutations affecting the emitted stream. """ __slots__ = ("style", "path_items", "_transform", "_clipping_path") def __init__(self): self.style: GraphicsStyle = GraphicsStyle() self.path_items: list[Renderable] = [] self._transform: Optional[Transform] = None self._clipping_path: Optional[ClippingPath] = None def __deepcopy__(self, memo): copied = self.__class__() copied.style = deepcopy(self.style, memo) copied.path_items = deepcopy(self.path_items, memo) copied._transform = deepcopy(self.transform, memo) copied._clipping_path = deepcopy(self.clipping_path, memo) return copied @property def transform(self) -> Optional[Transform]: return self._transform @transform.setter def transform(self, tf: Transform) -> None: self._transform = tf @property def clipping_path(self) -> Optional[ClippingPath]: """The `ClippingPath` for this graphics context.""" return self._clipping_path @clipping_path.setter def clipping_path(self, new_clipath: ClippingPath) -> None: self._clipping_path = new_clipath def add_item(self, item: Renderable, _copy: bool = True) -> None: """ Add a path element to this graphics context. Args: item: the path element to add. May be a primitive element or another `GraphicsContext` or a `PaintedPath`. _copy (bool): if true (the default), the item will be copied before being appended. This prevents modifications to a referenced object from "retroactively" altering its style/shape and should be disabled with caution. """ if _copy: item = deepcopy(item) self.path_items.append(item) def remove_last_item(self) -> None: del self.path_items[-1] def merge(self, other_context: "GraphicsContext") -> None: """Copy another `GraphicsContext`'s path items into this one.""" self.path_items.extend(other_context.path_items) @force_nodocument def build_render_list( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, debug_stream=None, pfx: Optional[str] = None, _push_stack: bool = True, ): """ Build a list composed of all all the individual elements rendered. This is used by `PaintedPath` and `ClippingPath` to reuse the `GraphicsContext` rendering process while still being able to inject some path specific items (e.g. the painting directive) before the render is collapsed into a single string. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). _push_stack (bool): if True, wrap the resulting render list in a push/pop graphics stack directive pair. Returns: `tuple[list[str], last_item]` where `last_item` is the past path element in this `GraphicsContext` """ render_list = [] if self.path_items: if debug_stream is not None: debug_stream.write(f"{self.__class__.__name__}") merged_style = style.__class__.merge(style, self.style) if debug_stream is not None: if self._transform: debug_stream.write(f"({self._transform})") styles_dbg = [] for attr in merged_style.MERGE_PROPERTIES: val = getattr(merged_style, attr) if val is not merged_style.INHERIT: if getattr(self.style, attr) is merged_style.INHERIT: inherited = " (inherited)" else: inherited = "" styles_dbg.append(f"{attr}: {val}{inherited}") if styles_dbg: debug_stream.write(" {\n") for style_dbg_line in styles_dbg: debug_stream.write(pfx + " ") debug_stream.write(style_dbg_line) debug_stream.write("\n") debug_stream.write(pfx + "}┐\n") else: debug_stream.write("\n") NO_EMIT_SET = (None, merged_style.INHERIT) emit_style = self.style if merged_style.allow_transparency != self.style.allow_transparency: emit_style = deepcopy(self.style) emit_style.allow_transparency = merged_style.allow_transparency # in order to decouple the dash pattern and the dash phase at the API layer, # we have to perform additional logic here to recombine them. We can rely # on these being serializable because we always get a sane style on the # drawing context. dash_pattern = merged_style.stroke_dash_pattern dash_phase = merged_style.stroke_dash_phase emit_dash = None if ( dash_pattern != style.stroke_dash_pattern or dash_phase != style.stroke_dash_phase ): if emit_style is self.style: emit_style = deepcopy(emit_style) emit_style.stroke_dash_pattern = dash_pattern emit_style.stroke_dash_phase = dash_phase emit_dash = (dash_pattern, dash_phase) if ( emit_style.soft_mask and emit_style.soft_mask is not GraphicsStyle.INHERIT and emit_style.soft_mask.object_id == 0 ): emit_style.soft_mask.object_id = resource_registry.register_soft_mask( emit_style.soft_mask ) style_dict_name = resource_registry.register_graphics_style(emit_style) if style_dict_name is not None: render_list.append(f"{render_pdf_primitive(style_dict_name)} gs") # we can't set color in the graphics state context dictionary, so we have to # manually inherit it and emit it here. fill_color = self.style.fill_color stroke_color = self.style.stroke_color if fill_color not in NO_EMIT_SET: if isinstance(fill_color, GradientPaint): render_list.append( fill_color.emit_fill( resource_registry, self.bounding_box( initial_point, style=self.style, expand_for_stroke=False )[0], ) ) else: render_list.append(fill_color.serialize().lower()) if stroke_color not in NO_EMIT_SET: if isinstance(stroke_color, GradientPaint): render_list.append( stroke_color.emit_stroke( resource_registry, self.bounding_box( initial_point, style=self.style, expand_for_stroke=False )[0], ) ) else: render_list.append(stroke_color.serialize().upper()) if emit_dash is not None: render_list.append( render_pdf_primitive(emit_dash[0]) + f" {number_to_str(emit_dash[1])} d" ) if debug_stream: if self.clipping_path is not None: debug_stream.write(pfx + " ├─ ") rendered_cpath, _, __ = self.clipping_path.render_debug( resource_registry, merged_style, last_item, initial_point, debug_stream, pfx + " │ ", ) if rendered_cpath: render_list.append(rendered_cpath) for item in self.path_items[:-1]: debug_stream.write(pfx + " ├─ ") rendered, last_item, initial_point = item.render_debug( resource_registry, merged_style, last_item, initial_point, debug_stream, pfx + " │ ", ) if rendered: render_list.append(rendered) debug_stream.write(pfx + " └─ ") rendered, last_item, initial_point = self.path_items[-1].render_debug( resource_registry, merged_style, last_item, initial_point, debug_stream, pfx + " ", ) if rendered: render_list.append(rendered) else: if self.clipping_path is not None: rendered_cpath, _, __ = self.clipping_path.render( resource_registry, merged_style, last_item, initial_point ) if rendered_cpath: render_list.append(rendered_cpath) for item in self.path_items: rendered, last_item, initial_point = item.render( resource_registry, merged_style, last_item, initial_point ) if rendered: render_list.append(rendered) # insert transform before points if self.transform is not None: render_list.insert(0, self.transform.render(last_item)[0]) if _push_stack: render_list.insert(0, "q") render_list.append("Q") return render_list, last_item, initial_point def bounding_box( self, start: Point, style: Optional[GraphicsStyle] = None, expand_for_stroke=True, ) -> tuple[BoundingBox, Point]: """ Compute bbox of all path items. We: 1) recurse with accumulated CTM, 2) merge child bboxes already transformed to this level, 3) at the end, expand once for stroke using the worst-case CTM row norms. """ I = Transform.identity() def walk( ctx: "GraphicsContext", current_point: Point, ambient_style: Optional[GraphicsStyle], accum_tf: Transform, ) -> tuple[BoundingBox, Point, float, float]: bbox = BoundingBox.empty() tf = accum_tf @ (ctx.transform or I) merged_style = ( ambient_style.__class__.merge(ambient_style, ctx.style) if ambient_style else ctx.style ) max_nx, max_ny = tf.row_norms() for item in ctx.path_items: if isinstance(item, GraphicsContext): child_bbox, end_point, cnx, cny = walk( item, current_point, merged_style, tf ) bbox = bbox.merge(child_bbox) # child bbox already in this space current_point = end_point max_nx = max(max_nx, cnx) max_ny = max(max_ny, cny) elif hasattr(item, "bounding_box"): item_bbox, end_point = item.bounding_box(current_point) bbox = bbox.merge(item_bbox.transformed(tf)) current_point = end_point return bbox, current_point, max_nx, max_ny # 1) geometric + collect CTM scales geom_bbox, end_pt, nx, ny = walk(self, start, style, I) final_bbox = geom_bbox if expand_for_stroke: # 2) expand once for stroke with the effective style at *this* level effective_style = ( style.__class__.merge(style, self.style) if style else self.style ) final_bbox = geom_bbox.expanded_to_stroke( effective_style, row_norms=(nx, ny) ) return final_bbox, end_pt def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, debug_stream=None, pfx=None, _push_stack=True, ) -> tuple[str, Renderable, Point]: render_list, last_item, initial_point = self.build_render_list( resource_registry, style, last_item, initial_point, debug_stream, pfx, _push_stack=_push_stack, ) return " ".join(render_list), last_item, initial_point def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx, _push_stack=True, ): return self.render( resource_registry, style, last_item, initial_point, debug_stream, pfx, _push_stack=_push_stack, )
var style
-
Expand source code Browse git
class GraphicsContext: """ Page-level container that collects drawable items and renders them into a PDF content stream. Converts model coordinates to PDF user space by applying the provided `scale` and a vertical flip so (0, 0) is the top-left of the page. Wraps output in a saved graphics state (`q … Q`) and registers any required resources (graphics state dictionaries, soft masks, dash pattern). Child items are typically `GraphicsContext`, `PaintedPath`, or `PaintComposite` objects added via `add_item()`. By default, items are deep-copied on insert to avoid later mutations affecting the emitted stream. """ __slots__ = ("style", "path_items", "_transform", "_clipping_path") def __init__(self): self.style: GraphicsStyle = GraphicsStyle() self.path_items: list[Renderable] = [] self._transform: Optional[Transform] = None self._clipping_path: Optional[ClippingPath] = None def __deepcopy__(self, memo): copied = self.__class__() copied.style = deepcopy(self.style, memo) copied.path_items = deepcopy(self.path_items, memo) copied._transform = deepcopy(self.transform, memo) copied._clipping_path = deepcopy(self.clipping_path, memo) return copied @property def transform(self) -> Optional[Transform]: return self._transform @transform.setter def transform(self, tf: Transform) -> None: self._transform = tf @property def clipping_path(self) -> Optional[ClippingPath]: """The `ClippingPath` for this graphics context.""" return self._clipping_path @clipping_path.setter def clipping_path(self, new_clipath: ClippingPath) -> None: self._clipping_path = new_clipath def add_item(self, item: Renderable, _copy: bool = True) -> None: """ Add a path element to this graphics context. Args: item: the path element to add. May be a primitive element or another `GraphicsContext` or a `PaintedPath`. _copy (bool): if true (the default), the item will be copied before being appended. This prevents modifications to a referenced object from "retroactively" altering its style/shape and should be disabled with caution. """ if _copy: item = deepcopy(item) self.path_items.append(item) def remove_last_item(self) -> None: del self.path_items[-1] def merge(self, other_context: "GraphicsContext") -> None: """Copy another `GraphicsContext`'s path items into this one.""" self.path_items.extend(other_context.path_items) @force_nodocument def build_render_list( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, debug_stream=None, pfx: Optional[str] = None, _push_stack: bool = True, ): """ Build a list composed of all all the individual elements rendered. This is used by `PaintedPath` and `ClippingPath` to reuse the `GraphicsContext` rendering process while still being able to inject some path specific items (e.g. the painting directive) before the render is collapsed into a single string. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). _push_stack (bool): if True, wrap the resulting render list in a push/pop graphics stack directive pair. Returns: `tuple[list[str], last_item]` where `last_item` is the past path element in this `GraphicsContext` """ render_list = [] if self.path_items: if debug_stream is not None: debug_stream.write(f"{self.__class__.__name__}") merged_style = style.__class__.merge(style, self.style) if debug_stream is not None: if self._transform: debug_stream.write(f"({self._transform})") styles_dbg = [] for attr in merged_style.MERGE_PROPERTIES: val = getattr(merged_style, attr) if val is not merged_style.INHERIT: if getattr(self.style, attr) is merged_style.INHERIT: inherited = " (inherited)" else: inherited = "" styles_dbg.append(f"{attr}: {val}{inherited}") if styles_dbg: debug_stream.write(" {\n") for style_dbg_line in styles_dbg: debug_stream.write(pfx + " ") debug_stream.write(style_dbg_line) debug_stream.write("\n") debug_stream.write(pfx + "}┐\n") else: debug_stream.write("\n") NO_EMIT_SET = (None, merged_style.INHERIT) emit_style = self.style if merged_style.allow_transparency != self.style.allow_transparency: emit_style = deepcopy(self.style) emit_style.allow_transparency = merged_style.allow_transparency # in order to decouple the dash pattern and the dash phase at the API layer, # we have to perform additional logic here to recombine them. We can rely # on these being serializable because we always get a sane style on the # drawing context. dash_pattern = merged_style.stroke_dash_pattern dash_phase = merged_style.stroke_dash_phase emit_dash = None if ( dash_pattern != style.stroke_dash_pattern or dash_phase != style.stroke_dash_phase ): if emit_style is self.style: emit_style = deepcopy(emit_style) emit_style.stroke_dash_pattern = dash_pattern emit_style.stroke_dash_phase = dash_phase emit_dash = (dash_pattern, dash_phase) if ( emit_style.soft_mask and emit_style.soft_mask is not GraphicsStyle.INHERIT and emit_style.soft_mask.object_id == 0 ): emit_style.soft_mask.object_id = resource_registry.register_soft_mask( emit_style.soft_mask ) style_dict_name = resource_registry.register_graphics_style(emit_style) if style_dict_name is not None: render_list.append(f"{render_pdf_primitive(style_dict_name)} gs") # we can't set color in the graphics state context dictionary, so we have to # manually inherit it and emit it here. fill_color = self.style.fill_color stroke_color = self.style.stroke_color if fill_color not in NO_EMIT_SET: if isinstance(fill_color, GradientPaint): render_list.append( fill_color.emit_fill( resource_registry, self.bounding_box( initial_point, style=self.style, expand_for_stroke=False )[0], ) ) else: render_list.append(fill_color.serialize().lower()) if stroke_color not in NO_EMIT_SET: if isinstance(stroke_color, GradientPaint): render_list.append( stroke_color.emit_stroke( resource_registry, self.bounding_box( initial_point, style=self.style, expand_for_stroke=False )[0], ) ) else: render_list.append(stroke_color.serialize().upper()) if emit_dash is not None: render_list.append( render_pdf_primitive(emit_dash[0]) + f" {number_to_str(emit_dash[1])} d" ) if debug_stream: if self.clipping_path is not None: debug_stream.write(pfx + " ├─ ") rendered_cpath, _, __ = self.clipping_path.render_debug( resource_registry, merged_style, last_item, initial_point, debug_stream, pfx + " │ ", ) if rendered_cpath: render_list.append(rendered_cpath) for item in self.path_items[:-1]: debug_stream.write(pfx + " ├─ ") rendered, last_item, initial_point = item.render_debug( resource_registry, merged_style, last_item, initial_point, debug_stream, pfx + " │ ", ) if rendered: render_list.append(rendered) debug_stream.write(pfx + " └─ ") rendered, last_item, initial_point = self.path_items[-1].render_debug( resource_registry, merged_style, last_item, initial_point, debug_stream, pfx + " ", ) if rendered: render_list.append(rendered) else: if self.clipping_path is not None: rendered_cpath, _, __ = self.clipping_path.render( resource_registry, merged_style, last_item, initial_point ) if rendered_cpath: render_list.append(rendered_cpath) for item in self.path_items: rendered, last_item, initial_point = item.render( resource_registry, merged_style, last_item, initial_point ) if rendered: render_list.append(rendered) # insert transform before points if self.transform is not None: render_list.insert(0, self.transform.render(last_item)[0]) if _push_stack: render_list.insert(0, "q") render_list.append("Q") return render_list, last_item, initial_point def bounding_box( self, start: Point, style: Optional[GraphicsStyle] = None, expand_for_stroke=True, ) -> tuple[BoundingBox, Point]: """ Compute bbox of all path items. We: 1) recurse with accumulated CTM, 2) merge child bboxes already transformed to this level, 3) at the end, expand once for stroke using the worst-case CTM row norms. """ I = Transform.identity() def walk( ctx: "GraphicsContext", current_point: Point, ambient_style: Optional[GraphicsStyle], accum_tf: Transform, ) -> tuple[BoundingBox, Point, float, float]: bbox = BoundingBox.empty() tf = accum_tf @ (ctx.transform or I) merged_style = ( ambient_style.__class__.merge(ambient_style, ctx.style) if ambient_style else ctx.style ) max_nx, max_ny = tf.row_norms() for item in ctx.path_items: if isinstance(item, GraphicsContext): child_bbox, end_point, cnx, cny = walk( item, current_point, merged_style, tf ) bbox = bbox.merge(child_bbox) # child bbox already in this space current_point = end_point max_nx = max(max_nx, cnx) max_ny = max(max_ny, cny) elif hasattr(item, "bounding_box"): item_bbox, end_point = item.bounding_box(current_point) bbox = bbox.merge(item_bbox.transformed(tf)) current_point = end_point return bbox, current_point, max_nx, max_ny # 1) geometric + collect CTM scales geom_bbox, end_pt, nx, ny = walk(self, start, style, I) final_bbox = geom_bbox if expand_for_stroke: # 2) expand once for stroke with the effective style at *this* level effective_style = ( style.__class__.merge(style, self.style) if style else self.style ) final_bbox = geom_bbox.expanded_to_stroke( effective_style, row_norms=(nx, ny) ) return final_bbox, end_pt def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, debug_stream=None, pfx=None, _push_stack=True, ) -> tuple[str, Renderable, Point]: render_list, last_item, initial_point = self.build_render_list( resource_registry, style, last_item, initial_point, debug_stream, pfx, _push_stack=_push_stack, ) return " ".join(render_list), last_item, initial_point def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx, _push_stack=True, ): return self.render( resource_registry, style, last_item, initial_point, debug_stream, pfx, _push_stack=_push_stack, )
prop transform : Transform | None
-
Expand source code Browse git
@property def transform(self) -> Optional[Transform]: return self._transform
Methods
def add_item(self,
item: Renderable) ‑> None-
Expand source code Browse git
def add_item(self, item: Renderable, _copy: bool = True) -> None: """ Add a path element to this graphics context. Args: item: the path element to add. May be a primitive element or another `GraphicsContext` or a `PaintedPath`. _copy (bool): if true (the default), the item will be copied before being appended. This prevents modifications to a referenced object from "retroactively" altering its style/shape and should be disabled with caution. """ if _copy: item = deepcopy(item) self.path_items.append(item)
Add a path element to this graphics context.
Args
item
- the path element to add. May be a primitive element or another
GraphicsContext
or aPaintedPath
. _copy
:bool
- if true (the default), the item will be copied before being appended. This prevents modifications to a referenced object from "retroactively" altering its style/shape and should be disabled with caution.
def bounding_box(self,
start: Point,
style: GraphicsStyle | None = None,
expand_for_stroke=True) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box( self, start: Point, style: Optional[GraphicsStyle] = None, expand_for_stroke=True, ) -> tuple[BoundingBox, Point]: """ Compute bbox of all path items. We: 1) recurse with accumulated CTM, 2) merge child bboxes already transformed to this level, 3) at the end, expand once for stroke using the worst-case CTM row norms. """ I = Transform.identity() def walk( ctx: "GraphicsContext", current_point: Point, ambient_style: Optional[GraphicsStyle], accum_tf: Transform, ) -> tuple[BoundingBox, Point, float, float]: bbox = BoundingBox.empty() tf = accum_tf @ (ctx.transform or I) merged_style = ( ambient_style.__class__.merge(ambient_style, ctx.style) if ambient_style else ctx.style ) max_nx, max_ny = tf.row_norms() for item in ctx.path_items: if isinstance(item, GraphicsContext): child_bbox, end_point, cnx, cny = walk( item, current_point, merged_style, tf ) bbox = bbox.merge(child_bbox) # child bbox already in this space current_point = end_point max_nx = max(max_nx, cnx) max_ny = max(max_ny, cny) elif hasattr(item, "bounding_box"): item_bbox, end_point = item.bounding_box(current_point) bbox = bbox.merge(item_bbox.transformed(tf)) current_point = end_point return bbox, current_point, max_nx, max_ny # 1) geometric + collect CTM scales geom_bbox, end_pt, nx, ny = walk(self, start, style, I) final_bbox = geom_bbox if expand_for_stroke: # 2) expand once for stroke with the effective style at *this* level effective_style = ( style.__class__.merge(style, self.style) if style else self.style ) final_bbox = geom_bbox.expanded_to_stroke( effective_style, row_norms=(nx, ny) ) return final_bbox, end_pt
Compute bbox of all path items. We: 1) recurse with accumulated CTM, 2) merge child bboxes already transformed to this level, 3) at the end, expand once for stroke using the worst-case CTM row norms.
def build_render_list(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point,
debug_stream=None,
pfx: str | None = None)-
Expand source code Browse git
@force_nodocument def build_render_list( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, debug_stream=None, pfx: Optional[str] = None, _push_stack: bool = True, ): """ Build a list composed of all all the individual elements rendered. This is used by `PaintedPath` and `ClippingPath` to reuse the `GraphicsContext` rendering process while still being able to inject some path specific items (e.g. the painting directive) before the render is collapsed into a single string. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). _push_stack (bool): if True, wrap the resulting render list in a push/pop graphics stack directive pair. Returns: `tuple[list[str], last_item]` where `last_item` is the past path element in this `GraphicsContext` """ render_list = [] if self.path_items: if debug_stream is not None: debug_stream.write(f"{self.__class__.__name__}") merged_style = style.__class__.merge(style, self.style) if debug_stream is not None: if self._transform: debug_stream.write(f"({self._transform})") styles_dbg = [] for attr in merged_style.MERGE_PROPERTIES: val = getattr(merged_style, attr) if val is not merged_style.INHERIT: if getattr(self.style, attr) is merged_style.INHERIT: inherited = " (inherited)" else: inherited = "" styles_dbg.append(f"{attr}: {val}{inherited}") if styles_dbg: debug_stream.write(" {\n") for style_dbg_line in styles_dbg: debug_stream.write(pfx + " ") debug_stream.write(style_dbg_line) debug_stream.write("\n") debug_stream.write(pfx + "}┐\n") else: debug_stream.write("\n") NO_EMIT_SET = (None, merged_style.INHERIT) emit_style = self.style if merged_style.allow_transparency != self.style.allow_transparency: emit_style = deepcopy(self.style) emit_style.allow_transparency = merged_style.allow_transparency # in order to decouple the dash pattern and the dash phase at the API layer, # we have to perform additional logic here to recombine them. We can rely # on these being serializable because we always get a sane style on the # drawing context. dash_pattern = merged_style.stroke_dash_pattern dash_phase = merged_style.stroke_dash_phase emit_dash = None if ( dash_pattern != style.stroke_dash_pattern or dash_phase != style.stroke_dash_phase ): if emit_style is self.style: emit_style = deepcopy(emit_style) emit_style.stroke_dash_pattern = dash_pattern emit_style.stroke_dash_phase = dash_phase emit_dash = (dash_pattern, dash_phase) if ( emit_style.soft_mask and emit_style.soft_mask is not GraphicsStyle.INHERIT and emit_style.soft_mask.object_id == 0 ): emit_style.soft_mask.object_id = resource_registry.register_soft_mask( emit_style.soft_mask ) style_dict_name = resource_registry.register_graphics_style(emit_style) if style_dict_name is not None: render_list.append(f"{render_pdf_primitive(style_dict_name)} gs") # we can't set color in the graphics state context dictionary, so we have to # manually inherit it and emit it here. fill_color = self.style.fill_color stroke_color = self.style.stroke_color if fill_color not in NO_EMIT_SET: if isinstance(fill_color, GradientPaint): render_list.append( fill_color.emit_fill( resource_registry, self.bounding_box( initial_point, style=self.style, expand_for_stroke=False )[0], ) ) else: render_list.append(fill_color.serialize().lower()) if stroke_color not in NO_EMIT_SET: if isinstance(stroke_color, GradientPaint): render_list.append( stroke_color.emit_stroke( resource_registry, self.bounding_box( initial_point, style=self.style, expand_for_stroke=False )[0], ) ) else: render_list.append(stroke_color.serialize().upper()) if emit_dash is not None: render_list.append( render_pdf_primitive(emit_dash[0]) + f" {number_to_str(emit_dash[1])} d" ) if debug_stream: if self.clipping_path is not None: debug_stream.write(pfx + " ├─ ") rendered_cpath, _, __ = self.clipping_path.render_debug( resource_registry, merged_style, last_item, initial_point, debug_stream, pfx + " │ ", ) if rendered_cpath: render_list.append(rendered_cpath) for item in self.path_items[:-1]: debug_stream.write(pfx + " ├─ ") rendered, last_item, initial_point = item.render_debug( resource_registry, merged_style, last_item, initial_point, debug_stream, pfx + " │ ", ) if rendered: render_list.append(rendered) debug_stream.write(pfx + " └─ ") rendered, last_item, initial_point = self.path_items[-1].render_debug( resource_registry, merged_style, last_item, initial_point, debug_stream, pfx + " ", ) if rendered: render_list.append(rendered) else: if self.clipping_path is not None: rendered_cpath, _, __ = self.clipping_path.render( resource_registry, merged_style, last_item, initial_point ) if rendered_cpath: render_list.append(rendered_cpath) for item in self.path_items: rendered, last_item, initial_point = item.render( resource_registry, merged_style, last_item, initial_point ) if rendered: render_list.append(rendered) # insert transform before points if self.transform is not None: render_list.insert(0, self.transform.render(last_item)[0]) if _push_stack: render_list.insert(0, "q") render_list.append("Q") return render_list, last_item, initial_point
Build a list composed of all all the individual elements rendered.
This is used by
PaintedPath
andClippingPath
to reuse theGraphicsContext
rendering process while still being able to inject some path specific items (e.g. the painting directive) before the render is collapsed into a single string.Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
_push_stack
:bool
- if True, wrap the resulting render list in a push/pop graphics stack directive pair.
Returns
tuple[list[str], last_item]
wherelast_item
is the past path element in thisGraphicsContext
def merge(self,
other_context: GraphicsContext) ‑> None-
Expand source code Browse git
def merge(self, other_context: "GraphicsContext") -> None: """Copy another `GraphicsContext`'s path items into this one.""" self.path_items.extend(other_context.path_items)
Copy another
GraphicsContext
's path items into this one. def remove_last_item(self) ‑> None
-
Expand source code Browse git
def remove_last_item(self) -> None: del self.path_items[-1]
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point,
debug_stream=None,
pfx=None) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, debug_stream=None, pfx=None, _push_stack=True, ) -> tuple[str, Renderable, Point]: render_list, last_item, initial_point = self.build_render_list( resource_registry, style, last_item, initial_point, debug_stream, pfx, _push_stack=_push_stack, ) return " ".join(render_list), last_item, initial_point
def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx, _push_stack=True, ): return self.render( resource_registry, style, last_item, initial_point, debug_stream, pfx, _push_stack=_push_stack, )
class GraphicsStyle
-
Expand source code Browse git
class GraphicsStyle: """ A class representing various style attributes that determine drawing appearance. This class uses the convention that the global Python singleton ellipsis (`...`) is exclusively used to represent values that are inherited from the parent style. This is to disambiguate the value None which is used for several values to signal an explicitly disabled style. An example of this is the fill/stroke color styles, which use None as hints to the auto paint style detection code. """ _PRIVATE_SLOTS = ( "_allow_transparency", "_auto_close", "_fill_color", "_intersection_rule", "_paint_rule", "_stroke_color", "_stroke_dash_pattern", "_stroke_dash_phase", ) __slots__ = _PRIVATE_SLOTS + tuple( k.value for k in PDFStyleKeys # we do not store STROKE_DASH_PATTERN under its PDF key; it's in _stroke_dash_pattern if k is not PDFStyleKeys.STROKE_DASH_PATTERN ) INHERIT = ... """Singleton specifying a style parameter should be inherited from the parent context.""" # order is be important here because some of these properties are entangled, e.g. # fill_color and fill_opacity MERGE_PROPERTIES = ( "paint_rule", "allow_transparency", "auto_close", "intersection_rule", "fill_color", "fill_opacity", "stroke_color", "stroke_opacity", "blend_mode", "stroke_width", "stroke_cap_style", "stroke_join_style", "stroke_miter_limit", "stroke_dash_pattern", "stroke_dash_phase", "soft_mask", ) """An ordered collection of properties to use when merging two GraphicsStyles.""" TRANSPARENCY_KEYS = ( PDFStyleKeys.FILL_ALPHA.value, PDFStyleKeys.STROKE_ALPHA.value, PDFStyleKeys.BLEND_MODE.value, PDFStyleKeys.SOFT_MASK.value, ) """An ordered collection of attributes not to emit in no transparency mode.""" PDF_STYLE_KEYS = ( *(k.value for k in PDFStyleKeys if k is not PDFStyleKeys.STROKE_DASH_PATTERN), ) """An ordered collection of keys to directly emit when serializing the style.""" _PAINT_RULE_LOOKUP = { frozenset({}): PathPaintRule.DONT_PAINT, frozenset({"stroke"}): PathPaintRule.STROKE, frozenset({"fill", IntersectionRule.NONZERO}): PathPaintRule.FILL_NONZERO, frozenset({"fill", IntersectionRule.EVENODD}): PathPaintRule.FILL_EVENODD, frozenset( {"stroke", "fill", IntersectionRule.NONZERO} ): PathPaintRule.STROKE_FILL_NONZERO, frozenset( {"stroke", "fill", IntersectionRule.EVENODD} ): PathPaintRule.STROKE_FILL_EVENODD, } """A dictionary for resolving `PathPaintRule.AUTO`""" @classmethod def merge(cls, parent: "GraphicsStyle", child: "GraphicsStyle") -> "GraphicsStyle": """ Merge parent and child into a single GraphicsStyle. The result contains the properties of the parent as overridden by any properties explicitly set on the child. If both the parent and the child specify to inherit a given property, that property will preserve the inherit value. """ new = cls() for prop in cls.MERGE_PROPERTIES: cval = getattr(child, prop) if cval is cls.INHERIT: setattr(new, prop, getattr(parent, prop)) else: setattr(new, prop, cval) return new def __init__(self): self.allow_transparency = self.INHERIT self.paint_rule = self.INHERIT self.auto_close = self.INHERIT self.intersection_rule = self.INHERIT self.fill_color = self.INHERIT self.fill_opacity = self.INHERIT self.stroke_color = self.INHERIT self.stroke_opacity = self.INHERIT self.blend_mode = self.INHERIT self.stroke_width = self.INHERIT self.stroke_cap_style = self.INHERIT self.stroke_join_style = self.INHERIT self.stroke_miter_limit = self.INHERIT self.stroke_dash_pattern = self.INHERIT self.stroke_dash_phase = self.INHERIT self.soft_mask = self.INHERIT def __deepcopy__(self, memo): copied = self.__class__() for prop in self.MERGE_PROPERTIES: setattr(copied, prop, getattr(self, prop)) return copied def __setattr__(self, name, value): if not hasattr(self.__class__, name): raise AttributeError( f'{self.__class__} does not have style "{name}" (a typo?)' ) super().__setattr__(name, value) # at some point it probably makes sense to turn this into a general compliance # property, but for now this is the simple approach. @property def allow_transparency(self): return self._allow_transparency # pylint: disable=no-member @allow_transparency.setter def allow_transparency(self, new): return super().__setattr__("_allow_transparency", new) # If these are used in a nested graphics context inside of a painting path # operation, they are no-ops. However, they can be used for outer GraphicsContexts # that painting paths inherit from. @property def paint_rule(self): """The paint rule to use for this path/group.""" return self._paint_rule # pylint: disable=no-member @paint_rule.setter def paint_rule(self, new): if new is None: super().__setattr__("_paint_rule", PathPaintRule.DONT_PAINT) elif new is self.INHERIT: super().__setattr__("_paint_rule", new) else: super().__setattr__("_paint_rule", PathPaintRule.coerce(new)) @property def auto_close(self): """If True, unclosed paths will be automatically closed before stroking.""" return self._auto_close # pylint: disable=no-member @auto_close.setter def auto_close(self, new): if new not in {True, False, self.INHERIT}: raise TypeError(f"auto_close must be a bool or self.INHERIT, not {new}") super().__setattr__("_auto_close", new) @property def intersection_rule(self): """The desired intersection rule for this path/group.""" return self._intersection_rule # pylint: disable=no-member @intersection_rule.setter def intersection_rule(self, new): # don't allow None for this one. if new is self.INHERIT: super().__setattr__("_intersection_rule", new) else: super().__setattr__("_intersection_rule", IntersectionRule.coerce(new)) @property def fill_color(self): """ The desired fill color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._fill_color # pylint: disable=no-member @fill_color.setter def fill_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_fill_color", color) if getattr(color, "a", None) is not None: self.fill_opacity = color.a elif (color is None) or (color is self.INHERIT): super().__setattr__("_fill_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def fill_opacity(self): """The desired fill opacity for this path/group.""" return getattr(self, PDFStyleKeys.FILL_ALPHA.value) @fill_opacity.setter def fill_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.FILL_ALPHA.value, new) @property def stroke_color(self): """ The desired stroke color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._stroke_color # pylint: disable=no-member @stroke_color.setter def stroke_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_stroke_color", color) if getattr(color, "a", None) is not None: self.stroke_opacity = color.a if self.stroke_width is self.INHERIT: self.stroke_width = 1 elif (color is None) or (color is self.INHERIT): super().__setattr__("_stroke_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def stroke_opacity(self): """The desired stroke opacity for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_ALPHA.value) @stroke_opacity.setter def stroke_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.STROKE_ALPHA.value, new) @property def blend_mode(self): """The desired blend mode for this path/group.""" return getattr(self, PDFStyleKeys.BLEND_MODE.value) @blend_mode.setter def blend_mode(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.BLEND_MODE.value, value) else: super().__setattr__( PDFStyleKeys.BLEND_MODE.value, BlendMode.coerce(value).value ) @property def stroke_width(self): """The desired stroke width for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_WIDTH.value) @stroke_width.setter def stroke_width(self, width): if not isinstance( width, (int, float, decimal.Decimal, type(None), type(self.INHERIT)), ): raise TypeError(f"stroke_width must be a number, not {type(width)}") super().__setattr__(PDFStyleKeys.STROKE_WIDTH.value, width) @property def stroke_cap_style(self): """The desired stroke cap style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_CAP_STYLE.value) @stroke_cap_style.setter def stroke_cap_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_CAP_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_CAP_STYLE.value, StrokeCapStyle.coerce(value) ) @property def stroke_join_style(self): """The desired stroke join style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_JOIN_STYLE.value) @stroke_join_style.setter def stroke_join_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_JOIN_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_JOIN_STYLE.value, StrokeJoinStyle.coerce(value), ) @property def stroke_miter_limit(self): """The desired stroke miter limit for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_MITER_LIMIT.value) @stroke_miter_limit.setter def stroke_miter_limit(self, value): if (value is self.INHERIT) or isinstance(value, NumberClass): super().__setattr__(PDFStyleKeys.STROKE_MITER_LIMIT.value, value) else: raise TypeError(f"{value} is not a number") @property def stroke_dash_pattern(self): """The desired stroke dash pattern for this path/group.""" return self._stroke_dash_pattern # pylint: disable=no-member @stroke_dash_pattern.setter def stroke_dash_pattern(self, value): if value is None: result = () elif value is self.INHERIT: result = value elif isinstance(value, NumberClass): result = (value,) else: try: accum = [] for item in value: if not isinstance(item, NumberClass): raise TypeError( f"stroke_dash_pattern {value} sequence has non-numeric value" ) accum.append(item) except TypeError: raise TypeError( f"stroke_dash_pattern {value} must be a number or sequence of numbers" ) from None result = (*accum,) super().__setattr__("_stroke_dash_pattern", result) @property def stroke_dash_phase(self): """The desired stroke dash pattern phase offset for this path/group.""" return self._stroke_dash_phase # pylint: disable=no-member @stroke_dash_phase.setter def stroke_dash_phase(self, value): if value is self.INHERIT or isinstance(value, NumberClass): return super().__setattr__("_stroke_dash_phase", value) raise TypeError(f"{value} isn't a number or GraphicsStyle.INHERIT") @property def soft_mask(self): return getattr(self, PDFStyleKeys.SOFT_MASK.value) @soft_mask.setter def soft_mask(self, value): if value is self.INHERIT or isinstance(value, PaintSoftMask): return super().__setattr__(PDFStyleKeys.SOFT_MASK.value, value) raise TypeError(f"{value} isn't a PaintSoftMask or GraphicsStyle.INHERIT") def serialize(self) -> Optional[Raw]: """ Convert this style object to a PDF dictionary with appropriate style keys. Only explicitly specified values are emitted. """ result = OrderedDict() for key in self.PDF_STYLE_KEYS: value = getattr(self, key, self.INHERIT) if (value is not self.INHERIT) and (value is not None): # None is used for out-of-band signaling on these, e.g. a stroke_width # of None doesn't need to land here because it signals the # PathPaintRule auto resolution only. result[key] = value # There is additional logic in GraphicsContext to ensure that this will work if self.stroke_dash_pattern and self.stroke_dash_pattern is not self.INHERIT: result[PDFStyleKeys.STROKE_DASH_PATTERN.value] = [ self.stroke_dash_pattern, self.stroke_dash_phase, ] if self.allow_transparency is False: for key in self.TRANSPARENCY_KEYS: if key in result: del result[key] if result: # Only insert this key if there is at least one other item in the result so # that we don't junk up the output PDF with empty ExtGState dictionaries. type_name = Name("Type") result[type_name] = Name("ExtGState") result.move_to_end(type_name, last=False) return render_pdf_primitive(result) # this signals to the graphics state registry that there is nothing to # register. This is a success case. return None @force_nodocument def resolve_paint_rule(self) -> PathPaintRule: """ Resolve `PathPaintRule.AUTO` to a real paint rule based on this style. Returns: the resolved `PathPaintRule`. """ if self.paint_rule is PathPaintRule.AUTO: want = set() if self.stroke_width is not None and self.stroke_color is not None: want.add("stroke") if self.fill_color is not None: want.add("fill") # we need to guarantee that this will not be None. The default will # be "nonzero". assert self.intersection_rule is not None want.add(self.intersection_rule) try: rule = self._PAINT_RULE_LOOKUP[frozenset(want)] except KeyError: # don't default to DONT_PAINT because that's almost certainly not a very # good default. rule = PathPaintRule.STROKE_FILL_NONZERO elif self.paint_rule is self.INHERIT: # this shouldn't happen under normal usage, but certain API (ab)use can end # up in this state. We can't resolve anything meaningful, so fall back to a # sane(?) default. rule = PathPaintRule.STROKE_FILL_NONZERO else: rule = self.paint_rule return rule
A class representing various style attributes that determine drawing appearance.
This class uses the convention that the global Python singleton ellipsis (
…
) is exclusively used to represent values that are inherited from the parent style. This is to disambiguate the value None which is used for several values to signal an explicitly disabled style. An example of this is the fill/stroke color styles, which use None as hints to the auto paint style detection code.Class variables
var INHERIT
-
Singleton specifying a style parameter should be inherited from the parent context.
var MERGE_PROPERTIES
-
An ordered collection of properties to use when merging two GraphicsStyles.
var PDF_STYLE_KEYS
-
An ordered collection of keys to directly emit when serializing the style.
var TRANSPARENCY_KEYS
-
An ordered collection of attributes not to emit in no transparency mode.
Static methods
def merge(parent: GraphicsStyle,
child: GraphicsStyle) ‑> GraphicsStyle-
Merge parent and child into a single GraphicsStyle.
The result contains the properties of the parent as overridden by any properties explicitly set on the child. If both the parent and the child specify to inherit a given property, that property will preserve the inherit value.
Instance variables
var BM
-
Expand source code Browse git
class GraphicsStyle: """ A class representing various style attributes that determine drawing appearance. This class uses the convention that the global Python singleton ellipsis (`...`) is exclusively used to represent values that are inherited from the parent style. This is to disambiguate the value None which is used for several values to signal an explicitly disabled style. An example of this is the fill/stroke color styles, which use None as hints to the auto paint style detection code. """ _PRIVATE_SLOTS = ( "_allow_transparency", "_auto_close", "_fill_color", "_intersection_rule", "_paint_rule", "_stroke_color", "_stroke_dash_pattern", "_stroke_dash_phase", ) __slots__ = _PRIVATE_SLOTS + tuple( k.value for k in PDFStyleKeys # we do not store STROKE_DASH_PATTERN under its PDF key; it's in _stroke_dash_pattern if k is not PDFStyleKeys.STROKE_DASH_PATTERN ) INHERIT = ... """Singleton specifying a style parameter should be inherited from the parent context.""" # order is be important here because some of these properties are entangled, e.g. # fill_color and fill_opacity MERGE_PROPERTIES = ( "paint_rule", "allow_transparency", "auto_close", "intersection_rule", "fill_color", "fill_opacity", "stroke_color", "stroke_opacity", "blend_mode", "stroke_width", "stroke_cap_style", "stroke_join_style", "stroke_miter_limit", "stroke_dash_pattern", "stroke_dash_phase", "soft_mask", ) """An ordered collection of properties to use when merging two GraphicsStyles.""" TRANSPARENCY_KEYS = ( PDFStyleKeys.FILL_ALPHA.value, PDFStyleKeys.STROKE_ALPHA.value, PDFStyleKeys.BLEND_MODE.value, PDFStyleKeys.SOFT_MASK.value, ) """An ordered collection of attributes not to emit in no transparency mode.""" PDF_STYLE_KEYS = ( *(k.value for k in PDFStyleKeys if k is not PDFStyleKeys.STROKE_DASH_PATTERN), ) """An ordered collection of keys to directly emit when serializing the style.""" _PAINT_RULE_LOOKUP = { frozenset({}): PathPaintRule.DONT_PAINT, frozenset({"stroke"}): PathPaintRule.STROKE, frozenset({"fill", IntersectionRule.NONZERO}): PathPaintRule.FILL_NONZERO, frozenset({"fill", IntersectionRule.EVENODD}): PathPaintRule.FILL_EVENODD, frozenset( {"stroke", "fill", IntersectionRule.NONZERO} ): PathPaintRule.STROKE_FILL_NONZERO, frozenset( {"stroke", "fill", IntersectionRule.EVENODD} ): PathPaintRule.STROKE_FILL_EVENODD, } """A dictionary for resolving `PathPaintRule.AUTO`""" @classmethod def merge(cls, parent: "GraphicsStyle", child: "GraphicsStyle") -> "GraphicsStyle": """ Merge parent and child into a single GraphicsStyle. The result contains the properties of the parent as overridden by any properties explicitly set on the child. If both the parent and the child specify to inherit a given property, that property will preserve the inherit value. """ new = cls() for prop in cls.MERGE_PROPERTIES: cval = getattr(child, prop) if cval is cls.INHERIT: setattr(new, prop, getattr(parent, prop)) else: setattr(new, prop, cval) return new def __init__(self): self.allow_transparency = self.INHERIT self.paint_rule = self.INHERIT self.auto_close = self.INHERIT self.intersection_rule = self.INHERIT self.fill_color = self.INHERIT self.fill_opacity = self.INHERIT self.stroke_color = self.INHERIT self.stroke_opacity = self.INHERIT self.blend_mode = self.INHERIT self.stroke_width = self.INHERIT self.stroke_cap_style = self.INHERIT self.stroke_join_style = self.INHERIT self.stroke_miter_limit = self.INHERIT self.stroke_dash_pattern = self.INHERIT self.stroke_dash_phase = self.INHERIT self.soft_mask = self.INHERIT def __deepcopy__(self, memo): copied = self.__class__() for prop in self.MERGE_PROPERTIES: setattr(copied, prop, getattr(self, prop)) return copied def __setattr__(self, name, value): if not hasattr(self.__class__, name): raise AttributeError( f'{self.__class__} does not have style "{name}" (a typo?)' ) super().__setattr__(name, value) # at some point it probably makes sense to turn this into a general compliance # property, but for now this is the simple approach. @property def allow_transparency(self): return self._allow_transparency # pylint: disable=no-member @allow_transparency.setter def allow_transparency(self, new): return super().__setattr__("_allow_transparency", new) # If these are used in a nested graphics context inside of a painting path # operation, they are no-ops. However, they can be used for outer GraphicsContexts # that painting paths inherit from. @property def paint_rule(self): """The paint rule to use for this path/group.""" return self._paint_rule # pylint: disable=no-member @paint_rule.setter def paint_rule(self, new): if new is None: super().__setattr__("_paint_rule", PathPaintRule.DONT_PAINT) elif new is self.INHERIT: super().__setattr__("_paint_rule", new) else: super().__setattr__("_paint_rule", PathPaintRule.coerce(new)) @property def auto_close(self): """If True, unclosed paths will be automatically closed before stroking.""" return self._auto_close # pylint: disable=no-member @auto_close.setter def auto_close(self, new): if new not in {True, False, self.INHERIT}: raise TypeError(f"auto_close must be a bool or self.INHERIT, not {new}") super().__setattr__("_auto_close", new) @property def intersection_rule(self): """The desired intersection rule for this path/group.""" return self._intersection_rule # pylint: disable=no-member @intersection_rule.setter def intersection_rule(self, new): # don't allow None for this one. if new is self.INHERIT: super().__setattr__("_intersection_rule", new) else: super().__setattr__("_intersection_rule", IntersectionRule.coerce(new)) @property def fill_color(self): """ The desired fill color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._fill_color # pylint: disable=no-member @fill_color.setter def fill_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_fill_color", color) if getattr(color, "a", None) is not None: self.fill_opacity = color.a elif (color is None) or (color is self.INHERIT): super().__setattr__("_fill_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def fill_opacity(self): """The desired fill opacity for this path/group.""" return getattr(self, PDFStyleKeys.FILL_ALPHA.value) @fill_opacity.setter def fill_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.FILL_ALPHA.value, new) @property def stroke_color(self): """ The desired stroke color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._stroke_color # pylint: disable=no-member @stroke_color.setter def stroke_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_stroke_color", color) if getattr(color, "a", None) is not None: self.stroke_opacity = color.a if self.stroke_width is self.INHERIT: self.stroke_width = 1 elif (color is None) or (color is self.INHERIT): super().__setattr__("_stroke_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def stroke_opacity(self): """The desired stroke opacity for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_ALPHA.value) @stroke_opacity.setter def stroke_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.STROKE_ALPHA.value, new) @property def blend_mode(self): """The desired blend mode for this path/group.""" return getattr(self, PDFStyleKeys.BLEND_MODE.value) @blend_mode.setter def blend_mode(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.BLEND_MODE.value, value) else: super().__setattr__( PDFStyleKeys.BLEND_MODE.value, BlendMode.coerce(value).value ) @property def stroke_width(self): """The desired stroke width for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_WIDTH.value) @stroke_width.setter def stroke_width(self, width): if not isinstance( width, (int, float, decimal.Decimal, type(None), type(self.INHERIT)), ): raise TypeError(f"stroke_width must be a number, not {type(width)}") super().__setattr__(PDFStyleKeys.STROKE_WIDTH.value, width) @property def stroke_cap_style(self): """The desired stroke cap style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_CAP_STYLE.value) @stroke_cap_style.setter def stroke_cap_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_CAP_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_CAP_STYLE.value, StrokeCapStyle.coerce(value) ) @property def stroke_join_style(self): """The desired stroke join style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_JOIN_STYLE.value) @stroke_join_style.setter def stroke_join_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_JOIN_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_JOIN_STYLE.value, StrokeJoinStyle.coerce(value), ) @property def stroke_miter_limit(self): """The desired stroke miter limit for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_MITER_LIMIT.value) @stroke_miter_limit.setter def stroke_miter_limit(self, value): if (value is self.INHERIT) or isinstance(value, NumberClass): super().__setattr__(PDFStyleKeys.STROKE_MITER_LIMIT.value, value) else: raise TypeError(f"{value} is not a number") @property def stroke_dash_pattern(self): """The desired stroke dash pattern for this path/group.""" return self._stroke_dash_pattern # pylint: disable=no-member @stroke_dash_pattern.setter def stroke_dash_pattern(self, value): if value is None: result = () elif value is self.INHERIT: result = value elif isinstance(value, NumberClass): result = (value,) else: try: accum = [] for item in value: if not isinstance(item, NumberClass): raise TypeError( f"stroke_dash_pattern {value} sequence has non-numeric value" ) accum.append(item) except TypeError: raise TypeError( f"stroke_dash_pattern {value} must be a number or sequence of numbers" ) from None result = (*accum,) super().__setattr__("_stroke_dash_pattern", result) @property def stroke_dash_phase(self): """The desired stroke dash pattern phase offset for this path/group.""" return self._stroke_dash_phase # pylint: disable=no-member @stroke_dash_phase.setter def stroke_dash_phase(self, value): if value is self.INHERIT or isinstance(value, NumberClass): return super().__setattr__("_stroke_dash_phase", value) raise TypeError(f"{value} isn't a number or GraphicsStyle.INHERIT") @property def soft_mask(self): return getattr(self, PDFStyleKeys.SOFT_MASK.value) @soft_mask.setter def soft_mask(self, value): if value is self.INHERIT or isinstance(value, PaintSoftMask): return super().__setattr__(PDFStyleKeys.SOFT_MASK.value, value) raise TypeError(f"{value} isn't a PaintSoftMask or GraphicsStyle.INHERIT") def serialize(self) -> Optional[Raw]: """ Convert this style object to a PDF dictionary with appropriate style keys. Only explicitly specified values are emitted. """ result = OrderedDict() for key in self.PDF_STYLE_KEYS: value = getattr(self, key, self.INHERIT) if (value is not self.INHERIT) and (value is not None): # None is used for out-of-band signaling on these, e.g. a stroke_width # of None doesn't need to land here because it signals the # PathPaintRule auto resolution only. result[key] = value # There is additional logic in GraphicsContext to ensure that this will work if self.stroke_dash_pattern and self.stroke_dash_pattern is not self.INHERIT: result[PDFStyleKeys.STROKE_DASH_PATTERN.value] = [ self.stroke_dash_pattern, self.stroke_dash_phase, ] if self.allow_transparency is False: for key in self.TRANSPARENCY_KEYS: if key in result: del result[key] if result: # Only insert this key if there is at least one other item in the result so # that we don't junk up the output PDF with empty ExtGState dictionaries. type_name = Name("Type") result[type_name] = Name("ExtGState") result.move_to_end(type_name, last=False) return render_pdf_primitive(result) # this signals to the graphics state registry that there is nothing to # register. This is a success case. return None @force_nodocument def resolve_paint_rule(self) -> PathPaintRule: """ Resolve `PathPaintRule.AUTO` to a real paint rule based on this style. Returns: the resolved `PathPaintRule`. """ if self.paint_rule is PathPaintRule.AUTO: want = set() if self.stroke_width is not None and self.stroke_color is not None: want.add("stroke") if self.fill_color is not None: want.add("fill") # we need to guarantee that this will not be None. The default will # be "nonzero". assert self.intersection_rule is not None want.add(self.intersection_rule) try: rule = self._PAINT_RULE_LOOKUP[frozenset(want)] except KeyError: # don't default to DONT_PAINT because that's almost certainly not a very # good default. rule = PathPaintRule.STROKE_FILL_NONZERO elif self.paint_rule is self.INHERIT: # this shouldn't happen under normal usage, but certain API (ab)use can end # up in this state. We can't resolve anything meaningful, so fall back to a # sane(?) default. rule = PathPaintRule.STROKE_FILL_NONZERO else: rule = self.paint_rule return rule
var CA
-
Expand source code Browse git
class GraphicsStyle: """ A class representing various style attributes that determine drawing appearance. This class uses the convention that the global Python singleton ellipsis (`...`) is exclusively used to represent values that are inherited from the parent style. This is to disambiguate the value None which is used for several values to signal an explicitly disabled style. An example of this is the fill/stroke color styles, which use None as hints to the auto paint style detection code. """ _PRIVATE_SLOTS = ( "_allow_transparency", "_auto_close", "_fill_color", "_intersection_rule", "_paint_rule", "_stroke_color", "_stroke_dash_pattern", "_stroke_dash_phase", ) __slots__ = _PRIVATE_SLOTS + tuple( k.value for k in PDFStyleKeys # we do not store STROKE_DASH_PATTERN under its PDF key; it's in _stroke_dash_pattern if k is not PDFStyleKeys.STROKE_DASH_PATTERN ) INHERIT = ... """Singleton specifying a style parameter should be inherited from the parent context.""" # order is be important here because some of these properties are entangled, e.g. # fill_color and fill_opacity MERGE_PROPERTIES = ( "paint_rule", "allow_transparency", "auto_close", "intersection_rule", "fill_color", "fill_opacity", "stroke_color", "stroke_opacity", "blend_mode", "stroke_width", "stroke_cap_style", "stroke_join_style", "stroke_miter_limit", "stroke_dash_pattern", "stroke_dash_phase", "soft_mask", ) """An ordered collection of properties to use when merging two GraphicsStyles.""" TRANSPARENCY_KEYS = ( PDFStyleKeys.FILL_ALPHA.value, PDFStyleKeys.STROKE_ALPHA.value, PDFStyleKeys.BLEND_MODE.value, PDFStyleKeys.SOFT_MASK.value, ) """An ordered collection of attributes not to emit in no transparency mode.""" PDF_STYLE_KEYS = ( *(k.value for k in PDFStyleKeys if k is not PDFStyleKeys.STROKE_DASH_PATTERN), ) """An ordered collection of keys to directly emit when serializing the style.""" _PAINT_RULE_LOOKUP = { frozenset({}): PathPaintRule.DONT_PAINT, frozenset({"stroke"}): PathPaintRule.STROKE, frozenset({"fill", IntersectionRule.NONZERO}): PathPaintRule.FILL_NONZERO, frozenset({"fill", IntersectionRule.EVENODD}): PathPaintRule.FILL_EVENODD, frozenset( {"stroke", "fill", IntersectionRule.NONZERO} ): PathPaintRule.STROKE_FILL_NONZERO, frozenset( {"stroke", "fill", IntersectionRule.EVENODD} ): PathPaintRule.STROKE_FILL_EVENODD, } """A dictionary for resolving `PathPaintRule.AUTO`""" @classmethod def merge(cls, parent: "GraphicsStyle", child: "GraphicsStyle") -> "GraphicsStyle": """ Merge parent and child into a single GraphicsStyle. The result contains the properties of the parent as overridden by any properties explicitly set on the child. If both the parent and the child specify to inherit a given property, that property will preserve the inherit value. """ new = cls() for prop in cls.MERGE_PROPERTIES: cval = getattr(child, prop) if cval is cls.INHERIT: setattr(new, prop, getattr(parent, prop)) else: setattr(new, prop, cval) return new def __init__(self): self.allow_transparency = self.INHERIT self.paint_rule = self.INHERIT self.auto_close = self.INHERIT self.intersection_rule = self.INHERIT self.fill_color = self.INHERIT self.fill_opacity = self.INHERIT self.stroke_color = self.INHERIT self.stroke_opacity = self.INHERIT self.blend_mode = self.INHERIT self.stroke_width = self.INHERIT self.stroke_cap_style = self.INHERIT self.stroke_join_style = self.INHERIT self.stroke_miter_limit = self.INHERIT self.stroke_dash_pattern = self.INHERIT self.stroke_dash_phase = self.INHERIT self.soft_mask = self.INHERIT def __deepcopy__(self, memo): copied = self.__class__() for prop in self.MERGE_PROPERTIES: setattr(copied, prop, getattr(self, prop)) return copied def __setattr__(self, name, value): if not hasattr(self.__class__, name): raise AttributeError( f'{self.__class__} does not have style "{name}" (a typo?)' ) super().__setattr__(name, value) # at some point it probably makes sense to turn this into a general compliance # property, but for now this is the simple approach. @property def allow_transparency(self): return self._allow_transparency # pylint: disable=no-member @allow_transparency.setter def allow_transparency(self, new): return super().__setattr__("_allow_transparency", new) # If these are used in a nested graphics context inside of a painting path # operation, they are no-ops. However, they can be used for outer GraphicsContexts # that painting paths inherit from. @property def paint_rule(self): """The paint rule to use for this path/group.""" return self._paint_rule # pylint: disable=no-member @paint_rule.setter def paint_rule(self, new): if new is None: super().__setattr__("_paint_rule", PathPaintRule.DONT_PAINT) elif new is self.INHERIT: super().__setattr__("_paint_rule", new) else: super().__setattr__("_paint_rule", PathPaintRule.coerce(new)) @property def auto_close(self): """If True, unclosed paths will be automatically closed before stroking.""" return self._auto_close # pylint: disable=no-member @auto_close.setter def auto_close(self, new): if new not in {True, False, self.INHERIT}: raise TypeError(f"auto_close must be a bool or self.INHERIT, not {new}") super().__setattr__("_auto_close", new) @property def intersection_rule(self): """The desired intersection rule for this path/group.""" return self._intersection_rule # pylint: disable=no-member @intersection_rule.setter def intersection_rule(self, new): # don't allow None for this one. if new is self.INHERIT: super().__setattr__("_intersection_rule", new) else: super().__setattr__("_intersection_rule", IntersectionRule.coerce(new)) @property def fill_color(self): """ The desired fill color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._fill_color # pylint: disable=no-member @fill_color.setter def fill_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_fill_color", color) if getattr(color, "a", None) is not None: self.fill_opacity = color.a elif (color is None) or (color is self.INHERIT): super().__setattr__("_fill_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def fill_opacity(self): """The desired fill opacity for this path/group.""" return getattr(self, PDFStyleKeys.FILL_ALPHA.value) @fill_opacity.setter def fill_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.FILL_ALPHA.value, new) @property def stroke_color(self): """ The desired stroke color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._stroke_color # pylint: disable=no-member @stroke_color.setter def stroke_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_stroke_color", color) if getattr(color, "a", None) is not None: self.stroke_opacity = color.a if self.stroke_width is self.INHERIT: self.stroke_width = 1 elif (color is None) or (color is self.INHERIT): super().__setattr__("_stroke_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def stroke_opacity(self): """The desired stroke opacity for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_ALPHA.value) @stroke_opacity.setter def stroke_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.STROKE_ALPHA.value, new) @property def blend_mode(self): """The desired blend mode for this path/group.""" return getattr(self, PDFStyleKeys.BLEND_MODE.value) @blend_mode.setter def blend_mode(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.BLEND_MODE.value, value) else: super().__setattr__( PDFStyleKeys.BLEND_MODE.value, BlendMode.coerce(value).value ) @property def stroke_width(self): """The desired stroke width for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_WIDTH.value) @stroke_width.setter def stroke_width(self, width): if not isinstance( width, (int, float, decimal.Decimal, type(None), type(self.INHERIT)), ): raise TypeError(f"stroke_width must be a number, not {type(width)}") super().__setattr__(PDFStyleKeys.STROKE_WIDTH.value, width) @property def stroke_cap_style(self): """The desired stroke cap style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_CAP_STYLE.value) @stroke_cap_style.setter def stroke_cap_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_CAP_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_CAP_STYLE.value, StrokeCapStyle.coerce(value) ) @property def stroke_join_style(self): """The desired stroke join style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_JOIN_STYLE.value) @stroke_join_style.setter def stroke_join_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_JOIN_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_JOIN_STYLE.value, StrokeJoinStyle.coerce(value), ) @property def stroke_miter_limit(self): """The desired stroke miter limit for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_MITER_LIMIT.value) @stroke_miter_limit.setter def stroke_miter_limit(self, value): if (value is self.INHERIT) or isinstance(value, NumberClass): super().__setattr__(PDFStyleKeys.STROKE_MITER_LIMIT.value, value) else: raise TypeError(f"{value} is not a number") @property def stroke_dash_pattern(self): """The desired stroke dash pattern for this path/group.""" return self._stroke_dash_pattern # pylint: disable=no-member @stroke_dash_pattern.setter def stroke_dash_pattern(self, value): if value is None: result = () elif value is self.INHERIT: result = value elif isinstance(value, NumberClass): result = (value,) else: try: accum = [] for item in value: if not isinstance(item, NumberClass): raise TypeError( f"stroke_dash_pattern {value} sequence has non-numeric value" ) accum.append(item) except TypeError: raise TypeError( f"stroke_dash_pattern {value} must be a number or sequence of numbers" ) from None result = (*accum,) super().__setattr__("_stroke_dash_pattern", result) @property def stroke_dash_phase(self): """The desired stroke dash pattern phase offset for this path/group.""" return self._stroke_dash_phase # pylint: disable=no-member @stroke_dash_phase.setter def stroke_dash_phase(self, value): if value is self.INHERIT or isinstance(value, NumberClass): return super().__setattr__("_stroke_dash_phase", value) raise TypeError(f"{value} isn't a number or GraphicsStyle.INHERIT") @property def soft_mask(self): return getattr(self, PDFStyleKeys.SOFT_MASK.value) @soft_mask.setter def soft_mask(self, value): if value is self.INHERIT or isinstance(value, PaintSoftMask): return super().__setattr__(PDFStyleKeys.SOFT_MASK.value, value) raise TypeError(f"{value} isn't a PaintSoftMask or GraphicsStyle.INHERIT") def serialize(self) -> Optional[Raw]: """ Convert this style object to a PDF dictionary with appropriate style keys. Only explicitly specified values are emitted. """ result = OrderedDict() for key in self.PDF_STYLE_KEYS: value = getattr(self, key, self.INHERIT) if (value is not self.INHERIT) and (value is not None): # None is used for out-of-band signaling on these, e.g. a stroke_width # of None doesn't need to land here because it signals the # PathPaintRule auto resolution only. result[key] = value # There is additional logic in GraphicsContext to ensure that this will work if self.stroke_dash_pattern and self.stroke_dash_pattern is not self.INHERIT: result[PDFStyleKeys.STROKE_DASH_PATTERN.value] = [ self.stroke_dash_pattern, self.stroke_dash_phase, ] if self.allow_transparency is False: for key in self.TRANSPARENCY_KEYS: if key in result: del result[key] if result: # Only insert this key if there is at least one other item in the result so # that we don't junk up the output PDF with empty ExtGState dictionaries. type_name = Name("Type") result[type_name] = Name("ExtGState") result.move_to_end(type_name, last=False) return render_pdf_primitive(result) # this signals to the graphics state registry that there is nothing to # register. This is a success case. return None @force_nodocument def resolve_paint_rule(self) -> PathPaintRule: """ Resolve `PathPaintRule.AUTO` to a real paint rule based on this style. Returns: the resolved `PathPaintRule`. """ if self.paint_rule is PathPaintRule.AUTO: want = set() if self.stroke_width is not None and self.stroke_color is not None: want.add("stroke") if self.fill_color is not None: want.add("fill") # we need to guarantee that this will not be None. The default will # be "nonzero". assert self.intersection_rule is not None want.add(self.intersection_rule) try: rule = self._PAINT_RULE_LOOKUP[frozenset(want)] except KeyError: # don't default to DONT_PAINT because that's almost certainly not a very # good default. rule = PathPaintRule.STROKE_FILL_NONZERO elif self.paint_rule is self.INHERIT: # this shouldn't happen under normal usage, but certain API (ab)use can end # up in this state. We can't resolve anything meaningful, so fall back to a # sane(?) default. rule = PathPaintRule.STROKE_FILL_NONZERO else: rule = self.paint_rule return rule
var LC
-
Expand source code Browse git
class GraphicsStyle: """ A class representing various style attributes that determine drawing appearance. This class uses the convention that the global Python singleton ellipsis (`...`) is exclusively used to represent values that are inherited from the parent style. This is to disambiguate the value None which is used for several values to signal an explicitly disabled style. An example of this is the fill/stroke color styles, which use None as hints to the auto paint style detection code. """ _PRIVATE_SLOTS = ( "_allow_transparency", "_auto_close", "_fill_color", "_intersection_rule", "_paint_rule", "_stroke_color", "_stroke_dash_pattern", "_stroke_dash_phase", ) __slots__ = _PRIVATE_SLOTS + tuple( k.value for k in PDFStyleKeys # we do not store STROKE_DASH_PATTERN under its PDF key; it's in _stroke_dash_pattern if k is not PDFStyleKeys.STROKE_DASH_PATTERN ) INHERIT = ... """Singleton specifying a style parameter should be inherited from the parent context.""" # order is be important here because some of these properties are entangled, e.g. # fill_color and fill_opacity MERGE_PROPERTIES = ( "paint_rule", "allow_transparency", "auto_close", "intersection_rule", "fill_color", "fill_opacity", "stroke_color", "stroke_opacity", "blend_mode", "stroke_width", "stroke_cap_style", "stroke_join_style", "stroke_miter_limit", "stroke_dash_pattern", "stroke_dash_phase", "soft_mask", ) """An ordered collection of properties to use when merging two GraphicsStyles.""" TRANSPARENCY_KEYS = ( PDFStyleKeys.FILL_ALPHA.value, PDFStyleKeys.STROKE_ALPHA.value, PDFStyleKeys.BLEND_MODE.value, PDFStyleKeys.SOFT_MASK.value, ) """An ordered collection of attributes not to emit in no transparency mode.""" PDF_STYLE_KEYS = ( *(k.value for k in PDFStyleKeys if k is not PDFStyleKeys.STROKE_DASH_PATTERN), ) """An ordered collection of keys to directly emit when serializing the style.""" _PAINT_RULE_LOOKUP = { frozenset({}): PathPaintRule.DONT_PAINT, frozenset({"stroke"}): PathPaintRule.STROKE, frozenset({"fill", IntersectionRule.NONZERO}): PathPaintRule.FILL_NONZERO, frozenset({"fill", IntersectionRule.EVENODD}): PathPaintRule.FILL_EVENODD, frozenset( {"stroke", "fill", IntersectionRule.NONZERO} ): PathPaintRule.STROKE_FILL_NONZERO, frozenset( {"stroke", "fill", IntersectionRule.EVENODD} ): PathPaintRule.STROKE_FILL_EVENODD, } """A dictionary for resolving `PathPaintRule.AUTO`""" @classmethod def merge(cls, parent: "GraphicsStyle", child: "GraphicsStyle") -> "GraphicsStyle": """ Merge parent and child into a single GraphicsStyle. The result contains the properties of the parent as overridden by any properties explicitly set on the child. If both the parent and the child specify to inherit a given property, that property will preserve the inherit value. """ new = cls() for prop in cls.MERGE_PROPERTIES: cval = getattr(child, prop) if cval is cls.INHERIT: setattr(new, prop, getattr(parent, prop)) else: setattr(new, prop, cval) return new def __init__(self): self.allow_transparency = self.INHERIT self.paint_rule = self.INHERIT self.auto_close = self.INHERIT self.intersection_rule = self.INHERIT self.fill_color = self.INHERIT self.fill_opacity = self.INHERIT self.stroke_color = self.INHERIT self.stroke_opacity = self.INHERIT self.blend_mode = self.INHERIT self.stroke_width = self.INHERIT self.stroke_cap_style = self.INHERIT self.stroke_join_style = self.INHERIT self.stroke_miter_limit = self.INHERIT self.stroke_dash_pattern = self.INHERIT self.stroke_dash_phase = self.INHERIT self.soft_mask = self.INHERIT def __deepcopy__(self, memo): copied = self.__class__() for prop in self.MERGE_PROPERTIES: setattr(copied, prop, getattr(self, prop)) return copied def __setattr__(self, name, value): if not hasattr(self.__class__, name): raise AttributeError( f'{self.__class__} does not have style "{name}" (a typo?)' ) super().__setattr__(name, value) # at some point it probably makes sense to turn this into a general compliance # property, but for now this is the simple approach. @property def allow_transparency(self): return self._allow_transparency # pylint: disable=no-member @allow_transparency.setter def allow_transparency(self, new): return super().__setattr__("_allow_transparency", new) # If these are used in a nested graphics context inside of a painting path # operation, they are no-ops. However, they can be used for outer GraphicsContexts # that painting paths inherit from. @property def paint_rule(self): """The paint rule to use for this path/group.""" return self._paint_rule # pylint: disable=no-member @paint_rule.setter def paint_rule(self, new): if new is None: super().__setattr__("_paint_rule", PathPaintRule.DONT_PAINT) elif new is self.INHERIT: super().__setattr__("_paint_rule", new) else: super().__setattr__("_paint_rule", PathPaintRule.coerce(new)) @property def auto_close(self): """If True, unclosed paths will be automatically closed before stroking.""" return self._auto_close # pylint: disable=no-member @auto_close.setter def auto_close(self, new): if new not in {True, False, self.INHERIT}: raise TypeError(f"auto_close must be a bool or self.INHERIT, not {new}") super().__setattr__("_auto_close", new) @property def intersection_rule(self): """The desired intersection rule for this path/group.""" return self._intersection_rule # pylint: disable=no-member @intersection_rule.setter def intersection_rule(self, new): # don't allow None for this one. if new is self.INHERIT: super().__setattr__("_intersection_rule", new) else: super().__setattr__("_intersection_rule", IntersectionRule.coerce(new)) @property def fill_color(self): """ The desired fill color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._fill_color # pylint: disable=no-member @fill_color.setter def fill_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_fill_color", color) if getattr(color, "a", None) is not None: self.fill_opacity = color.a elif (color is None) or (color is self.INHERIT): super().__setattr__("_fill_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def fill_opacity(self): """The desired fill opacity for this path/group.""" return getattr(self, PDFStyleKeys.FILL_ALPHA.value) @fill_opacity.setter def fill_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.FILL_ALPHA.value, new) @property def stroke_color(self): """ The desired stroke color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._stroke_color # pylint: disable=no-member @stroke_color.setter def stroke_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_stroke_color", color) if getattr(color, "a", None) is not None: self.stroke_opacity = color.a if self.stroke_width is self.INHERIT: self.stroke_width = 1 elif (color is None) or (color is self.INHERIT): super().__setattr__("_stroke_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def stroke_opacity(self): """The desired stroke opacity for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_ALPHA.value) @stroke_opacity.setter def stroke_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.STROKE_ALPHA.value, new) @property def blend_mode(self): """The desired blend mode for this path/group.""" return getattr(self, PDFStyleKeys.BLEND_MODE.value) @blend_mode.setter def blend_mode(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.BLEND_MODE.value, value) else: super().__setattr__( PDFStyleKeys.BLEND_MODE.value, BlendMode.coerce(value).value ) @property def stroke_width(self): """The desired stroke width for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_WIDTH.value) @stroke_width.setter def stroke_width(self, width): if not isinstance( width, (int, float, decimal.Decimal, type(None), type(self.INHERIT)), ): raise TypeError(f"stroke_width must be a number, not {type(width)}") super().__setattr__(PDFStyleKeys.STROKE_WIDTH.value, width) @property def stroke_cap_style(self): """The desired stroke cap style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_CAP_STYLE.value) @stroke_cap_style.setter def stroke_cap_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_CAP_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_CAP_STYLE.value, StrokeCapStyle.coerce(value) ) @property def stroke_join_style(self): """The desired stroke join style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_JOIN_STYLE.value) @stroke_join_style.setter def stroke_join_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_JOIN_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_JOIN_STYLE.value, StrokeJoinStyle.coerce(value), ) @property def stroke_miter_limit(self): """The desired stroke miter limit for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_MITER_LIMIT.value) @stroke_miter_limit.setter def stroke_miter_limit(self, value): if (value is self.INHERIT) or isinstance(value, NumberClass): super().__setattr__(PDFStyleKeys.STROKE_MITER_LIMIT.value, value) else: raise TypeError(f"{value} is not a number") @property def stroke_dash_pattern(self): """The desired stroke dash pattern for this path/group.""" return self._stroke_dash_pattern # pylint: disable=no-member @stroke_dash_pattern.setter def stroke_dash_pattern(self, value): if value is None: result = () elif value is self.INHERIT: result = value elif isinstance(value, NumberClass): result = (value,) else: try: accum = [] for item in value: if not isinstance(item, NumberClass): raise TypeError( f"stroke_dash_pattern {value} sequence has non-numeric value" ) accum.append(item) except TypeError: raise TypeError( f"stroke_dash_pattern {value} must be a number or sequence of numbers" ) from None result = (*accum,) super().__setattr__("_stroke_dash_pattern", result) @property def stroke_dash_phase(self): """The desired stroke dash pattern phase offset for this path/group.""" return self._stroke_dash_phase # pylint: disable=no-member @stroke_dash_phase.setter def stroke_dash_phase(self, value): if value is self.INHERIT or isinstance(value, NumberClass): return super().__setattr__("_stroke_dash_phase", value) raise TypeError(f"{value} isn't a number or GraphicsStyle.INHERIT") @property def soft_mask(self): return getattr(self, PDFStyleKeys.SOFT_MASK.value) @soft_mask.setter def soft_mask(self, value): if value is self.INHERIT or isinstance(value, PaintSoftMask): return super().__setattr__(PDFStyleKeys.SOFT_MASK.value, value) raise TypeError(f"{value} isn't a PaintSoftMask or GraphicsStyle.INHERIT") def serialize(self) -> Optional[Raw]: """ Convert this style object to a PDF dictionary with appropriate style keys. Only explicitly specified values are emitted. """ result = OrderedDict() for key in self.PDF_STYLE_KEYS: value = getattr(self, key, self.INHERIT) if (value is not self.INHERIT) and (value is not None): # None is used for out-of-band signaling on these, e.g. a stroke_width # of None doesn't need to land here because it signals the # PathPaintRule auto resolution only. result[key] = value # There is additional logic in GraphicsContext to ensure that this will work if self.stroke_dash_pattern and self.stroke_dash_pattern is not self.INHERIT: result[PDFStyleKeys.STROKE_DASH_PATTERN.value] = [ self.stroke_dash_pattern, self.stroke_dash_phase, ] if self.allow_transparency is False: for key in self.TRANSPARENCY_KEYS: if key in result: del result[key] if result: # Only insert this key if there is at least one other item in the result so # that we don't junk up the output PDF with empty ExtGState dictionaries. type_name = Name("Type") result[type_name] = Name("ExtGState") result.move_to_end(type_name, last=False) return render_pdf_primitive(result) # this signals to the graphics state registry that there is nothing to # register. This is a success case. return None @force_nodocument def resolve_paint_rule(self) -> PathPaintRule: """ Resolve `PathPaintRule.AUTO` to a real paint rule based on this style. Returns: the resolved `PathPaintRule`. """ if self.paint_rule is PathPaintRule.AUTO: want = set() if self.stroke_width is not None and self.stroke_color is not None: want.add("stroke") if self.fill_color is not None: want.add("fill") # we need to guarantee that this will not be None. The default will # be "nonzero". assert self.intersection_rule is not None want.add(self.intersection_rule) try: rule = self._PAINT_RULE_LOOKUP[frozenset(want)] except KeyError: # don't default to DONT_PAINT because that's almost certainly not a very # good default. rule = PathPaintRule.STROKE_FILL_NONZERO elif self.paint_rule is self.INHERIT: # this shouldn't happen under normal usage, but certain API (ab)use can end # up in this state. We can't resolve anything meaningful, so fall back to a # sane(?) default. rule = PathPaintRule.STROKE_FILL_NONZERO else: rule = self.paint_rule return rule
var LJ
-
Expand source code Browse git
class GraphicsStyle: """ A class representing various style attributes that determine drawing appearance. This class uses the convention that the global Python singleton ellipsis (`...`) is exclusively used to represent values that are inherited from the parent style. This is to disambiguate the value None which is used for several values to signal an explicitly disabled style. An example of this is the fill/stroke color styles, which use None as hints to the auto paint style detection code. """ _PRIVATE_SLOTS = ( "_allow_transparency", "_auto_close", "_fill_color", "_intersection_rule", "_paint_rule", "_stroke_color", "_stroke_dash_pattern", "_stroke_dash_phase", ) __slots__ = _PRIVATE_SLOTS + tuple( k.value for k in PDFStyleKeys # we do not store STROKE_DASH_PATTERN under its PDF key; it's in _stroke_dash_pattern if k is not PDFStyleKeys.STROKE_DASH_PATTERN ) INHERIT = ... """Singleton specifying a style parameter should be inherited from the parent context.""" # order is be important here because some of these properties are entangled, e.g. # fill_color and fill_opacity MERGE_PROPERTIES = ( "paint_rule", "allow_transparency", "auto_close", "intersection_rule", "fill_color", "fill_opacity", "stroke_color", "stroke_opacity", "blend_mode", "stroke_width", "stroke_cap_style", "stroke_join_style", "stroke_miter_limit", "stroke_dash_pattern", "stroke_dash_phase", "soft_mask", ) """An ordered collection of properties to use when merging two GraphicsStyles.""" TRANSPARENCY_KEYS = ( PDFStyleKeys.FILL_ALPHA.value, PDFStyleKeys.STROKE_ALPHA.value, PDFStyleKeys.BLEND_MODE.value, PDFStyleKeys.SOFT_MASK.value, ) """An ordered collection of attributes not to emit in no transparency mode.""" PDF_STYLE_KEYS = ( *(k.value for k in PDFStyleKeys if k is not PDFStyleKeys.STROKE_DASH_PATTERN), ) """An ordered collection of keys to directly emit when serializing the style.""" _PAINT_RULE_LOOKUP = { frozenset({}): PathPaintRule.DONT_PAINT, frozenset({"stroke"}): PathPaintRule.STROKE, frozenset({"fill", IntersectionRule.NONZERO}): PathPaintRule.FILL_NONZERO, frozenset({"fill", IntersectionRule.EVENODD}): PathPaintRule.FILL_EVENODD, frozenset( {"stroke", "fill", IntersectionRule.NONZERO} ): PathPaintRule.STROKE_FILL_NONZERO, frozenset( {"stroke", "fill", IntersectionRule.EVENODD} ): PathPaintRule.STROKE_FILL_EVENODD, } """A dictionary for resolving `PathPaintRule.AUTO`""" @classmethod def merge(cls, parent: "GraphicsStyle", child: "GraphicsStyle") -> "GraphicsStyle": """ Merge parent and child into a single GraphicsStyle. The result contains the properties of the parent as overridden by any properties explicitly set on the child. If both the parent and the child specify to inherit a given property, that property will preserve the inherit value. """ new = cls() for prop in cls.MERGE_PROPERTIES: cval = getattr(child, prop) if cval is cls.INHERIT: setattr(new, prop, getattr(parent, prop)) else: setattr(new, prop, cval) return new def __init__(self): self.allow_transparency = self.INHERIT self.paint_rule = self.INHERIT self.auto_close = self.INHERIT self.intersection_rule = self.INHERIT self.fill_color = self.INHERIT self.fill_opacity = self.INHERIT self.stroke_color = self.INHERIT self.stroke_opacity = self.INHERIT self.blend_mode = self.INHERIT self.stroke_width = self.INHERIT self.stroke_cap_style = self.INHERIT self.stroke_join_style = self.INHERIT self.stroke_miter_limit = self.INHERIT self.stroke_dash_pattern = self.INHERIT self.stroke_dash_phase = self.INHERIT self.soft_mask = self.INHERIT def __deepcopy__(self, memo): copied = self.__class__() for prop in self.MERGE_PROPERTIES: setattr(copied, prop, getattr(self, prop)) return copied def __setattr__(self, name, value): if not hasattr(self.__class__, name): raise AttributeError( f'{self.__class__} does not have style "{name}" (a typo?)' ) super().__setattr__(name, value) # at some point it probably makes sense to turn this into a general compliance # property, but for now this is the simple approach. @property def allow_transparency(self): return self._allow_transparency # pylint: disable=no-member @allow_transparency.setter def allow_transparency(self, new): return super().__setattr__("_allow_transparency", new) # If these are used in a nested graphics context inside of a painting path # operation, they are no-ops. However, they can be used for outer GraphicsContexts # that painting paths inherit from. @property def paint_rule(self): """The paint rule to use for this path/group.""" return self._paint_rule # pylint: disable=no-member @paint_rule.setter def paint_rule(self, new): if new is None: super().__setattr__("_paint_rule", PathPaintRule.DONT_PAINT) elif new is self.INHERIT: super().__setattr__("_paint_rule", new) else: super().__setattr__("_paint_rule", PathPaintRule.coerce(new)) @property def auto_close(self): """If True, unclosed paths will be automatically closed before stroking.""" return self._auto_close # pylint: disable=no-member @auto_close.setter def auto_close(self, new): if new not in {True, False, self.INHERIT}: raise TypeError(f"auto_close must be a bool or self.INHERIT, not {new}") super().__setattr__("_auto_close", new) @property def intersection_rule(self): """The desired intersection rule for this path/group.""" return self._intersection_rule # pylint: disable=no-member @intersection_rule.setter def intersection_rule(self, new): # don't allow None for this one. if new is self.INHERIT: super().__setattr__("_intersection_rule", new) else: super().__setattr__("_intersection_rule", IntersectionRule.coerce(new)) @property def fill_color(self): """ The desired fill color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._fill_color # pylint: disable=no-member @fill_color.setter def fill_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_fill_color", color) if getattr(color, "a", None) is not None: self.fill_opacity = color.a elif (color is None) or (color is self.INHERIT): super().__setattr__("_fill_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def fill_opacity(self): """The desired fill opacity for this path/group.""" return getattr(self, PDFStyleKeys.FILL_ALPHA.value) @fill_opacity.setter def fill_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.FILL_ALPHA.value, new) @property def stroke_color(self): """ The desired stroke color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._stroke_color # pylint: disable=no-member @stroke_color.setter def stroke_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_stroke_color", color) if getattr(color, "a", None) is not None: self.stroke_opacity = color.a if self.stroke_width is self.INHERIT: self.stroke_width = 1 elif (color is None) or (color is self.INHERIT): super().__setattr__("_stroke_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def stroke_opacity(self): """The desired stroke opacity for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_ALPHA.value) @stroke_opacity.setter def stroke_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.STROKE_ALPHA.value, new) @property def blend_mode(self): """The desired blend mode for this path/group.""" return getattr(self, PDFStyleKeys.BLEND_MODE.value) @blend_mode.setter def blend_mode(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.BLEND_MODE.value, value) else: super().__setattr__( PDFStyleKeys.BLEND_MODE.value, BlendMode.coerce(value).value ) @property def stroke_width(self): """The desired stroke width for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_WIDTH.value) @stroke_width.setter def stroke_width(self, width): if not isinstance( width, (int, float, decimal.Decimal, type(None), type(self.INHERIT)), ): raise TypeError(f"stroke_width must be a number, not {type(width)}") super().__setattr__(PDFStyleKeys.STROKE_WIDTH.value, width) @property def stroke_cap_style(self): """The desired stroke cap style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_CAP_STYLE.value) @stroke_cap_style.setter def stroke_cap_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_CAP_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_CAP_STYLE.value, StrokeCapStyle.coerce(value) ) @property def stroke_join_style(self): """The desired stroke join style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_JOIN_STYLE.value) @stroke_join_style.setter def stroke_join_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_JOIN_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_JOIN_STYLE.value, StrokeJoinStyle.coerce(value), ) @property def stroke_miter_limit(self): """The desired stroke miter limit for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_MITER_LIMIT.value) @stroke_miter_limit.setter def stroke_miter_limit(self, value): if (value is self.INHERIT) or isinstance(value, NumberClass): super().__setattr__(PDFStyleKeys.STROKE_MITER_LIMIT.value, value) else: raise TypeError(f"{value} is not a number") @property def stroke_dash_pattern(self): """The desired stroke dash pattern for this path/group.""" return self._stroke_dash_pattern # pylint: disable=no-member @stroke_dash_pattern.setter def stroke_dash_pattern(self, value): if value is None: result = () elif value is self.INHERIT: result = value elif isinstance(value, NumberClass): result = (value,) else: try: accum = [] for item in value: if not isinstance(item, NumberClass): raise TypeError( f"stroke_dash_pattern {value} sequence has non-numeric value" ) accum.append(item) except TypeError: raise TypeError( f"stroke_dash_pattern {value} must be a number or sequence of numbers" ) from None result = (*accum,) super().__setattr__("_stroke_dash_pattern", result) @property def stroke_dash_phase(self): """The desired stroke dash pattern phase offset for this path/group.""" return self._stroke_dash_phase # pylint: disable=no-member @stroke_dash_phase.setter def stroke_dash_phase(self, value): if value is self.INHERIT or isinstance(value, NumberClass): return super().__setattr__("_stroke_dash_phase", value) raise TypeError(f"{value} isn't a number or GraphicsStyle.INHERIT") @property def soft_mask(self): return getattr(self, PDFStyleKeys.SOFT_MASK.value) @soft_mask.setter def soft_mask(self, value): if value is self.INHERIT or isinstance(value, PaintSoftMask): return super().__setattr__(PDFStyleKeys.SOFT_MASK.value, value) raise TypeError(f"{value} isn't a PaintSoftMask or GraphicsStyle.INHERIT") def serialize(self) -> Optional[Raw]: """ Convert this style object to a PDF dictionary with appropriate style keys. Only explicitly specified values are emitted. """ result = OrderedDict() for key in self.PDF_STYLE_KEYS: value = getattr(self, key, self.INHERIT) if (value is not self.INHERIT) and (value is not None): # None is used for out-of-band signaling on these, e.g. a stroke_width # of None doesn't need to land here because it signals the # PathPaintRule auto resolution only. result[key] = value # There is additional logic in GraphicsContext to ensure that this will work if self.stroke_dash_pattern and self.stroke_dash_pattern is not self.INHERIT: result[PDFStyleKeys.STROKE_DASH_PATTERN.value] = [ self.stroke_dash_pattern, self.stroke_dash_phase, ] if self.allow_transparency is False: for key in self.TRANSPARENCY_KEYS: if key in result: del result[key] if result: # Only insert this key if there is at least one other item in the result so # that we don't junk up the output PDF with empty ExtGState dictionaries. type_name = Name("Type") result[type_name] = Name("ExtGState") result.move_to_end(type_name, last=False) return render_pdf_primitive(result) # this signals to the graphics state registry that there is nothing to # register. This is a success case. return None @force_nodocument def resolve_paint_rule(self) -> PathPaintRule: """ Resolve `PathPaintRule.AUTO` to a real paint rule based on this style. Returns: the resolved `PathPaintRule`. """ if self.paint_rule is PathPaintRule.AUTO: want = set() if self.stroke_width is not None and self.stroke_color is not None: want.add("stroke") if self.fill_color is not None: want.add("fill") # we need to guarantee that this will not be None. The default will # be "nonzero". assert self.intersection_rule is not None want.add(self.intersection_rule) try: rule = self._PAINT_RULE_LOOKUP[frozenset(want)] except KeyError: # don't default to DONT_PAINT because that's almost certainly not a very # good default. rule = PathPaintRule.STROKE_FILL_NONZERO elif self.paint_rule is self.INHERIT: # this shouldn't happen under normal usage, but certain API (ab)use can end # up in this state. We can't resolve anything meaningful, so fall back to a # sane(?) default. rule = PathPaintRule.STROKE_FILL_NONZERO else: rule = self.paint_rule return rule
var LW
-
Expand source code Browse git
class GraphicsStyle: """ A class representing various style attributes that determine drawing appearance. This class uses the convention that the global Python singleton ellipsis (`...`) is exclusively used to represent values that are inherited from the parent style. This is to disambiguate the value None which is used for several values to signal an explicitly disabled style. An example of this is the fill/stroke color styles, which use None as hints to the auto paint style detection code. """ _PRIVATE_SLOTS = ( "_allow_transparency", "_auto_close", "_fill_color", "_intersection_rule", "_paint_rule", "_stroke_color", "_stroke_dash_pattern", "_stroke_dash_phase", ) __slots__ = _PRIVATE_SLOTS + tuple( k.value for k in PDFStyleKeys # we do not store STROKE_DASH_PATTERN under its PDF key; it's in _stroke_dash_pattern if k is not PDFStyleKeys.STROKE_DASH_PATTERN ) INHERIT = ... """Singleton specifying a style parameter should be inherited from the parent context.""" # order is be important here because some of these properties are entangled, e.g. # fill_color and fill_opacity MERGE_PROPERTIES = ( "paint_rule", "allow_transparency", "auto_close", "intersection_rule", "fill_color", "fill_opacity", "stroke_color", "stroke_opacity", "blend_mode", "stroke_width", "stroke_cap_style", "stroke_join_style", "stroke_miter_limit", "stroke_dash_pattern", "stroke_dash_phase", "soft_mask", ) """An ordered collection of properties to use when merging two GraphicsStyles.""" TRANSPARENCY_KEYS = ( PDFStyleKeys.FILL_ALPHA.value, PDFStyleKeys.STROKE_ALPHA.value, PDFStyleKeys.BLEND_MODE.value, PDFStyleKeys.SOFT_MASK.value, ) """An ordered collection of attributes not to emit in no transparency mode.""" PDF_STYLE_KEYS = ( *(k.value for k in PDFStyleKeys if k is not PDFStyleKeys.STROKE_DASH_PATTERN), ) """An ordered collection of keys to directly emit when serializing the style.""" _PAINT_RULE_LOOKUP = { frozenset({}): PathPaintRule.DONT_PAINT, frozenset({"stroke"}): PathPaintRule.STROKE, frozenset({"fill", IntersectionRule.NONZERO}): PathPaintRule.FILL_NONZERO, frozenset({"fill", IntersectionRule.EVENODD}): PathPaintRule.FILL_EVENODD, frozenset( {"stroke", "fill", IntersectionRule.NONZERO} ): PathPaintRule.STROKE_FILL_NONZERO, frozenset( {"stroke", "fill", IntersectionRule.EVENODD} ): PathPaintRule.STROKE_FILL_EVENODD, } """A dictionary for resolving `PathPaintRule.AUTO`""" @classmethod def merge(cls, parent: "GraphicsStyle", child: "GraphicsStyle") -> "GraphicsStyle": """ Merge parent and child into a single GraphicsStyle. The result contains the properties of the parent as overridden by any properties explicitly set on the child. If both the parent and the child specify to inherit a given property, that property will preserve the inherit value. """ new = cls() for prop in cls.MERGE_PROPERTIES: cval = getattr(child, prop) if cval is cls.INHERIT: setattr(new, prop, getattr(parent, prop)) else: setattr(new, prop, cval) return new def __init__(self): self.allow_transparency = self.INHERIT self.paint_rule = self.INHERIT self.auto_close = self.INHERIT self.intersection_rule = self.INHERIT self.fill_color = self.INHERIT self.fill_opacity = self.INHERIT self.stroke_color = self.INHERIT self.stroke_opacity = self.INHERIT self.blend_mode = self.INHERIT self.stroke_width = self.INHERIT self.stroke_cap_style = self.INHERIT self.stroke_join_style = self.INHERIT self.stroke_miter_limit = self.INHERIT self.stroke_dash_pattern = self.INHERIT self.stroke_dash_phase = self.INHERIT self.soft_mask = self.INHERIT def __deepcopy__(self, memo): copied = self.__class__() for prop in self.MERGE_PROPERTIES: setattr(copied, prop, getattr(self, prop)) return copied def __setattr__(self, name, value): if not hasattr(self.__class__, name): raise AttributeError( f'{self.__class__} does not have style "{name}" (a typo?)' ) super().__setattr__(name, value) # at some point it probably makes sense to turn this into a general compliance # property, but for now this is the simple approach. @property def allow_transparency(self): return self._allow_transparency # pylint: disable=no-member @allow_transparency.setter def allow_transparency(self, new): return super().__setattr__("_allow_transparency", new) # If these are used in a nested graphics context inside of a painting path # operation, they are no-ops. However, they can be used for outer GraphicsContexts # that painting paths inherit from. @property def paint_rule(self): """The paint rule to use for this path/group.""" return self._paint_rule # pylint: disable=no-member @paint_rule.setter def paint_rule(self, new): if new is None: super().__setattr__("_paint_rule", PathPaintRule.DONT_PAINT) elif new is self.INHERIT: super().__setattr__("_paint_rule", new) else: super().__setattr__("_paint_rule", PathPaintRule.coerce(new)) @property def auto_close(self): """If True, unclosed paths will be automatically closed before stroking.""" return self._auto_close # pylint: disable=no-member @auto_close.setter def auto_close(self, new): if new not in {True, False, self.INHERIT}: raise TypeError(f"auto_close must be a bool or self.INHERIT, not {new}") super().__setattr__("_auto_close", new) @property def intersection_rule(self): """The desired intersection rule for this path/group.""" return self._intersection_rule # pylint: disable=no-member @intersection_rule.setter def intersection_rule(self, new): # don't allow None for this one. if new is self.INHERIT: super().__setattr__("_intersection_rule", new) else: super().__setattr__("_intersection_rule", IntersectionRule.coerce(new)) @property def fill_color(self): """ The desired fill color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._fill_color # pylint: disable=no-member @fill_color.setter def fill_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_fill_color", color) if getattr(color, "a", None) is not None: self.fill_opacity = color.a elif (color is None) or (color is self.INHERIT): super().__setattr__("_fill_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def fill_opacity(self): """The desired fill opacity for this path/group.""" return getattr(self, PDFStyleKeys.FILL_ALPHA.value) @fill_opacity.setter def fill_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.FILL_ALPHA.value, new) @property def stroke_color(self): """ The desired stroke color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._stroke_color # pylint: disable=no-member @stroke_color.setter def stroke_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_stroke_color", color) if getattr(color, "a", None) is not None: self.stroke_opacity = color.a if self.stroke_width is self.INHERIT: self.stroke_width = 1 elif (color is None) or (color is self.INHERIT): super().__setattr__("_stroke_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def stroke_opacity(self): """The desired stroke opacity for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_ALPHA.value) @stroke_opacity.setter def stroke_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.STROKE_ALPHA.value, new) @property def blend_mode(self): """The desired blend mode for this path/group.""" return getattr(self, PDFStyleKeys.BLEND_MODE.value) @blend_mode.setter def blend_mode(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.BLEND_MODE.value, value) else: super().__setattr__( PDFStyleKeys.BLEND_MODE.value, BlendMode.coerce(value).value ) @property def stroke_width(self): """The desired stroke width for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_WIDTH.value) @stroke_width.setter def stroke_width(self, width): if not isinstance( width, (int, float, decimal.Decimal, type(None), type(self.INHERIT)), ): raise TypeError(f"stroke_width must be a number, not {type(width)}") super().__setattr__(PDFStyleKeys.STROKE_WIDTH.value, width) @property def stroke_cap_style(self): """The desired stroke cap style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_CAP_STYLE.value) @stroke_cap_style.setter def stroke_cap_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_CAP_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_CAP_STYLE.value, StrokeCapStyle.coerce(value) ) @property def stroke_join_style(self): """The desired stroke join style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_JOIN_STYLE.value) @stroke_join_style.setter def stroke_join_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_JOIN_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_JOIN_STYLE.value, StrokeJoinStyle.coerce(value), ) @property def stroke_miter_limit(self): """The desired stroke miter limit for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_MITER_LIMIT.value) @stroke_miter_limit.setter def stroke_miter_limit(self, value): if (value is self.INHERIT) or isinstance(value, NumberClass): super().__setattr__(PDFStyleKeys.STROKE_MITER_LIMIT.value, value) else: raise TypeError(f"{value} is not a number") @property def stroke_dash_pattern(self): """The desired stroke dash pattern for this path/group.""" return self._stroke_dash_pattern # pylint: disable=no-member @stroke_dash_pattern.setter def stroke_dash_pattern(self, value): if value is None: result = () elif value is self.INHERIT: result = value elif isinstance(value, NumberClass): result = (value,) else: try: accum = [] for item in value: if not isinstance(item, NumberClass): raise TypeError( f"stroke_dash_pattern {value} sequence has non-numeric value" ) accum.append(item) except TypeError: raise TypeError( f"stroke_dash_pattern {value} must be a number or sequence of numbers" ) from None result = (*accum,) super().__setattr__("_stroke_dash_pattern", result) @property def stroke_dash_phase(self): """The desired stroke dash pattern phase offset for this path/group.""" return self._stroke_dash_phase # pylint: disable=no-member @stroke_dash_phase.setter def stroke_dash_phase(self, value): if value is self.INHERIT or isinstance(value, NumberClass): return super().__setattr__("_stroke_dash_phase", value) raise TypeError(f"{value} isn't a number or GraphicsStyle.INHERIT") @property def soft_mask(self): return getattr(self, PDFStyleKeys.SOFT_MASK.value) @soft_mask.setter def soft_mask(self, value): if value is self.INHERIT or isinstance(value, PaintSoftMask): return super().__setattr__(PDFStyleKeys.SOFT_MASK.value, value) raise TypeError(f"{value} isn't a PaintSoftMask or GraphicsStyle.INHERIT") def serialize(self) -> Optional[Raw]: """ Convert this style object to a PDF dictionary with appropriate style keys. Only explicitly specified values are emitted. """ result = OrderedDict() for key in self.PDF_STYLE_KEYS: value = getattr(self, key, self.INHERIT) if (value is not self.INHERIT) and (value is not None): # None is used for out-of-band signaling on these, e.g. a stroke_width # of None doesn't need to land here because it signals the # PathPaintRule auto resolution only. result[key] = value # There is additional logic in GraphicsContext to ensure that this will work if self.stroke_dash_pattern and self.stroke_dash_pattern is not self.INHERIT: result[PDFStyleKeys.STROKE_DASH_PATTERN.value] = [ self.stroke_dash_pattern, self.stroke_dash_phase, ] if self.allow_transparency is False: for key in self.TRANSPARENCY_KEYS: if key in result: del result[key] if result: # Only insert this key if there is at least one other item in the result so # that we don't junk up the output PDF with empty ExtGState dictionaries. type_name = Name("Type") result[type_name] = Name("ExtGState") result.move_to_end(type_name, last=False) return render_pdf_primitive(result) # this signals to the graphics state registry that there is nothing to # register. This is a success case. return None @force_nodocument def resolve_paint_rule(self) -> PathPaintRule: """ Resolve `PathPaintRule.AUTO` to a real paint rule based on this style. Returns: the resolved `PathPaintRule`. """ if self.paint_rule is PathPaintRule.AUTO: want = set() if self.stroke_width is not None and self.stroke_color is not None: want.add("stroke") if self.fill_color is not None: want.add("fill") # we need to guarantee that this will not be None. The default will # be "nonzero". assert self.intersection_rule is not None want.add(self.intersection_rule) try: rule = self._PAINT_RULE_LOOKUP[frozenset(want)] except KeyError: # don't default to DONT_PAINT because that's almost certainly not a very # good default. rule = PathPaintRule.STROKE_FILL_NONZERO elif self.paint_rule is self.INHERIT: # this shouldn't happen under normal usage, but certain API (ab)use can end # up in this state. We can't resolve anything meaningful, so fall back to a # sane(?) default. rule = PathPaintRule.STROKE_FILL_NONZERO else: rule = self.paint_rule return rule
var ML
-
Expand source code Browse git
class GraphicsStyle: """ A class representing various style attributes that determine drawing appearance. This class uses the convention that the global Python singleton ellipsis (`...`) is exclusively used to represent values that are inherited from the parent style. This is to disambiguate the value None which is used for several values to signal an explicitly disabled style. An example of this is the fill/stroke color styles, which use None as hints to the auto paint style detection code. """ _PRIVATE_SLOTS = ( "_allow_transparency", "_auto_close", "_fill_color", "_intersection_rule", "_paint_rule", "_stroke_color", "_stroke_dash_pattern", "_stroke_dash_phase", ) __slots__ = _PRIVATE_SLOTS + tuple( k.value for k in PDFStyleKeys # we do not store STROKE_DASH_PATTERN under its PDF key; it's in _stroke_dash_pattern if k is not PDFStyleKeys.STROKE_DASH_PATTERN ) INHERIT = ... """Singleton specifying a style parameter should be inherited from the parent context.""" # order is be important here because some of these properties are entangled, e.g. # fill_color and fill_opacity MERGE_PROPERTIES = ( "paint_rule", "allow_transparency", "auto_close", "intersection_rule", "fill_color", "fill_opacity", "stroke_color", "stroke_opacity", "blend_mode", "stroke_width", "stroke_cap_style", "stroke_join_style", "stroke_miter_limit", "stroke_dash_pattern", "stroke_dash_phase", "soft_mask", ) """An ordered collection of properties to use when merging two GraphicsStyles.""" TRANSPARENCY_KEYS = ( PDFStyleKeys.FILL_ALPHA.value, PDFStyleKeys.STROKE_ALPHA.value, PDFStyleKeys.BLEND_MODE.value, PDFStyleKeys.SOFT_MASK.value, ) """An ordered collection of attributes not to emit in no transparency mode.""" PDF_STYLE_KEYS = ( *(k.value for k in PDFStyleKeys if k is not PDFStyleKeys.STROKE_DASH_PATTERN), ) """An ordered collection of keys to directly emit when serializing the style.""" _PAINT_RULE_LOOKUP = { frozenset({}): PathPaintRule.DONT_PAINT, frozenset({"stroke"}): PathPaintRule.STROKE, frozenset({"fill", IntersectionRule.NONZERO}): PathPaintRule.FILL_NONZERO, frozenset({"fill", IntersectionRule.EVENODD}): PathPaintRule.FILL_EVENODD, frozenset( {"stroke", "fill", IntersectionRule.NONZERO} ): PathPaintRule.STROKE_FILL_NONZERO, frozenset( {"stroke", "fill", IntersectionRule.EVENODD} ): PathPaintRule.STROKE_FILL_EVENODD, } """A dictionary for resolving `PathPaintRule.AUTO`""" @classmethod def merge(cls, parent: "GraphicsStyle", child: "GraphicsStyle") -> "GraphicsStyle": """ Merge parent and child into a single GraphicsStyle. The result contains the properties of the parent as overridden by any properties explicitly set on the child. If both the parent and the child specify to inherit a given property, that property will preserve the inherit value. """ new = cls() for prop in cls.MERGE_PROPERTIES: cval = getattr(child, prop) if cval is cls.INHERIT: setattr(new, prop, getattr(parent, prop)) else: setattr(new, prop, cval) return new def __init__(self): self.allow_transparency = self.INHERIT self.paint_rule = self.INHERIT self.auto_close = self.INHERIT self.intersection_rule = self.INHERIT self.fill_color = self.INHERIT self.fill_opacity = self.INHERIT self.stroke_color = self.INHERIT self.stroke_opacity = self.INHERIT self.blend_mode = self.INHERIT self.stroke_width = self.INHERIT self.stroke_cap_style = self.INHERIT self.stroke_join_style = self.INHERIT self.stroke_miter_limit = self.INHERIT self.stroke_dash_pattern = self.INHERIT self.stroke_dash_phase = self.INHERIT self.soft_mask = self.INHERIT def __deepcopy__(self, memo): copied = self.__class__() for prop in self.MERGE_PROPERTIES: setattr(copied, prop, getattr(self, prop)) return copied def __setattr__(self, name, value): if not hasattr(self.__class__, name): raise AttributeError( f'{self.__class__} does not have style "{name}" (a typo?)' ) super().__setattr__(name, value) # at some point it probably makes sense to turn this into a general compliance # property, but for now this is the simple approach. @property def allow_transparency(self): return self._allow_transparency # pylint: disable=no-member @allow_transparency.setter def allow_transparency(self, new): return super().__setattr__("_allow_transparency", new) # If these are used in a nested graphics context inside of a painting path # operation, they are no-ops. However, they can be used for outer GraphicsContexts # that painting paths inherit from. @property def paint_rule(self): """The paint rule to use for this path/group.""" return self._paint_rule # pylint: disable=no-member @paint_rule.setter def paint_rule(self, new): if new is None: super().__setattr__("_paint_rule", PathPaintRule.DONT_PAINT) elif new is self.INHERIT: super().__setattr__("_paint_rule", new) else: super().__setattr__("_paint_rule", PathPaintRule.coerce(new)) @property def auto_close(self): """If True, unclosed paths will be automatically closed before stroking.""" return self._auto_close # pylint: disable=no-member @auto_close.setter def auto_close(self, new): if new not in {True, False, self.INHERIT}: raise TypeError(f"auto_close must be a bool or self.INHERIT, not {new}") super().__setattr__("_auto_close", new) @property def intersection_rule(self): """The desired intersection rule for this path/group.""" return self._intersection_rule # pylint: disable=no-member @intersection_rule.setter def intersection_rule(self, new): # don't allow None for this one. if new is self.INHERIT: super().__setattr__("_intersection_rule", new) else: super().__setattr__("_intersection_rule", IntersectionRule.coerce(new)) @property def fill_color(self): """ The desired fill color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._fill_color # pylint: disable=no-member @fill_color.setter def fill_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_fill_color", color) if getattr(color, "a", None) is not None: self.fill_opacity = color.a elif (color is None) or (color is self.INHERIT): super().__setattr__("_fill_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def fill_opacity(self): """The desired fill opacity for this path/group.""" return getattr(self, PDFStyleKeys.FILL_ALPHA.value) @fill_opacity.setter def fill_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.FILL_ALPHA.value, new) @property def stroke_color(self): """ The desired stroke color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._stroke_color # pylint: disable=no-member @stroke_color.setter def stroke_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_stroke_color", color) if getattr(color, "a", None) is not None: self.stroke_opacity = color.a if self.stroke_width is self.INHERIT: self.stroke_width = 1 elif (color is None) or (color is self.INHERIT): super().__setattr__("_stroke_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def stroke_opacity(self): """The desired stroke opacity for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_ALPHA.value) @stroke_opacity.setter def stroke_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.STROKE_ALPHA.value, new) @property def blend_mode(self): """The desired blend mode for this path/group.""" return getattr(self, PDFStyleKeys.BLEND_MODE.value) @blend_mode.setter def blend_mode(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.BLEND_MODE.value, value) else: super().__setattr__( PDFStyleKeys.BLEND_MODE.value, BlendMode.coerce(value).value ) @property def stroke_width(self): """The desired stroke width for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_WIDTH.value) @stroke_width.setter def stroke_width(self, width): if not isinstance( width, (int, float, decimal.Decimal, type(None), type(self.INHERIT)), ): raise TypeError(f"stroke_width must be a number, not {type(width)}") super().__setattr__(PDFStyleKeys.STROKE_WIDTH.value, width) @property def stroke_cap_style(self): """The desired stroke cap style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_CAP_STYLE.value) @stroke_cap_style.setter def stroke_cap_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_CAP_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_CAP_STYLE.value, StrokeCapStyle.coerce(value) ) @property def stroke_join_style(self): """The desired stroke join style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_JOIN_STYLE.value) @stroke_join_style.setter def stroke_join_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_JOIN_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_JOIN_STYLE.value, StrokeJoinStyle.coerce(value), ) @property def stroke_miter_limit(self): """The desired stroke miter limit for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_MITER_LIMIT.value) @stroke_miter_limit.setter def stroke_miter_limit(self, value): if (value is self.INHERIT) or isinstance(value, NumberClass): super().__setattr__(PDFStyleKeys.STROKE_MITER_LIMIT.value, value) else: raise TypeError(f"{value} is not a number") @property def stroke_dash_pattern(self): """The desired stroke dash pattern for this path/group.""" return self._stroke_dash_pattern # pylint: disable=no-member @stroke_dash_pattern.setter def stroke_dash_pattern(self, value): if value is None: result = () elif value is self.INHERIT: result = value elif isinstance(value, NumberClass): result = (value,) else: try: accum = [] for item in value: if not isinstance(item, NumberClass): raise TypeError( f"stroke_dash_pattern {value} sequence has non-numeric value" ) accum.append(item) except TypeError: raise TypeError( f"stroke_dash_pattern {value} must be a number or sequence of numbers" ) from None result = (*accum,) super().__setattr__("_stroke_dash_pattern", result) @property def stroke_dash_phase(self): """The desired stroke dash pattern phase offset for this path/group.""" return self._stroke_dash_phase # pylint: disable=no-member @stroke_dash_phase.setter def stroke_dash_phase(self, value): if value is self.INHERIT or isinstance(value, NumberClass): return super().__setattr__("_stroke_dash_phase", value) raise TypeError(f"{value} isn't a number or GraphicsStyle.INHERIT") @property def soft_mask(self): return getattr(self, PDFStyleKeys.SOFT_MASK.value) @soft_mask.setter def soft_mask(self, value): if value is self.INHERIT or isinstance(value, PaintSoftMask): return super().__setattr__(PDFStyleKeys.SOFT_MASK.value, value) raise TypeError(f"{value} isn't a PaintSoftMask or GraphicsStyle.INHERIT") def serialize(self) -> Optional[Raw]: """ Convert this style object to a PDF dictionary with appropriate style keys. Only explicitly specified values are emitted. """ result = OrderedDict() for key in self.PDF_STYLE_KEYS: value = getattr(self, key, self.INHERIT) if (value is not self.INHERIT) and (value is not None): # None is used for out-of-band signaling on these, e.g. a stroke_width # of None doesn't need to land here because it signals the # PathPaintRule auto resolution only. result[key] = value # There is additional logic in GraphicsContext to ensure that this will work if self.stroke_dash_pattern and self.stroke_dash_pattern is not self.INHERIT: result[PDFStyleKeys.STROKE_DASH_PATTERN.value] = [ self.stroke_dash_pattern, self.stroke_dash_phase, ] if self.allow_transparency is False: for key in self.TRANSPARENCY_KEYS: if key in result: del result[key] if result: # Only insert this key if there is at least one other item in the result so # that we don't junk up the output PDF with empty ExtGState dictionaries. type_name = Name("Type") result[type_name] = Name("ExtGState") result.move_to_end(type_name, last=False) return render_pdf_primitive(result) # this signals to the graphics state registry that there is nothing to # register. This is a success case. return None @force_nodocument def resolve_paint_rule(self) -> PathPaintRule: """ Resolve `PathPaintRule.AUTO` to a real paint rule based on this style. Returns: the resolved `PathPaintRule`. """ if self.paint_rule is PathPaintRule.AUTO: want = set() if self.stroke_width is not None and self.stroke_color is not None: want.add("stroke") if self.fill_color is not None: want.add("fill") # we need to guarantee that this will not be None. The default will # be "nonzero". assert self.intersection_rule is not None want.add(self.intersection_rule) try: rule = self._PAINT_RULE_LOOKUP[frozenset(want)] except KeyError: # don't default to DONT_PAINT because that's almost certainly not a very # good default. rule = PathPaintRule.STROKE_FILL_NONZERO elif self.paint_rule is self.INHERIT: # this shouldn't happen under normal usage, but certain API (ab)use can end # up in this state. We can't resolve anything meaningful, so fall back to a # sane(?) default. rule = PathPaintRule.STROKE_FILL_NONZERO else: rule = self.paint_rule return rule
var SA
-
Expand source code Browse git
class GraphicsStyle: """ A class representing various style attributes that determine drawing appearance. This class uses the convention that the global Python singleton ellipsis (`...`) is exclusively used to represent values that are inherited from the parent style. This is to disambiguate the value None which is used for several values to signal an explicitly disabled style. An example of this is the fill/stroke color styles, which use None as hints to the auto paint style detection code. """ _PRIVATE_SLOTS = ( "_allow_transparency", "_auto_close", "_fill_color", "_intersection_rule", "_paint_rule", "_stroke_color", "_stroke_dash_pattern", "_stroke_dash_phase", ) __slots__ = _PRIVATE_SLOTS + tuple( k.value for k in PDFStyleKeys # we do not store STROKE_DASH_PATTERN under its PDF key; it's in _stroke_dash_pattern if k is not PDFStyleKeys.STROKE_DASH_PATTERN ) INHERIT = ... """Singleton specifying a style parameter should be inherited from the parent context.""" # order is be important here because some of these properties are entangled, e.g. # fill_color and fill_opacity MERGE_PROPERTIES = ( "paint_rule", "allow_transparency", "auto_close", "intersection_rule", "fill_color", "fill_opacity", "stroke_color", "stroke_opacity", "blend_mode", "stroke_width", "stroke_cap_style", "stroke_join_style", "stroke_miter_limit", "stroke_dash_pattern", "stroke_dash_phase", "soft_mask", ) """An ordered collection of properties to use when merging two GraphicsStyles.""" TRANSPARENCY_KEYS = ( PDFStyleKeys.FILL_ALPHA.value, PDFStyleKeys.STROKE_ALPHA.value, PDFStyleKeys.BLEND_MODE.value, PDFStyleKeys.SOFT_MASK.value, ) """An ordered collection of attributes not to emit in no transparency mode.""" PDF_STYLE_KEYS = ( *(k.value for k in PDFStyleKeys if k is not PDFStyleKeys.STROKE_DASH_PATTERN), ) """An ordered collection of keys to directly emit when serializing the style.""" _PAINT_RULE_LOOKUP = { frozenset({}): PathPaintRule.DONT_PAINT, frozenset({"stroke"}): PathPaintRule.STROKE, frozenset({"fill", IntersectionRule.NONZERO}): PathPaintRule.FILL_NONZERO, frozenset({"fill", IntersectionRule.EVENODD}): PathPaintRule.FILL_EVENODD, frozenset( {"stroke", "fill", IntersectionRule.NONZERO} ): PathPaintRule.STROKE_FILL_NONZERO, frozenset( {"stroke", "fill", IntersectionRule.EVENODD} ): PathPaintRule.STROKE_FILL_EVENODD, } """A dictionary for resolving `PathPaintRule.AUTO`""" @classmethod def merge(cls, parent: "GraphicsStyle", child: "GraphicsStyle") -> "GraphicsStyle": """ Merge parent and child into a single GraphicsStyle. The result contains the properties of the parent as overridden by any properties explicitly set on the child. If both the parent and the child specify to inherit a given property, that property will preserve the inherit value. """ new = cls() for prop in cls.MERGE_PROPERTIES: cval = getattr(child, prop) if cval is cls.INHERIT: setattr(new, prop, getattr(parent, prop)) else: setattr(new, prop, cval) return new def __init__(self): self.allow_transparency = self.INHERIT self.paint_rule = self.INHERIT self.auto_close = self.INHERIT self.intersection_rule = self.INHERIT self.fill_color = self.INHERIT self.fill_opacity = self.INHERIT self.stroke_color = self.INHERIT self.stroke_opacity = self.INHERIT self.blend_mode = self.INHERIT self.stroke_width = self.INHERIT self.stroke_cap_style = self.INHERIT self.stroke_join_style = self.INHERIT self.stroke_miter_limit = self.INHERIT self.stroke_dash_pattern = self.INHERIT self.stroke_dash_phase = self.INHERIT self.soft_mask = self.INHERIT def __deepcopy__(self, memo): copied = self.__class__() for prop in self.MERGE_PROPERTIES: setattr(copied, prop, getattr(self, prop)) return copied def __setattr__(self, name, value): if not hasattr(self.__class__, name): raise AttributeError( f'{self.__class__} does not have style "{name}" (a typo?)' ) super().__setattr__(name, value) # at some point it probably makes sense to turn this into a general compliance # property, but for now this is the simple approach. @property def allow_transparency(self): return self._allow_transparency # pylint: disable=no-member @allow_transparency.setter def allow_transparency(self, new): return super().__setattr__("_allow_transparency", new) # If these are used in a nested graphics context inside of a painting path # operation, they are no-ops. However, they can be used for outer GraphicsContexts # that painting paths inherit from. @property def paint_rule(self): """The paint rule to use for this path/group.""" return self._paint_rule # pylint: disable=no-member @paint_rule.setter def paint_rule(self, new): if new is None: super().__setattr__("_paint_rule", PathPaintRule.DONT_PAINT) elif new is self.INHERIT: super().__setattr__("_paint_rule", new) else: super().__setattr__("_paint_rule", PathPaintRule.coerce(new)) @property def auto_close(self): """If True, unclosed paths will be automatically closed before stroking.""" return self._auto_close # pylint: disable=no-member @auto_close.setter def auto_close(self, new): if new not in {True, False, self.INHERIT}: raise TypeError(f"auto_close must be a bool or self.INHERIT, not {new}") super().__setattr__("_auto_close", new) @property def intersection_rule(self): """The desired intersection rule for this path/group.""" return self._intersection_rule # pylint: disable=no-member @intersection_rule.setter def intersection_rule(self, new): # don't allow None for this one. if new is self.INHERIT: super().__setattr__("_intersection_rule", new) else: super().__setattr__("_intersection_rule", IntersectionRule.coerce(new)) @property def fill_color(self): """ The desired fill color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._fill_color # pylint: disable=no-member @fill_color.setter def fill_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_fill_color", color) if getattr(color, "a", None) is not None: self.fill_opacity = color.a elif (color is None) or (color is self.INHERIT): super().__setattr__("_fill_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def fill_opacity(self): """The desired fill opacity for this path/group.""" return getattr(self, PDFStyleKeys.FILL_ALPHA.value) @fill_opacity.setter def fill_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.FILL_ALPHA.value, new) @property def stroke_color(self): """ The desired stroke color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._stroke_color # pylint: disable=no-member @stroke_color.setter def stroke_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_stroke_color", color) if getattr(color, "a", None) is not None: self.stroke_opacity = color.a if self.stroke_width is self.INHERIT: self.stroke_width = 1 elif (color is None) or (color is self.INHERIT): super().__setattr__("_stroke_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def stroke_opacity(self): """The desired stroke opacity for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_ALPHA.value) @stroke_opacity.setter def stroke_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.STROKE_ALPHA.value, new) @property def blend_mode(self): """The desired blend mode for this path/group.""" return getattr(self, PDFStyleKeys.BLEND_MODE.value) @blend_mode.setter def blend_mode(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.BLEND_MODE.value, value) else: super().__setattr__( PDFStyleKeys.BLEND_MODE.value, BlendMode.coerce(value).value ) @property def stroke_width(self): """The desired stroke width for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_WIDTH.value) @stroke_width.setter def stroke_width(self, width): if not isinstance( width, (int, float, decimal.Decimal, type(None), type(self.INHERIT)), ): raise TypeError(f"stroke_width must be a number, not {type(width)}") super().__setattr__(PDFStyleKeys.STROKE_WIDTH.value, width) @property def stroke_cap_style(self): """The desired stroke cap style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_CAP_STYLE.value) @stroke_cap_style.setter def stroke_cap_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_CAP_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_CAP_STYLE.value, StrokeCapStyle.coerce(value) ) @property def stroke_join_style(self): """The desired stroke join style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_JOIN_STYLE.value) @stroke_join_style.setter def stroke_join_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_JOIN_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_JOIN_STYLE.value, StrokeJoinStyle.coerce(value), ) @property def stroke_miter_limit(self): """The desired stroke miter limit for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_MITER_LIMIT.value) @stroke_miter_limit.setter def stroke_miter_limit(self, value): if (value is self.INHERIT) or isinstance(value, NumberClass): super().__setattr__(PDFStyleKeys.STROKE_MITER_LIMIT.value, value) else: raise TypeError(f"{value} is not a number") @property def stroke_dash_pattern(self): """The desired stroke dash pattern for this path/group.""" return self._stroke_dash_pattern # pylint: disable=no-member @stroke_dash_pattern.setter def stroke_dash_pattern(self, value): if value is None: result = () elif value is self.INHERIT: result = value elif isinstance(value, NumberClass): result = (value,) else: try: accum = [] for item in value: if not isinstance(item, NumberClass): raise TypeError( f"stroke_dash_pattern {value} sequence has non-numeric value" ) accum.append(item) except TypeError: raise TypeError( f"stroke_dash_pattern {value} must be a number or sequence of numbers" ) from None result = (*accum,) super().__setattr__("_stroke_dash_pattern", result) @property def stroke_dash_phase(self): """The desired stroke dash pattern phase offset for this path/group.""" return self._stroke_dash_phase # pylint: disable=no-member @stroke_dash_phase.setter def stroke_dash_phase(self, value): if value is self.INHERIT or isinstance(value, NumberClass): return super().__setattr__("_stroke_dash_phase", value) raise TypeError(f"{value} isn't a number or GraphicsStyle.INHERIT") @property def soft_mask(self): return getattr(self, PDFStyleKeys.SOFT_MASK.value) @soft_mask.setter def soft_mask(self, value): if value is self.INHERIT or isinstance(value, PaintSoftMask): return super().__setattr__(PDFStyleKeys.SOFT_MASK.value, value) raise TypeError(f"{value} isn't a PaintSoftMask or GraphicsStyle.INHERIT") def serialize(self) -> Optional[Raw]: """ Convert this style object to a PDF dictionary with appropriate style keys. Only explicitly specified values are emitted. """ result = OrderedDict() for key in self.PDF_STYLE_KEYS: value = getattr(self, key, self.INHERIT) if (value is not self.INHERIT) and (value is not None): # None is used for out-of-band signaling on these, e.g. a stroke_width # of None doesn't need to land here because it signals the # PathPaintRule auto resolution only. result[key] = value # There is additional logic in GraphicsContext to ensure that this will work if self.stroke_dash_pattern and self.stroke_dash_pattern is not self.INHERIT: result[PDFStyleKeys.STROKE_DASH_PATTERN.value] = [ self.stroke_dash_pattern, self.stroke_dash_phase, ] if self.allow_transparency is False: for key in self.TRANSPARENCY_KEYS: if key in result: del result[key] if result: # Only insert this key if there is at least one other item in the result so # that we don't junk up the output PDF with empty ExtGState dictionaries. type_name = Name("Type") result[type_name] = Name("ExtGState") result.move_to_end(type_name, last=False) return render_pdf_primitive(result) # this signals to the graphics state registry that there is nothing to # register. This is a success case. return None @force_nodocument def resolve_paint_rule(self) -> PathPaintRule: """ Resolve `PathPaintRule.AUTO` to a real paint rule based on this style. Returns: the resolved `PathPaintRule`. """ if self.paint_rule is PathPaintRule.AUTO: want = set() if self.stroke_width is not None and self.stroke_color is not None: want.add("stroke") if self.fill_color is not None: want.add("fill") # we need to guarantee that this will not be None. The default will # be "nonzero". assert self.intersection_rule is not None want.add(self.intersection_rule) try: rule = self._PAINT_RULE_LOOKUP[frozenset(want)] except KeyError: # don't default to DONT_PAINT because that's almost certainly not a very # good default. rule = PathPaintRule.STROKE_FILL_NONZERO elif self.paint_rule is self.INHERIT: # this shouldn't happen under normal usage, but certain API (ab)use can end # up in this state. We can't resolve anything meaningful, so fall back to a # sane(?) default. rule = PathPaintRule.STROKE_FILL_NONZERO else: rule = self.paint_rule return rule
var SMask
-
Expand source code Browse git
class GraphicsStyle: """ A class representing various style attributes that determine drawing appearance. This class uses the convention that the global Python singleton ellipsis (`...`) is exclusively used to represent values that are inherited from the parent style. This is to disambiguate the value None which is used for several values to signal an explicitly disabled style. An example of this is the fill/stroke color styles, which use None as hints to the auto paint style detection code. """ _PRIVATE_SLOTS = ( "_allow_transparency", "_auto_close", "_fill_color", "_intersection_rule", "_paint_rule", "_stroke_color", "_stroke_dash_pattern", "_stroke_dash_phase", ) __slots__ = _PRIVATE_SLOTS + tuple( k.value for k in PDFStyleKeys # we do not store STROKE_DASH_PATTERN under its PDF key; it's in _stroke_dash_pattern if k is not PDFStyleKeys.STROKE_DASH_PATTERN ) INHERIT = ... """Singleton specifying a style parameter should be inherited from the parent context.""" # order is be important here because some of these properties are entangled, e.g. # fill_color and fill_opacity MERGE_PROPERTIES = ( "paint_rule", "allow_transparency", "auto_close", "intersection_rule", "fill_color", "fill_opacity", "stroke_color", "stroke_opacity", "blend_mode", "stroke_width", "stroke_cap_style", "stroke_join_style", "stroke_miter_limit", "stroke_dash_pattern", "stroke_dash_phase", "soft_mask", ) """An ordered collection of properties to use when merging two GraphicsStyles.""" TRANSPARENCY_KEYS = ( PDFStyleKeys.FILL_ALPHA.value, PDFStyleKeys.STROKE_ALPHA.value, PDFStyleKeys.BLEND_MODE.value, PDFStyleKeys.SOFT_MASK.value, ) """An ordered collection of attributes not to emit in no transparency mode.""" PDF_STYLE_KEYS = ( *(k.value for k in PDFStyleKeys if k is not PDFStyleKeys.STROKE_DASH_PATTERN), ) """An ordered collection of keys to directly emit when serializing the style.""" _PAINT_RULE_LOOKUP = { frozenset({}): PathPaintRule.DONT_PAINT, frozenset({"stroke"}): PathPaintRule.STROKE, frozenset({"fill", IntersectionRule.NONZERO}): PathPaintRule.FILL_NONZERO, frozenset({"fill", IntersectionRule.EVENODD}): PathPaintRule.FILL_EVENODD, frozenset( {"stroke", "fill", IntersectionRule.NONZERO} ): PathPaintRule.STROKE_FILL_NONZERO, frozenset( {"stroke", "fill", IntersectionRule.EVENODD} ): PathPaintRule.STROKE_FILL_EVENODD, } """A dictionary for resolving `PathPaintRule.AUTO`""" @classmethod def merge(cls, parent: "GraphicsStyle", child: "GraphicsStyle") -> "GraphicsStyle": """ Merge parent and child into a single GraphicsStyle. The result contains the properties of the parent as overridden by any properties explicitly set on the child. If both the parent and the child specify to inherit a given property, that property will preserve the inherit value. """ new = cls() for prop in cls.MERGE_PROPERTIES: cval = getattr(child, prop) if cval is cls.INHERIT: setattr(new, prop, getattr(parent, prop)) else: setattr(new, prop, cval) return new def __init__(self): self.allow_transparency = self.INHERIT self.paint_rule = self.INHERIT self.auto_close = self.INHERIT self.intersection_rule = self.INHERIT self.fill_color = self.INHERIT self.fill_opacity = self.INHERIT self.stroke_color = self.INHERIT self.stroke_opacity = self.INHERIT self.blend_mode = self.INHERIT self.stroke_width = self.INHERIT self.stroke_cap_style = self.INHERIT self.stroke_join_style = self.INHERIT self.stroke_miter_limit = self.INHERIT self.stroke_dash_pattern = self.INHERIT self.stroke_dash_phase = self.INHERIT self.soft_mask = self.INHERIT def __deepcopy__(self, memo): copied = self.__class__() for prop in self.MERGE_PROPERTIES: setattr(copied, prop, getattr(self, prop)) return copied def __setattr__(self, name, value): if not hasattr(self.__class__, name): raise AttributeError( f'{self.__class__} does not have style "{name}" (a typo?)' ) super().__setattr__(name, value) # at some point it probably makes sense to turn this into a general compliance # property, but for now this is the simple approach. @property def allow_transparency(self): return self._allow_transparency # pylint: disable=no-member @allow_transparency.setter def allow_transparency(self, new): return super().__setattr__("_allow_transparency", new) # If these are used in a nested graphics context inside of a painting path # operation, they are no-ops. However, they can be used for outer GraphicsContexts # that painting paths inherit from. @property def paint_rule(self): """The paint rule to use for this path/group.""" return self._paint_rule # pylint: disable=no-member @paint_rule.setter def paint_rule(self, new): if new is None: super().__setattr__("_paint_rule", PathPaintRule.DONT_PAINT) elif new is self.INHERIT: super().__setattr__("_paint_rule", new) else: super().__setattr__("_paint_rule", PathPaintRule.coerce(new)) @property def auto_close(self): """If True, unclosed paths will be automatically closed before stroking.""" return self._auto_close # pylint: disable=no-member @auto_close.setter def auto_close(self, new): if new not in {True, False, self.INHERIT}: raise TypeError(f"auto_close must be a bool or self.INHERIT, not {new}") super().__setattr__("_auto_close", new) @property def intersection_rule(self): """The desired intersection rule for this path/group.""" return self._intersection_rule # pylint: disable=no-member @intersection_rule.setter def intersection_rule(self, new): # don't allow None for this one. if new is self.INHERIT: super().__setattr__("_intersection_rule", new) else: super().__setattr__("_intersection_rule", IntersectionRule.coerce(new)) @property def fill_color(self): """ The desired fill color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._fill_color # pylint: disable=no-member @fill_color.setter def fill_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_fill_color", color) if getattr(color, "a", None) is not None: self.fill_opacity = color.a elif (color is None) or (color is self.INHERIT): super().__setattr__("_fill_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def fill_opacity(self): """The desired fill opacity for this path/group.""" return getattr(self, PDFStyleKeys.FILL_ALPHA.value) @fill_opacity.setter def fill_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.FILL_ALPHA.value, new) @property def stroke_color(self): """ The desired stroke color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._stroke_color # pylint: disable=no-member @stroke_color.setter def stroke_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_stroke_color", color) if getattr(color, "a", None) is not None: self.stroke_opacity = color.a if self.stroke_width is self.INHERIT: self.stroke_width = 1 elif (color is None) or (color is self.INHERIT): super().__setattr__("_stroke_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def stroke_opacity(self): """The desired stroke opacity for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_ALPHA.value) @stroke_opacity.setter def stroke_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.STROKE_ALPHA.value, new) @property def blend_mode(self): """The desired blend mode for this path/group.""" return getattr(self, PDFStyleKeys.BLEND_MODE.value) @blend_mode.setter def blend_mode(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.BLEND_MODE.value, value) else: super().__setattr__( PDFStyleKeys.BLEND_MODE.value, BlendMode.coerce(value).value ) @property def stroke_width(self): """The desired stroke width for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_WIDTH.value) @stroke_width.setter def stroke_width(self, width): if not isinstance( width, (int, float, decimal.Decimal, type(None), type(self.INHERIT)), ): raise TypeError(f"stroke_width must be a number, not {type(width)}") super().__setattr__(PDFStyleKeys.STROKE_WIDTH.value, width) @property def stroke_cap_style(self): """The desired stroke cap style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_CAP_STYLE.value) @stroke_cap_style.setter def stroke_cap_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_CAP_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_CAP_STYLE.value, StrokeCapStyle.coerce(value) ) @property def stroke_join_style(self): """The desired stroke join style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_JOIN_STYLE.value) @stroke_join_style.setter def stroke_join_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_JOIN_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_JOIN_STYLE.value, StrokeJoinStyle.coerce(value), ) @property def stroke_miter_limit(self): """The desired stroke miter limit for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_MITER_LIMIT.value) @stroke_miter_limit.setter def stroke_miter_limit(self, value): if (value is self.INHERIT) or isinstance(value, NumberClass): super().__setattr__(PDFStyleKeys.STROKE_MITER_LIMIT.value, value) else: raise TypeError(f"{value} is not a number") @property def stroke_dash_pattern(self): """The desired stroke dash pattern for this path/group.""" return self._stroke_dash_pattern # pylint: disable=no-member @stroke_dash_pattern.setter def stroke_dash_pattern(self, value): if value is None: result = () elif value is self.INHERIT: result = value elif isinstance(value, NumberClass): result = (value,) else: try: accum = [] for item in value: if not isinstance(item, NumberClass): raise TypeError( f"stroke_dash_pattern {value} sequence has non-numeric value" ) accum.append(item) except TypeError: raise TypeError( f"stroke_dash_pattern {value} must be a number or sequence of numbers" ) from None result = (*accum,) super().__setattr__("_stroke_dash_pattern", result) @property def stroke_dash_phase(self): """The desired stroke dash pattern phase offset for this path/group.""" return self._stroke_dash_phase # pylint: disable=no-member @stroke_dash_phase.setter def stroke_dash_phase(self, value): if value is self.INHERIT or isinstance(value, NumberClass): return super().__setattr__("_stroke_dash_phase", value) raise TypeError(f"{value} isn't a number or GraphicsStyle.INHERIT") @property def soft_mask(self): return getattr(self, PDFStyleKeys.SOFT_MASK.value) @soft_mask.setter def soft_mask(self, value): if value is self.INHERIT or isinstance(value, PaintSoftMask): return super().__setattr__(PDFStyleKeys.SOFT_MASK.value, value) raise TypeError(f"{value} isn't a PaintSoftMask or GraphicsStyle.INHERIT") def serialize(self) -> Optional[Raw]: """ Convert this style object to a PDF dictionary with appropriate style keys. Only explicitly specified values are emitted. """ result = OrderedDict() for key in self.PDF_STYLE_KEYS: value = getattr(self, key, self.INHERIT) if (value is not self.INHERIT) and (value is not None): # None is used for out-of-band signaling on these, e.g. a stroke_width # of None doesn't need to land here because it signals the # PathPaintRule auto resolution only. result[key] = value # There is additional logic in GraphicsContext to ensure that this will work if self.stroke_dash_pattern and self.stroke_dash_pattern is not self.INHERIT: result[PDFStyleKeys.STROKE_DASH_PATTERN.value] = [ self.stroke_dash_pattern, self.stroke_dash_phase, ] if self.allow_transparency is False: for key in self.TRANSPARENCY_KEYS: if key in result: del result[key] if result: # Only insert this key if there is at least one other item in the result so # that we don't junk up the output PDF with empty ExtGState dictionaries. type_name = Name("Type") result[type_name] = Name("ExtGState") result.move_to_end(type_name, last=False) return render_pdf_primitive(result) # this signals to the graphics state registry that there is nothing to # register. This is a success case. return None @force_nodocument def resolve_paint_rule(self) -> PathPaintRule: """ Resolve `PathPaintRule.AUTO` to a real paint rule based on this style. Returns: the resolved `PathPaintRule`. """ if self.paint_rule is PathPaintRule.AUTO: want = set() if self.stroke_width is not None and self.stroke_color is not None: want.add("stroke") if self.fill_color is not None: want.add("fill") # we need to guarantee that this will not be None. The default will # be "nonzero". assert self.intersection_rule is not None want.add(self.intersection_rule) try: rule = self._PAINT_RULE_LOOKUP[frozenset(want)] except KeyError: # don't default to DONT_PAINT because that's almost certainly not a very # good default. rule = PathPaintRule.STROKE_FILL_NONZERO elif self.paint_rule is self.INHERIT: # this shouldn't happen under normal usage, but certain API (ab)use can end # up in this state. We can't resolve anything meaningful, so fall back to a # sane(?) default. rule = PathPaintRule.STROKE_FILL_NONZERO else: rule = self.paint_rule return rule
prop allow_transparency
-
Expand source code Browse git
@property def allow_transparency(self): return self._allow_transparency # pylint: disable=no-member
prop auto_close
-
Expand source code Browse git
@property def auto_close(self): """If True, unclosed paths will be automatically closed before stroking.""" return self._auto_close # pylint: disable=no-member
If True, unclosed paths will be automatically closed before stroking.
prop blend_mode
-
Expand source code Browse git
@property def blend_mode(self): """The desired blend mode for this path/group.""" return getattr(self, PDFStyleKeys.BLEND_MODE.value)
The desired blend mode for this path/group.
var ca
-
Expand source code Browse git
class GraphicsStyle: """ A class representing various style attributes that determine drawing appearance. This class uses the convention that the global Python singleton ellipsis (`...`) is exclusively used to represent values that are inherited from the parent style. This is to disambiguate the value None which is used for several values to signal an explicitly disabled style. An example of this is the fill/stroke color styles, which use None as hints to the auto paint style detection code. """ _PRIVATE_SLOTS = ( "_allow_transparency", "_auto_close", "_fill_color", "_intersection_rule", "_paint_rule", "_stroke_color", "_stroke_dash_pattern", "_stroke_dash_phase", ) __slots__ = _PRIVATE_SLOTS + tuple( k.value for k in PDFStyleKeys # we do not store STROKE_DASH_PATTERN under its PDF key; it's in _stroke_dash_pattern if k is not PDFStyleKeys.STROKE_DASH_PATTERN ) INHERIT = ... """Singleton specifying a style parameter should be inherited from the parent context.""" # order is be important here because some of these properties are entangled, e.g. # fill_color and fill_opacity MERGE_PROPERTIES = ( "paint_rule", "allow_transparency", "auto_close", "intersection_rule", "fill_color", "fill_opacity", "stroke_color", "stroke_opacity", "blend_mode", "stroke_width", "stroke_cap_style", "stroke_join_style", "stroke_miter_limit", "stroke_dash_pattern", "stroke_dash_phase", "soft_mask", ) """An ordered collection of properties to use when merging two GraphicsStyles.""" TRANSPARENCY_KEYS = ( PDFStyleKeys.FILL_ALPHA.value, PDFStyleKeys.STROKE_ALPHA.value, PDFStyleKeys.BLEND_MODE.value, PDFStyleKeys.SOFT_MASK.value, ) """An ordered collection of attributes not to emit in no transparency mode.""" PDF_STYLE_KEYS = ( *(k.value for k in PDFStyleKeys if k is not PDFStyleKeys.STROKE_DASH_PATTERN), ) """An ordered collection of keys to directly emit when serializing the style.""" _PAINT_RULE_LOOKUP = { frozenset({}): PathPaintRule.DONT_PAINT, frozenset({"stroke"}): PathPaintRule.STROKE, frozenset({"fill", IntersectionRule.NONZERO}): PathPaintRule.FILL_NONZERO, frozenset({"fill", IntersectionRule.EVENODD}): PathPaintRule.FILL_EVENODD, frozenset( {"stroke", "fill", IntersectionRule.NONZERO} ): PathPaintRule.STROKE_FILL_NONZERO, frozenset( {"stroke", "fill", IntersectionRule.EVENODD} ): PathPaintRule.STROKE_FILL_EVENODD, } """A dictionary for resolving `PathPaintRule.AUTO`""" @classmethod def merge(cls, parent: "GraphicsStyle", child: "GraphicsStyle") -> "GraphicsStyle": """ Merge parent and child into a single GraphicsStyle. The result contains the properties of the parent as overridden by any properties explicitly set on the child. If both the parent and the child specify to inherit a given property, that property will preserve the inherit value. """ new = cls() for prop in cls.MERGE_PROPERTIES: cval = getattr(child, prop) if cval is cls.INHERIT: setattr(new, prop, getattr(parent, prop)) else: setattr(new, prop, cval) return new def __init__(self): self.allow_transparency = self.INHERIT self.paint_rule = self.INHERIT self.auto_close = self.INHERIT self.intersection_rule = self.INHERIT self.fill_color = self.INHERIT self.fill_opacity = self.INHERIT self.stroke_color = self.INHERIT self.stroke_opacity = self.INHERIT self.blend_mode = self.INHERIT self.stroke_width = self.INHERIT self.stroke_cap_style = self.INHERIT self.stroke_join_style = self.INHERIT self.stroke_miter_limit = self.INHERIT self.stroke_dash_pattern = self.INHERIT self.stroke_dash_phase = self.INHERIT self.soft_mask = self.INHERIT def __deepcopy__(self, memo): copied = self.__class__() for prop in self.MERGE_PROPERTIES: setattr(copied, prop, getattr(self, prop)) return copied def __setattr__(self, name, value): if not hasattr(self.__class__, name): raise AttributeError( f'{self.__class__} does not have style "{name}" (a typo?)' ) super().__setattr__(name, value) # at some point it probably makes sense to turn this into a general compliance # property, but for now this is the simple approach. @property def allow_transparency(self): return self._allow_transparency # pylint: disable=no-member @allow_transparency.setter def allow_transparency(self, new): return super().__setattr__("_allow_transparency", new) # If these are used in a nested graphics context inside of a painting path # operation, they are no-ops. However, they can be used for outer GraphicsContexts # that painting paths inherit from. @property def paint_rule(self): """The paint rule to use for this path/group.""" return self._paint_rule # pylint: disable=no-member @paint_rule.setter def paint_rule(self, new): if new is None: super().__setattr__("_paint_rule", PathPaintRule.DONT_PAINT) elif new is self.INHERIT: super().__setattr__("_paint_rule", new) else: super().__setattr__("_paint_rule", PathPaintRule.coerce(new)) @property def auto_close(self): """If True, unclosed paths will be automatically closed before stroking.""" return self._auto_close # pylint: disable=no-member @auto_close.setter def auto_close(self, new): if new not in {True, False, self.INHERIT}: raise TypeError(f"auto_close must be a bool or self.INHERIT, not {new}") super().__setattr__("_auto_close", new) @property def intersection_rule(self): """The desired intersection rule for this path/group.""" return self._intersection_rule # pylint: disable=no-member @intersection_rule.setter def intersection_rule(self, new): # don't allow None for this one. if new is self.INHERIT: super().__setattr__("_intersection_rule", new) else: super().__setattr__("_intersection_rule", IntersectionRule.coerce(new)) @property def fill_color(self): """ The desired fill color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._fill_color # pylint: disable=no-member @fill_color.setter def fill_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_fill_color", color) if getattr(color, "a", None) is not None: self.fill_opacity = color.a elif (color is None) or (color is self.INHERIT): super().__setattr__("_fill_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def fill_opacity(self): """The desired fill opacity for this path/group.""" return getattr(self, PDFStyleKeys.FILL_ALPHA.value) @fill_opacity.setter def fill_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.FILL_ALPHA.value, new) @property def stroke_color(self): """ The desired stroke color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._stroke_color # pylint: disable=no-member @stroke_color.setter def stroke_color(self, color): if isinstance(color, str): color = color_from_hex_string(color) if isinstance(color, (DeviceRGB, DeviceGray, DeviceCMYK, GradientPaint)): super().__setattr__("_stroke_color", color) if getattr(color, "a", None) is not None: self.stroke_opacity = color.a if self.stroke_width is self.INHERIT: self.stroke_width = 1 elif (color is None) or (color is self.INHERIT): super().__setattr__("_stroke_color", color) else: raise TypeError(f"{color} doesn't look like a drawing color") @property def stroke_opacity(self): """The desired stroke opacity for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_ALPHA.value) @stroke_opacity.setter def stroke_opacity(self, new): if new not in {None, self.INHERIT}: check_range(new) super().__setattr__(PDFStyleKeys.STROKE_ALPHA.value, new) @property def blend_mode(self): """The desired blend mode for this path/group.""" return getattr(self, PDFStyleKeys.BLEND_MODE.value) @blend_mode.setter def blend_mode(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.BLEND_MODE.value, value) else: super().__setattr__( PDFStyleKeys.BLEND_MODE.value, BlendMode.coerce(value).value ) @property def stroke_width(self): """The desired stroke width for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_WIDTH.value) @stroke_width.setter def stroke_width(self, width): if not isinstance( width, (int, float, decimal.Decimal, type(None), type(self.INHERIT)), ): raise TypeError(f"stroke_width must be a number, not {type(width)}") super().__setattr__(PDFStyleKeys.STROKE_WIDTH.value, width) @property def stroke_cap_style(self): """The desired stroke cap style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_CAP_STYLE.value) @stroke_cap_style.setter def stroke_cap_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_CAP_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_CAP_STYLE.value, StrokeCapStyle.coerce(value) ) @property def stroke_join_style(self): """The desired stroke join style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_JOIN_STYLE.value) @stroke_join_style.setter def stroke_join_style(self, value): if value is self.INHERIT: super().__setattr__(PDFStyleKeys.STROKE_JOIN_STYLE.value, value) else: super().__setattr__( PDFStyleKeys.STROKE_JOIN_STYLE.value, StrokeJoinStyle.coerce(value), ) @property def stroke_miter_limit(self): """The desired stroke miter limit for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_MITER_LIMIT.value) @stroke_miter_limit.setter def stroke_miter_limit(self, value): if (value is self.INHERIT) or isinstance(value, NumberClass): super().__setattr__(PDFStyleKeys.STROKE_MITER_LIMIT.value, value) else: raise TypeError(f"{value} is not a number") @property def stroke_dash_pattern(self): """The desired stroke dash pattern for this path/group.""" return self._stroke_dash_pattern # pylint: disable=no-member @stroke_dash_pattern.setter def stroke_dash_pattern(self, value): if value is None: result = () elif value is self.INHERIT: result = value elif isinstance(value, NumberClass): result = (value,) else: try: accum = [] for item in value: if not isinstance(item, NumberClass): raise TypeError( f"stroke_dash_pattern {value} sequence has non-numeric value" ) accum.append(item) except TypeError: raise TypeError( f"stroke_dash_pattern {value} must be a number or sequence of numbers" ) from None result = (*accum,) super().__setattr__("_stroke_dash_pattern", result) @property def stroke_dash_phase(self): """The desired stroke dash pattern phase offset for this path/group.""" return self._stroke_dash_phase # pylint: disable=no-member @stroke_dash_phase.setter def stroke_dash_phase(self, value): if value is self.INHERIT or isinstance(value, NumberClass): return super().__setattr__("_stroke_dash_phase", value) raise TypeError(f"{value} isn't a number or GraphicsStyle.INHERIT") @property def soft_mask(self): return getattr(self, PDFStyleKeys.SOFT_MASK.value) @soft_mask.setter def soft_mask(self, value): if value is self.INHERIT or isinstance(value, PaintSoftMask): return super().__setattr__(PDFStyleKeys.SOFT_MASK.value, value) raise TypeError(f"{value} isn't a PaintSoftMask or GraphicsStyle.INHERIT") def serialize(self) -> Optional[Raw]: """ Convert this style object to a PDF dictionary with appropriate style keys. Only explicitly specified values are emitted. """ result = OrderedDict() for key in self.PDF_STYLE_KEYS: value = getattr(self, key, self.INHERIT) if (value is not self.INHERIT) and (value is not None): # None is used for out-of-band signaling on these, e.g. a stroke_width # of None doesn't need to land here because it signals the # PathPaintRule auto resolution only. result[key] = value # There is additional logic in GraphicsContext to ensure that this will work if self.stroke_dash_pattern and self.stroke_dash_pattern is not self.INHERIT: result[PDFStyleKeys.STROKE_DASH_PATTERN.value] = [ self.stroke_dash_pattern, self.stroke_dash_phase, ] if self.allow_transparency is False: for key in self.TRANSPARENCY_KEYS: if key in result: del result[key] if result: # Only insert this key if there is at least one other item in the result so # that we don't junk up the output PDF with empty ExtGState dictionaries. type_name = Name("Type") result[type_name] = Name("ExtGState") result.move_to_end(type_name, last=False) return render_pdf_primitive(result) # this signals to the graphics state registry that there is nothing to # register. This is a success case. return None @force_nodocument def resolve_paint_rule(self) -> PathPaintRule: """ Resolve `PathPaintRule.AUTO` to a real paint rule based on this style. Returns: the resolved `PathPaintRule`. """ if self.paint_rule is PathPaintRule.AUTO: want = set() if self.stroke_width is not None and self.stroke_color is not None: want.add("stroke") if self.fill_color is not None: want.add("fill") # we need to guarantee that this will not be None. The default will # be "nonzero". assert self.intersection_rule is not None want.add(self.intersection_rule) try: rule = self._PAINT_RULE_LOOKUP[frozenset(want)] except KeyError: # don't default to DONT_PAINT because that's almost certainly not a very # good default. rule = PathPaintRule.STROKE_FILL_NONZERO elif self.paint_rule is self.INHERIT: # this shouldn't happen under normal usage, but certain API (ab)use can end # up in this state. We can't resolve anything meaningful, so fall back to a # sane(?) default. rule = PathPaintRule.STROKE_FILL_NONZERO else: rule = self.paint_rule return rule
prop fill_color
-
Expand source code Browse git
@property def fill_color(self): """ The desired fill color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._fill_color # pylint: disable=no-member
The desired fill color for this path/group.
When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well.
prop fill_opacity
-
Expand source code Browse git
@property def fill_opacity(self): """The desired fill opacity for this path/group.""" return getattr(self, PDFStyleKeys.FILL_ALPHA.value)
The desired fill opacity for this path/group.
prop intersection_rule
-
Expand source code Browse git
@property def intersection_rule(self): """The desired intersection rule for this path/group.""" return self._intersection_rule # pylint: disable=no-member
The desired intersection rule for this path/group.
prop paint_rule
-
Expand source code Browse git
@property def paint_rule(self): """The paint rule to use for this path/group.""" return self._paint_rule # pylint: disable=no-member
The paint rule to use for this path/group.
prop soft_mask
-
Expand source code Browse git
@property def soft_mask(self): return getattr(self, PDFStyleKeys.SOFT_MASK.value)
prop stroke_cap_style
-
Expand source code Browse git
@property def stroke_cap_style(self): """The desired stroke cap style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_CAP_STYLE.value)
The desired stroke cap style for this path/group.
prop stroke_color
-
Expand source code Browse git
@property def stroke_color(self): """ The desired stroke color for this path/group. When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well. """ return self._stroke_color # pylint: disable=no-member
The desired stroke color for this path/group.
When setting this property, if the color specifies an opacity value, that will be used to set the fill_opacity property as well.
prop stroke_dash_pattern
-
Expand source code Browse git
@property def stroke_dash_pattern(self): """The desired stroke dash pattern for this path/group.""" return self._stroke_dash_pattern # pylint: disable=no-member
The desired stroke dash pattern for this path/group.
prop stroke_dash_phase
-
Expand source code Browse git
@property def stroke_dash_phase(self): """The desired stroke dash pattern phase offset for this path/group.""" return self._stroke_dash_phase # pylint: disable=no-member
The desired stroke dash pattern phase offset for this path/group.
prop stroke_join_style
-
Expand source code Browse git
@property def stroke_join_style(self): """The desired stroke join style for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_JOIN_STYLE.value)
The desired stroke join style for this path/group.
prop stroke_miter_limit
-
Expand source code Browse git
@property def stroke_miter_limit(self): """The desired stroke miter limit for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_MITER_LIMIT.value)
The desired stroke miter limit for this path/group.
prop stroke_opacity
-
Expand source code Browse git
@property def stroke_opacity(self): """The desired stroke opacity for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_ALPHA.value)
The desired stroke opacity for this path/group.
prop stroke_width
-
Expand source code Browse git
@property def stroke_width(self): """The desired stroke width for this path/group.""" return getattr(self, PDFStyleKeys.STROKE_WIDTH.value)
The desired stroke width for this path/group.
Methods
def resolve_paint_rule(self) ‑> PathPaintRule
-
Expand source code Browse git
@force_nodocument def resolve_paint_rule(self) -> PathPaintRule: """ Resolve `PathPaintRule.AUTO` to a real paint rule based on this style. Returns: the resolved `PathPaintRule`. """ if self.paint_rule is PathPaintRule.AUTO: want = set() if self.stroke_width is not None and self.stroke_color is not None: want.add("stroke") if self.fill_color is not None: want.add("fill") # we need to guarantee that this will not be None. The default will # be "nonzero". assert self.intersection_rule is not None want.add(self.intersection_rule) try: rule = self._PAINT_RULE_LOOKUP[frozenset(want)] except KeyError: # don't default to DONT_PAINT because that's almost certainly not a very # good default. rule = PathPaintRule.STROKE_FILL_NONZERO elif self.paint_rule is self.INHERIT: # this shouldn't happen under normal usage, but certain API (ab)use can end # up in this state. We can't resolve anything meaningful, so fall back to a # sane(?) default. rule = PathPaintRule.STROKE_FILL_NONZERO else: rule = self.paint_rule return rule
Resolve
PathPaintRule.AUTO
to a real paint rule based on this style.Returns
the resolved
PathPaintRule
. def serialize(self) ‑> Raw | None
-
Expand source code Browse git
def serialize(self) -> Optional[Raw]: """ Convert this style object to a PDF dictionary with appropriate style keys. Only explicitly specified values are emitted. """ result = OrderedDict() for key in self.PDF_STYLE_KEYS: value = getattr(self, key, self.INHERIT) if (value is not self.INHERIT) and (value is not None): # None is used for out-of-band signaling on these, e.g. a stroke_width # of None doesn't need to land here because it signals the # PathPaintRule auto resolution only. result[key] = value # There is additional logic in GraphicsContext to ensure that this will work if self.stroke_dash_pattern and self.stroke_dash_pattern is not self.INHERIT: result[PDFStyleKeys.STROKE_DASH_PATTERN.value] = [ self.stroke_dash_pattern, self.stroke_dash_phase, ] if self.allow_transparency is False: for key in self.TRANSPARENCY_KEYS: if key in result: del result[key] if result: # Only insert this key if there is at least one other item in the result so # that we don't junk up the output PDF with empty ExtGState dictionaries. type_name = Name("Type") result[type_name] = Name("ExtGState") result.move_to_end(type_name, last=False) return render_pdf_primitive(result) # this signals to the graphics state registry that there is nothing to # register. This is a success case. return None
Convert this style object to a PDF dictionary with appropriate style keys.
Only explicitly specified values are emitted.
class HorizontalLine (x: int | float | decimal.Decimal)
-
Expand source code Browse git
class HorizontalLine(NamedTuple): """ A path line element that takes its ordinate from the end of the previous element. See: `PaintedPath.horizontal_line_to` """ x: Number """The abscissa of the horizontal line's end point.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of a horizontal line from the start point to the new x.""" end = Point(float(self.x), start.y) return BoundingBox.from_points([start, end]), end @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `Line`. """ # pylint: disable=unused-argument end_point = Point(x=self.x, y=last_item.end_point.y) return _render_line(end_point), Line(end_point), initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `HorizontalLine.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
A path line element that takes its ordinate from the end of the previous element.
Ancestors
- builtins.tuple
Instance variables
var x : int | float | decimal.Decimal
-
Expand source code Browse git
class HorizontalLine(NamedTuple): """ A path line element that takes its ordinate from the end of the previous element. See: `PaintedPath.horizontal_line_to` """ x: Number """The abscissa of the horizontal line's end point.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of a horizontal line from the start point to the new x.""" end = Point(float(self.x), start.y) return BoundingBox.from_points([start, end]), end @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `Line`. """ # pylint: disable=unused-argument end_point = Point(x=self.x, y=last_item.end_point.y) return _render_line(end_point), Line(end_point), initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `HorizontalLine.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
The abscissa of the horizontal line's end point.
Methods
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of a horizontal line from the start point to the new x.""" end = Point(float(self.x), start.y) return BoundingBox.from_points([start, end]), end
Compute the bounding box of a horizontal line from the start point to the new x.
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
@force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `Line`. """ # pylint: disable=unused-argument end_point = Point(x=self.x, y=last_item.end_point.y) return _render_line(end_point), Line(end_point), initial_point
Render this path element to its PDF representation.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
Returns
a tuple of
(str, new_last_item)
, wherenew_last_item
is the resolvedLine
. def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
@force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `HorizontalLine.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
HorizontalLine.render()
.
class ImplicitClose
-
Expand source code Browse git
class ImplicitClose(NamedTuple): """ A path close element that is conditionally rendered depending on the value of `GraphicsStyle.auto_close`. """ # pylint: disable=no-self-use def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Return an empty bounding box; Close does not affect the geometry.""" return BoundingBox.empty(), start # pylint: disable=no-self-use @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is whatever the old last_item was. """ # pylint: disable=unused-argument if style.auto_close: return "h", last_item, initial_point return "", last_item, initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `ImplicitClose.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {rendered}\n") return rendered, resolved, initial_point
A path close element that is conditionally rendered depending on the value of
GraphicsStyle.auto_close
.Ancestors
- builtins.tuple
Methods
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Return an empty bounding box; Close does not affect the geometry.""" return BoundingBox.empty(), start
Return an empty bounding box; Close does not affect the geometry.
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
@force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is whatever the old last_item was. """ # pylint: disable=unused-argument if style.auto_close: return "h", last_item, initial_point return "", last_item, initial_point
Render this path element to its PDF representation.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
Returns
a tuple of
(str, new_last_item)
, wherenew_last_item
is whatever the old last_item was. def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
@force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `ImplicitClose.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {rendered}\n") return rendered, resolved, initial_point
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
ImplicitClose.render()
.
class Line (pt: Point)
-
Expand source code Browse git
class Line(NamedTuple): """ A path line element. This draws a straight line from the end point of the previous path element to the point specified by `pt`. See: `PaintedPath.line_to` """ pt: Point """The point to which the line is drawn.""" @property def end_point(self) -> Point: """The end point of this path element.""" return self.pt def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of a line from the start point to the end point.""" return BoundingBox.from_points([start, self.pt]), self.pt @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is `self` """ # pylint: disable=unused-argument return _render_line(self.pt), self, initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Line.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(str(self) + "\n") return rendered, resolved, initial_point
A path line element.
This draws a straight line from the end point of the previous path element to the point specified by
pt
.Ancestors
- builtins.tuple
Instance variables
prop end_point : Point
-
Expand source code Browse git
@property def end_point(self) -> Point: """The end point of this path element.""" return self.pt
The end point of this path element.
var pt : Point
-
Expand source code Browse git
class Line(NamedTuple): """ A path line element. This draws a straight line from the end point of the previous path element to the point specified by `pt`. See: `PaintedPath.line_to` """ pt: Point """The point to which the line is drawn.""" @property def end_point(self) -> Point: """The end point of this path element.""" return self.pt def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of a line from the start point to the end point.""" return BoundingBox.from_points([start, self.pt]), self.pt @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is `self` """ # pylint: disable=unused-argument return _render_line(self.pt), self, initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Line.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(str(self) + "\n") return rendered, resolved, initial_point
The point to which the line is drawn.
Methods
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of a line from the start point to the end point.""" return BoundingBox.from_points([start, self.pt]), self.pt
Compute the bounding box of a line from the start point to the end point.
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
@force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is `self` """ # pylint: disable=unused-argument return _render_line(self.pt), self, initial_point
Render this path element to its PDF representation.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
Returns
a tuple of
(str, new_last_item)
, wherenew_last_item
isself
def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
@force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Line.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(str(self) + "\n") return rendered, resolved, initial_point
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
Line.render()
.
class Move (pt: Point)
-
Expand source code Browse git
class Move(NamedTuple): """ A path move element. If a path has been created but not yet painted, this will create a new subpath. See: `PaintedPath.move_to` """ pt: Point """The point to which to move.""" @property def end_point(self) -> Point: """The end point of this path element.""" return self.pt # pylint: disable=unused-argument def bounding_box(self, start) -> tuple[BoundingBox, Point]: bbox = BoundingBox.empty() return bbox, self.pt @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is `self` """ return _render_move(self.pt), self, self.pt # pylint: disable=unused-argument @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Move.render`. """ rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(str(self) + "\n") return rendered, resolved, initial_point
A path move element.
If a path has been created but not yet painted, this will create a new subpath.
Ancestors
- builtins.tuple
Instance variables
prop end_point : Point
-
Expand source code Browse git
@property def end_point(self) -> Point: """The end point of this path element.""" return self.pt
The end point of this path element.
var pt : Point
-
Expand source code Browse git
class Move(NamedTuple): """ A path move element. If a path has been created but not yet painted, this will create a new subpath. See: `PaintedPath.move_to` """ pt: Point """The point to which to move.""" @property def end_point(self) -> Point: """The end point of this path element.""" return self.pt # pylint: disable=unused-argument def bounding_box(self, start) -> tuple[BoundingBox, Point]: bbox = BoundingBox.empty() return bbox, self.pt @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is `self` """ return _render_move(self.pt), self, self.pt # pylint: disable=unused-argument @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Move.render`. """ rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(str(self) + "\n") return rendered, resolved, initial_point
The point to which to move.
Methods
def bounding_box(self, start) ‑> tuple[BoundingBox, Point]
-
Expand source code Browse git
def bounding_box(self, start) -> tuple[BoundingBox, Point]: bbox = BoundingBox.empty() return bbox, self.pt
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
@force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is `self` """ return _render_move(self.pt), self, self.pt
Render this path element to its PDF representation.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
Returns
a tuple of
(str, new_last_item)
, wherenew_last_item
isself
def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
@force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Move.render`. """ rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(str(self) + "\n") return rendered, resolved, initial_point
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
Move.render()
.
class PaintComposite (backdrop,
source,
operation: CompositingOperation)-
Expand source code Browse git
class PaintComposite: @dataclass(frozen=True) class _Step: draw: str # "source" or "backdrop" mask_from: Optional[str] # "source" | "backdrop" | None invert: bool = False _MODES = { CompositingOperation.SOURCE_OVER: ( _Step("backdrop", None), _Step("source", None), ), CompositingOperation.DESTINATION_OVER: ( _Step("source", None), _Step("backdrop", None), ), CompositingOperation.SOURCE_IN: (_Step("source", "backdrop"),), CompositingOperation.DESTINATION_IN: (_Step("backdrop", "source"),), CompositingOperation.SOURCE_OUT: (_Step("source", "backdrop", True),), CompositingOperation.DESTINATION_OUT: (_Step("backdrop", "source", True),), CompositingOperation.SOURCE_ATOP: ( _Step("backdrop", "source", True), _Step("source", "backdrop"), ), CompositingOperation.DESTINATION_ATOP: ( _Step("source", "backdrop", True), _Step("backdrop", "source"), ), CompositingOperation.XOR: ( _Step("source", "backdrop", True), _Step("backdrop", "source", True), ), CompositingOperation.CLEAR: tuple(), } def __init__(self, backdrop, source, operation: CompositingOperation): if not isinstance(backdrop, PaintedPath) or not isinstance(source, PaintedPath): raise TypeError("PaintComposite requires two PaintedPath instances.") self.backdrop = backdrop self.source = source self.mode = operation if self.mode not in self._MODES: raise NotImplementedError( f"Compositing mode '{self.mode.value}' is not yet supported." ) @classmethod def _with_mask( cls, path: PaintedPath, mask_from: PaintedPath, invert: bool ) -> PaintedPath: p = deepcopy(path) p.style.soft_mask = PaintSoftMask(mask_from, invert=invert) return p def _pick(self, which: str) -> PaintedPath: return self.source if which == "source" else self.backdrop def render( self, resource_registry, style, last_item, initial_point, debug_stream=None, pfx=None, ): steps = self._MODES[self.mode] if not steps: # CLEAR return "", last_item, initial_point parts = [] for st in steps: path = self._pick(st.draw) if st.mask_from is not None: path = self._with_mask(path, self._pick(st.mask_from), st.invert) s, last_item, initial_point = path.render( resource_registry, style, last_item, initial_point, debug_stream, pfx ) parts.append(s) return " ".join(parts), last_item, initial_point def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): debug_stream.write(f"{pfx}<PaintComposite mode={self.mode}>\n") return self.render( resource_registry, style, last_item, initial_point, debug_stream, pfx )
Methods
def render(self, resource_registry, style, last_item, initial_point, debug_stream=None, pfx=None)
-
Expand source code Browse git
def render( self, resource_registry, style, last_item, initial_point, debug_stream=None, pfx=None, ): steps = self._MODES[self.mode] if not steps: # CLEAR return "", last_item, initial_point parts = [] for st in steps: path = self._pick(st.draw) if st.mask_from is not None: path = self._with_mask(path, self._pick(st.mask_from), st.invert) s, last_item, initial_point = path.render( resource_registry, style, last_item, initial_point, debug_stream, pfx ) parts.append(s) return " ".join(parts), last_item, initial_point
def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): debug_stream.write(f"{pfx}<PaintComposite mode={self.mode}>\n") return self.render( resource_registry, style, last_item, initial_point, debug_stream, pfx )
class PaintSoftMask (mask_path: PaintedPath,
invert: bool = False)-
Expand source code Browse git
class PaintSoftMask: """ Wraps a vector path as a PDF soft mask (SMask) that can be attached to a graphics state. The provided `mask_path` is deep-copied and forced to render as an opaque grayscale fill (white, alpha=1, nonzero rule, transparency disabled). During rendering, the mask’s content stream is generated and its resource dictionary is collected so it can be embedded as a Form XObject and referenced from an ExtGState. Notes: - This implementation is currently hard-coded to use **/S /Alpha** (alpha soft mask). **/S /Luminosity** is not implemented; support can be added in the future by switching the `/S` key and adjusting how the mask content is prepared. - The mask’s style is intentionally overridden to a solid white fill with full opacity so the path shape itself defines coverage. """ __slots__ = ("mask_path", "invert", "resources", "object_id") def __init__( self, mask_path: PaintedPath, invert: bool = False, ): self.mask_path = deepcopy(mask_path) # Force opaque grayscale style self.mask_path.style.paint_rule = PathPaintRule.FILL_NONZERO self.mask_path.style.fill_opacity = 1 self.mask_path.style.fill_color = "#ffffff" self.mask_path.style.allow_transparency = False self.invert = invert self.resources = [] self.object_id = 0 def serialize(self): tr = ( " /TR <</FunctionType 2 /Domain [0 1] /Range [0 1] /C0 [1] /C1 [0] /N 1>>" if self.invert else "" ) return f"<</S /Alpha /G {self.object_id} 0 R{tr}>>" def get_bounding_box(self) -> tuple[float, float, float, float]: bounding_box, _ = self.mask_path.bounding_box(Point(0, 0)) return bounding_box.to_tuple() def get_resource_dictionary(self, gfxstate_objs_per_name): """Get the resource dictionary for this soft mask.""" resource_dict = {} for resource_type, resource_id in self.resources: if resource_type.value not in resource_dict: resource_dict[resource_type.value] = {} resource_dict[resource_type.value][resource_id] = f"{resource_id} 0 R" ret = "" for key, resources in resource_dict.items(): ret += ( Name(key).serialize() + "<<" + "".join( f"{Name(resource_id).serialize()} {gfxstate_objs_per_name[resource_id]} 0 R" for resource_id in resources ) + ">>" ) return "<<" + ret + ">>" def render(self, resource_registry): stream, _, _ = self.mask_path.render( resource_registry, style=self.mask_path.style, last_item=None, initial_point=Point(0, 0), ) self.resources = resource_registry.scan_stream(stream) return stream
Wraps a vector path as a PDF soft mask (SMask) that can be attached to a graphics state.
The provided
mask_path
is deep-copied and forced to render as an opaque grayscale fill (white, alpha=1, nonzero rule, transparency disabled). During rendering, the mask’s content stream is generated and its resource dictionary is collected so it can be embedded as a Form XObject and referenced from an ExtGState.Notes: - This implementation is currently hard-coded to use /S /Alpha (alpha soft mask). /S /Luminosity is not implemented; support can be added in the future by switching the
/S
key and adjusting how the mask content is prepared. - The mask’s style is intentionally overridden to a solid white fill with full opacity so the path shape itself defines coverage.Instance variables
var invert
-
Expand source code Browse git
class PaintSoftMask: """ Wraps a vector path as a PDF soft mask (SMask) that can be attached to a graphics state. The provided `mask_path` is deep-copied and forced to render as an opaque grayscale fill (white, alpha=1, nonzero rule, transparency disabled). During rendering, the mask’s content stream is generated and its resource dictionary is collected so it can be embedded as a Form XObject and referenced from an ExtGState. Notes: - This implementation is currently hard-coded to use **/S /Alpha** (alpha soft mask). **/S /Luminosity** is not implemented; support can be added in the future by switching the `/S` key and adjusting how the mask content is prepared. - The mask’s style is intentionally overridden to a solid white fill with full opacity so the path shape itself defines coverage. """ __slots__ = ("mask_path", "invert", "resources", "object_id") def __init__( self, mask_path: PaintedPath, invert: bool = False, ): self.mask_path = deepcopy(mask_path) # Force opaque grayscale style self.mask_path.style.paint_rule = PathPaintRule.FILL_NONZERO self.mask_path.style.fill_opacity = 1 self.mask_path.style.fill_color = "#ffffff" self.mask_path.style.allow_transparency = False self.invert = invert self.resources = [] self.object_id = 0 def serialize(self): tr = ( " /TR <</FunctionType 2 /Domain [0 1] /Range [0 1] /C0 [1] /C1 [0] /N 1>>" if self.invert else "" ) return f"<</S /Alpha /G {self.object_id} 0 R{tr}>>" def get_bounding_box(self) -> tuple[float, float, float, float]: bounding_box, _ = self.mask_path.bounding_box(Point(0, 0)) return bounding_box.to_tuple() def get_resource_dictionary(self, gfxstate_objs_per_name): """Get the resource dictionary for this soft mask.""" resource_dict = {} for resource_type, resource_id in self.resources: if resource_type.value not in resource_dict: resource_dict[resource_type.value] = {} resource_dict[resource_type.value][resource_id] = f"{resource_id} 0 R" ret = "" for key, resources in resource_dict.items(): ret += ( Name(key).serialize() + "<<" + "".join( f"{Name(resource_id).serialize()} {gfxstate_objs_per_name[resource_id]} 0 R" for resource_id in resources ) + ">>" ) return "<<" + ret + ">>" def render(self, resource_registry): stream, _, _ = self.mask_path.render( resource_registry, style=self.mask_path.style, last_item=None, initial_point=Point(0, 0), ) self.resources = resource_registry.scan_stream(stream) return stream
var mask_path
-
Expand source code Browse git
class PaintSoftMask: """ Wraps a vector path as a PDF soft mask (SMask) that can be attached to a graphics state. The provided `mask_path` is deep-copied and forced to render as an opaque grayscale fill (white, alpha=1, nonzero rule, transparency disabled). During rendering, the mask’s content stream is generated and its resource dictionary is collected so it can be embedded as a Form XObject and referenced from an ExtGState. Notes: - This implementation is currently hard-coded to use **/S /Alpha** (alpha soft mask). **/S /Luminosity** is not implemented; support can be added in the future by switching the `/S` key and adjusting how the mask content is prepared. - The mask’s style is intentionally overridden to a solid white fill with full opacity so the path shape itself defines coverage. """ __slots__ = ("mask_path", "invert", "resources", "object_id") def __init__( self, mask_path: PaintedPath, invert: bool = False, ): self.mask_path = deepcopy(mask_path) # Force opaque grayscale style self.mask_path.style.paint_rule = PathPaintRule.FILL_NONZERO self.mask_path.style.fill_opacity = 1 self.mask_path.style.fill_color = "#ffffff" self.mask_path.style.allow_transparency = False self.invert = invert self.resources = [] self.object_id = 0 def serialize(self): tr = ( " /TR <</FunctionType 2 /Domain [0 1] /Range [0 1] /C0 [1] /C1 [0] /N 1>>" if self.invert else "" ) return f"<</S /Alpha /G {self.object_id} 0 R{tr}>>" def get_bounding_box(self) -> tuple[float, float, float, float]: bounding_box, _ = self.mask_path.bounding_box(Point(0, 0)) return bounding_box.to_tuple() def get_resource_dictionary(self, gfxstate_objs_per_name): """Get the resource dictionary for this soft mask.""" resource_dict = {} for resource_type, resource_id in self.resources: if resource_type.value not in resource_dict: resource_dict[resource_type.value] = {} resource_dict[resource_type.value][resource_id] = f"{resource_id} 0 R" ret = "" for key, resources in resource_dict.items(): ret += ( Name(key).serialize() + "<<" + "".join( f"{Name(resource_id).serialize()} {gfxstate_objs_per_name[resource_id]} 0 R" for resource_id in resources ) + ">>" ) return "<<" + ret + ">>" def render(self, resource_registry): stream, _, _ = self.mask_path.render( resource_registry, style=self.mask_path.style, last_item=None, initial_point=Point(0, 0), ) self.resources = resource_registry.scan_stream(stream) return stream
var object_id
-
Expand source code Browse git
class PaintSoftMask: """ Wraps a vector path as a PDF soft mask (SMask) that can be attached to a graphics state. The provided `mask_path` is deep-copied and forced to render as an opaque grayscale fill (white, alpha=1, nonzero rule, transparency disabled). During rendering, the mask’s content stream is generated and its resource dictionary is collected so it can be embedded as a Form XObject and referenced from an ExtGState. Notes: - This implementation is currently hard-coded to use **/S /Alpha** (alpha soft mask). **/S /Luminosity** is not implemented; support can be added in the future by switching the `/S` key and adjusting how the mask content is prepared. - The mask’s style is intentionally overridden to a solid white fill with full opacity so the path shape itself defines coverage. """ __slots__ = ("mask_path", "invert", "resources", "object_id") def __init__( self, mask_path: PaintedPath, invert: bool = False, ): self.mask_path = deepcopy(mask_path) # Force opaque grayscale style self.mask_path.style.paint_rule = PathPaintRule.FILL_NONZERO self.mask_path.style.fill_opacity = 1 self.mask_path.style.fill_color = "#ffffff" self.mask_path.style.allow_transparency = False self.invert = invert self.resources = [] self.object_id = 0 def serialize(self): tr = ( " /TR <</FunctionType 2 /Domain [0 1] /Range [0 1] /C0 [1] /C1 [0] /N 1>>" if self.invert else "" ) return f"<</S /Alpha /G {self.object_id} 0 R{tr}>>" def get_bounding_box(self) -> tuple[float, float, float, float]: bounding_box, _ = self.mask_path.bounding_box(Point(0, 0)) return bounding_box.to_tuple() def get_resource_dictionary(self, gfxstate_objs_per_name): """Get the resource dictionary for this soft mask.""" resource_dict = {} for resource_type, resource_id in self.resources: if resource_type.value not in resource_dict: resource_dict[resource_type.value] = {} resource_dict[resource_type.value][resource_id] = f"{resource_id} 0 R" ret = "" for key, resources in resource_dict.items(): ret += ( Name(key).serialize() + "<<" + "".join( f"{Name(resource_id).serialize()} {gfxstate_objs_per_name[resource_id]} 0 R" for resource_id in resources ) + ">>" ) return "<<" + ret + ">>" def render(self, resource_registry): stream, _, _ = self.mask_path.render( resource_registry, style=self.mask_path.style, last_item=None, initial_point=Point(0, 0), ) self.resources = resource_registry.scan_stream(stream) return stream
var resources
-
Expand source code Browse git
class PaintSoftMask: """ Wraps a vector path as a PDF soft mask (SMask) that can be attached to a graphics state. The provided `mask_path` is deep-copied and forced to render as an opaque grayscale fill (white, alpha=1, nonzero rule, transparency disabled). During rendering, the mask’s content stream is generated and its resource dictionary is collected so it can be embedded as a Form XObject and referenced from an ExtGState. Notes: - This implementation is currently hard-coded to use **/S /Alpha** (alpha soft mask). **/S /Luminosity** is not implemented; support can be added in the future by switching the `/S` key and adjusting how the mask content is prepared. - The mask’s style is intentionally overridden to a solid white fill with full opacity so the path shape itself defines coverage. """ __slots__ = ("mask_path", "invert", "resources", "object_id") def __init__( self, mask_path: PaintedPath, invert: bool = False, ): self.mask_path = deepcopy(mask_path) # Force opaque grayscale style self.mask_path.style.paint_rule = PathPaintRule.FILL_NONZERO self.mask_path.style.fill_opacity = 1 self.mask_path.style.fill_color = "#ffffff" self.mask_path.style.allow_transparency = False self.invert = invert self.resources = [] self.object_id = 0 def serialize(self): tr = ( " /TR <</FunctionType 2 /Domain [0 1] /Range [0 1] /C0 [1] /C1 [0] /N 1>>" if self.invert else "" ) return f"<</S /Alpha /G {self.object_id} 0 R{tr}>>" def get_bounding_box(self) -> tuple[float, float, float, float]: bounding_box, _ = self.mask_path.bounding_box(Point(0, 0)) return bounding_box.to_tuple() def get_resource_dictionary(self, gfxstate_objs_per_name): """Get the resource dictionary for this soft mask.""" resource_dict = {} for resource_type, resource_id in self.resources: if resource_type.value not in resource_dict: resource_dict[resource_type.value] = {} resource_dict[resource_type.value][resource_id] = f"{resource_id} 0 R" ret = "" for key, resources in resource_dict.items(): ret += ( Name(key).serialize() + "<<" + "".join( f"{Name(resource_id).serialize()} {gfxstate_objs_per_name[resource_id]} 0 R" for resource_id in resources ) + ">>" ) return "<<" + ret + ">>" def render(self, resource_registry): stream, _, _ = self.mask_path.render( resource_registry, style=self.mask_path.style, last_item=None, initial_point=Point(0, 0), ) self.resources = resource_registry.scan_stream(stream) return stream
Methods
def get_bounding_box(self) ‑> tuple[float, float, float, float]
-
Expand source code Browse git
def get_bounding_box(self) -> tuple[float, float, float, float]: bounding_box, _ = self.mask_path.bounding_box(Point(0, 0)) return bounding_box.to_tuple()
def get_resource_dictionary(self, gfxstate_objs_per_name)
-
Expand source code Browse git
def get_resource_dictionary(self, gfxstate_objs_per_name): """Get the resource dictionary for this soft mask.""" resource_dict = {} for resource_type, resource_id in self.resources: if resource_type.value not in resource_dict: resource_dict[resource_type.value] = {} resource_dict[resource_type.value][resource_id] = f"{resource_id} 0 R" ret = "" for key, resources in resource_dict.items(): ret += ( Name(key).serialize() + "<<" + "".join( f"{Name(resource_id).serialize()} {gfxstate_objs_per_name[resource_id]} 0 R" for resource_id in resources ) + ">>" ) return "<<" + ret + ">>"
Get the resource dictionary for this soft mask.
def render(self, resource_registry)
-
Expand source code Browse git
def render(self, resource_registry): stream, _, _ = self.mask_path.render( resource_registry, style=self.mask_path.style, last_item=None, initial_point=Point(0, 0), ) self.resources = resource_registry.scan_stream(stream) return stream
def serialize(self)
-
Expand source code Browse git
def serialize(self): tr = ( " /TR <</FunctionType 2 /Domain [0 1] /Range [0 1] /C0 [1] /C1 [0] /N 1>>" if self.invert else "" ) return f"<</S /Alpha /G {self.object_id} 0 R{tr}>>"
class PaintedPath (x: float = 0, y: float = 0)
-
Expand source code Browse git
class PaintedPath: """ A path to be drawn by the PDF renderer. A painted path is defined by a style and an arbitrary sequence of path elements, which include the primitive path elements (`Move`, `Line`, `BezierCurve`, ...) as well as arbitrarily nested `GraphicsContext` containing their own sequence of primitive path elements and `GraphicsContext`. """ __slots__ = ( "_root_graphics_context", "_graphics_context", "_closed", "_close_context", "_starter_move", ) def __init__(self, x: float = 0, y: float = 0) -> None: self._root_graphics_context: GraphicsContext = GraphicsContext() self._graphics_context: GraphicsContext = self._root_graphics_context self._closed: bool = True self._close_context: GraphicsContext = self._graphics_context self._starter_move: Renderable = Move(Point(x, y)) def __deepcopy__(self, memo): # there's no real way to recover the matching current _graphics_context after # copying the root context, but that's ok because we can just disallow copying # of paths under modification as that is almost certainly wrong usage. if self._graphics_context is not self._root_graphics_context: raise RuntimeError(f"cannot copy path {self} while it is being modified") copied = self.__class__() copied._root_graphics_context = deepcopy(self._root_graphics_context, memo) copied._graphics_context = copied._root_graphics_context copied._closed = self._closed copied._close_context = copied._graphics_context return copied @property def style(self) -> GraphicsStyle: """The `GraphicsStyle` applied to all elements of this path.""" return self._root_graphics_context.style @property def transform(self) -> Optional[Transform]: """The `Transform` that applies to all of the elements of this path.""" return self._root_graphics_context.transform @transform.setter def transform(self, tf: Transform) -> None: self._root_graphics_context.transform = tf @property def auto_close(self) -> bool: """If true, the path should automatically close itself before painting.""" return self.style.auto_close @auto_close.setter def auto_close(self, should: bool) -> None: self.style.auto_close = should @property def paint_rule(self) -> PathPaintRule: """Manually specify the `PathPaintRule` to use for rendering the path.""" return self.style.paint_rule @paint_rule.setter def paint_rule(self, style: PathPaintRule) -> None: self.style.paint_rule = style @property def clipping_path(self): """Set the clipping path for this path.""" return self._root_graphics_context.clipping_path @clipping_path.setter def clipping_path(self, new_clipath): self._root_graphics_context.clipping_path = new_clipath @contextmanager def _new_graphics_context(self, _attach=True): old_graphics_context = self._graphics_context new_graphics_context = GraphicsContext() self._graphics_context = new_graphics_context try: yield new_graphics_context if _attach: old_graphics_context.add_item(new_graphics_context) finally: self._graphics_context = old_graphics_context @contextmanager def transform_group(self, transform): """ Apply the provided `Transform` to all points added within this context. """ with self._new_graphics_context() as ctxt: ctxt.transform = transform yield self def add_path_element(self, item, _copy=True): """ Add the given element as a path item of this path. Args: item: the item to add to this path. _copy (bool): if true (the default), the item will be copied before being appended. This prevents modifications to a referenced object from "retroactively" altering its style/shape and should be disabled with caution. """ if self._starter_move is not None: self._closed = False self._graphics_context.add_item(self._starter_move, _copy=False) self._close_context = self._graphics_context self._starter_move = None self._graphics_context.add_item(item, _copy=_copy) def remove_last_path_element(self): self._graphics_context.remove_last_item() def rectangle(self, x, y, w, h, rx=0, ry=0): """ Append a rectangle as a closed subpath to the current path. If the width or the height are 0, the rectangle will be collapsed to a line (unless they're both 0, in which case it's collapsed to nothing). Args: x (Number): the abscissa of the starting corner of the rectangle. y (Number): the ordinate of the starting corner of the rectangle. w (Number): the width of the rectangle (if 0, the rectangle will be rendered as a vertical line). h (Number): the height of the rectangle (if 0, the rectangle will be rendered as a horizontal line). rx (Number): the x-radius of the rectangle rounded corner (if 0 the corners will not be rounded). ry (Number): the y-radius of the rectangle rounded corner (if 0 the corners will not be rounded). Returns: The path, to allow chaining method calls. """ self._insert_implicit_close_if_open() self.add_path_element( RoundedRectangle(Point(x, y), Point(w, h), Point(rx, ry)), _copy=False ) self._closed = True self.move_to(x, y) return self def circle(self, cx, cy, r): """ Append a circle as a closed subpath to the current path. Args: cx (Number): the abscissa of the circle's center point. cy (Number): the ordinate of the circle's center point. r (Number): the radius of the circle. Returns: The path, to allow chaining method calls. """ return self.ellipse(cx, cy, r, r) def ellipse(self, cx, cy, rx, ry): """ Append an ellipse as a closed subpath to the current path. Args: cx (Number): the abscissa of the ellipse's center point. cy (Number): the ordinate of the ellipse's center point. rx (Number): the x-radius of the ellipse. ry (Number): the y-radius of the ellipse. Returns: The path, to allow chaining method calls. """ self._insert_implicit_close_if_open() self.add_path_element(Ellipse(Point(rx, ry), Point(cx, cy)), _copy=False) self._closed = True self.move_to(cx, cy) return self def move_to(self, x, y): """ Start a new subpath or move the path starting point. If no path elements have been added yet, this will change the path starting point. If path elements have been added, this will insert an implicit close in order to start a new subpath. Args: x (Number): abscissa of the (sub)path starting point. y (Number): ordinate of the (sub)path starting point. Returns: The path, to allow chaining method calls. """ self._insert_implicit_close_if_open() self._starter_move = Move(Point(x, y)) return self def move_relative(self, x, y): """ Start a new subpath or move the path start point relative to the previous point. If no path elements have been added yet, this will change the path starting point. If path elements have been added, this will insert an implicit close in order to start a new subpath. This will overwrite an absolute move_to as long as no non-move path items have been appended. The relative position is resolved from the previous item when the path is being rendered, or from 0, 0 if it is the first item. Args: x (Number): abscissa of the (sub)path starting point relative to the previous point. y (Number): ordinate of the (sub)path starting point relative to the previous point. """ self._insert_implicit_close_if_open() if self._starter_move is not None: self._closed = False self._graphics_context.add_item(self._starter_move, _copy=False) self._close_context = self._graphics_context self._starter_move = RelativeMove(Point(x, y)) return self def line_to(self, x, y): """ Append a straight line to this path. Args: x (Number): abscissa the line's end point. y (Number): ordinate of the line's end point. Returns: The path, to allow chaining method calls. """ self.add_path_element(Line(Point(x, y)), _copy=False) return self def line_relative(self, dx, dy): """ Append a straight line whose end is computed as an offset from the end of the previous path element. Args: x (Number): abscissa the line's end point relative to the end point of the previous path element. y (Number): ordinate of the line's end point relative to the end point of the previous path element. Returns: The path, to allow chaining method calls. """ self.add_path_element(RelativeLine(Point(dx, dy)), _copy=False) return self def horizontal_line_to(self, x): """ Append a straight horizontal line to the given abscissa. The ordinate is retrieved from the end point of the previous path element. Args: x (Number): abscissa of the line's end point. Returns: The path, to allow chaining method calls. """ self.add_path_element(HorizontalLine(x), _copy=False) return self def horizontal_line_relative(self, dx): """ Append a straight horizontal line to the given offset from the previous path element. The ordinate is retrieved from the end point of the previous path element. Args: x (Number): abscissa of the line's end point relative to the end point of the previous path element. Returns: The path, to allow chaining method calls. """ self.add_path_element(RelativeHorizontalLine(dx), _copy=False) return self def vertical_line_to(self, y): """ Append a straight vertical line to the given ordinate. The abscissa is retrieved from the end point of the previous path element. Args: y (Number): ordinate of the line's end point. Returns: The path, to allow chaining method calls. """ self.add_path_element(VerticalLine(y), _copy=False) return self def vertical_line_relative(self, dy): """ Append a straight vertical line to the given offset from the previous path element. The abscissa is retrieved from the end point of the previous path element. Args: y (Number): ordinate of the line's end point relative to the end point of the previous path element. Returns: The path, to allow chaining method calls. """ self.add_path_element(RelativeVerticalLine(dy), _copy=False) return self def curve_to(self, x1, y1, x2, y2, x3, y3): """ Append a cubic Bézier curve to this path. Args: x1 (Number): abscissa of the first control point y1 (Number): ordinate of the first control point x2 (Number): abscissa of the second control point y2 (Number): ordinate of the second control point x3 (Number): abscissa of the end point y3 (Number): ordinate of the end point Returns: The path, to allow chaining method calls. """ ctrl1 = Point(x1, y1) ctrl2 = Point(x2, y2) end = Point(x3, y3) self.add_path_element(BezierCurve(ctrl1, ctrl2, end), _copy=False) return self def curve_relative(self, dx1, dy1, dx2, dy2, dx3, dy3): """ Append a cubic Bézier curve whose points are expressed relative to the end point of the previous path element. E.g. with a start point of (0, 0), given (1, 1), (2, 2), (3, 3), the output curve would have the points: (0, 0) c1 (1, 1) c2 (3, 3) e (6, 6) Args: dx1 (Number): abscissa of the first control point relative to the end point of the previous path element dy1 (Number): ordinate of the first control point relative to the end point of the previous path element dx2 (Number): abscissa offset of the second control point relative to the end point of the previous path element dy2 (Number): ordinate offset of the second control point relative to the end point of the previous path element dx3 (Number): abscissa offset of the end point relative to the end point of the previous path element dy3 (Number): ordinate offset of the end point relative to the end point of the previous path element Returns: The path, to allow chaining method calls. """ c1d = Point(dx1, dy1) c2d = Point(dx2, dy2) end = Point(dx3, dy3) self.add_path_element(RelativeBezierCurve(c1d, c2d, end), _copy=False) return self def quadratic_curve_to(self, x1, y1, x2, y2): """ Append a cubic Bézier curve mimicking the specified quadratic Bézier curve. Args: x1 (Number): abscissa of the control point y1 (Number): ordinate of the control point x2 (Number): abscissa of the end point y2 (Number): ordinate of the end point Returns: The path, to allow chaining method calls. """ ctrl = Point(x1, y1) end = Point(x2, y2) self.add_path_element(QuadraticBezierCurve(ctrl, end), _copy=False) return self def quadratic_curve_relative(self, dx1, dy1, dx2, dy2): """ Append a cubic Bézier curve mimicking the specified quadratic Bézier curve. Args: dx1 (Number): abscissa of the control point relative to the end point of the previous path element dy1 (Number): ordinate of the control point relative to the end point of the previous path element dx2 (Number): abscissa offset of the end point relative to the end point of the previous path element dy2 (Number): ordinate offset of the end point relative to the end point of the previous path element Returns: The path, to allow chaining method calls. """ ctrl = Point(dx1, dy1) end = Point(dx2, dy2) self.add_path_element(RelativeQuadraticBezierCurve(ctrl, end), _copy=False) return self def arc_to(self, rx, ry, rotation, large_arc, positive_sweep, x, y): """ Append an elliptical arc from the end of the previous path point to the specified end point. The arc is approximated using Bézier curves, so it is not perfectly accurate. However, the error is small enough to not be noticeable at any reasonable (and even most unreasonable) scales, with a worst-case deviation of around 3‱. Notes: - The signs of the radii arguments (`rx` and `ry`) are ignored (i.e. their absolute values are used instead). - If either radius is 0, then a straight line will be emitted instead of an arc. - If the radii are too small for the arc to reach from the current point to the specified end point (`x` and `y`), then they will be proportionally scaled up until they are big enough, which will always result in a half-ellipse arc (i.e. an 180 degree sweep) Args: rx (Number): radius in the x-direction. ry (Number): radius in the y-direction. rotation (Number): angle (in degrees) that the arc should be rotated clockwise from the principle axes. This parameter does not have a visual effect in the case that `rx == ry`. large_arc (bool): if True, the arc will cover a sweep angle of at least 180 degrees. Otherwise, the sweep angle will be at most 180 degrees. positive_sweep (bool): if True, the arc will be swept over a positive angle, i.e. clockwise. Otherwise, the arc will be swept over a negative angle. x (Number): abscissa of the arc's end point. y (Number): ordinate of the arc's end point. """ if rx == 0 or ry == 0: return self.line_to(x, y) radii = Point(abs(rx), abs(ry)) large_arc = bool(large_arc) rotation = math.radians(rotation) positive_sweep = bool(positive_sweep) end = Point(x, y) self.add_path_element( Arc(radii, rotation, large_arc, positive_sweep, end), _copy=False ) return self def arc_relative(self, rx, ry, rotation, large_arc, positive_sweep, dx, dy): """ Append an elliptical arc from the end of the previous path point to an offset point. The arc is approximated using Bézier curves, so it is not perfectly accurate. However, the error is small enough to not be noticeable at any reasonable (and even most unreasonable) scales, with a worst-case deviation of around 3‱. Notes: - The signs of the radii arguments (`rx` and `ry`) are ignored (i.e. their absolute values are used instead). - If either radius is 0, then a straight line will be emitted instead of an arc. - If the radii are too small for the arc to reach from the current point to the specified end point (`x` and `y`), then they will be proportionally scaled up until they are big enough, which will always result in a half-ellipse arc (i.e. an 180 degree sweep) Args: rx (Number): radius in the x-direction. ry (Number): radius in the y-direction. rotation (Number): angle (in degrees) that the arc should be rotated clockwise from the principle axes. This parameter does not have a visual effect in the case that `rx == ry`. large_arc (bool): if True, the arc will cover a sweep angle of at least 180 degrees. Otherwise, the sweep angle will be at most 180 degrees. positive_sweep (bool): if True, the arc will be swept over a positive angle, i.e. clockwise. Otherwise, the arc will be swept over a negative angle. dx (Number): abscissa of the arc's end point relative to the end point of the previous path element. dy (Number): ordinate of the arc's end point relative to the end point of the previous path element. """ if rx == 0 or ry == 0: return self.line_relative(dx, dy) radii = Point(abs(rx), abs(ry)) large_arc = bool(large_arc) rotation = math.radians(rotation) positive_sweep = bool(positive_sweep) end = Point(dx, dy) self.add_path_element( RelativeArc(radii, rotation, large_arc, positive_sweep, end), _copy=False ) return self def close(self): """ Explicitly close the current (sub)path. """ self.add_path_element(Close(), _copy=False) self._closed = True self.move_relative(0, 0) def _insert_implicit_close_if_open(self): if not self._closed: self._close_context.add_item(ImplicitClose(), _copy=False) self._close_context = self._graphics_context self._closed = True def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this painted path, including nested contexts and transformations.""" return self._root_graphics_context.bounding_box(start, self.style) def render( self, resource_registry, style, last_item, initial_point, debug_stream=None, pfx=None, ): self._insert_implicit_close_if_open() ( render_list, last_item, initial_point, ) = self._root_graphics_context.build_render_list( resource_registry, style, last_item, initial_point, debug_stream, pfx ) paint_rule = GraphicsStyle.merge(style, self.style).resolve_paint_rule() render_list.insert(-1, paint_rule.value) return " ".join(render_list), last_item, initial_point def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `PaintedPath.render`. """ return self.render( resource_registry, style, last_item, initial_point, debug_stream, pfx )
A path to be drawn by the PDF renderer.
A painted path is defined by a style and an arbitrary sequence of path elements, which include the primitive path elements (
Move
,Line
,BezierCurve
, …) as well as arbitrarily nestedGraphicsContext
containing their own sequence of primitive path elements andGraphicsContext
.Subclasses
Instance variables
prop auto_close : bool
-
Expand source code Browse git
@property def auto_close(self) -> bool: """If true, the path should automatically close itself before painting.""" return self.style.auto_close
If true, the path should automatically close itself before painting.
prop clipping_path
-
Expand source code Browse git
@property def clipping_path(self): """Set the clipping path for this path.""" return self._root_graphics_context.clipping_path
Set the clipping path for this path.
prop paint_rule : PathPaintRule
-
Expand source code Browse git
@property def paint_rule(self) -> PathPaintRule: """Manually specify the `PathPaintRule` to use for rendering the path.""" return self.style.paint_rule
Manually specify the
PathPaintRule
to use for rendering the path. prop style : GraphicsStyle
-
Expand source code Browse git
@property def style(self) -> GraphicsStyle: """The `GraphicsStyle` applied to all elements of this path.""" return self._root_graphics_context.style
The
GraphicsStyle
applied to all elements of this path. prop transform : Transform | None
-
Expand source code Browse git
@property def transform(self) -> Optional[Transform]: """The `Transform` that applies to all of the elements of this path.""" return self._root_graphics_context.transform
The
Transform
that applies to all of the elements of this path.
Methods
def add_path_element(self, item)
-
Expand source code Browse git
def add_path_element(self, item, _copy=True): """ Add the given element as a path item of this path. Args: item: the item to add to this path. _copy (bool): if true (the default), the item will be copied before being appended. This prevents modifications to a referenced object from "retroactively" altering its style/shape and should be disabled with caution. """ if self._starter_move is not None: self._closed = False self._graphics_context.add_item(self._starter_move, _copy=False) self._close_context = self._graphics_context self._starter_move = None self._graphics_context.add_item(item, _copy=_copy)
Add the given element as a path item of this path.
Args
item
- the item to add to this path.
_copy
:bool
- if true (the default), the item will be copied before being appended. This prevents modifications to a referenced object from "retroactively" altering its style/shape and should be disabled with caution.
def arc_relative(self, rx, ry, rotation, large_arc, positive_sweep, dx, dy)
-
Expand source code Browse git
def arc_relative(self, rx, ry, rotation, large_arc, positive_sweep, dx, dy): """ Append an elliptical arc from the end of the previous path point to an offset point. The arc is approximated using Bézier curves, so it is not perfectly accurate. However, the error is small enough to not be noticeable at any reasonable (and even most unreasonable) scales, with a worst-case deviation of around 3‱. Notes: - The signs of the radii arguments (`rx` and `ry`) are ignored (i.e. their absolute values are used instead). - If either radius is 0, then a straight line will be emitted instead of an arc. - If the radii are too small for the arc to reach from the current point to the specified end point (`x` and `y`), then they will be proportionally scaled up until they are big enough, which will always result in a half-ellipse arc (i.e. an 180 degree sweep) Args: rx (Number): radius in the x-direction. ry (Number): radius in the y-direction. rotation (Number): angle (in degrees) that the arc should be rotated clockwise from the principle axes. This parameter does not have a visual effect in the case that `rx == ry`. large_arc (bool): if True, the arc will cover a sweep angle of at least 180 degrees. Otherwise, the sweep angle will be at most 180 degrees. positive_sweep (bool): if True, the arc will be swept over a positive angle, i.e. clockwise. Otherwise, the arc will be swept over a negative angle. dx (Number): abscissa of the arc's end point relative to the end point of the previous path element. dy (Number): ordinate of the arc's end point relative to the end point of the previous path element. """ if rx == 0 or ry == 0: return self.line_relative(dx, dy) radii = Point(abs(rx), abs(ry)) large_arc = bool(large_arc) rotation = math.radians(rotation) positive_sweep = bool(positive_sweep) end = Point(dx, dy) self.add_path_element( RelativeArc(radii, rotation, large_arc, positive_sweep, end), _copy=False ) return self
Append an elliptical arc from the end of the previous path point to an offset point.
The arc is approximated using Bézier curves, so it is not perfectly accurate. However, the error is small enough to not be noticeable at any reasonable (and even most unreasonable) scales, with a worst-case deviation of around 3‱.
Notes
- The signs of the radii arguments (
rx
andry
) are ignored (i.e. their absolute values are used instead). - If either radius is 0, then a straight line will be emitted instead of an arc.
- If the radii are too small for the arc to reach from the current point to
the specified end point (
x
andy
), then they will be proportionally scaled up until they are big enough, which will always result in a half-ellipse arc (i.e. an 180 degree sweep)
Args
rx
:Number
- radius in the x-direction.
ry
:Number
- radius in the y-direction.
rotation
:Number
- angle (in degrees) that the arc should be rotated
clockwise from the principle axes. This parameter does not have
a visual effect in the case that
rx == ry
. large_arc
:bool
- if True, the arc will cover a sweep angle of at least 180 degrees. Otherwise, the sweep angle will be at most 180 degrees.
positive_sweep
:bool
- if True, the arc will be swept over a positive angle, i.e. clockwise. Otherwise, the arc will be swept over a negative angle.
dx
:Number
- abscissa of the arc's end point relative to the end point of the previous path element.
dy
:Number
- ordinate of the arc's end point relative to the end point of the previous path element.
- The signs of the radii arguments (
def arc_to(self, rx, ry, rotation, large_arc, positive_sweep, x, y)
-
Expand source code Browse git
def arc_to(self, rx, ry, rotation, large_arc, positive_sweep, x, y): """ Append an elliptical arc from the end of the previous path point to the specified end point. The arc is approximated using Bézier curves, so it is not perfectly accurate. However, the error is small enough to not be noticeable at any reasonable (and even most unreasonable) scales, with a worst-case deviation of around 3‱. Notes: - The signs of the radii arguments (`rx` and `ry`) are ignored (i.e. their absolute values are used instead). - If either radius is 0, then a straight line will be emitted instead of an arc. - If the radii are too small for the arc to reach from the current point to the specified end point (`x` and `y`), then they will be proportionally scaled up until they are big enough, which will always result in a half-ellipse arc (i.e. an 180 degree sweep) Args: rx (Number): radius in the x-direction. ry (Number): radius in the y-direction. rotation (Number): angle (in degrees) that the arc should be rotated clockwise from the principle axes. This parameter does not have a visual effect in the case that `rx == ry`. large_arc (bool): if True, the arc will cover a sweep angle of at least 180 degrees. Otherwise, the sweep angle will be at most 180 degrees. positive_sweep (bool): if True, the arc will be swept over a positive angle, i.e. clockwise. Otherwise, the arc will be swept over a negative angle. x (Number): abscissa of the arc's end point. y (Number): ordinate of the arc's end point. """ if rx == 0 or ry == 0: return self.line_to(x, y) radii = Point(abs(rx), abs(ry)) large_arc = bool(large_arc) rotation = math.radians(rotation) positive_sweep = bool(positive_sweep) end = Point(x, y) self.add_path_element( Arc(radii, rotation, large_arc, positive_sweep, end), _copy=False ) return self
Append an elliptical arc from the end of the previous path point to the specified end point.
The arc is approximated using Bézier curves, so it is not perfectly accurate. However, the error is small enough to not be noticeable at any reasonable (and even most unreasonable) scales, with a worst-case deviation of around 3‱.
Notes
- The signs of the radii arguments (
rx
andry
) are ignored (i.e. their absolute values are used instead). - If either radius is 0, then a straight line will be emitted instead of an arc.
- If the radii are too small for the arc to reach from the current point to
the specified end point (
x
andy
), then they will be proportionally scaled up until they are big enough, which will always result in a half-ellipse arc (i.e. an 180 degree sweep)
Args
rx
:Number
- radius in the x-direction.
ry
:Number
- radius in the y-direction.
rotation
:Number
- angle (in degrees) that the arc should be rotated
clockwise from the principle axes. This parameter does not have
a visual effect in the case that
rx == ry
. large_arc
:bool
- if True, the arc will cover a sweep angle of at least 180 degrees. Otherwise, the sweep angle will be at most 180 degrees.
positive_sweep
:bool
- if True, the arc will be swept over a positive angle, i.e. clockwise. Otherwise, the arc will be swept over a negative angle.
x
:Number
- abscissa of the arc's end point.
y
:Number
- ordinate of the arc's end point.
- The signs of the radii arguments (
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this painted path, including nested contexts and transformations.""" return self._root_graphics_context.bounding_box(start, self.style)
Compute the bounding box of this painted path, including nested contexts and transformations.
def circle(self, cx, cy, r)
-
Expand source code Browse git
def circle(self, cx, cy, r): """ Append a circle as a closed subpath to the current path. Args: cx (Number): the abscissa of the circle's center point. cy (Number): the ordinate of the circle's center point. r (Number): the radius of the circle. Returns: The path, to allow chaining method calls. """ return self.ellipse(cx, cy, r, r)
Append a circle as a closed subpath to the current path.
Args
cx
:Number
- the abscissa of the circle's center point.
cy
:Number
- the ordinate of the circle's center point.
r
:Number
- the radius of the circle.
Returns
The path, to allow chaining method calls.
def close(self)
-
Expand source code Browse git
def close(self): """ Explicitly close the current (sub)path. """ self.add_path_element(Close(), _copy=False) self._closed = True self.move_relative(0, 0)
Explicitly close the current (sub)path.
def curve_relative(self, dx1, dy1, dx2, dy2, dx3, dy3)
-
Expand source code Browse git
def curve_relative(self, dx1, dy1, dx2, dy2, dx3, dy3): """ Append a cubic Bézier curve whose points are expressed relative to the end point of the previous path element. E.g. with a start point of (0, 0), given (1, 1), (2, 2), (3, 3), the output curve would have the points: (0, 0) c1 (1, 1) c2 (3, 3) e (6, 6) Args: dx1 (Number): abscissa of the first control point relative to the end point of the previous path element dy1 (Number): ordinate of the first control point relative to the end point of the previous path element dx2 (Number): abscissa offset of the second control point relative to the end point of the previous path element dy2 (Number): ordinate offset of the second control point relative to the end point of the previous path element dx3 (Number): abscissa offset of the end point relative to the end point of the previous path element dy3 (Number): ordinate offset of the end point relative to the end point of the previous path element Returns: The path, to allow chaining method calls. """ c1d = Point(dx1, dy1) c2d = Point(dx2, dy2) end = Point(dx3, dy3) self.add_path_element(RelativeBezierCurve(c1d, c2d, end), _copy=False) return self
Append a cubic Bézier curve whose points are expressed relative to the end point of the previous path element.
E.g. with a start point of (0, 0), given (1, 1), (2, 2), (3, 3), the output curve would have the points:
(0, 0) c1 (1, 1) c2 (3, 3) e (6, 6)
Args
dx1
:Number
- abscissa of the first control point relative to the end point of the previous path element
dy1
:Number
- ordinate of the first control point relative to the end point of the previous path element
dx2
:Number
- abscissa offset of the second control point relative to the end point of the previous path element
dy2
:Number
- ordinate offset of the second control point relative to the end point of the previous path element
dx3
:Number
- abscissa offset of the end point relative to the end point of the previous path element
dy3
:Number
- ordinate offset of the end point relative to the end point of the previous path element
Returns
The path, to allow chaining method calls.
def curve_to(self, x1, y1, x2, y2, x3, y3)
-
Expand source code Browse git
def curve_to(self, x1, y1, x2, y2, x3, y3): """ Append a cubic Bézier curve to this path. Args: x1 (Number): abscissa of the first control point y1 (Number): ordinate of the first control point x2 (Number): abscissa of the second control point y2 (Number): ordinate of the second control point x3 (Number): abscissa of the end point y3 (Number): ordinate of the end point Returns: The path, to allow chaining method calls. """ ctrl1 = Point(x1, y1) ctrl2 = Point(x2, y2) end = Point(x3, y3) self.add_path_element(BezierCurve(ctrl1, ctrl2, end), _copy=False) return self
Append a cubic Bézier curve to this path.
Args
x1
:Number
- abscissa of the first control point
y1
:Number
- ordinate of the first control point
x2
:Number
- abscissa of the second control point
y2
:Number
- ordinate of the second control point
x3
:Number
- abscissa of the end point
y3
:Number
- ordinate of the end point
Returns
The path, to allow chaining method calls.
def ellipse(self, cx, cy, rx, ry)
-
Expand source code Browse git
def ellipse(self, cx, cy, rx, ry): """ Append an ellipse as a closed subpath to the current path. Args: cx (Number): the abscissa of the ellipse's center point. cy (Number): the ordinate of the ellipse's center point. rx (Number): the x-radius of the ellipse. ry (Number): the y-radius of the ellipse. Returns: The path, to allow chaining method calls. """ self._insert_implicit_close_if_open() self.add_path_element(Ellipse(Point(rx, ry), Point(cx, cy)), _copy=False) self._closed = True self.move_to(cx, cy) return self
Append an ellipse as a closed subpath to the current path.
Args
cx
:Number
- the abscissa of the ellipse's center point.
cy
:Number
- the ordinate of the ellipse's center point.
rx
:Number
- the x-radius of the ellipse.
ry
:Number
- the y-radius of the ellipse.
Returns
The path, to allow chaining method calls.
def horizontal_line_relative(self, dx)
-
Expand source code Browse git
def horizontal_line_relative(self, dx): """ Append a straight horizontal line to the given offset from the previous path element. The ordinate is retrieved from the end point of the previous path element. Args: x (Number): abscissa of the line's end point relative to the end point of the previous path element. Returns: The path, to allow chaining method calls. """ self.add_path_element(RelativeHorizontalLine(dx), _copy=False) return self
Append a straight horizontal line to the given offset from the previous path element. The ordinate is retrieved from the end point of the previous path element.
Args
x
:Number
- abscissa of the line's end point relative to the end point of the previous path element.
Returns
The path, to allow chaining method calls.
def horizontal_line_to(self, x)
-
Expand source code Browse git
def horizontal_line_to(self, x): """ Append a straight horizontal line to the given abscissa. The ordinate is retrieved from the end point of the previous path element. Args: x (Number): abscissa of the line's end point. Returns: The path, to allow chaining method calls. """ self.add_path_element(HorizontalLine(x), _copy=False) return self
Append a straight horizontal line to the given abscissa. The ordinate is retrieved from the end point of the previous path element.
Args
x
:Number
- abscissa of the line's end point.
Returns
The path, to allow chaining method calls.
def line_relative(self, dx, dy)
-
Expand source code Browse git
def line_relative(self, dx, dy): """ Append a straight line whose end is computed as an offset from the end of the previous path element. Args: x (Number): abscissa the line's end point relative to the end point of the previous path element. y (Number): ordinate of the line's end point relative to the end point of the previous path element. Returns: The path, to allow chaining method calls. """ self.add_path_element(RelativeLine(Point(dx, dy)), _copy=False) return self
Append a straight line whose end is computed as an offset from the end of the previous path element.
Args
x
:Number
- abscissa the line's end point relative to the end point of the previous path element.
y
:Number
- ordinate of the line's end point relative to the end point of the previous path element.
Returns
The path, to allow chaining method calls.
def line_to(self, x, y)
-
Expand source code Browse git
def line_to(self, x, y): """ Append a straight line to this path. Args: x (Number): abscissa the line's end point. y (Number): ordinate of the line's end point. Returns: The path, to allow chaining method calls. """ self.add_path_element(Line(Point(x, y)), _copy=False) return self
Append a straight line to this path.
Args
x
:Number
- abscissa the line's end point.
y
:Number
- ordinate of the line's end point.
Returns
The path, to allow chaining method calls.
def move_relative(self, x, y)
-
Expand source code Browse git
def move_relative(self, x, y): """ Start a new subpath or move the path start point relative to the previous point. If no path elements have been added yet, this will change the path starting point. If path elements have been added, this will insert an implicit close in order to start a new subpath. This will overwrite an absolute move_to as long as no non-move path items have been appended. The relative position is resolved from the previous item when the path is being rendered, or from 0, 0 if it is the first item. Args: x (Number): abscissa of the (sub)path starting point relative to the previous point. y (Number): ordinate of the (sub)path starting point relative to the previous point. """ self._insert_implicit_close_if_open() if self._starter_move is not None: self._closed = False self._graphics_context.add_item(self._starter_move, _copy=False) self._close_context = self._graphics_context self._starter_move = RelativeMove(Point(x, y)) return self
Start a new subpath or move the path start point relative to the previous point.
If no path elements have been added yet, this will change the path starting point. If path elements have been added, this will insert an implicit close in order to start a new subpath.
This will overwrite an absolute move_to as long as no non-move path items have been appended. The relative position is resolved from the previous item when the path is being rendered, or from 0, 0 if it is the first item.
Args
x
:Number
- abscissa of the (sub)path starting point relative to the previous point.
y
:Number
- ordinate of the (sub)path starting point relative to the previous point.
def move_to(self, x, y)
-
Expand source code Browse git
def move_to(self, x, y): """ Start a new subpath or move the path starting point. If no path elements have been added yet, this will change the path starting point. If path elements have been added, this will insert an implicit close in order to start a new subpath. Args: x (Number): abscissa of the (sub)path starting point. y (Number): ordinate of the (sub)path starting point. Returns: The path, to allow chaining method calls. """ self._insert_implicit_close_if_open() self._starter_move = Move(Point(x, y)) return self
Start a new subpath or move the path starting point.
If no path elements have been added yet, this will change the path starting point. If path elements have been added, this will insert an implicit close in order to start a new subpath.
Args
x
:Number
- abscissa of the (sub)path starting point.
y
:Number
- ordinate of the (sub)path starting point.
Returns
The path, to allow chaining method calls.
def quadratic_curve_relative(self, dx1, dy1, dx2, dy2)
-
Expand source code Browse git
def quadratic_curve_relative(self, dx1, dy1, dx2, dy2): """ Append a cubic Bézier curve mimicking the specified quadratic Bézier curve. Args: dx1 (Number): abscissa of the control point relative to the end point of the previous path element dy1 (Number): ordinate of the control point relative to the end point of the previous path element dx2 (Number): abscissa offset of the end point relative to the end point of the previous path element dy2 (Number): ordinate offset of the end point relative to the end point of the previous path element Returns: The path, to allow chaining method calls. """ ctrl = Point(dx1, dy1) end = Point(dx2, dy2) self.add_path_element(RelativeQuadraticBezierCurve(ctrl, end), _copy=False) return self
Append a cubic Bézier curve mimicking the specified quadratic Bézier curve.
Args
dx1
:Number
- abscissa of the control point relative to the end point of the previous path element
dy1
:Number
- ordinate of the control point relative to the end point of the previous path element
dx2
:Number
- abscissa offset of the end point relative to the end point of the previous path element
dy2
:Number
- ordinate offset of the end point relative to the end point of the previous path element
Returns
The path, to allow chaining method calls.
def quadratic_curve_to(self, x1, y1, x2, y2)
-
Expand source code Browse git
def quadratic_curve_to(self, x1, y1, x2, y2): """ Append a cubic Bézier curve mimicking the specified quadratic Bézier curve. Args: x1 (Number): abscissa of the control point y1 (Number): ordinate of the control point x2 (Number): abscissa of the end point y2 (Number): ordinate of the end point Returns: The path, to allow chaining method calls. """ ctrl = Point(x1, y1) end = Point(x2, y2) self.add_path_element(QuadraticBezierCurve(ctrl, end), _copy=False) return self
Append a cubic Bézier curve mimicking the specified quadratic Bézier curve.
Args
x1
:Number
- abscissa of the control point
y1
:Number
- ordinate of the control point
x2
:Number
- abscissa of the end point
y2
:Number
- ordinate of the end point
Returns
The path, to allow chaining method calls.
def rectangle(self, x, y, w, h, rx=0, ry=0)
-
Expand source code Browse git
def rectangle(self, x, y, w, h, rx=0, ry=0): """ Append a rectangle as a closed subpath to the current path. If the width or the height are 0, the rectangle will be collapsed to a line (unless they're both 0, in which case it's collapsed to nothing). Args: x (Number): the abscissa of the starting corner of the rectangle. y (Number): the ordinate of the starting corner of the rectangle. w (Number): the width of the rectangle (if 0, the rectangle will be rendered as a vertical line). h (Number): the height of the rectangle (if 0, the rectangle will be rendered as a horizontal line). rx (Number): the x-radius of the rectangle rounded corner (if 0 the corners will not be rounded). ry (Number): the y-radius of the rectangle rounded corner (if 0 the corners will not be rounded). Returns: The path, to allow chaining method calls. """ self._insert_implicit_close_if_open() self.add_path_element( RoundedRectangle(Point(x, y), Point(w, h), Point(rx, ry)), _copy=False ) self._closed = True self.move_to(x, y) return self
Append a rectangle as a closed subpath to the current path.
If the width or the height are 0, the rectangle will be collapsed to a line (unless they're both 0, in which case it's collapsed to nothing).
Args
x
:Number
- the abscissa of the starting corner of the rectangle.
y
:Number
- the ordinate of the starting corner of the rectangle.
w
:Number
- the width of the rectangle (if 0, the rectangle will be rendered as a vertical line).
h
:Number
- the height of the rectangle (if 0, the rectangle will be rendered as a horizontal line).
rx
:Number
- the x-radius of the rectangle rounded corner (if 0 the corners will not be rounded).
ry
:Number
- the y-radius of the rectangle rounded corner (if 0 the corners will not be rounded).
Returns
The path, to allow chaining method calls.
def remove_last_path_element(self)
-
Expand source code Browse git
def remove_last_path_element(self): self._graphics_context.remove_last_item()
def render(self, resource_registry, style, last_item, initial_point, debug_stream=None, pfx=None)
-
Expand source code Browse git
def render( self, resource_registry, style, last_item, initial_point, debug_stream=None, pfx=None, ): self._insert_implicit_close_if_open() ( render_list, last_item, initial_point, ) = self._root_graphics_context.build_render_list( resource_registry, style, last_item, initial_point, debug_stream, pfx ) paint_rule = GraphicsStyle.merge(style, self.style).resolve_paint_rule() render_list.insert(-1, paint_rule.value) return " ".join(render_list), last_item, initial_point
def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `PaintedPath.render`. """ return self.render( resource_registry, style, last_item, initial_point, debug_stream, pfx )
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
PaintedPath.render()
. def transform_group(self, transform)
-
Expand source code Browse git
@contextmanager def transform_group(self, transform): """ Apply the provided `Transform` to all points added within this context. """ with self._new_graphics_context() as ctxt: ctxt.transform = transform yield self
Apply the provided
Transform
to all points added within this context. def vertical_line_relative(self, dy)
-
Expand source code Browse git
def vertical_line_relative(self, dy): """ Append a straight vertical line to the given offset from the previous path element. The abscissa is retrieved from the end point of the previous path element. Args: y (Number): ordinate of the line's end point relative to the end point of the previous path element. Returns: The path, to allow chaining method calls. """ self.add_path_element(RelativeVerticalLine(dy), _copy=False) return self
Append a straight vertical line to the given offset from the previous path element. The abscissa is retrieved from the end point of the previous path element.
Args
y
:Number
- ordinate of the line's end point relative to the end point of the previous path element.
Returns
The path, to allow chaining method calls.
def vertical_line_to(self, y)
-
Expand source code Browse git
def vertical_line_to(self, y): """ Append a straight vertical line to the given ordinate. The abscissa is retrieved from the end point of the previous path element. Args: y (Number): ordinate of the line's end point. Returns: The path, to allow chaining method calls. """ self.add_path_element(VerticalLine(y), _copy=False) return self
Append a straight vertical line to the given ordinate. The abscissa is retrieved from the end point of the previous path element.
Args
y
:Number
- ordinate of the line's end point.
Returns
The path, to allow chaining method calls.
class QuadraticBezierCurve (ctrl: Point,
end: Point)-
Expand source code Browse git
class QuadraticBezierCurve(NamedTuple): """ A quadratic Bézier curve path element. This draws a Bézier curve parameterized by the end point of the previous path element, one off-curve control point, and an end point. See: `PaintedPath.quadratic_curve_to` """ ctrl: Point """The curve's control point.""" end: Point """The curve's end point.""" @property def end_point(self) -> Point: """The end point of this path element.""" return self.end def to_cubic_curve(self, start_point: Point) -> BezierCurve: ctrl = self.ctrl end = self.end ctrl1 = Point( x=start_point.x + 2 * (ctrl.x - start_point.x) / 3, y=start_point.y + 2 * (ctrl.y - start_point.y) / 3, ) ctrl2 = Point( x=end.x + 2 * (ctrl.x - end.x) / 3, y=end.y + 2 * (ctrl.y - end.y) / 3, ) return BezierCurve(ctrl1, ctrl2, end) def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this quadratic Bézier curve by converting it to a cubic Bézier.""" cubic = self.to_cubic_curve(start) return cubic.bounding_box(start) @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is `self`. """ return ( self.to_cubic_curve(last_item.end_point).render( resource_registry, style, last_item, initial_point )[0], self, initial_point, ) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `QuadraticBezierCurve.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write( f"{self} resolved to {self.to_cubic_curve(last_item.end_point)}\n" ) return rendered, resolved, initial_point
A quadratic Bézier curve path element.
This draws a Bézier curve parameterized by the end point of the previous path element, one off-curve control point, and an end point.
Ancestors
- builtins.tuple
Instance variables
var ctrl : Point
-
Expand source code Browse git
class QuadraticBezierCurve(NamedTuple): """ A quadratic Bézier curve path element. This draws a Bézier curve parameterized by the end point of the previous path element, one off-curve control point, and an end point. See: `PaintedPath.quadratic_curve_to` """ ctrl: Point """The curve's control point.""" end: Point """The curve's end point.""" @property def end_point(self) -> Point: """The end point of this path element.""" return self.end def to_cubic_curve(self, start_point: Point) -> BezierCurve: ctrl = self.ctrl end = self.end ctrl1 = Point( x=start_point.x + 2 * (ctrl.x - start_point.x) / 3, y=start_point.y + 2 * (ctrl.y - start_point.y) / 3, ) ctrl2 = Point( x=end.x + 2 * (ctrl.x - end.x) / 3, y=end.y + 2 * (ctrl.y - end.y) / 3, ) return BezierCurve(ctrl1, ctrl2, end) def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this quadratic Bézier curve by converting it to a cubic Bézier.""" cubic = self.to_cubic_curve(start) return cubic.bounding_box(start) @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is `self`. """ return ( self.to_cubic_curve(last_item.end_point).render( resource_registry, style, last_item, initial_point )[0], self, initial_point, ) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `QuadraticBezierCurve.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write( f"{self} resolved to {self.to_cubic_curve(last_item.end_point)}\n" ) return rendered, resolved, initial_point
The curve's control point.
var end : Point
-
Expand source code Browse git
class QuadraticBezierCurve(NamedTuple): """ A quadratic Bézier curve path element. This draws a Bézier curve parameterized by the end point of the previous path element, one off-curve control point, and an end point. See: `PaintedPath.quadratic_curve_to` """ ctrl: Point """The curve's control point.""" end: Point """The curve's end point.""" @property def end_point(self) -> Point: """The end point of this path element.""" return self.end def to_cubic_curve(self, start_point: Point) -> BezierCurve: ctrl = self.ctrl end = self.end ctrl1 = Point( x=start_point.x + 2 * (ctrl.x - start_point.x) / 3, y=start_point.y + 2 * (ctrl.y - start_point.y) / 3, ) ctrl2 = Point( x=end.x + 2 * (ctrl.x - end.x) / 3, y=end.y + 2 * (ctrl.y - end.y) / 3, ) return BezierCurve(ctrl1, ctrl2, end) def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this quadratic Bézier curve by converting it to a cubic Bézier.""" cubic = self.to_cubic_curve(start) return cubic.bounding_box(start) @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is `self`. """ return ( self.to_cubic_curve(last_item.end_point).render( resource_registry, style, last_item, initial_point )[0], self, initial_point, ) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `QuadraticBezierCurve.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write( f"{self} resolved to {self.to_cubic_curve(last_item.end_point)}\n" ) return rendered, resolved, initial_point
The curve's end point.
prop end_point : Point
-
Expand source code Browse git
@property def end_point(self) -> Point: """The end point of this path element.""" return self.end
The end point of this path element.
Methods
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this quadratic Bézier curve by converting it to a cubic Bézier.""" cubic = self.to_cubic_curve(start) return cubic.bounding_box(start)
Compute the bounding box of this quadratic Bézier curve by converting it to a cubic Bézier.
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
@force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is `self`. """ return ( self.to_cubic_curve(last_item.end_point).render( resource_registry, style, last_item, initial_point )[0], self, initial_point, )
Render this path element to its PDF representation.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
Returns
a tuple of
(str, new_last_item)
, wherenew_last_item
isself
. def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
@force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `QuadraticBezierCurve.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write( f"{self} resolved to {self.to_cubic_curve(last_item.end_point)}\n" ) return rendered, resolved, initial_point
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
QuadraticBezierCurve.render()
. def to_cubic_curve(self,
start_point: Point) ‑> BezierCurve-
Expand source code Browse git
def to_cubic_curve(self, start_point: Point) -> BezierCurve: ctrl = self.ctrl end = self.end ctrl1 = Point( x=start_point.x + 2 * (ctrl.x - start_point.x) / 3, y=start_point.y + 2 * (ctrl.y - start_point.y) / 3, ) ctrl2 = Point( x=end.x + 2 * (ctrl.x - end.x) / 3, y=end.y + 2 * (ctrl.y - end.y) / 3, ) return BezierCurve(ctrl1, ctrl2, end)
class Rectangle (org: Point,
size: Point)-
Expand source code Browse git
class Rectangle(NamedTuple): """A pdf primitive rectangle.""" org: Point """The top-left corner of the rectangle.""" size: Point """The width and height of the rectangle.""" # pylint: disable=unused-argument def bounding_box(self, start=None) -> tuple[BoundingBox, Point]: """Compute the bounding box of this rectangle.""" x0, y0 = self.org.x, self.org.y x1 = float(x0) + float(self.size.x) y1 = float(y0) + float(self.size.y) bbox = BoundingBox.from_points( [ Point(x0, y0), Point(x1, y0), Point(x0, y1), Point(x1, y1), ] ) return bbox, self.org @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a `Line` back to the rectangle's origin. """ return ( f"{self.org.render()} {self.size.render()} re", Line(self.org), initial_point, ) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Rectangle.render`. """ rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {rendered}\n") return rendered, resolved, initial_point
A pdf primitive rectangle.
Ancestors
- builtins.tuple
Instance variables
var org : Point
-
Expand source code Browse git
class Rectangle(NamedTuple): """A pdf primitive rectangle.""" org: Point """The top-left corner of the rectangle.""" size: Point """The width and height of the rectangle.""" # pylint: disable=unused-argument def bounding_box(self, start=None) -> tuple[BoundingBox, Point]: """Compute the bounding box of this rectangle.""" x0, y0 = self.org.x, self.org.y x1 = float(x0) + float(self.size.x) y1 = float(y0) + float(self.size.y) bbox = BoundingBox.from_points( [ Point(x0, y0), Point(x1, y0), Point(x0, y1), Point(x1, y1), ] ) return bbox, self.org @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a `Line` back to the rectangle's origin. """ return ( f"{self.org.render()} {self.size.render()} re", Line(self.org), initial_point, ) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Rectangle.render`. """ rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {rendered}\n") return rendered, resolved, initial_point
The top-left corner of the rectangle.
var size : Point
-
Expand source code Browse git
class Rectangle(NamedTuple): """A pdf primitive rectangle.""" org: Point """The top-left corner of the rectangle.""" size: Point """The width and height of the rectangle.""" # pylint: disable=unused-argument def bounding_box(self, start=None) -> tuple[BoundingBox, Point]: """Compute the bounding box of this rectangle.""" x0, y0 = self.org.x, self.org.y x1 = float(x0) + float(self.size.x) y1 = float(y0) + float(self.size.y) bbox = BoundingBox.from_points( [ Point(x0, y0), Point(x1, y0), Point(x0, y1), Point(x1, y1), ] ) return bbox, self.org @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a `Line` back to the rectangle's origin. """ return ( f"{self.org.render()} {self.size.render()} re", Line(self.org), initial_point, ) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Rectangle.render`. """ rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {rendered}\n") return rendered, resolved, initial_point
The width and height of the rectangle.
Methods
def bounding_box(self, start=None) ‑> tuple[BoundingBox, Point]
-
Expand source code Browse git
def bounding_box(self, start=None) -> tuple[BoundingBox, Point]: """Compute the bounding box of this rectangle.""" x0, y0 = self.org.x, self.org.y x1 = float(x0) + float(self.size.x) y1 = float(y0) + float(self.size.y) bbox = BoundingBox.from_points( [ Point(x0, y0), Point(x1, y0), Point(x0, y1), Point(x1, y1), ] ) return bbox, self.org
Compute the bounding box of this rectangle.
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
@force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a `Line` back to the rectangle's origin. """ return ( f"{self.org.render()} {self.size.render()} re", Line(self.org), initial_point, )
Render this path element to its PDF representation.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
Returns
a tuple of
(str, new_last_item)
, wherenew_last_item
is aLine
back to the rectangle's origin. def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
@force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `Rectangle.render`. """ rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {rendered}\n") return rendered, resolved, initial_point
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
Rectangle.render()
.
class RelativeArc (radii: Point,
rotation: int | float | decimal.Decimal,
large: bool,
sweep: bool,
end: Point)-
Expand source code Browse git
class RelativeArc(NamedTuple): """ An elliptical arc path element. The arc is drawn from the end of the current path element to its specified end point using a number of parameters to determine how it is constructed. See: `PaintedPath.arc_relative` """ radii: Point """ The x- and y-radii of the arc. If `radii.x == radii.y` the arc will be circular. """ rotation: Number """The rotation of the arc's major/minor axes relative to the coordinate frame.""" large: bool """If True, sweep the arc over an angle greater than or equal to 180 degrees.""" sweep: bool """If True, the arc is swept in the positive angular direction.""" end: Point """The end point of the arc relative to the end of the previous path element.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of the resolved arc from the given start point.""" end_point = start + self.end arc = Arc( radii=self.radii, rotation=self.rotation, large=self.large, sweep=self.sweep, end=end_point, ) return arc.bounding_box(start) @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `BezierCurve`. """ return Arc( self.radii, self.rotation, self.large, self.sweep, last_item.end_point + self.end, ).render(resource_registry, style, last_item, initial_point) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeArc.render`. """ # newline is intentionally missing here debug_stream.write(f"{self} resolved to ") return Arc( self.radii, self.rotation, self.large, self.sweep, last_item.end_point + self.end, ).render_debug( resource_registry, style, last_item, initial_point, debug_stream, pfx )
An elliptical arc path element.
The arc is drawn from the end of the current path element to its specified end point using a number of parameters to determine how it is constructed.
Ancestors
- builtins.tuple
Instance variables
var end : Point
-
Expand source code Browse git
class RelativeArc(NamedTuple): """ An elliptical arc path element. The arc is drawn from the end of the current path element to its specified end point using a number of parameters to determine how it is constructed. See: `PaintedPath.arc_relative` """ radii: Point """ The x- and y-radii of the arc. If `radii.x == radii.y` the arc will be circular. """ rotation: Number """The rotation of the arc's major/minor axes relative to the coordinate frame.""" large: bool """If True, sweep the arc over an angle greater than or equal to 180 degrees.""" sweep: bool """If True, the arc is swept in the positive angular direction.""" end: Point """The end point of the arc relative to the end of the previous path element.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of the resolved arc from the given start point.""" end_point = start + self.end arc = Arc( radii=self.radii, rotation=self.rotation, large=self.large, sweep=self.sweep, end=end_point, ) return arc.bounding_box(start) @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `BezierCurve`. """ return Arc( self.radii, self.rotation, self.large, self.sweep, last_item.end_point + self.end, ).render(resource_registry, style, last_item, initial_point) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeArc.render`. """ # newline is intentionally missing here debug_stream.write(f"{self} resolved to ") return Arc( self.radii, self.rotation, self.large, self.sweep, last_item.end_point + self.end, ).render_debug( resource_registry, style, last_item, initial_point, debug_stream, pfx )
The end point of the arc relative to the end of the previous path element.
var large : bool
-
Expand source code Browse git
class RelativeArc(NamedTuple): """ An elliptical arc path element. The arc is drawn from the end of the current path element to its specified end point using a number of parameters to determine how it is constructed. See: `PaintedPath.arc_relative` """ radii: Point """ The x- and y-radii of the arc. If `radii.x == radii.y` the arc will be circular. """ rotation: Number """The rotation of the arc's major/minor axes relative to the coordinate frame.""" large: bool """If True, sweep the arc over an angle greater than or equal to 180 degrees.""" sweep: bool """If True, the arc is swept in the positive angular direction.""" end: Point """The end point of the arc relative to the end of the previous path element.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of the resolved arc from the given start point.""" end_point = start + self.end arc = Arc( radii=self.radii, rotation=self.rotation, large=self.large, sweep=self.sweep, end=end_point, ) return arc.bounding_box(start) @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `BezierCurve`. """ return Arc( self.radii, self.rotation, self.large, self.sweep, last_item.end_point + self.end, ).render(resource_registry, style, last_item, initial_point) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeArc.render`. """ # newline is intentionally missing here debug_stream.write(f"{self} resolved to ") return Arc( self.radii, self.rotation, self.large, self.sweep, last_item.end_point + self.end, ).render_debug( resource_registry, style, last_item, initial_point, debug_stream, pfx )
If True, sweep the arc over an angle greater than or equal to 180 degrees.
var radii : Point
-
Expand source code Browse git
class RelativeArc(NamedTuple): """ An elliptical arc path element. The arc is drawn from the end of the current path element to its specified end point using a number of parameters to determine how it is constructed. See: `PaintedPath.arc_relative` """ radii: Point """ The x- and y-radii of the arc. If `radii.x == radii.y` the arc will be circular. """ rotation: Number """The rotation of the arc's major/minor axes relative to the coordinate frame.""" large: bool """If True, sweep the arc over an angle greater than or equal to 180 degrees.""" sweep: bool """If True, the arc is swept in the positive angular direction.""" end: Point """The end point of the arc relative to the end of the previous path element.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of the resolved arc from the given start point.""" end_point = start + self.end arc = Arc( radii=self.radii, rotation=self.rotation, large=self.large, sweep=self.sweep, end=end_point, ) return arc.bounding_box(start) @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `BezierCurve`. """ return Arc( self.radii, self.rotation, self.large, self.sweep, last_item.end_point + self.end, ).render(resource_registry, style, last_item, initial_point) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeArc.render`. """ # newline is intentionally missing here debug_stream.write(f"{self} resolved to ") return Arc( self.radii, self.rotation, self.large, self.sweep, last_item.end_point + self.end, ).render_debug( resource_registry, style, last_item, initial_point, debug_stream, pfx )
The x- and y-radii of the arc. If
radii.x == radii.y
the arc will be circular. var rotation : int | float | decimal.Decimal
-
Expand source code Browse git
class RelativeArc(NamedTuple): """ An elliptical arc path element. The arc is drawn from the end of the current path element to its specified end point using a number of parameters to determine how it is constructed. See: `PaintedPath.arc_relative` """ radii: Point """ The x- and y-radii of the arc. If `radii.x == radii.y` the arc will be circular. """ rotation: Number """The rotation of the arc's major/minor axes relative to the coordinate frame.""" large: bool """If True, sweep the arc over an angle greater than or equal to 180 degrees.""" sweep: bool """If True, the arc is swept in the positive angular direction.""" end: Point """The end point of the arc relative to the end of the previous path element.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of the resolved arc from the given start point.""" end_point = start + self.end arc = Arc( radii=self.radii, rotation=self.rotation, large=self.large, sweep=self.sweep, end=end_point, ) return arc.bounding_box(start) @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `BezierCurve`. """ return Arc( self.radii, self.rotation, self.large, self.sweep, last_item.end_point + self.end, ).render(resource_registry, style, last_item, initial_point) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeArc.render`. """ # newline is intentionally missing here debug_stream.write(f"{self} resolved to ") return Arc( self.radii, self.rotation, self.large, self.sweep, last_item.end_point + self.end, ).render_debug( resource_registry, style, last_item, initial_point, debug_stream, pfx )
The rotation of the arc's major/minor axes relative to the coordinate frame.
var sweep : bool
-
Expand source code Browse git
class RelativeArc(NamedTuple): """ An elliptical arc path element. The arc is drawn from the end of the current path element to its specified end point using a number of parameters to determine how it is constructed. See: `PaintedPath.arc_relative` """ radii: Point """ The x- and y-radii of the arc. If `radii.x == radii.y` the arc will be circular. """ rotation: Number """The rotation of the arc's major/minor axes relative to the coordinate frame.""" large: bool """If True, sweep the arc over an angle greater than or equal to 180 degrees.""" sweep: bool """If True, the arc is swept in the positive angular direction.""" end: Point """The end point of the arc relative to the end of the previous path element.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of the resolved arc from the given start point.""" end_point = start + self.end arc = Arc( radii=self.radii, rotation=self.rotation, large=self.large, sweep=self.sweep, end=end_point, ) return arc.bounding_box(start) @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `BezierCurve`. """ return Arc( self.radii, self.rotation, self.large, self.sweep, last_item.end_point + self.end, ).render(resource_registry, style, last_item, initial_point) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeArc.render`. """ # newline is intentionally missing here debug_stream.write(f"{self} resolved to ") return Arc( self.radii, self.rotation, self.large, self.sweep, last_item.end_point + self.end, ).render_debug( resource_registry, style, last_item, initial_point, debug_stream, pfx )
If True, the arc is swept in the positive angular direction.
Methods
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of the resolved arc from the given start point.""" end_point = start + self.end arc = Arc( radii=self.radii, rotation=self.rotation, large=self.large, sweep=self.sweep, end=end_point, ) return arc.bounding_box(start)
Compute the bounding box of the resolved arc from the given start point.
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
@force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `BezierCurve`. """ return Arc( self.radii, self.rotation, self.large, self.sweep, last_item.end_point + self.end, ).render(resource_registry, style, last_item, initial_point)
Render this path element to its PDF representation.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
Returns
a tuple of
(str, new_last_item)
, wherenew_last_item
is a resolvedBezierCurve
. def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
@force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeArc.render`. """ # newline is intentionally missing here debug_stream.write(f"{self} resolved to ") return Arc( self.radii, self.rotation, self.large, self.sweep, last_item.end_point + self.end, ).render_debug( resource_registry, style, last_item, initial_point, debug_stream, pfx )
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
RelativeArc.render()
.
class RelativeBezierCurve (c1: Point,
c2: Point,
end: Point)-
Expand source code Browse git
class RelativeBezierCurve(NamedTuple): """ A cubic Bézier curve path element whose points are specified relative to the end point of the previous path element. See: `PaintedPath.curve_relative` """ c1: Point """ The curve's first control point relative to the end of the previous path element. """ c2: Point """ The curve's second control point relative to the end of the previous path element. """ end: Point """The curve's end point relative to the end of the previous path element.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this relative cubic Bézier curve. Args: start (Point): The starting point of the curve (i.e., the end of the previous path element). Returns: A tuple containing: - BoundingBox: the axis-aligned bounding box containing the entire curve. - Point: the end point of the curve. """ # Resolve absolute coordinates p0 = start p1 = start + self.c1 p2 = start + self.c2 p3 = start + self.end tx = [0, 1] + _cubic_bezier_critical_ts_1d(p0.x, p1.x, p2.x, p3.x) ty = [0, 1] + _cubic_bezier_critical_ts_1d(p0.y, p1.y, p2.y, p3.y) xs = [ _eval_cubic_bezier_1d(t, float(p0.x), float(p1.x), float(p2.x), float(p3.x)) for t in tx ] ys = [ _eval_cubic_bezier_1d(t, float(p0.y), float(p1.y), float(p2.y), float(p3.y)) for t in ty ] bbox = BoundingBox(min(xs), min(ys), max(xs), max(ys)) return bbox, p3 @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `BezierCurve`. """ # pylint: disable=unused-argument last_point = last_item.end_point c1 = last_point + self.c1 c2 = last_point + self.c2 end = last_point + self.end return ( _render_curve(c1, c2, end), BezierCurve(c1=c1, c2=c2, end=end), initial_point, ) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeBezierCurve.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
A cubic Bézier curve path element whose points are specified relative to the end point of the previous path element.
Ancestors
- builtins.tuple
Instance variables
var c1 : Point
-
Expand source code Browse git
class RelativeBezierCurve(NamedTuple): """ A cubic Bézier curve path element whose points are specified relative to the end point of the previous path element. See: `PaintedPath.curve_relative` """ c1: Point """ The curve's first control point relative to the end of the previous path element. """ c2: Point """ The curve's second control point relative to the end of the previous path element. """ end: Point """The curve's end point relative to the end of the previous path element.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this relative cubic Bézier curve. Args: start (Point): The starting point of the curve (i.e., the end of the previous path element). Returns: A tuple containing: - BoundingBox: the axis-aligned bounding box containing the entire curve. - Point: the end point of the curve. """ # Resolve absolute coordinates p0 = start p1 = start + self.c1 p2 = start + self.c2 p3 = start + self.end tx = [0, 1] + _cubic_bezier_critical_ts_1d(p0.x, p1.x, p2.x, p3.x) ty = [0, 1] + _cubic_bezier_critical_ts_1d(p0.y, p1.y, p2.y, p3.y) xs = [ _eval_cubic_bezier_1d(t, float(p0.x), float(p1.x), float(p2.x), float(p3.x)) for t in tx ] ys = [ _eval_cubic_bezier_1d(t, float(p0.y), float(p1.y), float(p2.y), float(p3.y)) for t in ty ] bbox = BoundingBox(min(xs), min(ys), max(xs), max(ys)) return bbox, p3 @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `BezierCurve`. """ # pylint: disable=unused-argument last_point = last_item.end_point c1 = last_point + self.c1 c2 = last_point + self.c2 end = last_point + self.end return ( _render_curve(c1, c2, end), BezierCurve(c1=c1, c2=c2, end=end), initial_point, ) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeBezierCurve.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
The curve's first control point relative to the end of the previous path element.
var c2 : Point
-
Expand source code Browse git
class RelativeBezierCurve(NamedTuple): """ A cubic Bézier curve path element whose points are specified relative to the end point of the previous path element. See: `PaintedPath.curve_relative` """ c1: Point """ The curve's first control point relative to the end of the previous path element. """ c2: Point """ The curve's second control point relative to the end of the previous path element. """ end: Point """The curve's end point relative to the end of the previous path element.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this relative cubic Bézier curve. Args: start (Point): The starting point of the curve (i.e., the end of the previous path element). Returns: A tuple containing: - BoundingBox: the axis-aligned bounding box containing the entire curve. - Point: the end point of the curve. """ # Resolve absolute coordinates p0 = start p1 = start + self.c1 p2 = start + self.c2 p3 = start + self.end tx = [0, 1] + _cubic_bezier_critical_ts_1d(p0.x, p1.x, p2.x, p3.x) ty = [0, 1] + _cubic_bezier_critical_ts_1d(p0.y, p1.y, p2.y, p3.y) xs = [ _eval_cubic_bezier_1d(t, float(p0.x), float(p1.x), float(p2.x), float(p3.x)) for t in tx ] ys = [ _eval_cubic_bezier_1d(t, float(p0.y), float(p1.y), float(p2.y), float(p3.y)) for t in ty ] bbox = BoundingBox(min(xs), min(ys), max(xs), max(ys)) return bbox, p3 @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `BezierCurve`. """ # pylint: disable=unused-argument last_point = last_item.end_point c1 = last_point + self.c1 c2 = last_point + self.c2 end = last_point + self.end return ( _render_curve(c1, c2, end), BezierCurve(c1=c1, c2=c2, end=end), initial_point, ) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeBezierCurve.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
The curve's second control point relative to the end of the previous path element.
var end : Point
-
Expand source code Browse git
class RelativeBezierCurve(NamedTuple): """ A cubic Bézier curve path element whose points are specified relative to the end point of the previous path element. See: `PaintedPath.curve_relative` """ c1: Point """ The curve's first control point relative to the end of the previous path element. """ c2: Point """ The curve's second control point relative to the end of the previous path element. """ end: Point """The curve's end point relative to the end of the previous path element.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this relative cubic Bézier curve. Args: start (Point): The starting point of the curve (i.e., the end of the previous path element). Returns: A tuple containing: - BoundingBox: the axis-aligned bounding box containing the entire curve. - Point: the end point of the curve. """ # Resolve absolute coordinates p0 = start p1 = start + self.c1 p2 = start + self.c2 p3 = start + self.end tx = [0, 1] + _cubic_bezier_critical_ts_1d(p0.x, p1.x, p2.x, p3.x) ty = [0, 1] + _cubic_bezier_critical_ts_1d(p0.y, p1.y, p2.y, p3.y) xs = [ _eval_cubic_bezier_1d(t, float(p0.x), float(p1.x), float(p2.x), float(p3.x)) for t in tx ] ys = [ _eval_cubic_bezier_1d(t, float(p0.y), float(p1.y), float(p2.y), float(p3.y)) for t in ty ] bbox = BoundingBox(min(xs), min(ys), max(xs), max(ys)) return bbox, p3 @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `BezierCurve`. """ # pylint: disable=unused-argument last_point = last_item.end_point c1 = last_point + self.c1 c2 = last_point + self.c2 end = last_point + self.end return ( _render_curve(c1, c2, end), BezierCurve(c1=c1, c2=c2, end=end), initial_point, ) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeBezierCurve.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
The curve's end point relative to the end of the previous path element.
Methods
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this relative cubic Bézier curve. Args: start (Point): The starting point of the curve (i.e., the end of the previous path element). Returns: A tuple containing: - BoundingBox: the axis-aligned bounding box containing the entire curve. - Point: the end point of the curve. """ # Resolve absolute coordinates p0 = start p1 = start + self.c1 p2 = start + self.c2 p3 = start + self.end tx = [0, 1] + _cubic_bezier_critical_ts_1d(p0.x, p1.x, p2.x, p3.x) ty = [0, 1] + _cubic_bezier_critical_ts_1d(p0.y, p1.y, p2.y, p3.y) xs = [ _eval_cubic_bezier_1d(t, float(p0.x), float(p1.x), float(p2.x), float(p3.x)) for t in tx ] ys = [ _eval_cubic_bezier_1d(t, float(p0.y), float(p1.y), float(p2.y), float(p3.y)) for t in ty ] bbox = BoundingBox(min(xs), min(ys), max(xs), max(ys)) return bbox, p3
Compute the bounding box of this relative cubic Bézier curve.
Args
start
:Point
- The starting point of the curve (i.e., the end of the previous path element).
Returns
A tuple containing: - BoundingBox: the axis-aligned bounding box containing the entire curve. - Point: the end point of the curve.
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
@force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `BezierCurve`. """ # pylint: disable=unused-argument last_point = last_item.end_point c1 = last_point + self.c1 c2 = last_point + self.c2 end = last_point + self.end return ( _render_curve(c1, c2, end), BezierCurve(c1=c1, c2=c2, end=end), initial_point, )
Render this path element to its PDF representation.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
Returns
a tuple of
(str, new_last_item)
, wherenew_last_item
is the resolvedBezierCurve
. def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
@force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeBezierCurve.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
RelativeBezierCurve.render()
.
class RelativeHorizontalLine (x: int | float | decimal.Decimal)
-
Expand source code Browse git
class RelativeHorizontalLine(NamedTuple): """ A path line element that takes its ordinate from the end of the previous element and computes its abscissa offset from the end of that element. See: `PaintedPath.horizontal_line_relative` """ x: Number """ The abscissa of the horizontal line's end point relative to the abscissa of the previous path element. """ def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of a relative horizontal line.""" end = Point(float(start.x) + float(self.x), start.y) bbox = BoundingBox.from_points([start, end]) return bbox, end @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `Line`. """ # pylint: disable=unused-argument end_point = Point(x=last_item.end_point.x + self.x, y=last_item.end_point.y) return _render_line(end_point), Line(end_point), initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeHorizontalLine.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
A path line element that takes its ordinate from the end of the previous element and computes its abscissa offset from the end of that element.
Ancestors
- builtins.tuple
Instance variables
var x : int | float | decimal.Decimal
-
Expand source code Browse git
class RelativeHorizontalLine(NamedTuple): """ A path line element that takes its ordinate from the end of the previous element and computes its abscissa offset from the end of that element. See: `PaintedPath.horizontal_line_relative` """ x: Number """ The abscissa of the horizontal line's end point relative to the abscissa of the previous path element. """ def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of a relative horizontal line.""" end = Point(float(start.x) + float(self.x), start.y) bbox = BoundingBox.from_points([start, end]) return bbox, end @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `Line`. """ # pylint: disable=unused-argument end_point = Point(x=last_item.end_point.x + self.x, y=last_item.end_point.y) return _render_line(end_point), Line(end_point), initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeHorizontalLine.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
The abscissa of the horizontal line's end point relative to the abscissa of the previous path element.
Methods
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of a relative horizontal line.""" end = Point(float(start.x) + float(self.x), start.y) bbox = BoundingBox.from_points([start, end]) return bbox, end
Compute the bounding box of a relative horizontal line.
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
@force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `Line`. """ # pylint: disable=unused-argument end_point = Point(x=last_item.end_point.x + self.x, y=last_item.end_point.y) return _render_line(end_point), Line(end_point), initial_point
Render this path element to its PDF representation.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
Returns
a tuple of
(str, new_last_item)
, wherenew_last_item
is the resolvedLine
. def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
@force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeHorizontalLine.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
RelativeHorizontalLine.render()
.
class RelativeLine (pt: Point)
-
Expand source code Browse git
class RelativeLine(NamedTuple): """ A path line element with an endpoint relative to the end of the previous element. This draws a straight line from the end point of the previous path element to the point specified by `last_item.end_point + pt`. The absolute coordinates of the end point are resolved during the rendering process. See: `PaintedPath.line_relative` """ pt: Point """The endpoint of the line relative to the previous path element.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of a relative line from the start point to the new end point.""" return BoundingBox.from_points([start, start + self.pt]), start + self.pt @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `Line`. """ # pylint: disable=unused-argument point = last_item.end_point + self.pt return _render_line(point), Line(point), initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeLine.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
A path line element with an endpoint relative to the end of the previous element.
This draws a straight line from the end point of the previous path element to the point specified by
last_item.end_point + pt
. The absolute coordinates of the end point are resolved during the rendering process.Ancestors
- builtins.tuple
Instance variables
var pt : Point
-
Expand source code Browse git
class RelativeLine(NamedTuple): """ A path line element with an endpoint relative to the end of the previous element. This draws a straight line from the end point of the previous path element to the point specified by `last_item.end_point + pt`. The absolute coordinates of the end point are resolved during the rendering process. See: `PaintedPath.line_relative` """ pt: Point """The endpoint of the line relative to the previous path element.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of a relative line from the start point to the new end point.""" return BoundingBox.from_points([start, start + self.pt]), start + self.pt @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `Line`. """ # pylint: disable=unused-argument point = last_item.end_point + self.pt return _render_line(point), Line(point), initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeLine.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
The endpoint of the line relative to the previous path element.
Methods
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of a relative line from the start point to the new end point.""" return BoundingBox.from_points([start, start + self.pt]), start + self.pt
Compute the bounding box of a relative line from the start point to the new end point.
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
@force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `Line`. """ # pylint: disable=unused-argument point = last_item.end_point + self.pt return _render_line(point), Line(point), initial_point
Render this path element to its PDF representation.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
Returns
a tuple of
(str, new_last_item)
, wherenew_last_item
is the resolvedLine
. def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
@force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeLine.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
RelativeLine.render()
.
class RelativeMove (pt: Point)
-
Expand source code Browse git
class RelativeMove(NamedTuple): """ A path move element with an end point relative to the end of the previous path element. If a path has been created but not yet painted, this will create a new subpath. See: `PaintedPath.move_relative` """ pt: Point """The offset by which to move.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """RelativeMove doesn't draw anything, so it has no bounding box.""" return BoundingBox.empty(), start + self.pt @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `Move` """ # pylint: disable=unused-argument point = last_item.end_point + self.pt return _render_move(point), Move(point), point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeMove.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
A path move element with an end point relative to the end of the previous path element.
If a path has been created but not yet painted, this will create a new subpath.
Ancestors
- builtins.tuple
Instance variables
var pt : Point
-
Expand source code Browse git
class RelativeMove(NamedTuple): """ A path move element with an end point relative to the end of the previous path element. If a path has been created but not yet painted, this will create a new subpath. See: `PaintedPath.move_relative` """ pt: Point """The offset by which to move.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """RelativeMove doesn't draw anything, so it has no bounding box.""" return BoundingBox.empty(), start + self.pt @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `Move` """ # pylint: disable=unused-argument point = last_item.end_point + self.pt return _render_move(point), Move(point), point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeMove.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
The offset by which to move.
Methods
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """RelativeMove doesn't draw anything, so it has no bounding box.""" return BoundingBox.empty(), start + self.pt
RelativeMove doesn't draw anything, so it has no bounding box.
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
@force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `Move` """ # pylint: disable=unused-argument point = last_item.end_point + self.pt return _render_move(point), Move(point), point
Render this path element to its PDF representation.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
Returns
a tuple of
(str, new_last_item)
, wherenew_last_item
is the resolvedMove
def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
@force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeMove.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
RelativeMove.render()
.
class RelativeQuadraticBezierCurve (ctrl: Point,
end: Point)-
Expand source code Browse git
class RelativeQuadraticBezierCurve(NamedTuple): """ A quadratic Bézier curve path element whose points are specified relative to the end point of the previous path element. See: `PaintedPath.quadratic_curve_relative` """ ctrl: Point """The curve's control point relative to the end of the previous path element.""" end: Point """The curve's end point relative to the end of the previous path element.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this relative quadratic Bézier curve.""" ctrl = start + self.ctrl end = start + self.end return QuadraticBezierCurve(ctrl=ctrl, end=end).bounding_box(start) @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `QuadraticBezierCurve`. """ last_point = last_item.end_point ctrl = last_point + self.ctrl end = last_point + self.end absolute = QuadraticBezierCurve(ctrl=ctrl, end=end) return absolute.render(resource_registry, style, last_item, initial_point) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeQuadraticBezierCurve.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write( f"{self} resolved to {resolved} " f"then to {resolved.to_cubic_curve(last_item.end_point)}\n" ) return rendered, resolved, initial_point
A quadratic Bézier curve path element whose points are specified relative to the end point of the previous path element.
Ancestors
- builtins.tuple
Instance variables
var ctrl : Point
-
Expand source code Browse git
class RelativeQuadraticBezierCurve(NamedTuple): """ A quadratic Bézier curve path element whose points are specified relative to the end point of the previous path element. See: `PaintedPath.quadratic_curve_relative` """ ctrl: Point """The curve's control point relative to the end of the previous path element.""" end: Point """The curve's end point relative to the end of the previous path element.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this relative quadratic Bézier curve.""" ctrl = start + self.ctrl end = start + self.end return QuadraticBezierCurve(ctrl=ctrl, end=end).bounding_box(start) @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `QuadraticBezierCurve`. """ last_point = last_item.end_point ctrl = last_point + self.ctrl end = last_point + self.end absolute = QuadraticBezierCurve(ctrl=ctrl, end=end) return absolute.render(resource_registry, style, last_item, initial_point) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeQuadraticBezierCurve.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write( f"{self} resolved to {resolved} " f"then to {resolved.to_cubic_curve(last_item.end_point)}\n" ) return rendered, resolved, initial_point
The curve's control point relative to the end of the previous path element.
var end : Point
-
Expand source code Browse git
class RelativeQuadraticBezierCurve(NamedTuple): """ A quadratic Bézier curve path element whose points are specified relative to the end point of the previous path element. See: `PaintedPath.quadratic_curve_relative` """ ctrl: Point """The curve's control point relative to the end of the previous path element.""" end: Point """The curve's end point relative to the end of the previous path element.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this relative quadratic Bézier curve.""" ctrl = start + self.ctrl end = start + self.end return QuadraticBezierCurve(ctrl=ctrl, end=end).bounding_box(start) @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `QuadraticBezierCurve`. """ last_point = last_item.end_point ctrl = last_point + self.ctrl end = last_point + self.end absolute = QuadraticBezierCurve(ctrl=ctrl, end=end) return absolute.render(resource_registry, style, last_item, initial_point) @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeQuadraticBezierCurve.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write( f"{self} resolved to {resolved} " f"then to {resolved.to_cubic_curve(last_item.end_point)}\n" ) return rendered, resolved, initial_point
The curve's end point relative to the end of the previous path element.
Methods
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this relative quadratic Bézier curve.""" ctrl = start + self.ctrl end = start + self.end return QuadraticBezierCurve(ctrl=ctrl, end=end).bounding_box(start)
Compute the bounding box of this relative quadratic Bézier curve.
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
@force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `QuadraticBezierCurve`. """ last_point = last_item.end_point ctrl = last_point + self.ctrl end = last_point + self.end absolute = QuadraticBezierCurve(ctrl=ctrl, end=end) return absolute.render(resource_registry, style, last_item, initial_point)
Render this path element to its PDF representation.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
Returns
a tuple of
(str, new_last_item)
, wherenew_last_item
is the resolvedQuadraticBezierCurve
. def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
@force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeQuadraticBezierCurve.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write( f"{self} resolved to {resolved} " f"then to {resolved.to_cubic_curve(last_item.end_point)}\n" ) return rendered, resolved, initial_point
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
RelativeQuadraticBezierCurve.render()
.
class RelativeVerticalLine (y: int | float | decimal.Decimal)
-
Expand source code Browse git
class RelativeVerticalLine(NamedTuple): """ A path line element that takes its abscissa from the end of the previous element and computes its ordinate offset from the end of that element. See: `PaintedPath.vertical_line_relative` """ y: Number """ The ordinate of the vertical line's end point relative to the ordinate of the previous path element. """ def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this relative vertical line.""" end = Point(start.x, float(start.y) + float(self.y)) bbox = BoundingBox.from_points([start, end]) return bbox, end @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `Line`. """ # pylint: disable=unused-argument end_point = Point(x=last_item.end_point.x, y=last_item.end_point.y + self.y) return _render_line(end_point), Line(end_point), initial_point # pylint: disable=unused-argument @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeVerticalLine.render`. """ rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
A path line element that takes its abscissa from the end of the previous element and computes its ordinate offset from the end of that element.
Ancestors
- builtins.tuple
Instance variables
var y : int | float | decimal.Decimal
-
Expand source code Browse git
class RelativeVerticalLine(NamedTuple): """ A path line element that takes its abscissa from the end of the previous element and computes its ordinate offset from the end of that element. See: `PaintedPath.vertical_line_relative` """ y: Number """ The ordinate of the vertical line's end point relative to the ordinate of the previous path element. """ def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this relative vertical line.""" end = Point(start.x, float(start.y) + float(self.y)) bbox = BoundingBox.from_points([start, end]) return bbox, end @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `Line`. """ # pylint: disable=unused-argument end_point = Point(x=last_item.end_point.x, y=last_item.end_point.y + self.y) return _render_line(end_point), Line(end_point), initial_point # pylint: disable=unused-argument @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeVerticalLine.render`. """ rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
The ordinate of the vertical line's end point relative to the ordinate of the previous path element.
Methods
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this relative vertical line.""" end = Point(start.x, float(start.y) + float(self.y)) bbox = BoundingBox.from_points([start, end]) return bbox, end
Compute the bounding box of this relative vertical line.
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
@force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `Line`. """ # pylint: disable=unused-argument end_point = Point(x=last_item.end_point.x, y=last_item.end_point.y + self.y) return _render_line(end_point), Line(end_point), initial_point
Render this path element to its PDF representation.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
Returns
a tuple of
(str, new_last_item)
, wherenew_last_item
is the resolvedLine
. def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
@force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RelativeVerticalLine.render`. """ rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
RelativeVerticalLine.render()
.
class Renderable (*args, **kwargs)
-
Expand source code Browse git
@runtime_checkable class Renderable(Protocol): """ Structural type for things that can render themselves into PDF operators and report a geometric bounding box. """ def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: "Renderable", initial_point: Point, ) -> tuple[str, "Renderable", Point]: ... def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: ...
Structural type for things that can render themselves into PDF operators and report a geometric bounding box.
Ancestors
- typing.Protocol
- typing.Generic
Methods
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: ...
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, 'Renderable', Point]-
Expand source code Browse git
def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: "Renderable", initial_point: Point, ) -> tuple[str, "Renderable", Point]: ...
class RoundedRectangle (org: Point,
size: Point,
corner_radii: Point)-
Expand source code Browse git
class RoundedRectangle(NamedTuple): """ A rectangle with rounded corners. See: `PaintedPath.rectangle` """ org: Point """The top-left corner of the rectangle.""" size: Point """The width and height of the rectangle.""" corner_radii: Point """The x- and y-radius of the corners.""" def _decompose(self) -> list[Renderable]: items = [] if (self.size.x == 0) and (self.size.y == 0): pass elif (self.size.x == 0) or (self.size.y == 0): items.append(Move(self.org)) items.append(Line(self.org + self.size)) items.append(Close()) elif (self.corner_radii.x == 0) or (self.corner_radii.y == 0): items.append(Rectangle(self.org, self.size)) else: x, y = self.org w, h = self.size rx, ry = self.corner_radii sign_width = (self.size.x >= 0) - (self.size.x < 0) sign_height = (self.size.y >= 0) - (self.size.y < 0) if abs(rx) > abs(w): rx = self.size.x if abs(ry) > abs(h): ry = self.size.y rx = sign_width * abs(rx) ry = sign_height * abs(ry) arc_rad = Point(rx, ry) items.append(Move(Point(x + rx, y))) items.append(Line(Point(x + w - rx, y))) items.append(Arc(arc_rad, 0, False, True, Point(x + w, y + ry))) items.append(Line(Point(x + w, y + h - ry))) items.append(Arc(arc_rad, 0, False, True, Point(x + w - rx, y + h))) items.append(Line(Point(x + rx, y + h))) items.append(Arc(arc_rad, 0, False, True, Point(x, y + h - ry))) items.append(Line(Point(x, y + ry))) items.append(Arc(arc_rad, 0, False, True, Point(x + rx, y))) items.append(Close()) return items def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this rounded rectangle by decomposing into primitives and merging their individual bounding boxes. """ bbox = BoundingBox.empty() current_point = start for item in self._decompose(): b, current_point = item.bounding_box(current_point) bbox = bbox.merge(b) return bbox, self.org @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `Line`. """ components = self._decompose() if not components: return "", last_item, initial_point render_list = [] for item in components: rendered, last_item, initial_point = item.render( resource_registry, style, last_item, initial_point ) render_list.append(rendered) return " ".join(render_list), Line(self.org), initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RoundedRectangle.render`. """ components = self._decompose() debug_stream.write(f"{self} resolved to:\n") if not components: debug_stream.write(pfx + " └─ nothing\n") return "", last_item, initial_point render_list = [] for item in components[:-1]: rendered, last_item, initial_point = item.render( resource_registry, style, last_item, initial_point ) debug_stream.write(pfx + f" ├─ {item}\n") render_list.append(rendered) rendered, last_item, initial_point = components[-1].render( resource_registry, style, last_item, initial_point ) debug_stream.write(pfx + f" └─ {components[-1]}\n") render_list.append(rendered) return " ".join(render_list), Line(self.org), initial_point
A rectangle with rounded corners.
Ancestors
- builtins.tuple
Instance variables
var corner_radii : Point
-
Expand source code Browse git
class RoundedRectangle(NamedTuple): """ A rectangle with rounded corners. See: `PaintedPath.rectangle` """ org: Point """The top-left corner of the rectangle.""" size: Point """The width and height of the rectangle.""" corner_radii: Point """The x- and y-radius of the corners.""" def _decompose(self) -> list[Renderable]: items = [] if (self.size.x == 0) and (self.size.y == 0): pass elif (self.size.x == 0) or (self.size.y == 0): items.append(Move(self.org)) items.append(Line(self.org + self.size)) items.append(Close()) elif (self.corner_radii.x == 0) or (self.corner_radii.y == 0): items.append(Rectangle(self.org, self.size)) else: x, y = self.org w, h = self.size rx, ry = self.corner_radii sign_width = (self.size.x >= 0) - (self.size.x < 0) sign_height = (self.size.y >= 0) - (self.size.y < 0) if abs(rx) > abs(w): rx = self.size.x if abs(ry) > abs(h): ry = self.size.y rx = sign_width * abs(rx) ry = sign_height * abs(ry) arc_rad = Point(rx, ry) items.append(Move(Point(x + rx, y))) items.append(Line(Point(x + w - rx, y))) items.append(Arc(arc_rad, 0, False, True, Point(x + w, y + ry))) items.append(Line(Point(x + w, y + h - ry))) items.append(Arc(arc_rad, 0, False, True, Point(x + w - rx, y + h))) items.append(Line(Point(x + rx, y + h))) items.append(Arc(arc_rad, 0, False, True, Point(x, y + h - ry))) items.append(Line(Point(x, y + ry))) items.append(Arc(arc_rad, 0, False, True, Point(x + rx, y))) items.append(Close()) return items def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this rounded rectangle by decomposing into primitives and merging their individual bounding boxes. """ bbox = BoundingBox.empty() current_point = start for item in self._decompose(): b, current_point = item.bounding_box(current_point) bbox = bbox.merge(b) return bbox, self.org @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `Line`. """ components = self._decompose() if not components: return "", last_item, initial_point render_list = [] for item in components: rendered, last_item, initial_point = item.render( resource_registry, style, last_item, initial_point ) render_list.append(rendered) return " ".join(render_list), Line(self.org), initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RoundedRectangle.render`. """ components = self._decompose() debug_stream.write(f"{self} resolved to:\n") if not components: debug_stream.write(pfx + " └─ nothing\n") return "", last_item, initial_point render_list = [] for item in components[:-1]: rendered, last_item, initial_point = item.render( resource_registry, style, last_item, initial_point ) debug_stream.write(pfx + f" ├─ {item}\n") render_list.append(rendered) rendered, last_item, initial_point = components[-1].render( resource_registry, style, last_item, initial_point ) debug_stream.write(pfx + f" └─ {components[-1]}\n") render_list.append(rendered) return " ".join(render_list), Line(self.org), initial_point
The x- and y-radius of the corners.
var org : Point
-
Expand source code Browse git
class RoundedRectangle(NamedTuple): """ A rectangle with rounded corners. See: `PaintedPath.rectangle` """ org: Point """The top-left corner of the rectangle.""" size: Point """The width and height of the rectangle.""" corner_radii: Point """The x- and y-radius of the corners.""" def _decompose(self) -> list[Renderable]: items = [] if (self.size.x == 0) and (self.size.y == 0): pass elif (self.size.x == 0) or (self.size.y == 0): items.append(Move(self.org)) items.append(Line(self.org + self.size)) items.append(Close()) elif (self.corner_radii.x == 0) or (self.corner_radii.y == 0): items.append(Rectangle(self.org, self.size)) else: x, y = self.org w, h = self.size rx, ry = self.corner_radii sign_width = (self.size.x >= 0) - (self.size.x < 0) sign_height = (self.size.y >= 0) - (self.size.y < 0) if abs(rx) > abs(w): rx = self.size.x if abs(ry) > abs(h): ry = self.size.y rx = sign_width * abs(rx) ry = sign_height * abs(ry) arc_rad = Point(rx, ry) items.append(Move(Point(x + rx, y))) items.append(Line(Point(x + w - rx, y))) items.append(Arc(arc_rad, 0, False, True, Point(x + w, y + ry))) items.append(Line(Point(x + w, y + h - ry))) items.append(Arc(arc_rad, 0, False, True, Point(x + w - rx, y + h))) items.append(Line(Point(x + rx, y + h))) items.append(Arc(arc_rad, 0, False, True, Point(x, y + h - ry))) items.append(Line(Point(x, y + ry))) items.append(Arc(arc_rad, 0, False, True, Point(x + rx, y))) items.append(Close()) return items def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this rounded rectangle by decomposing into primitives and merging their individual bounding boxes. """ bbox = BoundingBox.empty() current_point = start for item in self._decompose(): b, current_point = item.bounding_box(current_point) bbox = bbox.merge(b) return bbox, self.org @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `Line`. """ components = self._decompose() if not components: return "", last_item, initial_point render_list = [] for item in components: rendered, last_item, initial_point = item.render( resource_registry, style, last_item, initial_point ) render_list.append(rendered) return " ".join(render_list), Line(self.org), initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RoundedRectangle.render`. """ components = self._decompose() debug_stream.write(f"{self} resolved to:\n") if not components: debug_stream.write(pfx + " └─ nothing\n") return "", last_item, initial_point render_list = [] for item in components[:-1]: rendered, last_item, initial_point = item.render( resource_registry, style, last_item, initial_point ) debug_stream.write(pfx + f" ├─ {item}\n") render_list.append(rendered) rendered, last_item, initial_point = components[-1].render( resource_registry, style, last_item, initial_point ) debug_stream.write(pfx + f" └─ {components[-1]}\n") render_list.append(rendered) return " ".join(render_list), Line(self.org), initial_point
The top-left corner of the rectangle.
var size : Point
-
Expand source code Browse git
class RoundedRectangle(NamedTuple): """ A rectangle with rounded corners. See: `PaintedPath.rectangle` """ org: Point """The top-left corner of the rectangle.""" size: Point """The width and height of the rectangle.""" corner_radii: Point """The x- and y-radius of the corners.""" def _decompose(self) -> list[Renderable]: items = [] if (self.size.x == 0) and (self.size.y == 0): pass elif (self.size.x == 0) or (self.size.y == 0): items.append(Move(self.org)) items.append(Line(self.org + self.size)) items.append(Close()) elif (self.corner_radii.x == 0) or (self.corner_radii.y == 0): items.append(Rectangle(self.org, self.size)) else: x, y = self.org w, h = self.size rx, ry = self.corner_radii sign_width = (self.size.x >= 0) - (self.size.x < 0) sign_height = (self.size.y >= 0) - (self.size.y < 0) if abs(rx) > abs(w): rx = self.size.x if abs(ry) > abs(h): ry = self.size.y rx = sign_width * abs(rx) ry = sign_height * abs(ry) arc_rad = Point(rx, ry) items.append(Move(Point(x + rx, y))) items.append(Line(Point(x + w - rx, y))) items.append(Arc(arc_rad, 0, False, True, Point(x + w, y + ry))) items.append(Line(Point(x + w, y + h - ry))) items.append(Arc(arc_rad, 0, False, True, Point(x + w - rx, y + h))) items.append(Line(Point(x + rx, y + h))) items.append(Arc(arc_rad, 0, False, True, Point(x, y + h - ry))) items.append(Line(Point(x, y + ry))) items.append(Arc(arc_rad, 0, False, True, Point(x + rx, y))) items.append(Close()) return items def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this rounded rectangle by decomposing into primitives and merging their individual bounding boxes. """ bbox = BoundingBox.empty() current_point = start for item in self._decompose(): b, current_point = item.bounding_box(current_point) bbox = bbox.merge(b) return bbox, self.org @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `Line`. """ components = self._decompose() if not components: return "", last_item, initial_point render_list = [] for item in components: rendered, last_item, initial_point = item.render( resource_registry, style, last_item, initial_point ) render_list.append(rendered) return " ".join(render_list), Line(self.org), initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RoundedRectangle.render`. """ components = self._decompose() debug_stream.write(f"{self} resolved to:\n") if not components: debug_stream.write(pfx + " └─ nothing\n") return "", last_item, initial_point render_list = [] for item in components[:-1]: rendered, last_item, initial_point = item.render( resource_registry, style, last_item, initial_point ) debug_stream.write(pfx + f" ├─ {item}\n") render_list.append(rendered) rendered, last_item, initial_point = components[-1].render( resource_registry, style, last_item, initial_point ) debug_stream.write(pfx + f" └─ {components[-1]}\n") render_list.append(rendered) return " ".join(render_list), Line(self.org), initial_point
The width and height of the rectangle.
Methods
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """ Compute the bounding box of this rounded rectangle by decomposing into primitives and merging their individual bounding boxes. """ bbox = BoundingBox.empty() current_point = start for item in self._decompose(): b, current_point = item.bounding_box(current_point) bbox = bbox.merge(b) return bbox, self.org
Compute the bounding box of this rounded rectangle by decomposing into primitives and merging their individual bounding boxes.
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
@force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is a resolved `Line`. """ components = self._decompose() if not components: return "", last_item, initial_point render_list = [] for item in components: rendered, last_item, initial_point = item.render( resource_registry, style, last_item, initial_point ) render_list.append(rendered) return " ".join(render_list), Line(self.org), initial_point
Render this path element to its PDF representation.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
Returns
a tuple of
(str, new_last_item)
, wherenew_last_item
is a resolvedLine
. def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
@force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `RoundedRectangle.render`. """ components = self._decompose() debug_stream.write(f"{self} resolved to:\n") if not components: debug_stream.write(pfx + " └─ nothing\n") return "", last_item, initial_point render_list = [] for item in components[:-1]: rendered, last_item, initial_point = item.render( resource_registry, style, last_item, initial_point ) debug_stream.write(pfx + f" ├─ {item}\n") render_list.append(rendered) rendered, last_item, initial_point = components[-1].render( resource_registry, style, last_item, initial_point ) debug_stream.write(pfx + f" └─ {components[-1]}\n") render_list.append(rendered) return " ".join(render_list), Line(self.org), initial_point
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
RoundedRectangle.render()
.
class VerticalLine (y: int | float | decimal.Decimal)
-
Expand source code Browse git
class VerticalLine(NamedTuple): """ A path line element that takes its abscissa from the end of the previous element. See: `PaintedPath.vertical_line_to` """ y: Number """The ordinate of the vertical line's end point.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this vertical line.""" end = Point(start.x, float(self.y)) bbox = BoundingBox.from_points([start, end]) return bbox, end @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `Line`. """ # pylint: disable=unused-argument end_point = Point(x=last_item.end_point.x, y=float(self.y)) return _render_line(end_point), Line(end_point), initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `VerticalLine.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
A path line element that takes its abscissa from the end of the previous element.
Ancestors
- builtins.tuple
Instance variables
var y : int | float | decimal.Decimal
-
Expand source code Browse git
class VerticalLine(NamedTuple): """ A path line element that takes its abscissa from the end of the previous element. See: `PaintedPath.vertical_line_to` """ y: Number """The ordinate of the vertical line's end point.""" def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this vertical line.""" end = Point(start.x, float(self.y)) bbox = BoundingBox.from_points([start, end]) return bbox, end @force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `Line`. """ # pylint: disable=unused-argument end_point = Point(x=last_item.end_point.x, y=float(self.y)) return _render_line(end_point), Line(end_point), initial_point @force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `VerticalLine.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
The ordinate of the vertical line's end point.
Methods
def bounding_box(self,
start: Point) ‑> tuple[BoundingBox, Point]-
Expand source code Browse git
def bounding_box(self, start: Point) -> tuple[BoundingBox, Point]: """Compute the bounding box of this vertical line.""" end = Point(start.x, float(self.y)) bbox = BoundingBox.from_points([start, end]) return bbox, end
Compute the bounding box of this vertical line.
def render(self,
resource_registry: ResourceCatalog,
style: GraphicsStyle,
last_item: Renderable,
initial_point: Point) ‑> tuple[str, Renderable, Point]-
Expand source code Browse git
@force_nodocument def render( self, resource_registry: "ResourceCatalog", style: GraphicsStyle, last_item: Renderable, initial_point: Point, ) -> tuple[str, Renderable, Point]: """ Render this path element to its PDF representation. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command Returns: a tuple of `(str, new_last_item)`, where `new_last_item` is the resolved `Line`. """ # pylint: disable=unused-argument end_point = Point(x=last_item.end_point.x, y=float(self.y)) return _render_line(end_point), Line(end_point), initial_point
Render this path element to its PDF representation.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
Returns
a tuple of
(str, new_last_item)
, wherenew_last_item
is the resolvedLine
. def render_debug(self, resource_registry, style, last_item, initial_point, debug_stream, pfx)
-
Expand source code Browse git
@force_nodocument def render_debug( self, resource_registry, style, last_item, initial_point, debug_stream, pfx ): """ Render this path element to its PDF representation and produce debug information. Args: resource_registry (ResourceCatalog): the owner's graphics state dictionary registry. style (GraphicsStyle): the current resolved graphics style last_item: the previous path element. initial_point: last position set by a "M" or "m" command debug_stream (io.TextIO): the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr). pfx (str): the current debug output prefix string (only needed if emitting more than one line). Returns: The same tuple as `VerticalLine.render`. """ # pylint: disable=unused-argument rendered, resolved, initial_point = self.render( resource_registry, style, last_item, initial_point ) debug_stream.write(f"{self} resolved to {resolved}\n") return rendered, resolved, initial_point
Render this path element to its PDF representation and produce debug information.
Args
resource_registry
:ResourceCatalog
- the owner's graphics state dictionary registry.
style
:GraphicsStyle
- the current resolved graphics style
last_item
- the previous path element.
initial_point
- last position set by a "M" or "m" command
debug_stream
:io.TextIO
- the stream to which the debug output should be written. This is not guaranteed to be seekable (e.g. it may be stdout or stderr).
pfx
:str
- the current debug output prefix string (only needed if emitting more than one line).
Returns
The same tuple as
VerticalLine.render()
.