"""
Create a quadruped robot with 9 separate bodies and 8 degrees of freedom.

The robot consists of:
- 1 torso body (body_sid=0)
- 4 upper leg bodies (body_sid=1,2,3,4) tilted at 45 degrees outward
- 4 lower leg bodies (body_sid=5,6,7,8)

Joints:
- 4 hip hinge joints connecting torso to upper legs
- 4 knee hinge joints connecting upper legs to lower legs
"""

import os
import pickle
from typing import Tuple

import numpy as np
from scipy.spatial.transform import Rotation

from rise import (
    RQuat3rf,
    RVec3rf,
    RS_NULL_INDEX,
    RSE_StructureConstraintType,
    RS_StructureBodyConfig,
    RS_StructureConstraintConfig,
    RS_StructureConfig,
)


def create_body_config(
    body_sid: int,
    x_voxels: int,
    y_voxels: int,
    z_voxels: int,
    relative_origin_position: Tuple[float, float, float],
    relative_orientation: Tuple[float, float, float, float],
    is_rigid: bool = True,
    soft_shell_thickness: int = 0,
    voxel_size: float = 0.01,
) -> Tuple[RS_StructureBodyConfig, dict]:
    """
    Create a body config with rigid core and optional soft voxel shell.

    Args:
        body_sid: Body structure ID
        x_voxels, y_voxels, z_voxels: Voxel grid dimensions for the rigid core
        relative_origin_position: Position of body origin relative to structure origin
                                  (this is the position of the rigid core's corner)
        relative_orientation: Orientation quaternion (x, y, z, w)
        segment_bid: Segment body ID for all voxels
        is_rigid: Whether the core voxels are rigid (True) or soft (False)
        soft_shell_thickness: Number of soft voxel layers to add around the rigid core
        voxel_size: Size of each voxel in meters (needed for origin adjustment)

    Returns:
        Tuple of (body_config, body_info) where body_info contains:
            - rigid_size: (x, y, z) dimensions of rigid core in voxels
            - total_size: (x, y, z) dimensions including soft shell in voxels
            - rigid_origin_offset: offset from body origin to rigid core origin
    """
    body_config = RS_StructureBodyConfig()
    body_config.body_sid = body_sid

    # Calculate total dimensions including soft shell
    if soft_shell_thickness > 0:
        total_x = x_voxels + 2 * soft_shell_thickness
        total_y = y_voxels + 2 * soft_shell_thickness
        total_z = z_voxels + 2 * soft_shell_thickness

        # Adjust origin position to account for soft shell
        # The soft shell extends in all directions in the BODY'S LOCAL coordinate system,
        # so we need to rotate the offset by the body's orientation before subtracting
        shell_offset = soft_shell_thickness * voxel_size
        shell_offset_local = np.array([shell_offset, shell_offset, shell_offset])
        # Rotate shell offset by body orientation to get offset in structure space
        r = Rotation.from_quat(relative_orientation)
        shell_offset_structure = r.apply(shell_offset_local)
        adjusted_origin = (
            relative_origin_position[0] - shell_offset_structure[0],
            relative_origin_position[1] - shell_offset_structure[1],
            relative_origin_position[2] - shell_offset_structure[2],
        )
    else:
        total_x, total_y, total_z = x_voxels, y_voxels, z_voxels
        adjusted_origin = relative_origin_position
        shell_offset = 0

    body_config.relative_origin_position = RVec3rf(*adjusted_origin)
    body_config.relative_orientation = RQuat3rf(*relative_orientation)
    body_config.x_voxels = total_x
    body_config.y_voxels = total_y
    body_config.z_voxels = total_z

    # Fill voxels: outer shell is soft, inner core is rigid
    # IMPORTANT: Each segment_bid can only have ONE segment_type (all soft or all rigid)
    soft_segment_bid = 0
    rigid_segment_bid = 1

    for iz in range(total_z):
        for iy in range(total_y):
            for ix in range(total_x):
                # Check if this voxel is in the outer shell (soft) or inner core (rigid)
                if soft_shell_thickness > 0:
                    # Check distance from all 6 faces
                    is_in_shell = (
                        ix < soft_shell_thickness
                        or ix >= total_x - soft_shell_thickness
                        or iy < soft_shell_thickness
                        or iy >= total_y - soft_shell_thickness
                        or iz < soft_shell_thickness
                        or iz >= total_z - soft_shell_thickness
                    )
                    if is_in_shell:
                        body_config.material_reference_sid.append(0)  # Soft material
                        body_config.segment_bid.append(soft_segment_bid)
                        body_config.segment_type.append(0)  # Soft
                    else:
                        body_config.material_reference_sid.append(1)  # Rigid material
                        body_config.segment_bid.append(rigid_segment_bid)
                        body_config.segment_type.append(1 if is_rigid else 0)
                else:
                    body_config.material_reference_sid.append(1)  # Rigid material
                    body_config.segment_bid.append(rigid_segment_bid)
                    body_config.segment_type.append(1 if is_rigid else 0)

    # Return body info for visualization
    body_info = {
        "rigid_size": (x_voxels, y_voxels, z_voxels),
        "total_size": (total_x, total_y, total_z),
        "soft_shell_thickness": soft_shell_thickness,
        "rigid_origin_offset": shell_offset,
    }

    return body_config, body_info


