Skip to content

cherenkov-plenoscope/spherical_coordinates

Repository files navigation

Spherical Coordinates

TestStatus PyPiStatus BlackStyle BlackPackStyle MITLicenseBadge

A python package to transform the representations of pointings. It adopts the naming and definitions of KIT's CORSIKA. The transformations support both scalar and array-like in- and outputs, as in numpy.

img_frame

Install

pip install spherical_coordinates

Coordinates systems

The shower simulation CORSIKA has a mindset of particles running down in the atmosphere towards the surface of the earth. This is, the particles have (mostly) momentum into negative z direction and are running towards the xy plane. Because of this, the spherical coordinates used by CORSIKA point towards the negative z direction by default (for phi=0, theta=0).

On the other hand, astronomy has a mindset of looking up into the sky, into positive z direction away from the xy plane. Because of this, the spherical coordiantes used in astronomy point towards the positive z direction by default (for azimuth=0, zenith distance=0).

CORSIKA's spherical coordinates are phi-theta. They are used in e.g. the input of CORSIKA and can be defined in the steering card as PHIP and THETAP. Note in the figure how theta starts to open from the negative z axis.

On the other hand, astronomy's spherical coordinates are azimuth-zenith distance. (Astronomy has many coordinate systems but to discuss the pointing of a telescope on earth, azimuth and zenith are rather common). Note in the figure how zenith starts to open from the positive z axis.

Transformations between spherical and cartesian

Directions in spherical coordiantes azimuth-zenith have a corresponding cartesian direction vector [cx, cy, cz]. This python package is mostly about the transformations between this cartesian direction vector and the spehrical coordinates.

Az = azimuth
Zd = zenith distance

To go from spherical to cartesian:

cx = sin(Zd) cos(Az)
cy = sin(Zd) sin(Az)
cz = cos(Zd)

This implies the vector [cx, cy, cz] has length 1.

sqrt(cx^2 + cy^2 + cz^2) = 1

To go back from cartesian to spherical:

Az = arctan2(cy, cx)
Zd = arccos(cz)

Sometimes cz is omitted in case the usecase is always in the positive z hemisphere.

cz = sqrt(1.0 - cx^2 - cy^2)

CORSIKA

The relation to CORSIKA's coordinates is:

phi = Az - PI
theta = Zd

for spherical coordinates and

cx = -ux
cy = -vy
cz = -wz

for cartesian coordinates.

Relations inside of CORSIKA are:

ux = sin(theta) cos(phi)
vy = sin(theta) sin(phi)
wz = (-1.0) cos(theta)
phi = arctan2(vy, ux)
theta = (-1.0) arccos(wz)

Inside the CORSIKA manual there is only u and v. Here we rename u to ux and v to vy to make clear what dimension of the cartesian vector they corrspond to. Also we added wz for the z component.

Usage

import spherical_coordinates

cx, cy, cz = spherical_coordinates.az_zd_to_cx_cy_cz(
    azimuth_rad=0.2,
    zenith_rad=0.3,
)
print(cx, cy, cz)
0.28962947762551555 0.058710801693826566 0.955336489125606

az, zd = spherical_coordinates.cx_cy_cz_to_az_zd(cx=cx, cy=cy, cz=cz)
print(az, zd)
0.20000000000000015 0.30000000000000016

Hemisphere in positive Z

Often, CORSIKA assumes that all directions point above the x-y plane in what case it neglects the Z-component. For example this is the case in the output files for particles or Cherenkov photons where only (cx and cy) are given. To not always restore the missing cz component manually, there are transformations for this limited, but unfortunately common case. I suspect CORSIKA omitted the cz to reduce storage space. For new projects, I would not recommand this.

import spherical_coordinates

cx, cy = spherical_coordinates.az_zd_to_cx_cy(
    azimuth_rad=0.2,
    zenith_rad=0.3,
)
print(cx, cy, cz)
0.28962947762551555 0.058710801693826566

az, zd = spherical_coordinates.cx_cy_to_az_zd(cx=cx, cy=cy)
print(az, zd)
0.20000000000000015 0.30000000000000016

However, for pointings below the x-y plane this will fail:

import spherical_coordinates

zenith_below_xy_plane_rad = 2.0
cx, cy = spherical_coordinates.az_zd_to_cx_cy(
    azimuth_rad=0.2,
    zenith_rad=zenith_below_xy_plane_rad,
)
az, zd = spherical_coordinates.cx_cy_to_az_zd(cx=cx, cy=cy)
print(zd)
1.1415926535897933
# zd is now the projection into the upper hemisphere

To restore cz there is:

import spherical_coordinates

cz = spherical_coordinates.restore_cz(cx=0.1, cy=0.2)
print(cz)

what assumes the lengths is 1.0:

cz = sqrt{1.0 - cx * cx - cy * cy}

Angles Between

Quickly estimate the angle between two pointings.

import spherical_coordinates

delta = spherical_coordinates.angle_between_cx_cy_cz(
    cx1=0.5, cy1=0.5, cz1=0.7071, cx2=0.7071, cy2=0.0, cz2=0.7071
)
print(delta*180/3.14159, "DEG")
31.399818557245204 DEG

delta = spherical_coordinates.angle_between_cx_cy(
    cx1=0.5, cy1=0.5, cx2=0.7071, cy2=0.0
)
print(delta*180/3.14159, "DEG")
31.399818557245204 DEG

delta = spherical_coordinates.angle_between_az_zd(
    azimuth1_rad=1.2, zenith1_rad=0.2, azimuth2_rad=-0.5, zenith2_rad=0.7
)
print(delta*180/3.14159, "DEG")
42.852624700439804 DEG

Random

Draw random, uniform directions from within a cone.

import spherical_coordinates
import numpy as np

prng = np.random.Generator(np.random.PCG64(132))

az, zd = spherical_coordinates.random.uniform_az_zd_in_cone(
    prng=prng,
    azimuth_rad=0,
    zenith_rad=0,
    min_half_angle_rad=0,
    max_half_angle_rad=np.pi,
    size=10,
)

Azimuth Range

From my experience, some parts in CORSIKA expect the azimuth angle to be in the so called 'least absolute residue'. This is:

- PI < azimuth_rad <= +PI

For this spherical_coordinates has a range limiter:

import spherical_coordinates

az = spherical_coordinates.azimuth_range(azimuth_rad=123.4)
print(az*180/3.1415, "DEG")
-129.7046334064967 DEG

About

Transform spherical coordinates as it is done in KIT's CORSIKA

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages