Source code for xoppylib.srcalc.conic

"""
Conic optical surface for ray tracing (mirrors and lenses).
"""

import numpy

from numpy.testing import assert_equal, assert_almost_equal

# OE surface in form of conic equation:
#      ccc[0]*X^2 + ccc[1]*Y^2 + ccc[2]*Z^2 +
#      ccc[3]*X*Y + ccc[4]*Y*Z + ccc[5]*X*Z  +
#      ccc[6]*X   + ccc[7]*Y   + ccc[8]*Z + ccc[9] = 0


[docs]class Conic(object): def __init__(self, ccc=numpy.zeros(10)): if ccc is not None: self.ccc = ccc.copy() else: self.ccc = numpy.zeros(10)
[docs] @classmethod def initialize_from_coefficients(cls, ccc): if numpy.array(ccc).size != 10: raise Exception("Invalid coefficients (dimension must be 10)") return Conic(ccc=ccc)
[docs] @classmethod def initialize_as_plane(cls): return Conic(numpy.array([0,0,0,0,0,0,0,0,-1.,0]))
# # initializers from focal distances #
[docs] @classmethod def initialize_as_sphere_from_focal_distances(cls,p, q, theta1, cylindrical=0, cylangle=0.0, switch_convexity=0): ccc = Conic() ccc.set_sphere_from_focal_distances(p,q,theta1) if cylindrical: ccc.set_cylindrical(cylangle) if switch_convexity: ccc.switch_convexity() return ccc
[docs] @classmethod def initialize_as_ellipsoid_from_focal_distances(cls,p, q, theta1, cylindrical=0, cylangle=0.0, switch_convexity=0): ccc = Conic() ccc.set_ellipsoid_from_focal_distances(p,q,theta1) if cylindrical: ccc.set_cylindrical(cylangle) if switch_convexity: ccc.switch_convexity() return ccc
[docs] @classmethod def initialize_as_paraboloid_from_focal_distances(cls,p, q, theta1, cylindrical=0, cylangle=0.0, switch_convexity=0): ccc = Conic() ccc.set_paraboloid_from_focal_distances(p,q,theta1) if cylindrical: ccc.set_cylindrical(cylangle) if switch_convexity: ccc.switch_convexity() return ccc
[docs] @classmethod def initialize_as_hyperboloid_from_focal_distances(cls,p, q, theta1, cylindrical=0, cylangle=0.0, switch_convexity=0): ccc = Conic() ccc.set_hyperboloid_from_focal_distances(p,q,theta1) if cylindrical: ccc.set_cylindrical(cylangle) if switch_convexity: ccc.switch_convexity() return ccc
# # initializars from surface parameters #
[docs] @classmethod def initialize_as_sphere_from_curvature_radius(cls, radius, cylindrical=0, cylangle=0.0, switch_convexity=0): ccc = Conic() ccc.set_sphere_from_curvature_radius(radius) if cylindrical: ccc.set_cylindrical(cylangle) if switch_convexity: ccc.switch_convexity() return ccc
[docs] def duplicate(self): return Conic.initialize_from_coefficients(self.ccc.copy())
# # getters #
[docs] def get_coefficients(self): return self.ccc.copy()
# # setters #
[docs] def set_coefficients(self,ccc): if numpy.array(ccc).size != 10: raise Exception("Invalid coefficients (dimension must be 10)") self.ccc = ccc
[docs] def vector_reflection(self,v1,normal): tmp = v1 * normal tmp2 = tmp[0,:] + tmp[1,:] + tmp[2,:] tmp3 = normal.copy() for jj in (0,1,2): tmp3[jj,:] = tmp3[jj,:] * tmp2 v2 = v1 - 2 * tmp3 v2mod = numpy.sqrt(v2[0,:]**2 + v2[1,:]**2 + v2[2,:]**2) v2 /= v2mod return v2
[docs] def get_normal(self,x2): # ; # ; Calculates the normal at intercept points x2 [see shadow's normal.F] # ; normal = numpy.zeros_like(x2) normal[0,:] = 2 * self.ccc[1-1] * x2[0,:] + self.ccc[4-1] * x2[1,:] + self.ccc[6-1] * x2[2,:] + self.ccc[7-1] normal[1,:] = 2 * self.ccc[2-1] * x2[1,:] + self.ccc[4-1] * x2[0,:] + self.ccc[5-1] * x2[2,:] + self.ccc[8-1] normal[2,:] = 2 * self.ccc[3-1] * x2[2,:] + self.ccc[5-1] * x2[1,:] + self.ccc[6-1] * x2[0,:] + self.ccc[9-1] normalmod = numpy.sqrt( normal[0,:]**2 + normal[1,:]**2 + normal[2,:]**2 ) normal[0,:] /= normalmod normal[1,:] /= normalmod normal[2,:] /= normalmod return normal
[docs] def apply_specular_reflection_on_beam(self,newbeam): # ; # ; TRACING... # ; x1 = newbeam.get_columns([1,2,3]) # numpy.array(a3.getshcol([1,2,3])) v1 = newbeam.get_columns([4,5,6]) # numpy.array(a3.getshcol([4,5,6])) flag = newbeam.get_column(10) # numpy.array(a3.getshonecol(10)) t1,t2 = self.calculate_intercept(x1,v1) t, iflag = self.choose_solution(t1,t2,reference_distance=-newbeam.get_column(2).mean()) # for i in range(t.size): # print(">>>> solutions: ",t1[i],t2[i],t[i]) x2 = x1 + v1 * t for i in range(flag.size): if iflag[i] < 0: flag[i] = -100 # ; # ; Calculates the normal at each intercept [see shadow's normal.F] # ; normal = self.get_normal(x2) # ; # ; reflection # ; v2 = self.vector_reflection(v1,normal) # ; # ; writes the mirr.XX file # ; newbeam.set_column(1, x2[0]) newbeam.set_column(2, x2[1]) newbeam.set_column(3, x2[2]) newbeam.set_column(4, v2[0]) newbeam.set_column(5, v2[1]) newbeam.set_column(6, v2[2]) newbeam.set_column(10, flag ) return newbeam
[docs] def calculate_intercept(self,XIN,VIN,keep=0): # # FUNCTION conicintercept,ccc,xIn1,vIn1,iflag,keep=keep # # # # # # ;+ # # ; # # ; NAME: # # ; CONICINTERCEPT # # ; # # ; PURPOSE: # # ; This function Calculates the intersection of a # # ; conic (defined by its 10 coefficients in ccc) # # ; with a straight line, defined by a point xIn and # # ; an unitary direction vector vIn # # ; # # ; CATEGORY: # # ; SHADOW tools # # ; # # ; CALLING SEQUENCE: # # ; t = conicIntercept(ccc,xIn,vIn,iFlag) # # ; # # ; INPUTS: # # ; ccc: the array with the 10 coefficients defining the # # ; conic. # # ; xIn: a vector DblArr(3) or stack of vectors DblArr(3,nvectors) # # ; vIn: a vector DblArr(3) or stack of vectors DblArr(3,nvectors) # # ; # # ; OUTPUTS # # ; t the "travelled" distance between xIn and the surface # # ; # # ; OUTPUT KEYWORD PARAMETERS # # ; IFLAG: A flag (negative if no intersection) # # ; # # ; KEYWORD PARAMETERS # # ; keep: 0 [default] keep the max t from both solutions # # ; 1 keep the MIN t from both solutions # # ; 2 keep the first solution # # ; 3 keep the second solution # # ; ALGORITHM: # # ; Adapted from SHADOW/INTERCEPT # # ; # # ; Equation of the conic: # # ; # # ; c[0]*X^2 + c[1]*Y^2 + c[2]*Z^2 + # # ; c[3]*X*Y + c[4]*Y*Z + c[5]*X*Z + # # ; c[6]*X + c[7]*Y + c[8]*Z + c[9] = 0 # # ; # # ; NOTE that the vectors, that are usually DblArr(3) can be # # ; stacks of vectors DblArr(3,nvectors). In such a case, # # ; the routine returns t # # ; # # ; # # ; AUTHOR: # # ; M. Sanchez del Rio srio@esrf.eu, Sept. 29, 2009 # # ; # # ; MODIFICATION HISTORY: # # ; # # ;- # # # # # # ;CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC # # ;C # # ;C subroutine intercept ( xin, vin, tpar, iflag) # # ;C # # ;C purpose computes the intercepts onto the mirror surface # # ;C # # ;C arguments xin ray starting position mirror RF # # ;C vin ray direction mirror RF # # ;C tpar distance from start of # # ;C intercept # # ;C iflag input 1 ordinary case # # ;C -1 ripple case # # ;C iflag output 0 success # # ;C -1 complex sol. # # ;C # # ;CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC # # CCC = self.ccc if XIN.shape==(3,): XIN.shape = (3,1) if VIN.shape==(3,): VIN.shape = (3,1) AA = CCC[1-1]*VIN[1-1,:]**2 \ + CCC[2-1]*VIN[2-1,:]**2 \ + CCC[3-1]*VIN[3-1,:]**2 \ + CCC[4-1]*VIN[1-1,:]*VIN[2-1,:] \ + CCC[5-1]*VIN[2-1,:]*VIN[3-1,:] \ + CCC[6-1]*VIN[1-1,:]*VIN[3-1,:] BB = CCC[1-1] * XIN[1-1,:] * VIN[1-1,:]*2 \ + CCC[2-1] * XIN[2-1,:] * VIN[2-1,:]*2 \ + CCC[3-1] * XIN[3-1,:] * VIN[3-1,:]*2 \ + CCC[4-1] * (XIN[2-1,:] * VIN[1-1,:] \ + XIN[1-1,:] * VIN[2-1,:]) \ + CCC[5-1]*(XIN[3-1,:]*VIN[2-1,:] \ + XIN[2-1,:]*VIN[3-1,:]) \ + CCC[6-1]*(XIN[1-1,:]*VIN[3-1,:] \ + XIN[3-1,:]*VIN[1-1,:]) \ + CCC[7-1] * VIN[1-1,:] \ + CCC[8-1] * VIN[2-1,:] \ + CCC[9-1] * VIN[3-1,:] CC = CCC[1-1] * XIN[1-1,:]**2 \ + CCC[2-1] * XIN[2-1,:]**2 \ + CCC[3-1] * XIN[3-1,:]**2 \ + CCC[4-1] * XIN[2-1,:] * XIN[1-1,:] \ + CCC[5-1] * XIN[2-1,:] * XIN[3-1,:] \ + CCC[6-1] * XIN[1-1,:]*XIN[3-1,:] \ + CCC[7-1] * XIN[1-1,:] \ + CCC[8-1] * XIN[2-1,:] \ + CCC[9-1] * XIN[3-1,:] \ + CCC[10-1] # ;C # ;C Solve now the second deg. equation ** # ;C TPAR = numpy.zeros_like(AA) TPAR1 = numpy.zeros_like(AA) TPAR2 = numpy.zeros_like(AA) IFLAG = numpy.ones_like(AA) # shape_x = x.shape # TODO: remove loop! for i in range(AA.size): if numpy.abs(AA[i]) < 1e-15: TPAR1[i] = - CC[i] / BB[i] TPAR2[i] = TPAR1[i] else: DENOM = 0.5 / AA[i] DETER = BB[i] ** 2 - CC[i] * AA[i] * 4 if DETER < 0.0: TPAR[i] = 0.0 IFLAG[i] = -1 else: TPAR1[i] = -(BB[i] + numpy.sqrt(DETER)) * DENOM TPAR2[i] = -(BB[i] - numpy.sqrt(DETER)) * DENOM if TPAR2.size == 1: TPAR2 = numpy.asscalar(TPAR2) return TPAR1.real,TPAR2.real
[docs] def choose_solution(self,TPAR1,TPAR2,reference_distance=10.0): #todo remove this nasty thing TPAR = numpy.zeros_like(TPAR1) I_FLAG = numpy.ones_like(TPAR1) for i in range(TPAR1.size): if ( numpy.abs(TPAR1[i]-reference_distance) <= numpy.abs(TPAR2[i]-reference_distance)): TPAR[i] = TPAR1[i] else: TPAR[i] = TPAR2[i] return TPAR,I_FLAG
[docs] def z_vs_xy(self,x,y): ccc = self.ccc AA = ccc[2] * numpy.ones_like(x) BB = ccc[4] * y + ccc[5] * x + ccc[8] CC = ccc[0]*x**2 + ccc[1]*y**2 + ccc[3]*x*y + ccc[6]*x + ccc[7]*y + ccc[9] shape_x = x.shape AAf = AA.flatten() BBf = BB.flatten() CCf = CC.flatten() TPAR1 = numpy.zeros_like(CCf,dtype=complex) TPAR2 = numpy.zeros_like(CCf,dtype=complex) for i in range(AAf.size): roots = numpy.roots([CCf[i],BBf[i],AAf[i]]) TPAR1[i] = roots[0] TPAR2[i] = roots[1] # print("Found solutions: ",TPAR1[i],TPAR2[i]) TPAR2.shape = shape_x if TPAR2.size == 1: TPAR2 = numpy.asscalar(TPAR2) return TPAR2.real
[docs] def set_cylindrical(self,CIL_ANG): COS_CIL = numpy.cos(CIL_ANG) SIN_CIL = numpy.sin(CIL_ANG) A_1 = self.ccc[1-1] A_2 = self.ccc[2-1] A_3 = self.ccc[3-1] A_4 = self.ccc[4-1] A_5 = self.ccc[5-1] A_6 = self.ccc[6-1] A_7 = self.ccc[7-1] A_8 = self.ccc[8-1] A_9 = self.ccc[9-1] A_10 = self.ccc[10-1] self.ccc[1-1] = A_1 * SIN_CIL**4 + A_2 * COS_CIL**2 * SIN_CIL**2 - A_4 * COS_CIL * SIN_CIL**3 self.ccc[2-1] = A_2 * COS_CIL**4 + A_1 * COS_CIL**2 * SIN_CIL**2 - A_4 * COS_CIL**3 * SIN_CIL self.ccc[3-1] = A_3 # Z^2 self.ccc[4-1] = - 2*A_1 * COS_CIL * SIN_CIL**3 - 2 * A_2 * COS_CIL**3 * SIN_CIL + 2 * A_4 * COS_CIL**2 *SIN_CIL**2 # X Y self.ccc[5-1] = A_5 * COS_CIL**2 - A_6 * COS_CIL * SIN_CIL # Y Z self.ccc[6-1] = A_6 * SIN_CIL**2 - A_5 * COS_CIL * SIN_CIL # X Z self.ccc[7-1] = A_7 * SIN_CIL**2 - A_8 * COS_CIL * SIN_CIL # X self.ccc[8-1] = A_8 * COS_CIL**2 - A_7 * COS_CIL * SIN_CIL # Y self.ccc[9-1] = A_9 # Z self.ccc[10-1]= A_10
[docs] def switch_convexity(self): self.ccc[5-1] = - self.ccc[5-1] self.ccc[6-1] = - self.ccc[6-1] self.ccc[9-1] = - self.ccc[9-1]
[docs] def set_sphere_from_focal_distances(self, ssour, simag, theta_grazing, verbose=True): # todo: implement also sagittal bending print("Theta grazing is: %f" %(theta_grazing)) theta = (numpy.pi/2) - theta_grazing rmirr = ssour * simag * 2 / numpy.cos(theta) / (ssour + simag) if verbose: txt = "" txt += "p=%f, q=%f, theta_grazing=%f rad, theta_normal=%f rad\n"%(ssour,simag,theta_grazing,theta) txt += "Radius= %f \n"%(rmirr) print(txt) self.ccc[1-1] = 1.0 # X^2 # = 0 in cylinder case self.ccc[2-1] = 1.0 # Y^2 self.ccc[3-1] = 1.0 # Z^2 self.ccc[4-1] = .0 # X*Y # = 0 in cylinder case self.ccc[5-1] = .0 # Y*Z self.ccc[6-1] = .0 # X*Z # = 0 in cylinder case self.ccc[7-1] = .0 # X # = 0 in cylinder case self.ccc[8-1] = .0 # Y self.ccc[9-1] = -2 * rmirr # Z self.ccc[10-1] = .0 # G
[docs] def set_sphere_from_curvature_radius(self,rmirr): self.ccc[1-1] = 1.0 # X^2 # = 0 in cylinder case self.ccc[2-1] = 1.0 # Y^2 self.ccc[3-1] = 1.0 # Z^2 self.ccc[4-1] = .0 # X*Y # = 0 in cylinder case self.ccc[5-1] = .0 # Y*Z self.ccc[6-1] = .0 # X*Z # = 0 in cylinder case self.ccc[7-1] = .0 # X # = 0 in cylinder case self.ccc[8-1] = .0 # Y self.ccc[9-1] = -2 * rmirr # Z self.ccc[10-1] = .0 # G
[docs] def set_ellipsoid_from_focal_distances(self, ssour, simag, theta_grazing, verbose=True): theta = (numpy.pi/2) - theta_grazing COSTHE = numpy.cos(theta) SINTHE = numpy.sin(theta) AXMAJ = ( ssour + simag )/2 AXMIN = numpy.sqrt( simag * ssour) * COSTHE AFOCI = numpy.sqrt( AXMAJ**2 - AXMIN**2 ) ECCENT = AFOCI/AXMAJ # ;C # ;C The center is computed on the basis of the object and image positions # ;C YCEN = (ssour - simag) * 0.5 / ECCENT ZCEN = -numpy.sqrt( 1 - YCEN**2 / AXMAJ**2) * AXMIN # ;C # ;C Computes now the normal in the mirror center. # ;C RNCEN = numpy.zeros(3) RNCEN[1-1] = 0.0 RNCEN[2-1] = -2 * YCEN / AXMAJ**2 RNCEN[3-1] = -2 * ZCEN / AXMIN**2 # ;CALL NORM(RNCEN,RNCEN) RNCEN = RNCEN / numpy.sqrt((RNCEN**2).sum()) # ;C # ;C Computes the tangent versor in the mirror center. # ;C RTCEN = numpy.zeros(3) RTCEN[1-1] = 0.0 RTCEN[2-1] = RNCEN[3-1] RTCEN[3-1] = -RNCEN[2-1] if verbose: txt = "" txt += "p=%f, q=%f, theta_grazing=%f rad, theta_normal=%f rad\n" % (ssour, simag, theta_grazing, theta) txt += 'Ellipsoid of revolution a=%f \n'%AXMAJ txt += 'Ellipsoid of revolution b=%f \n'%AXMIN txt += 'Ellipsoid of revolution c=sqrt(a^2-b^2)=%f \n'%AFOCI txt += 'Ellipsoid of revolution focal distance c^2=%f \n'%(AFOCI**2) txt += 'Ellipsoid of revolution excentricity: %f \n'%ECCENT txt += 'Optical element center at: [0,%f,%f]\n'%(YCEN,ZCEN) txt += 'Optical element normal: [%f,%f,%f]\n'%(RNCEN[0],RNCEN[1],RNCEN[2]) txt += 'Optical element tangent: [%f,%f,%f]\n'%(RTCEN[0],RTCEN[1],RTCEN[2]) print(txt) # ;C Computes now the quadric coefficient with the mirror center # ;C located at (0,0,0) and normal along (0,0,1) # ;C A = 1 / AXMIN**2 B = 1 / AXMAJ**2 C = A self.ccc[0] = A self.ccc[1] = B * RTCEN[2-1]**2 + C * RTCEN[3-1]**2 self.ccc[2] = B * RNCEN[2-1]**2 + C * RNCEN[3-1]**2 self.ccc[3] = 0.0 self.ccc[4] = 2 * (B * RNCEN[2-1] * RTCEN[2-1] + C * RNCEN[3-1] * RTCEN[3-1]) self.ccc[5] = 0.0 self.ccc[6] = 0.0 self.ccc[7] = 0.0 self.ccc[8] = 2 * (B * YCEN * RNCEN[2-1] + C * ZCEN * RNCEN[3-1]) self.ccc[9] = 0.0
[docs] def set_paraboloid_from_focal_distances(self, SSOUR, SIMAG, theta_grazing, infinity_location="", verbose=True): # ;C # ;C Computes the parabola # ;C theta = (numpy.pi/2) - theta_grazing COSTHE = numpy.cos(theta) SINTHE = numpy.sin(theta) if infinity_location == "": if SSOUR <= SIMAG: location = "q" else: location = "p" if location=="q": PARAM = 2 * SSOUR * COSTHE**2 YCEN = -SSOUR * SINTHE**2 ZCEN = -2 * SSOUR * SINTHE * COSTHE fact = -1.0 elif location == "p": PARAM = 2 * SIMAG * COSTHE**2 YCEN = - SIMAG * SINTHE**2 ZCEN = -2 * SIMAG * SINTHE * COSTHE fact = 1.0 if verbose: txt = "" if location == "p": txt += "Source is at infinity\n" txt += "q=%f, theta_grazing=%f rad, theta_normal=%f rad\n" % (SIMAG, theta_grazing, theta) else: txt += "Image is at infinity\n" txt += "p=%f, theta_grazing=%f rad, theta_normal=%f rad\n" % (SSOUR, theta_grazing, theta) txt += 'Parabloid of revolution PARAM=%f \n'%PARAM txt += 'Optical element center at: [0,%f,%f]\n'%(YCEN,ZCEN) print(txt) self.ccc[0] = 1.0 self.ccc[1] = COSTHE**2 self.ccc[2] = SINTHE**2 self.ccc[3] = 0.0 self.ccc[4] = 2 * fact * COSTHE * SINTHE self.ccc[5] = 0.0 self.ccc[6] = 0.0 self.ccc[7] = 0.0 self.ccc[8] = 2 * ZCEN * SINTHE - 2 * PARAM * COSTHE self.ccc[9] = 0.0
[docs] def set_hyperboloid_from_focal_distances(self, SSOUR, SIMAG, theta_grazing, verbose=True): theta = (numpy.pi/2) - theta_grazing COSTHE = numpy.cos(theta) SINTHE = numpy.sin(theta) AXMAJ = (SSOUR - SIMAG)/2 # ;C # ;C If AXMAJ > 0, then we are on the left branch of the hyp. Else we # ;C are onto the right one. We have to discriminate between the two cases # ;C In particular, if AXMAJ.LT.0 then the hiperb. will be convex. # ;C AFOCI = 0.5 * numpy.sqrt( SSOUR**2 + SIMAG**2 + 2 * SSOUR * SIMAG * numpy.cos(2 * theta) ) # ;; why this works better? # ;; AFOCI = 0.5D0*SQRT( SSOUR^2 + SIMAG^2 - 2*SSOUR*SIMAG*COS(2*THETA) ) AXMIN = numpy.sqrt( AFOCI**2 - AXMAJ**2 ) ECCENT = AFOCI / numpy.abs( AXMAJ ) BRANCH = -1.0 #; branch=+1,-1 # ;C # ;C Computes the center coordinates in the hiperbola RF. # ;C # ;IF AXMAJ GT 0.0D0 THEN BEGIN # ; YCEN = ( AXMAJ - SSOUR )/ECCENT ; < 0 # ;ENDIF ELSE BEGIN # ; YCEN = ( SSOUR - AXMAJ )/ECCENT ; > 0 # ;ENDELSE if AXMAJ>0: YCEN = (SSOUR - AXMAJ) / ECCENT else: YCEN = (SSOUR - AXMAJ) / ECCENT #YCEN = numpy.abs( SSOUR - AXMAJ ) / ECCENT * BRANCH ZCEN_ARG = numpy.abs( YCEN**2 / AXMAJ**2 - 1.0) if ZCEN_ARG > 1.0e-14: ZCEN = -AXMIN * numpy.sqrt(ZCEN_ARG) # < 0 else: ZCEN = 0.0 # ; # ; THIS GIVES BETTER LOOKING HYPERBOLA BUT WORSE TRACING. WHY? # ;YCEN=ABS(YCEN) # ;ZCEN=ABS(ZCEN) # ;C # ;C Computes now the normal in the same RF. The signs are forced to # ;C suit our RF. # ;C RNCEN = numpy.zeros(3) RNCEN[1-1] = 0.0 RNCEN[2-1] = -numpy.abs( YCEN ) / AXMAJ**2 # < 0 RNCEN[3-1] = -ZCEN / AXMIN**2 # > 0 RNCEN = RNCEN / numpy.sqrt((RNCEN**2).sum()) # ;C # ;C Computes the tangent in the same RF # ;C RTCEN = numpy.zeros(3) RTCEN[1-1] = 0.0 RTCEN[2-1] = -RNCEN[3-1] # > 0 RTCEN[3-1] = RNCEN[2-1] # > 0 # txt = [txt, $ # String('Rev Hyperboloid a: ', $ # AXMAJ, Format='(A40,G20.15)'), $ # String('Rev Hyperboloid b: ', $ # AXMIN, Format='(A40,G20.15)'), $ # String('Rev Hyperboloid c: ', $ # AFOCI, Format='(A40,G20.15)'), $ # String('Rev Hyperboloid focal discance c^2: ', $ # AFOCI^2, Format='(A40,G20.15)'), $ # String('Rev Hyperboloid excentricity: ', $ # ECCENT, Format='(A40,G20.15)'), $ # 'Mirror BRANCH: '+String(branch), $ # 'Mirror center at: '+vect2string([0,YCEN,ZCEN]), $ # 'Mirror normal: '+vect2string(RNCEN), $ # 'Mirror tangent: '+vect2string(RTCEN) ] if verbose: txt = "" txt += "p=%f, q=%f, theta_grazing=%f rad, theta_normal=%f rad\n" % (SSOUR, SIMAG, theta_grazing, theta) txt += 'Hyperboloid of revolution a=%f \n'%AXMAJ txt += 'Hyperboloid of revolution b=%f \n'%AXMIN txt += 'Hyperboloid of revolution c=%f \n'%AFOCI txt += 'Hyperboloid of revolution focal distance c^2=%f \n'%(AFOCI**2) txt += 'Hyperboloid of revolution excentricity: %f \n'%ECCENT txt += 'Optical element center at: [0,%f,%f]\n'%(YCEN,ZCEN) txt += 'Optical element normal: [%f,%f,%f]\n'%(RNCEN[0],RNCEN[1],RNCEN[2]) txt += 'Optical element tangent: [%f,%f,%f]\n'%(RTCEN[0],RTCEN[1],RTCEN[2]) print(txt) # ;C # ;C Coefficients of the canonical form # ;C A = -1 / AXMIN**2 B = 1 / AXMAJ**2 C = A # ;C # ;C Rotate now in the mirror RF. The equations are the same as for the # ;C ellipse case. # ;C self.ccc[0] = A self.ccc[1] = B * RTCEN[2-1]**2 + C * RTCEN[3-1]**2 self.ccc[2] = B * RNCEN[2-1]**2 + C * RNCEN[3-1]**2 self.ccc[3] = 0.0 self.ccc[4] = 2 * (B *RNCEN[2-1] * RTCEN[2-1] + C * RNCEN[3-1] * RTCEN[3-1]) self.ccc[5] = 0.0 self.ccc[6] = 0.0 self.ccc[7] = 0.0 self.ccc[8] = 2 * (B * YCEN * RNCEN[2-1] + C * ZCEN * RNCEN[3-1]) self.ccc[9] = 0.0
# # info #
[docs] def info(self): """ :return: """ txt = "" txt += "++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++\n" txt += "OE surface in form of conic equation: \n" txt += " ccc[0]*X^2 + ccc[1]*Y^2 + ccc[2]*Z^2 \n" txt += " ccc[3]*X*Y + ccc[4]*Y*Z + ccc[5]*X*Z \n" txt += " ccc[6]*X + ccc[7]*Y + ccc[8]*Z + ccc[9] = 0 \n" txt += " with \n" txt += " c[0] = %g \n "%self.ccc[0] txt += " c[1] = %g \n "%self.ccc[1] txt += " c[2] = %g \n "%self.ccc[2] txt += " c[3] = %g \n "%self.ccc[3] txt += " c[4] = %g \n "%self.ccc[4] txt += " c[5] = %g \n "%self.ccc[5] txt += " c[6] = %g \n "%self.ccc[6] txt += " c[7] = %g \n "%self.ccc[7] txt += " c[8] = %g \n "%self.ccc[8] txt += " c[9] = %g \n "%self.ccc[9] txt += "++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++'n" return txt
[docs] def rotation_surface_conic(self, alpha, axis ): if axis == 'x': self.rotation_surface_conic_x(alpha) elif axis == 'y': self.rotation_surface_conic_y(alpha) elif axis == 'z': self.rotation_surface_conic_z(alpha)
[docs] def rotation_surface_conic_x(self, alpha): a = numpy.cos(alpha) b = numpy.sin(alpha) c0 = self.ccc[0] c1 = self.ccc[1] * a ** 2 + self.ccc[2] * b ** 2 + self.ccc[4] * a * b c2 = self.ccc[1] * b ** 2 + self.ccc[2] * a ** 2 - self.ccc[4] * a * b c3 = self.ccc[3] * a + self.ccc[5] * b c4 = - 2 * self.ccc[1] * a * b + 2 * self.ccc[2] * a * b + self.ccc[4] * (a ** 2 - b ** 2) c5 = - self.ccc[3] * b + self.ccc[5] * a c6 = self.ccc[6] c7 = self.ccc[7] * a + self.ccc[8] * b c8 = - self.ccc[7] * b + self.ccc[8] * a c9 = self.ccc[9] self.ccc = numpy.array([c0, c1, c2, c3, c4, c5, c6, c7, c8, c9])
[docs] def rotation_surface_conic_y(self,alpha): a = numpy.cos(alpha) b = numpy.sin(alpha) c0 = self.ccc[0] * a**2 + self.ccc[2] * b**2 - self.ccc[5] * a * b c1 = self.ccc[1] c2 = self.ccc[0] * b**2 + self.ccc[2] * a**2 + self.ccc[5] * a * b c3 = self.ccc[3] * a - self.ccc[4] * b c4 = self.ccc[3] * b + self.ccc[4] * a c5 = 2 * self.ccc[0] * a * b - 2 * self.ccc[2] * a * b + self.ccc[5] * (a**2 - b**2) c6 = self.ccc[6] *a - self.ccc[8] * b c7 = self.ccc[7] c8 = self.ccc[6] * b + self.ccc[8] * a c9 = self.ccc[9] self.ccc = numpy.array([c0, c1, c2, c3, c4, c5, c6, c7, c8, c9])
[docs] def rotation_surface_conic_z(self, alpha): a = numpy.cos(alpha) b = numpy.sin(alpha) c0 = self.ccc[0] * a ** 2 + self.ccc[1] * b ** 2 + self.ccc[3] * a * b c1 = self.ccc[0] * b ** 2 + self.ccc[1] * a ** 2 - self.ccc[3] * a * b c2 = self.ccc[2] c3 = - 2 * self.ccc[0] * a * b + 2 * self.ccc[1] * a * b + self.ccc[3] * (a ** 2 - b ** 2) c4 = self.ccc[4] * a - self.ccc[5] * b c5 = self.ccc[4] * b + self.ccc[5] * a c6 = self.ccc[6] * a + self.ccc[7] * b c7 = - self.ccc[6] * b + self.ccc[7] * a c8 = self.ccc[8] c9 = self.ccc[9] self.ccc = numpy.array([c0, c1, c2, c3, c4, c5, c6, c7, c8, c9])
[docs] def translation_surface_conic (self, x0, axis = 'x'): if axis == 'x': self.translation_surface_conic_x(x0) elif axis == 'y': self.translation_surface_conic_y(x0) elif axis == 'z': self.translation_surface_conic_z(x0)
[docs] def translation_surface_conic_x(self, x0): c6 = - 2 * self.ccc[0] * x0 + self.ccc[6] c7 = - self.ccc[3] * x0 + self.ccc[7] c8 = - self.ccc[5] * x0 + self.ccc[8] c9 = self.ccc[0] * x0**2 + self.ccc[9] - self.ccc[6] * x0 self.ccc = numpy.array([self.ccc[0], self.ccc[1], self.ccc[2], self.ccc[3], self.ccc[4], self.ccc[5], c6, c7, c8, c9])
[docs] def translation_surface_conic_y(self, y0): c6 = - self.ccc[3] * y0 + self.ccc[6] c7 = - 2 * self.ccc[1] * y0 + self.ccc[7] c8 = - self.ccc[4] * y0 + self.ccc[8] c9 = self.ccc[1] * y0**2 + self.ccc[9] - self.ccc[7] * y0 self.ccc = numpy.array([self.ccc[0], self.ccc[1], self.ccc[2], self.ccc[3], self.ccc[4], self.ccc[5], c6, c7, c8, c9])
[docs] def translation_surface_conic_z(self, z0): c6 = - self.ccc[5] * z0 + self.ccc[6] c7 = - self.ccc[4] * z0 + self.ccc[7] c8 = - 2 * self.ccc[2] * z0 + self.ccc[8] c9 = self.ccc[2] * z0**2 + self.ccc[9] - self.ccc[8] * z0 self.ccc = numpy.array([self.ccc[0], self.ccc[1], self.ccc[2], self.ccc[3], self.ccc[4], self.ccc[5], c6, c7, c8, c9])
[docs] def height(self,y=0,x=0,return_solution=0): """ :param y: a scalar, vector or mesh :param x: a scalar, vector or mesh y and x must be homogeneous, otherwise an error will occur: both scalars both mesh one scalar and another vector :param return_solution: 0 = guess the solution with zero at pole, 1 = get first solution 2 = get second solution :return: the height scalar/vector/mesh depending on inputs """ aa = self.ccc[2] bb = self.ccc[4] * y + self.ccc[5] * x + self.ccc[8] cc = self.ccc[0] * x**2 + self.ccc[1] * y**2 + self.ccc[3] * x * y + \ self.ccc[6] * x + self.ccc[7] * y + self.ccc[9] if aa != 0: discr = bb**2 - 4 * aa * cc + 0j s1 = (-bb + numpy.sqrt(discr)) / 2 / aa s2 = (-bb - numpy.sqrt(discr)) / 2 / aa if return_solution == 0: # select the solution close to zero at pole if numpy.abs(s1).min() < numpy.abs(s2).min(): ss = s1 else: ss = s2 elif return_solution == 1: ss = s1 else: ss = s2 else: ss = -cc / bb return numpy.real(ss)
if __name__ == "__main__": p = 13.73 + 13.599 q = 2.64 theta1 = 0.02181 # ccc = Conic.initialize_as_sphere_from_focal_distances(p, q, theta1, cylindrical=0, cylangle=0.0, switch_convexity=0) # ccc = Conic.initialize_as_ellipsoid_from_focal_distances(p, q, theta1, cylindrical=0, cylangle=0.0, switch_convexity=0) # ccc = Conic.initialize_as_paraboloid_from_focal_distances(p, q, theta1, cylindrical=0, cylangle=0.0, switch_convexity=0) ccc = Conic.initialize_as_hyperboloid_from_focal_distances(p, q, theta1, cylindrical=0, cylangle=0.0, switch_convexity=0) # print(ccc.info()) y = numpy.linspace(-0.15,0.15,100) z = ccc.height(y) from srxraylib.plot.gol import plot plot(y,z) # # # y = numpy.linspace(-0.15, 0.15, 200) x = numpy.linspace(-0.15, 0.15, 100) Y = numpy.outer(numpy.ones_like(x),y) X = numpy.outer(x,numpy.ones_like(y)) Z = ccc.height(Y,X) from srxraylib.plot.gol import plot_image plot_image(Z,x,y)