Skip to content

Elements

The specific elements on the calendar plot.

Includes x-axis, y-axis, and their settings, as well as the calendar events.

CalendarEvent dataclass

Something on the calendar.

Plots rectangles on matplotlib axis.

Parameters:

Name Type Description Default
day int

The day of the week. 0 is Monday.

required
start float

The start hour of the event.

required
end float | None

The end hour of the event.

None
duration float | None

The duration of the event. Only used if end is None.

None
days int

The number of days the event spans. Default is 1.

1

Examples:

Plot event from calendar data

calendar_data = CalendarData(day=0, start=0, end=2.5)
event = CalendarEvent.from_calendar_data(calendar_data=calendar_data, cmap=...)
event.plot(ax=ax)

Plot a single calendar event from vocab

event = CalendarEvent.from_vocab("00 01")
event.plot(ax=ax)
Source code in latent_calendar/plot/elements.py
@dataclass
class CalendarEvent:
    """Something on the calendar.

    Plots rectangles on matplotlib axis.

    Args:
        day: The day of the week. 0 is Monday.
        start: The start hour of the event.
        end: The end hour of the event.
        duration: The duration of the event. Only used if end is None.
        days: The number of days the event spans. Default is 1.

    Examples:
        Plot event from calendar data

        ```python
        calendar_data = CalendarData(day=0, start=0, end=2.5)
        event = CalendarEvent.from_calendar_data(calendar_data=calendar_data, cmap=...)
        event.plot(ax=ax)
        ```

        Plot a single calendar event from vocab

        ```python
        event = CalendarEvent.from_vocab("00 01")
        event.plot(ax=ax)
        ```

    """

    day: int
    start: float
    end: float | None = None
    duration: float | None = None
    days: int = 1

    def __post_init__(self) -> None:
        if self.day not in range(DAYS_IN_WEEK):
            raise ValueError("Day must be between 0 and 6")

        if self.end is None and self.duration is None:
            raise ValueError("Either end or duration must be provided")

        if self.end is not None and self.duration is not None:
            raise ValueError("Only one of end or duration can be provided")

        if self.end is None:
            self.end = self.start + (self.duration / 60)
            self.duration = None

        if self.days not in range(1, DAYS_IN_WEEK + 1):
            raise ValueError("Days must be between 1 and 7")

    @classmethod
    def from_calendar_data(
        cls,
        calendar_data: CalendarData,
    ) -> "CalendarEvent":
        return cls(
            day=calendar_data.day,
            start=calendar_data.start,
            end=calendar_data.end,
        )

    @classmethod
    def from_vocab(
        cls,
        vocab: str,
        duration: float = 60.0,
    ) -> "CalendarEvent":
        """Constructor from vocab string in order to plot on an axis.

        Args:
            vocab: The vocab string.
            duration: The duration of the event.

        Example:
            Plot on an axis

            ```python
            event = CalendarEvent.from_vocab("00 01")
            event.plot(ax=ax)
            ```

            Plot a two and half hour window

            ```python
            event = CalendarEvent.from_vocab("00 01", hours=2.5)
            event.plot(ax=ax)
            ```

        """
        day, hour = get_day_hour(vocab=vocab)

        return cls(
            day=day,
            start=hour,
            end=hour + (duration / 60),
        )

    @property
    def multiday_tour(self) -> bool:
        if self.end == HOURS_IN_DAY:
            return False

        return self.end % HOURS_IN_DAY < self.start

    @property
    def multiweek_tour(self) -> bool:
        return self.day + self.days > DAYS_IN_WEEK

    def _cap_event_at_week_end(self) -> None:
        self.days = DAYS_IN_WEEK - self.day

    def _create_next_week_event(self) -> "CalendarEvent":
        return CalendarEvent(
            day=0,
            start=self.start,
            end=self.end,
            days=self.days - DAYS_IN_WEEK + self.day,
        )

    def _cap_event_at_midnight(self) -> None:
        self.end = min(HOURS_IN_DAY, self.end)

    def _create_next_day_event(self) -> "CalendarEvent":
        """In the case of tour going into the next day, this is the next item."""
        return CalendarEvent(
            day=(self.day + 1) % DAYS_IN_WEEK,
            start=0,
            end=self.end % HOURS_IN_DAY,
            days=self.days,
        )

    def separate_events(self) -> list["CalendarEvent"]:
        """Return list of events that represent the one event across different days.

        Examples:
            A single event that goes from 23:00 to 01:00 will be split into two events.

            ```python
            event = CalendarEvent(day=0, start=23, duration=2 * 60)

            events = event.separate_events()
            ```

        """
        event = replace(self)
        events = [event]

        if event.multiday_tour:
            events.append(event._create_next_day_event())
            event._cap_event_at_midnight()

        for event in events:
            if not event.multiweek_tour:
                continue

            events.append(event._create_next_week_event())
            event._cap_event_at_week_end()

        return events

    def _create_matplotlib_rectangle(
        self, monday_start: bool, lw, fill: bool, linestyle, fillcolor, alpha, **kwargs
    ) -> plt.Rectangle:
        """Create a rectangle matplotlib instance from the event."""
        height = self.end - self.start
        assert height > 0.0, (
            f"The rectangle doesn't have positive height. Hour start {self.start} > Hour end {self.end}"
        )

        x = self.day if monday_start else (self.day + 1) % DAYS_IN_WEEK
        rect_kwargs = {
            "xy": [x, self.start],
            "width": self.days,
            "height": height,
        }

        rect_kwargs["edgecolor"] = "black"
        rect_kwargs["lw"] = lw
        rect_kwargs["fill"] = fill
        rect_kwargs["linestyle"] = linestyle
        rect_kwargs["facecolor"] = fillcolor
        rect_kwargs["alpha"] = alpha

        rect_kwargs.update(kwargs)

        return plt.Rectangle(**rect_kwargs)

    def plot(
        self,
        ax: plt.Axes,
        monday_start: bool = True,
        lw: float = 0.1,
        fill: bool = True,
        linestyle=None,
        fillcolor=None,
        alpha=None,
        **kwargs,
    ) -> None:
        """Put the CalendarEvent instance onto an axis.

        Options for kwargs [here](https://matplotlib.org/stable/api/_as_gen/matplotlib.patches.Rectangle.html).

        Args:
            ax: Axis to plot on
            monday_start: Whether to start the week on Monday or Sunday.
            lw: The line width of the event.
            fill: Whether to fill the event.
            linestyle: The line style of the event.
            fillcolor: The color of the event.
            alpha: The alpha of the event.
            kwargs: Addtional kwargs for the Patch instances or to override.

        """
        for event in self.separate_events():
            rectangle = event._create_matplotlib_rectangle(
                monday_start=monday_start,
                lw=lw,
                fill=fill,
                linestyle=linestyle,
                fillcolor=fillcolor,
                alpha=alpha,
                **kwargs,
            )
            if "label" in kwargs:
                kwargs.pop("label")

            ax.add_patch(rectangle)

