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 pyquaternion import Quaternion
from nuscenes.nuscenes import NuScenes
from nuscenes.utils.data_classes import LidarPointCloud

from utils import compute_alphashape, hexbin_plot

data_root = '/path/to/nuscenes'
out_root = '/your/out/dir'

nusc = NuScenes(version='v1.0-trainval', dataroot=data_root, verbose=True)

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)

# Iterate over all scenes in dataset
for scene in tqdm(nusc.scene):
    
    # Read first sample token (i.e., frame) of the scene
    sample_token = scene['first_sample_token']

    # Aggregate point clouds over frames    
    points = []

    # Iterate over sample tokens (i.e., frames) in the scene
    while sample_token != '':
        sample = nusc.get('sample', sample_token)

        # Get LiDAR
        pointsensor_token = sample['data']['LIDAR_TOP']
        pointsensor = nusc.get('sample_data', pointsensor_token)
        
        # Load point cloud
        pcl_path = os.path.join(nusc.dataroot, pointsensor['filename'])
        if pointsensor['sensor_modality'] == 'lidar':
            pc = LidarPointCloud.from_file(pcl_path)
            pc.subsample(0.1)

        # Points live in the point sensor frame. So they need to be transformed via global to the image plane.
        # First step: transform the pointcloud to the ego vehicle frame for the timestamp of the sweep.
        cs_record = nusc.get('calibrated_sensor', pointsensor['calibrated_sensor_token'])
        pc.rotate(Quaternion(cs_record['rotation']).rotation_matrix)
        pc.translate(np.array(cs_record['translation']))

        # Second step: transform from ego to the global frame.
        poserecord = nusc.get('ego_pose', pointsensor['ego_pose_token'])
        pc.rotate(Quaternion(poserecord['rotation']).rotation_matrix)
        pc.translate(np.array(poserecord['translation']))

        points.append(pc.points[:3, :])

        # Move to next sample (i.e., frame)
        sample_token = sample['next']    
    
    pointclouds = np.hstack(points)
    
    concave_hull = compute_alphashape(pointclouds, save_path=os.path.join(f'{out_root}/bounds/', f'{scene["token"]}.pkl'))
    hexbin_plot(pointclouds, save_path_png=os.path.join(f'{out_root}/maps_with_bounds', f'{scene["token"]}.png'), shape=concave_hull)
