Source code for prismo.monitors.base

"""
Base classes for electromagnetic field monitors.

This module defines the base Monitor class and related utilities for
monitoring and recording field data during FDTD simulations.
"""

from abc import ABC, abstractmethod
from typing import Optional

import numpy as np

from prismo.core.fields import ElectromagneticFields
from prismo.core.grid import YeeGrid


[docs] class Monitor(ABC): """ Abstract base class for all field monitors. Parameters ---------- center : Tuple[float, float, float] Physical coordinates of the monitor center (x, y, z) in meters. size : Tuple[float, float, float] Physical dimensions of the monitor region (Lx, Ly, Lz) in meters. For point monitors, use (0, 0, 0). name : str, optional Name of the monitor for identification. """
[docs] def __init__( self, center: tuple[float, float, float], size: tuple[float, float, float], name: Optional[str] = None, ): self.center = center self.size = size self.name = name or f"{self.__class__.__name__}_{id(self)}" # Set during initialization by the simulation self._grid: Optional[YeeGrid] = None self._monitor_region: dict[str, np.ndarray] = {}
[docs] def initialize(self, grid: YeeGrid) -> None: """ Initialize the monitor on a specific grid. Parameters ---------- grid : YeeGrid The grid on which to initialize the monitor. """ self._grid = grid self._setup_monitor_region()
def _setup_monitor_region(self) -> None: """ Set up the monitor region in grid coordinates. This method computes the grid indices where the monitor records data. """ if self._grid is None: raise RuntimeError("Monitor must be initialized with a grid first") # Convert physical coordinates to grid indices x_min, x_max, y_min, y_max, z_min, z_max = self._compute_monitor_region() # Store monitor region indices for each field component self._monitor_region = { "Ex": self._grid.get_component_indices( "Ex", x_min, x_max, y_min, y_max, z_min, z_max ), "Ey": self._grid.get_component_indices( "Ey", x_min, x_max, y_min, y_max, z_min, z_max ), "Ez": self._grid.get_component_indices( "Ez", x_min, x_max, y_min, y_max, z_min, z_max ), "Hx": self._grid.get_component_indices( "Hx", x_min, x_max, y_min, y_max, z_min, z_max ), "Hy": self._grid.get_component_indices( "Hy", x_min, x_max, y_min, y_max, z_min, z_max ), "Hz": self._grid.get_component_indices( "Hz", x_min, x_max, y_min, y_max, z_min, z_max ), } def _compute_monitor_region(self) -> tuple[int, int, int, int, int, int]: """ Compute the grid index bounds for the monitor region. Returns ------- Tuple[int, int, int, int, int, int] Grid index bounds (x_min, x_max, y_min, y_max, z_min, z_max). """ if self._grid is None: raise RuntimeError("Monitor must be initialized with a grid first") # Get physical half-sizes half_Lx = self.size[0] / 2 half_Ly = self.size[1] / 2 half_Lz = self.size[2] / 2 # Convert physical coordinates to grid indices x_min, y_min, z_min = self._grid.point_to_index( ( self.center[0] - half_Lx, self.center[1] - half_Ly, self.center[2] - half_Lz, ) ) x_max, y_max, z_max = self._grid.point_to_index( ( self.center[0] + half_Lx, self.center[1] + half_Ly, self.center[2] + half_Lz, ) ) # Ensure we have at least one cell for point monitors if x_min == x_max: x_max = x_min + 1 if y_min == y_max: y_max = y_min + 1 if z_min == z_max and self._grid.is_3d: z_max = z_min + 1 return x_min, x_max, y_min, y_max, z_min, z_max
[docs] @abstractmethod def update(self, fields: ElectromagneticFields, time: float, dt: float) -> None: """ Record field data at the current time step. Parameters ---------- fields : ElectromagneticFields Electromagnetic fields to record. time : float Current simulation time in seconds. dt : float Time step in seconds. """ pass