Path and CrossSection#

You can create a Path in gdsfactory and extrude it with an arbitrary CrossSection.

Let us create a path:

  • Create a blank Path.

  • Append points to the Path by either using the built-in functions (arc(), straight(), euler() …) or by providing your own lists of points.

  • Specify CrossSection with layers and offsets.

  • Extrude Path with a CrossSection to create a Component with the path polygons in it.

import matplotlib.pyplot as plt
import numpy as np

import gdsfactory as gf

Path#

The first step is to generate the list of points we want the path to follow. Let us now start out by creating a blank Path and using the built-in functions to make a few smooth turns.

p1 = gf.path.straight(length=5)

# This creates a curved path segment using an Euler bend profile,
# which is a curve with a continuously changing radius designed to minimize light loss. This specific bend turns by 45 degrees.
# By setting use_eff=False, you are telling the function to ignore complex calculations and instead create a simpler bend with a constant, user-specified radius.
p2 = gf.path.euler(radius=5, angle=45, p=0.5, use_eff=False)

# The + operator is used to concatenate the two paths.
# It takes the second path (p2) and appends it to the end of the first path (p1), ensuring a smooth, continuous transition.
p = p1 + p2
f = p.plot()
../_images/a93a214a7df47fda01907f95f205dc108b8ec5d74c41e582aeca9d1d43a0d282.png
p1 = gf.path.straight(length=5)
p2 = gf.path.euler(radius=5, angle=45, p=0.5, use_eff=False)
p = p2 + p1
f = p.plot()
../_images/b49d0a47dfd4c221b0a6ca031a5858b72bba5c9280176fee53ac56322da036d8.png
# Note: -angle rotations correspond to a clockwise turn.
P = gf.Path()
P += gf.path.arc(radius=10, angle=90)  # Circular arc.
P += gf.path.straight(length=10)  # Straight section.
P += gf.path.euler(radius=3, angle=-90)  # Euler bend (aka "racetrack" curve).
P += gf.path.straight(length=40)
P += gf.path.arc(radius=8, angle=-45)
P += gf.path.straight(length=10)
P += gf.path.arc(radius=8, angle=45)
P += gf.path.straight(length=10)

f = P.plot()
../_images/22f3bf77b522a80591fb982f2699c71d0ce442f2a2ba6e01b6ba93361a42ad16.png
p2 = P.copy().rotate(45)
f = p2.plot()
../_images/82d135e88383bd250a535ca9ebe6f7dba08bc822e54e0329d08d44350c092c27.png
P.points - p2.points
array([[  0.        ,   0.        ],
       [  0.07775818,  -0.18109421],
       [  0.16015338,  -0.36012627],
       [  0.24713097,  -0.53697746],
       [  0.33863327,  -0.71153054],
       [  0.43459961,  -0.88366975],
       [  0.53496636,  -1.05328096],
       [  0.63966697,  -1.2202517 ],
       [  0.74863202,  -1.38447127],
       [  0.86178925,  -1.54583076],
       [  0.97906364,  -1.7042232 ],
       [  1.10037742,  -1.85954356],
       [  1.22565016,  -2.01168884],
       [  1.35479879,  -2.16055818],
       [  1.48773767,  -2.30605285],
       [  1.62437867,  -2.44807638],
       [  1.76463118,  -2.58653461],
       [  1.9084022 ,  -2.72133573],
       [  2.0555964 ,  -2.85239035],
       [  2.20611618,  -2.97961158],
       [  2.35986175,  -3.10291506],
       [  2.51673115,  -3.22221903],
       [  2.67662037,  -3.33744439],
       [  2.83942339,  -3.44851474],
       [  3.00503227,  -3.55535642],
       [  3.1733372 ,  -3.6578986 ],
       [  3.34422657,  -3.75607328],
       [  3.51758709,  -3.84981537],
       [  3.69330379,  -3.93906271],
       [  3.87126017,  -4.02375613],
       [  4.05133823,  -4.10383946],
       [  4.23341856,  -4.1792596 ],
       [  4.41738044,  -4.24996655],
       [  4.60310189,  -4.31591343],
       [  4.79045976,  -4.3770565 ],
       [  4.97932982,  -4.43335522],
       [  5.16958684,  -4.48477228],
       [  5.36110467,  -4.53127356],
       [  5.55375631,  -4.57282824],
       [  5.74741403,  -4.60940877],
       [  5.94194941,  -4.64099089],
       [  6.13723348,  -4.66755366],
       [  6.33313674,  -4.68907946],
       [  6.52952929,  -4.70555403],
       [  6.72628092,  -4.71696644],
       [  6.92326117,  -4.72330911],
       [  7.12033942,  -4.72457786],
       [  7.317385  ,  -4.72077183],
       [  7.51426725,  -4.71189355],
       [  7.71085564,  -4.69794891],
       [  7.90701981,  -4.67894715],
       [  8.10262968,  -4.65490087],
       [  8.29755556,  -4.62582602],
       [  8.4916682 ,  -4.59174187],
       [  8.6848389 ,  -4.55267103],
       [  8.87693955,  -4.50863939],
       [  9.0678428 ,  -4.45967617],
       [  9.25742205,  -4.40581381],
       [  9.44555161,  -4.34708805],
       [  9.63210673,  -4.28353781],
       [  9.81696372,  -4.21520523],
       [ 10.        ,  -4.14213562],
       [ 17.07106781,  -1.21320344],
       [ 17.22259744,  -1.15062977],
       [ 17.3745295 ,  -1.0890412 ],
       [ 17.52724719,  -1.02943054],
       [ 17.68109502,  -0.9728054 ],
       [ 17.83635884,  -0.92019408],
       [ 17.99324531,  -0.87264941],
       [ 18.15186066,  -0.83125013],
       [ 18.31218874,  -0.79709882],
       [ 18.47406876,  -0.77131591],
       [ 18.63717281,  -0.75502883],
       [ 18.80098407,  -0.74935572],
       [ 18.98113405,  -0.75643383],
       [ 19.16017334,  -0.77762453],
       [ 19.33699811,  -0.81279716],
       [ 19.51051817,  -0.86173488],
       [ 19.67966373,  -0.92413597],
       [ 19.84339193,  -0.9996157 ],
       [ 20.00069333,  -1.08770872],
       [ 20.15059814,  -1.18787191],
       [ 20.29218212,  -1.29948772],
       [ 20.42457237,  -1.42186801],
       [ 20.53639293,  -1.54171156],
       [ 20.6402082 ,  -1.66856024],
       [ 20.73644339,  -1.80125797],
       [ 20.82566384,  -1.93877567],
       [ 20.90854811,  -2.08020737],
       [ 20.98586445,  -2.22476202],
       [ 21.05845073,  -2.37175194],
       [ 21.12719755,  -2.5205788 ],
       [ 21.19303416,  -2.67071762],
       [ 21.25691665,  -2.82169951],
       [ 21.31981802,  -2.97309339],
       [ 33.03554677, -31.25736464],
       [ 33.1091836 , -31.44370627],
       [ 33.17668408, -31.63235747],
       [ 33.23797592, -31.82311621],
       [ 33.29299349, -32.01577822],
       [ 33.34167789, -32.21013721],
       [ 33.38397696, -32.40598505],
       [ 33.41984543, -32.60311201],
       [ 33.44924488, -32.80130701],
       [ 33.47214384, -33.00035782],
       [ 33.48851777, -33.20005129],
       [ 33.49834914, -33.40017358],
       [ 33.50162744, -33.60051039],
       [ 33.49834914, -33.80084721],
       [ 33.48851777, -34.0009695 ],
       [ 33.47214384, -34.20066296],
       [ 33.44924488, -34.39971377],
       [ 33.41984543, -34.59790877],
       [ 33.38397696, -34.79503574],
       [ 33.34167789, -34.99088357],
       [ 33.29299349, -35.18524256],
       [ 33.23797592, -35.37790458],
       [ 33.17668408, -35.56866332],
       [ 33.1091836 , -35.75731451],
       [ 33.03554677, -35.94365614],
       [ 30.10661458, -43.01472396],
       [ 30.03297776, -43.20106559],
       [ 29.96547728, -43.38971678],
       [ 29.90418544, -43.58047552],
       [ 29.84916786, -43.77313754],
       [ 29.80048347, -43.96749653],
       [ 29.75818439, -44.16334436],
       [ 29.72231592, -44.36047132],
       [ 29.69291647, -44.55866633],
       [ 29.67001752, -44.75771713],
       [ 29.65364359, -44.9574106 ],
       [ 29.64381221, -45.15753289],
       [ 29.64053392, -45.35786971],
       [ 29.64381221, -45.55820652],
       [ 29.65364359, -45.75832881],
       [ 29.67001752, -45.95802228],
       [ 29.69291647, -46.15707309],
       [ 29.72231592, -46.35526809],
       [ 29.75818439, -46.55239505],
       [ 29.80048347, -46.74824289],
       [ 29.84916786, -46.94260187],
       [ 29.90418544, -47.13526389],
       [ 29.96547728, -47.32602263],
       [ 30.03297776, -47.51467382],
       [ 30.10661458, -47.70101546],
       [ 33.03554677, -54.77208327]])

