# This code is part of KQCircuits
# Copyright (C) 2021 IQM Finland Oy
#
# This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public
# License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later
# version.
#
# This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied
# warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License along with this program. If not, see
# https://www.gnu.org/licenses/gpl-3.0.html.
#
# The software distribution should follow IQM trademark policy for open-source software
# (meetiqm.com/iqm-open-source-trademark-policy). IQM welcomes contributions to the code.
# Please see our contribution agreements for individuals (meetiqm.com/iqm-individual-contributor-license-agreement)
# and organizations (meetiqm.com/iqm-organization-contributor-license-agreement).
import math
from kqcircuits.pya_resolver import pya
from kqcircuits.util.parameters import Param, pdt, add_parameters_from
from kqcircuits.elements.element import Element
from kqcircuits.elements.waveguide_coplanar_straight import WaveguideCoplanarStraight
from kqcircuits.elements.waveguide_coplanar_curved import WaveguideCoplanarCurved
[docs]
@add_parameters_from(WaveguideCoplanarStraight, "add_metal", "ground_grid_in_trace")
class WaveguideCoplanar(Element):
    """The PCell declaration for an arbitrary coplanar waveguide.
    Coplanar waveguide defined by the width of the center conductor and gap. It can follow any segmented lines with
    predefined bending radios. It actually consists of straight and bent PCells. Termination lengths are lengths of
    extra ground gaps for opened transmission lines
    The ``path`` parameter defines the waypoints of the waveguide. When a DPath is supplied, the waypoints can be edited
    in the KLayout GUI with the Partial tool. Alternatively, a list of DPoint can be supplied, in which case the
    guiding shape is not visible in the GUI. This is useful for code-generated (sub)cells where graphical editing is not
    possible or desired.
    Warning:
        Arbitrary angle bents can have very small gaps between bends and straight segments due to
        precision of arithmetic.
    .. MARKERS_FOR_PNG 20,-10 50,0
    """
    path = Param(pdt.TypeShape, "TLine", pya.DPath([pya.DPoint(0, 0), pya.DPoint(100, 0)], 0))
    term1 = Param(pdt.TypeDouble, "Termination length start", 0, unit="μm")
    term2 = Param(pdt.TypeDouble, "Termination length end", 0, unit="μm")
[docs]
    def can_create_from_shape_impl(self):
        return self.shape.is_path() 
[docs]
    def parameters_from_shape_impl(self):
        points = [pya.DPoint(point * self.layout.dbu) for point in self.shape.each_point()]
        self.path = pya.DPath(points, 1) 
[docs]
    def produce_waveguide(self):
        if isinstance(self.path, list):
            points = [pya.DPoint(p) if isinstance(p, pya.DVector) else p for p in self.path]
        else:
            points = list(self.path.each_point())
        if len(points) < 2:
            self.raise_error_on_cell(
                "Need at least 2 points for a waveguide.", points[0] if len(points) == 1 else pya.DPoint()
            )
        eps_length = 0.5 * self.layout.dbu
        last_cut_dist = 0.0
        # For each segment except the last
        for i in range(0, len(points) - 2):
            # Check if straight can fit between points[i] and points[i + 1]
            v1, _, alpha1, alpha2, corner_pos = self.get_corner_data(points[i], points[i + 1], points[i + 2], self.r)
            alpha = (alpha2 - alpha1 + math.pi) % (2 * math.pi) - math.pi  # turn angle (between -pi and pi) in radians
            # distance between points[i + 1] and beginning of the straight
            cut_dist = self.r * math.tan(abs(alpha) / 2)
            straight_length = v1.length() - last_cut_dist - cut_dist
            if straight_length < -eps_length:
                self.raise_error_on_cell(
                    "Straight segment cannot fit. Try decreasing the turn radius.", points[i] + v1 / 2
                )
            # Straight segment before corner
            if straight_length > eps_length:
                start_point = points[i] + last_cut_dist / v1.length() * v1
                transf = pya.DCplxTrans(1, math.degrees(alpha1), False, start_point)
                WaveguideCoplanarStraight.build_geometry(self, transf, straight_length)
            # Curved segment at the corner
            if 2 * cut_dist > eps_length:
                transf = pya.DCplxTrans(1, math.degrees(alpha1) + (90 if alpha < 0 else -90), False, corner_pos)
                WaveguideCoplanarCurved.build_geometry(self, transf, alpha)
            # Prepare for next iteration
            last_cut_dist = cut_dist
        # Check if straight can fit between the last two points
        v1 = points[-1] - points[-2]
        straight_length = v1.length() - last_cut_dist
        if straight_length < -eps_length:
            self.raise_error_on_cell(
                "Straight segment cannot fit. Try decreasing the turn radius.", points[-2] + v1 / 2
            )
        # Straight segment at the end
        if straight_length > eps_length:
            start_point = points[-2] + last_cut_dist / v1.length() * v1
            transf = pya.DCplxTrans(1, math.degrees(math.atan2(v1.y, v1.x)), False, start_point)
            WaveguideCoplanarStraight.build_geometry(self, transf, straight_length)
        # Termination before the first segment
        WaveguideCoplanar.produce_end_termination(self, points[1], points[0], self.term1)
        self.add_port("a", points[0], points[0] - points[1])
        # Terminate the end
        WaveguideCoplanar.produce_end_termination(self, points[-2], points[-1], self.term2)
        self.add_port("b", points[-1], points[-1] - points[-2]) 
[docs]
    def build(self):
        self.produce_waveguide() 
