import os
import bpy
import random
import math
from mathutils import Vector


R_MIN = 2.2
R_MAX = 4.2
ANGLE_MIN = 200
ANGLE_MAX = 340
MIN_DISTANCE = 1.5


def clear_temp_frogs_collection():
    if "Temp_Frogs" in bpy.data.collections:
        collection = bpy.data.collections["Temp_Frogs"]

        for obj in list(collection.objects):
            bpy.data.objects.remove(obj, do_unlink=True)
    else:
        temp_frogs = bpy.data.collections.new("Temp_Frogs")
        bpy.context.scene.collection.children.link(temp_frogs)
        print("Created new 'Temp_Frogs' collection")


def deep_copy_object(obj, name):
    new_obj = obj.copy()
    new_obj.name = name
    new_obj.hide_render = False

    if obj.data:
        new_obj.data = obj.data.copy()

    for child in obj.children:
        new_child = deep_copy_object(child, f"{name}_{child.name}")
        new_child.parent = new_obj

    return new_obj


def calculate_distance(pos1, pos2):
    return math.sqrt(
        (pos1[0] - pos2[0]) ** 2 + (pos1[1] - pos2[1]) ** 2 + (pos1[2] - pos2[2]) ** 2
    )


def generate_valid_position(existing_positions):
    max_attempts = 100
    attempt = 0

    while attempt < max_attempts:
        radius = random.uniform(R_MIN, R_MAX)
        angle_rad = math.radians(random.uniform(ANGLE_MIN, ANGLE_MAX))

        x = radius * math.cos(angle_rad)
        y = radius * math.sin(angle_rad)
        position = (x, y, 0)

        valid = True
        for existing_pos in existing_positions:
            if calculate_distance(position, existing_pos) < MIN_DISTANCE:
                valid = False
                break

        if valid:
            return position

        attempt += 1

    print(
        f"Warning: Could not find position with MIN_DISTANCE={MIN_DISTANCE} after {max_attempts} attempts"
    )
    return position


def create_random_frogs(
    num_frogs,
    looking_at_sword,
):
    if "Frog_2" not in bpy.data.objects:
        print("Error: 'Frog_2' object not found in the scene")
        return

    source_obj = bpy.data.objects["Frog_2"]
    collection = bpy.data.collections["Temp_Frogs"]

    print(f"Creating {num_frogs} frogs")

    frog_positions = []

    for i in range(num_frogs):
        new_obj = deep_copy_object(source_obj, f"Frog_2_{i+1}")

        collection.objects.link(new_obj)
        for child in new_obj.children:
            collection.objects.link(child)

        position = generate_valid_position(frog_positions)
        frog_positions.append(position)

        new_obj.location = Vector(position)

        should_look_at_origin = i < looking_at_sword

        new_obj.rotation_mode = "XYZ"
        if should_look_at_origin:
            angle = math.atan2(-position[1], -position[0])
            new_obj.rotation_euler[2] = angle + math.pi / 2
        else:
            angle = math.atan2(position[1], position[0])
            new_obj.rotation_euler[2] = angle + math.pi / 2


def create_new_task():
    num_frogs = int(os.getenv("NUM_FROGS"))
    looking_at_sword = int(os.getenv("LOOKING_AT_SWORD")) - 1

    clear_temp_frogs_collection()
    create_random_frogs(
        num_frogs,
        looking_at_sword,
    )


create_new_task()