You can also modify our path in the same ways as any other gdsfactory object:

  • Manipulation with move(), rotate(), mirror(), etc

  • Accessing properties like xmin, y, center, bbox, etc

P.movey(10)
P.xmin = 20
f = P.plot()
../_images/fc66d6569aeae0e229cbf699881f901eb401fc781624e5da8c57cbfc44b2ecc9.png

You can also check the length of the curve with the length() method:

P.length()
105.341

CrossSection#

Now that you have got your path defined, the next step is to define the cross-section of the path. To do this, you can create a blank CrossSection and add whatever cross-sections you want to it. You can then combine the Path and the CrossSection using the gf.path.extrude() function to generate a component:

Option 1: Single layer and width cross-section#

The simplest option is to just set the cross-section to be a constant width by passing a number to extrude() like so:

# Extrude the Path and the cross-section.
# The extrude function converts a 1D path into a 2D shape by giving it a specified width.

c = gf.path.extrude(P, layer=(1, 0), width=1.5)
c.plot()
../_images/3f4470409c589d6a06563bf44bdb9d82e28eae02aa3a8edb66c1e08f964a09c8.png

Option 2: Arbitrary Cross-section#

You can also extrude an arbitrary cross_section.

Now, what if we want a more complicated straight? For instance, in some photonic applications it is helpful to have a shallow etch that appears on either side of the straight (often called a trench or sleeve). Additionally, it might be nice to have a port on either end of the center section so we can snap other geometries to it. Let us try adding something like that in:

p = gf.path.straight()

# The code first defines three gf.Section objects, each representing a part of the total cross-section.
# s0: The central core section. It is 1 µm wide, centered at an offset of 0, and is on layer (1, 0).
# s1: A side section. It is 2 µm wide, its center is offset by +2 µm from the main centerline, and it is on layer (2, 0).
# s2: Another side section, identical to s1 but offset by -2 µm.
s0 = gf.Section(width=1, offset=0, layer=(1, 0), port_names=("in", "out"))
s1 = gf.Section(width=2, offset=2, layer=(2, 0))
s2 = gf.Section(width=2, offset=-2, layer=(2, 0))
x = gf.CrossSection(sections=(s0, s1, s2))

c = gf.path.extrude(p, cross_section=x)
c.draw_ports()
c.plot()
../_images/bbcff2164721406f6d95e284234d044cc4180ae119ebda175f0036fe5785e5e9.png

If you add more ports to a cross-section it also exposes its ports.

p = gf.path.straight()

# Add a few "sections" to the cross-section.
s0 = gf.Section(width=1, offset=0, layer=(1, 0), port_names=("in", "out"))
s1 = gf.Section(width=2, offset=2, layer=(2, 0), port_names=("e1", "e2"))
s2 = gf.Section(width=2, offset=-2, layer=(2, 0))
x = gf.CrossSection(sections=(s0, s1, s2))

c = gf.path.extrude(p, cross_section=x)
c.draw_ports()
c.plot()
../_images/0486a038f5b841760e96c01fc8c6d2e37a867d117ad47d6ed139b9d131fd5236.png
p = gf.path.arc() # A 1D path in the shape of a 90-degree circular arc is created. This defines the centerline for the extrusion.

# Combine the Path and the cross-section.
b = gf.path.extrude(p, cross_section=x)
b.plot()
../_images/5280ccd027e465203e11add5a8b948eaf64763a24f3dca111085952184a64589.png

⚠️ Warning! for GS routing. You need to add a centered port for routing to work correctly.

p = gf.path.straight()

# Add GS routing sections.
# GS routing is a specific auto-routing algorithm used in gdsfactory to create smooth, low-loss waveguide connections between component ports.
s0 = gf.Section(width=2, offset=0, layer=(2, 0), port_names=("g1", "g2"))
s1 = gf.Section(width=2, offset=4, layer=(2, 0))
x = gf.CrossSection(sections=(s0, s1), radius=8)
c = gf.path.extrude(p, cross_section=x)
pad = c
c_copy = c.copy()
c_copy.draw_ports()
c_copy.plot()
../_images/7ba51180bc06d55365591fa3a605119fd3d3d66b0571b8f45453cdb2029fbed0.png
# Do not do this for GS routing. See solution below.
c2 = gf.Component()
pad1 = c2 << pad
pad2 = c2 << pad
pad2.move((100, 100))


