import numpy as np
import pandas as pd
import pymap3d as pm
from pymap3d.ned import ned2ecef, ecef2ned
from pymap3d.ecef import ecef2geodetic, geodetic2ecef


# Constants for WGS-84
a = 6378137.0  # Earth semi-major axis (meters)
e2 = 0.00669437999  # First eccentricity squared

def read_pcl(file_path):
    pcl = np.fromfile(file_path, dtype=np.float32).reshape(-1, 4).T
    indices = pcl[3, :] > 0  # Reflectance > 0
    pcl = pcl[:3, indices]
    return pcl

def read_gps_imu_file(file_path):
    """Read a GPS/IMU data file and return a structured Pandas DataFrame."""
    columns = [
        "lat", "lon", "alt", "roll", "pitch", "yaw", 
        "vn", "ve", "vf", "vl", "vu", 
        "ax", "ay", "az", "af", "al", "au", 
        "wx", "wy", "wz", "wf", "wl", "wu",
        "pos_accuracy", "vel_accuracy", 
        "navstat", "numsats", "posmode", "velmode", "orimode"
    ]
    
    # Read the data into a Pandas DataFrame
    data = pd.read_csv(file_path, sep=r"\s+", header=None, names=columns)
    
    return data

def read_calib_file(file_path):
    """Read a camera calibration data file and return transformation matrices LiDAR->Camera and IMU->LiDAR in homogeneous coordinates"""
    # Open the file
    with open(file_path, 'r') as file:
        lines = file.readlines()
    
    # Extract relevant lines for transformation matrices
    Tr_velo_cam_line = lines[-2].strip().split()[1:]  # Skip 'Tr_velo_cam' label and split values
    Tr_imu_velo_line = lines[-1].strip().split()[1:]  # Skip 'Tr_imu_velo' label and split values
    
    # Convert lines into 4x4 numpy arrays
    Tr_velo_cam = np.array(Tr_velo_cam_line, dtype=np.float32).reshape(3, 4)
    Tr_imu_velo = np.array(Tr_imu_velo_line, dtype=np.float32).reshape(3, 4)
    
    # Add the last row for homogeneous transformation (row of [0, 0, 0, 1])
    Tr_velo_cam = np.vstack([Tr_velo_cam, [0, 0, 0, 1]])
    Tr_imu_velo = np.vstack([Tr_imu_velo, [0, 0, 0, 1]])
    
    return Tr_velo_cam, Tr_imu_velo

def rotation_matrix(roll, pitch, yaw):
    """Compute the 3D rotation matrix from roll, pitch, yaw."""
    R_x = np.array([
        [1, 0, 0],
        [0, np.cos(roll), np.sin(roll)],
        [0, -np.sin(roll), np.cos(roll)]
    ])
    
    R_y = np.array([
        [np.cos(pitch), 0, -np.sin(pitch)],
        [0, 1, 0],
        [np.sin(pitch), 0, np.cos(pitch)]
    ])
    
    R_z = np.array([
        [np.cos(yaw), np.sin(yaw), 0],
        [-np.sin(yaw), np.cos(yaw), 0],
        [0, 0, 1]
    ])
    
    return R_z @ R_y @ R_x

def ned_to_imu(roll, pitch, yaw):
    """Compute the homogeneous transformation matrix from NED to IMU."""
    R_ned_imu_000 = np.zeros((3, 3))
    R_ned_imu_000[0, 1] = 1
    R_ned_imu_000[1, 0] = 1
    R_ned_imu_000[2, 2] = -1

    R_ned_imu = rotation_matrix(roll, pitch, yaw) @ R_ned_imu_000
    
    # Homogeneous transformation matrix
    T = np.eye(4)
    T[:3, :3] = R_ned_imu

    return T

def imu_to_ecef(lat, lon, alt, roll, pitch, yaw, points):
    points = np.linalg.inv(ned_to_imu(roll, pitch, yaw)) @ points
    n, e, d = points[0, :], points[1, :], points[2, :]
    points = ned2ecef(n, e, d, lat, lon, alt)
    points = np.vstack((np.array(points), np.ones_like(n)))

    return points

def ecef_to_imu(lat, lon, alt, roll, pitch, yaw, points):
    x, y, z = points[0, :], points[1, :], points[2, :]
    points = ecef2ned(x, y, z, lat, lon, alt)
    points = np.vstack((np.array(points), np.ones_like(x)))
    points = ned_to_imu(roll, pitch, yaw) @ points

    return points

def 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, points):
    points = imu_to_ecef(lat_t, lon_t, alt_t, roll_t, pitch_t, yaw_t, points)
    points = ecef_to_imu(lat_0, lon_0, alt_0, roll_0, pitch_0, yaw_0, points)

    return points