Source code for igp2.opendrive.elements.eulerspiral
# -*- coding: utf-8 -*-
import numpy as np
from scipy import special
__author__ = "Benjamin Orthen, Stefan Urban"
__copyright__ = "TUM Cyber-Physical Systems Group"
__credits__ = ["Priority Program SPP 1835 Cooperative Interacting Automobiles"]
__version__ = "1.2.0"
__maintainer__ = "Sebastian Maierhofer"
__email__ = "commonroad-i06@in.tum.de"
__status__ = "Released"
[docs]
class EulerSpiral:
""" """
def __init__(self, gamma):
self._gamma = gamma
[docs]
@staticmethod
def createFromLengthAndCurvature(length, curvStart, curvEnd):
"""Create an EulerSpiral from a given length with curveStart
and curvEnd. This is how the OpenDrive format specifies
EulerSpirals.
Args:
length: Length of EulerSpiral.
curvStart: Curvature at start of EulerSpiral.
curvEnd: Curvature at end of EulerSpiral.
Returns:
EulerSpiral: A new Clothoid.
"""
# if length is zero, assume zero curvature
if length == 0:
return EulerSpiral(0)
return EulerSpiral(1 * (curvEnd - curvStart) / length)
[docs]
def calc(self, s, x0=0, y0=0, kappa0=0, theta0=0):
"""
Args:
s:
x0: (Default value = 0)
y0: (Default value = 0)
kappa0: (Default value = 0)
theta0: (Default value = 0)
Returns:
"""
# Start
C0 = x0 + 1j * y0
if self._gamma == 0 and kappa0 == 0:
# Straight line
Cs = C0 + np.exp(1j * theta0 * s)
elif self._gamma == 0 and kappa0 != 0:
# Arc
Cs = C0 + np.exp(1j * theta0) / kappa0 * (
np.sin(kappa0 * s) + 1j * (1 - np.cos(kappa0 * s))
)
else:
# Fresnel integrals
Cs = self._calc_fresnel_integral(s, kappa0, theta0, C0)
# Tangent at each point
theta = self._gamma * s ** 2 / 2 + kappa0 * s + theta0
return (Cs.real, Cs.imag, theta)
def _calc_fresnel_integral(self, s, kappa0, theta0, C0):
Sa, Ca = special.fresnel(
(kappa0 + self._gamma * s) / np.sqrt(np.pi * np.abs(self._gamma))
)
Sb, Cb = special.fresnel(kappa0 / np.sqrt(np.pi * np.abs(self._gamma)))
# Euler Spiral
Cs1 = np.sqrt(np.pi / np.abs(self._gamma)) * np.exp(
1j * (theta0 - kappa0 ** 2 / 2 / self._gamma)
)
Cs2 = np.sign(self._gamma) * (Ca - Cb) + 1j * Sa - 1j * Sb
Cs = C0 + Cs1 * Cs2
return Cs