# The gf.routing.route_bundle function is a powerful auto-router for creating multiple, parallel waveguide connections.
# [pad1.ports["g2"]]: A list of the starting ports for the routes.
# [pad2.ports["g1"]]: A list of the ending ports for the routes.
# sort_ports=True: An option that helps the router find the optimal, non-crossing paths when routing multiple waveguides.
# bend='bend_euler': Specifies that any curves in the route should be smooth Euler bends.
gf.routing.route_bundle(c2, [pad1.ports["g2"]], [pad2.ports["g1"]], cross_section=x, sort_ports=True, bend='bend_euler')
c2.plot()
/home/runner/work/gdsfactory/gdsfactory/gdsfactory/components/bends/bend_euler.py:108: UserWarning: {'width': 2.0} ignored for cross_section 'xs_375bea09'
  x = gf.get_cross_section(cross_section, width=width or x.width)
/home/runner/work/gdsfactory/gdsfactory/gdsfactory/components/waveguides/straight.py:31: UserWarning: {'width': 2.0} ignored for cross_section 'xs_375bea09'
  x = gf.get_cross_section(cross_section, width=width)
../_images/c70bdc39cae47433a7947b53d059ad781bc89c17dd51266b0d640a05dfb2115f.png

For GSG routing it works well because the port is at the center. GSG routing is a specific auto-routing algorithm used in gdsfactory to create smooth, S-bend-like waveguide connections between component ports.

p = gf.path.straight()

# Add a few "sections" to the cross-section
g = gf.Section(width=2, offset=0, layer=(2, 0), port_names=("e1", "e2"), port_types=('electrical', 'electrical'))
s0 = gf.Section(width=2, offset=-4, layer=(2, 0))
s1 = gf.Section(width=2, offset=4, layer=(2, 0))
x = gf.CrossSection(sections=(g, s0, s1), radius=8)
c = gf.path.extrude(p, cross_section=x)
c_copy = c.copy()
c_copy.draw_ports()
c_copy.plot()
../_images/1d703f48fd5edf8c3b58bb4adcb59a83a0ad9a5da1f7e303ac9ad71c37aa5aef.png
c2 = gf.Component()
pad1 = c2 << c
pad2 = c2 << c
pad2.move((100, 100))
gf.routing.route_bundle(c2, [pad1.ports["e2"]], [pad2.ports["e1"]], cross_section=x, port_type='electrical')
c2.plot()
/home/runner/work/gdsfactory/gdsfactory/gdsfactory/components/bends/bend_euler.py:108: UserWarning: {'width': 2.0} ignored for cross_section 'xs_f4227c9f'
  x = gf.get_cross_section(cross_section, width=width or x.width)
/home/runner/work/gdsfactory/gdsfactory/gdsfactory/components/waveguides/straight.py:31: UserWarning: {'width': 2.0} ignored for cross_section 'xs_f4227c9f'
  x = gf.get_cross_section(cross_section, width=width)
../_images/b56c7a083f5362127e0bd8820f020cad63a7abfe2549d05471cf672f4cc19404.png

For GS routing the recommended solution is adding a dummy / abstract layer in the middle, where we add the ports.

p = gf.path.straight()

# Add GS routing sections.
# 99, 0 is an abstract layer that can be used to add ports to the path.
s0 = gf.Section(width=2, offset=0, layer=(99, 0), port_names=("e1", "e2"))
s1 = gf.Section(width=2, offset=-4, layer=(2, 0))
s2 = gf.Section(width=2, offset=+4, layer=(2, 0))
x = gf.CrossSection(sections=(s0, s1, s2), radius=8)
c = gf.path.extrude(p, cross_section=x)
pad = c
c_copy = c.copy()
c_copy.draw_ports()
c_copy.plot()
../_images/7eadf2aa2d3433c5d9af6f812e98f8ce4c1f02b9eeea224647c0c3793f673a20.png
c2 = gf.Component()
pad1 = c2 << c
pad2 = c2 << c
pad2.move((100, 100))
gf.routing.route_bundle(c2,
    [pad1.ports["e2"]],
    [pad2.ports["e1"]],
    cross_section=x,
    port_type='optical',
    bend='bend_euler',
    raise_on_error=True,
)
c2.plot()
/home/runner/work/gdsfactory/gdsfactory/gdsfactory/components/bends/bend_euler.py:108: UserWarning: {'width': 2.0} ignored for cross_section 'xs_3c55915b'
  x = gf.get_cross_section(cross_section, width=width or x.width)
/home/runner/work/gdsfactory/gdsfactory/gdsfactory/components/waveguides/straight.py:31: UserWarning: {'width': 2.0} ignored for cross_section 'xs_3c55915b'
  x = gf.get_cross_section(cross_section, width=width)
../_images/191074f6b814ecc3d0914983b00bb9faeadb9bf0e147ad4752789f14878bd984.png

Option 3: Cross-section with ComponentAlongPath#

You can also place components along a path, which is useful for wiring vias. A via is a vertical electrical connection that goes through the insulating layers of an integrated circuit to connect different layers of horizontal metal wiring.

import gdsfactory as gf
from gdsfactory.cross_section import ComponentAlongPath

# Create the path.
p = gf.path.straight()
p += gf.path.arc(10)
p += gf.path.straight()

# Define a cross-section containing a via.
via = ComponentAlongPath(
    component=gf.c.rectangle(size=(1, 1), centered=True), spacing=5, padding=2
)
s = gf.Section(width=0.5, offset=0, layer=(1, 0), port_names=("in", "out"))
x = gf.CrossSection(sections=(s,), components_along_path=(via,))

# Combine the path with the cross-section.
c = gf.path.extrude(p, cross_section=x)
c.plot()
../_images/2299ab03e14e4f08efcabc985c1f53532a808fb31dc93422ae14603ba4bf3492.png
import gdsfactory as gf
from gdsfactory.cross_section import ComponentAlongPath

# Create the path.
p = gf.path.straight()
p += gf.path.arc(10)
p += gf.path.straight()

# Define a cross-section with a via.
via0 = ComponentAlongPath(component=gf.c.via1(), spacing=5, padding=2, offset=0)
viap = ComponentAlongPath(component=gf.c.via1(), spacing=5, padding=2, offset=+2)
vian = ComponentAlongPath(component=gf.c.via1(), spacing=5, padding=2, offset=-2)
x = gf.CrossSection(sections=[s], components_along_path=(via0, viap, vian))

# Combine the path with the cross-section.
c = gf.path.extrude(p, cross_section=x)
c.plot()
../_images/89d3a05fb75ef4601ae53a0ee32c90ff1d217ed9779467f149f4c2047685e18f.png

Path#

You can pass append() lists of path segments. This makes it easy to combine paths very quickly. Below we show 3 examples using this functionality:

Example 1: Assemble a complex path by making a list of paths and passing it to append().

import gdsfactory as gf

P = gf.Path()

