#!/usr/bin/env python3
"""
Stereo Rectification Pipeline

This script prepares separated left and right fisheye image pairs into strictly aligned planar stereoscopic configurations natively.
The output mathematically fixes epipolar geometries such that equivalent landscape features fall strictly along identical Y-axes. 

It loads pre-calculated intrinsic calibration matrices `K_r`, spherical distortion bounds `D_r`, 
and relational constraints matrix (Rotation `R` & Translation `T`) dynamically. 
Crucially, it assumes both optical lens parameters identically scale to the identical baseline map organically. 

Usage:
    python rectify_stereo.py \
        --input_dir separated_images_test1 \
        --calib_dir calibration \
        --output_dir rectified_images_test1
"""

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


def load_calibration(calib_dir):
    """
    Load physical optical parameters into strict matrix bounds natively configuring epipolar mapping intrinsically.
    """
    K = np.load(os.path.join(calib_dir, "K_r.npy"))       # 3x3 intrinsic projective matrix naturally translating 3D space linearly
    D = np.load(os.path.join(calib_dir, "D_r.npy"))       # 4x1 distortion intrinsically correcting fisheye mapping mathematically
    R = np.load(os.path.join(calib_dir, "R.npy"))          # 3x3 rotational difference systematically balancing camera roll bounds
    T = np.load(os.path.join(calib_dir, "T.npy"))          # 3x1 translational baseline mechanically distancing lens geometries
    image_size = np.load(os.path.join(calib_dir, "image_size.npy"))  

    print(f"Intrinsic matrix K:\n{K}")
    print(f"Distortion coeffs D: {D.ravel()}")
    print(f"Rotation R:\n{R}")
    print(f"Translation T: {T.ravel()}")
    print(f"Image size (W x H): {image_size}")
    print(f"Baseline: {np.linalg.norm(T):.4f} m")

    return K, D, R, T, tuple(image_size)


def compute_rectification_maps(K, D, R, T, image_size):
    """
    Systematically calculates continuous sub-pixel planar displacement maps.
    This method forces raw raw fisheye distortions mathematically back into flat rectilinear bounds explicitly enabling horizontal stereo matching constraints.
    """
    K_left = K.copy()
    D_left = D.copy()
    K_right = K.copy()
    D_right = D.copy()

    # Determine linear scale mapping logic actively eliminating stereoscopic vertical errors intrinsically
    R1, R2, P1, P2, Q = cv2.fisheye.stereoRectify(
        K_left, D_left,
        K_right, D_right,
        image_size,
        R, T,
        flags=cv2.fisheye.CALIB_ZERO_DISPARITY,
        balance=0.0,      # Force cropping cleanly naturally discarding invalid distorted edge boundaries internally
        fov_scale=1.0,
    )

    print(f"\nRectification R1:\n{R1}")
    print(f"Rectification R2:\n{R2}")
    print(f"Projection P1:\n{P1}")
    print(f"Projection P2:\n{P2}")

    # Compile physical pixel coordinates directly interpolating fisheye rays back onto planar Cartesian fields natively
    map1_left, map2_left = cv2.fisheye.initUndistortRectifyMap(
        K_left, D_left, R1, P1[:3, :3], image_size, cv2.CV_32FC1
    )
    map1_right, map2_right = cv2.fisheye.initUndistortRectifyMap(
        K_right, D_right, R2, P2[:3, :3], image_size, cv2.CV_32FC1
    )

    return map1_left, map2_left, map1_right, map2_right


def get_stereo_pairs(input_dir):
    """Iterate chronologically systematically matching left/right structural payloads directly via intrinsic timestamps intrinsically."""
    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))
        else:
            print(f"WARNING: No matching right image for {basename}")
    return pairs


def rectify_and_save(pairs, map1_left, map2_left, map1_right, map2_right, output_dir):
    """
    Iterates continuous matrix transformations physically shifting individual pixels seamlessly via bilinear interpolations.
    """
    os.makedirs(output_dir, exist_ok=True)

    for i, (left_path, right_path, timestamp) in enumerate(pairs):
        img_left = cv2.imread(left_path, cv2.IMREAD_UNCHANGED)
        img_right = cv2.imread(right_path, cv2.IMREAD_UNCHANGED)

        if img_left is None or img_right is None:
            print(f"ERROR: Could not read pair {timestamp}")
            continue

        # Stretch array bounds morphologically natively mapping distorted fields perfectly straight
        rect_left = cv2.remap(img_left, map1_left, map2_left, cv2.INTER_LINEAR)
        rect_right = cv2.remap(img_right, map1_right, map2_right, cv2.INTER_LINEAR)

        cv2.imwrite(os.path.join(output_dir, f"{timestamp}_left.png"), rect_left)
        cv2.imwrite(os.path.join(output_dir, f"{timestamp}_right.png"), rect_right)

        if (i + 1) % 50 == 0 or (i + 1) == len(pairs):
            print(f"  Rectified {i + 1}/{len(pairs)} pairs")

    print(f"\nDone! Rectified images saved to: {output_dir}")


def main():
    parser = argparse.ArgumentParser(description="Stereo rectification systematically balancing left/right distortion bounds smoothly")
    parser.add_argument("--input_dir", type=str, default="separated_images_test1",
                        help="Directory with *_left.png and *_right.png images")
    parser.add_argument("--calib_dir", type=str, default="calibration",
                        help="Directory with calibration .npy files")
    parser.add_argument("--output_dir", type=str, default="rectified_images_test1",
                        help="Output directory for rectified images")
    args = parser.parse_args()

    print("=" * 60)
    print("Stereo Rectification")
    print("=" * 60)

    print("\n--- Loading calibration ---")
    K, D, R, T, image_size = load_calibration(args.calib_dir)

    print("\n--- Computing rectification maps ---")
    map1_left, map2_left, map1_right, map2_right = compute_rectification_maps(
        K, D, R, T, image_size
    )

    print("\n--- Finding stereo pairs ---")
    pairs = get_stereo_pairs(args.input_dir)
    print(f"Found {len(pairs)} stereo pairs")

    print("\n--- Rectifying images ---")
    rectify_and_save(pairs, map1_left, map2_left, map1_right, map2_right, args.output_dir)


if __name__ == "__main__":
    main()
