from typing import Optional

from shapely.geometry import LineString, GeometryCollection
from shapely import Point as P
from shapely import segmentize
import shapely

from loguru import logger


# Define each character with grid-based lines, normalized to fit in a 1x1 grid
CHARACTER_SEGMENTS = {
    "A": [(P(0, 0), P(0.25, 0.5)), (P(0.25, 0.5), P(0.5, 1)), (P(0.5, 1), P(0.75, 0.5)), (P(0.75, 0.5), P(1, 0)), (P(0.25, 0.5), P(0.75, 0.5))],
    "B": [(P(0, 0), P(0, 1)), (P(0, 1), P(0.75, 1)), (P(0.75, 1), P(0.75, 0.5)), (P(0.75, 0.5), P(0, 0.5)),
          (P(0, 0.5), P(0.75, 0.5)), (P(0.75, 0.5), P(0.75, 0)), (P(0.75, 0), P(0, 0))],
    "C": [(P(1, 1), P(0, 1)), (P(0, 1), P(0, 0)), (P(0, 0), P(1, 0))],
    "D": [(P(0, 0), P(0, 1)), (P(0, 1), P(0.75, 0.75)), (P(0.75, 0.75), P(0.75, 0.25)), (P(0.75, 0.25), P(0, 0))],
    "E": [(P(0, 0), P(0, 1)), (P(0, 1), P(1, 1)), (P(0, 0.5), P(0.75, 0.5)), (P(0, 0), P(1, 0))],
    "F": [(P(0, 0), P(0, 1)), (P(0, 1), P(1, 1)), (P(0, 0.5), P(0.75, 0.5))],
    "G": [(P(1, 1), P(0, 1)), (P(0, 1), P(0, 0)), (P(0, 0), P(1, 0)), (P(1, 0), P(1, 0.5)), (P(1, 0.5), P(0.5, 0.5))],
    "H": [(P(0, 0), P(0, 1)), (P(1, 0), P(1, 1)), (P(0, 0.5), P(1, 0.5))],
    "I": [(P(0.5, 1), P(0.5, 0))],
    "J": [(P(1, 1), P(1, 0)), (P(1, 0), P(0.25, 0)), (P(0.25, 0), P(0, 0.25))],
    "K": [(P(0, 0), P(0, 1)), (P(0, 0.5), P(1, 1)), (P(0, 0.5), P(1, 0))],
    "L": [(P(0, 1), P(0, 0)), (P(0, 0), P(1, 0))],
    "M": [(P(0, 0), P(0, 1)), (P(0, 1), P(0.5, 0.5)), (P(0.5, 0.5), P(1, 1)), (P(1, 1), P(1, 0))],
    "N": [(P(0, 0), P(0, 1)), (P(0, 1), P(1, 0)), (P(1, 0), P(1, 1))],
    "O": [(P(0, 0), P(0, 1)), (P(0, 1), P(1, 1)), (P(1, 1), P(1, 0)), (P(1, 0), P(0, 0))],
    "P": [(P(0, 0), P(0, 1)), (P(0, 1), P(1, 1)), (P(1, 1), P(1, 0.5)), (P(1, 0.5), P(0, 0.5))],
    "Q": [(P(0, 0), P(0, 1)), (P(0, 1), P(1, 1)), (P(1, 1), P(1, 0)), (P(1, 0), P(0, 0)), (P(0.5, 0.25), P(1, -0.25))],
    "R": [(P(0, 0), P(0, 1)), (P(0, 1), P(1, 1)), (P(1, 1), P(1, 0.5)), (P(1, 0.5), P(0, 0.5)), (P(0, 0.5), P(1, 0))],
    "S": [(P(1, 1), P(0, 1)), (P(0, 1), P(0, 0.5)), (P(0, 0.5), P(1, 0.5)), (P(1, 0.5), P(1, 0)), (P(1, 0), P(0, 0))],
    "T": [(P(0, 1), P(1, 1)), (P(0.5, 1), P(0.5, 0))],
    "U": [(P(0, 1), P(0, 0)), (P(0, 0), P(1, 0)), (P(1, 0), P(1, 1))],
    "V": [(P(0, 1), P(0.5, 0)), (P(0.5, 0), P(1, 1))],
    "W": [(P(0, 1), P(0.25, 0)), (P(0.25, 0), P(0.5, 0.5)), (P(0.5, 0.5), P(0.75, 0)), (P(0.75, 0), P(1, 1))],
    "X": [(P(0, 1), P(1, 0)), (P(0, 0), P(1, 1))],
    "Y": [(P(0, 1), P(0.5, 0.5)), (P(1, 1), P(0.5, 0.5)), (P(0.5, 0.5), P(0.5, 0))],
    "Z": [(P(0, 1), P(1, 1)), (P(1, 1), P(0, 0)), (P(0, 0), P(1, 0))],
}

