import numpy as np
from typing import Tuple
from rise import *


def create_cube(
    name: str,
    position: Tuple[float, float, float],
    size: float,
    orientation: Tuple[float, float, float, float] = (0, 0, 0, 1),
    material_name: str = "material_1",
    voxel_size: float = 0.01,
    is_fixed: bool = True,
) -> RS_StructureConfig:
    """
    create a cube object

    parameters:
        name: the name of the object
        position: the position of the object in the global coordinate system (x, y, z)
        size: the side length of the cube
        orientation: the direction of the object (x, y, z, w)
        material_name: the name of the material
        voxel_size: the size of the voxel
        is_fixed: whether the object is fixed

    return:
        the RS_StructureConfig object
    """
    # calculate the number of voxels
    voxel_count = int(np.ceil(size / voxel_size))

    # create the structure config
    structure_config = RS_StructureConfig()
    structure_config.name = name
    structure_config.is_fixed = is_fixed
    structure_config.voxel_size = voxel_size
    structure_config.origin_position = RVec3rf(*position)
    structure_config.orientation = RQuat3rf(*orientation)
    structure_config.material_references.append(material_name)

    # create the volume config
    body_config = RS_StructureBodyConfig()
    body_config.body_sid = 0
    body_config.relative_origin_position = RVec3rf(0, 0, 0)
    body_config.relative_orientation = RQuat3rf(0, 0, 0, 1)
    body_config.x_voxels = voxel_count
    body_config.y_voxels = voxel_count
    body_config.z_voxels = voxel_count

    # fill the voxel data
    total_voxels = voxel_count * voxel_count * voxel_count
    for _ in range(total_voxels):
        body_config.material_reference_sid.append(0)
        body_config.segment_bid.append(0)
        body_config.segment_type.append(0)  # 0 represents soft body

    # add the volume config to the structure
    structure_config.bodies.append(body_config)

    return structure_config


def create_cuboid(
    name: str,
    position: Tuple[float, float, float],
    size: Tuple[float, float, float],
    orientation: Tuple[float, float, float, float] = (0, 0, 0, 1),
    material_name: str = "material_0",
    voxel_size: float = 0.01,
    is_fixed: bool = True,
) -> RS_StructureConfig:
    """
    create a cuboid object

    parameters:
        name: the name of the object
        position: the position of the object in the global coordinate system (x, y, z)
        size: the size of the cuboid (x_size, y_size, z_size)
        orientation: the direction of the object (x, y, z, w)
        material_name: the name of the material
        voxel_size: the size of the voxel
        is_fixed: whether the object is fixed

    return:
        the RS_StructureConfig object
    """
    # calculate the number of voxels
    x_voxels = int(np.ceil(size[0] / voxel_size))
    y_voxels = int(np.ceil(size[1] / voxel_size))
    z_voxels = int(np.ceil(size[2] / voxel_size))

    # create the structure config
    structure_config = RS_StructureConfig()
    structure_config.name = name
    structure_config.is_fixed = is_fixed
    structure_config.voxel_size = voxel_size
    structure_config.origin_position = RVec3rf(*position)
    structure_config.orientation = RQuat3rf(*orientation)
    structure_config.material_references.append(material_name)

    # create the volume config
    body_config = RS_StructureBodyConfig()
    body_config.body_sid = 0
    body_config.relative_origin_position = RVec3rf(0, 0, 0)
    body_config.relative_orientation = RQuat3rf(0, 0, 0, 1)
    body_config.x_voxels = x_voxels
    body_config.y_voxels = y_voxels
    body_config.z_voxels = z_voxels

    # fill the voxel data
    total_voxels = x_voxels * y_voxels * z_voxels
    for _ in range(total_voxels):
        body_config.material_reference_sid.append(0)
        body_config.segment_bid.append(0)
        body_config.segment_type.append(0)  # 0 represents soft body

    # add the volume config to the structure
    structure_config.bodies.append(body_config)

    return structure_config


