Source code for pygrin.pygrin

# pylint: disable=invalid-name
# pylint: disable=too-many-arguments
"""
Gradient Index lens calculations and plots.

More documentation at <https://pygrin.readthedocs.io>

Typical usage::

    import pygin

    length = 7               # mm
    diameter = 2             # mm
    r = np.linspace(-1,1,11) # mm
    n_0 = 1.48               # refractive index at r=0
    theta_i = 0              # launch angle
    pitch = 0.25             # quarter pitch lens

    pygrin.plot_principal_planes(n_0, pitch, length, diameter)
    for r_i in r:
        z,r = pygrin.meridional_curve(n_0, pitch, length, r_i, theta_i)
        plt.plot(z,r,color='blue')
    plt.show()

Functions to locate focal points and cardinal points::

    BFL(n_0, pitch, length)
    EFL(n_0, pitch, length)
    FFL(n_0, pitch, length)
    NA(n_0, pitch, length, diameter)
    cardinal_points(n_0, pitch, length)

Functions to find properties of GRIN lens::

    gradient(pitch, length)
    period(grad, length)
    max_angle(n_0, pitch, length, diameter)
    ABCD(n_0, pitch, length, z)
    image_distance(n_0, pitch, length, s)
    image_mag(n_0, pitch, length, s)

Functions to determine refractive index profile::

    hyperbolic_secant_profile_index(n_0, alpha, r)
    parabolic_profile_index(n_0, pitch, length, r)

Functions to help raytrace through GRIN lens::

    full_meridional_curve(n_0, pitch, length, z_obj, r_obj, r_lens)
    meridional_curve(n_0, pitch, length, r_i, theta_i)
    plot_principal_planes(n_0, pitch, length, diameter)
"""

import numpy as np
import matplotlib.pyplot as plt
import matplotlib.patches as patches

__all__ = ('ABCD',
           'BFL',
           'EFL',
           'FFL',
           'NA',
           'cardinal_points',
           'full_meridional_curve',
           'gradient',
           'hyperbolic_secant_profile_index',
           'image_distance',
           'image_mag',
           'max_angle',
           'meridional_curve',
           'parabolic_profile_index',
           'period',
           'plot_principal_planes',
           )