# Create the basic Path components.
left_turn = gf.path.euler(radius=4, angle=90)
right_turn = gf.path.euler(radius=4, angle=-90)
straight = gf.path.straight(length=10)

# Assemble a complex path by making a list of paths and passing it to `append()`.
# .append([...]): This method takes a list of path factories and adds them sequentially to the end of the existing path P.
# Each new segment starts where the previous one ended, creating a single, continuous path.
P.append(
    [
        straight,
        left_turn,
        straight,
        right_turn,
        straight,
        straight,
        right_turn,
        left_turn,
        straight,
    ]
)

f = P.plot()
../_images/42d0adfc570368c54dada09ae38a2a9f680d6b243237dd8da42570408fe31f1a.png
P = (
    straight
    + left_turn
    + straight
    + right_turn
    + straight
    + straight
    + right_turn
    + left_turn
    + straight
)
f = P.plot()
../_images/4a263967a653548dd4f513f87d4950ef1c35b6545a088c8ca455853fba2a481f.png

Example 2: Create an “S-turn” just by making a list of [left_turn, right_turn].

P = gf.Path()

# Create an "S-turn" by making a list.
s_turn = [left_turn, right_turn]

P.append(s_turn)
f = P.plot()
../_images/f94f1df9460a99007c9d01ff65157bf70f890a78f999665f33fff2c6613cc6cd.png

Example 3: Repeat the S-turn 3 times by nesting our S-turn list in another list. Nesting means placing one data structure inside another of the same type. In this context, it means creating a “list of lists.”

P = gf.Path()

# Create an "S-turn" using a list.
s_turn = [left_turn, right_turn]

# Repeat the S-turn 3 times by nesting our S-turn list 3x times in another list.
triple_s_turn = [s_turn, s_turn, s_turn]

P.append(triple_s_turn)
f = P.plot()
../_images/e2f70eea0cf52bb349f0621a60411cd0dd52072b42142f2d03c0edefc8b6965a.png

Note you can also use the Path() constructor to immediately construct your Path:

P = gf.Path([straight, left_turn, straight, right_turn, straight])
f = P.plot()
../_images/3bb94c21741076b345bd5e949635214ff0c7b626aae1861da71a57d1cee6faa4.png

Waypoint smooth paths#

You can also build smooth paths between waypoints with the smooth() function.

points = np.array([(20, 10), (40, 10), (20, 40), (50, 40), (50, 20), (70, 20)])
plt.plot(points[:, 0], points[:, 1], ".-")

# This functionensures that one unit on the x-axis is the same length as one unit on the y-axis.
# This is crucial for plots where the geometric shape is important.
plt.axis("equal")
(np.float64(17.5), np.float64(72.5), np.float64(8.5), np.float64(41.5))
../_images/0fc79e2d2bf710ea8288d0f741d9a2479fa478b905974905c2cbf4f0ba885785.png
points = np.array([(20, 10), (40, 10), (20, 40), (50, 40), (50, 20), (70, 20)])

P = gf.path.smooth(
    points=points,
    radius=2,
    bend=gf.path.euler,  # Alternatively, use pp.arc, which will create a constant-radius bend.
    use_eff=False,
)
f = P.plot()
../_images/967f14a521b787c14e9ff5e2c29cfff1fd18cb219514d27fcbd135d4ed64b740.png

Waypoint sharp paths#

It is also possible to make more traditional angular paths (e.g. electrical wires) in a few different ways.

Example 1: Using a simple list of points:

P = gf.Path([(20, 10), (30, 10), (40, 30), (50, 30), (50, 20), (70, 20)])
f = P.plot()
../_images/e94327418bdff0e5f02606a18c3d13e13b82b5fa5419a458c663535b6d2a951c.png

Example 2: Using the “turn and move” method, where you manipulate the end angle of the path so that when you append points to it they are in the correct direction. Note: It is crucial that the number of points per straight section is set to 2 (gf.path.straight(length, num_pts = 2)) otherwise the extrusion algorithm will show defects.

P = gf.Path()
P += gf.path.straight(length=10, npoints=2)
P.end_angle += 90  # "Turn" 90 deg (left).
P += gf.path.straight(length=10, npoints=2)  # "Walk" length of 10.
P.end_angle += -135  # "Turn" -135 degrees (right).
P += gf.path.straight(length=15, npoints=2)  # "Walk" length of 15.
P.end_angle = 0  # Force the direction to be 0 degrees.
P += gf.path.straight(length=10, npoints=2) 
f = P.plot()
../_images/013e85ae008f3ccfafddcd28aad2503c7818f678be4c8e100636b045cbc0510e.png
s0 = gf.Section(width=1, offset=0, layer=(1, 0))
s1 = gf.Section(width=1.5, offset=2.5, layer=(2, 0))
s2 = gf.Section(width=1.5, offset=-2.5, layer=(3, 0))
X = gf.CrossSection(sections=[s0, s1, s2])
c = gf.path.extrude(P, X)
c.plot()
../_images/1a7067db91492a14202d96aabf708cdb24fdab85210a85eb5d5db6ea46bb27af.png

Custom curves#

Now let us have some fun and try to make a loop-de-loop structure with parallel straights and several ports.

To create a new type of curve we simply make a function that produces an array of points. The best way to do that is to create a function which allows you to specify a large number of points along that curve – in the case shown below, the looploop() function outputs 1000 points along a looping path. Later, if we want to reduce the number of points in our geometry we can easily simplify the path.

def looploop(num_pts=1000):
    """Simple limacon looping curve."""

    # This line creates an array of num_pts evenly spaced numbers ranging from -π to 0. This array represents the angle t in polar coordinates.
    t = np.linspace(-np.pi, 0, num_pts) 
    r = 20 + 25 * np.sin(t) # This line calculates the radius r for each corresponding angle t using the polar equation for a limaçon curve.

    # # These lines convert the polar coordinates (r, t) into standard Cartesian coordinates (x, y), which are needed for plotting.
    x = r * np.cos(t) 
    y = r * np.sin(t) 

    # The separate x and y arrays are combined into a single NumPy array of coordinate pairs, which is then returned by the function.
    return np.array((x, y)).T


# Create the path points.
P = gf.Path()
P.append(gf.path.arc(radius=10, angle=90))
P.append(gf.path.straight())
P.append(gf.path.arc(radius=5, angle=-90))
P.append(looploop(num_pts=1000))
P.rotate(-45)

# Create the cross-section.
s0 = gf.Section(width=1, offset=0, layer=(1, 0), port_names=("in", "out"))
s1 = gf.Section(width=0.5, offset=2, layer=(2, 0))
s2 = gf.Section(width=0.5, offset=4, layer=(3, 0))
s3 = gf.Section(width=1, offset=0, layer=(4, 0))
X = gf.CrossSection(sections=(s0, s1, s2, s3))

c = gf.path.extrude(P, X)
c.plot()
../_images/4b05521ef3b265bd48256e134d98f8a23ae4dde566d411870e61bcfd93b7d73e.png