from_vocab(vocab, duration=60.0) classmethod

Constructor from vocab string in order to plot on an axis.

Parameters:

Name Type Description Default
vocab str

The vocab string.

required
duration float

The duration of the event.

60.0
Example

Plot on an axis

event = CalendarEvent.from_vocab("00 01")
event.plot(ax=ax)

Plot a two and half hour window

event = CalendarEvent.from_vocab("00 01", hours=2.5)
event.plot(ax=ax)
Source code in latent_calendar/plot/elements.py
@classmethod
def from_vocab(
    cls,
    vocab: str,
    duration: float = 60.0,
) -> "CalendarEvent":
    """Constructor from vocab string in order to plot on an axis.

    Args:
        vocab: The vocab string.
        duration: The duration of the event.

    Example:
        Plot on an axis

        ```python
        event = CalendarEvent.from_vocab("00 01")
        event.plot(ax=ax)
        ```

        Plot a two and half hour window

        ```python
        event = CalendarEvent.from_vocab("00 01", hours=2.5)
        event.plot(ax=ax)
        ```

    """
    day, hour = get_day_hour(vocab=vocab)

    return cls(
        day=day,
        start=hour,
        end=hour + (duration / 60),
    )

plot(ax, monday_start=True, lw=0.1, fill=True, linestyle=None, fillcolor=None, alpha=None, **kwargs)

Put the CalendarEvent instance onto an axis.

Options for kwargs here.

Parameters:

Name Type Description Default
ax Axes

Axis to plot on

required
monday_start bool

Whether to start the week on Monday or Sunday.

True
lw float

The line width of the event.

0.1
fill bool

Whether to fill the event.

True
linestyle

The line style of the event.

None
fillcolor

The color of the event.

None
alpha

The alpha of the event.

None
kwargs

Addtional kwargs for the Patch instances or to override.

