Source code for data_morph.data.dataset

"""Class representing a dataset for morphing."""

from __future__ import annotations

from numbers import Number

import matplotlib.pyplot as plt
import pandas as pd
from matplotlib.axes import Axes

from ..bounds.bounding_box import BoundingBox
from ..bounds.interval import Interval
from ..plotting.style import plot_with_custom_style


[docs] class Dataset: """ Class for representing a starting dataset and bounds. .. plot:: :caption: Upon creation, these bounds are automatically calculated. Use :meth:`plot` to generate this visualization. from data_morph.data.loader import DataLoader _ = DataLoader.load_dataset('panda').plot(show_bounds=True) Parameters ---------- name : str The name to use for the dataset. df : pandas.DataFrame DataFrame containing columns x and y. scale : numbers.Number, optional The factor to scale the data by (can be used to speed up morphing). Values in the data's x and y columns will be divided by this value. See Also -------- :class:`.DataLoader` Utility for creating :class:`Dataset` objects from CSV files. """ _REQUIRED_COLUMNS = ('x', 'y') def __init__( self, name: str, df: pd.DataFrame, scale: Number | None = None, ) -> None: self.df: pd.DataFrame = self._validate_data(df).pipe(self._scale_data, scale) """pandas.DataFrame: DataFrame containing columns x and y.""" self.name: str = name """str: The name to use for the dataset.""" self.data_bounds: BoundingBox = self._derive_data_bounds() """BoundingBox: The bounds of the data.""" self.morph_bounds: BoundingBox = self._derive_morphing_bounds() """BoundingBox: The limits for the morphing process.""" self.plot_bounds: BoundingBox = self._derive_plotting_bounds() """BoundingBox: The bounds to use when plotting the morphed data.""" def __repr__(self) -> str: return f'<{self.__class__.__name__} name={self.name} scaled={self._scaled}>' def _derive_data_bounds(self) -> BoundingBox: """ Derive bounds based on the data. Returns ------- BoundingBox The bounds of the data. """ return BoundingBox( *[ Interval([self.df[dim].min(), self.df[dim].max()], inclusive=False) for dim in self._REQUIRED_COLUMNS ] ) def _derive_morphing_bounds(self) -> BoundingBox: """ Derive morphing bounds based on the data. Returns ------- BoundingBox The bounds of the morphing process. """ # TODO: range * 0.2 is still a bit arbitrary (need to take into account density at the edges) # could also make this a parameter to __init__() morph_bounds = self.data_bounds.clone() x_offset, y_offset = (offset * 0.2 for offset in self.data_bounds.range) morph_bounds.adjust_bounds(x=x_offset, y=y_offset) return morph_bounds def _derive_plotting_bounds(self) -> BoundingBox: """ Derive plotting bounds based on the morphing bounds. Returns ------- BoundingBox The bounds of the plot. """ # TODO: range * 0.2 is still a bit arbitrary (need to take into account density at the edges) # could also make this a parameter to __init__() x_offset, y_offset = (offset * 0.2 for offset in self.data_bounds.range) plot_bounds = self.morph_bounds.clone() plot_bounds.adjust_bounds(x=x_offset, y=y_offset) plot_bounds.align_aspect_ratio() return plot_bounds def _scale_data(self, df: pd.DataFrame, scale: Number) -> pd.DataFrame: """ Apply scaling to the data. Parameters ---------- df : pandas.DataFrame The data to scale. scale : numbers.Number, optional The factor to scale the data by (can be used to speed up morphing). Values in the data's x and y columns will be divided by this value. Returns ------- pandas.DataFrame The scaled data. """ if scale is None: self._scaled = False return df if isinstance(scale, bool) or not isinstance(scale, Number): raise TypeError('scale must be a numeric value.') if not scale: raise ValueError('scale must be non-zero.') scaled_df = df.assign(x=df.x.div(scale), y=df.y.div(scale)) self._scaled = True return scaled_df def _validate_data(self, data: pd.DataFrame) -> pd.DataFrame: """ Validate the data. Parameters ---------- data : pandas.DataFrame DataFrame for morphing. Returns ------- pandas.DataFrame DataFrame provided it contains columns x and y. """ required = set(self._REQUIRED_COLUMNS) missing_columns = required.difference(data.columns) if missing_columns: case_insensitive_missing = missing_columns.difference( data.columns.str.lower() ) if case_insensitive_missing: raise ValueError( 'Columns "x" and "y" are required for datasets. The provided ' 'dataset is missing the following column(s): ' f"{', '.join(sorted(missing_columns))}." ) data = data.rename(columns={col.upper(): col for col in missing_columns}) return data
[docs] @plot_with_custom_style def plot( self, ax: Axes | None = None, show_bounds: bool = True, title: str = 'default' ) -> Axes: """ Plot the dataset and its bounds. Parameters ---------- ax : matplotlib.axes.Axes, optional An optional :class:`~matplotlib.axes.Axes` object to plot on. show_bounds : bool, default ``True`` Whether to plot the bounds of the dataset. title : str, optional Title to use for the plot. The default will call ``str()`` on the Dataset. Pass ``None`` to leave the plot untitled. Returns ------- matplotlib.axes.Axes The :class:`~matplotlib.axes.Axes` object containing the plot. """ if not ax: fig, ax = plt.subplots(layout='constrained') fig.get_layout_engine().set(w_pad=0.2, h_pad=0.2) ax.axis('equal') ax.scatter(self.df.x, self.df.y, s=2, color='black') ax.set(xlabel='', ylabel='', title=self if title == 'default' else title) if show_bounds: scale_base = 85 # data bounds x_offset = self.data_bounds.x_bounds.range / scale_base y_offset = self.data_bounds.y_bounds.range / scale_base data_rectangle = [ self.data_bounds.x_bounds[0] - x_offset, self.data_bounds.y_bounds[0] - y_offset, ] ax.add_patch( plt.Rectangle( data_rectangle, width=self.data_bounds.x_bounds.range + x_offset * 2, height=self.data_bounds.y_bounds.range + y_offset * 2, ec='blue', linewidth=2, fill=False, ) ) ax.text( (self.df.x.max() + self.df.x.min()) / 2, self.df.y.max() + self.data_bounds.y_bounds.range / scale_base, 'DATA BOUNDS', color='blue', va='bottom', ha='center', ) # morph bounds morph_rectangle = [ self.morph_bounds.x_bounds[0], self.morph_bounds.y_bounds[0], ] ax.add_patch( plt.Rectangle( morph_rectangle, width=self.morph_bounds.x_bounds.range, height=self.morph_bounds.y_bounds.range, ec='red', linewidth=2, fill=False, ) ) ax.text( *morph_rectangle, ' MORPH BOUNDS', color='red', va='bottom', ha='left' ) # plot bounds plot_rectangle = [ self.plot_bounds.x_bounds[0], self.plot_bounds.y_bounds[0], ] ax.add_patch( plt.Rectangle( plot_rectangle, width=self.plot_bounds.x_bounds.range, height=self.plot_bounds.y_bounds.range, ec='#7CA1CC', linewidth=2, fill=False, ) ) ax.text( *plot_rectangle, ' PLOT BOUNDS', color='#7CA1CC', va='bottom', ha='left' ) ax.autoscale() return ax