You can create Paths from any array of points – just be sure that they form smooth curves! If we examine our path P we can see that we have effortlessly created a long list of points:

path_points = P.points  # Curve points are stored as a numpy array in P.points.
print(np.shape(path_points))  # The shape of the array is Nx2.
print(len(P))  # Equivalently, use len(P) to see how many points are inside.
(1092, 2)
1092

Simplifying / reducing point usage#

One of the primary concerns of generating smooth curves is that too many points are generated, inflating file sizes and making boolean operations computationally expensive. Fortunately, PHIDL has a fast implementation of the Ramer-Douglas–Peucker algorithm that lets you reduce the number of points in a curve without changing its shape. All that needs to be done when you make a component() is extruding the path with a cross_section, you need to specify the simplify argument.

If we specify simplify = 1e-3, the number of points in the line drops from 12,000 to 4,000, and the remaining points form a line that is identical to within 1e-3 distance from the original (for the default 1 micron unit size, this corresponds to 1 nanometer resolution):

# The remaining points form a identical line to within `1e-3` from the original.
c = gf.path.extrude(p=P, cross_section=X, simplify=1e-3)
c.plot()
../_images/9328925198fc2b0304ae54579685d1d5267b468ef418c73b61e798c2e379e47b.png

Let us say we need fewer points. We can increase the simplify tolerance by specifying simplify = 1e-1. This drops the number of points to ~400 points and they form a line that is identical to within 1e-1 distance from the original:

c = gf.path.extrude(P, cross_section=X, simplify=1e-1)
c.plot()
../_images/c2f2dc641179fe11500ac4588d65c719a67e2f8f793eb188a4faf4ebb58b1daf.png

Taken to absurdity, what happens if we set simplify = 0.3? Once again, the ~200 remaining points form a line that is within 0.3 units from the original – but that line will look pretty bad.

c = gf.path.extrude(P, cross_section=X, simplify=0.3)
c.plot()
../_images/4f6b7f2d047be5d4f5af0bd74425a20d618fae0521accdb6dede988db6f8e1ed.png

Curvature calculation#

The Path class has a curvature() method that computes the curvature K of your smooth path (K = 1/(radius of curvature)). This can be helpful for verifying that your curves transition smoothly such as in track-transition curves (also known as “Euler” bends in the photonics world). Euler bends have lower mode-mismatch loss as explained in this paper

Note this curvature is numerically computed, so areas in which the curvature jumps instantaneously (such as between an arc and a straight segment) will be slightly interpolated, and sudden changes in point density along the curve can cause discontinuities.

straight_points = 100

P = gf.Path()
P.append(
    [
        gf.path.straight(
            length=10, npoints=straight_points
        ),  # Should have a curvature of 0
        gf.path.euler(
            radius=3, angle=90, p=0.5, use_eff=False
        ),  # Euler straight-to-bend transition with min. bend radius of 3 (max curvature of 1/3)
        gf.path.straight(
            length=10, npoints=straight_points
        ),  # Should have a curvature of 0
        gf.path.arc(radius=10, angle=90),  # Should have a curvature of 1/10
        gf.path.arc(radius=5, angle=-90),  # Should have a curvature of -1/5
        gf.path.straight(
            length=2, npoints=straight_points
        ),  # Should have a curvature of 0
    ]
)

f = P.plot()
../_images/062c9eee59d2fb815c2a64fa5fa777071f353ff984e52a438de5466a9704e672.png

Arc paths are equivalent to bend_circular and euler paths are equivalent to bend_euler.

# The .curvature() method of the Path object P is called.
# It returns two arrays: s, which contains the cumulative distance (arc length) at each point along the path,
# and K, which contains the corresponding curvature at that point. (Curvature is the reciprocal of the bend radius).
s, K = P.curvature()

# This plots the arc length s on the x-axis and the curvature K on the y-axis.
# The ".-" format string specifies that the plot should be a line with a dot marker at each data point.
plt.plot(s, K, ".-")
plt.xlabel("Position along curve (arc length)")
plt.ylabel("Curvature")
Text(0, 0.5, 'Curvature')
../_images/563b94e4f0949d1b6a673e1114e5745fa517808eec00b4f577d853ae532755f2.png
P = gf.path.euler(radius=3, angle=90, p=1.0, use_eff=False)
P.append(gf.path.euler(radius=3, angle=90, p=0.2, use_eff=False))
P.append(gf.path.euler(radius=3, angle=90, p=0.0, use_eff=False))
P.plot()
../_images/c07e2a8e0cb7353d5035f659ccb76d4229f9cd4b136e092d3a0ed8214a4a1745.png
s, K = P.curvature()
plt.plot(s, K, ".-")
plt.xlabel("Position along curve (arc length)")
plt.ylabel("Curvature")
Text(0, 0.5, 'Curvature')
../_images/0a9552c885a70de283e75a208c8bfe86eb9b16ab61ce8c13085a801ca54c2e88.png

You can compare two 90 degrees euler bends with 180 euler bend.

A 180 euler bend is shorter, and has less loss than two 90 degrees euler bend.

straight_points = 100

P = gf.Path()
P.append(
    [
        gf.path.euler(radius=3, angle=90, p=1, use_eff=False),
        gf.path.euler(radius=3, angle=90, p=1, use_eff=False),
        gf.path.straight(length=6, npoints=100),
        gf.path.euler(radius=3, angle=180, p=1, use_eff=False),
    ]
)

f = P.plot()
../_images/1de91b4bbf41aaf51930190fc2247b90c57412f7ea92493759cd71a0b5f2943a.png
s, K = P.curvature()
plt.plot(s, K, ".-")
plt.xlabel("Position along curve (arc length)")
plt.ylabel("Curvature")
Text(0, 0.5, 'Curvature')
../_images/d4e5edab173c24bd5ec6d5a3ab7680d3a146e65b055957a477139ff0d5066878.png

Transitioning between cross-sections#

Often a critical element of building paths is being able to transition between cross-sections. You can use the transition() function to do exactly this: You simply feed it two CrossSections and it will output a new CrossSection that smoothly transitions between the two.

Let us start off by creating two cross-sections we want to transition between. Note we give all the cross-sectional elements names by specifying the name argument in the add() function – this is important because the transition function will try to match names between the two input cross-sections, and any names not present in both inputs will be skipped.

# Create our first Cross-section.
import gdsfactory as gf

s0 = gf.Section(width=1.2, offset=0, layer=(2, 0), name="core", port_names=("o1", "o2"))
s1 = gf.Section(width=2.2, offset=0, layer=(3, 0), name="etch")
s2 = gf.Section(width=1.1, offset=3, layer=(1, 0), name="wg2")
X1 = gf.CrossSection(sections=[s0, s1, s2])

