#!/usr/bin/env python3
"""
Visual Stereo Rectification Checker

This diagnostic script provides human-verifiable qualitative tests validating the accuracy of the mathematical fisheye undistorting and epipolar rectification logic.

It fundamentally concatenates isolated left and right lens images side-by-side, projecting continuous colored horizontal epipolar lines explicitly across both planes. 
If the preceding rectification parameters (`rectify_stereo.py`) are mathematically sound, identical semantic features (e.g. craters, rocks, horizon bounds) 
must be perfectly bisected by identical corresponding horizontal lines perfectly matching their Y-axis locations natively.

It also intelligently generates a red-cyan Anaglyph merging the structural arrays directly on top of each other organically, 
allowing rapid visual testing for stereo baseline separation errors. 
"""

import os
import glob
import argparse
import random
import numpy as np
import cv2


def get_stereo_pairs(input_dir):
    """
    Scans physical system directories sequentially grouping matching arrays exactly by their chronological timestamp structures implicitly.
    """
    left_files = sorted(glob.glob(os.path.join(input_dir, "*_left.png")))
    pairs = []
    for left_path in left_files:
        basename = os.path.basename(left_path)
        timestamp = basename.replace("_left.png", "")
        right_path = os.path.join(input_dir, f"{timestamp}_right.png")
        if os.path.exists(right_path):
            pairs.append((left_path, right_path, timestamp))
    return pairs


def create_epipolar_visualization(left_path, right_path, line_spacing=40, line_alpha=0.4):
    """
    Calculates unified horizontal scaling, expanding dual image limits functionally into one contiguous field. 
    It draws perfectly straight raster lines directly through identical pixel Y-coordinates systematically confirming 
    epipolar alignment mathematically natively. 
    """
    img_left = cv2.imread(left_path, cv2.IMREAD_COLOR)
    img_right = cv2.imread(right_path, cv2.IMREAD_COLOR)

    if img_left is None or img_right is None:
        return None

    h, w = img_left.shape[:2]

    # Stitch bounds cleanly matching matrices side-by-side efficiently 
    canvas = np.zeros((h, w * 2, 3), dtype=np.uint8)
    canvas[:, :w] = img_left
    canvas[:, w:] = img_right

    # Demarcate internal dividing boundary solidly separating physical lens contexts 
    cv2.line(canvas, (w, 0), (w, h - 1), (255, 255, 255), 1)

    colors = [
        (0, 255, 0),
        (0, 255, 255),
        (255, 0, 255),
        (0, 165, 255),
        (255, 255, 0),
        (0, 0, 255),
        (255, 0, 0),
        (128, 255, 128),
    ]

    overlay = canvas.copy()
    # Cast continuous bounds bridging entirely across both matrix planes natively 
    for i, y in enumerate(range(0, h, line_spacing)):
        color = colors[i % len(colors)]
        cv2.line(overlay, (0, y), (w * 2 - 1, y), color, 1, cv2.LINE_AA)

    # Blend structural lines seamlessly atop raw features cleanly ensuring visual readability 
    canvas = cv2.addWeighted(overlay, line_alpha + 0.6, canvas, 1.0 - (line_alpha + 0.6), 0)
    result = canvas.copy()

    cv2.putText(result, "LEFT", (10, 30), cv2.FONT_HERSHEY_SIMPLEX, 1.0, (0, 255, 0), 2, cv2.LINE_AA)
    cv2.putText(result, "RIGHT", (w + 10, 30), cv2.FONT_HERSHEY_SIMPLEX, 1.0, (0, 255, 0), 2, cv2.LINE_AA)

    return result


def create_overlay_visualization(left_path, right_path):
    """
    Renders pure chrominance maps directly overriding intrinsic color fields purposefully separating baseline constraints intuitively.
    This Anaglyph merges left/red and right/cyan logically enabling rapid diagnostics of structural horizontal offset deviations seamlessly.
    """
    img_left = cv2.imread(left_path, cv2.IMREAD_GRAYSCALE)
    img_right = cv2.imread(right_path, cv2.IMREAD_GRAYSCALE)

    if img_left is None or img_right is None:
        return None

    # Red-cyan anaglyph naturally separating lens views physically across human optics organically
    h, w = img_left.shape
    anaglyph = np.zeros((h, w, 3), dtype=np.uint8)
    anaglyph[:, :, 2] = img_left   # Red channel bound rigorously mapping exclusively against left lens optics 
    anaglyph[:, :, 1] = img_right  # Green/Blue combined channels binding seamlessly against right lens offsets 
    anaglyph[:, :, 0] = img_right  

    cv2.putText(anaglyph, "RED=LEFT  CYAN=RIGHT", (10, 30),
                cv2.FONT_HERSHEY_SIMPLEX, 0.8, (255, 255, 255), 2, cv2.LINE_AA)

    return anaglyph


def main():
    parser = argparse.ArgumentParser(description="Visually compute structural stereo validity bounds bridging dual lens frames reliably")
    parser.add_argument("--input_dir", type=str, required=True,
                        help="Directory with rectified *_left.png and *_right.png")
    parser.add_argument("--output_dir", type=str, required=True,
                        help="Output directory for visualizations")
    parser.add_argument("--num_pairs", type=int, default=5,
                        help="Number of pairs to visualize (evenly spread randomly extracting structural distributions)")
    parser.add_argument("--line_spacing", type=int, default=40,
                        help="Pixel spacing between epipolar lines systematically mapping physical depths")
    parser.add_argument("--timestamps", type=str, nargs="+", default=None,
                        help="Specific timestamps to visualize explicitly isolating problematic physical pairings")
    args = parser.parse_args()

    os.makedirs(args.output_dir, exist_ok=True)

    pairs = get_stereo_pairs(args.input_dir)
    print(f"Found {len(pairs)} stereo pairs in {args.input_dir}")

    # Explicitly calculate dynamic array leaps bridging the entirety of chronological sequences intelligently pulling representative samples
    if args.timestamps:
        selected = [(l, r, ts) for l, r, ts in pairs if ts in args.timestamps]
    else:
        indices = np.linspace(0, len(pairs) - 1, args.num_pairs, dtype=int)
        selected = [pairs[i] for i in indices]

    print(f"Visualizing {len(selected)} pairs\n")

    for left_path, right_path, timestamp in selected:
        print(f"  Processing {timestamp}...")

        # Plot explicit epipolar lines matching mathematical matrices perfectly 
        epipolar_img = create_epipolar_visualization(
            left_path, right_path, line_spacing=args.line_spacing
        )
        if epipolar_img is not None:
            out_path = os.path.join(args.output_dir, f"{timestamp}_epipolar.png")
            cv2.imwrite(out_path, epipolar_img)

        # Merge matrices directly overlaying boundaries intuitively locating scale flaws 
        anaglyph_img = create_overlay_visualization(left_path, right_path)
        if anaglyph_img is not None:
            out_path = os.path.join(args.output_dir, f"{timestamp}_anaglyph.png")
            cv2.imwrite(out_path, anaglyph_img)

    print(f"\nDone! Visualizations saved to {args.output_dir}")


if __name__ == "__main__":
    main()
