import cv2
import matplotlib.pyplot as plt
import matplotlib.patches as patches
import os
from typing import List, Dict


def visualize_grounding_boxes(image_path: str, 
                             grounding_boxes: List[Dict], 
                             output_path: str,
                             figsize: tuple = (10, 8),
                             font_size: int = 10,
                             abs_coords: bool = True) -> str:
    """
    Visualize grounding boxes on an image and save the result.
    
    Args:
        image_path: Path to input image
        grounding_boxes: List of box info, format: [{"bbox_2d": [x1, y1, x2, y2], "label": "..."}]
        output_path: Output path for the visualization
        figsize: Figure size for display
        font_size: Font size for labels
        abs_coords: Whether coordinates are absolute. True=absolute pixel coords, False=normalized (0-1000)
        
    Returns:
        str: Path to saved visualization
    """
    # Read image
    image = cv2.imread(image_path)
    if image is None:
        raise ValueError(f"Cannot read image: {image_path}")
    
    # Get image dimensions
    img_height, img_width = image.shape[:2]
    
    # Convert color space (BGR -> RGB)
    image_rgb = cv2.cvtColor(image, cv2.COLOR_BGR2RGB)
    
    # Create matplotlib figure
    fig, ax = plt.subplots(1, 1, figsize=figsize)
    ax.imshow(image_rgb)
    
    # Assign different colors to different labels
    colors = ['red', 'blue', 'green', 'orange', 'purple', 'brown', 'pink', 'gray', 'cyan', 'magenta']
    
    for i, box_info in enumerate(grounding_boxes):
        # Get bounding box coordinates
        bbox = box_info.get('bbox_2d', [])
        if len(bbox) != 4:
            print(f"Warning: Bounding box {i} has invalid format, skipping")
            continue
            
        x1, y1, x2, y2 = bbox
        
        # If normalized coordinates (0-1000), convert to absolute pixel coordinates
        if not abs_coords:
            x1 = x1 * img_width / 1000
            y1 = y1 * img_height / 1000
            x2 = x2 * img_width / 1000
            y2 = y2 * img_height / 1000
            
        # Ensure coordinates are within valid range and convert to integers
        x1 = max(0, min(int(x1), img_width - 1))
        y1 = max(0, min(int(y1), img_height - 1))
        x2 = max(0, min(int(x2), img_width))
        y2 = max(0, min(int(y2), img_height))
        
        # Ensure bounding box is valid
        if x2 <= x1 or y2 <= y1:
            continue
        
        # Select color
        color = colors[i % len(colors)]
        
        # Draw bounding box
        rect = patches.Rectangle((x1, y1), x2-x1, y2-y1, 
                               linewidth=4, edgecolor=color, facecolor='none', 
                               linestyle='-', alpha=1.0)
        ax.add_patch(rect)
        
        # Prepare label text
        label_text = box_info.get('label', f'Object {i+1}')
        
        # Draw label text
        if label_text:
            ax.text(x1, y1-5, label_text, 
                   fontsize=font_size, color='white', 
                   bbox=dict(boxstyle="round,pad=0.3", facecolor=color, alpha=0.8))
    
    ax.set_title('Grounding Boxes', fontsize=14)
    ax.axis('off')
    plt.tight_layout()
    
    # Create output directory
    os.makedirs(os.path.dirname(output_path), exist_ok=True)
    
    # Save image
    plt.savefig(output_path, bbox_inches='tight', dpi=150, facecolor='white')
    plt.close()
    
    return output_path


def generate_visualization_path(original_image_path: str, output_dir: str = "visualizations") -> str:
    """
    Generate visualization output path based on original image path.
    
    Args:
        original_image_path: Path to original image
        output_dir: Output directory
        
    Returns:
        str: Generated visualization path
    """
    # Get original image filename
    basename = os.path.basename(original_image_path)
    name, ext = os.path.splitext(basename)
    
    # Generate new filename
    vis_filename = f"{name}_grounded{ext}"
    
    # Generate full path
    vis_path = os.path.join(output_dir, vis_filename)
    
    return vis_path