{}
Source code in latent_calendar/plot/elements.py
def plot(
    self,
    ax: plt.Axes,
    monday_start: bool = True,
    lw: float = 0.1,
    fill: bool = True,
    linestyle=None,
    fillcolor=None,
    alpha=None,
    **kwargs,
) -> None:
    """Put the CalendarEvent instance onto an axis.

    Options for kwargs [here](https://matplotlib.org/stable/api/_as_gen/matplotlib.patches.Rectangle.html).

    Args:
        ax: Axis to plot on
        monday_start: Whether to start the week on Monday or Sunday.
        lw: The line width of the event.
        fill: Whether to fill the event.
        linestyle: The line style of the event.
        fillcolor: The color of the event.
        alpha: The alpha of the event.
        kwargs: Addtional kwargs for the Patch instances or to override.

    """
    for event in self.separate_events():
        rectangle = event._create_matplotlib_rectangle(
            monday_start=monday_start,
            lw=lw,
            fill=fill,
            linestyle=linestyle,
            fillcolor=fillcolor,
            alpha=alpha,
            **kwargs,
        )
        if "label" in kwargs:
            kwargs.pop("label")

        ax.add_patch(rectangle)

separate_events()

Return list of events that represent the one event across different days.

Examples:

A single event that goes from 23:00 to 01:00 will be split into two events.

event = CalendarEvent(day=0, start=23, duration=2 * 60)

events = event.separate_events()
Source code in latent_calendar/plot/elements.py
def separate_events(self) -> list["CalendarEvent"]:
    """Return list of events that represent the one event across different days.

    Examples:
        A single event that goes from 23:00 to 01:00 will be split into two events.

        ```python
        event = CalendarEvent(day=0, start=23, duration=2 * 60)

        events = event.separate_events()
        ```

    """
    event = replace(self)
    events = [event]

    if event.multiday_tour:
        events.append(event._create_next_day_event())
        event._cap_event_at_midnight()

    for event in events:
        if not event.multiweek_tour:
            continue

        events.append(event._create_next_week_event())
        event._cap_event_at_week_end()

    return events

DayLabeler dataclass

Day of the week axis.

This is typically the x-axis.

Parameters:

Name Type Description Default
day_start int

The day to start the plot at.

0
days_of_week list[str]

The names of the days of the week.

create_default_days()
rotation float | None

The rotation of the day labels.

45
display bool

Whether to display the day labels.

True
Source code in latent_calendar/plot/elements.py
@dataclass
class DayLabeler:
    """Day of the week axis.

    This is typically the x-axis.

    Args:
        day_start: The day to start the plot at.
        days_of_week: The names of the days of the week.
        rotation: The rotation of the day labels.
        display: Whether to display the day labels.

    """

    day_start: int = 0
    days_of_week: list[str] = field(default_factory=create_default_days)
    rotation: float | None = 45
    display: bool = True

    def __post_init__(self) -> None:
        if self.day_start not in range(DAYS_IN_WEEK):
            msg = "'day_start' value must be 0: Monday or 6: Sunday"
            raise ValueError(msg)

    @property
    def day_labels(self) -> list[str]:
        """What is added to the plot. If this is display, empty ticks."""
        if not self.display:
            return ["" for _ in range(DAYS_IN_WEEK)]

        return self.days_of_week[self.day_start :] + self.days_of_week[: self.day_start]

    def create_labels(self, ax: plt.Axes, axis: str = "x") -> None:
        """Create the labels for the plot."""
        getattr(ax, f"set_{axis}lim")(0, DAYS_IN_WEEK)
        getattr(ax, f"set_{axis}ticks")(
            [i + 0.5 for i in range(DAYS_IN_WEEK)],
            self.day_labels,
            rotation=self.rotation,
        )

        if axis == "y":
            ax.invert_yaxis()

day_labels property

What is added to the plot. If this is display, empty ticks.

create_labels(ax, axis='x')

Create the labels for the plot.

Source code in latent_calendar/plot/elements.py
def create_labels(self, ax: plt.Axes, axis: str = "x") -> None:
    """Create the labels for the plot."""
    getattr(ax, f"set_{axis}lim")(0, DAYS_IN_WEEK)
    getattr(ax, f"set_{axis}ticks")(
        [i + 0.5 for i in range(DAYS_IN_WEEK)],
        self.day_labels,
        rotation=self.rotation,
    )

    if axis == "y":
        ax.invert_yaxis()

DisplaySettings dataclass

