import os
import cv2
import argparse
import glob
from tqdm import tqdm
def main():
    parser = argparse.ArgumentParser()
    parser.add_argument("--data_root", type=str, required=True, help="data_root")
    parser.add_argument("--fps", type=int, default=30, help="fps")
    args = parser.parse_args()

    frame_dir = os.path.join(args.data_root, "frame_cam00")
    assert os.path.isdir(frame_dir), f"fail to find {frame_dir}"

    img_paths = sorted(
        glob.glob(os.path.join(frame_dir, "frame_*.png")),
        key=lambda x: int(os.path.splitext(os.path.basename(x))[0].split("_")[1])
    )
    assert img_paths, f"fail to read {frame_dir}/frame_xxx.png"

    first_img = cv2.imread(img_paths[0], cv2.IMREAD_UNCHANGED)
    if first_img is None:
        raise ValueError(f"fail to read {img_paths[0]}")
    height, width = first_img.shape[:2]

    out_path = os.path.join(args.data_root, "smoke_white_bg.mp4")
    fourcc = cv2.VideoWriter_fourcc(*"mp4v")
    writer = cv2.VideoWriter(out_path, fourcc, args.fps, (width, height))

    for img_path in tqdm(img_paths):
        img = cv2.imread(img_path, cv2.IMREAD_UNCHANGED)
        if img is None:
            print(f"skip {img_path}")
            continue

        img_inv = 255 - img
        if img_inv.ndim == 2:  
            img_inv = cv2.cvtColor(img_inv, cv2.COLOR_GRAY2BGR)
        elif img_inv.ndim == 3:
            if img_inv.shape[2] == 4:  # RGBA/BGRA -> BGR
                img_inv = cv2.cvtColor(img_inv, cv2.COLOR_BGRA2BGR)
            elif img_inv.shape[2] == 3:
                pass
            else:
                raise ValueError(f"Unexpected channel count: {img_inv.shape}")
        else:
            raise ValueError(f"Unexpected image shape: {img_inv.shape}")
        writer.write(img_inv)

    writer.release()
    print(f"video: {out_path}")

if __name__ == "__main__":
    main()