import pybullet as p
import numpy as np

def create_real_intersection(road_length=4.2672, road_width=0.6096, road_height=0.1):
    """
    Creates a road intersection in PyBullet with roads, stop lines, walls, and stop signs.
    
    Args:
        road_length: Length of the roads (default: 4.2672) - 14 feet
        road_width: Width of the roads (default: 0.6096) - 24 inches
        road_height: Height/thickness of the road surface (default: 0.1)
    """
    # Move intersection up by adjusting z position
    z_offset = 0.05
    
    # Visual properties
    road_color = [0.1, 0.1, 0.1, 1]  # Dark gray
    line_color = [1, 1, 1, 1]        # White
    wall_color = [0, 0, 0, 0]        # Black
    
    ############################## Roads ##############################
    
    # Create the roads (North-South and East-West)
    road_shapes = [
        # North-South road
        [0, 0, z_offset, 0, 0, 0, road_width, road_length, road_height],
        # East-West road
        [0, 0, z_offset, 0, 0, 0, road_length, road_width, road_height]
    ]
    
    # Create collision and visual shapes for roads
    for shape in road_shapes:
        visual_shape = p.createVisualShape(
            shapeType=p.GEOM_BOX,
            halfExtents=[shape[6]/2, shape[7]/2, shape[8]/2],
            rgbaColor=road_color
        )
        collision_shape = p.createCollisionShape(
            shapeType=p.GEOM_BOX,
            halfExtents=[shape[6]/2, shape[7]/2, shape[8]/2]
        )
        p.createMultiBody(
            baseMass=0,
            baseCollisionShapeIndex=collision_shape,
            baseVisualShapeIndex=visual_shape,
            basePosition=[shape[0], shape[1], shape[2]],
            baseOrientation=p.getQuaternionFromEuler([shape[3], shape[4], shape[5]])
        )
    ############################## End of roads ##############################

    ############################## Stop lines ##############################
    
    # Add stop lines on all lanes
    stop_line_half_extents = [0.3, 0.05, 0.002]  # Increased height for better visibility
    stop_line_color = [1, 0, 0, 1]  # Red color for stop lines

    # Define stop line positions for each lane - moved back by 0.1 units
    stop_line_positions = [
        # North lane (horizontal line) - moved back (south)
        [0, road_width/4 + 0.3, z_offset + 0.05, 0, 0, 0],
        # South lane (horizontal line) - moved back (north)
        [0, -road_width/4 - 0.3, z_offset + 0.05, 0, 0, 0],
        # East lane (vertical line) - moved back (west)
        [road_width/4 + 0.3, 0, z_offset + 0.05, 0, 0, np.pi/2],
        # West lane (vertical line) - moved back (east)
        [-road_width/4 - 0.3, 0, z_offset + 0.05, 0, 0, np.pi/2]
    ]

    # Create stop lines
    for pos in stop_line_positions:
        stop_line_vis = p.createVisualShape(
            shapeType=p.GEOM_BOX,
            halfExtents=stop_line_half_extents,
            rgbaColor=stop_line_color
        )
        p.createMultiBody(
            baseMass=0,
            baseVisualShapeIndex=stop_line_vis,
            basePosition=pos[:3],
            baseOrientation=p.getQuaternionFromEuler(pos[3:])
        )

    ############################## End of stop lines ##############################

    ############################## Walls along the roads ##############################
    
    wall_height = 0.5  # Height of the walls
    wall_thickness = 0.1  # Thickness of the walls
    wall_length = road_length - 1.0  # Length of each wall segment
    wall_offset = 0.5  # Offset from the intersection
    
    # Define wall positions (along the roads)
    wall_configs = [
        # North road walls (2 walls)
        # Upper East wall
        [road_width/2, road_length/4 + wall_offset/2, wall_height/2, 
         wall_thickness, wall_length/2, wall_height],
        # Upper West wall
        [-road_width/2, road_length/4 + wall_offset/2, wall_height/2, 
         wall_thickness, wall_length/2, wall_height],
         
        # South road walls (2 walls)
        # Lower East wall
        [road_width/2, -road_length/4 - wall_offset/2, wall_height/2, 
         wall_thickness, wall_length/2, wall_height],
        # Lower West wall
        [-road_width/2, -road_length/4 - wall_offset/2, wall_height/2, 
         wall_thickness, wall_length/2, wall_height],
         
        # East road walls (2 walls)
        # Right Upper wall
        [road_length/4 + wall_offset/2, road_width/2, wall_height/2, 
         wall_length/2, wall_thickness, wall_height],
        # Right Lower wall
        [road_length/4 + wall_offset/2, -road_width/2, wall_height/2, 
         wall_length/2, wall_thickness, wall_height],
         
        # West road walls (2 walls)
        # Left Upper wall
        [-road_length/4 - wall_offset/2, road_width/2, wall_height/2, 
         wall_length/2, wall_thickness, wall_height],
        # Left Lower wall
        [-road_length/4 - wall_offset/2, -road_width/2, wall_height/2, 
         wall_length/2, wall_thickness, wall_height],
    ]
    
    
    for wall in wall_configs:
        visual_shape = p.createVisualShape(
            shapeType=p.GEOM_BOX,
            halfExtents=[wall[3]/2, wall[4]/2, wall[5]/2],
            rgbaColor=wall_color
        )
        collision_shape = p.createCollisionShape(
            shapeType=p.GEOM_BOX,
            halfExtents=[wall[3]/2, wall[4]/2, wall[5]/2]
        )
        p.createMultiBody(
            baseMass=0,  # Static object
            baseCollisionShapeIndex=collision_shape,
            baseVisualShapeIndex=visual_shape,
            basePosition=[wall[0], wall[1], wall[2]]
        )

    ############################## Outer boundary walls ##############################
    
    boundary_height = 1.016  # Increased height of the boundary walls
    boundary_thickness = 0.1  # Thickness of the boundary walls
    boundary_color = [1, 1, 1, 1]  # White color
    boundary_extension = 0.2  # Extension for the walls
    boundary_offset = 0.1
    
    # Define boundary wall positions (surrounding the entire intersection)
    boundary_configs = [
        # North boundary
        [0, road_length/2 + boundary_offset, boundary_height/2, 
         road_length + 2*boundary_thickness + boundary_extension, boundary_thickness, boundary_height],
        # South boundary
        [0, -road_length/2 - boundary_offset, boundary_height/2, 
         road_length + 2*boundary_thickness + boundary_extension, boundary_thickness, boundary_height],
        # East boundary
        [road_length/2 + boundary_offset, 0, boundary_height/2, 
         boundary_thickness, road_length + 2*boundary_thickness + boundary_extension, boundary_height],
        # West boundary
        [-road_length/2 - boundary_offset, 0, boundary_height/2, 
         boundary_thickness, road_length + 2*boundary_thickness + boundary_extension, boundary_height],
    ]
    # Create boundary walls
    for wall in boundary_configs:
        visual_shape = p.createVisualShape(
            shapeType=p.GEOM_BOX,
            halfExtents=[wall[3]/2, wall[4]/2, wall[5]/2],
            rgbaColor=boundary_color
        )
        collision_shape = p.createCollisionShape(
            shapeType=p.GEOM_BOX,
            halfExtents=[wall[3]/2, wall[4]/2, wall[5]/2]
        )
        p.createMultiBody(
            baseMass=0,  # Static object
            baseCollisionShapeIndex=collision_shape,
            baseVisualShapeIndex=visual_shape,
            basePosition=[wall[0], wall[1], wall[2]]
        )
    ############################## End of outer boundary walls ##############################

    ############################## Floor ##############################
    
    # Add a white floor beneath the intersection
    floor_size = road_length + 2.0  # Make the floor larger than the intersection
    floor_height = 0.01
    floor_color = [1, 1, 1, 1]  # White color
    
    floor_visual = p.createVisualShape(
        shapeType=p.GEOM_BOX,
        halfExtents=[floor_size/2, floor_size/2, floor_height/2],
        rgbaColor=floor_color
    )
    floor_collision = p.createCollisionShape(
        shapeType=p.GEOM_BOX,
        halfExtents=[floor_size/2, floor_size/2, floor_height/2]
    )
    
    # Position the floor slightly below the road to avoid z-fighting
    p.createMultiBody(
        baseMass=0,  # Static object
        baseCollisionShapeIndex=floor_collision,
        baseVisualShapeIndex=floor_visual,
        basePosition=[0, 0, -floor_height/2],  # Position below the road
        baseOrientation=p.getQuaternionFromEuler([0, 0, 0])
    )
    
    ############################## End of floor ##############################

