fieldplot.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415
  1. #!/usr/bin/env python
  2. # -*- coding: UTF-8 -*-
  3. #
  4. # Copyright (C) 2009-2015 Ovidio Peña Rodríguez <ovidio@bytesfall.com>
  5. # Copyright (C) 2013-2015 Konstantin Ladutenko <kostyfisik@gmail.com>
  6. #
  7. # This file is part of python-scattnlay
  8. #
  9. # This program is free software: you can redistribute it and/or modify
  10. # it under the terms of the GNU General Public License as published by
  11. # the Free Software Foundation, either version 3 of the License, or
  12. # (at your option) any later version.
  13. #
  14. # This program is distributed in the hope that it will be useful,
  15. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  16. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  17. # GNU General Public License for more details.
  18. #
  19. # The only additional remark is that we expect that all publications
  20. # describing work using this software, or all commercial products
  21. # using it, cite the following reference:
  22. # [1] O. Pena and U. Pal, "Scattering of electromagnetic radiation by
  23. # a multilayered sphere," Computer Physics Communications,
  24. # vol. 180, Nov. 2009, pp. 2348-2354.
  25. #
  26. # You should have received a copy of the GNU General Public License
  27. # along with this program. If not, see <http://www.gnu.org/licenses/>.
  28. # Several functions to plot field and streamlines (power flow lines).
  29. import scattnlay
  30. from scattnlay import fieldnlay
  31. from scattnlay import scattnlay
  32. import numpy as np
  33. import cmath
  34. def unit_vector(vector):
  35. """ Returns the unit vector of the vector. """
  36. return vector / np.linalg.norm(vector)
  37. def angle_between(v1, v2):
  38. """ Returns the angle in radians between vectors 'v1' and 'v2'::
  39. >>> angle_between((1, 0, 0), (0, 1, 0))
  40. 1.5707963267948966
  41. >>> angle_between((1, 0, 0), (1, 0, 0))
  42. 0.0
  43. >>> angle_between((1, 0, 0), (-1, 0, 0))
  44. 3.141592653589793
  45. """
  46. v1_u = unit_vector(v1)
  47. v2_u = unit_vector(v2)
  48. angle = np.arccos(np.dot(v1_u, v2_u))
  49. if np.isnan(angle):
  50. if (v1_u == v2_u).all():
  51. return 0.0
  52. else:
  53. return np.pi
  54. return angle
  55. ###############################################################################
  56. def GetFlow3D(x0, y0, z0, max_length, max_angle, x, m, pl):
  57. # Initial position
  58. flow_x = [x0]
  59. flow_y = [y0]
  60. flow_z = [z0]
  61. max_step = x[-1] / 3
  62. min_step = x[0] / 2000
  63. # max_step = min_step
  64. step = min_step * 2.0
  65. if max_step < min_step:
  66. max_step = min_step
  67. coord = np.vstack(([flow_x[-1]], [flow_y[-1]], [flow_z[-1]])).transpose()
  68. terms, E, H = fieldnlay(np.array([x]), np.array([m]), coord, pl=pl)
  69. Ec, Hc = E[0, 0, :], H[0, 0, :]
  70. S = np.cross(Ec, Hc.conjugate()).real
  71. Snorm_prev = S / np.linalg.norm(S)
  72. Sprev = S
  73. length = 0
  74. dpos = step
  75. count = 0
  76. while length < max_length:
  77. count = count + 1
  78. if (count > 4000): # Limit length of the absorbed power streamlines
  79. break
  80. if step < max_step:
  81. step = step * 2.0
  82. r = np.sqrt(flow_x[-1]**2 + flow_y[-1]**2 + flow_z[-1]**2)
  83. while step > min_step:
  84. # Evaluate displacement from previous poynting vector
  85. dpos = step
  86. dx = dpos * Snorm_prev[0]
  87. dy = dpos * Snorm_prev[1]
  88. dz = dpos * Snorm_prev[2]
  89. # Test the next position not to turn\chang size for more than
  90. # max_angle
  91. coord = np.vstack(([flow_x[-1] + dx], [flow_y[-1] + dy],
  92. [flow_z[-1] + dz])).transpose()
  93. terms, E, H = fieldnlay(np.array([x]), np.array([m]), coord, pl=pl)
  94. Ec, Hc = E[0, 0, :], H[0, 0, :]
  95. Eth = max(np.absolute(Ec)) / 1e10
  96. Hth = max(np.absolute(Hc)) / 1e10
  97. for i in xrange(0, len(Ec)):
  98. if abs(Ec[i]) < Eth:
  99. Ec[i] = 0 + 0j
  100. if abs(Hc[i]) < Hth:
  101. Hc[i] = 0 + 0j
  102. S = np.cross(Ec, Hc.conjugate()).real
  103. if not np.isfinite(S).all():
  104. break
  105. Snorm = S / np.linalg.norm(S)
  106. diff = (S - Sprev) / max(np.linalg.norm(S), np.linalg.norm(Sprev))
  107. if np.linalg.norm(diff) < max_angle:
  108. # angle = angle_between(Snorm, Snorm_prev)
  109. # if abs(angle) < max_angle:
  110. break
  111. step = step / 2.0
  112. # 3. Save result
  113. Sprev = S
  114. Snorm_prev = Snorm
  115. dx = dpos * Snorm_prev[0]
  116. dy = dpos * Snorm_prev[1]
  117. dz = dpos * Snorm_prev[2]
  118. length = length + step
  119. flow_x.append(flow_x[-1] + dx)
  120. flow_y.append(flow_y[-1] + dy)
  121. flow_z.append(flow_z[-1] + dz)
  122. return np.array(flow_x), np.array(flow_y), np.array(flow_z)
  123. ###############################################################################
  124. def GetField(crossplane, npts, factor, x, m, pl, mode_n=-1, mode_type=-1, inner_only = False):
  125. """
  126. crossplane: XZ, YZ, XY, or XYZ (half is XZ, half is YZ)
  127. npts: number of point in each direction
  128. factor: ratio of plotting size to outer size of the particle
  129. x: size parameters for particle layers
  130. m: relative index values for particle layers
  131. """
  132. scan = np.linspace(-factor*x[-1], factor*x[-1], npts)
  133. zero = np.zeros(npts*npts, dtype = np.float64)
  134. if crossplane=='XZ':
  135. coordX, coordZ = np.meshgrid(scan, scan)
  136. coordX.resize(npts * npts)
  137. coordZ.resize(npts * npts)
  138. coordY = zero
  139. coordPlot1 = coordX
  140. coordPlot2 = coordZ
  141. elif crossplane == 'YZ':
  142. coordY, coordZ = np.meshgrid(scan, scan)
  143. coordY.resize(npts * npts)
  144. coordZ.resize(npts * npts)
  145. coordX = zero
  146. coordPlot1 = coordY
  147. coordPlot2 = coordZ
  148. elif crossplane == 'XY':
  149. coordX, coordY = np.meshgrid(scan, scan)
  150. coordX.resize(npts * npts)
  151. coordY.resize(npts * npts)
  152. coordZ = zero
  153. coordPlot1 = coordY
  154. coordPlot2 = coordX
  155. elif crossplane=='XYZ':
  156. coordX, coordZ = np.meshgrid(scan, scan)
  157. coordY, coordZ = np.meshgrid(scan, scan)
  158. coordPlot1, coordPlot2 = np.meshgrid(scan, scan)
  159. coordPlot1.resize(npts * npts)
  160. coordPlot2.resize(npts * npts)
  161. half=npts//2
  162. # coordX = np.copy(coordX)
  163. # coordY = np.copy(coordY)
  164. coordX[:,:half]=0
  165. coordY[:,half:]=0
  166. coordX.resize(npts*npts)
  167. coordY.resize(npts*npts)
  168. coordZ.resize(npts*npts)
  169. else:
  170. print("Unknown crossplane")
  171. import sys
  172. sys.exit()
  173. coord = np.vstack((coordX, coordY, coordZ)).transpose()
  174. terms, E, H = fieldnlay(np.array([x]), np.array([m]), coord, pl=pl
  175. , mode_n=mode_n, mode_type=mode_type
  176. )
  177. Ec = E[0, :, :]
  178. Hc = H[0, :, :]
  179. if inner_only:
  180. mask = [ [np.sum(c**2)>x[-1]**2]*3 for c in coord]
  181. np.place(Ec, mask, 0)
  182. np.place(Hc, mask, 0)
  183. P = []
  184. P = np.array(map(lambda n: np.linalg.norm(np.cross(Ec[n], Hc[n])).real,
  185. range(0, len(E[0]))))
  186. # for n in range(0, len(E[0])):
  187. # P.append(np.linalg.norm( np.cross(Ec[n], np.conjugate(Hc[n]) ).real/2 ))
  188. return Ec, Hc, P, coordPlot1, coordPlot2
  189. ###############################################################################
  190. def fieldplot(fig, ax, x, m, WL, comment='', WL_units=' ', crossplane='XZ',
  191. field_to_plot='Pabs', npts=101, factor=2.1, flow_total=11,
  192. is_flow_extend=True, pl=-1, outline_width=1, subplot_label=' ',
  193. mode_n=-1, mode_type=-1, isStream = False,minlength=0.5 , inner_only = False):
  194. print ("WL=",WL," xm:", x,m)
  195. Ec, Hc, P, coordX, coordZ = GetField(crossplane, npts, factor, x, m, pl, mode_n, mode_type, inner_only)
  196. Er = np.absolute(Ec)
  197. Hr = np.absolute(Hc)
  198. try:
  199. from matplotlib import cm
  200. from matplotlib.colors import LogNorm
  201. if field_to_plot == 'Pabs':
  202. Eabs_data = np.resize(P, (npts, npts)).T
  203. label = r'$\operatorname{Re}(E \times H)$'
  204. elif field_to_plot == 'Eabs':
  205. Eabs = np.sqrt(Er[:, 0]**2 + Er[:, 1]**2 + Er[:, 2]**2)
  206. label = r'$|E|$'
  207. # Eabs = np.real(Hc[:, 0])
  208. # label = r'$Re(H_x)$'
  209. # Eabs = np.real(Hc[:, 1])
  210. # label = r'$Re(H_y)$'
  211. # Eabs = np.real(Ec[:, 1])
  212. # label = r'$Re(E_y)$'
  213. # Eabs = np.real(Ec[:, 0])
  214. # label = r'$Re(E_x)$'
  215. Eabs_data = np.resize(Eabs, (npts, npts)).T
  216. #Eabs_data = np.flipud(np.resize(Eabs, (npts, npts)))
  217. elif field_to_plot == 'Habs':
  218. Habs = np.sqrt(Hr[:, 0]**2 + Hr[:, 1]**2 + Hr[:, 2]**2)
  219. Eabs_data = np.resize(Habs, (npts, npts)).T
  220. label = r'$|H|$'
  221. elif field_to_plot == 'angleEx':
  222. Eangle = np.angle(Ec[:, 0]) / np.pi * 180
  223. Eabs_data = np.resize(Eangle, (npts, npts)).T
  224. label = r'$arg(E_x)$'
  225. elif field_to_plot == 'angleHy':
  226. Hangle = np.angle(Hc[:, 1]) / np.pi * 180
  227. Eabs_data = np.resize(Hangle, (npts, npts)).T
  228. label = r'$arg(H_y)$'
  229. # Rescale to better show the axes
  230. scale_x = np.linspace(
  231. min(coordX) * WL / 2.0 / np.pi, max(coordX) * WL / 2.0 / np.pi, npts)
  232. scale_z = np.linspace(
  233. min(coordZ) * WL / 2.0 / np.pi, max(coordZ) * WL / 2.0 / np.pi, npts)
  234. # Define scale ticks
  235. min_tick = np.amin(Eabs_data[~np.isnan(Eabs_data)])
  236. #min_tick = 0.1
  237. max_tick = np.amax(Eabs_data[~np.isnan(Eabs_data)])
  238. #max_tick = 60
  239. scale_ticks = np.linspace(min_tick, max_tick, 5)
  240. #scale_ticks = np.power(10.0, np.linspace(np.log10(min_tick), np.log10(max_tick), 6))
  241. #scale_ticks = [0.1,0.3,1,3,10, max_tick]
  242. # Interpolation can be 'nearest', 'bilinear' or 'bicubic'
  243. #ax.set_title(label)
  244. # build a rectangle in axes coords
  245. ax.annotate(subplot_label, xy=(0.0, 1.1), xycoords='axes fraction', # fontsize=10,
  246. horizontalalignment='left', verticalalignment='top')
  247. # ax.text(right, top, subplot_label,
  248. # horizontalalignment='right',
  249. # verticalalignment='bottom',
  250. # transform=ax.transAxes)
  251. cax = ax.imshow(Eabs_data
  252. #, interpolation='nearest'
  253. , interpolation='none'
  254. #, interpolation='quadric'
  255. , cmap=cm.jet,
  256. origin='lower', vmin=min_tick, vmax=max_tick, extent=(min(scale_x), max(scale_x), min(scale_z), max(scale_z))
  257. # ,norm = LogNorm()
  258. )
  259. ax.axis("image")
  260. # # Add colorbar
  261. # cbar = fig.colorbar(cax, ticks=[a for a in scale_ticks], ax=ax
  262. # #,fraction=0.45
  263. # )
  264. # # vertically oriented colorbar
  265. # if 'angle' in field_to_plot:
  266. # cbar.ax.set_yticklabels(['%3.0f' % (a) for a in scale_ticks])
  267. # else:
  268. # cbar.ax.set_yticklabels(['%g' % (a) for a in scale_ticks])
  269. # # pos = list(cbar.ax.get_position().bounds)
  270. # #fig.text(pos[0] - 0.02, 0.925, '|E|/|E$_0$|', fontsize = 14)
  271. lp2 = -10.0
  272. lp1 = -1.0
  273. if crossplane == 'XZ':
  274. ax.set_xlabel('Z, ' + WL_units, labelpad=lp1)
  275. ax.set_ylabel('X, ' + WL_units, labelpad=lp2)
  276. elif crossplane == 'YZ':
  277. ax.set_xlabel('Z, ' + WL_units, labelpad=lp1)
  278. ax.set_ylabel('Y, ' + WL_units, labelpad=lp2)
  279. elif crossplane=='XYZ':
  280. ax.set_xlabel(r'$Z,$'+WL_units)
  281. ax.set_ylabel(r'$Y:X,$'+WL_units)
  282. elif crossplane == 'XY':
  283. ax.set_xlabel('X, ' + WL_units, labelpad=lp1)
  284. ax.set_ylabel('Y, ' + WL_units, labelpad=lp2)
  285. # # This part draws the nanoshell
  286. from matplotlib import patches
  287. from matplotlib.path import Path
  288. for xx in x:
  289. r = xx * WL / 2.0 / np.pi
  290. s1 = patches.Arc((0, 0), 2.0 * r, 2.0 * r, angle=0.0, zorder=1.8,
  291. theta1=0.0, theta2=360.0, linewidth=outline_width*1.8, color='black')
  292. ax.add_patch(s1)
  293. #
  294. # for flow in range(0,flow_total):
  295. # flow_x, flow_z = GetFlow(scale_x, scale_z, Ec, Hc,
  296. # min(scale_x)+flow*(scale_x[-1]-scale_x[0])/(flow_total-1),
  297. # min(scale_z),
  298. # #0.0,
  299. # npts*16)
  300. # verts = np.vstack((flow_z, flow_x)).transpose().tolist()
  301. # #codes = [Path.CURVE4]*len(verts)
  302. # codes = [Path.LINETO]*len(verts)
  303. # codes[0] = Path.MOVETO
  304. # path = Path(verts, codes)
  305. # patch = patches.PathPatch(path, facecolor='none', lw=1, edgecolor='yellow')
  306. # ax.add_patch(patch)
  307. if (not crossplane == 'XY') and flow_total > 0:
  308. from matplotlib.path import Path
  309. scanSP = np.linspace(-factor * x[-1], factor * x[-1], npts)
  310. min_SP = -factor * x[-1]
  311. step_SP = 2.0 * factor * x[-1] / (flow_total - 1)
  312. x0, y0, z0 = 0, 0, 0
  313. max_length = factor * x[-1] * 10
  314. # max_length=factor*x[-1]*5
  315. max_angle = np.pi / 160
  316. if is_flow_extend:
  317. rg = range(0, flow_total * 5 + 1)
  318. else:
  319. rg = range(0, flow_total)
  320. for flow in rg:
  321. if is_flow_extend:
  322. f = min_SP*2 + flow*step_SP
  323. else:
  324. f = min_SP + flow*step_SP
  325. if crossplane=='XZ':
  326. x0 = f
  327. elif crossplane=='YZ':
  328. y0 = f
  329. elif crossplane=='XYZ':
  330. x0 = 0
  331. y0 = 0
  332. if f > 0:
  333. x0 = f
  334. else:
  335. y0 = f
  336. z0 = min_SP
  337. # x0 = x[-1]/20
  338. flow_xSP, flow_ySP, flow_zSP = GetFlow3D(
  339. x0, y0, z0, max_length, max_angle, x, m, pl)
  340. if crossplane == 'XZ':
  341. flow_z_plot = flow_zSP * WL / 2.0 / np.pi
  342. flow_f_plot = flow_xSP * WL / 2.0 / np.pi
  343. elif crossplane == 'YZ':
  344. flow_z_plot = flow_zSP * WL / 2.0 / np.pi
  345. flow_f_plot = flow_ySP * WL / 2.0 / np.pi
  346. elif crossplane=='XYZ':
  347. if f > 0:
  348. flow_z_plot = flow_zSP*WL/2.0/np.pi
  349. flow_f_plot = flow_xSP*WL/2.0/np.pi
  350. else:
  351. flow_z_plot = flow_zSP*WL/2.0/np.pi
  352. flow_f_plot = flow_ySP*WL/2.0/np.pi
  353. verts = np.vstack(
  354. (flow_z_plot, flow_f_plot)).transpose().tolist()
  355. codes = [Path.LINETO] * len(verts)
  356. codes[0] = Path.MOVETO
  357. path = Path(verts, codes)
  358. #patch = patches.PathPatch(path, facecolor='none', lw=0.2, edgecolor='white',zorder = 2.7)
  359. patch = patches.PathPatch(
  360. path, facecolor='none', lw=outline_width, edgecolor='white', zorder=1.9, alpha=0.7)
  361. # patch = patches.PathPatch(
  362. # path, facecolor='none', lw=0.7, edgecolor='white', zorder=1.9, alpha=0.7)
  363. ax.add_patch(patch)
  364. # ax.plot(flow_z_plot, flow_f_plot, 'x', ms=2, mew=0.1,
  365. # linewidth=0.5, color='k', fillstyle='none')
  366. finally:
  367. terms, Qext, Qsca, Qabs, Qbk, Qpr, g, Albedo, S1, S2 = scattnlay(
  368. np.array([x]), np.array([m]))
  369. print("Qsca = " + str(Qsca))
  370. if isStream == True and field_to_plot in ["Eabs", "Habs"]:
  371. if field_to_plot == "Eabs": field = np.real(Ec)
  372. else: field = np.real(Hc)
  373. V = field[:,2].reshape((npts, npts)).T # z-component
  374. if crossplane == "XZ": U = field[:,0].reshape((npts, npts)).T
  375. if crossplane == "YZ": U = field[:,1].reshape((npts, npts)).T
  376. #print(U[:,0])
  377. X = coordX.reshape((npts, npts))[0,:]*WL/2.0/np.pi
  378. Z = coordZ.reshape((npts, npts))[:,0]*WL/2.0/np.pi
  379. lw = np.sqrt((V**2+U**2)/(np.max(V**2+U**2)))
  380. pts2 = npts*3.0
  381. ax.streamplot(X, Z, V, U, color="white", linewidth=lw*4.0+1.2,
  382. #cmap=plt.cm.inferno,
  383. density=1.1, arrowstyle='->', arrowsize=1.5
  384. ,minlength=minlength
  385. ,maxlength=25
  386. , zorder=1.3
  387. )
  388. #ax.quiver(X[::15], Z[::15], V[::15,::15], U[::15,::15], units='width',color="white")
  389. #