[docs]
    @staticmethod
    def get_corner_data(point1, point2, point3, r):
        """Returns data needed to create a curved waveguide at path corner.
        Args:
            point1: point before corner
            point2: corner point
            point3: point after corner
            r: curve radius
        Returns:
            A tuple (``v1``, ``v2``, ``alpha1``, ``alpha2``, ``corner_pos``), where
            * ``v1``: the vector (`point2` - `point1`)
            * ``v2``: the vector (`point3` - `point2`)
            * ``alpha1``: angle between `v1` and positive x-axis
            * ``alpha2``: angle between `v2` and positive x-axis
            * ``corner_pos``: position where the curved waveguide should be placed
        """
        v1 = point2 - point1
        v2 = point3 - point2
        alpha1 = math.atan2(v1.y, v1.x)
        alpha2 = math.atan2(v2.y, v2.x)
        alpha = (alpha2 - alpha1 + math.pi) % (2 * math.pi) - math.pi  # turn angle (between -pi and pi) in radians
        alphacorner = alpha1 + (alpha + math.pi) / 2  # corner middle angle plus 90 degrees
        distcorner = (r if alpha > 0 else -r) / math.cos(alpha / 2)
        corner_pos = point2 + pya.DVector(math.cos(alphacorner) * distcorner, math.sin(alphacorner) * distcorner)
        return v1, v2, alpha1, alpha2, corner_pos 
[docs]
    @staticmethod
    def produce_end_termination(elem, point_1, point_2, term_len, face_index=0):
        """Produces termination for a waveguide.
        The termination consists of a rectangular polygon in the metal gap layer, and grid avoidance around it.
        One edge of the polygon is centered at point_2, and the polygon extends to length "term_len" in the
        direction of (point_2 - point_1).
        Args:
            elem: Element from which the waveguide parameters for the termination are taken
            point_1: DPoint before point_2, used only to determine the direction
            point_2: DPoint after which termination is produced
            term_len (double): termination length, assumed positive
            face_index (int): face index of the face in elem where the termination is created
        """
        a = elem.a
        b = elem.b
        v = (point_2 - point_1) * (1 / point_1.distance(point_2))
        u = pya.DTrans.R270.trans(v)
        shift_start = pya.DTrans(pya.DVector(point_2))
        if term_len > 0:
            poly = pya.DPolygon(
                [
                    pya.DPoint(u * (a / 2 + b)),
                    pya.DPoint(u * (a / 2 + b) + v * term_len),
                    pya.DPoint(u * (-a / 2 - b) + v * term_len),
                    pya.DPoint(u * (-a / 2 - b)),
                ]
            )
            elem.cell.shapes(elem.layout.layer(elem.face(face_index)["base_metal_gap_wo_grid"])).insert(
                poly.transform(shift_start)
            )
        # protection
        term_len += elem.margin
        poly2 = pya.DPolygon(
            [
                pya.DPoint(u * (a / 2 + b + elem.margin)),
                pya.DPoint(u * (a / 2 + b + elem.margin) + v * term_len),
                pya.DPoint(u * (-a / 2 - b - elem.margin) + v * term_len),
                pya.DPoint(u * (-a / 2 - b - elem.margin)),
            ]
        )
        elem.add_protection(poly2.transform(shift_start), face_index) 
[docs]
    @staticmethod
    def is_continuous(waveguide_cell, annotation_layer, tolerance):
        """Returns true if the given waveguide is determined to be continuous, false otherwise.
        The waveguide is considered continuous if the endpoints of its every segment (except first and last) are close
        enough to the endpoints of neighboring segments. The waveguide segments are not necessarily ordered correctly
        when iterating through the cells using begin_shapes_rec. This means we must compare the endpoints of each
        waveguide segment to the endpoints of all other waveguide segments.
        Args:
            waveguide_cell: Cell of the waveguide.
            annotation_layer: unsigned int representing the annotation layer
            tolerance: maximum allowed distance between connected waveguide segments
        """
        is_continuous = True
        # find the two endpoints for every waveguide segment
        endpoints = []  # endpoints of waveguide segment i are contained in endpoints[i][0] and endpoints[i][1]
        shapes_iter = waveguide_cell.begin_shapes_rec(annotation_layer)
        while not shapes_iter.at_end():
            shape = shapes_iter.shape()
            if shape.is_path():
                dtrans = shapes_iter.dtrans()  # transformation from shape coordinates to waveguide_cell coordinates
                pts = shape.each_dpoint()
                first_point = dtrans * next(pts, None)
                last_point = first_point.dup()
                for pt in pts:
                    last_point = pt
                last_point = dtrans * last_point
                endpoints.append([first_point, last_point])
            shapes_iter.next()
        # for every waveguide segment endpoint, try to find another endpoint which is close to it
        num_segments = len(endpoints)
        num_non_connected_points = 0
        for i in range(num_segments):
            def find_connected_point(point):
                """Tries to find a waveguide segment endpoint close enough to the given point."""
                found_connected_point = False
                for j in range(num_segments):
                    # pylint: disable=cell-var-from-loop
                    if i != j and (
                        point.distance(endpoints[j][1]) < tolerance or point.distance(endpoints[j][0]) < tolerance
                    ):
                        # print(f"{point} | {endpoints[j][1]} | {endpoints[j][0]}")
                        found_connected_point = True
                        break
                if not found_connected_point:
                    nonlocal num_non_connected_points
                    num_non_connected_points += 1
            if endpoints[i][0].distance(endpoints[i][1]) != 0:  # we ignore any zero-length segments
                find_connected_point(endpoints[i][0])
                find_connected_point(endpoints[i][1])
            # we can have up to 2 non-connected points, because ends of the waveguide don't have to be connected
            if num_non_connected_points > 2:
                is_continuous = False
                break
        return is_continuous