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

from tqdm import tqdm

from utils import read_pcl, hexbin_plot, read_calib_file, read_gps_imu_file, imut_to_imu0, compute_alphashape

data_root = '/path/to/KITTI/training/velodyne'
out_root = '/your/out/dir'

os.makedirs(f'{out_root}/maps/', exist_ok=True)
os.makedirs(f'{out_root}/bounds/', exist_ok=True)
os.makedirs(f'{out_root}/maps_with_bounds/', exist_ok=True)

for sequence_id in tqdm(sorted(os.listdir(data_root))):
    
    # Read camera extrinsics
    T_velo_cam, T_imu_velo = read_calib_file(os.path.join(data_root, f'{sequence_id}.txt').replace('velodyne', 'calib_02'))

    imu_data = read_gps_imu_file(os.path.join(data_root, f'{sequence_id}.txt').replace('velodyne', 'oxts_02'))
    lat_0 = imu_data['lat'].iloc[[0]].item()
    lon_0 = imu_data['lon'].iloc[[0]].item()
    alt_0 = imu_data['alt'].iloc[[0]].item()
    roll_0 = imu_data['roll'].iloc[[0]].item()
    pitch_0 = imu_data['pitch'].iloc[[0]].item()
    yaw_0 = imu_data['yaw'].iloc[[0]].item()
    
    pointclouds = []
    for filename in sorted(os.listdir(os.path.join(data_root, sequence_id))):
        file_path = os.path.join(data_root, sequence_id, filename)
        pcl = read_pcl(file_path)
        # pcl = pcl[:, ::100]
        
        frame_num = int(filename.split('.')[0])
        lat_t = imu_data['lat'].iloc[[frame_num]].item()
        lon_t = imu_data['lon'].iloc[[frame_num]].item()
        alt_t = imu_data['alt'].iloc[[frame_num]].item()
        roll_t = imu_data['roll'].iloc[[frame_num]].item()
        pitch_t = imu_data['pitch'].iloc[[frame_num]].item()
        yaw_t = imu_data['yaw'].iloc[[frame_num]].item()        

        # Transform points into LiDAR frame at time 0
        pcl = np.vstack((pcl, np.ones((1, pcl.shape[1]))))
        pcl = np.linalg.inv(T_imu_velo) @ pcl
        pcl = imut_to_imu0(lat_0, lon_0, alt_0, roll_0, pitch_0, yaw_0, 
                           lat_t, lon_t, alt_t, roll_t, pitch_t, yaw_t, pcl)
        pcl = T_imu_velo @ pcl
        pcl = pcl[:-1, :]

        pointclouds.append(pcl)
        
    pointclouds = np.hstack(pointclouds)
    # concave_hull = compute_alphashape(pointclouds, save_path=os.path.join(f'{out_root}/bounds/', f'{sequence_id}.pkl'))
    # hexbin_plot(pointclouds, save_path_png=f'{out_root}/maps_with_bounds/{sequence_id}.png', shape=concave_hull)
    hexbin_plot(pointclouds, save_path_png=f'{out_root}/maps/{sequence_id}.png')
    