def create_sphere(
    name: str,
    position: Tuple[float, float, float],
    radius: float,
    orientation: Tuple[float, float, float, float] = (0, 0, 0, 1),
    material_name: str = "material_0",
    voxel_size: float = 0.01,
    is_fixed: bool = True,
) -> RS_StructureConfig:
    """
    create a sphere object

    parameters:
        name: the name of the object
        position: the position of the object in the global coordinate system (x, y, z)
        radius: the radius of the sphere
        orientation: the direction of the object (x, y, z, w)
        material_name: the name of the material
        voxel_size: the size of the voxel
        is_fixed: whether the object is fixed

    return:
        the RS_StructureConfig object
    """
    # calculate the number of voxels (the number of voxels of the diameter)
    diameter_voxels = int(np.ceil(2 * radius / voxel_size))

    # create the structure config
    structure_config = RS_StructureConfig()
    structure_config.name = name
    structure_config.is_fixed = is_fixed
    structure_config.voxel_size = voxel_size
    structure_config.origin_position = RVec3rf(*position)
    structure_config.orientation = RQuat3rf(*orientation)
    structure_config.material_references.append(material_name)

    # create the volume config
    body_config = RS_StructureBodyConfig()
    body_config.body_sid = 0
    body_config.relative_origin_position = RVec3rf(0, 0, 0)
    body_config.relative_orientation = RQuat3rf(0, 0, 0, 1)
    body_config.x_voxels = diameter_voxels
    body_config.y_voxels = diameter_voxels
    body_config.z_voxels = diameter_voxels

    # calculate the position of the sphere center in the voxel coordinate system
    center = diameter_voxels / 2.0
    radius_squared = (radius / voxel_size) ** 2

    # initialize the voxel data as empty
    total_voxels = diameter_voxels**3

    for _ in range(total_voxels):
        body_config.material_reference_sid.append(RS_NULL_INDEX)
        body_config.segment_bid.append(RS_NULL_INDEX)
        body_config.segment_type.append(RS_NULL_INDEX)

    # fill the sphere voxels
    for x in range(diameter_voxels):
        for y in range(diameter_voxels):
            for z in range(diameter_voxels):
                # calculate the distance squared between the voxel center and the sphere center
                dx = x - center
                dy = y - center
                dz = z - center
                dist_squared = dx * dx + dy * dy + dz * dz

                # if the voxel is inside the sphere, fill it
                if dist_squared <= radius_squared:
                    idx = (
                        x + y * diameter_voxels + z * diameter_voxels * diameter_voxels
                    )
                    body_config.material_reference_sid[idx] = 0
                    body_config.segment_bid[idx] = 0
                    body_config.segment_type[idx] = 0  # 0 represents soft body

    # add the volume config to the structure
    structure_config.bodies.append(body_config)

    return structure_config


def create_triangular_prism(
    name: str,
    position: Tuple[float, float, float],
    base_size: Tuple[float, float],
    height: float,
    orientation: Tuple[float, float, float, float] = (0, 0, 0, 1),
    material_name: str = "material_0",
    voxel_size: float = 0.01,
    is_fixed: bool = True,
) -> RS_StructureConfig:
    """
    create a triangular prism object

    parameters:
        name: the name of the object
        position: the position of the object in the global coordinate system (x, y, z)
        base_size: the size of the bottom rectangle (x_size, y_size)
        height: the height of the triangular prism
        orientation: the direction of the object (x, y, z, w)
        material_name: the name of the material
        voxel_size: the size of the voxel
        is_fixed: whether the object is fixed

    return:
        the RS_StructureConfig object
    """
    # calculate the number of voxels
    x_voxels = int(np.ceil(base_size[0] / voxel_size))
    y_voxels = int(np.ceil(base_size[1] / voxel_size))
    z_voxels = int(np.ceil(height / voxel_size))

    # create the structure config
    structure_config = RS_StructureConfig()
    structure_config.name = name
    structure_config.is_fixed = is_fixed
    structure_config.voxel_size = voxel_size
    structure_config.expansion_rate_signal_num = 0
    structure_config.rotation_angle_signal_num = 0
    structure_config.origin_position = RVec3rf(*position)
    structure_config.orientation = RQuat3rf(*orientation)
    structure_config.material_references.append(material_name)

    # create the volume config
    body_config = RS_StructureBodyConfig()
    body_config.body_sid = 0
    body_config.relative_origin_position = RVec3rf(0, 0, 0)
    body_config.relative_orientation = RQuat3rf(0, 0, 0, 1)
    body_config.x_voxels = x_voxels
    body_config.y_voxels = y_voxels
    body_config.z_voxels = z_voxels

    # initialize the voxel data as empty
    total_voxels = x_voxels * y_voxels * z_voxels

    for _ in range(total_voxels):
        body_config.material_reference_sid.append(RS_NULL_INDEX)
        body_config.segment_bid.append(RS_NULL_INDEX)
        body_config.segment_type.append(0)

    # fill the triangular prism voxels - the slope is from y=0 to y=y_voxels, the height is from z=0 to z=z_voxels
    for x in range(x_voxels):
        for y in range(y_voxels):
            # calculate the maximum height at the current y position (triangle)
            max_z = int(z_voxels * y / y_voxels)

            for z in range(max_z + 1):
                idx = x + y * x_voxels + z * x_voxels * y_voxels
                body_config.material_reference_sid[idx] = 0
                body_config.segment_bid[idx] = 0
                body_config.segment_type[idx] = 0  # 0 represents soft body

    # add the volume config to the structure
    structure_config.bodies.append(body_config)

    return structure_config
