#!/usr/bin/env python

import numpy as np

# avoid having to write all this in the convex_hull algorithm
def crossp(o, a, b):
    return (a[0] - o[0]) * (b[1] - o[1]) - (a[1] - o[1]) * (b[0] - o[0])


# Graham scan algorithm to compute the convex hull
def convex_hull(points):
    # Sort points by y-coordinate (and x-coordinate to break ties)
    points = sorted(points, key=lambda p: (p[1], p[0]))

    # Build the lower hull
    lower_hull = []
    for p in points:
        # Pop if we make a non-left turn
        while len(lower_hull) >= 2 and crossp(lower_hull[-2], lower_hull[-1], p) <= 0:
            lower_hull.pop()
        lower_hull.append(p)

    # Build the upper hull
    upper_hull = []
    for p in reversed(points):
        while len(upper_hull) >= 2 and crossp(upper_hull[-2], upper_hull[-1], p) <= 0:
            upper_hull.pop()
        upper_hull.append(p)

    # Concatenate lower and upper hull, and remove the last point of each because it's repeated
    return np.array(lower_hull[:-1] + upper_hull[:-1])


def point_on_line_segment(p, a, b):
    p, a, b = np.array(p), np.array(a), np.array(b)

    # vector from a to b, and from a to p
    ab = b - a
    ap = p - a

    # check colinearity of the vectors
    cross = ab[0] * ap[1] - ab[1] * ap[0]
    if not np.isclose(cross, 0):
        return -np.inf

    # |ab|^2 and ap∙ab
    dot_ab_ab = np.dot(ab, ab)
    dot_ap_ab = np.dot(ap, ab)

    # if |ab|^2 = 0, then the segment is degenerate (i.e. a == b)
    if np.isclose(dot_ab_ab, 0):
        return 0 if np.allclose(a, p) else None

    # calculate projection factor, i.e. where on the line, starting at a, the
    # point lies
    return dot_ap_ab / dot_ab_ab


class Polygon:
    def __init__(self, vertices):
        self.vertices = vertices
        self.edges = list(zip(vertices, vertices[1:] + vertices[:1]))
        self.xlims = [np.inf, -np.inf]
        self.ylims = [np.inf, -np.inf]
        for v in vertices:
            self.xlims[0] = v[0] if v[0] < self.xlims[0] else self.xlims[0]
            self.xlims[1] = v[0] if v[0] > self.xlims[1] else self.xlims[1]
            self.ylims[0] = v[1] if v[1] < self.ylims[0] else self.ylims[0]
            self.ylims[1] = v[1] if v[1] > self.ylims[1] else self.ylims[1]

    def is_inside_single(self, x, y):
        # test if within bounding box
        if x < self.xlims[0] or self.xlims[1] < x:
            return False
        if y < self.ylims[0] or self.ylims[1] < y:
            return False

        # test if on any edge, and if yes, exit early that the point is
        # "inside". this also covers testing if the point is on a vertex or not
        p = [x, y]
        for e in self.edges:
            t = point_on_line_segment(p, e[0], e[1])
            if (t is not None) and (0 <= t <= 1):
                return True

        # Ray-casting algorithm to check if point is inside the polygon
        intersections = 0
        for v0, v1 in self.edges:
            # Ensure v1 is below v2
            if v0[1] > v1[1]:
                v0, v1 = v1, v0

            # Check if the ray intersects the edge (v1, v2)
            if y == v0[1] or y == v1[1]:
                # Move the point slightly to avoid ambiguity if it's on a vertex
                y += 1e-9

            if y > v0[1] and y <= v1[1] and (v1[1] != v0[1]):  # Ray passes between v1 and v2
                x_intersect = v0[0] + (y - v0[1]) * (v1[0] - v0[0]) / (v1[1] - v0[1])
                if x < x_intersect:  # Count intersection only if it is to the right of the point
                    intersections += 1

        # If the number of intersections is odd, the point is inside
        return intersections % 2 == 1


    def is_inside(self, x, y):
        if np.isscalar(x) and np.isscalar(y):
            return self.is_inside_single(x, y)

        # TODO: numpy vectorize
        x, y = np.asarray(x), np.asarray(y)
        result = np.zeros_like(x, dtype=bool)
        for i in range(len(x)):
            _x = x[i]
            _y = y[i]
            result[i] = self.is_inside_single(_x, _y)

        return result


def generate_star_polygon(center, radius_outer, radius_inner, num_spikes=7):
    cx, cy = center
    vertices = []
    angle_step = np.pi / num_spikes  # Step between outer and inner vertices

    for i in range(2 * num_spikes):
        angle = i * angle_step
        radius = radius_outer if i % 2 == 0 else radius_inner
        x = cx + radius * np.cos(angle)
        y = cy + radius * np.sin(angle)
        vertices.append((x, y))

    return Polygon(vertices)


class BoxArena:
    def __init__(self, xlims=[-1.,1.], ylims=[-1.,1.]):
        self.xlims = xlims
        self.ylims = ylims

    def is_inside(self, x, y):
        x = np.asarray(x)
        y = np.asarray(y)
        return (self.xlims[0] <= x) & (x <= self.xlims[1]) \
             & (self.ylims[0] <= y) & (y <= self.ylims[1])

    def get_bbox(self):
        return self.xlims, self.ylims


class CircleArena:
    def __init__(self, center=[0.,0.], radius=1.):
        self.center = center
        self.radius = radius
        self.sq_radius = radius**2
        self.xlims = [self.center[0]- self.radius, self.center[0] + self.radius]
        self.ylims = [self.center[1]- self.radius, self.center[1] + self.radius]

    def is_inside(self, x, y):
        sq_dist = (self.center[0] - x)**2 + (self.center[1] - y)**2
        return (sq_dist <= self.sq_radius)

    def get_bbox(self):
        return self.xlims, self.ylims


class TMaze:
    def __init__(self):
        self.polygon = Polygon([
            [1.0, 0.0],
            [2.0, 0.0],
            [2.0, 2.0],
            [3.0, 2.0],
            [3.0, 3.0],
            [0.0, 3.0],
            [0.0, 2.0],
            [1.0, 2.0]])

    def is_inside(self, x, y):
        return self.polygon.is_inside(x, y)

    def get_bbox(self):
        return self.polygon.xlims, self.polygon.ylims


class AMaze:
    def __init__(self):
        self.polygon_outer = Polygon([
            [0., 0.],
            [1., 0.],
            [1., 1.],
            [2., 1.],
            [2., 0.],
            [3., 0.],
            [3., 4.],
            [0., 4.],
        ])
        self.polygon_inner = Polygon([
            [1., 2.],
            [2., 2.],
            [2., 3.],
            [1., 3.],
        ])
        self.xlims = [0., 3.]
        self.ylims = [0., 4.]

    def is_inside(self, x, y):
        return ~self.polygon_inner.is_inside(x, y) & self.polygon_outer.is_inside(x, y)

    def get_bbox(self):
        return self.xlims, self.ylims



