"""
Create a bipedal robot with 7 separate bodies and 6 degrees of freedom.

The robot consists of:
- 1 torso body (body_sid=0)
- 2 upper leg bodies (body_sid=1,2) - left and right
- 2 lower leg bodies (body_sid=3,4) - left and right
- 2 foot bodies (body_sid=5,6) - left and right

Joints:
- 2 hip hinge joints connecting torso to upper legs
- 2 knee hinge joints connecting upper legs to lower legs
- 2 ankle hinge joints connecting lower legs to feet
"""

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
        relative_orientation: Orientation quaternion (x, y, z, w)
        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

    Returns:
        Tuple of (body_config, body_info)
    """
    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
    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(*relative_origin_position)
    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
    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):
                body_config.material_reference_sid.append(0)

                # Check if this voxel is in the outer shell (soft) or inner core (rigid)
                if soft_shell_thickness > 0:
                    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.segment_bid.append(soft_segment_bid)
                        body_config.segment_type.append(0)  # Soft
                    else:
                        body_config.segment_bid.append(rigid_segment_bid)
                        body_config.segment_type.append(1 if is_rigid else 0)
                else:
                    body_config.segment_bid.append(rigid_segment_bid)
                    body_config.segment_type.append(1 if is_rigid else 0)

    body_info = {
        "rigid_size": (x_voxels, y_voxels, z_voxels),
        "total_size": (total_x, total_y, total_z),
        "soft_shell_thickness": soft_shell_thickness,
    }

    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.
    """
    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.
    """
    r = Rotation.from_quat(quat)
    return r.apply(point)


def create_biped_6dof(
    structure_name: str,
    position: Tuple[float, float, float],
    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,
    soft_shell_thickness: int = 1,
    upper_leg_y_offset_voxels: int = 5,
    lower_leg_z_offset_voxels: int = 5,
    foot_z_offset_voxels: int = 5,
) -> Tuple[RS_StructureConfig, dict]:
    """
    Build a bipedal structure with 7 separate bodies:
    - Body 0: torso
    - Bodies 1-2: upper legs (left, right)
    - Bodies 3-4: lower legs (left, right)
    - Bodies 5-6: feet (left, right)

    Args:
        structure_name: Name of the structure
        position: Origin position in world coordinates
        material_name: Name of 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
        soft_shell_thickness: Number of soft voxel layers around rigid core
        upper_leg_y_offset_voxels: Additional Y offset for upper legs (outward, in voxels)
        lower_leg_z_offset_voxels: Additional Z offset for lower legs (downward, in voxels)
        foot_z_offset_voxels: Additional Z offset for feet (downward, in voxels)

    Returns:
        Tuple of (structure_config, structure_dict)
    """
    # Dimensions in voxels
    torso_x, torso_y, torso_z = 5, 8, 15
    upper_leg_x, upper_leg_y, upper_leg_z = 2, 2, 10
    lower_leg_x, lower_leg_y, lower_leg_z = 2, 2, 10
    foot_x, foot_y, foot_z = 5, 8, 1

    # Calculate dimensions in meters
    torso_size = (
        np.array([torso_x, torso_y, torso_z]) * voxel_size
        + 2 * soft_shell_thickness * voxel_size
    )
    upper_leg_size = (
        np.array([upper_leg_x, upper_leg_y, upper_leg_z]) * voxel_size
        + 2 * soft_shell_thickness * voxel_size
    )
    lower_leg_size = (
        np.array([lower_leg_x, lower_leg_y, lower_leg_z]) * voxel_size
        + 2 * soft_shell_thickness * voxel_size
    )
    foot_size = (
        np.array([foot_x, foot_y, foot_z]) * voxel_size
        + 2 * soft_shell_thickness * voxel_size
    )

    # Convert offset parameters from voxels to meters
    upper_leg_y_offset_size = upper_leg_y_offset_voxels * voxel_size
    lower_leg_z_offset_size = lower_leg_z_offset_voxels * voxel_size
    foot_z_offset_size = foot_z_offset_voxels * 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)
    structure_config.material_references.append(material_name)

    all_bodies = []  # List of (body_sid, body_config, body_info)

    # =========================================================================
    # Body 0: Torso
    # =========================================================================
    # Position torso so it's centered and at appropriate height
    # Total height includes all leg segments and their z-offsets
    torso_center_height_struct = (
        foot_size[2]
        + foot_z_offset_size
        + lower_leg_size[2]
        + lower_leg_z_offset_size
        + upper_leg_size[2]
    )
    torso_origin_struct = (
        -torso_size[0] / 2.0,
        -torso_size[1] / 2.0,
        torso_center_height_struct - torso_size[2] / 2.0,
    )

    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_struct,
        relative_orientation=(0.0, 0.0, 0.0, 1.0),
        soft_shell_thickness=soft_shell_thickness,
        voxel_size=voxel_size,
    )
    all_bodies.append((0, torso_body, torso_info))

    # =========================================================================
    # upper legs
    # =========================================================================
    # Hip positions in torso local coordinates
    torso_origin_to_hip_anchor = [
        (
            0.0,
            torso_size[1] / 2.0,
            torso_size[2] / 2.0,
        ),  # Left hip (at -X edge, middle height)
        (
            torso_size[0],
            torso_size[1] / 2.0,
            torso_size[2] / 2.0,
        ),  # Right hip (at +X edge, middle height)
    ]

    # Storage for joint anchors
    upper_leg_origin_to_hip_anchor = []
    knee_anchors_upper_leg = []
    lower_leg_origin_to_knee_anchor = []
    lower_leg_to_ankle_anchors = []
    foot_origin_to_ankle_anchors = []

    upper_leg_origin_to_knee_anchor = []

    for leg_idx, torso_origin_to_hip_joint in enumerate(torso_origin_to_hip_anchor):
        # Determine if this is left (0) or right (1) leg
        is_left = leg_idx == 0
        hip_joint_x_offset = (
            -upper_leg_y_offset_size if is_left else upper_leg_y_offset_size
        )

        # =====================================================================
        # Upper Leg (vertical, extending downward)
        # =====================================================================
        # Hip joint position in structure space
        hip_anchor_struct = np.array(
            [
                torso_origin_struct[0] + torso_origin_to_hip_joint[0],
                torso_origin_struct[1] + torso_origin_to_hip_joint[1],
                torso_origin_struct[2] + torso_origin_to_hip_joint[2],
            ]
        )

        # Upper leg extends downward from hip
        # Position upper leg so its top-center aligns with hip joint
        upper_leg_origin = hip_anchor_struct - np.array(
            [
                upper_leg_size[0] if is_left else 0.0,
                upper_leg_size[1] / 2.0,
                upper_leg_size[2],
            ]
        )
        upper_leg_origin[0] += hip_joint_x_offset

        upper_leg_quat = (0.0, 0.0, 0.0, 1.0)

        upper_leg_sid = leg_idx + 1  # Bodies 1, 2
        upper_leg_body, upper_leg_info = create_body_config(
            body_sid=upper_leg_sid,
            x_voxels=upper_leg_x,
            y_voxels=upper_leg_y,
            z_voxels=upper_leg_z,
            relative_origin_position=tuple(upper_leg_origin),
            relative_orientation=upper_leg_quat,
            soft_shell_thickness=soft_shell_thickness,
            voxel_size=voxel_size,
        )
        all_bodies.append((upper_leg_sid, upper_leg_body, upper_leg_info))

        upper_leg_origin_to_hip_anchor_local = hip_anchor_struct - upper_leg_origin
        upper_leg_origin_to_hip_anchor.append(upper_leg_origin_to_hip_anchor_local)

        # =====================================================================
        # Lower Leg (vertical, extending downward)
        # =====================================================================
        # Knee joint at bottom of upper leg (local anchor position)
        upper_leg_origin_to_knee_anchor_local = (
            upper_leg_size[0] / 2.0,
            upper_leg_size[1] / 2.0,
            -lower_leg_z_offset_size,
        )
        upper_leg_origin_to_knee_anchor.append(upper_leg_origin_to_knee_anchor_local)

        # Calculate knee joint position in structure coordinates
        knee_anchor_struct = upper_leg_origin + np.array(
            upper_leg_origin_to_knee_anchor_local
        )

        # Lower leg extends downward from knee
        lower_leg_origin_struct = knee_anchor_struct - np.array(
            [
                lower_leg_size[0] / 2.0,
                lower_leg_size[1] / 2.0,
                lower_leg_size[2],
            ]
        )
        lower_leg_quat = (0.0, 0.0, 0.0, 1.0)

        # Create lower leg body
        lower_leg_sid = leg_idx + 3  # Bodies 3, 4
        lower_leg_body, lower_leg_info = create_body_config(
            body_sid=lower_leg_sid,
            x_voxels=lower_leg_x,
            y_voxels=lower_leg_y,
            z_voxels=lower_leg_z,
            relative_origin_position=tuple(lower_leg_origin_struct),
            relative_orientation=lower_leg_quat,
            soft_shell_thickness=soft_shell_thickness,
            voxel_size=voxel_size,
        )
        all_bodies.append((lower_leg_sid, lower_leg_body, lower_leg_info))

        lower_leg_origin_to_knee_anchor_local = (
            lower_leg_size[0] / 2.0,
            lower_leg_size[1] / 2.0,
            lower_leg_size[2],  # Top of lower leg
        )
        lower_leg_origin_to_knee_anchor.append(lower_leg_origin_to_knee_anchor_local)

        # =====================================================================
        # Foot (flat platform at bottom)
        # =====================================================================
        # Ankle joint at bottom of lower leg
        ankle_anchor_struct = lower_leg_origin_struct + np.array(
            [
                lower_leg_size[0] / 2.0,
                lower_leg_size[1] / 2.0,
                -lower_leg_z_offset_size,  # Bottom of lower leg
            ]
        )

        # Foot positioned so its top-center aligns with ankle
        foot_origin = ankle_anchor_struct - np.array(
            [
                foot_size[0] / 2.0,
                foot_size[1] / 2.0,
                foot_size[2],
            ]
        )
        foot_quat = (0.0, 0.0, 0.0, 1.0)

        # Create foot body
        foot_sid = leg_idx + 5  # Bodies 5, 6
        foot_body, foot_info = create_body_config(
            body_sid=foot_sid,
            x_voxels=foot_x,
            y_voxels=foot_y,
            z_voxels=foot_z,
            relative_origin_position=tuple(foot_origin),
            relative_orientation=foot_quat,
            soft_shell_thickness=soft_shell_thickness,
            voxel_size=voxel_size,
        )
        all_bodies.append((foot_sid, foot_body, foot_info))

        # Ankle anchors
        lower_leg_ankle_anchor_local = (
            lower_leg_size[0] / 2.0,
            lower_leg_size[1] / 2.0,
            -lower_leg_z_offset_size,  # Bottom of lower leg
        )
        lower_leg_to_ankle_anchors.append(lower_leg_ankle_anchor_local)

        foot_origin_to_ankle_anchor_loca = (
            foot_size[0] / 2.0,
            foot_size[1] / 2.0,
            foot_size[2],  # Top of foot
        )
        foot_origin_to_ankle_anchors.append(foot_origin_to_ankle_anchor_loca)

    # =========================================================================
    # Sort bodies by body_sid and append to structure_config
    # =========================================================================
    all_bodies.sort(key=lambda x: x[0])
    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(2):
        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(*torso_origin_to_hip_anchor[leg_idx])

        # Upper leg side
        constraint.b_body_sid = leg_idx + 1
        constraint.b_segment_bid = 1
        constraint.b_local_anchor = RVec3rf(*upper_leg_origin_to_hip_anchor[leg_idx])

        # Hinge axis (X-axis for forward/backward swing)
        # Left leg (idx=0): +X axis, Right leg (idx=1): -X axis
        if leg_idx == 0:  # Left leg
            constraint.hinge_a_local_axis = RVec3rf(1.0, 0.0, 0.0)
            constraint.hinge_b_local_axis = RVec3rf(-1.0, 0.0, 0.0)
        else:  # Right leg
            constraint.hinge_a_local_axis = RVec3rf(-1.0, 0.0, 0.0)
            constraint.hinge_b_local_axis = RVec3rf(1.0, 0.0, 0.0)

        constraint.hinge_rotation_angle_signal_sid = leg_idx
        constraint.hinge_min = -0.6  # ~-34 degrees
        constraint.hinge_max = 0.6  # ~+34 degrees
        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(2):
        constraint = RS_StructureConstraintConfig()
        constraint.type = RSE_StructureConstraintType.RSE_HINGE_JOINT

        # Upper leg side
        constraint.a_body_sid = leg_idx + 1
        constraint.a_segment_bid = 1
        constraint.a_local_anchor = RVec3rf(*upper_leg_origin_to_knee_anchor[leg_idx])

        # Lower leg side
        constraint.b_body_sid = leg_idx + 3
        constraint.b_segment_bid = 1
        constraint.b_local_anchor = RVec3rf(*lower_leg_origin_to_knee_anchor[leg_idx])

        # Hinge axis (X-axis for bending)
        # Left leg (idx=0): +X axis, Right leg (idx=1): -X axis
        if leg_idx == 0:  # Left leg
            constraint.hinge_a_local_axis = RVec3rf(1.0, 0.0, 0.0)
            constraint.hinge_b_local_axis = RVec3rf(-1.0, 0.0, 0.0)
            constraint.hinge_min = -1.2  # ~-69 degrees
            constraint.hinge_max = 0.2  # ~+11 degrees

        else:  # Right leg
            constraint.hinge_a_local_axis = RVec3rf(-1.0, 0.0, 0.0)
            constraint.hinge_b_local_axis = RVec3rf(1.0, 0.0, 0.0)
            constraint.hinge_min = -0.2  # ~-69 degrees
            constraint.hinge_max = 1.2  # ~+11 degrees

        constraint.hinge_rotation_angle_signal_sid = 2 + leg_idx
        constraint.hinge_max_torque = 6.0

        structure_config.constraints.append(constraint)

    # =========================================================================
    # Create ankle hinge joints (lower legs to feet)
    # =========================================================================
    for leg_idx in range(2):
        constraint = RS_StructureConstraintConfig()
        constraint.type = RSE_StructureConstraintType.RSE_HINGE_JOINT

        # Lower leg side
        constraint.a_body_sid = leg_idx + 3
        constraint.a_segment_bid = 1
        constraint.a_local_anchor = RVec3rf(*lower_leg_to_ankle_anchors[leg_idx])

        # Foot side
        constraint.b_body_sid = leg_idx + 5
        constraint.b_segment_bid = 1
        constraint.b_local_anchor = RVec3rf(*foot_origin_to_ankle_anchors[leg_idx])

        # Hinge axis (X-axis for foot pitch)
        # Left leg (idx=0): +X axis, Right leg (idx=1): -X axis
        if leg_idx == 0:  # Left leg
            constraint.hinge_a_local_axis = RVec3rf(1.0, 0.0, 0.0)
            constraint.hinge_b_local_axis = RVec3rf(-1.0, 0.0, 0.0)
        else:  # Right leg
            constraint.hinge_a_local_axis = RVec3rf(-1.0, 0.0, 0.0)
            constraint.hinge_b_local_axis = RVec3rf(1.0, 0.0, 0.0)

        constraint.hinge_rotation_angle_signal_sid = 4 + leg_idx
        constraint.hinge_min = -0.4  # ~-23 degrees
        constraint.hinge_max = 0.4  # ~+23 degrees
        constraint.hinge_max_torque = 6.0

        structure_config.constraints.append(constraint)

    structure_config.rotation_angle_signal_num = 6

    # =========================================================================
    # Build auxiliary structure dict (for mesh building / visualization)
    # =========================================================================
    combined_size = 64
    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)

    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))

    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_struct) + 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
    connections = []

    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,
        "voxel_size": voxel_size,
    }

    return structure_config, structure


if __name__ == "__main__":
    # Test the function
    soft_shell_thickness = 1
    upper_leg_y_offset_voxels = 0
    lower_leg_z_offset_voxels = 1
    foot_z_offset_voxels = 0

    structure_config, structure = create_biped_6dof(
        "biped_6dof",
        (0.0, 0.0, 0.0),
        "material_0",
        soft_shell_thickness=soft_shell_thickness,
        upper_leg_y_offset_voxels=upper_leg_y_offset_voxels,
        lower_leg_z_offset_voxels=lower_leg_z_offset_voxels,
        foot_z_offset_voxels=foot_z_offset_voxels,
    )

    print("=" * 60)
    print("Biped 6-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: {soft_shell_thickness} voxels")
    print(f"Upper leg Y offset: {upper_leg_y_offset_voxels} voxels (outward)")
    print(f"Lower leg Z offset: {lower_leg_z_offset_voxels} voxels (downward)")
    print(f"Foot Z offset: {foot_z_offset_voxels} voxels (downward)")

    print("\nBodies:")
    body_names = [
        "Torso",
        "Left Upper Leg",
        "Right Upper Leg",
        "Left Lower Leg",
        "Right Lower Leg",
        "Left Foot",
        "Right Foot",
    ]
    body_infos = structure.get("body_infos", [])
    for i, body in enumerate(structure_config.bodies):
        pos = body.relative_origin_position
        ori = body.relative_orientation
        body_name = body_names[i] if i < len(body_names) else f"Body {i}"
        print(
            f"  {body_name} (body_sid={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:")
    constraint_names = [
        "Left Hip",
        "Right Hip",
        "Left Knee",
        "Right Knee",
        "Left Ankle",
        "Right Ankle",
    ]
    for i, constraint in enumerate(structure_config.constraints):
        constraint_name = (
            constraint_names[i] if i < len(constraint_names) else f"Joint {i}"
        )
        print(
            f"  {constraint_name} (joint {i}): "
            f"body {constraint.a_body_sid} <-> body {constraint.b_body_sid}"
        )
        print(f"    Signal SID: {constraint.hinge_rotation_angle_signal_sid}")
        print(
            f"    Range: [{constraint.hinge_min:.2f}, {constraint.hinge_max:.2f}] radians"
        )

    print("\nTo visualize the robot, create a visualization script similar to:")
    print("  visualize_quadruped_8dof.py or visualize_ant_8dof.py")

    # Save robot config to data/robot_config/biped_6dof.data
    output_path = os.path.join(
        os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__)))),
        "data",
        "robot_config",
        "biped_6dof.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}")