def add(p1: P, p2: P):
    return P(p1.x + p2.x, p1.y + p2.y)

class Characters():
    def __init__(self,
                 scale: float = 1.0,
                 spacing: float = 1.5,
                 max_segment_length: float = 0.4,
                 mirror: bool = True,
                 characters: Optional[str | list[str]] = None):
        self.scale = scale
        self.spacing = spacing
        self.max_segment_length = max_segment_length
        self.current_offset = 0
        self.mirror = mirror
        self.mirror_point = P(0,0)
        self.transformed_alphabet = {}
        self.characters = characters
        if self.characters is not None:
            self.__prepare_characters()
        self.__transform_alphabet()

    # After specifying the scale, starting point etc., transform the dictionary character_segment to a class representative dictionary
    def __transform_alphabet(self):
        if self.characters is not None:
            for key in self.characters:
                self.transformed_alphabet[key] = self.__compute_character_geometry(CHARACTER_SEGMENTS[key])
        else:
            for key, character in CHARACTER_SEGMENTS.items():
                self.transformed_alphabet[key] = self.__compute_character_geometry(character)

    def __compute_character_geometry(self, character) -> list[tuple[P,P]]:
        transformed_lines = []
        for line in character:
            line = LineString(line)
            line = shapely.affinity.translate(line, yoff=-1) # FIXXME
            line = shapely.affinity.scale(line, xfact=self.scale, yfact=-self.scale, origin=(0,0)) ## FIXXME
            transformed_lines.append(line)
        return transformed_lines


    # Segmentize the Linestrings if they're longer than max_segment_length
    def __segmentize(self, tuple_segment: tuple[P, P]):
        return segmentize(tuple_segment, self.max_segment_length)

    # Transform an input character to a geometry
    def __character_to_geometry(self, character: str, index: int = 0):
        segments = self.transformed_alphabet.get(character.upper(), [])
        segments = [shapely.affinity.translate(self.__segmentize(segment_tuple), xoff=self.__character_offset(index)) for segment_tuple in segments]
        return segments

    def get_word(self, word: str) -> list[LineString]:
        word_geometry = []
        for index, character in enumerate(list(word)):
            word_geometry += self.__character_to_geometry(character, index)
        return word_geometry

    def get_positions(self, word: str, center: Optional[tuple[float,float]] = None) -> list[P]:
        segments = self.get_word(word)
        if center:
            _, _, width, height = GeometryCollection(segments).convex_hull.bounds
            assert width < center[0] and height < center[1], f"Cannot center word with bounds {width}x{height} on surface with dimensions: {center}"
            align_x = (center[0] - width) / 2
            align_y = (center[1] - height) / 2
            segments = [shapely.affinity.translate(segment, xoff=align_x, yoff=align_y) for segment in segments]

        return list(set([P(coordinates) for segment in segments for coordinates in segment.coords]))

    def __prepare_characters(self):
        if isinstance(self.characters, str):
            self.characters = list(self.characters.upper())
        elif isinstance(self.characters, list):
            self.characters = [c.upper() for c in self.characters]

    def __character_offset(self, index) -> float:
        return (self.scale + self.spacing) * index
