Source code for gdsfactory.routing.route_quad

"""Route for electrical based on phidl.routing.route_quad."""

from __future__ import annotations

import numpy as np
import numpy.typing as npt

import gdsfactory as gf
from gdsfactory.typings import Port


def _get_rotated_basis(
    theta: float,
) -> tuple[npt.NDArray[np.float64], npt.NDArray[np.float64]]:
    """Returns basis vectors rotated CCW by theta (in degrees)."""
    theta = np.radians(theta)
    e1 = np.array([np.cos(theta), np.sin(theta)])
    e2 = np.array([-1 * np.sin(theta), np.cos(theta)])
    return e1, e2


[docs] def route_quad( component: gf.Component, port1: Port, port2: Port, width1: float | None = None, width2: float | None = None, layer: gf.typings.LayerSpec = "M1", manhattan_target_step: float | None = None, ) -> None: """Routes a basic quadrilateral polygon directly between two ports. Args: component: Component to add the route to. port1: Port to start route. port2 : Port objects to end route. width1: Width of quadrilateral at ports. If None, uses port widths. width2: Width of quadrilateral at ports. If None, uses port widths. layer: Layer to put the route on. manhattan_target_step: if not none, min step to manhattanize the polygon .. plot:: :include-source: import gdsfactory as gf c = gf.Component() pad1 = c << gf.components.pad(size=(50, 50)) pad2 = c << gf.components.pad(size=(10, 10)) pad2.dmovex(100) pad2.dmovey(50) gf.routing.route_quad( c, pad1.ports["e2"], pad2.ports["e4"], width1=None, width2=None, ) c.plot() """ def get_port_edges( port: Port, width: float ) -> tuple[npt.NDArray[np.float64], npt.NDArray[np.float64]]: _, e1 = _get_rotated_basis(port.orientation) pt1 = port.dcenter + e1 * width / 2 pt2 = port.dcenter - e1 * width / 2 return pt1, pt2 if width1 is None: width1 = port1.dwidth if width2 is None: width2 = port2.dwidth vertices = np.array(get_port_edges(port1, width1) + get_port_edges(port2, width2)) center = np.mean(vertices, axis=0) displacements = vertices - center # sort vertices by angle from center of quadrilateral to make convex polygon angles = np.array([np.arctan2(disp[0], disp[1]) for disp in displacements]) sorted_vertices = [ vert for _, vert in sorted(zip(angles, vertices), key=lambda x: x[0]) ] if manhattan_target_step: component.add_polygon( sorted_vertices, layer=layer, ) else: component.add_polygon(points=sorted_vertices, layer=layer)
if __name__ == "__main__": from gdsfactory.components import pad c = gf.Component() pad1 = c << pad(size=(50, 50)) pad2 = c << pad(size=(10, 10)) pad2.movex(100) pad2.movey(50) route_quad( c, pad1.ports["e2"], pad2.ports["e4"], width1=None, width2=None, manhattan_target_step=0.1, ) # c = gf.Component("route") # pad1 = c << gf.components.pad(size=(50, 50)) # pad2 = c << gf.components.pad(size=(10, 10)) # pad2.dmovex(100) # pad2.dmovey(50) # route_gnd = c << route_quad( # pad1.ports["e2"], # pad2.ports["e4"], # width1=None, # width2=None, # manhattan_min_step=0.1, # ) c.show() # test_manhattan_route_quad()