# Create the second Cross-section that we want to transition to.
s0 = gf.Section(width=1, offset=0, layer=(2, 0), name="core", port_names=("o1", "o2"))
s1 = gf.Section(width=3.5, offset=0, layer=(3, 0), name="etch")
s2 = gf.Section(width=3, offset=5, layer=(1, 0), name="wg2")
X2 = gf.CrossSection(sections=[s0, s1, s2])

# To show the cross-sections, let us now create two paths and create components by extruding them.
P1 = gf.path.straight(length=5)
P2 = gf.path.straight(length=5)
wg1 = gf.path.extrude(P1, X1)
wg2 = gf.path.extrude(P2, X2)

# Place both cross-section components and quickplot them,
# Quickplot is designed to create a wide variety of complex graphs with a simple, concise syntax, making it ideal for quick data exploration.
c = gf.Component()
wg1ref = c << wg1
wg2ref = c << wg2
wg2ref.movex(7.5)

c.plot()
../_images/b21857f417721fe9bcc23c6a10c13ada41ce4bb20d5b7225bf6f421228b9513b.png

Now we can create the transitional cross-section by calling a transition() with these two cross-sections as the input. If we want the width to vary as a smooth sinusoid between the sections, we can set width_type to 'sine' (alternatively we could also use 'linear').

# Create the transitional cross-section.
Xtrans = gf.path.transition(cross_section1=X1, cross_section2=X2, width_type="sine")

# Create a Path for the transitional cross-section to follow.
P3 = gf.path.straight(length=15, npoints=100)

# Use the transitional cross-section to create a component.
straight_transition = gf.path.extrude_transition(P3, Xtrans)
straight_transition.plot()
../_images/34a5e9ebd3e85f316b03638ccfab4ff042dd7c4d1f877cc6e3c5a338f3db1efd.png

Now that we have all of our components, let us proceed to connect() everything and see what it looks like:

c = gf.Component("transition_demo")

wg1ref = c << wg1
wgtref = c << straight_transition
wg2ref = c << wg2

wgtref.connect("o1", wg1ref.ports["o2"])
wg2ref.connect("o1", wgtref.ports["o2"])

c.plot()
../_images/82ae4355ceaad56f6bc15c6bc594f31e23c1c728c3c5e3a880ea2e1d745e6e1f.png

Note that since transition() outputs a Transition, we can make the transition follow an arbitrary path:

# Transition along a curving path.
P4 = gf.path.euler(radius=25, angle=45, p=0.5, use_eff=False)
wg_trans = gf.path.extrude_transition(P4, Xtrans)

c = gf.Component("demo_transition")
wg1_ref = c << wg1  # First cross-section component.
wg2_ref = c << wg2
wgt_ref = c << wg_trans

wgt_ref.connect("o1", wg1_ref.ports["o2"])
wg2_ref.connect("o1", wgt_ref.ports["o2"])

c.plot()
../_images/35882b9af36c9b0a451dcd468fa0aa960948ee41cd0c4cf57c0e6b69bd76ce14.png

You can also extrude an arbitrary transition:

w1 = 1
w2 = 5
x1 = gf.get_cross_section("strip", width=w1)
x2 = gf.get_cross_section("strip", width=w2)
transition = gf.path.transition(x1, x2)
p = gf.path.arc(radius=10)
c = gf.path.extrude_transition(p, transition)
c.plot()
../_images/6688ce3d2786b1338e966c7b0618c87e3b228f44a17cb5873bae44c9d2f20bf9.png

Asymmetric transition#

In some cases, you may want the edges of the transition to follow a different function. This can be done by using the transition_asymmetric() function. In this case, the argument width_type of transition is split into width_type1, corresponding to the lower edge, and width_type2, corresponding to the upper edge of the transition. As in the case of transition(), the user can define their own transition function.

Let us look at an example where the upper edge follows the sinusoidal (default) transition of the width, while the lower follows a user-defined polynomial.

import gdsfactory as gf

# Define a custom polynomial transition function from y1 -> y2, for t ∈ [0,1].
def polynomial(t: float, y1: float, y2: float) -> float:
        return (y2 - y1) * t**3 + y1

w1 = 2
w2 = 6
length = 10
cs1 = gf.get_cross_section("strip", width=w1)
cs2 = gf.get_cross_section("strip", width=w2)

transition = gf.path.transition_asymmetric(
    cs1, cs2, width_type1=polynomial, width_type2="sine")
p = gf.path.straight(length, npoints=100)
c = gf.path.extrude_transition(p, transition)

c.plot()
../_images/df5553def67d02d14237eadcb0d337a4b5d16ecda431f7ccf1d626c883cbd87d.png

Variable width / offset#

In some instances, you may want to vary the width or offset of the path’s cross-section as it travels. This can be accomplished by giving the CrossSection arguments that are functions or lists. Let us say we wanted a width that varies sinusoidally along the length of the Path. To do this, we need to make a width function that is parameterized from 0 to 1: for an example function my_width_fun(t) where the width at t==0 is the width at the beginning of the path and the width at t==1 is the width at the end.

import numpy as np

import gdsfactory as gf


def my_custom_width_fun(t):
    # Note: Custom width/offset functions MUST be vectorizable --
    # you must be able to call them with an array input like my_custom_width_fun([0, 0.1, 0.2, 0.3, 0.4]).
    num_periods = 5

    # np.cos(...): This is the core cosine function from the NumPy library, which generates a wave that oscillates between -1 and 1.
    # 2 * np.pi * t * num_periods: This part calculates the angle (in radians) for the cosine function.
    # It determines the frequency of the wave, i.e., how many full cycles (num_periods) it completes over a given time t.
    # This adds a vertical offset of 3 to the wave. Instead of oscillating between -1 and 1, the wave now oscillates between 2 (3 - 1) and 4 (3 + 1).
    return 3 + np.cos(2 * np.pi * t * num_periods)


P = gf.path.straight(length=40, npoints=30)

#Create two cross-sections: one fixed width, one modulated by my_custom_offset_fun.
s0 = gf.Section(width=3, offset=-6, layer=(2, 0))
s1 = gf.Section(width=0, width_function=my_custom_width_fun, offset=0, layer=(1, 0))
X = gf.CrossSection(sections=(s0, s1))

# # Extrude the path to create the component.
c = gf.path.extrude(P, cross_section=X)
c.plot()
../_images/1d9301928958cad0ea56db62728fa4145da0dfba3f76e8c0ae1f0a0320be80fd.png

We can do the same thing with the offset argument:

def my_custom_offset_fun(t):
    num_periods = 3
    return 3 + np.cos(2 * np.pi * t * num_periods)


P = gf.path.straight(length=40, npoints=30)

s0 = gf.Section(width=1, offset=0, layer=(1, 0))
s1 = gf.Section(
    width=1,
    offset_function=my_custom_offset_fun,
    layer=(2, 0),
    port_names=("clad1", "clad2"),
)
X = gf.CrossSection(sections=(s0, s1))

