Source code for data_morph.shapes.circles.rings

"""Rings shape."""

from __future__ import annotations

from typing import TYPE_CHECKING

import numpy as np

from ...plotting.style import plot_with_custom_style
from ..bases.shape import Shape
from .circle import Circle

if TYPE_CHECKING:
    from numbers import Number

    from matplotlib.axes import Axes

    from ..data.dataset import Dataset


[docs] class Rings(Shape): """ Class representing rings comprising three concentric circles. .. plot:: :scale: 75 :caption: This shape is generated using the panda dataset. from data_morph.data.loader import DataLoader from data_morph.shapes.circles import Rings _ = Rings(DataLoader.load_dataset('panda')).plot() Parameters ---------- dataset : Dataset The starting dataset to morph into other shapes. See Also -------- Circle : The individual rings are represented as circles. """ def __init__(self, dataset: Dataset) -> None: self.circles: list[Circle] = [ Circle(dataset, radius) for radius in self._derive_radii(dataset) ] """The individual rings represented by :class:`Circle` objects.""" self._centers = np.array([circle.center for circle in self.circles]) self._radii = np.array([circle.radius for circle in self.circles]) def __repr__(self) -> str: return self._recursive_repr('circles') @staticmethod def _derive_radii(dataset: Dataset) -> np.ndarray: """ Derive the radii for the circles in the rings. Parameters ---------- dataset : Dataset The starting dataset to morph into. Returns ------- np.ndarray The radii for the circles in the rings. """ stdev = (min(dataset.data_bounds.range) + min(dataset.morph_bounds.range)) / 4 return np.linspace(stdev, 0, 3, endpoint=False)
[docs] def distance(self, x: Number, y: Number) -> float: """ Calculate the minimum absolute distance between any of this shape's circles' edges and a point (x, y). Parameters ---------- x, y : numbers.Number Coordinates of a point in 2D space. Returns ------- float The minimum absolute distance between any of this shape's circles' edges and the point (x, y). """ point = np.array([x, y]) return np.min( np.abs(np.linalg.norm(self._centers - point, axis=1) - self._radii) )
[docs] @plot_with_custom_style def plot(self, ax: Axes | None = None) -> Axes: """ Plot the shape. Parameters ---------- ax : matplotlib.axes.Axes, optional An optional :class:`~matplotlib.axes.Axes` object to plot on. Returns ------- matplotlib.axes.Axes The :class:`~matplotlib.axes.Axes` object containing the plot. """ for circle in self.circles: ax = circle.plot(ax) return ax