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


MIN_X = -0.7
MAX_X = 0.7
MIN_Y = -0.7
MAX_Y = 0.7
MIN_Z = 1.0
MAX_Z = 1.5
MIN_DISTANCE = 0.3

SIMULATE_END = 250


NUM_LONG_HEX = int(os.getenv("NUM_LONG_HEX", random.randint(3, 7)))
NUM_SHORT_HEX = int(os.getenv("NUM_SHORT_HEX", random.randint(3, 7)))


def clear_temp_bolts_collection():
    if "Temp_Bolts" in bpy.data.collections:
        collection = bpy.data.collections["Temp_Bolts"]

        for obj in list(collection.objects):
            bpy.data.objects.remove(obj, do_unlink=True)
    else:
        temp_bolts = bpy.data.collections.new("Temp_Bolts")
        bpy.context.scene.collection.children.link(temp_bolts)
        print("Created new 'Temp_Bolts' 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:
        x = random.uniform(MIN_X, MAX_X)
        y = random.uniform(MIN_Y, MAX_Y)
        z = random.uniform(MIN_Z, MAX_Z)
        position = (x, y, z)

        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 (
        random.uniform(MIN_X, MAX_X),
        random.uniform(MIN_Y, MAX_Y),
        random.uniform(MIN_Z, MAX_Z),
    )


def create_random_bolts(bolt_type, num_bolts, bolt_positions):
    if bolt_type not in bpy.data.objects:
        print(f"Error: '{bolt_type}' object not found in the scene")
        return bolt_positions

    source_obj = bpy.data.objects[bolt_type]
    collection = bpy.data.collections["Temp_Bolts"]

    print(f"Creating {num_bolts} {bolt_type} bolts")

    for i in range(num_bolts):
        new_obj = deep_copy_object(source_obj, f"{bolt_type}_{i+1}")

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

        position = generate_valid_position(bolt_positions)
        bolt_positions.append(position)

        new_obj.location = Vector(position)

        new_obj.rotation_mode = "XYZ"
        new_obj.rotation_euler[0] = random.uniform(0, 2 * math.pi)
        new_obj.rotation_euler[1] = random.uniform(0, 2 * math.pi)
        new_obj.rotation_euler[2] = random.uniform(0, 2 * math.pi)

    return bolt_positions


def simulate_physics():
    bpy.context.scene.frame_end = SIMULATE_END

    for i in range(1, SIMULATE_END + 1):
        bpy.context.scene.frame_set(i)

    print(f"Physics simulation completed up to frame {SIMULATE_END}")


def create_new_task():
    clear_temp_bolts_collection()

    bolt_positions = []

    bolt_positions = create_random_bolts("Long_Hex", NUM_LONG_HEX, bolt_positions)
    bolt_positions = create_random_bolts("Short_Hex", NUM_SHORT_HEX, bolt_positions)

    simulate_physics()


create_new_task()