c = gf.path.extrude(P, cross_section=X)
c.plot()
../_images/9c7edd0e993524269a4e8d6a0b269386f7fe71f19b5aecfedf67fdaf28db353b.png

Offsetting a Path#

Sometimes it is convenient to start with a simple path and offset the line it follows to suit your needs (without using a custom-offset cross-section). Here, we start with two copies of a simple straight path and use the offset() function to directly modify each path.

def my_custom_offset_fun(t):

    num_periods = 3
    return 2 + np.cos(2 * np.pi * t * num_periods)


P1 = gf.path.straight(npoints=101)
P1.offset(offset=my_custom_offset_fun)
f = P1.plot()
../_images/a240b2466d65d65922e3f9d2c9703e489eff432ea1356e4c84833f8a6a08915b.png
P2 = P1.copy()  # Make a copy of the path.
P2.mirror((1, 0))  # Mirror across X-axis.
f2 = P2.plot()
../_images/87421a85750c7b68f832285a64168c07ad105a7eebfa0bc34e86ae0a6c6afe37.png
P = gf.path.arc(radius=10, angle=45)

s0 = gf.Section(width=1, offset=3, layer=(2, 0), name="waveguide")
s1 = gf.Section(width=1, offset=0, layer=(1, 0), name="heater", port_names=("o1", "o2"))
X = gf.CrossSection(sections=(s0, s1))
c = gf.path.extrude(P, X)
c.plot()
../_images/ce7f0fcad31494dcaf1a58fc01107c2f2027c1928b5e3c81f9b6758eefa00654.png
P = gf.Path()
P.append(gf.path.arc(radius=10, angle=90))  # Circular arc.
P.append(gf.path.straight(length=10))  # Straight section.
P.append(gf.path.euler(radius=3, angle=-90))  # Euler bend (aka "racetrack" curve).
P.append(gf.path.straight(length=40))
P.append(gf.path.arc(radius=8, angle=-45))
P.append(gf.path.straight(length=10))
P.append(gf.path.arc(radius=8, angle=45))
P.append(gf.path.straight(length=10))

f = P.plot()
../_images/22f3bf77b522a80591fb982f2699c71d0ce442f2a2ba6e01b6ba93361a42ad16.png
c = gf.path.extrude(P, width=1, layer=(2, 0))
c.plot()
../_images/bbf9d4bb47e1c93341c55c5036128cdb04aac66ffd4095c21a78053e463f0525.png
s0 = gf.Section(width=2, offset=0, layer=(2, 0))
xs = gf.CrossSection(sections=(s0,))
c = gf.path.extrude(P, xs)
c.plot()
../_images/3c345472314f187baac04d538da32566375304ab08644f4899e1430f7e06e9ec.png
p = gf.path.straight(length=10, npoints=101)
s0 = gf.Section(width=1, offset=0, layer=(1, 0), port_names=("o1", "o2"), name="core")
s1 = gf.Section(width=3, offset=0, layer=(3, 0), name="slab")
x1 = gf.CrossSection(sections=(s0, s1))
c = gf.path.extrude(p, x1)
c.plot()
../_images/6dfe5bbc29ffe2f88d52c1d2256e46eeb365b034a96734d20585d212406d8ba1.png
s0 = gf.Section(
    width=1 + 3, offset=0, layer=(1, 0), port_names=("o1", "o2"), name="core"
)
s1 = gf.Section(width=3 + 3, offset=0, layer=(3, 0), name="slab")
x2 = gf.CrossSection(sections=(s0, s1))
c2 = gf.path.extrude(p, x2)
c2.plot()
../_images/83cb58841685287445de1941c4a37e6783a760cd8043a3012d70e199c1f64f2e.png
t = gf.path.transition(x1, x2)
c3 = gf.path.extrude_transition(p, t)
c3.plot()
../_images/b08c2615fdb7746715e7bfd43e087183e43a587dd92d118c49cff5b6db12697a.png
c4 = gf.Component()
start_ref = c4 << c
trans_ref = c4 << c3
end_ref = c4 << c2

trans_ref.connect("o1", start_ref.ports["o2"])
end_ref.connect("o1", trans_ref.ports["o2"])
c4.plot()
../_images/3deed45b0c5d37bc0f6a57aea51bcc80eaaaa3ebaa7e23c98a7da6e3987378ce.png

Creating new cross_sections#

You can create functions that return a cross_section in 2 ways:

  • Customize an existing cross-section for example gf.cross_section.strip.

  • Define a function that returns a cross_section.

  • Define a CrossSection object.

What parameters do cross_section take?

help(gf.cross_section.cross_section)
Help on function cross_section in module gdsfactory.cross_section:

cross_section(width: 'float' = 0.5, offset: 'float' = 0, layer: 'typings.LayerSpec' = 'WG', sections: 'Sections | None' = None, port_names: 'typings.IOPorts' = ('o1', 'o2'), port_types: 'typings.IOPorts' = ('optical', 'optical'), bbox_layers: 'typings.LayerSpecs | None' = None, bbox_offsets: 'typings.Floats | None' = None, cladding_layers: 'typings.LayerSpecs | None' = None, cladding_offsets: 'typings.Floats | None' = None, cladding_simplify: 'typings.Floats | None' = None, cladding_centers: 'typings.Floats | None' = None, radius: 'float | None' = 10.0, radius_min: 'float | None' = None, main_section_name: 'str' = '_default') -> 'CrossSection'
    Return CrossSection.
    
    Args:
        width: main Section width (um).
        offset: main Section center offset (um).
        layer: main section layer.
        sections: list of Sections(width, offset, layer, ports).
        port_names: for input and output ('o1', 'o2').
        port_types: for input and output: electrical, optical, vertical_te ...
        bbox_layers: list of layers bounding boxes to extrude.
        bbox_offsets: list of offset from bounding box edge.
        cladding_layers: list of layers to extrude.
        cladding_offsets: list of offset from main Section edge.
        cladding_simplify: Optional Tolerance value for the simplification algorithm.                 All points that can be removed without changing the resulting.                 polygon by more than the value listed here will be removed.
        cladding_centers: center offset for each cladding layer. Defaults to 0.
        radius: routing bend radius (um).
        radius_min: min acceptable bend radius.
        main_section_name: name of the main section. Defaults to _default
    
    .. plot::
        :include-source:
    
        import gdsfactory as gf
    
        xs = gf.cross_section.cross_section(width=0.5, offset=0, layer='WG')
        p = gf.path.arc(radius=10, angle=45)
        c = p.extrude(xs)
        c.plot()
    
    .. code::
    
    
           ┌────────────────────────────────────────────────────────────┐
           │                                                            │
           │                                                            │
           │                   boox_layer                               │
           │                                                            │
           │         ┌──────────────────────────────────────┐           │
           │         │                            ▲         │bbox_offset│
           │         │                            │         ├──────────►│
           │         │           cladding_offset  │         │           │
           │         │                            │         │           │
           │         ├─────────────────────────▲──┴─────────┤           │
           │         │                         │            │           │
        ─ ─┤         │           core   width  │            │           ├─ ─ center
           │         │                         │            │           │
           │         ├─────────────────────────▼────────────┤           │
           │         │                                      │           │
           │         │                                      │           │
           │         │                                      │           │
           │         │                                      │           │
           │         └──────────────────────────────────────┘           │
           │                                                            │
           │                                                            │
           │                                                            │
           └────────────────────────────────────────────────────────────┘