[docs]def gradient(pitch, length): """ Gradient of a grin lens based on its pitch and length. Args: pitch: pitch or period of the lens [unitless] length: length of grin lens [mm] Returns: float: the gradient characterizing the index of refraction profile [1/mm] """ return 2 * np.pi * pitch / length
[docs]def period(grad, length): """ Period or pitch of a grin lens based on its gradient and length. Args: grad: geometric gradient of the lens [1/mm] length: length of grin lens [mm] Returns: float: the pitch or period of the grin lens [unitless] """ return length * grad / (2 * np.pi)
[docs]def parabolic_profile_index(n_0, pitch, length, r): """ Index of a parabolic grin lens at a particular radius. Args: n_0: index of refraction at center of grin lens [unitless] pitch: pitch or period of the lens [unitless] length: axial length of the lens [mm] r: distance from center of lens [mm] Returns: float: the index of a parabolic grin lens at r [unitless] """ return n_0 * (1 - 2 * (np.pi * pitch * r / length)**2)
[docs]def hyperbolic_secant_profile_index(n_0, alpha, r): """ Index of a hyperbolic secant grin lens at a particular radius. Args: n_0: index of refraction at center of grin lens [unitless] alpha: parameter (like gradient for parabolic lens) [1/mm] r: distance from center of lens [mm] Returns: float: the index of a parabolic grin lens at r [unitless] """ return np.sqrt(1 + (n_0**2 - 1.0)/np.cosh(alpha*r)**2)
[docs]def EFL(n_0, pitch, length): """ Effective focal length of a grin lens. Args: n_0: index of refraction at center of grin lens [unitless] pitch: pitch or period of the lens [unitless] length: axial length of the lens [mm] Returns: float: the effective focal length of the grin lens [mm] """ return length / np.sin(2 * np.pi * pitch) / (2 * np.pi * pitch * n_0)
[docs]def FFL(n_0, pitch, length): """ Front focal length of a grin lens. Args: n_0: index of refraction at center of grin lens [unitless] pitch: pitch or period of the lens [unitless] length: axial length of the lens [mm] Returns: float: the front focal length of the grin lens [mm] """ return -length / np.tan(2 * np.pi * pitch) / (2 * np.pi * pitch * n_0)
[docs]def BFL(n_0, pitch, length): """ Back focal length of a grin lens. Args: n_0: index of refraction at center of grin lens [unitless] pitch: pitch or period of the lens [unitless] length: axial length of the lens [mm] Returns: float: the back focal length of the grin lens [mm] """ return length + length / np.tan(2 * np.pi * pitch) / (2 * np.pi * pitch * n_0)
[docs]def max_angle(n_0, pitch, length, diameter): """ Maximum acceptance angle of a grin lens in air. Args: n_0: index of refraction at center of grin lens [unitless] pitch: pitch or period of the lens [unitless] length: axial length of the lens [mm] diameter: diameter of the lens [mm] Returns: float: the maximum acceptance angle of the lens in air [radians] """ return n_0 * np.sqrt(1 - np.cosh(diameter * pitch * np.pi / length)**-2)
[docs]def NA(n_0, pitch, length, diameter): """ Numerical aperture of a grin lens in air. Args: n_0: index of refraction at center of grin lens [unitless] pitch: pitch or period of the lens [unitless] length: axial length of the lens [mm] diameter: diameter of the lens [mm] Returns: float: the numerical aperture of the grin lens in air [unitless] """ return np.sin(max_angle(n_0, pitch, length, diameter))
[docs]def ABCD(n_0, pitch, length, z): """ ABCD matrix for meridonal ray propagation. Args: n_0: index of refraction at center of grin lens [unitless] pitch: pitch or period of the lens [unitless] length: axial length of the lens [mm] z: distance within lens from front surface [mm] Returns: float: the ABCD matrix for meridonal ray propagation [radians] """ g = gradient(pitch, length) cos = np.cos(g * z) sin = np.sin(g * z) return np.array([[cos, sin / g / n_0], [-n_0 * g * sin, cos]])
[docs]def image_distance(n_0, pitch, length, s): """ Image distance for an object. Args: n_0: index of refraction at center of grin lens [unitless] pitch: pitch or period of the lens [unitless] length: axial length of the lens [mm] s: distance from front of lens to object [mm] Returns: float: the image distance from the back of the lens [mm] """ g = gradient(pitch, length) numer = s * np.cos(2 * np.pi * pitch) - np.sin(2 * np.pi * pitch) / g / n_0 denom = n_0 * g * s * np.sin(2 * np.pi * pitch) + np.cos(2 * np.pi * pitch) return numer / denom
[docs]def image_mag(n_0, pitch, length, s): """ Transverse magnification of an object located at s. Args: n_0: index of refraction at center of grin lens [unitless] pitch: pitch or period of the lens [unitless] length: axial length of the lens [mm] s: distance from front of lens to object [mm] Returns: float: the transvers magnification [unitless] """ g = gradient(pitch, length) twopp = 2 * np.pi * pitch return 1 / (g * n_0 * s * np.sin(twopp) - np.cos(twopp))
[docs]def cardinal_points(n_0, pitch, length, offset=0): """ Cardinal points of a grin lens relative to first surface. Args: n_0: index of refraction at center of grin lens [unitless] pitch: pitch or period of the lens [unitless] length: axial length of the lens [mm] offset: float (optional) origin relative to first lens surface Returns: float: location of the front focal point [mm] float: location of the first lens surface [mm] float: location of the first principal plane [mm] float: location of the second principal plane [mm] float: location of the second lens surface [mm] float: location of the back focal point [mm] """ efl = EFL(n_0, pitch, length) ffl = FFL(n_0, pitch, length) bfl = BFL(n_0, pitch, length) FF = offset + ffl FL = offset FPP = offset + ffl + efl SPP = offset + bfl - efl SL = offset + length BF = offset + bfl return FF, FL, FPP, SPP, SL, BF
[docs]def meridional_curve(n_0, pitch, length, r_i, theta_i, npoints=40): """ Points on path of a ray passing through a grin lens. Args: n_0: index of refraction at center of grin lens [unitless] pitch: pitch or period of the lens [unitless] length: axial length of the lens [mm] r_i: radial distance that ray hits grin lens [mm] theta_i: angle of incidence [radians] npoints: integer number of points in the returned curve Returns: z: array axial points along the curve inside the grin lens [mm] r: array radial points along the curve inside the grin lens [mm] """ z = np.linspace(0, length, npoints) V = np.array([r_i, n_0 * np.cos(np.pi / 2 - theta_i)]) # there must be a better way to do this r = np.zeros(npoints) for i in range(npoints): abcd = ABCD(n_0, pitch, length, z[i]) r[i], _ = np.dot(abcd, V) return z, r
[docs]def full_meridional_curve(n_0, pitch, length, z_obj, r_obj, r_lens, npoints=40): """ Points on a path from an object to image through a GRIN lens. The light ray starts at (z_obj,r_obj) and hits the front surface of the front face of the GRIN lens at (0,r_lens). Args: n_0: index of refraction at center of grin lens [unitless] pitch: pitch or period of the lens [unitless] length: axial length of the lens [mm] z_obj: axial position of the object [mm] r_obj: radius at which the ray leaves the object [mm] r_lens: radius at which the ray hits the lens [mm] npoints: integer (optional) number of points in the returned curve Returns: z: array axial points along path from object to image [mm] r: array radial points along path from object to image [mm] """ # angle in air theta_i = np.arctan((r_obj - r_lens) / z_obj) # angle inside lens at front surface n_lens = parabolic_profile_index(n_0, pitch, length, r_lens) theta_lens = np.arcsin(np.sin(theta_i) / n_lens) z, r = meridional_curve(n_0, pitch, length, r_lens, theta_lens, npoints=npoints - 2) # insert point at top of object z = np.insert(z, 0, z_obj) r = np.insert(r, 0, r_obj) # append point at image of object z_img = image_distance(n_0, pitch, length, z_obj) mag = image_mag(n_0, pitch, length, z_obj) z = np.insert(z, -1, length + z_img) r = np.insert(r, -1, mag * r_obj) return z, r
[docs]def plot_principal_planes(n_0, pitch, length, diameter): """ Create a plot for a grin lens showing the cardinal points. Args: n_0: index of refraction at center of grin lens [unitless] pitch: pitch or period of the lens [unitless] length: axial length of the lens [mm] diameter: diameter of the lens [mm] """ FF, FL, FPP, SPP, SL, BF = cardinal_points(n_0, pitch, length) radius = diameter / 2 rect = patches.Rectangle((FL, -radius), length, diameter, lw=0, facecolor='lightgray', alpha=0.3) plt.axes().add_patch(rect) plt.plot([FL, SL], [0, 0], lw=0.5, color='black') if np.abs(FPP) < 10 * length: plt.plot([FPP, FPP], [-radius, radius], ':k') plt.annotate('H ', xy=(FPP, -radius), ha='right', fontsize=16) if np.abs(SPP) < 10 * length: plt.plot([SPP, SPP], [-radius, radius], ':k') plt.annotate(" H'", xy=(SPP, -radius), ha='left', fontsize=16) if np.abs(FF) < 10 * length: plt.scatter([FF], [0], s=50, color='black') plt.annotate('f ', xy=(FF, 0), ha='right', fontsize=16) if np.abs(BF) < 10 * length: plt.scatter([BF], [0], s=50, color='black') plt.annotate(" f'", xy=(BF, 0), ha='left', fontsize=16) plt.title(r'pitch=%.2f, n$_0$=%.3f' % (pitch, n_0))