def create_real_intersection_walls(road_length=4.2672, road_width=0.6096, road_height=0.0254):
    """
    Creates a road intersection in PyBullet with roads, stop lines, walls, and stop signs.
    
    Args:
        road_length: Length of the roads (default: 4.2672) - 14 feet
        road_width: Width of the roads (default: 0.6096) - 24 inches
        road_height: Height/thickness of the road surface (default: 0.1)
    """
    # Move intersection up by adjusting z position
    z_offset = 0.01
    

    road_z = (0.01/2) + (road_height/2)

    # Visual properties
    road_color = [0.1, 0.1, 0.1, 1]  # Dark gray
    line_color = [1, 1, 1, 1]        # White
    wall_color = [1, 1, 1, 1]        # Gray
    
    ############################## Roads ##############################
    
    # Create the roads (North-South and East-West)
    road_shapes = [
        # North-South road
        [0, 0, road_z, 0, 0, 0, road_width, road_length, road_height],
        # East-West road
        [0, 0, road_z, 0, 0, 0, road_length, road_width, road_height]
    ]
    
    # Create collision and visual shapes for roads
    for shape in road_shapes:
        visual_shape = p.createVisualShape(
            shapeType=p.GEOM_BOX,
            halfExtents=[shape[6]/2, shape[7]/2, shape[8]/2],
            rgbaColor=road_color
        )
        collision_shape = p.createCollisionShape(
            shapeType=p.GEOM_BOX,
            halfExtents=[shape[6]/2, shape[7]/2, shape[8]/2]
        )
        p.createMultiBody(
            baseMass=0,
            baseCollisionShapeIndex=collision_shape,
            baseVisualShapeIndex=visual_shape,
            basePosition=[shape[0], shape[1], shape[2]],
            baseOrientation=p.getQuaternionFromEuler([shape[3], shape[4], shape[5]])
        )
    ############################## End of roads ##############################

    ############################## Stop lines ##############################
    
    # Add stop lines on all lanes
    stop_line_half_extents = [0.3, 0.05, 0.002]  # Increased height for better visibility
    stop_line_color = [1, 0, 0, 1]  # Red color for stop lines

    # Define stop line positions for each lane - moved back by 0.1 units
    stop_line_positions = [
        # North lane (horizontal line) - moved back (south)
        [0, road_width/4 + 0.3, road_z+.015 , 0, 0, 0],
        # South lane (horizontal line) - moved back (north)
        [0, -road_width/4 - 0.3, road_z+.015, 0, 0, 0],
        # East lane (vertical line) - moved back (west)
        [road_width/4 + 0.3, 0, road_z +.015, 0, 0, np.pi/2],
        # West lane (vertical line) - moved back (east)
        [-road_width/4 - 0.3, 0, road_z +.015, 0, 0, np.pi/2]
    ]

    # Create stop lines
    # for pos in stop_line_positions:
    #     stop_line_vis = p.createVisualShape(
    #         shapeType=p.GEOM_BOX,
    #         halfExtents=stop_line_half_extents,
    #         rgbaColor=stop_line_color
    #     )
    #     p.createMultiBody(
    #         baseMass=0,
    #         baseVisualShapeIndex=stop_line_vis,
    #         basePosition=pos[:3],
    #         baseOrientation=p.getQuaternionFromEuler(pos[3:])
    #     )

    ############################## End of stop lines ##############################

    ############################## Walls along the roads ##############################
    
    # wall_height = 0.1778  # Height of the walls (7 inches)
    # wall_thickness = 0.0476  # Thickness of the walls
    # wall_length = 1.016  # Length of each wall segment (40 inches)
    # wall_offset = 0.5  # Offset from the intersection
    
    # # Define wall positions (along the roads)
    # wall_configs = [
    #     # North road walls (2 walls)
    #     # Upper East wall
    #     [road_width/2, road_length/4 - 0.5 + wall_offset/2, wall_height/2, 
    #      wall_thickness, wall_length, wall_height],
    #     # Upper West wall
    #     [-road_width/2, road_length/4 - 0.5 + wall_offset/2, wall_height/2, 
    #      wall_thickness, wall_length, wall_height],
         
    #     # South road walls (2 walls)
    #     # Lower East wall
    #     [road_width/2, -road_length/4 +0.5 - wall_offset/2, wall_height/2, 
    #      wall_thickness, wall_length, wall_height],
    #     # Lower West wall
    #     [-road_width/2, -road_length/4 +0.5 - wall_offset/2, wall_height/2, 
    #      wall_thickness, wall_length, wall_height],
         
    #     # # East road walls (2 walls)
    #     # # Right Upper wall
    #     # [road_length/4 + wall_offset/2, road_width/2, wall_height/2, 
    #     #  wall_length/2, wall_thickness, wall_height],
    #     # # Right Lower wall
    #     # [road_length/4 + wall_offset/2, -road_width/2, wall_height/2, 
    #     #  wall_length/2, wall_thickness, wall_height],
         
    #     # # West road walls (2 walls)
    #     # # Left Upper wall
    #     # [-road_length/4 - wall_offset/2, road_width/2, wall_height/2, 
    #     #  wall_length/2, wall_thickness, wall_height],
    #     # # Left Lower wall
    #     # [-road_length/4 - wall_offset/2, -road_width/2, wall_height/2, 
    #     #  wall_length/2, wall_thickness, wall_height],
    # ]
    
    
    # for wall in wall_configs:
    #     visual_shape = p.createVisualShape(
    #         shapeType=p.GEOM_BOX,
    #         halfExtents=[wall[3]/2, wall[4]/2, wall[5]/2],
    #         rgbaColor=wall_color
    #     )
    #     collision_shape = p.createCollisionShape(
    #         shapeType=p.GEOM_BOX,
    #         halfExtents=[wall[3]/2, wall[4]/2, wall[5]/2]
    #     )
    #     p.createMultiBody(
    #         baseMass=0,  # Static object
    #         baseCollisionShapeIndex=collision_shape,
    #         baseVisualShapeIndex=visual_shape,
    #         basePosition=[wall[0], wall[1], wall[2]]
    #     )

    ############################## Outer boundary walls ##############################
    
    boundary_height = 1.016  # Increased height of the boundary walls
    boundary_thickness = 0.1  # Thickness of the boundary walls
    boundary_color = [1, 1, 1, 1]  # White color
    boundary_extension = 0.2  # Extension for the walls
    boundary_offset = 0.1
    
    # Define boundary wall positions (surrounding the entire intersection)
    boundary_configs = [
        # North boundary
        [0, road_length/2 + boundary_offset, boundary_height/2, 
         road_length + 2*boundary_thickness + boundary_extension, boundary_thickness, boundary_height*2],
        # South boundary
        [0, -road_length/2 - boundary_offset, boundary_height/2, 
         road_length + 2*boundary_thickness + boundary_extension, boundary_thickness, boundary_height],
        # East boundary
        [road_length/2 + boundary_offset, 0, boundary_height/2, 
         boundary_thickness, road_length + 2*boundary_thickness + boundary_extension, boundary_height],
        # West boundary
        [-road_length/2 - boundary_offset, 0, boundary_height/2, 
         boundary_thickness, road_length + 2*boundary_thickness + boundary_extension, boundary_height],
    ]
    # Create boundary walls
    for wall in boundary_configs:
        visual_shape = p.createVisualShape(
            shapeType=p.GEOM_BOX,
            halfExtents=[wall[3]/2, wall[4]/2, wall[5]/2],
            rgbaColor=boundary_color
        )
        collision_shape = p.createCollisionShape(
            shapeType=p.GEOM_BOX,
            halfExtents=[wall[3]/2, wall[4]/2, wall[5]/2]
        )
        p.createMultiBody(
            baseMass=0,  # Static object
            baseCollisionShapeIndex=collision_shape,
            baseVisualShapeIndex=visual_shape,
            basePosition=[wall[0], wall[1], wall[2]]
        )
    ############################## End of outer boundary walls ##############################

    ############################## Floor ##############################
    
    # Add a white floor beneath the intersection
    floor_size = road_length + 2.0  # Make the floor larger than the intersection
    floor_height = 0.01
    floor_color = [1, 1, 1, 1]  # White color
    
    floor_visual = p.createVisualShape(
        shapeType=p.GEOM_BOX,
        halfExtents=[floor_size/2, floor_size/2, floor_height/2],
        rgbaColor=floor_color
    )
    floor_collision = p.createCollisionShape(
        shapeType=p.GEOM_BOX,
        halfExtents=[floor_size/2, floor_size/2, floor_height/2]
    )
    
    # Position the floor slightly below the road to avoid z-fighting
    p.createMultiBody(
        baseMass=0,  # Static object
        baseCollisionShapeIndex=floor_collision,
        baseVisualShapeIndex=floor_visual,
        basePosition=[0, 0, 0],  # Position below the road
        baseOrientation=p.getQuaternionFromEuler([0, 0, 0])
    )
    
    ############################## End of floor ##############################

    ############################## Road lines ##############################
    
    # # Add road lines
    # line_width = 0.1
    # line_length = road_length/4
    # line_height = road_height + 0.001
    
    # line_shapes = [
    #     # North center line
    #     [0, road_length/4, z_offset + 0.001, 0, 0, 0, line_width, line_length, line_height],
    #     # South center line
    #     [0, -road_length/4, z_offset + 0.001, 0, 0, 0, line_width, line_length, line_height],
    #     # East center line
    #     [road_length/4, 0, z_offset + 0.001, 0, 0, 0, line_length, line_width, line_height],
    #     # West center line
    #     [-road_length/4, 0, z_offset + 0.001, 0, 0, 0, line_length, line_width, line_height]
    # ]
    
    # # Create lines
    # for line in line_shapes:
    #     visual_shape = p.createVisualShape(
    #         shapeType=p.GEOM_BOX,
    #         halfExtents=[line[6]/2, line[7]/2, line[8]/2],
    #         rgbaColor=line_color
    #     )
    #     p.createMultiBody(
    #         baseMass=0,
    #         baseVisualShapeIndex=visual_shape,
    #         basePosition=[line[0], line[1], line[2]],
    #         baseOrientation=p.getQuaternionFromEuler([line[3], line[4], line[5]])
    #     )
    ############################## End of road lines ##############################

    ############################## Stop signs ##############################
    
    # Add stop signs at each lane
    # COMMENTED OUT FOR NOW
    stop_sign_size = 0.1778  # 5 inches in meters
    pole_width = 0.04    
    pole_height = 0.762  # 30 inches in meters
    
    # Define stop sign positions for each lane
    stop_sign_positions = [
        # South lane - place closer to the stop line
        [0.4, -road_width/4 - 0.35, stop_sign_size/2 + pole_height, 0, 0, 0],
        # North lane - place closer to the stop line
        [0.4, road_width/4 + 0.4, stop_sign_size/2 + pole_height, 0, 0, np.pi],
        # East lane - place closer to the stop line
        # [road_width/4 + 0.25, 0.4, stop_sign_size/2 + pole_height, 0, 0, -np.pi/2],
        # West lane - place closer to the stop line 
        # [-road_width/4 - 0.4, 2.4, stop_sign_size/2 + pole_height, 0, 0, np.pi/2]
         [0, road_width + 0.9, stop_sign_size/2 + pole_height, 0, 0, 0]
    ]
    
    # Create stop signs and poles
    # for pos in stop_sign_positions:
    #     # Create the stop sign (red square)
    #     stop_sign_visual = p.createVisualShape(
    #         shapeType=p.GEOM_BOX,  
    #         halfExtents=[stop_sign_size/2, 0.005, stop_sign_size/2],
    #         rgbaColor=[0.8, 0, 0, 1]  
    #     )
    #     stop_sign_collision = p.createCollisionShape(
    #         shapeType=p.GEOM_BOX,
    #         halfExtents=[stop_sign_size/2, 0.005, stop_sign_size/2]
    #     )
        
    #     # Create the pole (gray box)
    #     pole_visual = p.createVisualShape(
    #         shapeType=p.GEOM_BOX,
    #         halfExtents=[pole_width/2, pole_width/2, pole_height/2],
    #         rgbaColor=[0.7, 0.7, 0.7, 1]  # Gray color
    #     )
    #     pole_collision = p.createCollisionShape(
    #         shapeType=p.GEOM_BOX,
    #         halfExtents=[pole_width/2, pole_width/2, pole_height/2]
    #     )
        
    #     # Create the stop sign and pole as separate bodies
    #     p.createMultiBody(
    #         baseMass=0,  # Static object
    #         baseCollisionShapeIndex=stop_sign_collision,
    #         baseVisualShapeIndex=stop_sign_visual,
    #         basePosition=pos[:3],
    #         baseOrientation=p.getQuaternionFromEuler(pos[3:])
    #     )
        
    #     # Position the pole below the sign
    #     pole_position = [pos[0], pos[1], pole_height/2]
    #     p.createMultiBody(
    #         baseMass=0,  # Static object
    #         baseCollisionShapeIndex=pole_collision,
    #         baseVisualShapeIndex=pole_visual,
    #         basePosition=pole_position,
    #         baseOrientation=p.getQuaternionFromEuler([0, 0, 0])
    #     )
    ############################## End of stop signs ##############################

