import shapely
from shapely import LineString, Polygon, GeometryCollection, Point, MultiPoint

import numpy as np

from molecule_movement import Matching

from loguru import logger

def compute_corridor(matching: Matching, corridor_width: float, parking_distance: float = 0.0, parking_buffer: float = 0.0) -> Polygon:
    matching_line = LineString(matching.anchors)
    left_bound = shapely.offset_curve(matching_line, -corridor_width / 2)
    right_bound = shapely.offset_curve(matching_line, corridor_width / 2)
    left_coordinates = list(left_bound.coords)
    right_coordinates = list(reversed(right_bound.coords))

    if len(left_coordinates) == 0 and len(right_coordinates) == 0:
        return matching.goal.polygon.buffer(parking_buffer)

    if parking_distance > 0.0:
        if LineString([left_coordinates[-1], left_coordinates[-2]]).length < parking_distance:
            #logger.trace(f"Length of left bound for molecule {matching.molecule.name} is too short for parking distance {parking_distance}.")
            left_coordinates.pop(-1)
        else:
            left_coordinates[-1] = shapely.line_interpolate_point(left_bound, left_bound.length - parking_distance)
        if LineString([right_coordinates[0], right_coordinates[1]]).length < parking_distance:
            #logger.trace(f"Length of left bound for molecule {matching.molecule.name} is too short for parking distance {parking_distance}.")
            right_coordinates.pop(0)
        else:
            right_coordinates[0] = shapely.line_interpolate_point(right_bound, right_bound.length - parking_distance)

    if len(left_coordinates) <= 1 and len(right_coordinates) <= 1:
        max_size = max(matching.molecule.get_size()[0], matching.molecule.get_size()[1])
        x, y = matching.molecule.polygon_at_start.buffer((corridor_width - max_size)/2).exterior.coords.xy
        head = MultiPoint([left_coordinates[0], right_coordinates[0], *[Point(i,j) for i,j in zip(x,y)]]).convex_hull
        x, y = matching.goal.polygon.buffer(parking_buffer).exterior.coords.xy
        goal = MultiPoint([left_coordinates[0], right_coordinates[0], *[Point(i,j) for i,j in zip(x,y)]]).convex_hull
        return shapely.unary_union(GeometryCollection([head, goal]))

    max_size = max(matching.molecule.get_size()[0], matching.molecule.get_size()[1])
    x, y = matching.molecule.polygon_at_start.buffer((corridor_width - max_size)/2).exterior.coords.xy
    head = MultiPoint([left_coordinates[0], right_coordinates[-1], *[Point(i,j) for i,j in zip(x,y)]]).convex_hull
    x, y = matching.goal.polygon.buffer(parking_buffer).exterior.coords.xy
    goal = MultiPoint([left_coordinates[-1], right_coordinates[0], *[Point(i,j) for i,j in zip(x,y)]]).convex_hull

    corridor = Polygon([*left_coordinates, *right_coordinates])
    return shapely.unary_union(GeometryCollection([corridor, head, goal]))

