import json
import os
import numpy as np
import sys
sys.path.append(os.path.abspath(os.path.join(os.path.dirname(__file__), '..')))

from utils import load_calibration, load_novatel_data, convert_novatel_to_pose, hexbin_plot, read_pcl, compute_alphashape


dataset_root = '/path/to/cadcd'
out_dir = '/your/out/dir/maps'
os.makedirs(out_dir, exist_ok=True)
os.makedirs(out_dir.replace('maps', 'bounds'), exist_ok=True)
os.makedirs(out_dir.replace('maps', 'maps_with_bounds'), exist_ok=True)

for date in os.listdir(dataset_root):
    # Load calibration matrices
    # (N.B. in CADC notation, T_A_B is the transformation from B to A, in our notation it's from A to B)
    extrinsics = load_calibration(os.path.join(dataset_root, date, 'calib'))['extrinsics']
    T_IMU_LIDAR = np.array(extrinsics['T_LIDAR_GPSIMU'])

    for scene_id in os.listdir(os.path.join(dataset_root, date)):
        if scene_id == 'calib':
            continue

        # Load inertial measurements of all frames in the scene
        novatel = load_novatel_data(os.path.join(dataset_root, date, scene_id, 'labeled/novatel/data/'))
        T_IMU_ENU_list = convert_novatel_to_pose(novatel)
        T_IMU0_ENU = np.array(T_IMU_ENU_list[0])
        yaw_0 = np.deg2rad(-1.0 * float(novatel[0][9]))

        pointclouds = []
        # Iterate over LiDAR swipes in the scene
        for frame_num, lidar_file in enumerate(sorted(os.listdir(os.path.join(dataset_root, date, scene_id, 'labeled/lidar_points/data/')))):
            # Read point cloud
            pcl = read_pcl(os.path.join(dataset_root, date, scene_id, 'labeled/lidar_points/data/', lidar_file))
            pcl = np.append(pcl, np.ones((1, pcl.shape[1])), axis=0)
            
            # Transform from LiDAR_t to LiDAR_0
            T_IMUt_ENU = np.array(T_IMU_ENU_list[frame_num])
            T_LIDARt_LIDAR0 = T_IMU_LIDAR @ np.linalg.inv(T_IMU0_ENU) @ T_IMUt_ENU @ np.linalg.inv(T_IMU_LIDAR)
            pcl = T_LIDARt_LIDAR0 @ pcl
            pcl = pcl[:3, :]
            pointclouds.append(pcl)
        
        # Combine and save point clouds
        pointclouds = np.hstack(pointclouds)
        # concave_hull = compute_alphashape(pointclouds, save_path=os.path.join(out_dir.replace('maps', 'bounds'), f'{date}_{scene_id}.pkl'))
        # hexbin_plot(pointclouds, save_path_png=os.path.join(out_dir.replace('maps', 'maps_with_bounds'), f'{date}_{scene_id}.png'), shape=concave_hull)
        hexbin_plot(pointclouds, save_path_png=os.path.join(out_dir, f'{date}_{scene_id}.png'))
        print(f'{date}_{scene_id}.png')
        pointclouds = []
        