''' 
Date: 2023-01-31 22:23:17
LastEditTime: 2023-04-03 19:00:37
Description: 
    Copyright (c) 2022-2023 Safebench Team

    This file is modified from <https://github.com/carla-simulator/scenario_runner/tree/master/srunner/tools>
    Copyright (c) 2018-2020 Intel Corporation

    This work is licensed under the terms of the MIT license.
    For a copy, see <https://opensource.org/licenses/MIT>
'''

from collections import deque
import math
import numpy as np
import carla
from agents.tools.misc import get_speed


class VehiclePIDController():
    """
        VehiclePIDController is the combination of two PID controllers
        (lateral and longitudinal) to perform the low level control a vehicle from client side
    """

    def __init__(self, vehicle, args_lateral, args_longitudinal, offset=0, max_throttle=0.75, max_brake=0.3, max_steering=0.8):
        """
            :param vehicle: actor to apply to local planner logic onto
            :param args_lateral: dictionary of arguments to set the lateral PID controller using the following semantics:
                K_P -- Proportional term
                K_D -- Differential term
                K_I -- Integral term
            :param args_longitudinal: dictionary of arguments to set the longitudinal PID controller using the following semantics:
                K_P -- Proportional term
                K_D -- Differential term
                K_I -- Integral term
            :param offset: If different than zero, the vehicle will drive displaced from the center line.
            Positive values imply a right offset while negative ones mean a left one. Numbers high enough
            to cause the vehicle to drive through other lanes might break the controller.
        """

        self.max_brake = max_brake
        self.max_throt = max_throttle
        self.max_steer = max_steering

        self._vehicle = vehicle
        self._world = self._vehicle.get_world()
        self.past_steering = self._vehicle.get_control().steer
        self._lon_controller = PIDLongitudinalController(self._vehicle, **args_longitudinal)
        self._lat_controller = PIDLateralController(self._vehicle, offset, **args_lateral)

    def run_step(self, target_speed, transform):
        """
            Execute one step of control invoking both lateral and longitudinal PID controllers to reach a target waypoint at a given target_speed.
                :param target_speed: desired vehicle speed
                :param waypoint: target location encoded as a waypoint
                :return: distance (in meters) to the waypoint
        """

        acceleration = self._lon_controller.run_step(target_speed)
        current_steering = self._lat_controller.run_step(transform)
        control = carla.VehicleControl()
        if acceleration >= 0.0:
            control.throttle = min(acceleration, self.max_throt)
            control.brake = 0.0
        else:
            control.throttle = 0.0
            control.brake = min(abs(acceleration), self.max_brake)

        # Steering regulation: changes cannot happen abruptly, can't steer too much.
        if current_steering > self.past_steering + 0.1:
            current_steering = self.past_steering + 0.1
        elif current_steering < self.past_steering - 0.1:
            current_steering = self.past_steering - 0.1

        if current_steering >= 0:
            steering = min(self.max_steer, current_steering)
        else:
            steering = max(-self.max_steer, current_steering)

        control.steer = steering
        control.hand_brake = False
        control.manual_gear_shift = False
        self.past_steering = steering

        return control

    def change_longitudinal_PID(self, args_longitudinal):
        """Changes the parameters of the PIDLongitudinalController"""
        self._lon_controller.change_parameters(**args_longitudinal)

    def change_lateral_PID(self, args_lateral):
        """Changes the parameters of the PIDLongitudinalController"""
        self._lon_controller.change_parameters(**args_lateral)


