# viz/qualitative.py
"""
Qualitative visualization utilities: create montages, annotate images with metrics.
"""
from PIL import Image, ImageDraw, ImageFont
import numpy as np
import os

def annotate_image(img_np, text, font_size=14):
    """
    img_np: [H,W,3] float in [0,1]
    """
    img = Image.fromarray((img_np * 255).astype('uint8'))
    draw = ImageDraw.Draw(img)
    try:
        font = ImageFont.truetype("DejaVuSans.ttf", font_size)
    except Exception:
        font = ImageFont.load_default()
    draw.text((5,5), text, fill=(255,255,255), font=font)
    return np.array(img).astype(np.float32)/255.0

def make_montage(image_list, captions, outpath="viz/montage.png", thumb=(256,256)):
    """
    image_list: list of numpy arrays [H,W,3] or PIL Images
    captions: list of strings
    """
    imgs = []
    for im in image_list:
        if isinstance(im, np.ndarray):
            pil = Image.fromarray((im*255).astype('uint8'))
        else:
            pil = im
        pil = pil.resize(thumb)
        imgs.append(pil)
    # grid layout
    cols = min(4, len(imgs))
    rows = (len(imgs)+cols-1)//cols
    W = cols * thumb[0]
    H = rows * thumb[1] + 20 * rows
    grid = Image.new('RGB', (W, H), color=(0,0,0))
    for idx, im in enumerate(imgs):
        x = (idx % cols) * thumb[0]
        y = (idx // cols) * thumb[1]
        grid.paste(im, (x, y))
    os.makedirs(os.path.dirname(outpath) or ".", exist_ok=True)
    grid.save(outpath)
    print(f"Saved montage to {outpath}")