def create_intersection(road_length=10.0, road_width=2.0, road_height=0.1):
    """
    Creates a road intersection in PyBullet with roads, stop lines, walls, and stop signs.
    
    Args:
        road_length: Length of the roads (default: 10.0)
        road_width: Width of the roads (default: 2.0)
        road_height: Height/thickness of the road surface (default: 0.1)
    """
    # Move intersection up by adjusting z position
    z_offset = 0.05
    
    # Visual properties
    road_color = [0.1, 0.1, 0.1, 1]  # Dark gray
    line_color = [1, 1, 1, 1]        # White
    wall_color = [0, 0, 0, 0]        # Black
    
    ############################## Roads ##############################
    
    # Create the roads (North-South and East-West)
    road_shapes = [
        # North-South road
        [0, 0, z_offset, 0, 0, 0, road_width, road_length, road_height],
        # East-West road
        [0, 0, z_offset, 0, 0, 0, road_length, road_width, road_height]
    ]
    
    # Create collision and visual shapes for roads
    for shape in road_shapes:
        visual_shape = p.createVisualShape(
            shapeType=p.GEOM_BOX,
            halfExtents=[shape[6]/2, shape[7]/2, shape[8]/2],
            rgbaColor=road_color
        )
        collision_shape = p.createCollisionShape(
            shapeType=p.GEOM_BOX,
            halfExtents=[shape[6]/2, shape[7]/2, shape[8]/2]
        )
        p.createMultiBody(
            baseMass=0,
            baseCollisionShapeIndex=collision_shape,
            baseVisualShapeIndex=visual_shape,
            basePosition=[shape[0], shape[1], shape[2]],
            baseOrientation=p.getQuaternionFromEuler([shape[3], shape[4], shape[5]])
        )
    ############################## End of roads ##############################

    ############################## Stop lines ##############################
    
    # Add stop lines on all lanes
    stop_line_half_extents = [0.3, 0.05, 0.001] 
    stop_line_color = [1, 0, 0, 1]  # Red color for stop lines

    # Define stop line positions for each lane - adjust to match reward function
    stop_line_positions = [
        # North lane (horizontal line)
        [-0.5, road_length/2 - 3.8, z_offset + 0.1, 0, 0, 0],
        # South lane (horizontal line)
        [0.5, -road_length/2 + 3.8, z_offset + 0.1, 0, 0, 0],
        # East lane (vertical line)
        [road_length/2 - 3.8, 0.5, z_offset + 0.1, 0, 0, np.pi/2],
        # West lane (vertical line)
        [-road_length/2 + 3.8, -0.5, z_offset + 0.1, 0, 0, np.pi/2]
    ]

    # Create stop lines
    for pos in stop_line_positions:
        stop_line_vis = p.createVisualShape(
            shapeType=p.GEOM_BOX,
            halfExtents=stop_line_half_extents,
            rgbaColor=stop_line_color
        )
        p.createMultiBody(
            baseMass=0,
            baseVisualShapeIndex=stop_line_vis,
            basePosition=pos[:3],
            baseOrientation=p.getQuaternionFromEuler(pos[3:])
        )

    ############################## End of stop lines ##############################

    ############################## Walls along the roads ##############################
    
    wall_height = 0.5  # Height of the walls
    wall_thickness = 0.1  # Thickness of the walls
    wall_length = 4.0  # Length of each wall segment
    wall_offset = 0.5  # Offset from the intersection
    
    # Define wall positions (along the roads)
    wall_configs = [
        # North road walls (2 walls)
        # Upper East wall
        [road_width/2, road_length/4 + wall_offset, wall_height/2, 
         wall_thickness, wall_length, wall_height],
        # Upper West wall
        [-road_width/2, road_length/4 + wall_offset, wall_height/2, 
         wall_thickness, wall_length, wall_height],
         
        # South road walls (2 walls)
        # Lower East wall
        [road_width/2, -road_length/4 - wall_offset, wall_height/2, 
         wall_thickness, wall_length, wall_height],
        # Lower West wall
        [-road_width/2, -road_length/4 - wall_offset, wall_height/2, 
         wall_thickness, wall_length, wall_height],
         
        # East road walls (2 walls)
        # Right Upper wall
        [road_length/4 + wall_offset, road_width/2, wall_height/2, 
         wall_length, wall_thickness, wall_height],
        # Right Lower wall
        [road_length/4 + wall_offset, -road_width/2, wall_height/2, 
         wall_length, wall_thickness, wall_height],
         
        # West road walls (2 walls)
        # Left Upper wall
        [-road_length/4 - wall_offset, road_width/2, wall_height/2, 
         wall_length, wall_thickness, wall_height],
        # Left Lower wall
        [-road_length/4 - wall_offset, -road_width/2, wall_height/2, 
         wall_length, wall_thickness, wall_height],
    ]
    
    
    for wall in wall_configs:
        visual_shape = p.createVisualShape(
            shapeType=p.GEOM_BOX,
            halfExtents=[wall[3]/2, wall[4]/2, wall[5]/2],
            rgbaColor=wall_color
        )
        collision_shape = p.createCollisionShape(
            shapeType=p.GEOM_BOX,
            halfExtents=[wall[3]/2, wall[4]/2, wall[5]/2]
        )
        p.createMultiBody(
            baseMass=0,  # Static object
            baseCollisionShapeIndex=collision_shape,
            baseVisualShapeIndex=visual_shape,
            basePosition=[wall[0], wall[1], wall[2]]
        )

    ############################## Outer boundary walls ##############################
    
    boundary_height = 1.5  # Increased height of the boundary walls
    boundary_thickness = 0.2  # Thickness of the boundary walls
    boundary_color = [1, 1, 1, 1]  # White color
    boundary_extension = 0.1  # Extension for the walls
    
    # Define boundary wall positions (surrounding the entire intersection)
    boundary_configs = [
        # North boundary
        [0, road_length/2 + 0.5, boundary_height/2, 
         road_length + 2*boundary_thickness + boundary_extension, boundary_thickness, boundary_height],
        # South boundary
        [0, -road_length/2 - 0.5, boundary_height/2, 
         road_length + 2*boundary_thickness + boundary_extension, boundary_thickness, boundary_height],
        # East boundary
        [road_length/2 + 0.5, 0, boundary_height/2, 
         boundary_thickness, road_length + 2*boundary_thickness + boundary_extension, boundary_height],
        # West boundary
        [-road_length/2 - 0.5, 0, boundary_height/2, 
         boundary_thickness, road_length + 2*boundary_thickness + boundary_extension, boundary_height],
    ]
    
    # Create boundary walls
    for wall in boundary_configs:
        visual_shape = p.createVisualShape(
            shapeType=p.GEOM_BOX,
            halfExtents=[wall[3]/2, wall[4]/2, wall[5]/2],
            rgbaColor=boundary_color
        )
        collision_shape = p.createCollisionShape(
            shapeType=p.GEOM_BOX,
            halfExtents=[wall[3]/2, wall[4]/2, wall[5]/2]
        )
        p.createMultiBody(
            baseMass=0,  # Static object
            baseCollisionShapeIndex=collision_shape,
            baseVisualShapeIndex=visual_shape,
            basePosition=[wall[0], wall[1], wall[2]]
        )
    ############################## End of outer boundary walls ##############################

    ############################## Road lines ##############################
    
    # Add road lines
    line_width = 0.1
    line_length = road_length/4
    line_height = road_height + 0.001
    
    line_shapes = [
        # North center line
        [0, road_length/4, z_offset + 0.001, 0, 0, 0, line_width, line_length, line_height],
        # South center line
        [0, -road_length/4, z_offset + 0.001, 0, 0, 0, line_width, line_length, line_height],
        # East center line
        [road_length/4, 0, z_offset + 0.001, 0, 0, 0, line_length, line_width, line_height],
        # West center line
        [-road_length/4, 0, z_offset + 0.001, 0, 0, 0, line_length, line_width, line_height]
    ]
    
    # Create lines
    for line in line_shapes:
        visual_shape = p.createVisualShape(
            shapeType=p.GEOM_BOX,
            halfExtents=[line[6]/2, line[7]/2, line[8]/2],
            rgbaColor=line_color
        )
        p.createMultiBody(
            baseMass=0,
            baseVisualShapeIndex=visual_shape,
            basePosition=[line[0], line[1], line[2]],
            baseOrientation=p.getQuaternionFromEuler([line[3], line[4], line[5]])
        )
    ############################## End of road lines ##############################

    ############################## Stop signs ##############################
    
    # Add stop signs at each lane
    stop_sign_height = 1.0  
    stop_sign_size = 0.3   
    pole_radius = 0.02    
    pole_height = 0.7    
    
    # Define stop sign positions for each lane
    stop_sign_positions = [
        # South lane (moved left)
        [-road_width/2 + 2, -road_length/2 + 4.0, stop_sign_height/4 + pole_height - 0.1, 0, 0, 0],
        # North lane (moved left)
        [-road_width/2 - 0.3, road_length/2 - 4.0, stop_sign_height/4 + pole_height - 0.1, 0, 0, np.pi],
        # East lane (moved left relative to direction of travel)
        [road_length/2 - 4.0, road_width/2 + 0.3, stop_sign_height/4 + pole_height - 0.1, 0, 0, -np.pi/2],
        # West lane (moved left relative to direction of travel)
        [-road_length/2 + 4.0, -road_width/2 - 0.1, stop_sign_height/4 + pole_height - 0.1, 0, 0, np.pi/2]
    ]
    
    # Create stop signs and poles
    for pos in stop_sign_positions:
        # Create the stop sign (red square)
        stop_sign_visual = p.createVisualShape(
            shapeType=p.GEOM_BOX,  
            halfExtents=[stop_sign_size/2, 0.02, stop_sign_size/2],
            rgbaColor=[0.8, 0, 0, 1]  
        )
        stop_sign_collision = p.createCollisionShape(
            shapeType=p.GEOM_BOX,
            halfExtents=[stop_sign_size/2, 0.02, stop_sign_size/2]
        )
        
        # Create the pole (gray cylinder)
        pole_visual = p.createVisualShape(
            shapeType=p.GEOM_CYLINDER,
            radius=pole_radius,
            length=pole_height,
            rgbaColor=[0.7, 0.7, 0.7, 1]  # Gray color
        )
        pole_collision = p.createCollisionShape(
            shapeType=p.GEOM_CYLINDER,
            radius=pole_radius,
            height=pole_height
        )
        
        # Create the stop sign and pole as separate bodies
        p.createMultiBody(
            baseMass=0,  # Static object
            baseCollisionShapeIndex=stop_sign_collision,
            baseVisualShapeIndex=stop_sign_visual,
            basePosition=pos[:3],
            baseOrientation=p.getQuaternionFromEuler(pos[3:])
        )
        
        # Position the pole below the sign
        pole_position = [pos[0], pos[1], pole_height/2]
        p.createMultiBody(
            baseMass=0,  # Static object
            baseCollisionShapeIndex=pole_collision,
            baseVisualShapeIndex=pole_visual,
            basePosition=pole_position,
            baseOrientation=p.getQuaternionFromEuler([0, 0, 0])
        )
    ############################## End of stop signs ##############################