class PIDLongitudinalController():
    """
        PIDLongitudinalController implements longitudinal control using a PID.
    """

    def __init__(self, vehicle, K_P=1.0, K_I=0.0, K_D=0.0, dt=0.03):
        """
            Constructor method.
                :param vehicle: actor to apply to local planner logic onto
                :param K_P: Proportional term
                :param K_D: Differential term
                :param K_I: Integral term
                :param dt: time differential in seconds
        """
        self._vehicle = vehicle
        self._k_p = K_P
        self._k_i = K_I
        self._k_d = K_D
        self._dt = dt
        self._error_buffer = deque(maxlen=10)

    def run_step(self, target_speed):
        """
            Execute one step of longitudinal control to reach a given target speed.
                :param target_speed: target speed in Km/h
                :param debug: boolean for debugging
                :return: throttle control
        """
        current_speed = get_speed(self._vehicle)
        return self._pid_control(target_speed, current_speed)

    def _pid_control(self, target_speed, current_speed):
        """
            Estimate the throttle/brake of the vehicle based on the PID equations

                :param target_speed:  target speed in Km/h
                :param current_speed: current speed of the vehicle in Km/h
                :return: throttle/brake control
        """
        error = target_speed - current_speed
        self._error_buffer.append(error)

        if len(self._error_buffer) >= 2:
            _de = (self._error_buffer[-1] - self._error_buffer[-2]) / self._dt
            _ie = sum(self._error_buffer) * self._dt
        else:
            _de = 0.0
            _ie = 0.0

        return np.clip((self._k_p * error) + (self._k_d * _de) + (self._k_i * _ie), -1.0, 1.0)

    def change_parameters(self, K_P, K_I, K_D, dt):
        self._k_p = K_P
        self._k_i = K_I
        self._k_d = K_D
        self._dt = dt


class PIDLateralController():
    """
        PIDLateralController implements lateral control using a PID.
    """

    def __init__(self, vehicle, offset=0, K_P=1.0, K_I=0.0, K_D=0.0, dt=0.03):
        """
            Constructor method.

                :param vehicle: actor to apply to local planner logic onto
                :param offset: distance to the center line. If might cause issues if the value
                    is large enough to make the vehicle invade other lanes.
                :param K_P: Proportional term
                :param K_D: Differential term
                :param K_I: Integral term
                :param dt: time differential in seconds
        """
        self._vehicle = vehicle
        self._k_p = K_P
        self._k_i = K_I
        self._k_d = K_D
        self._dt = dt
        self._offset = offset
        self._e_buffer = deque(maxlen=10)

    def run_step(self, transform):
        """
            Execute one step of lateral control to steer
            the vehicle towards a certain waypoin.

                :param transform: target waypoint
                :return: steering control in the range [-1, 1] where:
                -1 maximum steering to left
                +1 maximum steering to right
        """
        return self._pid_control(transform, self._vehicle.get_transform())

    def _pid_control(self, transform, vehicle_transform):
        """
            Estimate the steering angle of the vehicle based on the PID equations

                :param transform: target waypoint
                :param vehicle_transform: current transform of the vehicle
                :return: steering control in the range [-1, 1]
        """
        # Get the ego's location and forward vector
        ego_loc = vehicle_transform.location
        v_vec = vehicle_transform.get_forward_vector()
        v_vec = np.array([v_vec.x, v_vec.y, 0.0])

        # Get the vector vehicle-target_wp
        if self._offset != 0:
            # Displace the wp to the side
            w_tran = transform
            r_vec = w_tran.get_right_vector()
            w_loc = w_tran.location + carla.Location(x=self._offset*r_vec.x, y=self._offset*r_vec.y)
        else:
            w_loc = transform.location

        w_vec = np.array([w_loc.x - ego_loc.x, w_loc.y - ego_loc.y, 0.0])
        wv_linalg = np.linalg.norm(w_vec) * np.linalg.norm(v_vec)
        if wv_linalg == 0:
            _dot = 1
        else:
            _dot = math.acos(np.clip(np.dot(w_vec, v_vec) / (wv_linalg), -1.0, 1.0))
        _cross = np.cross(v_vec, w_vec)
        if _cross[2] < 0:
            _dot *= -1.0

        self._e_buffer.append(_dot)
        if len(self._e_buffer) >= 2:
            _de = (self._e_buffer[-1] - self._e_buffer[-2]) / self._dt
            _ie = sum(self._e_buffer) * self._dt
        else:
            _de = 0.0
            _ie = 0.0

        return np.clip((self._k_p * _dot) + (self._k_d * _de) + (self._k_i * _ie), -1.0, 1.0)

    def change_parameters(self, K_P, K_I, K_D, dt):
        self._k_p = K_P
        self._k_i = K_I
        self._k_d = K_D
        self._dt = dt