def quaternion_from_axis_angle(
    axis: np.ndarray, angle_rad: float
) -> Tuple[float, float, float, float]:
    """
    Create a quaternion (x, y, z, w) from axis-angle representation.

    Args:
        axis: Unit vector representing rotation axis
        angle_rad: Rotation angle in radians

    Returns:
        Quaternion as (x, y, z, w)
    """
    axis = axis / np.linalg.norm(axis)
    half_angle = angle_rad / 2.0
    sin_half = np.sin(half_angle)
    cos_half = np.cos(half_angle)
    return (
        float(sin_half * axis[0]),
        float(sin_half * axis[1]),
        float(sin_half * axis[2]),
        float(cos_half),
    )


def rotate_point_by_quaternion(
    point: np.ndarray, quat: Tuple[float, float, float, float]
) -> np.ndarray:
    """
    Rotate a point by a quaternion.

    Args:
        point: 3D point to rotate
        quat: Quaternion as (x, y, z, w)

    Returns:
        Rotated point
    """
    r = Rotation.from_quat(quat)
    return r.apply(point)


def create_quadruped_8dof(
    structure_name: str,
    position: Tuple[float, float, float],
    soft_material_name: str,
    rigid_material_name: str,
    orientation: Tuple[float, float, float, float] = (0.0, 0.0, 0.0, 1.0),
    voxel_size: float = 0.01,
    is_fixed: bool = False,
    leg_angle_deg: float = 45.0,
    soft_shell_thickness: int = 1,
) -> Tuple[RS_StructureConfig, dict]:
    """
    Build a quadruped structure with 9 separate bodies:
    - Body 0: torso
    - Bodies 1-4: upper legs (tilted outward at specified angle)
    - Bodies 5-8: lower legs

    Each body has its own voxel grid. Joints connect different bodies.
    Each body has a rigid core surrounded by a soft voxel shell.

    Note: Torso and upper legs always have 1 layer of soft voxels (fixed),
    while lower legs use the user-defined soft_shell_thickness parameter.

    Args:
        structure_name: Name of the structure
        position: Origin position in world coordinates
        soft_material_name: Name of soft material to use
        rigid_material_name: Name of rigid material to use
        orientation: Structure orientation quaternion (x, y, z, w)
        voxel_size: Size of each voxel in meters
        is_fixed: Whether the structure is fixed in space
        leg_angle_deg: Angle of upper legs from vertical (in degrees)
        soft_shell_thickness: Number of soft voxel layers around rigid core for lower legs only (default 1)

    Returns:
        Tuple of (structure_config, structure_dict)
    """
    leg_angle_rad = np.deg2rad(leg_angle_deg)

    # Dimensions in voxels
    torso_x, torso_y, torso_z = 30, 11, 7
    leg_x, leg_y, leg_z = 2, 2, 12

    # Calculate dimensions in meters
    torso_size = np.array([torso_x, torso_y, torso_z]) * voxel_size
    leg_size = np.array([leg_x, leg_y, leg_z]) * voxel_size

    torso_upper_leg_shell_thickness = 1
    torso_upper_leg_shell_offset = torso_upper_leg_shell_thickness * voxel_size
    lower_leg_shell_offset = soft_shell_thickness * voxel_size

    # Create structure config
    structure_config = RS_StructureConfig()
    structure_config.name = structure_name
    structure_config.is_fixed = is_fixed
    structure_config.voxel_size = voxel_size
    structure_config.origin_position = RVec3rf(*position)
    structure_config.orientation = RQuat3rf(*orientation)
    # Material ID 0: Soft material
    structure_config.material_references.append(soft_material_name)
    # Material ID 1: Rigid material
    structure_config.material_references.append(rigid_material_name)

    # =========================================================================
    # Body 0: Torso
    # =========================================================================
    # Position torso so its center is at (0, 0, torso_height_center)
    # The torso origin (voxel 0,0,0) is at the corner
    torso_height_center = (
        torso_size[2] / 2.0 + leg_size[2] * np.cos(leg_angle_rad) * 2.0
    )
    torso_origin = (
        -torso_size[0] / 2.0,
        -torso_size[1] / 2.0,
        torso_height_center - torso_size[2] / 2.0,
    )

    # Store all bodies temporarily, will sort by body_sid before appending
    all_bodies = []  # List of (body_sid, body_config, body_info)

    torso_body, torso_info = create_body_config(
        body_sid=0,
        x_voxels=torso_x,
        y_voxels=torso_y,
        z_voxels=torso_z,
        relative_origin_position=torso_origin,
        relative_orientation=(0.0, 0.0, 0.0, 1.0),
        soft_shell_thickness=torso_upper_leg_shell_thickness,
        voxel_size=voxel_size,
    )
    all_bodies.append((0, torso_body, torso_info))

    # =========================================================================
    # Define leg attachment points on torso
    # =========================================================================
    # Legs attach at the corners of the torso, on the Y-faces (front and back)
    # The hip joint is at the mid-height of the torso

    # Corner positions in torso local coordinates (relative to torso origin)
    # Format: (x_offset, y_offset, is_front)
    leg_corners = [
        (leg_x / 2 * voxel_size, 0.0, True),  # Front-left (body 1)
        (torso_size[0] - leg_x / 2 * voxel_size, 0.0, True),  # Front-right (body 2)
        (leg_x / 2 * voxel_size, torso_size[1], False),  # Back-left (body 3)
        (
            torso_size[0] - leg_x / 2 * voxel_size,
            torso_size[1],
            False,
        ),  # Back-right (body 4)
    ]

    # Hip joint height in torso local coords (mid-height of torso)
    hip_z_in_torso = torso_size[2] / 2.0

    # Store body configs and joint info for later
    upper_leg_bodies = []
    lower_leg_bodies = []
    hip_anchors_torso = []  # Hip anchor in torso local coords
    hip_anchors_leg = []  # Hip anchor in upper leg local coords
    knee_anchors_upper = []  # Knee anchor in upper leg local coords
    knee_anchors_lower = []  # Knee anchor in lower leg local coords
    upper_leg_orientations = []
    lower_leg_origins = []

    # Lateral offsets to prevent self-collision (in voxels)
    # Upper legs shift 2 voxels outward, lower legs shift 4 voxels outward
    upper_leg_lateral_offset_voxels = 3
    lower_leg_lateral_offset_voxels = 7  # Total offset from body center


    for leg_idx, (corner_x, corner_y, is_front) in enumerate(leg_corners):
        # =====================================================================
        # Calculate upper leg orientation
        # =====================================================================
        # All upper legs tilt in the same direction along the long side of torso
        # Rotate around Y-axis by positive angle (all legs tilt toward -X direction)

        tilt_axis = np.array([0.0, 1.0, 0.0])
        tilt_angle = leg_angle_rad

        upper_leg_quat = quaternion_from_axis_angle(tilt_axis, tilt_angle)
        upper_leg_orientations.append(upper_leg_quat)

        # =====================================================================
        # Calculate upper leg position
        # =====================================================================
        # Determine lateral offset direction based on front/back position
        # Front legs (indices 0, 1) shift in -Y direction (forward)
        # Back legs (indices 2, 3) shift in +Y direction (backward)
        lateral_direction = -1.0 if is_front else 1.0

        # Upper leg lateral offset in meters
        upper_leg_lateral_offset = (
            lateral_direction * upper_leg_lateral_offset_voxels * voxel_size
        )

        # Hip joint position in structure space (before considering torso origin)
        hip_joint_structure = np.array(
            [
                torso_origin[0] + corner_x,
                torso_origin[1] + corner_y + upper_leg_lateral_offset,
                torso_origin[2] + hip_z_in_torso,
            ]
        )

        # In the upper leg's local coordinate system (rotated), the leg extends
        # in -Z direction. The hip joint is at the top of the leg.
        # The leg origin (voxel 0,0,0) is at one corner of the leg.
        # We want the hip joint to be at the center-top of the leg.

        # Leg center offset in leg's local coords (before rotation)
        leg_center_top_local = np.array(
            [
                leg_size[0] / 2.0,  # Center in X
                leg_size[1] / 2.0,  # Center in Y
                leg_size[2],  # Top in Z
            ]
        )

        # Rotate this offset by the leg's orientation
        leg_center_top_rotated = rotate_point_by_quaternion(
            leg_center_top_local, upper_leg_quat
        )

        # Upper leg origin in structure space
        upper_leg_origin = hip_joint_structure - leg_center_top_rotated

        # =====================================================================
        # Create upper leg body config
        # =====================================================================
        upper_leg_sid = leg_idx + 1  # Bodies 1, 2, 3, 4
        upper_leg_body, upper_leg_info = create_body_config(
            body_sid=upper_leg_sid,
            x_voxels=leg_x,
            y_voxels=leg_y,
            z_voxels=leg_z,
            relative_origin_position=tuple(upper_leg_origin),
            relative_orientation=upper_leg_quat,
            soft_shell_thickness=torso_upper_leg_shell_thickness,
            voxel_size=voxel_size,
        )
        upper_leg_bodies.append(upper_leg_body)
        all_bodies.append((upper_leg_sid, upper_leg_body, upper_leg_info))

        # =====================================================================
        # Hip joint anchors
        # =====================================================================
        # In torso local coords (includes lateral offset along Y)
        # Add torso_upper_leg_shell_offset to account for soft shell around rigid core
        hip_anchor_torso = (
            corner_x + torso_upper_leg_shell_offset,
            corner_y + upper_leg_lateral_offset + torso_upper_leg_shell_offset,
            hip_z_in_torso + torso_upper_leg_shell_offset,
        )
        hip_anchors_torso.append(hip_anchor_torso)

        # In upper leg local coords (center-top of leg)
        # Add torso_upper_leg_shell_offset to account for soft shell around rigid core
        hip_anchor_leg = leg_center_top_local + np.array(
            [torso_upper_leg_shell_offset, torso_upper_leg_shell_offset, torso_upper_leg_shell_offset]
        )
        hip_anchors_leg.append(tuple(hip_anchor_leg))

        # =====================================================================
        # Calculate lower leg position
        # =====================================================================
        # Knee joint is at the bottom of the upper leg
        knee_joint_local_upper = np.array(
            [
                leg_size[0] / 2.0,  # Center in X
                leg_size[1] / 2.0,  # Center in Y
                0.0,  # Bottom in Z
            ]
        )

        # Knee joint in structure space
        knee_joint_structure = upper_leg_origin + rotate_point_by_quaternion(
            knee_joint_local_upper, upper_leg_quat
        )

        # Lower leg: -90 degrees tilt along the same axis as upper legs
        lower_leg_tilt_angle = np.deg2rad(-45)
        lower_leg_quat = quaternion_from_axis_angle(
            np.array([0.0, 1.0, 0.0]), lower_leg_tilt_angle
        )

        # Lower leg center-top position (in lower leg's local rotated frame)
        lower_leg_center_top_local = np.array(
            [
                leg_size[0] / 2.0,
                leg_size[1] / 2.0,
                leg_size[2],
            ]
        )

        # Rotate the offset by the lower leg's orientation
        lower_leg_center_top_rotated = rotate_point_by_quaternion(
            lower_leg_center_top_local, lower_leg_quat
        )

        # Additional lateral offset for lower leg (beyond upper leg offset)
        # Lower leg total offset is 4 voxels, upper leg is 2, so additional is 2
        lower_leg_additional_offset = (
            lateral_direction
            * (lower_leg_lateral_offset_voxels - upper_leg_lateral_offset_voxels)
            * voxel_size
        )

        # Lower leg origin in structure space (with additional lateral offset)
        lower_leg_origin = knee_joint_structure - lower_leg_center_top_rotated
        lower_leg_origin[1] += lower_leg_additional_offset  # Apply additional Y offset
        lower_leg_origins.append(lower_leg_origin)

        # =====================================================================
        # Create lower leg body config
        # =====================================================================
        lower_leg_sid = leg_idx + 5  # Bodies 5, 6, 7, 8
        lower_leg_body, lower_leg_info = create_body_config(
            body_sid=lower_leg_sid,
            x_voxels=leg_x,
            y_voxels=leg_y,
            z_voxels=leg_z,
            relative_origin_position=tuple(lower_leg_origin),
            relative_orientation=lower_leg_quat,
            soft_shell_thickness=soft_shell_thickness,
            voxel_size=voxel_size,
        )
        lower_leg_bodies.append(lower_leg_body)
        all_bodies.append((lower_leg_sid, lower_leg_body, lower_leg_info))

        # =====================================================================
        # Knee joint anchors
        # =====================================================================
        # In upper leg local coords (center-bottom)
        # Add torso_upper_leg_shell_offset to account for soft shell around rigid core
        knee_anchor_upper = knee_joint_local_upper + np.array(
            [torso_upper_leg_shell_offset, torso_upper_leg_shell_offset, torso_upper_leg_shell_offset]
        )
        knee_anchors_upper.append(tuple(knee_anchor_upper))

        # In lower leg local coords (center-top)
        # Need to adjust for the additional lateral offset of the lower leg
        # The lower leg body is shifted outward, so the anchor needs to be shifted inward
        # to keep the joint at the original knee position
        # Convert the world-space offset to lower leg local coordinates
        lower_leg_offset_world = np.array([0.0, lower_leg_additional_offset, 0.0])
        # Inverse rotation to get local coordinates (negate the rotation angle)
        inverse_lower_leg_quat = quaternion_from_axis_angle(
            np.array([0.0, 1.0, 0.0]), -lower_leg_tilt_angle
        )
        lower_leg_offset_local = rotate_point_by_quaternion(
            lower_leg_offset_world, inverse_lower_leg_quat
        )

        # Adjust the knee anchor by subtracting the offset (shift inward in local coords)
        # Add lower_leg_shell_offset to account for soft shell around rigid core (lower legs use user-defined thickness)
        knee_anchor_lower = (
            lower_leg_center_top_local
            - lower_leg_offset_local
            + np.array([lower_leg_shell_offset, lower_leg_shell_offset, lower_leg_shell_offset])
        )
        knee_anchors_lower.append(tuple(knee_anchor_lower))

    # =========================================================================
    # Sort bodies by body_sid and append to structure_config
    # =========================================================================
    # This ensures body_sid matches the list index in structure_config.bodies
    all_bodies.sort(key=lambda x: x[0])  # Sort by body_sid
    body_infos = []
    for body_sid, body_config, body_info in all_bodies:
        structure_config.bodies.append(body_config)
        body_infos.append(body_info)

    # =========================================================================
    # Create hip hinge joints (torso to upper legs)
    # =========================================================================
    for leg_idx in range(4):
        constraint = RS_StructureConstraintConfig()
        constraint.type = RSE_StructureConstraintType.RSE_HINGE_JOINT

        # Torso side
        constraint.a_body_sid = 0
        constraint.a_segment_bid = 1
        constraint.a_local_anchor = RVec3rf(*hip_anchors_torso[leg_idx])

        # Upper leg side (body_sid = 1, 2, 3, 4)
        constraint.b_body_sid = leg_idx + 1
        constraint.b_segment_bid = 1
        constraint.b_local_anchor = RVec3rf(*hip_anchors_leg[leg_idx])

        # Hinge axis (Y-axis in local coords for lateral swing)
        # Front legs rotate one way, back legs rotate the other
        if leg_idx < 2:  # Front legs
            constraint.hinge_a_local_axis = RVec3rf(0.0, 1.0, 0.0)
            constraint.hinge_b_local_axis = RVec3rf(0.0, -1.0, 0.0)
        else:  # Back legs
            constraint.hinge_a_local_axis = RVec3rf(0.0, -1.0, 0.0)
            constraint.hinge_b_local_axis = RVec3rf(0.0, 1.0, 0.0)

        constraint.hinge_rotation_angle_signal_sid = leg_idx
        constraint.hinge_min = -0.7
        constraint.hinge_max = 0.7

        constraint.hinge_max_torque = 6.0

        structure_config.constraints.append(constraint)

    # =========================================================================
    # Create knee hinge joints (upper legs to lower legs)
    # =========================================================================
    for leg_idx in range(4):
        constraint = RS_StructureConstraintConfig()
        constraint.type = RSE_StructureConstraintType.RSE_HINGE_JOINT

        # Upper leg side (body_sid = 1, 2, 3, 4)
        constraint.a_body_sid = leg_idx + 1
        constraint.a_segment_bid = 1
        constraint.a_local_anchor = RVec3rf(*knee_anchors_upper[leg_idx])

        # Lower leg side (body_sid = 5, 6, 7, 8)
        constraint.b_body_sid = leg_idx + 5
        constraint.b_segment_bid = 1
        constraint.b_local_anchor = RVec3rf(*knee_anchors_lower[leg_idx])

        # Hinge axis (Y-axis for knee bending)
        if leg_idx < 2:  # Front legs
            constraint.hinge_a_local_axis = RVec3rf(0.0, 1.0, 0.0)
            constraint.hinge_b_local_axis = RVec3rf(0.0, -1.0, 0.0)
        else:  # Back legs
            constraint.hinge_a_local_axis = RVec3rf(0.0, -1.0, 0.0)
            constraint.hinge_b_local_axis = RVec3rf(0.0, 1.0, 0.0)

        constraint.hinge_rotation_angle_signal_sid = 4 + leg_idx

        # Set hinge limits based on leg position
        # Front legs (bodies 5, 6): increase max range
        # Back legs (bodies 7, 8): increase min range
        if leg_idx < 2:  # Front legs (bodies 5, 6)
            constraint.hinge_min = -1.5
            constraint.hinge_max = 1.5
        else:  # Back legs (bodies 7, 8)
            constraint.hinge_min = -1.5
            constraint.hinge_max = 1.5

        constraint.hinge_max_torque = 6.0

        structure_config.constraints.append(constraint)

    structure_config.rotation_angle_signal_num = 8

    # =========================================================================
    # Build auxiliary structure dict (for mesh building / visualization)
    # =========================================================================
    # Create voxel arrays for visualization/mesh building
    # We'll create combined arrays that represent the full robot

    # Estimate bounding box for combined voxel grid
    max_extent = max(torso_x, torso_y, torso_z + leg_z * 2) + 10
    combined_size = 64  # Use same size as original for compatibility

    is_not_empty = np.zeros((combined_size, combined_size, combined_size), dtype=bool)
    is_rigid = np.zeros((combined_size, combined_size, combined_size), dtype=bool)
    segment_id = np.zeros((combined_size, combined_size, combined_size), dtype=int)

    # Helper to convert structure coords to voxel indices in combined grid
    def structure_to_voxel_idx(
        pos: np.ndarray, offset: np.ndarray
    ) -> Tuple[int, int, int]:
        """Convert structure position to voxel indices in combined grid."""
        voxel_pos = (pos / voxel_size + offset).astype(int)
        return tuple(np.clip(voxel_pos, 0, combined_size - 1))

    # Offset to center the combined grid
    grid_offset = np.array([combined_size // 2, combined_size // 2, combined_size // 4])

    # Fill torso voxels
    for ix in range(torso_x):
        for iy in range(torso_y):
            for iz in range(torso_z):
                local_pos = np.array([ix, iy, iz]) * voxel_size
                world_pos = np.array(torso_origin) + local_pos
                vx, vy, vz = structure_to_voxel_idx(world_pos, grid_offset)
                if (
                    0 <= vx < combined_size
                    and 0 <= vy < combined_size
                    and 0 <= vz < combined_size
                ):
                    is_rigid[vx, vy, vz] = True
                    is_not_empty[vx, vy, vz] = True
                    segment_id[vx, vy, vz] = 1

    # Build connections info (same format as original)
    connections = []

    # Hip connections
    for leg_idx in range(4):
        hip_pos_structure = np.array(torso_origin) + np.array(
            hip_anchors_torso[leg_idx]
        )
        hip_voxel = structure_to_voxel_idx(hip_pos_structure, grid_offset)

        connections.append(
            {
                "components": (0, leg_idx + 1),  # Torso body to upper leg body
                "position": np.array(hip_voxel, dtype=float),
                "axis": np.array(
                    [0.0, (1.0 if leg_idx < 2 else -1.0), 0.0], dtype=float
                ),
                "size": float(leg_x * leg_y),
            }
        )

    # Knee connections
    for leg_idx in range(4):
        upper_leg_origin = structure_config.bodies[leg_idx + 1].relative_origin_position
        upper_leg_quat = upper_leg_orientations[leg_idx]
        knee_local = np.array(knee_anchors_upper[leg_idx])
        knee_rotated = rotate_point_by_quaternion(knee_local, upper_leg_quat)
        knee_pos_structure = (
            np.array([upper_leg_origin.x, upper_leg_origin.y, upper_leg_origin.z])
            + knee_rotated
        )
        knee_voxel = structure_to_voxel_idx(knee_pos_structure, grid_offset)

        connections.append(
            {
                "components": (leg_idx + 1, leg_idx + 5),  # Upper leg to lower leg
                "position": np.array(knee_voxel, dtype=float),
                "axis": np.array(
                    [0.0, (1.0 if leg_idx < 2 else -1.0), 0.0], dtype=float
                ),
                "size": float(leg_x * leg_y),
            }
        )

    structure = {
        "is_not_empty": is_not_empty,
        "is_rigid": is_rigid,
        "segment_id": segment_id,
        "connections": connections,
        "body_infos": body_infos,
        "soft_shell_thickness": soft_shell_thickness,  # Lower legs thickness (torso & upper legs always use 1)
        "voxel_size": voxel_size,
    }

    return structure_config, structure


if __name__ == "__main__":
    # Test the function
    soft_shell_thickness = 2
    structure_config, structure = create_quadruped_8dof(
        "quadruped_8dof",
        (0.0, 0.0, 0.0),
        "material_0",
        "material_1",
        leg_angle_deg=45.0,
        soft_shell_thickness=soft_shell_thickness,
    )

    print("=" * 60)
    print("Quadruped 8-DOF Robot Generator")
    print("=" * 60)
    print(f"Structure name: {structure_config.name}")
    print(f"Number of bodies: {len(structure_config.bodies)}")
    print(f"Number of constraints: {len(structure_config.constraints)}")
    print(f"Rotation angle signals: {structure_config.rotation_angle_signal_num}")
    print(f"Soft shell thickness:")
    print(f"  - Torso & upper legs: 1 voxel (fixed)")
    print(f"  - Lower legs: {soft_shell_thickness} voxels (user-defined)")

    print("\nBodies:")
    body_infos = structure.get("body_infos", [])
    for i, body in enumerate(structure_config.bodies):
        pos = body.relative_origin_position
        ori = body.relative_orientation
        print(
            f"  Body {body.body_sid}: "
            f"{body.x_voxels}x{body.y_voxels}x{body.z_voxels} voxels (total)"
        )
        if i < len(body_infos):
            info = body_infos[i]
            print(
                f"    Rigid core: {info['rigid_size'][0]}x{info['rigid_size'][1]}x{info['rigid_size'][2]} voxels"
            )
        print(f"    Position: ({pos.x:.4f}, {pos.y:.4f}, {pos.z:.4f})")
        print(f"    Orientation: ({ori.x:.4f}, {ori.y:.4f}, {ori.z:.4f}, {ori.w:.4f})")

    print("\nConstraints:")
    for i, constraint in enumerate(structure_config.constraints):
        joint_type = "Hip" if i < 4 else "Knee"
        print(
            f"  {joint_type} Joint {i}: "
            f"body {constraint.a_body_sid} <-> body {constraint.b_body_sid}"
        )
        print(f"    Signal SID: {constraint.hinge_rotation_angle_signal_sid}")

    print("\nTo visualize the robot, run:")
    print("  python visualize_quadruped_8dof.py")

    # Save robot config to data/robot_config/quadruped_8dof.data
    output_path = os.path.join(
        os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__)))),
        "data",
        "robot_config",
        "quadruped_8dof.data",
    )
    os.makedirs(os.path.dirname(output_path), exist_ok=True)
    with open(output_path, "wb") as f:
        pickle.dump(structure_config, f)
    print(f"\nRobot config saved to: {output_path}")
