"""Class and custom data array for representing a scattering matrix wave port."""fromtypingimportOptional,Unionimportnumpyasnpimportpydantic.v1aspdfrom....components.baseimportcached_propertyfrom....components.data.data_arrayimportFreqDataArray,FreqModeDataArrayfrom....components.data.monitor_dataimportModeSolverDatafrom....components.data.sim_dataimportSimulationDatafrom....components.geometry.baseimportBoxfrom....components.grid.gridimportGridfrom....components.monitorimportFieldMonitor,ModeSolverMonitorfrom....components.simulationimportSimulationfrom....components.source.fieldimportModeSource,ModeSpecfrom....components.source.timeimportGaussianPulsefrom....components.typesimportBound,Direction,FreqArrayfrom....exceptionsimportValidationErrorfrom...microwaveimport(CurrentIntegralTypes,ImpedanceCalculator,VoltageIntegralTypes,)from...modeimportModeSolverfrom.base_terminalimportAbstractTerminalPort
[docs]classWavePort(AbstractTerminalPort,Box):"""Class representing a single wave port"""direction:Direction=pd.Field(...,title="Direction",description="'+' or '-', defining which direction is considered 'input'.",)mode_spec:ModeSpec=pd.Field(ModeSpec(),title="Mode Specification",description="Parameters to feed to mode solver which determine modes measured by monitor.",)mode_index:pd.NonNegativeInt=pd.Field(0,title="Mode Index",description="Index into the collection of modes returned by mode solver. "" Specifies which mode to inject using this source. ""If larger than ``mode_spec.num_modes``, ""``num_modes`` in the solver will be set to ``mode_index + 1``.",)voltage_integral:Optional[VoltageIntegralTypes]=pd.Field(None,title="Voltage Integral",description="Definition of voltage integral used to compute voltage and the characteristic impedance.",)current_integral:Optional[CurrentIntegralTypes]=pd.Field(None,title="Current Integral",description="Definition of current integral used to compute current and the characteristic impedance.",)@cached_propertydefinjection_axis(self):"""Injection axis of the port."""returnself.size.index(0.0)@cached_propertydef_field_monitor_name(self)->str:"""Return the name of the :class:`.FieldMonitor` associated with this port."""returnf"{self.name}_field"@cached_propertydef_mode_monitor_name(self)->str:"""Return the name of the :class:`.ModeMonitor` associated with this port."""returnf"{self.name}_mode"
[docs]defto_source(self,source_time:GaussianPulse,snap_center:float=None)->ModeSource:"""Create a mode source from the wave port."""center=list(self.center)ifsnap_center:center[self.injection_axis]=snap_centerreturnModeSource(center=center,size=self.size,source_time=source_time,mode_spec=self.mode_spec,mode_index=self.mode_index,direction=self.direction,name=self.name,)
[docs]defto_field_monitors(self,freqs:FreqArray,snap_center:float=None,grid:Grid=None)->list[FieldMonitor]:"""Field monitor to compute port voltage and current."""center=list(self.center)ifsnap_center:center[self.injection_axis]=snap_centerfield_mon=FieldMonitor(center=center,size=self.size,freqs=freqs,name=self._field_monitor_name,colocate=False,)return[field_mon]
[docs]defto_mode_solver_monitor(self,freqs:FreqArray)->ModeSolverMonitor:"""Mode solver monitor to compute modes that will be used to compute characteristic impedances."""mode_mon=ModeSolverMonitor(center=self.center,size=self.size,freqs=freqs,name=self._mode_monitor_name,colocate=False,mode_spec=self.mode_spec,direction=self.direction,)returnmode_mon
[docs]defto_mode_solver(self,simulation:Simulation,freqs:FreqArray)->ModeSolver:"""Helper to create a :class:`.ModeSolver` instance."""mode_solver=ModeSolver(simulation=simulation,plane=self.geometry,mode_spec=self.mode_spec,freqs=freqs,direction=self.direction,colocate=False,)returnmode_solver
[docs]defcompute_voltage(self,sim_data:SimulationData)->FreqDataArray:"""Helper to compute voltage across the port."""field_monitor=sim_data[self._field_monitor_name]returnself.voltage_integral.compute_voltage(field_monitor)
[docs]defcompute_current(self,sim_data:SimulationData)->FreqDataArray:"""Helper to compute current flowing through the port."""field_monitor=sim_data[self._field_monitor_name]returnself.current_integral.compute_current(field_monitor)
[docs]defcompute_port_impedance(self,sim_mode_data:Union[SimulationData,ModeSolverData])->FreqModeDataArray:"""Helper to compute impedance of port. The port impedance is computed from the transmission line mode, which should be TEM or at least quasi-TEM."""impedance_calc=ImpedanceCalculator(voltage_integral=self.voltage_integral,current_integral=self.current_integral)ifisinstance(sim_mode_data,SimulationData):mode_solver_data=sim_mode_data[self._mode_monitor_name]else:mode_solver_data=sim_mode_data# Filter out unwanted modes to reduce impedance computation effortmode_solver_data=mode_solver_data._isel(mode_index=[self.mode_index])impedance_array=impedance_calc.compute_impedance(mode_solver_data)returnimpedance_array
@staticmethoddef_within_port_bounds(path_bounds:Bound,port_bounds:Bound)->bool:"""Helper to check if one bounding box is completely within the other."""path_min=np.array(path_bounds[0])path_max=np.array(path_bounds[1])bound_min=np.array(port_bounds[0])bound_max=np.array(port_bounds[1])return(bound_min<=path_min).all()and(bound_max>=path_max).all()@pd.validator("voltage_integral","current_integral")def_validate_path_integrals_within_port(cls,val,values):"""Raise ``ValidationError`` when the supplied path integrals are not within the port bounds."""center=values["center"]size=values["size"]box=Box(center=center,size=size)ifvalandnotWavePort._within_port_bounds(val.bounds,box.bounds):raiseValidationError(f"'{cls.__name__}' must be setup with all path integrals defined within the bounds "f"of the port. Path bounds are '{val.bounds}', but port bounds are '{box.bounds}'.")returnval@pd.validator("current_integral",always=True)def_check_voltage_or_current(cls,val,values):"""Raise validation error if both ``voltage_integral`` and ``current_integral`` were not provided."""ifnotvalues.get("voltage_integral")andnotval:raiseValidationError("At least one of 'voltage_integral' or 'current_integral' must be provided.")returnval
[docs]@pd.validator("current_integral",always=True)defvalidate_current_integral_sign(cls,val,values):""" Validate that the sign of ``current_integral`` matches the port direction. """ifvalisNone:returnvaldirection=values.get("direction")name=values.get("name")ifval.sign!=direction:raiseValidationError(f"'current_integral' sign must match the '{name}' direction '{direction}'.")returnval