import gdsfactory as gf
from gdsfactory.cross_section import CrossSection, cross_section, xsection
from gdsfactory.typings import LayerSpec

@xsection
def pin(
    width: float = 0.5,
    layer: LayerSpec = "WG",
    radius: float = 10.0,
    radius_min: float = 5,
    layer_p: LayerSpec = (21, 0),
    layer_n: LayerSpec = (20, 0),
    width_p: float = 2,
    width_n: float = 2,
    offset_p: float = 1,
    offset_n: float = -1,
    **kwargs,
) -> CrossSection:
    """Return PIN cross_section."""
    sections = (
        gf.Section(layer=layer_p, width=width_p, offset=offset_p),
        gf.Section(layer=layer_n, width=width_n, offset=offset_n),
    )

    return cross_section(
        width=width,
        layer=layer,
        radius=radius,
        radius_min=radius_min,
        sections=sections,
        **kwargs,
    )
c = gf.components.straight(cross_section=pin)
c.plot()
../_images/6eb97469845e9bcd39cf6643a40f5e83c25fe2d9c113ea806948710a6ceb8c91.png
pin5 = gf.components.straight(cross_section=pin, length=5)
pin5.plot()
../_images/291e061780a0a1cf538caaae3cbfe0425182ab0cfc2e516c796a88684ed54efe.png
pin5 = gf.components.straight(cross_section="pin", length=5)
pin5.plot()
../_images/291e061780a0a1cf538caaae3cbfe0425182ab0cfc2e516c796a88684ed54efe.png

Finally, you can also pass the dictionary (dict) of most components that define the cross-section.

# Create our first cross-section
s0 = gf.Section(width=0.5, offset=0, layer=(1, 0), name="wg", port_names=("o1", "o2"))
s1 = gf.Section(width=0.2, offset=0, layer=(3, 0), name="slab")
x1 = gf.CrossSection(sections=(s0, s1))

# Create the second cross-section that we want to transition to.
s0 = gf.Section(width=0.5, offset=0, layer=(1, 0), name="wg", port_names=("o1", "o2"))
s1 = gf.Section(width=3.0, offset=0, layer=(3, 0), name="slab")
x2 = gf.CrossSection(sections=(s0, s1))

# To show the cross-sections, let us create two paths and create components by extruding them.
p1 = gf.path.straight(length=5)
p2 = gf.path.straight(length=5)
wg1 = gf.path.extrude(p1, x1)
wg2 = gf.path.extrude(p2, x2)

# Place both cross-section components and quickplot them.
c = gf.Component()
wg1ref = c << wg1
wg2ref = c << wg2
wg2ref.movex(7.5)

# Create the transitional cross-section.
xtrans = gf.path.transition(cross_section1=x1, cross_section2=x2, width_type="linear")
# Create a path for the transitional cross-section to follow.
p3 = gf.path.straight(length=15, npoints=100)

# Use the transitional cross-section to create a component.
straight_transition = gf.path.extrude_transition(p3, xtrans)
straight_transition.plot()
../_images/1e6d4c9491d3f1c55606ea9636feb5aa840fa6ad9a0f36dd0c80671fa04edfee.png
xtrans = gf.path.transition(
    cross_section1=x1, cross_section2=x2, width_type="parabolic"
)

p3 = gf.path.straight(length=15, npoints=100)


straight_transition = gf.path.extrude_transition(p3, xtrans)
straight_transition.plot()
../_images/3a9a7866fb1bb8155eef5cfc14b1b94d63b0515d98d93e6c30f54e2b3d4e3751.png
xtrans = gf.path.transition(cross_section1=x1, cross_section2=x2, width_type="sine")
p3 = gf.path.straight(length=15, npoints=100)


straight_transition = gf.path.extrude_transition(p3, xtrans)
straight_transition.plot()
../_images/5d366ef7a94bb084419e85aeb93471dccc06dffe06026067a7b96d8044fd210a.png
s = straight_transition.to_3d()
s.show()

The port location, width and orientation remains the same for a sheared component. However, an additional property, shear_angle is set to the value of the shear angle. In general, shear ports can be safely connected together.

bbox_layers vs cladding_layers#

For extruding waveguides you have two options:

  1. bbox_layers for squared bounding box.

  2. cladding_layers for extruding a layer that follows the shape of the path.

xs_bbox = gf.cross_section.cross_section(bbox_layers=((3, 0),), bbox_offsets=(3,))
w1 = gf.components.bend_euler(cross_section=xs_bbox)
w1.plot()
../_images/541483fce26221edd140f150a4d7de3add983c9c12c98870c0182c2e487fd09c.png
xs_clad = gf.cross_section.cross_section(cladding_layers=[(3, 0)], cladding_offsets=[3])
w2 = gf.components.bend_euler(cross_section=xs_clad)
w2.plot()
../_images/65a0c58e3e6a17dc48869584434f083a015c44ef616e842a16d37b4a0010d91a.png

Insets#

It is handy to be able to extrude a CrossSection along a Path, while each Section may have a particular inset relative to the main Section. An example of this is a waveguide with a heater.

import gdsfactory as gf


@xsection
def xs_waveguide_heater() -> gf.CrossSection:
    return gf.cross_section.cross_section(
        layer="WG",
        width=0.5,
        sections=(
            gf.cross_section.Section(
                name="heater",
                width=1,
                layer="HEATER",
                insets=(1, 2),
            ),
        ),
    )


c = gf.components.straight(cross_section=xs_waveguide_heater)
c.plot()
../_images/784f4fe7356df003425a55ec19cbb257a0258b3b42acb99d0968b429f1907f28.png
@xsection
def xs_waveguide_heater_with_ports() -> gf.CrossSection:
    return gf.cross_section.cross_section(
        layer="WG",
        width=0.5,
        sections=(
            gf.cross_section.Section(
                name="heater",
                width=1,
                layer="HEATER",
                insets=(1, 2),
                port_names=("e1", "e2"),
                port_types=("electrical", "electrical"),
            ),
        ),
    )


c = gf.components.straight(cross_section=xs_waveguide_heater_with_ports)
c.plot()
../_images/784f4fe7356df003425a55ec19cbb257a0258b3b42acb99d0968b429f1907f28.png