Source code for rayoptics.parax.paraxialdesign

#!/usr/bin/env python3
# -*- coding: utf-8 -*-
# Copyright © 2018 Michael J. Hayford
""" First order paraxial design space

.. Created on Sat Mar 31 21:14:42 2018

.. codeauthor: Michael J. Hayford
"""
import math
from itertools import zip_longest
import numpy as np

from rayoptics.typing import SeqPath
import rayoptics.optical.model_constants as mc
import rayoptics.parax.firstorder as fo
from rayoptics.parax.idealimager import ideal_imager_setup
from rayoptics.parax.specsheet import SpecSheet
from rayoptics.seq.gap import Gap
from rayoptics.elem.surface import Surface

from rayoptics.util.line_intersection import get_intersect
from rayoptics.util.misc_math import normalize
from rayoptics.util.misc_math import infinity_guard, is_kinda_big
from numpy.linalg import norm


[docs] def bbox_from_poly(poly): minx, miny = np.min(poly, axis=0) maxx, maxy = np.max(poly, axis=0) return np.array([[minx, miny], [maxx, maxy]])
[docs] def calculate_slope(x, y, line_type): """ x=ybar, y=y """ if line_type == 'stop': k = x/y return k, np.array([[1, -k], [0, 1]]).T elif line_type == 'object_image': k = y/x return k, np.array([[1, 0], [-k, 1]]).T else: k = 0 return 0, np.array([[1, 0], [0, 1]])
[docs] class ParaxialModel(): def __init__(self, opt_model, opt_inv=1.0, seq_mapping=None, **kwargs): self.opt_model = opt_model self.seq_model = opt_model.seq_model self.seq_mapping = seq_mapping self.layers = {'ifcs': self} if seq_mapping is None else None self.sys = [] self.ax = [] self.pr = [] self.ztwp = [] self.opt_inv = opt_inv self.y_star = None self.ybar_star = None def __json_encode__(self): attrs = dict(vars(self)) del attrs['opt_model'] del attrs['seq_model'] del attrs['layers'] return attrs
[docs] def sync_to_restore(self, opt_model): self.opt_model = opt_model self.seq_model = opt_model.seq_model if not hasattr(self, 'seq_mapping'): self.seq_mapping = None if not hasattr(self, 'layers'): self.layers = {'ifcs': self} if not hasattr(self, 'y_star'): self.y_star, self.ybar_star = self.calc_object_and_pupil(0)
[docs] def update_model(self, **kwargs): src_model = kwargs.get('src_model', None) num_ifcs = len(self.seq_model.ifcs) if (num_ifcs > 2 and (len(self.sys) != num_ifcs or src_model is not self)): self.build_lens() if kwargs.get('build', None) != 'update': self.layers = {'ifcs': self}
[docs] def sync_to_seq(self, seq_model): self.build_lens()
[docs] def set_from_specsheet(self, specsheet): sys = self.sys ax_ray = self.ax pr_ray = self.pr n_0 = sys[0][mc.indx] n_k = sys[-1][mc.indx] thi_0 = -self.y_star*self.ybar_star/self.opt_inv yu, yu_bar = specsheet.get_parax_start_data(thi_0, n_0, n_k) ax_ray[0][mc.ht] = y0 = yu[mc.ht] ax_ray[0][mc.slp] = nu0 = n_0*yu[mc.slp] pr_ray[0][mc.ht] = y_bar0 = yu_bar[mc.ht] pr_ray[0][mc.slp] = nu_bar0 = n_0*yu_bar[mc.slp] self.opt_inv = nu_bar0*y0 - nu0*y_bar0 self.paraxial_trace() self.update_parax_to_dgms()
[docs] def build_lens(self): # rebuild the `sys` description from the seq_model path sys = self.seq_path_to_paraxial_lens(self.seq_model.path()) self.sys = sys # precalculate the reduced forms of the paraxial axial and chief rays parax_data = self.opt_model['analysis_results']['parax_data'] if parax_data is not None: ax_ray, pr_ray, fod = parax_data self.opt_inv = fod.opt_inv self.ax = [] self.pr = [] for i in range(0, len(sys)): n = sys[i][mc.indx] self.ax.append([ax_ray[i][mc.ht], n*ax_ray[i][mc.slp]]) self.pr.append([pr_ray[i][mc.ht], n*pr_ray[i][mc.slp]]) self.update_parax_to_dgms()
[docs] def parax_from_dgms(self, dgm_list, opt_inv, rndx_and_imode=None): """Construct a diagram using `dgm_list`, a list of diagram vertices. """ Z, nu_nubar = dgm_list[0], dgm_list[1] max_nodes = len(Z) delta_nodes = max_nodes - len(nu_nubar) ax = [[0., 0.] for i in range(max_nodes)] pr = [[0., 0.] for i in range(max_nodes)] sys = [[0., 0., 1., 'transmit'] for i in range(max_nodes)] if rndx_and_imode is None: rndx_and_imode = [(1., 'transmit') for i in range(max_nodes)] hsni = zip_longest(Z, nu_nubar, rndx_and_imode) for i in range(max_nodes): ht_node, slp_node, ni = next(hsni) ax[i][mc.ht] = ht_node[1] ax[i][mc.slp] = (slp_node[1] if slp_node is not None else ax[i-1][mc.slp]) pr[i][mc.ht] = ht_node[0] pr[i][mc.slp] = (slp_node[0] if slp_node is not None else pr[i-1][mc.slp]) if i < max_nodes-1: tau = np.cross(Z[i+1], Z[i])/opt_inv else: tau = 0 if i > 0 and i < max_nodes-delta_nodes-1: pwr = np.cross(nu_nubar[i], nu_nubar[i-1])/opt_inv else: pwr = 0 sys[i] = [pwr, tau, *ni] self.ax = ax self.pr = pr self.sys = sys self.opt_inv = opt_inv self.sys[0][mc.rmd] = 'dummy' self.sys[-1][mc.rmd] = 'dummy' return self
[docs] def get_num_nodes(self): return len(self.ax)
[docs] def parax_to_nodes(self, type_sel): """ render the paraxial model into a node list """ nodes = [[x[type_sel], y[type_sel]] for x, y in zip(self.pr, self.ax)] nodes = np.array(nodes) return nodes
[docs] def update_parax_to_dgms(self): """ use the ax and pr to update the diagram and obj/pupil definition. """ self.ztwp = parax_to_dgms(self.ax, self.pr, self.sys, self.opt_inv) self.y_star, self.ybar_star = self.calc_object_and_pupil(0)
[docs] def replace_node_with_dgm(self, e_node, dgm, **kwargs): """ Update the parax model from the node list, `nodes`. """ prx, dgm_pkg = dgm pm, node_idx, type_sel = prx node_list, sys_data = dgm_pkg node_list = np.array(node_list) if len(node_list.shape) == 1 or len(node_list.shape) == 3: nodes = node_list[type_sel] else: nodes = node_list self.delete_node(node_idx) for i, ns in enumerate(zip(nodes, sys_data), start=node_idx-1): node, sys = ns self.add_node(i, node, type_sel, sys) return self
[docs] def nodes_to_parax(self, node_list, type_sel): """ Update the parax model from the node list, `nodes`. """ node_list = np.array(node_list) if len(node_list.shape) == 1 or len(node_list.shape) == 3: nodes = node_list[type_sel] else: nodes = node_list if type_sel == mc.ht: for i, node in enumerate(nodes): self.apply_ht_dgm_data(i, node) elif type_sel == mc.slp: self.pr[0][mc.ht] = self.opt_model['parax_model'].ybar_star self.ax[0][mc.ht] = 0. for i, node in enumerate(nodes): self.apply_slope_dgm_data(i, node) return self
[docs] def get_pt(self, idx, type_sel=mc.ht): return [self.pr[idx][type_sel], self.ax[idx][type_sel]]
[docs] def get_pt_np(self, idx, type_sel=mc.ht): return np.array([self.pr[idx][type_sel], self.ax[idx][type_sel]])
[docs] def set_pt(self, idx, pt, type_sel=mc.ht): self.pr[idx][type_sel] = pt[0] self.ax[idx][type_sel] = pt[1]
[docs] def apply_scale_factor(self, scale_factor): """ Apply scale factor to height diagram and update parax_model. """ type_sel = mc.ht nodes = self.parax_to_nodes(type_sel) nodes = scale_factor*nodes self.opt_inv *= scale_factor self.nodes_to_parax(nodes, type_sel)
[docs] def get_gap_for_node(self, node_idx, dgm_type): if self.seq_mapping is None: # just return the node in the seq_model gap_idx = node_idx else: # use the node_defs to map to the seq_model node_defs, map_to_ifcs = self.seq_mapping kernel = node_defs[node_idx] if len(kernel) == 1: gap_idx = kernel[0] elif len(kernel) == 2: prev_gap_idx, after_gap_idx = kernel gap_idx = prev_gap_idx if dgm_type == 'ht' else after_gap_idx elif len(kernel) == 3: idx, prev_gap_idx, after_gap_idx = kernel gap_idx = idx if idx != after_gap_idx else after_gap_idx return self.seq_model.gaps[gap_idx], self.seq_model.z_dir[gap_idx]
# --- add/delete points from diagram
[docs] def add_node(self, node, new_vertex, type_sel, sys_data): """ Add a node in the paraxial data structures """ new_node = node + 1 self.sys.insert(new_node, [0.0, 0.0, *sys_data]) ax_node = [0.0, 0.0] ax_node[type_sel] = new_vertex[1] self.ax.insert(new_node, ax_node) pr_node = [0.0, 0.0] pr_node[type_sel] = new_vertex[0] self.pr.insert(new_node, pr_node) if type_sel == mc.ht: self.apply_ht_dgm_data(new_node, new_vertex=new_vertex) if self.seq_mapping is not None: ht_nodes = self.parax_to_nodes(mc.ht) self.process_seq_mapping(ht_nodes) elif type_sel == mc.slp: self.apply_slope_dgm_data(new_node, new_vertex=new_vertex) if self.seq_mapping is not None: ht_nodes = self.parax_to_nodes(mc.ht) self.process_seq_mapping(ht_nodes) if sys_data[1] == 'reflect': for i in range(new_node, len(self.sys)): self.sys[i][mc.indx] = -self.sys[i][mc.indx] return new_node
[docs] def assign_object_to_node(self, node, new_node, type_sel, factory, **inputs): """ create a new element from `factory` and replace `node` with it """ # extract optical properties of node n = self.sys[new_node][mc.indx] power = self.sys[new_node][mc.pwr] thi = n*self.sys[new_node][mc.tau] sd = abs(self.ax[new_node][mc.ht]) + abs(self.pr[new_node][mc.ht]) # create an element with the node's properties descriptor = factory(power=power, sd=sd, prx=(self, new_node, type_sel)) seq, elm, e_nodez, dgm = descriptor # insert the path sequence and elements into the # sequential and element models kwargs = {'idx': node, 't': thi, **inputs, 'src_model': self} self.opt_model.insert_ifc_gp_ele(*descriptor, **kwargs) self.compute_signed_rindx() n_before = self.sys[new_node-1][mc.indx] thi_before = n_before*self.sys[new_node-1][mc.tau] self.seq_model.gaps[new_node-1].thi = thi_before ins_offset = 0 if inputs.get('insert', False) else -1 seq_end = node + len(seq) + ins_offset n_following = self.sys[seq_end][mc.indx] thi_following = n_following*self.sys[seq_end][mc.tau] self.seq_model.gaps[seq_end].thi = thi_following return descriptor, kwargs
[docs] def compute_signed_rindx(self): """Reset the state of the refractive index array. This method resets the signs of the refractive indices so that they are negative following an odd number of reflections, but positive otherwise. """ flip = 1 for ss in self.sys: ss[mc.indx] = abs(ss[mc.indx]) if ss[mc.rmd] == 'reflect': flip = -flip if flip < 0: ss[mc.indx] = -ss[mc.indx]
[docs] def replace_node_with_seq(self, node_idx, sys_seq, pp_info): """ replaces the data at node_idx with sys_seq """ sys = self.sys ax = self.ax pr = self.pr if len(sys_seq) == 1: sys[node_idx] = sys_seq[0] else: opt_inv = self.opt_inv power, efl, fl_obj, fl_img, pp1, ppk, pp_sep, ffl, bfl = pp_info sys[node_idx-1][mc.tau] -= pp1/sys[node_idx-1][mc.indx] # sys_seq[-1][tau] = sys[node_idx][tau] - ppk/sys_seq[-1][indx] p0 = [ax[node_idx][mc.ht], pr[node_idx][mc.ht]] pn = [ax[node_idx+1][mc.ht], pr[node_idx+1][mc.ht]] slp0 = [ax[node_idx][mc.slp], pr[node_idx][mc.slp]] for n, ss in enumerate(sys_seq[:-1], start=node_idx): sys.insert(n, ss) ax_ht = ax[n-1][mc.ht] + sys[n-1][mc.tau]*ax[n-1][mc.slp] ax_slp = ax[n-1][mc.slp] - ax_ht*sys[n][mc.pwr] ax.insert(n, [ax_ht, ax_slp]) pr_ht = pr[n-1][mc.ht] + sys[n-1][mc.tau]*pr[n-1][mc.slp] pr_slp = pr[n-1][mc.slp] - pr_ht*sys[n][mc.pwr] pr.insert(n, [pr_ht, pr_slp]) # replace the original node_idx data ax[n+1][mc.slp] = slp0[0] pr[n+1][mc.slp] = slp0[1] sys[n+1][mc.pwr] = (ax[n][mc.slp]*pr[n+1][mc.slp] - ax[n+1][mc.slp]*pr[n][mc.slp])/opt_inv # sys_seq[-1][pwr] p1 = [ax[n][mc.ht], pr[n][mc.ht]] p2 = [ax[n][mc.ht]+ax[n][mc.slp], pr[n][mc.ht]+pr[n][mc.slp]] p2int = np.array(get_intersect(p1, p2, p0, pn)) ax[n+1][mc.ht] = p2int[0] pr[n+1][mc.ht] = p2int[1] sys[n][mc.tau] = ( (ax[n][mc.ht]*pr[n+1][mc.ht] - ax[n+1][mc.ht]*pr[n][mc.ht])/opt_inv) sys[n+1][mc.tau] = (p2int[0]*pn[1] - pn[0]*p2int[1])/opt_inv
[docs] def get_object_for_node(self, node_idx): ''' basic 1:1 relationship between seq and parax model sequences ''' ifc = self.seq_model.ifcs[node_idx] e_node = self.opt_model.part_tree.parent_node(ifc) prx = self, node_idx, mc.ht dgm_pkg = [self.get_pt(node_idx)], [self.sys[node_idx][mc.indx:mc.rmd+1]] dgm = prx, dgm_pkg args = [[ifc, None, None, 1, 1]], [e_node.id], e_node, dgm kwargs = {'idx': node_idx} return args, kwargs
[docs] def delete_node(self, surf): """ delete the dgm node at position surf """ del self.sys[surf] del self.ax[surf] del self.pr[surf]
# --- edit diagram points
[docs] def apply_ht_dgm_data(self, surf, new_vertex=None): """ This routine calculates all data dependent on the input height coordinates (y,ybar) at surface surf. """ sys = self.sys ax_ray = self.ax pr_ray = self.pr opt_inv = self.opt_inv ht = mc.ht tau = mc.tau slp = mc.slp pwr = mc.pwr if new_vertex is not None: pr_ray[surf][ht] = new_vertex[0] ax_ray[surf][ht] = new_vertex[1] nsm1 = len(sys) - 1 if surf == 0: surf += 1 p = surf - 1 c = surf sys[p][tau] = ((ax_ray[p][ht]*pr_ray[c][ht] - ax_ray[c][ht]*pr_ray[p][ht]) / opt_inv) if sys[p][tau] == 0: if (surf > 1): p2 = surf - 2 ax_ray[p][slp] = ax_ray[p2][slp] pr_ray[p][slp] = pr_ray[p2][slp] else: ax_ray[p][slp] = (ax_ray[c][ht] - ax_ray[p][ht])/sys[p][tau] pr_ray[p][slp] = (pr_ray[c][ht] - pr_ray[p][ht])/sys[p][tau] if (surf > 1): p2 = surf - 2 sys[p][pwr] = ((ax_ray[p2][slp]*pr_ray[p][slp] - ax_ray[p][slp]*pr_ray[p2][slp]) / opt_inv) if (surf < nsm1): s = surf + 1 sys[c][tau] = (ax_ray[c][ht]*pr_ray[s][ht] - ax_ray[s][ht]*pr_ray[c][ht])/opt_inv if sys[c][tau] == 0: pass else: ax_ray[c][slp] = (ax_ray[s][ht] - ax_ray[c][ht])/sys[c][tau] pr_ray[c][slp] = (pr_ray[s][ht] - pr_ray[c][ht])/sys[c][tau] sys[c][pwr] = (ax_ray[p][slp]*pr_ray[c][slp] - ax_ray[c][slp]*pr_ray[p][slp])/opt_inv if s < nsm1: sys[s][pwr] = (ax_ray[c][slp]*pr_ray[s][slp] - ax_ray[s][slp]*pr_ray[c][slp])/opt_inv else: ax_ray[c][slp] = ax_ray[p][slp] pr_ray[c][slp] = pr_ray[p][slp] sys[c][pwr] = 0 sys[c][tau] = 0
[docs] def apply_slope_dgm_data(self, surf, new_vertex=None): """ This routine calculates all data dependent on the input slope coordinates (nu,nubar) at surface surf. """ sys = self.sys ax_ray = self.ax pr_ray = self.pr opt_inv = self.opt_inv ht = mc.ht tau = mc.tau slp = mc.slp pwr = mc.pwr if new_vertex is not None: pr_ray[surf][slp] = new_vertex[0] ax_ray[surf][slp] = new_vertex[1] nsm1 = len(sys) - 1 if nsm1 == 0: p = 0 c = 1 ax_ray[c][ht] = ax_ray[p][slp]*sys[p][tau] + ax_ray[p][ht] pr_ray[c][ht] = pr_ray[p][slp]*sys[p][tau] + pr_ray[p][ht] elif (surf < nsm1): if (surf == 0): surf += 1 p = surf - 1 c = surf sys[c][pwr] = (ax_ray[p][slp]*pr_ray[c][slp] - ax_ray[c][slp]*pr_ray[p][slp])/opt_inv if sys[c][pwr] == 0: ax_ray[c][ht] = ax_ray[p][slp]*sys[p][tau] + ax_ray[p][ht] pr_ray[c][ht] = pr_ray[p][slp]*sys[p][tau] + pr_ray[p][ht] else: ax_ray[c][ht] = (ax_ray[p][slp] - ax_ray[c][slp])/sys[c][pwr] pr_ray[c][ht] = (pr_ray[p][slp] - pr_ray[c][slp])/sys[c][pwr] sys[p][tau] = (ax_ray[p][ht]*pr_ray[c][ht] - ax_ray[c][ht]*pr_ray[p][ht])/opt_inv
# s = surf + 1 # ax_ray[s][ht] = ax_ray[c][slp]*sys[c][tau] + ax_ray[c][ht] # pr_ray[s][ht] = pr_ray[c][slp]*sys[c][tau] + pr_ray[c][ht] # elif (surf == nsm1): # c = surf # s = surf + 1 # ax_ray[s][ht] = ax_ray[c][slp]*sys[c][tau] + ax_ray[c][ht] # pr_ray[s][ht] = pr_ray[c][slp]*sys[c][tau] + pr_ray[c][ht]
[docs] def process_seq_mapping(self, nodes): """The composite `nodes` are mapped and applied to the ifcs layer. """ node_defs, map_to_ifcs = self.seq_mapping nodes_ifcs = [] for i, kernel in enumerate(map_to_ifcs): idx, nidx, t1, t2 = kernel if t1 == 0: new_node = t2*np.array(nodes[nidx]) else: l1_pt1 = np.array(nodes[nidx]) l1_pt2 = np.array(nodes[nidx+1]) new_intersection_pt = t1*(l1_pt2 - l1_pt1) + l1_pt1 new_node = t2*new_intersection_pt nodes_ifcs.append(new_node) # print(f'nodes_ifcs {nodes_ifcs}') self.opt_model['parax_model'].nodes_to_parax(nodes_ifcs, mc.ht)
[docs] def update_composite_node_fct(self, type_sel): """ returns the appropriate 'apply' fct for the `type_sel`. """ if type_sel == mc.ht: def apply_ht_data(node, new_vertex): self.apply_ht_dgm_data(node, new_vertex) if self.seq_mapping is not None: ht_nodes = self.parax_to_nodes(mc.ht) self.process_seq_mapping(ht_nodes) return apply_ht_data elif type_sel == mc.slp: def apply_slp_data(node, new_vertex): # editing only allowed for interface layer currently self.apply_slope_dgm_data(node, new_vertex) if self.seq_mapping is not None: ht_nodes = self.parax_to_nodes(mc.ht) self.process_seq_mapping(ht_nodes) return apply_slp_data
[docs] def update_rindex(self, surf): """Update the refractive index using the `gap` at *surf*.""" gap = self.seq_model.gaps[surf] wvl = self.seq_model.central_wavelength() self.sys[surf][mc.indx] = gap.medium.rindex(wvl)
# ParaxTrace() - This routine performs a paraxial raytrace from object # (surface 0) to image. The last operation is a # transfer to the image surface.
[docs] def paraxial_trace(self): """ regenerate paraxial axial and chief rays from power and reduced distance """ sys = self.sys ax_ray = self.ax pr_ray = self.pr ht = mc.ht tau = mc.tau slp = mc.slp pwr = mc.pwr nsm1 = len(sys) - 1 # Transfer from object c = 0 s = 1 ax_ray[s][ht] = ax_ray[c][ht] + sys[c][tau]*ax_ray[c][slp] pr_ray[s][ht] = pr_ray[c][ht] + sys[c][tau]*pr_ray[c][slp] for i in range(1, len(sys)): p = c c = s # Refraction ax_ray[c][slp] = ax_ray[p][slp] - ax_ray[c][ht]*sys[c][pwr] pr_ray[c][slp] = pr_ray[p][slp] - pr_ray[c][ht]*sys[c][pwr] # Transfer if (i < nsm1): s += 1 ax_ray[s][ht] = ax_ray[c][ht] + sys[c][tau]*ax_ray[c][slp] pr_ray[s][ht] = pr_ray[c][ht] + sys[c][tau]*pr_ray[c][slp]
# --- list output
[docs] def list_model(self): """ list the paraxial axial and chief rays, and power, reduced distance """ sys = self.sys ax_ray = self.ax pr_ray = self.pr header_str = ( " ax_ht pr_ht ax_slp pr_slp" " power tau index type") print(header_str) for i, aps in enumerate(zip(ax_ray, pr_ray, sys)): ax, pr, sy = aps print(f"{i:2}: {ax[mc.ht]:12.6g} {pr[mc.ht]:12.6g} " f"{ax[mc.slp]:12.6g} {pr[mc.slp]:12.6g} " f"{sy[mc.pwr]:13.7g} {sy[mc.tau]:12.6g} " f"{sy[mc.indx]:12.5f} {sy[mc.rmd]}")
[docs] def list_lens(self): """ list the paraxial axial and chief rays, and power, reduced distance """ sys = self.sys ax_ray = self.ax pr_ray = self.pr ht = mc.ht tau = mc.tau slp = mc.slp pwr = mc.pwr indx = mc.indx rmd = mc.rmd print(" ax_ray_ht ax_ray_slp") for i in range(0, len(ax_ray)): print("{:2}: {:12.6g} {:12.6g}".format(i, ax_ray[i][ht], ax_ray[i][slp])) print("\n pr_ray_ht pr_ray_slp") for i in range(0, len(pr_ray)): print("{:2}: {:12.6g} {:12.6g}".format(i, pr_ray[i][ht], pr_ray[i][slp])) print("\n power tau index type") for i in range(0, len(sys)): print("{:2}: {:13.7g} {:12.5g} {:12.5f} {}".format( i, sys[i][pwr], sys[i][tau], sys[i][indx], sys[i][rmd]))
[docs] def list_sys_seq(self): print(" c t medium mode") fmt = "{:2}: {:12.7g} {:#12.6g} {:12.5f} {:>10s}" sys = self.sys n_before = sys[0][mc.indx] for i, sg in enumerate(self.sys): n_after = sys[i][mc.indx] thi = n_after*sys[i][mc.tau] delta_n = n_after - n_before cv = sys[i][mc.pwr] if delta_n == 0 else sys[i][mc.pwr]/delta_n print(fmt.format(i, cv, thi, n_after, sys[i][mc.rmd])) n_before = n_after
[docs] def first_order_data(self): """List out the first order imaging properties of the model.""" self.opt_model['analysis_results']['parax_data'].fod.list_first_order_data()
# --- convert to/from sequential model
[docs] def seq_path_to_paraxial_lens(self, path: SeqPath): """ returns lists of power, reduced thickness, signed index and refract mode """ sys = [] for i, sg in enumerate(path): ifc, gap, _, rndx, z_dir = sg imode = ifc.interact_mode power = ifc.optical_power if gap: n_after = rndx if z_dir > 0 else -rndx tau = gap.thi/n_after sys.append([power, tau, n_after, imode]) else: sys.append([power, 0.0, sys[-1][mc.indx], imode]) return sys
[docs] def paraxial_lens_to_seq_model(self): """ Applies a paraxial lens spec (power, reduced distance) to the model """ sys = self.sys ax_ray = self.ax n_before = sys[0][mc.indx] slp_before = self.ax[0][mc.slp] for i, sg in enumerate(self.seq_model.path()): ifc, gap, _, _, _ = sg if gap: n_after = sys[i][mc.indx] slp_after = ax_ray[i][mc.slp] gap.thi = n_after*infinity_guard(sys[i][mc.tau]) ifc.set_optical_power(sys[i][mc.pwr], n_before, n_after) ifc.from_first_order(slp_before, slp_after, ax_ray[i][mc.ht]) n_before = n_after slp_before = slp_after
[docs] def lens_from_dgm(self, idx: int, bending=0., th=None, sd=1., **kwargs): """ Return single lens constructional parameters from parax_model. This method uses a method for thickening a lens element developed by Lopez-Lopez in his `PhD dissertation <https://repository.arizona.edu/handle/10150/582195>`_, *The application of the Delano* |ybar| *diagram to optical design*. The variables are named following the notation in the thesis. He used `z` for the |ybar| coordinates; the reduced slopes and system parameters include a (inverse) factor of the optical invariant, and are capitalized here: T, W, Pwr """ from opticalglass.modelglass import ModelGlass # type: ignore if 'med' in kwargs: med = kwargs['med'] if med is None: med = ModelGlass(1.517, 64.2, '517642') else: from rayoptics.seq.medium import decode_medium med = decode_medium(med) else: med = ModelGlass(1.517, 64.2, '517642') nom_wvl = kwargs.get('nom_wvl', 'd') rndx = med.rindex(nom_wvl) opt_inv = self.opt_inv z0 = np.array(self.get_pt(idx-1, type_sel=mc.ht)) z1 = np.array(self.get_pt(idx, type_sel=mc.ht)) z2 = np.array(self.get_pt(idx+1, type_sel=mc.ht)) To = np.cross(z1, z0) Wo = (z1 - z0)/To Ti = np.cross(z2, z1) Wi = (z2 - z1)/Ti Pwr = np.cross(Wi, Wo) # print(f"To={To:8.4f}, Ti={Ti:8.4f}, opt_inv={opt_inv:8.4f}") # print(f"Pwr={Pwr:8.6f}, efl={1/(Pwr*opt_inv):8.4f}") zf = -Wi/Pwr zfp = Wo/Pwr # print(f"zf={zf}, zfp={zfp}") sd = kwargs.get('sd', norm(z1)) th = kwargs.get('th', None) if th is None: th = sd/5 X = bending T_lens = (th/rndx)*opt_inv p = T_lens*Pwr Q = 1 + p*(X**2 - 1) # brkt = 1/(1 + np.sqrt(Q)) # print(f"X={X:8.4f}, T_lens={T_lens:8.4f}, p={p:8.4f}, Q={Q:8.4f}, brkt={brkt:8.4f}") FWo = (X + 1)/(1 + np.sqrt(Q)) FWi = -(X - 1)/(1 + np.sqrt(Q)) FZo = (X + np.sqrt(Q))/(X + 1) FZi = (X - np.sqrt(Q))/(X - 1) # print(f"FWo={FWo:8.4f}, FWi={FWi:8.4f}, " # f"FZo={FZo:8.4f}, FZi={FZi:8.4f}") Wx = FWo*Wi + FWi*Wo z10 = zf + FZo*zfp z11 = FZi*zf + zfp P10 = FWo*Pwr P11 = FWi*Pwr # To10 = np.cross(z10, z0) # T11i = np.cross(z2, z11) # pp1 = (To - To10)/opt_inv # ppk = (Ti - T11i)/opt_inv # print(f"z0={z0}, z10={z10}, z1={z1}, z11={z11}, z2={z2}") # print(f"To10={To10:8.4f}, T11i={T11i:8.4f}, " # f"pp1={pp1:8.4f}, ppk={ppk:8.4f}") cv1 = P10*opt_inv/(rndx-1) cv2 = P11*opt_inv/(1-rndx) thi = rndx*T_lens/opt_inv lens = cv1, cv2, thi, rndx, sd # print(f"cv1={cv1:8.6f}, cv2={cv2:8.6f}, thi={thi:8.4f}, " # f"rndx={rndx:6.4f}, sd={sd:8.4f}") dgm_pkg = [z10, z11], [[rndx, 'transmit'], [1.0, 'transmit']] return dgm_pkg, lens
[docs] def calc_object_and_pupil(self, idx: int): """ calculate axial intercepts of line between idx and idx+1 """ delta_ybar = self.pr[idx+1][mc.ht] - self.pr[idx][mc.ht] if delta_ybar == 0: y_star = np.inf ybar_star = self.pr[idx][mc.ht] else: k = (self.ax[idx+1][mc.ht] - self.ax[idx][mc.ht])/delta_ybar y_star = self.ax[idx][mc.ht] if k != 0: y_star -= k*self.pr[idx][mc.ht] else: y_star = self.ax[idx+1][mc.ht] ybar_star = (math.copysign(np.inf, -delta_ybar) if k == 0 else self.pr[idx][mc.ht] - self.ax[idx][mc.ht]/k) # print(f"{idx}: y_star={y_star}, ybar_star={ybar_star}, k={k}") return y_star, ybar_star
# --- power and thickness solves
[docs] def pwr_slope_solve(self, parax_ray, surf: int, slp_new: float) -> float: """ solve for power to give `parax_ray.slp_new` after `surf` """ p = parax_ray[surf-1] c = parax_ray[surf] pwr = (p[mc.slp] - slp_new)/c[mc.ht] return pwr
[docs] def pwr_ht_solve(self, parax_ray, surf: int, ht_new: float) -> float: """ solve for power to give `parax_ray.ht_new` on `surf` +1 """ sys = self.sys p = parax_ray[surf-1] c = parax_ray[surf] slp_new = (ht_new - c[mc.ht])/sys[surf][mc.tau] pwr = (p[mc.slp] - slp_new)/ht_new return pwr
[docs] def thi_ht_solve(self, parax_ray, surf: int, ht_new: float) -> float: """ solve for thi to give `parax_ray.ht_new` on `surf` +1 """ sys = self.sys c = parax_ray[surf] thi = sys[surf][mc.indx]*(ht_new - c[mc.ht])/c[mc.slp] return thi
[docs] def set_paraxial_focus(self, seq_model): """ Set the image at the paraxial image point using the final gap.""" pim = self.thi_ht_solve(self.ax, -1, 0) seq_model.gaps[-1].thi += pim
# --- calculations
[docs] def compute_principle_points_from_dgm(self, idx_os=0, idx_is=-2): """ Calculate focal points and principal points. Args: idx_os: object space gap index idx_is: image space gap index Returns: (power, efl, fl_obj, fl_img, pp1, ppk, pp_sep, ffl, bfl) - power: optical power of system - efl: effective focal length - fl_obj: object space focal length, f - fl_img: image space focal length, f' - pp1: distance from the 1st interface to the front principle plane - ppk: distance from the last interface to the rear principle plane - pp_sep: distance from the front principle plane to the rear principle plane - ffl: front focal length, distance from the 1st interface to the front focal point - bfl: back focal length, distance from the last interface to the back focal point """ pp_info = compute_principle_points_from_dgm(self.ztwp[0], self.sys, self.opt_inv, idx_os, idx_is) return pp_info
[docs] def apply_conjugate_shift(self, nodes, k, mat, line_type): sheared_nodes = self.calc_conjugate_shift(nodes, k, mat, line_type) if self.seq_mapping is not None: self.process_seq_mapping(sheared_nodes) parax_model = self.opt_model['parax_model'] parax_model.paraxial_lens_to_seq_model() self.opt_model['optical_spec'].sync_to_parax(parax_model)
[docs] def calc_conjugate_shift(self, nodes, k, mat, line_type): sys = self.sys ax = self.ax pr = self.pr opt_inv = self.opt_inv # apply the shear transformation to the original shape sheared_nodes = np.matmul(nodes, mat) # for an object shift, object and image distances change, # calculate them before revising the ray slope values if line_type == 'object_image': conj_line = np.array([0., 0.]), np.array([1., k]) # update object distance and object y and ybar sheared_obj = get_intersect(nodes[0], nodes[1], *conj_line) pr[0][mc.ht], ax[0][mc.ht] = sheared_obj[0], 0. sys[0][mc.tau] = ((-sheared_nodes[1][1]*sheared_obj[0]) / opt_inv) # update image distance and image y and ybar sheared_img = get_intersect(nodes[-2], nodes[-1], *conj_line) pr[-1][mc.ht], ax[-1][mc.ht] = sheared_img[0], 0. sys[-2][mc.tau] = ((sheared_nodes[-2][1]*sheared_img[0]) / opt_inv) else: # pupil shift, update object values here pr[0][mc.ht], ax[0][mc.ht] = sheared_nodes[0] # update height and slope values over the diagram for i, sheared_node in enumerate(sheared_nodes[1:-1], start=1): pr[i][mc.ht], ax[i][mc.ht] = sheared_node if sys[i-1][mc.tau] == 0: if i>1: ax[i-1][mc.slp] = ( ax[i-2][mc.slp] - ax[i-1][mc.ht]*sys[i-1][mc.pwr]) pr[i-1][mc.slp] = ( pr[i-2][mc.slp] - pr[i-1][mc.ht]*sys[i-1][mc.pwr]) else: ax[i-1][mc.slp] = ax[i-1][mc.slp] pr[i-1][mc.slp] = pr[i-1][mc.slp] else: pr[i-1][mc.slp] = ( (pr[i][mc.ht] - pr[i-1][mc.ht]) / sys[i-1][mc.tau]) ax[i-1][mc.slp] = ( (ax[i][mc.ht] - ax[i-1][mc.ht]) / sys[i-1][mc.tau]) # update final slope values pr[i][mc.slp] = (pr[i+1][mc.ht] - pr[i][mc.ht]) / sys[i][mc.tau] ax[i][mc.slp] = (ax[i+1][mc.ht] - ax[i][mc.ht]) / sys[i][mc.tau] pr[-1][mc.slp] = pr[-2][mc.slp] ax[-1][mc.slp] = ax[-2][mc.slp] return sheared_nodes
[docs] def match_pupil_and_conj(self, prx, **kwargs): pm, node, type_sel = prx opt_inv = pm.opt_inv z0 = np.array(pm.get_pt(node-1, type_sel=mc.ht)) z1 = np.array(pm.get_pt(node, type_sel=mc.ht)) z2 = np.array(pm.get_pt(node+1, type_sel=mc.ht)) To = np.cross(z1, z0) Wo = (z1 - z0)/To pp_info = self.compute_principle_points_from_dgm() power, efl, fl_obj, fl_img, pp1, ppk, pp_sep, ffl, bfl = pp_info T_pp1 = pp1 * opt_inv To_new = To - T_pp1 z1_new = z0 + To_new * Wo T_ppk = ppk * opt_inv To_new = To - T_pp1 z1_new = z0 + To_new * Wo nodes = self.parax_to_nodes(type_sel) shear = nodes[1] - z1_new k_oi, mat_oi = calculate_slope(nodes[1][0], shear[1], 'object_image') obj_shft = self.calc_conjugate_shift(nodes, k_oi, mat_oi, 'object_image') k_s, mat_s = calculate_slope(shear[0], obj_shft[1][1], 'stop') node_list = self.calc_conjugate_shift(obj_shft, k_s, mat_s, 'stop') sys_data = [[sys_i[mc.indx], sys_i[mc.indx]] for sys_i in self.sys] dgm_pkg = node_list[1:-1], sys_data[1:-1] dgm = prx, dgm_pkg return dgm
[docs] def paraxial_vignetting(self, rel_fov=1): """Calculate the vignetting factors using paraxial optics. """ sm = self.seq_model min_vly = 1, None min_vuy = 1, None for i, ifc in enumerate(sm.ifcs[:-1]): if self.ax[i][mc.ht] != 0: max_ap = ifc.surface_od() y = self.ax[i][mc.ht] ybar = rel_fov * self.pr[i][mc.ht] ratio = (max_ap - abs(ybar))/abs(y) if ratio > 0: if ybar < 0: if ratio < min_vly[0]: min_vly = ratio, i elif ybar > 0: if ratio < min_vuy[0]: min_vuy = ratio, i else: # ybar == 0 if ratio < min_vly[0]: min_vly = ratio, i if ratio < min_vuy[0]: min_vuy = ratio, i # print(f'{i:2d}: {ratio:8.3f} {ybar:8.3f} {y:8.3f} {max_ap:8.3f}') # print("min_vly:", min_vly, "min_vuy:", min_vuy) return min_vly, min_vuy
[docs] def dgm_sketch_to_parax_model(self, Z, opt_inv, *args, **inputs): opm = self.opt_model osp = opm['optical_spec'] self.opt_inv = opt_inv # handle infinite conjugates, both object and image spaces yo_star, yobar_star = calc_object_and_pupil_from_dgm(Z, 0) Z[0] = [yobar_star, 0] lstk = len(Z)-2 yk_star, ybark_star = calc_object_and_pupil_from_dgm(Z, lstk) Z[lstk+1] = [ybark_star, 0] # generate normalized diagram from Z Z, T, W, Pwr = update_from_dgm(Z, opt_inv, osp) self.ztwp = Z, T, W, Pwr # generate the parax_model from the normalized diagram aps = dgms_to_parax(Z, T, W, Pwr, opt_inv) self.ax, self.pr, self.sys = aps # build the other models from the parax_model factory = inputs['factory'] populate_model_from_parax(opm, factory, mc.ht) osp.setup_specs_using_dgms(self) ss = opm['specsheet'] if opm['specsheet'] is None: ss = SpecSheet('unknown') specsheet_from_dgm(self, osp, ss)
[docs] def nodes_to_new_model(*args, **inputs): """ Sketch a 2d polyline and populate an opt_model from it. """ opm = inputs['opt_model'] osp = opm['optical_spec'] pm = opm['parax_model'] fig = inputs['figure'] ele_factory = inputs['factory'] dgm = fig.diagram def on_finished(poly_data, line_artist): opt_inv = opt_inv_from_dgm_and_osp(poly_data, osp) pm.dgm_sketch_to_parax_model(poly_data, opt_inv, factory=ele_factory) fig.refresh_gui(build='rebuild', src_model=pm) gui_fct = inputs['gui_fct'] gui_fct(*args, figure=fig, do_on_finished=on_finished)
[docs] def populate_model_from_parax(opt_model, factory, type_sel): pm = opt_model['parax_model'] opt_model['seq_model'].reset() opt_model['ele_model'].reset() opt_model['part_tree'].root_node.children = [] opt_model.do_init_postproc() for i in range(1, len(pm.ax)-1): # extract optical properties of node n = pm.sys[i][mc.indx] power = pm.sys[i][mc.pwr] thi = n*pm.sys[i][mc.tau] sd = abs(pm.ax[i][mc.ht]) + abs(pm.pr[i][mc.ht]) # create an element with the node's properties descriptor = factory(power=power, sd=sd, prx=(pm, i, type_sel)) seq, ele, e_node, dgm = descriptor b4_idx = i-1 if i > 0 else 0 n_before = pm.sys[b4_idx][mc.indx] thi_before = n_before*infinity_guard(pm.sys[b4_idx][mc.tau]) # insert the path sequence and elements into the # sequential and element models kwargs = {'idx': b4_idx, 't': infinity_guard(thi), 'insert': True, 'src_model': pm, 'do_update': False} opt_model.insert_ifc_gp_ele(*descriptor, **kwargs) opt_model['seq_model'].gaps[b4_idx].thi = thi_before
[docs] def create_diagram_for_key(opm, key, type_sel): seq_mapping, dgm_list = generate_mapping_for_key(opm, key) prx_model = build_from_yybar(opm, dgm_list, type_sel, seq_mapping) return key, prx_model
[docs] def update_diagram_for_key(opm, key, type_sel): seq_mapping, dgm_list = generate_mapping_for_key(opm, key) if key in opm['parax_model'].layers: prx_model = opm['parax_model'].layers[key] prx_model.seq_mapping = seq_mapping rndx_and_imode = None if key == 'ifcs': rndx_and_imode = opm['seq_model'].get_rndx_and_imode() opt_inv = opm['parax_model'].opt_inv prx_model.parax_from_dgms(dgm_list, opt_inv, rndx_and_imode) else: prx_model = build_from_yybar(opm, dgm_list, type_sel, seq_mapping) opm['parax_model'].layers[key] = prx_model return key, prx_model
[docs] def generate_mapping_for_key(opm, key): """ generate all of the mappings, ht and slp, for *key*. """ # generate the node_defs for key pt = opm['part_tree'] if key == 'eles': air_nodes = pt.nodes_with_tag('#airgap') air_gap_list = [n.id.reference_idx() for n in air_nodes] node_defs = air_gaps_to_node_defs(air_gap_list) elif key == 'asm': air_gap_list = [n.id.reference_idx() for n in pt.root_node.children if '#airgap' in n.tag] node_defs = air_gaps_to_node_defs(air_gap_list) elif key == 'sys': num_nodes = len(opm['seq_model'].ifcs) node_defs = [(0,), (0, num_nodes-2), (num_nodes-1,)] elif key == 'ifcs': nodes = [opm['parax_model'].parax_to_nodes(mc.ht), opm['parax_model'].parax_to_nodes(mc.slp)] return None, nodes # print(f'{key}:\nnode_defs_orig {node_defs}') node_defs, ht_nodes = get_valid_ht_nodes(opm['parax_model'], node_defs) # print(f'node_defs {node_defs}\nnodes {nodes}') map_to_ifcs = gen_ifcs_node_mapping(opm['parax_model'], node_defs, ht_nodes) # print(f'map_to_ifcs {map_to_ifcs}\n') slp_nodes = slp_nodes_from_node_defs(opm['parax_model'], node_defs) seq_mapping = (node_defs, map_to_ifcs) dgm_list = [np.array(ht_nodes), np.array(slp_nodes)] return seq_mapping, dgm_list
[docs] def air_gaps_to_node_defs(air_gap_list): """ generate the node defs for composite layers, based on airgaps. """ prev_gap_idx = 0 node_defs = [(prev_gap_idx,)] for gap_idx in air_gap_list[1:]: if gap_idx - prev_gap_idx < 2: node_defs.append((gap_idx,)) else: node_defs.append((prev_gap_idx, gap_idx)) prev_gap_idx = gap_idx node_defs.append((gap_idx+1,)) return node_defs
[docs] def get_valid_ht_nodes(parax_model, node_defs_in): """ given the input node defs, replace non-physical thin lenses as needed.""" node_defs = None node_defs_new = node_defs_in nodes_new = ht_nodes_from_node_defs(parax_model, node_defs_new) while node_defs != node_defs_new: node_defs = node_defs_new nodes = nodes_new node_defs_new = scan_nodes(parax_model, node_defs, nodes) nodes_new = ht_nodes_from_node_defs(parax_model, node_defs_new) return node_defs_new, nodes_new
[docs] def ht_nodes_from_node_defs(parax_model, node_defs): """ produce a list of ht nodes given the parax_model and node_defs. `node_defs` is a list of tuples, each with either one or two indices. if there is a single index, it is to a node in `parax_model`. if there are 2 indices, the first is to the gap preceding the element; the second is to the gap following the element (also the last interface of the element). The node is calculated from the intersection of the diagram edges corresponding to these gaps. There is no guarentee that the nodes calculated here represent a physically realizable system, i.e. there may be virtual airspaces. """ nodes = [] for i, kernel in enumerate(node_defs): if len(kernel) == 1: nodes.append(parax_model.get_pt(kernel[0], type_sel=mc.ht)) elif len(kernel) == 2: prev_gap_idx, after_gap_idx = kernel l1_pt1 = parax_model.get_pt_np(prev_gap_idx) l1_pt2 = parax_model.get_pt_np(prev_gap_idx+1) if np.linalg.norm(l1_pt2-l1_pt1) == 0: l1_pt2 = l1_pt1 + parax_model.get_pt_np(prev_gap_idx, type_sel=mc.slp) l2_pt1 = parax_model.get_pt_np(after_gap_idx) l2_pt2 = parax_model.get_pt_np(after_gap_idx+1) if np.linalg.norm(l2_pt2-l2_pt1) == 0: l2_pt2 = l2_pt1 + parax_model.get_pt_np(after_gap_idx, type_sel=mc.slp) new_node = get_intersect(l1_pt1, l1_pt2, l2_pt1, l2_pt2) nodes.append(new_node) elif len(kernel) == 3: idx, prev_gap_idx, after_gap_idx = kernel nodes.append(parax_model.get_pt(idx)) return nodes
[docs] def slp_nodes_from_node_defs(parax_model, node_defs): """ produce a list of slopes given the parax_model and node_defs. `node_defs` is a list of tuples, each with either one or two indices. if there is a single index, it is to a node in `parax_model`. if there are 2 indices, the first is to the gap preceding the element; the second is to the gap following the element (also the last interface of the element). The node is calculated from the intersection of the diagram edges corresponding to these gaps. There is no guarentee that the nodes calculated here represent a physically realizable system, i.e. there may be virtual airspaces. """ nodes = [] for kernel in node_defs[:-1]: if len(kernel) == 1: after_gap_idx = kernel[0] elif len(kernel) == 2: prev_gap_idx, after_gap_idx = kernel elif len(kernel) == 3: idx, prev_gap_idx, after_gap_idx = kernel nodes.append(parax_model.get_pt(after_gap_idx, type_sel=mc.slp)) return nodes
[docs] def scan_nodes(parax_model, node_defs, nodes): """scan node defs for any invalid thin elements Replace the first invalid thin element found with two 3 element tuples, signifying a thick node. The first tuple element is the index to the node in the `parax_model` and the last two elements are the range of indices in the `parax_model` covered by the thick element. Return the updated node_def list. """ new_node_defs = node_defs.copy() xprods = np.cross(nodes[:-1], nodes[1:]) for i, kn in enumerate(zip(node_defs, nodes)): kernel, node = kn if len(kernel) == 2: prev_gap_idx, after_gap_idx = kernel prev = 1 if prev_gap_idx == 0 else prev_gap_idx l1_pt1 = parax_model.get_pt(prev) # l1_pt2 = parax_model.get_pt(prev_gap_idx+1) l2_pt1 = parax_model.get_pt(after_gap_idx) xprod1 = np.cross(l1_pt1, l2_pt1) # xprodt = np.cross(l1_pt2, l2_pt1) # print(f'{i}: {prev}-{after_gap_idx}: ifc: {xprod1}, t: {xprodt}, thin: {xprods[i-1]}') if xprods[i-1] > 0 or (prev_gap_idx != 0 and 2*xprod1 > xprods[i-1]): new_node_defs[i] = (prev_gap_idx+1, prev_gap_idx, after_gap_idx) new_node_defs.insert(i+1, (after_gap_idx, prev_gap_idx, after_gap_idx)) break return new_node_defs
[docs] def build_from_yybar(opm, dgm_list, type_sel, seq_mapping): """Returns a parax_model for the input nodes and seq_mapping. """ prx_model = ParaxialModel(opm, opt_inv=opm['pm'].opt_inv, seq_mapping=seq_mapping) rndx_and_imode = None if seq_mapping is None: rndx_and_imode = opm['seq_model'].get_rndx_and_imode() prx_model.parax_from_dgms(dgm_list, prx_model.opt_inv, rndx_and_imode) return prx_model
[docs] def gen_ifcs_node_mapping(parax_model, node_defs, nodes): """Create mapping between composite diagram and interface based diagram. Each node in the composite diagram is associated with one or a range of nodes in parax_model.layer['ifcs']. `node_defs` and `nodes` define the composite diagram. `node_defs` is a list of tuples, one per composite node, of length 1, 2, or 3. The number of entries is as follows: 1) the composite node maps directly to node idx in the 'ifcs' layer 2) the composite node is generated from the previous and following gap indices 3) the composite node is part of a thick node A **thick** node is what is used when reducing a range of interfaces to a single node requires virtual propagation distances. In this case, the first and last nodes in the range are retained in the composite diagram; interior nodes are scaled according to how the thick edge is stretched. Changes in the composite diagram are propagated to the underlying 'ifcs' layer by applying a 2D stretch to the nodes in the 'ifcs' layer. The 'ifcs' node is parameterized by the calculating the intersection of the composite edge with the vector from the origin through the composite node. The scale factors are: * t1: parametric distance of the intersection point along the composite edge * t2: fractional distance of the composite node to the intersection point The map_to_ifcs list connects the edge in the composite diagram to the 'ifcs' node and the scale factors needed to update the 'ifcs' node position when the composite diagram changes. """ def calc_scale_factors(pt1, pt2, pt_k): new_node = np.array(get_intersect(pt1, pt2, np.array([0., 0.]), pt_k)) t1 = norm(new_node - pt1)/norm(pt2 - pt1) t2 = norm(pt_k)/norm(new_node) return t1, t2 map_to_ifcs = [] for i, kernel in enumerate(node_defs): if len(kernel) == 1: # single node or mirror idx = kernel[0] map_to_ifcs.append((idx, i, 0., 1.)) elif len(kernel) == 2: # thin element group # get the defining vertices from the composite diagram l1_pt1 = np.array(nodes[i-1]) l1_pt2 = np.array(nodes[i]) l2_pt1 = np.array(nodes[i]) l2_pt2 = np.array(nodes[i+1]) prev_gap_idx, after_gap_idx = kernel for k in range(prev_gap_idx+1, after_gap_idx+1): pt_k = np.array(parax_model.get_pt(k, type_sel=mc.ht)) xprod = np.cross(nodes[i], pt_k) if xprod >= 0.0: t1, t2 = calc_scale_factors(l1_pt1, l1_pt2, pt_k) map_to_ifcs.append((k, i-1, t1, t2)) else: t1, t2 = calc_scale_factors(l2_pt1, l2_pt2, pt_k) map_to_ifcs.append((k, i, t1, t2)) elif len(kernel) == 3: # thick element group # A thick element group has 2 adjacent nodes. The first and last nodes # of the thick element correspond to the first and last nodes in the # interface diagram. Interface nodes between these are scaled along # the edge separating the 2 thick nodes. map_to_ifcs.append((idx, i, 0., 1.)) idx, prev_gap_idx, after_gap_idx = kernel if idx != after_gap_idx: # get the defining vertices from the composite diagram thick_pt1 = np.array(nodes[i]) thick_pt2 = np.array(nodes[i+1]) for k in range(idx+1, after_gap_idx): pt_k = np.array(parax_model.get_pt(k, type_sel=mc.ht)) t1, t2 = calc_scale_factors(thick_pt1, thick_pt2, pt_k) map_to_ifcs.append((k, i, t1, t2)) return map_to_ifcs
[docs] def opt_inv_from_dgm_and_osp(Z, osp, efl=1, t=None): """ Use the osp database to update the diagram data """ aperture_spec = osp['pupil'].derive_parax_params() pupil_oi_key, pupil_key, pupil_value = aperture_spec field_spec = osp['fov'].derive_parax_params() fov_oi_key, field_key, field_value = field_spec y1_star, ybar0_star = calc_object_and_pupil_from_dgm(Z, 0) num_nodes = len(Z) yk_star, ybark_star = calc_object_and_pupil_from_dgm(Z, num_nodes-2) opt_inv = 1 m = 0 if np.isinf(ybar0_star) else ybark_star/ybar0_star if fov_oi_key == 'object': if pupil_oi_key == 'object': if field_key == 'height' and pupil_key == 'slope': opt_inv = -pupil_value*ybar0_star elif field_key == 'slope' and pupil_key == 'height': opt_inv = y1_star*field_value elif field_key == 'height' and pupil_key == 'height': if t is not None: opt_inv = -y1_star*ybar0_star/t else: # pupil_oi_key == 'image' if field_key == 'height' and pupil_key == 'slope': opt_inv = -pupil_value*ybark_star elif field_key == 'slope' and pupil_key == 'height': opt_inv = yk_star*field_value elif field_key == 'height' and pupil_key == 'height': if t is not None: opt_inv = -yk_star*ybark_star/t else: # fov_oi_key == 'image' if pupil_oi_key == 'image': if field_key == 'height' and pupil_key == 'slope': opt_inv = -pupil_value*ybark_star elif field_key == 'slope' and pupil_key == 'height': opt_inv = yk_star*field_value elif field_key == 'height' and pupil_key == 'height': if t is not None: opt_inv = -yk_star*ybark_star/t else: # pupil_oi_key == 'object' if field_key == 'height' and pupil_key == 'slope': opt_inv = -pupil_value*ybar0_star elif field_key == 'slope' and pupil_key == 'height': opt_inv = y1_star*field_value elif field_key == 'height' and pupil_key == 'height': if t is not None: opt_inv = -y1_star*ybar0_star/t return opt_inv
[docs] def calc_object_and_pupil_from_dgm(Z, idx: int): """ calculate axial intercepts of line between idx and idx+1 """ y, ybar = 1, 0 del_Z = Z[idx+1] - Z[idx] if del_Z[ybar] == 0: y_star = infinity_guard(np.inf) ybar_star = Z[idx][ybar] else: k = del_Z[y] / del_Z[ybar] y_star = Z[idx][y] - k*Z[idx][ybar] if k == 0: inf_direction = -del_Z[ybar] if idx == 0 else del_Z[ybar] ybar_star = infinity_guard(math.copysign(np.inf, inf_direction)) else: ybar_star = Z[idx][ybar] - Z[idx][y]/k # print(f"{idx}: y_star={y_star}, ybar_star={ybar_star}, k={k}") return y_star, ybar_star
[docs] def compute_principle_points_from_dgm(Z, sys, opt_inv, idx_os=0, idx_is=-2): """ Calculate focal points and principal points. Args: Z: array of diagram points sys: the system data for the diagram opt_inv: optical invariant idx_os: object space gap index idx_is: image space gap index Returns: (power, efl, fl_obj, fl_img, pp1, ppk, pp_sep, ffl, bfl) - power: optical power of system - efl: effective focal length - fl_obj: object space focal length, f - fl_img: image space focal length, f' - pp1: distance from the 1st interface to the front principle plane - ppk: distance from the last interface to the rear principle plane - pp_sep: distance from the front principle plane to the rear principle plane - ffl: front focal length, distance from the 1st interface to the front focal point - bfl: back focal length, distance from the last interface to the back focal point """ def oal(): oal = 0 for s in sys: oal += s[mc.indx]*s[mc.tau] return oal n_o = sys[idx_os][mc.indx] n_i = sys[idx_is][mc.indx] z0 = Z[idx_os] z1o = Z[idx_os+1] z1i = Z[idx_is] z2 = Z[idx_is+1] z1 = np.array(get_intersect(z0, z1o, z1i, z2)) To = np.cross(z1, z1o) Wo = (z1 - z1o)/To Ti = np.cross(z1, z1i) Wi = (z1 - z1i)/Ti Pwr = np.cross(Wi, Wo) zfo = -Wi/Pwr zfi = Wo/Pwr power = Pwr*opt_inv pp1 = n_o*To/opt_inv ppk = n_i*Ti/opt_inv fl_obj = n_o/power fl_img = n_i/power efl = fl_img ffl = pp1 + (-fl_obj) bfl = ppk + fl_img pp_sep = oal() - pp1 + ppk pp_info = power, efl, fl_obj, fl_img, pp1, ppk, pp_sep, ffl, bfl # print(f"efl={pp_info[0]:8.4f}, ffl={pp_info[3]:8.4f}, bfl={pp_info[4]:8.4f}, " # f"pp1={pp_info[1]:8.4f}, ppk={pp_info[2]:8.4f}") return pp_info
[docs] def specsheet_from_dgm(parax_model, optical_spec, specsheet): """ update specsheet to contents of parax_model, while preserving inputs """ opt_inv = parax_model.opt_inv Z, T, W, Pwr = parax_model.ztwp y1_star, ybar0_star = calc_object_and_pupil_from_dgm(Z, 0) yk_star, ybark_star = calc_object_and_pupil_from_dgm(Z, -2) m = ybar0_star/ybark_star Pwr_tot = np.cross(W[-2], W[0]) efl = 1/(Pwr_tot*opt_inv) conj_type = 'finite' if is_kinda_big(T[0]/opt_inv): conj_type = 'infinite' specsheet.conjugate_type = conj_type # specsheet.imager_inputs contains values of independent variables of # the optical system. Augment these as needed to get a defined imager. imager_inputs = dict(specsheet.imager_inputs) num_imager_inputs = len(imager_inputs) if num_imager_inputs == 0: # no user inputs, use model values if conj_type == 'finite': imager_inputs['m'] = m imager_inputs['f'] = efl specsheet.frozen_imager_inputs = [False]*5 else: # conj_type == 'infinite' imager_inputs['s'] = -math.inf if efl != 0: imager_inputs['f'] = efl specsheet.frozen_imager_inputs = [True, True, True, True, False] elif num_imager_inputs == 1: # some/partial user input specification if conj_type == 'finite': # make sure that m is specified if 'm' in imager_inputs: imager_inputs['f'] = efl else: imager_inputs['m'] = m specsheet.frozen_imager_inputs = [False]*5 else: # conj_type == 'infinite' imager_inputs['s'] = -math.inf if efl != 0: imager_inputs['f'] = efl specsheet.frozen_imager_inputs = [True, True, True, True, False] specsheet.imager = ideal_imager_setup(**imager_inputs) ape_key, ape_value = optical_spec.pupil.get_input_for_specsheet() fld_key, fld_value = optical_spec.field_of_view.get_input_for_specsheet() etendue_inputs = specsheet.etendue_inputs etendue_inputs['aperture'][ape_key[0]][ape_key[1]] = ape_value etendue_inputs['field'][fld_key[0]][fld_key[1]] = fld_value specsheet.generate_from_inputs(imager_inputs, etendue_inputs) return specsheet
[docs] def update_from_dgm(Z, opt_inv, osp): """ Given a |ybar| diagram `Z`, generate the dual and params. """ y, ybar = 1, 0 delZo = Z[1] - Z[0] To = np.cross(Z[1], Z[0]) if is_kinda_big(delZo[ybar]): field_spec = osp['fov'].derive_parax_params() fov_oi_key, field_key, field_value = field_spec Wo = np.array([field_value/opt_inv, 0.]) elif is_kinda_big(delZo[y]): aperture_spec = osp['pupil'].derive_parax_params() pupil_oi_key, pupil_key, pupil_value = aperture_spec Wo = np.array([0, pupil_value]) else: Wo = delZo/To T = [To] W = [Wo] Pwr = [np.array(0.)] for i in range(1, len(Z)-1): delZ = Z[i+1] - Z[i] Ti = np.cross(Z[i+1], Z[i]) if is_kinda_big(delZ[ybar]): Pwr1 = Wo[y]/Z[i][y] Wi = Wo - Pwr1*Z[i] elif is_kinda_big(delZ[y]): Pwr1 = Wo[ybar]/Z[i][ybar] Wi = Wo - Pwr1*Z[i] else: Wi = delZ/Ti Pwr1 = np.cross(Wi, Wo) T.append(Ti) W.append(Wi) Pwr.append(Pwr1) Wo = Wi T.append(0) W.append(Wi) Pwr.append(0) return Z, np.array(T), np.array(W), np.array(Pwr)
[docs] def parax_to_dgms(ax, pr, sys, opt_inv): """ convert paraxial rays to normalized diagram. """ Z = [] T = [] W = [] Pwr = [] for pkg in zip(ax, pr, sys): ax_i, pr_i, sys_i = pkg Z.append(np.array([pr_i[mc.ht], ax_i[mc.ht]])) W.append(np.array([pr_i[mc.slp], ax_i[mc.slp]])/opt_inv) T.append(sys_i[mc.tau]*opt_inv) Pwr.append(sys_i[mc.pwr]/opt_inv) del T[-1:] del W[-1:] return Z, T, W, Pwr
[docs] def dgms_to_parax(Z, T, W, Pwr, opt_inv, rndx_and_imode=None): """ convert normalized diagrams to paraxial ray and first order system. """ y, ybar = 1, 0 ax = [] pr = [] sys = [] len_Z = len(Z) len_W = len_Z - 1 if rndx_and_imode is None: rndx_and_imode = [(1., 'transmit') for i in range(len_Z)] for i in range(len_Z): if i < len_W: W_i = W[i] T_i = T[i] ni = rndx_and_imode[i] else: W_i = W[i-1] T_i = 0 ni = rndx_and_imode[i-1] ax.append([Z[i][y], W_i[y]*opt_inv]) pr.append([Z[i][ybar], W_i[ybar]*opt_inv]) sys.append([Pwr[i]*opt_inv, T_i/opt_inv, *ni]) return ax, pr, sys