Source code for data_morph.bounds.bounding_box

"""Class for working with two-dimensional bounds."""

from __future__ import annotations

from collections.abc import Iterable
from numbers import Number

from ._utils import _validate_2d
from .interval import Interval


[docs] class BoundingBox: """ Class representing 2-dimensional range of numeric values. Parameters ---------- x_bounds, y_bounds : Interval | Iterable[numbers.Number] A 2-dimensional numeric iterable or an :class:`.Interval` object. inclusive : bool, default ``False`` Whether the bounds include the endpoints. Default is exclusive. If :class:`.Interval` objects are provided, their settings are used. """ def __init__( self, x_bounds: Interval | Iterable[Number], y_bounds: Interval | Iterable[Number], inclusive: Iterable[bool] = False, ) -> None: if x_bounds is None or y_bounds is None: raise ValueError('BoundingBox requires bounds for both dimensions.') if isinstance(inclusive, bool): inclusive = [inclusive] * 2 if not ( isinstance(inclusive, (tuple, list)) and len(inclusive) == 2 and all(isinstance(x, bool) for x in inclusive) ): raise ValueError( 'inclusive must be an iterable of 2 Boolean values' ' or a single Boolean value' ) self.x_bounds = ( x_bounds.clone() if isinstance(x_bounds, Interval) else Interval(x_bounds, inclusive[0]) ) """Interval: The bounds for the x direction.""" self.y_bounds = ( y_bounds.clone() if isinstance(y_bounds, Interval) else Interval(y_bounds, inclusive[1]) ) """Interval: The bounds for the y direction.""" def __contains__(self, value: Iterable[Number]) -> bool: """ Add support for using the ``in`` operator to check whether a two-dimensional point is in the bounding box. Parameters ---------- value : Iterable[numbers.Number] A two-dimensional point. Returns ------- bool Whether ``value`` is contained in the bounding box. """ x, y = _validate_2d(value, 'input') return x in self.x_bounds and y in self.y_bounds def __eq__(self, other: BoundingBox) -> bool: """ Check whether two :class:`.BoundingBox` objects are equivalent. Parameters ---------- other : BoundingBox A :class:`.BoundingBox` object. Returns ------- bool Whether the two :class:`.BoundingBox` objects are equivalent. """ if not isinstance(other, BoundingBox): raise TypeError('Equality is only defined between BoundingBox objects.') return self.x_bounds == other.x_bounds and self.y_bounds == other.y_bounds def __repr__(self) -> str: return '<BoundingBox>\n' f' x={self.x_bounds}' '\n' f' y={self.y_bounds}'
[docs] def adjust_bounds(self, x: Number | None = None, y: Number | None = None) -> None: """ Adjust bounding box range. Parameters ---------- x : numbers.Number, optional The amount to change the x bound range by (half will be applied to each end). y : numbers.Number, optional The amount to change the y bound range by (half will be applied to each end). See Also -------- :meth:`.Interval.adjust_bounds` : Method that performs the adjustment. """ if x: self.x_bounds.adjust_bounds(x) if y: self.y_bounds.adjust_bounds(y)
[docs] def align_aspect_ratio(self) -> None: """Align the aspect ratio to 1:1.""" x_range, y_range = self.range diff = x_range - y_range if diff < 0: self.adjust_bounds(x=-diff) elif diff > 0: self.adjust_bounds(y=diff)
@property def aspect_ratio(self) -> Number: """ Calculate the aspect ratio of the bounding box. Returns ------- numbers.Number The range in the x direction divided by the range in the y direction. """ x_range, y_range = self.range return x_range / y_range
[docs] def clone(self) -> BoundingBox: """ Clone this instance. Returns ------- BoundingBox A new :class:`.BoundingBox` instance with the same bounds. """ return BoundingBox( self.x_bounds.clone(), self.y_bounds.clone(), )
@property def range(self) -> Iterable[Number]: """ Calculate the range (width) of the bounding box in each direction. Returns ------- Iterable[numbers.Number] The range covered by the x and y bounds, respectively. """ return self.x_bounds.range, self.y_bounds.range