Small wrapper to hold the display settings in the plots.

Parameters:

Name Type Description Default
x bool

Whether to x axis the plot.

True
y bool

Whether to y axis the plot.

True
Source code in latent_calendar/plot/elements.py
@dataclass
class DisplaySettings:
    """Small wrapper to hold the display settings in the plots.

    Args:
        x: Whether to x axis the plot.
        y: Whether to y axis the plot.

    """

    x: bool = True
    y: bool = True

GridLines dataclass

Grid lines between the calendar for the plot.

Parameters:

Name Type Description Default
dow bool

Whether to add day of week grid lines.

False
hour bool

Whether to add hour grid lines.

False
color str

The color of the grid lines.

'black'
linestyle str

The style of the grid lines.

'--'
alpha float

The alpha of the grid lines.

0.2
Source code in latent_calendar/plot/elements.py
@dataclass
class GridLines:
    """Grid lines between the calendar for the plot.

    Args:
        dow: Whether to add day of week grid lines.
        hour: Whether to add hour grid lines.
        color: The color of the grid lines.
        linestyle: The style of the grid lines.
        alpha: The alpha of the grid lines.

    """

    dow: bool = False
    hour: bool = False
    color: str = "black"
    linestyle: str = "--"
    alpha: float = 0.2

    def configure_grid(self, ax: plt.Axes) -> None:
        if self.dow:
            for dow in range(DAYS_IN_WEEK):
                ax.axvline(
                    x=dow + 1,
                    color=self.color,
                    linestyle=self.linestyle,
                    alpha=self.alpha,
                )

        if self.hour:
            for hour in range(HOURS_IN_DAY):
                ax.axhline(
                    y=hour + 1,
                    color=self.color,
                    linestyle=self.linestyle,
                    alpha=self.alpha,
                )

TimeLabeler dataclass

This is time of day and all its settings in the plot.

This is typically the y-axis.

Parameters:

Name Type Description Default
hour_formatter HourFormatter

The formatter for the hour labels.

HourFormatter()
start int

The hour to start the plot at.

0
stride int

The number of hours to skip between ticks.

2
display bool

Whether to display the hour labels.

True
rotation float | None

The rotation of the hour labels.

0
Source code in latent_calendar/plot/elements.py
@dataclass
class TimeLabeler:
    """This is time of day and all its settings in the plot.

    This is typically the y-axis.

    Args:
        hour_formatter: The formatter for the hour labels.
        start: The hour to start the plot at.
        stride: The number of hours to skip between ticks.
        display: Whether to display the hour labels.
        rotation: The rotation of the hour labels.

    """

    hour_formatter: HourFormatter = field(default_factory=HourFormatter)
    start: int = 0
    stride: int = 2
    display: bool = True
    rotation: float | None = 0

    def get_hours(self) -> tuple[list[int], list[str]]:
        return range(HOURS_IN_DAY + 1)[:: self.stride]

    def create_labels(self, ax: plt.Axes, axis: str = "y") -> None:
        """Create the hour labels on the plot ax."""
        if axis not in {"x", "y"}:
            raise ValueError("Only supported for the x and y.")

        hours = self.get_hours()
        hour_name_func = self.label if self.display else self.empty_label
        hour_names = hour_name_func(hours)

        getattr(ax, f"set_{axis}ticks")(hours, hour_names, rotation=self.rotation)
        getattr(ax, f"set_{axis}lim")(0, HOURS_IN_DAY)

        if axis == "y":
            ax.invert_yaxis()

    def label(self, hrs: list[int]) -> list[str]:
        return [self.hour_formatter(hr) for hr in hrs]

    def empty_label(self, hrs: list[int]) -> list[str]:
        return ["" for _ in range(len(hrs))]

create_labels(ax, axis='y')

Create the hour labels on the plot ax.

Source code in latent_calendar/plot/elements.py
def create_labels(self, ax: plt.Axes, axis: str = "y") -> None:
    """Create the hour labels on the plot ax."""
    if axis not in {"x", "y"}:
        raise ValueError("Only supported for the x and y.")

    hours = self.get_hours()
    hour_name_func = self.label if self.display else self.empty_label
    hour_names = hour_name_func(hours)

    getattr(ax, f"set_{axis}ticks")(hours, hour_names, rotation=self.rotation)
    getattr(ax, f"set_{axis}lim")(0, HOURS_IN_DAY)

    if axis == "y":
        ax.invert_yaxis()

Comments