-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathtrapping_array_pcell.lym
More file actions
67 lines (60 loc) · 52.6 KB
/
trapping_array_pcell.lym
File metadata and controls
67 lines (60 loc) · 52.6 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
<?xml version="1.0" encoding="utf-8"?>
<!--
Trapping Array PCell Library v1.0.0
Install : Macro > Import... or copy to ~/.klayout/pymacros/
Usage : Libraries panel > "Trapping Array" > drag TrapArray_A or TrapArray_B
Ref : Ruyssen et al., Computers and Fluids 297 (2025) 106643
-->
<klayout-macro>
<description>Trapping Array PCell Library v1.0.0</description>
<version>1.0.0</version>
<category>pymacros</category>
<prolog/>
<epilog/>
<doc/>
<autorun>true</autorun>
<autorun-early>false</autorun-early>
<shortcut/>
<show-in-menu>false</show-in-menu>
<group-name/>
<menu-path/>
<interpreter>python</interpreter>
<dsl-interpreter-name/>
<text>"""
Trapping Array PCell Library v1.0.0
Ref: Ruyssen et al., Computers and Fluids 297 (2025) 106643
Registers "Trapping Array" in the Libraries panel with two PCells:
TrapArray_A - Fixed grid (fixed N_l x N_c)
TrapArray_B - Max traps (maximise count given pitch)
"""
import os, sys, importlib
_pkg = os.path.join(os.path.expanduser("~"), ".klayout",
"pymacros", "trapping_array")
_core = os.path.join(_pkg, "core")
for _d in (_core,):
os.makedirs(_d, exist_ok=True)
_M = {
'core/__init__.py': __import__("base64").b64decode('IiIiCnRyYXBwaW5nX2FycmF5LmNvcmUKPT09PT09PT09PT09PT09PT09PQpQdWJsaWMgQVBJIGZvciB0aGUgVHJhcHBpbmcgQXJyYXkgS0xheW91dCBsaWJyYXJ5LgoKUXVpY2sgc3RhcnQKLS0tLS0tLS0tLS0KICAgIGZyb20gdHJhcHBpbmdfYXJyYXkuY29yZSBpbXBvcnQgbWFrZV9maXhlZF9ncmlkLCBtYWtlX21heF90cmFwcywgZ2VuZXJhdGVfZ2RzCgogICAgIyBNb2RlIEEg4oCTIGZpeGVkIDfDlzE0IGdyaWQsIGRpcmVjdCBJTwogICAgZ3JpZCwgYnVpbGRlciA9IG1ha2VfZml4ZWRfZ3JpZCgKICAgICAgICBMPTE1MDAsIFc9MTIwMCwKICAgICAgICBOX2w9NywgTl9jPTE0LCBnYXBfeD0xMCwgZ2FwX3k9MTAsCiAgICAgICAgbF90cmFwPTMwLCB3X3RyYXA9MjUsIHdhbGxfdD0zLCBiYXNlX3Q9NCwgd19vX2Rvd249OCwKICAgICAgICBpb19tb2RlPSdkaXJlY3QnLCBDPTAuNDQsIHdfaW89MTUwLCBjaGFubmVsX2xlbj0xMDAsCiAgICApCiAgICBwcmludChidWlsZGVyLnN1bW1hcnkoKSkKICAgIGdlbmVyYXRlX2dkcyhncmlkLCBidWlsZGVyLCAiY2hpcF9BLmdkcyIpCgogICAgIyBNb2RlIEIg4oCTIG1heGltaXNlIHRyYXBzLCBkaXZlcmdpbmcgSU8KICAgIGdyaWQsIGJ1aWxkZXIgPSBtYWtlX21heF90cmFwcygKICAgICAgICBMPTE1MDAsIFc9MTIwMCwKICAgICAgICBEZWx0YV94PTQwLCBEZWx0YV95PTM1LAogICAgICAgIGxfdHJhcD0zMCwgd190cmFwPTI1LCB3YWxsX3Q9MywgYmFzZV90PTQsIHdfb19kb3duPTgsCiAgICAgICAgaW9fbW9kZT0nZGl2ZXJnaW5nJywgd19pbz0xNTAsIGxfZGl2PTMwMCwgY2hhbm5lbF9sZW49MTAwLAogICAgKQogICAgZ2VuZXJhdGVfZ2RzKGdyaWQsIGJ1aWxkZXIsICJjaGlwX0IuZ2RzIikKIiIiCgpmcm9tIC5ncmlkICAgIGltcG9ydCBBcnJheUdyaWQKZnJvbSAuYnVpbGRlciBpbXBvcnQgVHJhcHBpbmdBcnJheUJ1aWxkZXIKCmltcG9ydCBweWEKCl9fYWxsX18gPSBbCiAgICAiQXJyYXlHcmlkIiwKICAgICJUcmFwcGluZ0FycmF5QnVpbGRlciIsCiAgICAibWFrZV9maXhlZF9ncmlkIiwKICAgICJtYWtlX21heF90cmFwcyIsCiAgICAiZ2VuZXJhdGVfZ2RzIiwKXQoKCiMgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCiMgQ29udmVuaWVuY2UgZmFjdG9yeSBmdW5jdGlvbnMKIyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCmRlZiBtYWtlX2ZpeGVkX2dyaWQoCiAgICAjIENhdml0eQogICAgTCwgVywKICAgICMgR3JpZAogICAgTl9sLCBOX2MsIGdhcF94LCBnYXBfeSwKICAgIGdhcF94X21hcmdpbj1Ob25lLCBnYXBfeV9tYXJnaW49Tm9uZSwKICAgICMgVHJhcCBzaGFwZQogICAgbF90cmFwPTMwLjAsIHdfdHJhcD0yNS4wLCB3YWxsX3Q9My4wLCBiYXNlX3Q9NC4wLCB3X29fZG93bj04LjAsCiAgICAjIElPCiAgICBpb19tb2RlPSdkaXJlY3QnLCBDPTEuMCwgd19pbz0xMDAuMCwgY2hhbm5lbF9sZW49MTAwLjAsIGxfZGl2PTIwMC4wLAogICAgIyBEaXNvcmRlcgogICAgRF9mPTAuMCwgZGlzb3JkZXJfdHlwZT0ndW5pZm9ybScsIHNlZWQ9Tm9uZSwKKToKICAgICIiIgogICAgTW9kZSBBOiBmaXhlZCBOX2wgw5cgTl9jIHRyYXAgZ3JpZC4KCiAgICBSZXR1cm5zCiAgICAtLS0tLS0tCiAgICAoQXJyYXlHcmlkLCBUcmFwcGluZ0FycmF5QnVpbGRlcikKICAgICIiIgogICAgZ3JpZCA9IEFycmF5R3JpZCgKICAgICAgICAnZml4ZWRfZ3JpZCcsIEwsIFcsIGxfdHJhcCwgd190cmFwLAogICAgICAgIE5fbD1OX2wsIE5fYz1OX2MsIGdhcF94PWdhcF94LCBnYXBfeT1nYXBfeSwKICAgICAgICBnYXBfeF9tYXJnaW49Z2FwX3hfbWFyZ2luLCBnYXBfeV9tYXJnaW49Z2FwX3lfbWFyZ2luLAogICAgKQogICAgYnVpbGRlciA9IFRyYXBwaW5nQXJyYXlCdWlsZGVyKAogICAgICAgIGdyaWQsIGxfdHJhcCwgd190cmFwLCB3YWxsX3QsIGJhc2VfdCwgd19vX2Rvd24sCiAgICAgICAgaW9fbW9kZT1pb19tb2RlLCBDPUMsIHdfaW89d19pbywKICAgICAgICBjaGFubmVsX2xlbj1jaGFubmVsX2xlbiwgbF9kaXY9bF9kaXYsCiAgICAgICAgRF9mPURfZiwgZGlzb3JkZXJfdHlwZT1kaXNvcmRlcl90eXBlLCBzZWVkPXNlZWQsCiAgICApCiAgICByZXR1cm4gZ3JpZCwgYnVpbGRlcgoKCmRlZiBtYWtlX21heF90cmFwcygKICAgICMgQ2F2aXR5CiAgICBMLCBXLAogICAgIyBQaXRjaAogICAgRGVsdGFfeCwgRGVsdGFfeSwKICAgIG1hcmdpbl94PU5vbmUsIG1hcmdpbl95PU5vbmUsCiAgICAjIFRyYXAgc2hhcGUKICAgIGxfdHJhcD0zMC4wLCB3X3RyYXA9MjUuMCwgd2FsbF90PTMuMCwgYmFzZV90PTQuMCwgd19vX2Rvd249OC4wLAogICAgIyBJTwogICAgaW9fbW9kZT0nZGlyZWN0JywgQz0xLjAsIHdfaW89MTAwLjAsIGNoYW5uZWxfbGVuPTEwMC4wLCBsX2Rpdj0yMDAuMCwKICAgICMgRGlzb3JkZXIKICAgIERfZj0wLjAsIGRpc29yZGVyX3R5cGU9J3VuaWZvcm0nLCBzZWVkPU5vbmUsCik6CiAgICAiIiIKICAgIE1vZGUgQjogbWF4aW1pc2UgdHJhcCBjb3VudCBnaXZlbiBjYXZpdHkgYW5kIHBpdGNoLgoKICAgIFJldHVybnMKICAgIC0tLS0tLS0KICAgIChBcnJheUdyaWQsIFRyYXBwaW5nQXJyYXlCdWlsZGVyKQogICAgIiIiCiAgICBncmlkID0gQXJyYXlHcmlkKAogICAgICAgICdtYXhfdHJhcHMnLCBMLCBXLCBsX3RyYXAsIHdfdHJhcCwKICAgICAgICBEZWx0YV94PURlbHRhX3gsIERlbHRhX3k9RGVsdGFfeSwKICAgICAgICBtYXJnaW5feD1tYXJnaW5feCwgbWFyZ2luX3k9bWFyZ2luX3ksCiAgICApCiAgICBidWlsZGVyID0gVHJhcHBpbmdBcnJheUJ1aWxkZXIoCiAgICAgICAgZ3JpZCwgbF90cmFwLCB3X3RyYXAsIHdhbGxfdCwgYmFzZV90LCB3X29fZG93biwKICAgICAgICBpb19tb2RlPWlvX21vZGUsIEM9Qywgd19pbz13X2lvLAogICAgICAgIGNoYW5uZWxfbGVuPWNoYW5uZWxfbGVuLCBsX2Rpdj1sX2RpdiwKICAgICAgICBEX2Y9RF9mLCBkaXNvcmRlcl90eXBlPWRpc29yZGVyX3R5cGUsIHNlZWQ9c2VlZCwKICAgICkKICAgIHJldHVybiBncmlkLCBidWlsZGVyCgoKZGVmIGdlbmVyYXRlX2dkcyhncmlkLCBidWlsZGVyLCBmaWxlbmFtZT0idHJhcHBpbmdfYXJyYXkuZ2RzIiwKICAgICAgICAgICAgICAgICBkYnU9MC4wMDEsIGxheWVyX251bWJlcj0xLCBsYXllcl9kYXRhdHlwZT0wKToKICAgICIiIgogICAgQnVpbGQgdGhlIGZ1bGwgY2hpcCBsYXlvdXQgYW5kIHdyaXRlIGEgR0RTIGZpbGUuCgogICAgUGFyYW1ldGVycwogICAgLS0tLS0tLS0tLQogICAgZ3JpZCwgYnVpbGRlciA6IGZyb20gbWFrZV9maXhlZF9ncmlkIC8gbWFrZV9tYXhfdHJhcHMKICAgIGZpbGVuYW1lICAgICAgOiBvdXRwdXQgcGF0aCAoc3RyKQogICAgZGJ1ICAgICAgICAgICA6IGRhdGFiYXNlIHVuaXQgaW4gwrVtIChkZWZhdWx0IDAuMDAxID0gMSBubSkKICAgIGxheWVyX251bWJlciAgOiBHRFMgbGF5ZXIgbnVtYmVyIChkZWZhdWx0IDEpCiAgICBsYXllcl9kYXRhdHlwZTogR0RTIGRhdGF0eXBlIChkZWZhdWx0IDApCgogICAgUmV0dXJucwogICAgLS0tLS0tLQogICAgcHlhLkxheW91dAogICAgIiIiCiAgICBsYXlvdXQgICAgPSBweWEuTGF5b3V0KCkKICAgIGxheW91dC5kYnUgPSBkYnUKICAgIGxheWVyX2lkeCAgPSBsYXlvdXQubGF5ZXIobGF5ZXJfbnVtYmVyLCBsYXllcl9kYXRhdHlwZSkKICAgIGJ1aWxkZXIuYnVpbGQobGF5b3V0LCBsYXllcl9pZHgpCiAgICBsYXlvdXQud3JpdGUoZmlsZW5hbWUpCiAgICBwcmludChmIltUcmFwQXJyYXldIFdyaXR0ZW46IHtmaWxlbmFtZX0iKQogICAgcHJpbnQoYnVpbGRlci5zdW1tYXJ5KCkpCiAgICByZXR1cm4gbGF5b3V0Cg==').decode(),
'core/grid.py': __import__("base64").b64decode('"""
core/grid.py
------------
ArrayGrid: resolves the trap array geometry for both operating modes.
Pure Python — no pya dependency, fully testable without KLayout.

Mode A  fixed_grid
    Inputs : cavity (L, W), grid (N_l × N_c), gaps (gap_x, gap_y).
    Derived: Δx = l_trap + gap_x,  Δy = w_trap + gap_y.
    Margin : lateral  margin_y = Δy/2  (overridable).
             streamwise margin_x = gap_x (overridable).
    Array centred in cavity; staggered (odd columns shifted +Δx/2 in x).

Mode B  max_traps
    Inputs : cavity (L, W), pitch (Δx, Δy), optional margin overrides.
    Derived: N_c = floor((W - 2*margin_y - w_trap) / Δy) + 1
             N_l = floor((L - 2*margin_x - l_trap) / Δx) + 1
"""

import math
import random


class ArrayGrid:
    """
    Resolved trap array geometry.

    Attributes (all µm unless stated)
    ----------------------------------
    mode       : str   – 'fixed_grid' or 'max_traps'
    L, W       : float – cavity dimensions
    N_l, N_c   : int   – lines (streamwise) and columns (lateral)
    N_traps    : int   – N_l * N_c
    Delta_x    : float – streamwise pitch
    Delta_y    : float – lateral pitch
    l, w       : float – array bounding-box span (margins included)
    x0_array   : float – x of array BL corner (centred in cavity)
    y0_array   : float – y of array BL corner
    margin_x   : float – streamwise cavity-wall → nearest trap edge
    margin_y   : float – lateral   cavity-wall → nearest trap edge (= Δy/2)
    feasible   : bool
    warnings   : list[str]
    """

    def __init__(self, mode, L, W, l_trap, w_trap, **kw):
        self.mode    = mode
        self.L       = L
        self.W       = W
        self.l_trap  = l_trap
        self.w_trap  = w_trap
        self.warnings = []
        self.feasible = True

        if mode == 'fixed_grid':
            self._solve_fixed(**kw)
        elif mode == 'max_traps':
            self._solve_max(**kw)
        else:
            raise ValueError(
                f"Unknown mode '{mode}'. Use 'fixed_grid' or 'max_traps'.")

    # ------------------------------------------------------------------
    def _solve_fixed(self, N_l, N_c, gap_x, gap_y,
                     gap_x_margin=None, gap_y_margin=None):
        Delta_x  = self.l_trap + gap_x
        Delta_y  = self.w_trap + gap_y
        margin_y = Delta_y / 2.0 if gap_y_margin is None else float(gap_y_margin)
        margin_x = float(gap_x)  if gap_x_margin is None else float(gap_x_margin)

        w_needed = 2 * margin_y + (N_c - 1) * Delta_y + self.w_trap
        l_needed = 2 * margin_x + (N_l - 1) * Delta_x + self.l_trap

        if w_needed > self.W:
            self.warnings.append(
                f"Array width {w_needed:.1f} µm > cavity W = {self.W:.1f} µm. "
                f"Reduce N_c or gap_y.")
            self.feasible = False
        if l_needed > self.L:
            self.warnings.append(
                f"Array length {l_needed:.1f} µm > cavity L = {self.L:.1f} µm. "
                f"Reduce N_l or gap_x.")
            self.feasible = False

        self.N_l      = int(N_l)
        self.N_c      = int(N_c)
        self.N_traps  = self.N_l * self.N_c
        self.Delta_x  = Delta_x
        self.Delta_y  = Delta_y
        self.margin_x = margin_x
        self.margin_y = margin_y
        self.l        = l_needed
        self.w        = w_needed
        self.x0_array = (self.L - self.l) / 2.0
        self.y0_array = (self.W - self.w) / 2.0

    # ------------------------------------------------------------------
    def _solve_max(self, Delta_x, Delta_y, margin_x=None, margin_y=None):
        margin_y = Delta_y / 2.0 if margin_y is None else float(margin_y)
        margin_x = margin_y       if margin_x is None else float(margin_x)

        N_c = max(0, math.floor(
            (self.W - 2 * margin_y - self.w_trap) / Delta_y) + 1)
        N_l = max(0, math.floor(
            (self.L - 2 * margin_x - self.l_trap) / Delta_x) + 1)

        if N_c < 1 or N_l < 1:
            self.warnings.append(
                f"No traps fit in cavity {self.L} × {self.W} µm "
                f"with Δx={Delta_x}, Δy={Delta_y}, "
                f"margin_x={margin_x:.1f}, margin_y={margin_y:.1f} µm.")
            self.feasible = False
            N_c = max(1, N_c)
            N_l = max(1, N_l)

        self.N_l      = int(N_l)
        self.N_c      = int(N_c)
        self.N_traps  = self.N_l * self.N_c
        self.Delta_x  = float(Delta_x)
        self.Delta_y  = float(Delta_y)
        self.margin_x = margin_x
        self.margin_y = margin_y
        self.l        = 2 * margin_x + (N_l - 1) * Delta_x + self.l_trap
        self.w        = 2 * margin_y + (N_c - 1) * Delta_y + self.w_trap
        self.x0_array = (self.L - self.l) / 2.0
        self.y0_array = (self.W - self.w) / 2.0

    # ------------------------------------------------------------------
    def trap_positions(self, D_f=0.0, disorder_type='uniform', seed=None):
        """
        Return list of (x0, y0) bottom-left corners for all traps.

        Iteration order: column-major (ic outer, il inner), matching the
        stagger convention: column ic (0-indexed) is shifted by +Δx/2 in x
        when ic is odd.

        Disorder (D_f > 0): each position perturbed by
            (ψ_x · D_f · Δx,  ψ_y · D_f · Δy)
        where ψ is sampled from uniform[−1, 1] or clipped Gaussian(0, 0.3).

        Parameters
        ----------
        D_f          : float  [0–1]
        disorder_type: 'uniform' or 'gaussian'
        seed         : int | None  (for reproducibility)
        """
        rng = random.Random(seed)
        positions = []
        for ic in range(self.N_c):
            stagger = (self.Delta_x / 2.0) if (ic % 2 == 1) else 0.0
            for il in range(self.N_l):
                x0 = (self.x0_array + self.margin_x
                       + il * self.Delta_x + stagger)
                y0 = self.y0_array + self.margin_y + ic * self.Delta_y
                if D_f > 0.0:
                    x0 += _sample_psi(rng, disorder_type) * D_f * self.Delta_x
                    y0 += _sample_psi(rng, disorder_type) * D_f * self.Delta_y
                positions.append((x0, y0))
        return positions

    # ------------------------------------------------------------------
    def summary(self):
        """Return a human-readable description of the resolved geometry."""
        lines = [
            f"Mode        : {self.mode}",
            f"Cavity      : L = {self.L} µm,  W = {self.W} µm",
            f"Grid        : {self.N_l} lines × {self.N_c} columns"
            f"  =  {self.N_traps} traps",
            f"Pitch       : Δx = {self.Delta_x:.3f} µm,"
            f"  Δy = {self.Delta_y:.3f} µm",
            f"Array span  : {self.l:.3f} (x) × {self.w:.3f} (y) µm",
            f"Margins     : x ± {self.margin_x:.3f} µm,"
            f"  y ± {self.margin_y:.3f} µm",
            f"Array origin: ({self.x0_array:.3f}, {self.y0_array:.3f}) µm",
            f"Feasible    : {self.feasible}",
        ]
        for w in self.warnings:
            lines.append(f"  ⚠  {w}")
        return "\n".join(lines)

    def __repr__(self):
        return (f"ArrayGrid({self.mode}, "
                f"N={self.N_l}×{self.N_c}, "
                f"Δx={self.Delta_x:.2f}, Δy={self.Delta_y:.2f})")


# ---------------------------------------------------------------------------
# Helpers
# ---------------------------------------------------------------------------

def _sample_psi(rng, disorder_type):
    """Sample ψ ∈ [−1, 1] from the chosen distribution."""
    if disorder_type == 'gaussian':
        return max(-1.0, min(1.0, rng.gauss(0.0, 0.3)))
    return rng.uniform(-1.0, 1.0)
').decode(),
'core/builder.py': __import__("base64").b64decode('"""
core/builder.py
---------------
TrappingArrayBuilder: combines an ArrayGrid with trap-shape and IO
parameters to produce the full chip layout in a pya cell.
"""

from .primitives import to_dbu, rect, draw_trap
from .io_shapes import (draw_direct_channels, draw_diverging_channels,
                        diverging_half_angle)

import pya


class TrappingArrayBuilder:
    """
    Parameters
    ----------
    grid          : ArrayGrid
    l_trap        : float  – trap streamwise length (µm)
    w_trap        : float  – trap lateral outer width (µm)
    wall_t        : float  – pillar wall thickness (µm)
    base_t        : float  – base bar depth on downstream face (µm)
    w_o_down      : float  – downstream slit width (µm)
                             must satisfy 0 < w_o_down < w_trap − 2·wall_t
    io_mode       : str    – 'direct' or 'diverging'
    C             : float  – IO centring [0–1], direct mode only
                             (1 = channels aligned, 0 = maximum offset)
    w_io          : float  – inlet/outlet channel width (µm)
    channel_len   : float  – stub length protruding outside chip (µm)
    l_div         : float  – diverging section length (µm), diverging mode only
    D_f           : float  – disorder factor [0–1]  (0 = regular)
    disorder_type : str    – 'uniform' or 'gaussian'
    seed          : int | None  – random seed for disorder (None = random)

    Read-only properties
    --------------------
    w_o               : upstream opening width  = w_trap − 2·wall_t
    Delta_y_io        : inlet lateral offset (direct mode)
    div_half_angle_deg: diverging half-angle in degrees (diverging mode)
    """

    def __init__(self, grid, l_trap, w_trap, wall_t, base_t, w_o_down,
                 io_mode='direct', C=1.0, w_io=100.0, channel_len=100.0,
                 l_div=200.0, D_f=0.0, disorder_type='uniform', seed=None):

        w_o = w_trap - 2.0 * wall_t

        # --- validation ---
        if w_o <= 0.0:
            raise ValueError(
                f"w_trap ({w_trap}) − 2·wall_t ({wall_t}) = {w_o:.3f} ≤ 0. "
                "Increase w_trap or reduce wall_t.")
        if not (0.0 < w_o_down < w_o):
            raise ValueError(
                f"w_o_down ({w_o_down}) must be in "
                f"(0, w_o = {w_o:.3f}).")
        if base_t >= l_trap:
            raise ValueError(
                f"base_t ({base_t}) must be < l_trap ({l_trap}).")
        if io_mode not in ('direct', 'diverging'):
            raise ValueError(
                f"io_mode must be 'direct' or 'diverging', got '{io_mode}'.")
        if io_mode == 'diverging' and w_io >= grid.W:
            raise ValueError(
                f"w_io ({w_io}) must be < W ({grid.W}) in diverging mode.")
        if w_io <= 0.0:
            raise ValueError("w_io must be positive.")
        if channel_len < 0.0:
            raise ValueError("channel_len must be ≥ 0.")

        self.grid          = grid
        self.l_trap        = float(l_trap)
        self.w_trap        = float(w_trap)
        self.wall_t        = float(wall_t)
        self.base_t        = float(base_t)
        self.w_o_down      = float(w_o_down)
        self.w_o           = float(w_o)
        self.io_mode       = io_mode
        self.C             = float(C)
        self.w_io          = float(w_io)
        self.channel_len   = float(channel_len)
        self.l_div         = float(l_div)
        self.D_f           = float(D_f)
        self.disorder_type = disorder_type
        self.seed          = seed

        # derived IO quantities
        self.Delta_y_io = (1.0 - C) * (grid.W - w_io) / 2.0
        self.div_half_angle_deg = (
            diverging_half_angle(grid.W, w_io, l_div)
            if io_mode == 'diverging' else 0.0
        )

    # ------------------------------------------------------------------
    def build(self, layout, layer, cell_name="TrapArray"):
        """
        Create a new named cell in `layout`, populate it, and return it.

        Parameters
        ----------
        layout    : pya.Layout
        layer     : int  – layer index (from layout.layer(...))
        cell_name : str
        """
        cvt  = to_dbu(layout.dbu)
        cell = layout.create_cell(cell_name)
        self._draw_all(cell, layer, cvt)
        return cell

    def build_into(self, cell, layer):
        """
        Insert all shapes into an existing `cell`.
        Useful when composing multiple arrays into one layout (e.g. 4-in-parallel).
        """
        cvt = to_dbu(cell.layout().dbu)
        self._draw_all(cell, layer, cvt)

    # ------------------------------------------------------------------
    def _draw_all(self, cell, layer, cvt):
        self._draw_chamber(cell, layer, cvt)
        self._draw_io(cell, layer, cvt)
        self._draw_traps(cell, layer, cvt)

    def _draw_chamber(self, cell, layer, cvt):
        rect(cell, layer, cvt, 0, 0, self.grid.L, self.grid.W)

    def _draw_io(self, cell, layer, cvt):
        if self.io_mode == 'direct':
            draw_direct_channels(
                cell, layer, cvt,
                self.grid, self.w_io, self.channel_len, self.Delta_y_io)
        else:
            draw_diverging_channels(
                cell, layer, cvt,
                self.grid, self.w_io, self.channel_len, self.l_div)

    def _draw_traps(self, cell, layer, cvt):
        t  = self.wall_t
        positions = self.grid.trap_positions(
            D_f=self.D_f,
            disorder_type=self.disorder_type,
            seed=self.seed,
        )
        for (x0, y0) in positions:
            # clamp disordered traps to stay inside chamber with wall_t margin
            x0 = max(t, min(self.grid.L - self.l_trap - t, x0))
            y0 = max(t, min(self.grid.W - self.w_trap - t, y0))
            draw_trap(cell, layer, cvt,
                      x0, y0,
                      self.l_trap, self.w_trap,
                      self.wall_t, self.base_t, self.w_o_down)

    # ------------------------------------------------------------------
    def summary(self):
        """Return a human-readable description of the full chip geometry."""
        lines = [self.grid.summary(), ""]
        lines.append(f"IO mode     : {self.io_mode}")
        if self.io_mode == 'direct':
            lines.append(
                f"  C = {self.C}  →  Δy_io = {self.Delta_y_io:.3f} µm")
        else:
            lines.append(
                f"  l_div = {self.l_div} µm  "
                f"→  half-angle = {self.div_half_angle_deg:.2f}°")
        lines += [
            f"Channel     : w_io = {self.w_io} µm,"
            f"  stub = {self.channel_len} µm",
            f"Trap        : l_trap = {self.l_trap} µm,"
            f"  w_trap = {self.w_trap} µm,",
            f"              wall_t = {self.wall_t} µm,"
            f"  base_t = {self.base_t} µm",
            f"  w_o (upstream opening) = {self.w_o:.3f} µm",
            f"  w_o_down (downstream slit) = {self.w_o_down} µm",
            f"Disorder    : D_f = {self.D_f}  ({self.disorder_type})",
        ]
        return "\n".join(lines)

    def __repr__(self):
        return (f"TrappingArrayBuilder("
                f"grid={self.grid!r}, "
                f"io_mode='{self.io_mode}', "
                f"D_f={self.D_f})")
').decode(),
'core/primitives.py': __import__("base64").b64decode('IiIiCmNvcmUvcHJpbWl0aXZlcy5weQotLS0tLS0tLS0tLS0tLS0tLS0KTG93LWxldmVsIHB5YSBkcmF3aW5nIGhlbHBlcnMgYW5kIHRoZSBzaW5nbGUtdHJhcCBwb2x5Z29uIGdlbmVyYXRvci4KTm8gY2xhc3Mgc3RhdGU7IGFsbCBmdW5jdGlvbnMgYXJlIHB1cmUgZ2l2ZW4gdGhlaXIgYXJndW1lbnRzLgoKQ29vcmRpbmF0ZSBjb252ZW50aW9uIChzaGFyZWQgYWNyb3NzIHRoZSB3aG9sZSBsaWJyYXJ5KQogICAgeCA9IHN0cmVhbXdpc2UgIChpbmNyZWFzZXMgdG93YXJkIGlubGV0IC8gdXBzdHJlYW0pCiAgICB5ID0gbGF0ZXJhbAogICAgT3JpZ2luIGF0IGJvdHRvbS1sZWZ0IGNvcm5lciBvZiB0aGUgY2hhbWJlciByZWN0YW5nbGUuCgpUcmFwIHNoYXBlOiBpbnZlcnRlZC1VIHdpdGggZG93bnN0cmVhbSBzbGl0CiAgICBUd28gaG9yaXpvbnRhbCBwaWxsYXJzICh3YWxsX3QgdGhpY2spIHJ1bm5pbmcgdGhlIGZ1bGwgbF90cmFwIGluIHguCiAgICBBIGJhc2UgYmFyIChiYXNlX3QgZGVlcCBpbiB4KSBvbiB0aGUgZG93bnN0cmVhbSBmYWNlICh4ID0geDApLAogICAgc3BsaXQgYnkgYSBjZW50cmVkIHNsaXQgb2Ygd2lkdGggd19vX2Rvd24uCiAgICBUaGUgdXBzdHJlYW0gZmFjZSAoeCA9IHgwICsgbF90cmFwKSBpcyBmdWxseSBvcGVuIG92ZXIKICAgIHdfbyA9IHdfdHJhcCAtIDIqd2FsbF90LgoiIiIKCmltcG9ydCBweWEKCgpkZWYgdG9fZGJ1KGxheW91dF9vcl9kYnUpOgogICAgIiIiUmV0dXJuIGEgwrVt4oaSZGJ1IGNvbnZlcnRlciBmb3IgdGhlIGdpdmVuIGxheW91dCBvciBkYnUgZmxvYXQuIiIiCiAgICBkYnUgPSBsYXlvdXRfb3JfZGJ1IGlmIGlzaW5zdGFuY2UobGF5b3V0X29yX2RidSwgZmxvYXQpIGVsc2UgbGF5b3V0X29yX2RidS5kYnUKICAgIHJldHVybiBsYW1iZGEgdjogaW50KHJvdW5kKHYgLyBkYnUpKQoKCmRlZiByZWN0KGNlbGwsIGxheWVyLCBjdnQsIHgwLCB5MCwgeDEsIHkxKToKICAgICIiIkluc2VydCBhbiBheGlzLWFsaWduZWQgcmVjdGFuZ2xlICjCtW0gY29vcmRzLCBjb252ZXJ0ZWQgYnkgY3Z0KS4iIiIKICAgIHB0cyA9IFsKICAgICAgICBweWEuUG9pbnQoY3Z0KHgwKSwgY3Z0KHkwKSksCiAgICAgICAgcHlhLlBvaW50KGN2dCh4MSksIGN2dCh5MCkpLAogICAgICAgIHB5YS5Qb2ludChjdnQoeDEpLCBjdnQoeTEpKSwKICAgICAgICBweWEuUG9pbnQoY3Z0KHgwKSwgY3Z0KHkxKSksCiAgICBdCiAgICBjZWxsLnNoYXBlcyhsYXllcikuaW5zZXJ0KHB5YS5Qb2x5Z29uKHB0cykpCgoKZGVmIHBvbHlnb24oY2VsbCwgbGF5ZXIsIGN2dCwgcHRzX3VtKToKICAgICIiIkluc2VydCBhbiBhcmJpdHJhcnkgcG9seWdvbiBmcm9tIGEgbGlzdCBvZiAoeCwgeSkgwrVtIHR1cGxlcy4iIiIKICAgIHB0cyA9IFtweWEuUG9pbnQoY3Z0KHgpLCBjdnQoeSkpIGZvciB4LCB5IGluIHB0c191bV0KICAgIGNlbGwuc2hhcGVzKGxheWVyKS5pbnNlcnQocHlhLlBvbHlnb24ocHRzKSkKCgpkZWYgZHJhd190cmFwKGNlbGwsIGxheWVyLCBjdnQsIHgwLCB5MCwgbF90cmFwLCB3X3RyYXAsIHdhbGxfdCwgYmFzZV90LCB3X29fZG93bik6CiAgICAiIiIKICAgIERyYXcgYSBzaW5nbGUgdHJhcCBhdCBib3R0b20tbGVmdCBjb3JuZXIgKHgwLCB5MCkuCgogICAgUGFyYW1ldGVycwogICAgLS0tLS0tLS0tLQogICAgeDAsIHkwICAgOiDCtW0g4oCTIGJvdHRvbS1sZWZ0IG9mIHRyYXAgYm91bmRpbmcgYm94CiAgICBsX3RyYXAgICA6IMK1bSDigJMgdG90YWwgc3RyZWFtd2lzZSBsZW5ndGggKG9wZW4gZmFjZSBhdCB4MCtsX3RyYXApCiAgICB3X3RyYXAgICA6IMK1bSDigJMgdG90YWwgbGF0ZXJhbCBvdXRlciB3aWR0aAogICAgd2FsbF90ICAgOiDCtW0g4oCTIHBpbGxhciB3YWxsIHRoaWNrbmVzcyAodG9wIGFuZCBib3R0b20gaW4geSkKICAgIGJhc2VfdCAgIDogwrVtIOKAkyBiYXNlIGJhciBkZXB0aCAoZG93bnN0cmVhbSBmYWNlLCBpbiB4KQogICAgd19vX2Rvd24gOiDCtW0g4oCTIGRvd25zdHJlYW0gc2xpdCB3aWR0aCAoY2VudHJlZCB3aXRoaW4gaW50ZXJpb3IpCgogICAgV2FsbCBjb21wb25lbnRzCiAgICAtLS0tLS0tLS0tLS0tLS0KICAgIEJvdHRvbSBwaWxsYXIgOiB4IOKIiCBbeDAsICAgICAgICB4MCtsX3RyYXBdLCAgeSDiiIggW3kwLCAgICAgICAgICAgIHkwK3dhbGxfdF0KICAgIFRvcCAgICBwaWxsYXIgOiB4IOKIiCBbeDAsICAgICAgICB4MCtsX3RyYXBdLCAgeSDiiIggW3kwK3dfdHJhcC10LCAgIHkwK3dfdHJhcF0KICAgIEJhc2UgbGVmdCBzdHViOiB4IOKIiCBbeDAsICAgICAgICB4MCtiYXNlX3RdLCAgeSDiiIggW3kwK3dhbGxfdCwgICAgIHNsaXRfeTAgIF0KICAgIEJhc2UgcmdodCBzdHViOiB4IOKIiCBbeDAsICAgICAgICB4MCtiYXNlX3RdLCAgeSDiiIggW3NsaXRfeTEsICAgICAgIHkwK3dfdHJhcC10XQoKICAgIHdoZXJlCiAgICAgIHdfbyAgICAgPSB3X3RyYXAgLSAyKndhbGxfdCAgICAgICAgICAoaW50ZXJpb3IgLyB1cHN0cmVhbSBvcGVuaW5nIHdpZHRoKQogICAgICBzbGl0X3kwID0geTAgKyB3YWxsX3QgKyAod19vIC0gd19vX2Rvd24pIC8gMgogICAgICBzbGl0X3kxID0gc2xpdF95MCArIHdfb19kb3duCiAgICAiIiIKICAgIHQgICA9IHdhbGxfdAogICAgYnQgID0gYmFzZV90CiAgICB3X28gPSB3X3RyYXAgLSAyICogdAoKICAgIHNsaXRfeTAgPSB5MCArIHQgKyAod19vIC0gd19vX2Rvd24pIC8gMi4wCiAgICBzbGl0X3kxID0gc2xpdF95MCArIHdfb19kb3duCgogICAgIyBib3R0b20gcGlsbGFyICjiiJJ5IHNpZGUsIGZ1bGwgbGVuZ3RoKQogICAgcmVjdChjZWxsLCBsYXllciwgY3Z0LCB4MCwgeTAsIHgwICsgbF90cmFwLCB5MCArIHQpCgogICAgIyB0b3AgcGlsbGFyICgreSBzaWRlLCBmdWxsIGxlbmd0aCkKICAgIHJlY3QoY2VsbCwgbGF5ZXIsIGN2dCwgeDAsIHkwICsgd190cmFwIC0gdCwgeDAgKyBsX3RyYXAsIHkwICsgd190cmFwKQoKICAgICMgYmFzZSBiYXIgbGVmdCBzdHViIChiZWxvdyBzbGl0KQogICAgcmVjdChjZWxsLCBsYXllciwgY3Z0LCB4MCwgeTAgKyB0LCB4MCArIGJ0LCBzbGl0X3kwKQoKICAgICMgYmFzZSBiYXIgcmlnaHQgc3R1YiAoYWJvdmUgc2xpdCkKICAgIHJlY3QoY2VsbCwgbGF5ZXIsIGN2dCwgeDAsIHNsaXRfeTEsIHgwICsgYnQsIHkwICsgd190cmFwIC0gdCkK').decode(),
'core/io_shapes.py': __import__("base64").b64decode('IiIiCmNvcmUvaW9fc2hhcGVzLnB5Ci0tLS0tLS0tLS0tLS0tLS0tCklubGV0IGFuZCBvdXRsZXQgY2hhbm5lbCBkcmF3aW5nIGZvciBib3RoIElPIG1vZGVzLgoKJ2RpcmVjdCcgICAg4oCTIHJlY3Rhbmd1bGFyIHN0dWJzIGNvbm5lY3RpbmcgZGlyZWN0bHkgdG8gdGhlIGNoYW1iZXIgd2FsbHMuCiAgICAgICAgICAgICAgVGhlIGlubGV0IG1heSBiZSBsYXRlcmFsbHkgb2Zmc2V0IGJ5IM6UeV9pbyA9ICgx4oiSQynCtyhX4oiSd19pbykvMi4KICAgICAgICAgICAgICBUaGUgb3V0bGV0IGlzIGFsd2F5cyBjZW50cmVkIG9uIFcvMi4KCidkaXZlcmdpbmcnIOKAkyBzdHJhaWdodCB0cmFwZXpvaWRhbCB0cmFuc2l0aW9uIHNlY3Rpb25zIGZhbiB0aGUgY2hhbm5lbCBmcm9tCiAgICAgICAgICAgICAgd19pbyB0byB0aGUgZnVsbCBjaGFtYmVyIHdpZHRoIFcgb3ZlciBsZW5ndGggbF9kaXYsIHRoZW4gYQogICAgICAgICAgICAgIHJlY3Rhbmd1bGFyIHN0dWIgb2YgbGVuZ3RoIGNoYW5uZWxfbGVuIGJleW9uZCB0aGF0LgogICAgICAgICAgICAgIEJvdGggY2hhbm5lbHMgYXJlIGNlbnRyZWQgb24gVy8yOyBDIGlzIGlycmVsZXZhbnQuCiIiIgoKaW1wb3J0IG1hdGgKZnJvbSAucHJpbWl0aXZlcyBpbXBvcnQgcmVjdCwgcG9seWdvbgoKCmRlZiBkcmF3X2RpcmVjdF9jaGFubmVscyhjZWxsLCBsYXllciwgY3Z0LCBncmlkLCB3X2lvLCBjaGFubmVsX2xlbiwgRGVsdGFfeV9pbyk6CiAgICAiIiIKICAgIERyYXcgaW5sZXQgYW5kIG91dGxldCBhcyBwbGFpbiByZWN0YW5ndWxhciBzdHVicy4KCiAgICBJbmxldCAgKHJpZ2h0IHNpZGUsICt4KTogIGNlbnRyZWQgYXQgVy8yICsgRGVsdGFfeV9pbwogICAgT3V0bGV0IChsZWZ0ICBzaWRlLCDiiJJ4KTogIGNlbnRyZWQgYXQgVy8yCiAgICAiIiIKICAgIFcgID0gZ3JpZC5XCiAgICBMICA9IGdyaWQuTAogICAgY2wgPSBjaGFubmVsX2xlbgogICAgd2lvID0gd19pbwoKICAgICMgb3V0bGV0CiAgICB5Y19vdXQgPSBXIC8gMi4wCiAgICByZWN0KGNlbGwsIGxheWVyLCBjdnQsCiAgICAgICAgIC1jbCwgeWNfb3V0IC0gd2lvIC8gMi4wLAogICAgICAgICAwLjAsIHljX291dCArIHdpbyAvIDIuMCkKCiAgICAjIGlubGV0CiAgICB5Y19pbiA9IFcgLyAyLjAgKyBEZWx0YV95X2lvCiAgICByZWN0KGNlbGwsIGxheWVyLCBjdnQsCiAgICAgICAgIEwsICAgICAgeWNfaW4gLSB3aW8gLyAyLjAsCiAgICAgICAgIEwgKyBjbCwgeWNfaW4gKyB3aW8gLyAyLjApCgoKZGVmIGRyYXdfZGl2ZXJnaW5nX2NoYW5uZWxzKGNlbGwsIGxheWVyLCBjdnQsIGdyaWQsIHdfaW8sIGNoYW5uZWxfbGVuLCBsX2Rpdik6CiAgICAiIiIKICAgIERyYXcgaW5sZXQgYW5kIG91dGxldCBlYWNoIGFzIGEgc3RyYWlnaHQgdHJhcGV6b2lkICsgcmVjdGFuZ3VsYXIgc3R1YiwKICAgIGJvdGggY2VudHJlZCBvbiBXLzIuCgogICAgSW5sZXQgc2lkZSAoK3gpOgogICAgICAgIHN0dWIgICAgICA6IHgg4oiIIFtMK2xfZGl2LCBMK2xfZGl2K2NoYW5uZWxfbGVuXSwgIHdpZHRoIHdfaW8KICAgICAgICB0cmFwZXpvaWQgOiB3aWRlIGZhY2UgKFcpIGF0IHg9TCwgbmFycm93IGZhY2UgKHdfaW8pIGF0IHg9TCtsX2RpdgoKICAgIE91dGxldCBzaWRlICjiiJJ4KToKICAgICAgICBzdHViICAgICAgOiB4IOKIiCBb4oiSbF9kaXbiiJJjaGFubmVsX2xlbiwg4oiSbF9kaXZdLCAgd2lkdGggd19pbwogICAgICAgIHRyYXBlem9pZCA6IG5hcnJvdyBmYWNlICh3X2lvKSBhdCB4PeKIkmxfZGl2LCB3aWRlIGZhY2UgKFcpIGF0IHg9MAogICAgIiIiCiAgICBXICAgPSBncmlkLlcKICAgIEwgICA9IGdyaWQuTAogICAgY2wgID0gY2hhbm5lbF9sZW4KICAgIGxkICA9IGxfZGl2CiAgICB3aW8gPSB3X2lvCiAgICB5YyAgPSBXIC8gMi4wCgogICAgIyDilIDilIAgaW5sZXQg4pSA4pSACiAgICBwb2x5Z29uKGNlbGwsIGxheWVyLCBjdnQsIFsKICAgICAgICAoTCwgICAgICAwLjApLAogICAgICAgIChMLCAgICAgIFcpLAogICAgICAgIChMICsgbGQsIHljICsgd2lvIC8gMi4wKSwKICAgICAgICAoTCArIGxkLCB5YyAtIHdpbyAvIDIuMCksCiAgICBdKQogICAgcmVjdChjZWxsLCBsYXllciwgY3Z0LAogICAgICAgICBMICsgbGQsICAgICAgeWMgLSB3aW8gLyAyLjAsCiAgICAgICAgIEwgKyBsZCArIGNsLCB5YyArIHdpbyAvIDIuMCkKCiAgICAjIOKUgOKUgCBvdXRsZXQg4pSA4pSACiAgICBwb2x5Z29uKGNlbGwsIGxheWVyLCBjdnQsIFsKICAgICAgICAoLWxkLCB5YyAtIHdpbyAvIDIuMCksCiAgICAgICAgKC1sZCwgeWMgKyB3aW8gLyAyLjApLAogICAgICAgICgwLjAsIFcpLAogICAgICAgICgwLjAsIDAuMCksCiAgICBdKQogICAgcmVjdChjZWxsLCBsYXllciwgY3Z0LAogICAgICAgICAtbGQgLSBjbCwgeWMgLSB3aW8gLyAyLjAsCiAgICAgICAgIC1sZCwgICAgICB5YyArIHdpbyAvIDIuMCkKCgpkZWYgZGl2ZXJnaW5nX2hhbGZfYW5nbGUoVywgd19pbywgbF9kaXYpOgogICAgIiIiUmV0dXJuIHRoZSBoYWxmLWFuZ2xlIG9mIHRoZSBkaXZlcmdpbmcgc2VjdGlvbiBpbiBkZWdyZWVzLiIiIgogICAgaWYgbF9kaXYgPD0gMDoKICAgICAgICByZXR1cm4gOTAuMAogICAgcmV0dXJuIG1hdGguZGVncmVlcyhtYXRoLmF0YW4oKFcgLSB3X2lvKSAvICgyLjAgKiBsX2RpdikpKQo=').decode(),
'trapping_array_lib.py': __import__("base64").b64decode('"""
trapping_array_lib.py
=====================
KLayout PCell library — registers "Trapping Array" in the Libraries panel.

Two PCells:
  TrapArray_A  —  fixed N_l × N_c grid
  TrapArray_B  —  maximise trap count given pitch

Each PCell declares its parameters so KLayout renders the edit form
automatically when the user places or double-clicks an instance.
"""

import pya, sys, os

# Make sure the extracted core package is importable
_pkg = os.path.join(os.path.expanduser("~"), ".klayout", "pymacros", "trapping_array")
if _pkg not in sys.path:
    sys.path.insert(0, _pkg)

from core import make_fixed_grid, make_max_traps


# ---------------------------------------------------------------------------
# Shared helpers
# ---------------------------------------------------------------------------

def _layer_index(layout, layer_param):
    """Return a usable layer index from a LayerInfo parameter value."""
    try:
        return layout.layer(layer_param)
    except Exception:
        return layout.layer(1, 0)


def _build(layout, layer, grid, builder):
    """Insert all chip shapes into the PCell's own cell."""
    # PCells get a fresh cell from KLayout; we populate it in-place.
    # build_into() expects a cell whose layout() is accessible.
    cell = layout.create_cell("__tmp__")
    builder.build_into(cell, layer)
    return cell


# ---------------------------------------------------------------------------
# Mode A PCell
# ---------------------------------------------------------------------------

class TrapArrayA(pya.PCellDeclarationHelper):
    """
    Fixed N_l × N_c hydrodynamic trapping array.
    Parameters are declared once; KLayout builds the edit dialog automatically.
    """

    def __init__(self):
        super().__init__()

        # ── Cavity ──────────────────────────────────────────────────────
        self.param("L", self.TypeDouble, "Cavity streamwise length (µm)",
                   default=1500.0)
        self.param("W", self.TypeDouble, "Cavity lateral width (µm)",
                   default=1200.0)

        # ── Grid ────────────────────────────────────────────────────────
        self.param("N_l", self.TypeInt, "Lines — streamwise (N_l)",
                   default=7)
        self.param("N_c", self.TypeInt, "Columns — lateral (N_c)",
                   default=14)
        self.param("gap_x", self.TypeDouble, "Inter-trap gap x (µm)  →  Δx = l_trap + gap_x",
                   default=10.0)
        self.param("gap_y", self.TypeDouble, "Inter-trap gap y (µm)  →  Δy = w_trap + gap_y",
                   default=10.0)
        self.param("gap_x_margin", self.TypeDouble,
                   "Streamwise wall margin (µm)  [0 = use gap_x]",
                   default=0.0)
        self.param("gap_y_margin", self.TypeDouble,
                   "Lateral wall margin (µm)  [0 = use Δy/2]",
                   default=0.0)

        # ── Trap shape ───────────────────────────────────────────────────
        self.param("l_trap",   self.TypeDouble, "l_trap — streamwise length (µm)",  default=30.0)
        self.param("w_trap",   self.TypeDouble, "w_trap — lateral outer width (µm)", default=25.0)
        self.param("wall_t",   self.TypeDouble, "wall_t — pillar thickness (µm)",    default=3.0)
        self.param("base_t",   self.TypeDouble, "base_t — base bar depth (µm)",      default=4.0)
        self.param("w_o_down", self.TypeDouble, "w_o_down — downstream slit (µm)",   default=8.0)

        # ── IO ───────────────────────────────────────────────────────────
        self.param("io_mode", self.TypeString,
                   "IO mode: 'direct' or 'diverging'", default="direct")
        self.param("C", self.TypeDouble,
                   "Centring C [0–1] (direct mode only)  1=aligned", default=1.0)
        self.param("w_io",       self.TypeDouble, "Channel width (µm)",              default=100.0)
        self.param("channel_len",self.TypeDouble, "Stub length outside chip (µm)",   default=100.0)
        self.param("l_div",      self.TypeDouble, "Diverging section length (µm)",   default=200.0)

        # ── Disorder ─────────────────────────────────────────────────────
        self.param("D_f",           self.TypeDouble, "Disorder factor D_f [0–1]",    default=0.0)
        self.param("disorder_type", self.TypeString,
                   "Disorder type: 'uniform' or 'gaussian'", default="uniform")
        self.param("seed", self.TypeInt, "Random seed (0 = random)", default=42)

        # ── Layer ────────────────────────────────────────────────────────
        self.param("layer", self.TypeLayer, "GDS layer", default=pya.LayerInfo(1, 0))

    def display_text_impl(self):
        return f"TrapArray_A  {self.N_l}×{self.N_c}  {self.io_mode}"

    def coerce_parameters_impl(self):
        # Keep N_l, N_c positive
        if self.N_l < 1: self.N_l = 1
        if self.N_c < 1: self.N_c = 1
        # Clamp C
        self.C = max(0.0, min(1.0, self.C))
        # Clamp D_f
        self.D_f = max(0.0, min(1.0, self.D_f))

    def produce_impl(self):
        layer = _layer_index(self.layout, self.layer)
        gxm   = self.gap_x_margin if self.gap_x_margin > 0 else None
        gym   = self.gap_y_margin if self.gap_y_margin > 0 else None
        seed  = self.seed if self.seed > 0 else None

        try:
            grid, builder = make_fixed_grid(
                L=self.L, W=self.W,
                N_l=self.N_l, N_c=self.N_c,
                gap_x=self.gap_x, gap_y=self.gap_y,
                gap_x_margin=gxm, gap_y_margin=gym,
                l_trap=self.l_trap, w_trap=self.w_trap,
                wall_t=self.wall_t, base_t=self.base_t,
                w_o_down=self.w_o_down,
                io_mode=self.io_mode, C=self.C,
                w_io=self.w_io, channel_len=self.channel_len,
                l_div=self.l_div,
                D_f=self.D_f, disorder_type=self.disorder_type,
                seed=seed,
            )
        except (ValueError, TypeError) as exc:
            # Draw a warning marker so the cell isn't silently empty
            _draw_error_marker(self.cell, layer, self.layout.dbu, str(exc))
            return

        builder.build_into(self.cell, layer)


# ---------------------------------------------------------------------------
# Mode B PCell
# ---------------------------------------------------------------------------

class TrapArrayB(pya.PCellDeclarationHelper):
    """
    Maximise trap count given cavity and pitch.
    """

    def __init__(self):
        super().__init__()

        # ── Cavity ──────────────────────────────────────────────────────
        self.param("L", self.TypeDouble, "Cavity streamwise length (µm)",
                   default=1500.0)
        self.param("W", self.TypeDouble, "Cavity lateral width (µm)",
                   default=1200.0)

        # ── Pitch ────────────────────────────────────────────────────────
        self.param("Delta_x", self.TypeDouble,
                   "Δx — streamwise pitch (µm)  =  l_trap + gap_x", default=40.0)
        self.param("Delta_y", self.TypeDouble,
                   "Δy — lateral pitch (µm)  =  w_trap + gap_y", default=35.0)
        self.param("margin_x", self.TypeDouble,
                   "Streamwise wall margin (µm)  [0 = Δy/2]", default=0.0)
        self.param("margin_y", self.TypeDouble,
                   "Lateral wall margin (µm)  [0 = Δy/2]", default=0.0)

        # ── Trap shape ───────────────────────────────────────────────────
        self.param("l_trap",   self.TypeDouble, "l_trap — streamwise length (µm)",  default=30.0)
        self.param("w_trap",   self.TypeDouble, "w_trap — lateral outer width (µm)", default=25.0)
        self.param("wall_t",   self.TypeDouble, "wall_t — pillar thickness (µm)",    default=3.0)
        self.param("base_t",   self.TypeDouble, "base_t — base bar depth (µm)",      default=4.0)
        self.param("w_o_down", self.TypeDouble, "w_o_down — downstream slit (µm)",   default=8.0)

        # ── IO ───────────────────────────────────────────────────────────
        self.param("io_mode", self.TypeString,
                   "IO mode: 'direct' or 'diverging'", default="direct")
        self.param("C", self.TypeDouble,
                   "Centring C [0–1] (direct mode only)  1=aligned", default=1.0)
        self.param("w_io",        self.TypeDouble, "Channel width (µm)",             default=100.0)
        self.param("channel_len", self.TypeDouble, "Stub length outside chip (µm)",  default=100.0)
        self.param("l_div",       self.TypeDouble, "Diverging section length (µm)",  default=200.0)

        # ── Disorder ─────────────────────────────────────────────────────
        self.param("D_f",           self.TypeDouble, "Disorder factor D_f [0–1]",    default=0.0)
        self.param("disorder_type", self.TypeString,
                   "Disorder type: 'uniform' or 'gaussian'", default="uniform")
        self.param("seed", self.TypeInt, "Random seed (0 = random)", default=42)

        # ── Layer ────────────────────────────────────────────────────────
        self.param("layer", self.TypeLayer, "GDS layer", default=pya.LayerInfo(1, 0))

    def display_text_impl(self):
        return (f"TrapArray_B  Δx={self.Delta_x} Δy={self.Delta_y}"
                f"  {self.io_mode}")

    def coerce_parameters_impl(self):
        if self.Delta_x <= 0: self.Delta_x = 1.0
        if self.Delta_y <= 0: self.Delta_y = 1.0
        self.C   = max(0.0, min(1.0, self.C))
        self.D_f = max(0.0, min(1.0, self.D_f))

    def produce_impl(self):
        layer = _layer_index(self.layout, self.layer)
        mxb   = self.margin_x if self.margin_x > 0 else None
        myb   = self.margin_y if self.margin_y > 0 else None
        seed  = self.seed if self.seed > 0 else None

        try:
            grid, builder = make_max_traps(
                L=self.L, W=self.W,
                Delta_x=self.Delta_x, Delta_y=self.Delta_y,
                margin_x=mxb, margin_y=myb,
                l_trap=self.l_trap, w_trap=self.w_trap,
                wall_t=self.wall_t, base_t=self.base_t,
                w_o_down=self.w_o_down,
                io_mode=self.io_mode, C=self.C,
                w_io=self.w_io, channel_len=self.channel_len,
                l_div=self.l_div,
                D_f=self.D_f, disorder_type=self.disorder_type,
                seed=seed,
            )
        except (ValueError, TypeError) as exc:
            _draw_error_marker(self.cell, layer, self.layout.dbu, str(exc))
            return

        builder.build_into(self.cell, layer)


# ---------------------------------------------------------------------------
# Error marker (drawn when parameters are invalid)
# ---------------------------------------------------------------------------

def _draw_error_marker(cell, layer, dbu, message):
    """Draw a 100×100 µm cross so the user sees something went wrong."""
    cvt = lambda v: int(round(v / dbu))
    size = 100
    t    = 5
    pts_h = [
        pya.Point(cvt(-size//2), cvt(-t//2)),
        pya.Point(cvt( size//2), cvt(-t//2)),
        pya.Point(cvt( size//2), cvt( t//2)),
        pya.Point(cvt(-size//2), cvt( t//2)),
    ]
    pts_v = [
        pya.Point(cvt(-t//2), cvt(-size//2)),
        pya.Point(cvt( t//2), cvt(-size//2)),
        pya.Point(cvt( t//2), cvt( size//2)),
        pya.Point(cvt(-t//2), cvt( size//2)),
    ]
    cell.shapes(layer).insert(pya.Polygon(pts_h))
    cell.shapes(layer).insert(pya.Polygon(pts_v))
    print(f"[TrapArray] Parameter error: {message}")


# ---------------------------------------------------------------------------
# Library registration
# ---------------------------------------------------------------------------

class TrappingArrayLibrary(pya.Library):

    def __init__(self):
        super().__init__()
        self.description = "Microfluidic trapping arrays (Ruyssen et al. 2025)"
        self.layout().register_pcell("TrapArray_A — Fixed grid",   TrapArrayA())
        self.layout().register_pcell("TrapArray_B — Max traps",    TrapArrayB())
        self.register("Trapping Array")


# Instantiate — this registers the library with KLayout
_lib_instance = TrappingArrayLibrary()
').decode(),
}
_D = {
'core/__init__.py': os.path.join(_core, '__init__.py'),
'core/grid.py': os.path.join(_core, 'grid.py'),
'core/builder.py': os.path.join(_core, 'builder.py'),
'core/primitives.py': os.path.join(_core, 'primitives.py'),
'core/io_shapes.py': os.path.join(_core, 'io_shapes.py'),
'trapping_array_lib.py': os.path.join(_pkg, 'trapping_array_lib.py'),
}
for _k, _v in _D.items():
with open(_v, "w", encoding="utf-8") as _f:
_f.write(_M[_k])
if _pkg not in sys.path:
sys.path.insert(0, _pkg)
import trapping_array_lib as _tlib
importlib.reload(_tlib)
</text>
</klayout-macro>