App.vue 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883
  1. <template>
  2. <div id="app">
  3. <b-loading :active.sync="isLoading" :can-cancel="isLoading"/>
  4. <ShowInfo/>
  5. <section class="add_padding">
  6. <GetUnits v-bind:units="units"
  7. v-bind:source_units="source_units"
  8. v-bind:isSourceOtherUnits="isSourceOtherUnits"
  9. @unitsData="units=$event"
  10. @source_unitsData="source_units=$event"
  11. @isSourceOtherUnitsData="isSourceOtherUnits=$event"
  12. />
  13. <GetHostIndex v-bind:hostIndex="simulationSetup.hostIndex"
  14. @hostIndexData="simulationSetup.hostIndex=$event" />
  15. <GetSourceParameters v-bind:fromWL="simulationSetup.fromWL"
  16. v-bind:toWL="simulationSetup.toWL"
  17. v-bind:stepWL="simulationSetup.stepWL"
  18. v-bind:source_units="source_units"
  19. @fromWLData="simulationSetup.fromWL=$event"
  20. @toWLData="simulationSetup.toWL=$event"
  21. @stepWLData="simulationSetup.stepWL=$event"
  22. @source_unitsData="source_units=$event"
  23. />
  24. <GetMaterials v-bind:materials="materials"
  25. v-bind:plot_width="chart.layout.width"
  26. v-bind:plot_height="chart.layout.height"
  27. />
  28. <GetParticleParameters v-bind:layers="simulationSetup.layers"
  29. v-bind:units="units"
  30. v-bind:materials="materials"
  31. />
  32. <div class="field is-horizontal">
  33. <div class="field-label is-normal">
  34. <label class="label">Modes to plot</label>
  35. </div>
  36. <div class="field-body" >
  37. <b-field label="">
  38. <b-input v-model="simulationSetup.total_mode_n" type='number' min=1 style="width:7rem"/>
  39. </b-field>
  40. </div>
  41. </div>
  42. <div class="add_padding">
  43. <b-button class="is-primary is-medium" @click="runSimulation();">
  44. Run simulation
  45. </b-button><br>
  46. It took {{ ttime }} s.
  47. </div>
  48. <div class="field is-grouped is-grouped-multiline add_padding">
  49. <b-switch v-model="plotSelector.isPlotQsca">
  50. Qsca
  51. </b-switch>
  52. <b-switch v-model="plotSelector.isPlotQabs">
  53. Qabs
  54. </b-switch>
  55. <b-switch v-model="plotSelector.isPlotQext">
  56. Qext
  57. </b-switch>
  58. </div>
  59. <!-- <b-table :data="plotSelectorData" :columns="plotSelectorColumns" :mobile-cards="isShowInfo"></b-table>-->
  60. <table class="table is-narrow add_padding">
  61. <thead>
  62. <tr>
  63. <th v-for="mode_name in simulationRuntime.mode_n_names" v-bind:key="mode_name.name">
  64. {{ mode_name.name == 'type' ? '': mode_name.name}}
  65. </th>
  66. </tr>
  67. </thead>
  68. <tbody>
  69. <tr>
  70. <th>E</th>
  71. <td v-for="(mode,index) in plotSelector.isPlotModeE" v-bind:key="index">
  72. <template v-if="(simulationRuntime.total_mode_n_evaluated - 1) < index">
  73. <b-checkbox v-model="plotSelector.isPlotModeE[index]" disabled> </b-checkbox>
  74. </template>
  75. <template v-else>
  76. <b-checkbox v-model="plotSelector.isPlotModeE[index]"> </b-checkbox>
  77. </template>
  78. </td>
  79. </tr>
  80. <tr>
  81. <th>H</th>
  82. <td v-for="(mode,index) in plotSelector.isPlotModeH" v-bind:key="index">
  83. <template v-if="(simulationRuntime.total_mode_n_evaluated - 1) < index">
  84. <b-checkbox v-model="plotSelector.isPlotModeH[index]" disabled> </b-checkbox>
  85. </template>
  86. <template v-else>
  87. <b-checkbox v-model="plotSelector.isPlotModeH[index]"> </b-checkbox>
  88. </template>
  89. </td>
  90. </tr>
  91. </tbody>
  92. </table>
  93. </section>
  94. <div class="chart-container">
  95. <reactive-chart :chart="chart"/>
  96. </div>
  97. </div>
  98. </template>
  99. <script>
  100. // You should put *.wasm to public/ and *.js to src/ folder
  101. // To compile fibbonacci example use
  102. // emcc -O3 -s WASM=1 -s EXTRA_EXPORTED_RUNTIME_METHODS='["cwrap"]' -s ALLOW_MEMORY_GROWTH=1 -s MODULARIZE=1 -s 'EXPORT_NAME="fibonacci"' -o ./fibonacci.js fibonacci.c
  103. // for and example from https://gist.github.com/ashleygwilliams/32c31a3f5b8c87bf2894108b3534ee4f
  104. // import fibonacci from './fibonacci.js';
  105. // const module = fibonacci({
  106. // locateFile(path) {
  107. // console.log(path);
  108. // return path;
  109. // }
  110. // });
  111. // module.onRuntimeInitialized = () => {
  112. // console.log(module._fib(12));
  113. // };
  114. // // Test the size of wasm file
  115. // fetch('nmiejs.wasm'
  116. // ).then(response =>
  117. // response.arrayBuffer()
  118. // ).then(bytes =>
  119. // console.log(bytes)
  120. // );
  121. import { mapState } from 'vuex'
  122. import nmiejs from './nmiejs.js';
  123. // Test nmiejs if working
  124. (async () => {
  125. const module = await nmiejs({
  126. locateFile(path) {
  127. let deploy_path = process.env.BASE_URL;
  128. // '/themes/custom/physics/mie/';
  129. // console.log();
  130. // // let deploy_path = '';
  131. console.log(deploy_path + path);
  132. return deploy_path + path;
  133. }
  134. })
  135. const nmie = new module.nmie();
  136. nmie.ClearTarget();
  137. let R = 100.0;
  138. let reN = 4.0;
  139. let imN = 0.01;
  140. nmie.AddTargetLayerReIm(R, reN, imN)
  141. nmie.SetModeNmaxAndType(-1, -1);
  142. let WL = 800;
  143. nmie.SetWavelength(WL);
  144. nmie.RunMieCalculation();
  145. console.log(nmie.GetQsca());
  146. // outer_arc_points, radius_points, from_Rho, to_Rho,
  147. // from_Theta, to_Theta, from_Phi, to_Phi, isIgnoreAvailableNmax
  148. nmie.RunFieldCalculationPolar(2, 2, 0.1, 1.5, 0, 3.1415, 0, 3.1415, 0);
  149. console.log("Field Eabs:", nmie.GetFieldEabs());
  150. })();
  151. const range = (start, stop, step = 1) => Array(Math.ceil((stop - start) / step)).fill(start).map((x, y) => x + y * step);
  152. function rangeInt(size, startAt = 0) {
  153. return [...Array(size).keys()].map(i => i + startAt);
  154. }
  155. import ReactiveChart from "./components/ReactiveChart.vue";
  156. import ShowInfo from "./components/ShowInfo.vue";
  157. import GetMaterials from "./components/GetMaterials.vue";
  158. import GetHostIndex from "./components/GetHostIndex.vue";
  159. import GetUnits from "./components/GetUnits.vue";
  160. import GetSourceParameters from "./components/GetSourceParameters.vue";
  161. import GetParticleParameters from "./components/GetParticleParameters.vue";
  162. export default {
  163. name: 'app',
  164. components: {
  165. GetHostIndex,
  166. GetMaterials,
  167. GetSourceParameters,
  168. GetParticleParameters,
  169. GetUnits,
  170. ReactiveChart,
  171. ShowInfo
  172. },
  173. computed:
  174. mapState([
  175. 'simulationSetup'
  176. ]),
  177. data() {
  178. return {
  179. nmie: undefined,
  180. ttime: "",
  181. window: {
  182. width: 0,
  183. height: 0
  184. },
  185. units: 'nm',
  186. units_prev: 'nm',
  187. source_units: 'nm',
  188. source_units_prev: 'nm',
  189. materials: [],
  190. isSourceOtherUnits: false,
  191. simulationRuntime: {
  192. r_units: 'nm',
  193. r_source_units: 'nm',
  194. stepWL: 2,
  195. fromWL: 300.0,
  196. toWL: 1000.0,
  197. layers: [
  198. {
  199. R: 100.0,
  200. material: 'nk',
  201. reN: 4.0,
  202. imN: 0.01,
  203. index:0
  204. }
  205. ],
  206. mode_n: [],
  207. mode_n_names: [],
  208. mode_types: range(0, 2),
  209. mode_names: ['E', 'H'],
  210. WLs: [],
  211. Qsca: [],
  212. Qabs: [],
  213. Qext: [],
  214. Qsca_n: [[], []],
  215. Qabs_n: [[], []],
  216. layout: {},
  217. trace1: {},
  218. trace2: {},
  219. total_mode_n_evaluated: 4,
  220. },
  221. plotData: {
  222. pWLs: [],
  223. pQsca: [],
  224. pQabs: [],
  225. pQext: [],
  226. pQsca_n: [[], []],
  227. pQabs_n: [[], []],
  228. },
  229. changes: 0,
  230. isShowInfo: false,
  231. isLoading: true,
  232. isRunning: false,
  233. plotSelector: {
  234. isPlotModeE: [],
  235. isPlotModeH: [],
  236. isPlotQabs: true,
  237. isPlotQsca: true,
  238. isPlotQext: false,
  239. },
  240. // plotSelectorData: undefined,
  241. // plotSelectorColumns: undefined,
  242. // plotSelectorNames: undefined,
  243. chart: {
  244. uuid: "spectra",
  245. traces: [
  246. {
  247. y: [],
  248. line: {
  249. color: "#5e9e7e",
  250. width: 4,
  251. shape: "line"
  252. }
  253. }
  254. ],
  255. layout: {
  256. margin: {
  257. l:0,
  258. r:40,
  259. b:50,
  260. t:0
  261. },
  262. // paper_bgcolor: '#7f7f7f',
  263. // plot_bgcolor: '#c7c7c7',
  264. // title: 'reactive charts',
  265. xaxis: {
  266. // will be set on mount
  267. title: ''
  268. },
  269. yaxis: {
  270. title: 'Normalized cross-sections'
  271. },
  272. showlegend: true,
  273. legend: {
  274. orientation:"h",
  275. x: -.1,
  276. y: 1.05
  277. },
  278. // width: 100,
  279. height: 100
  280. }
  281. }
  282. };
  283. },
  284. created() {
  285. window.addEventListener('resize', this.handleResize);
  286. this.handleResize();
  287. },
  288. destroyed() {
  289. window.removeEventListener('resize', this.handleResize)
  290. },
  291. mounted() {
  292. this.setXaxisTitle();
  293. (async () => {
  294. const module = await nmiejs({
  295. locateFile(path) {
  296. let deploy_path = process.env.BASE_URL;
  297. // '/themes/custom/physics/mie/';
  298. // console.log();
  299. // // let deploy_path = '';
  300. console.log(deploy_path + path);
  301. return deploy_path + path;
  302. }
  303. })
  304. this.nmie = new module.nmie();
  305. this.isLoading = false;
  306. this.runMie();
  307. this.setIsPlotMode();
  308. this.setModeNNames();
  309. this.plotResults();
  310. })();
  311. },
  312. watch: {
  313. 'simulationSetup.layers': {
  314. handler: function () {
  315. this.checkConflictsWL();
  316. },
  317. deep: true
  318. },
  319. 'simulationSetup.toWL': {
  320. handler: function () {
  321. for (let i = 0; i < this.simulationSetup.layers.length; i++) {
  322. this.simulationSetup.layers[i].isMaterial_hasConflict = false;
  323. }
  324. this.checkConflictsWL();
  325. }
  326. },
  327. 'simulationSetup.fromWL': {
  328. handler: function () {
  329. for (let i = 0; i < this.simulationSetup.layers.length; i++) {
  330. this.simulationSetup.layers[i].isMaterial_hasConflict = false;
  331. }
  332. this.checkConflictsWL();
  333. }
  334. },
  335. plotSelector: {
  336. handler: function () {
  337. this.plotResults();
  338. },
  339. deep: true
  340. },
  341. 'simulationSetup.total_mode_n': function () {
  342. this.setIsPlotMode();
  343. this.setModeNNames();
  344. },
  345. units: {
  346. handler: function () {
  347. if (!this.isSourceOtherUnits) {
  348. this.source_units = this.units;
  349. }
  350. let u = this.units;
  351. let u_prev = this.units_prev;
  352. let layers = this.simulationSetup.layers;
  353. for (let i=0; i<layers.length; i++) {
  354. layers[i].R = this.convertUnits(u_prev, u, layers[i].R);
  355. }
  356. this.units_prev = this.units;
  357. }
  358. },
  359. source_units: {
  360. handler: function () {
  361. // this.$buefy.notification.open({
  362. // duration: 4000,
  363. // message: 'Source parameters were restored from last simulation for correct plotting.',
  364. // type: 'is-info',
  365. // position: 'is-top',
  366. // });
  367. this.setXaxisTitle();
  368. let u = this.source_units;
  369. let u_prev = this.source_units_prev;
  370. for (let i=0; i<this.plotData.pWLs.length; i++) {
  371. this.plotData.pWLs[i] = this.convertUnits(u_prev, u,
  372. this.plotData.pWLs[i]);
  373. }
  374. this.simulationSetup.fromWL = this.convertUnits(u_prev, u,
  375. this.simulationSetup.fromWL);
  376. this.simulationSetup.toWL = this.convertUnits(u_prev, u,
  377. this.simulationSetup.toWL);
  378. if (this.simulationSetup.fromWL > this.simulationSetup.toWL) {
  379. let tmp = this.simulationSetup.fromWL;
  380. this.simulationSetup.fromWL = this.simulationSetup.toWL;
  381. this.simulationSetup.toWL = tmp;
  382. }
  383. this.simulationSetup.stepWL = (this.simulationSetup.toWL-this.simulationSetup.fromWL)
  384. /this.plotData.pWLs.length;
  385. this.source_units_prev = this.source_units;
  386. // Update plotting
  387. this.plotSelector.isPlotQsca = !this.plotSelector.isPlotQsca
  388. setTimeout(
  389. () => {
  390. this.plotSelector.isPlotQsca = !this.plotSelector.isPlotQsca
  391. }, 2);
  392. }
  393. },
  394. isSourceOtherUnits: {
  395. handler: function () {
  396. // this.setEmptyChart();
  397. this.setXaxisTitle();
  398. if (!this.isSourceOtherUnits) {
  399. this.source_units = this.units;
  400. } else {
  401. this.source_units = 'THz'
  402. }
  403. }
  404. },
  405. window: {
  406. handler: function () {
  407. // this.chart.layout.width = this.window.width * 0.92;
  408. this.chart.layout.height = this.window.height * 0.95;
  409. // if (this.window.width < 600) this.chart.layout.width = this.window.width;
  410. if (this.window.height < 600) this.chart.layout.height = this.window.height;
  411. },
  412. deep: true
  413. }
  414. },
  415. methods: {
  416. checkConflictsWL (){
  417. this.simulationSetup.fromWL_hasConflict = false;
  418. this.simulationSetup.toWL_hasConflict = false;
  419. let fromWL = this.convertUnits2nm(this.source_units, this.simulationSetup.fromWL);
  420. let toWL = this.convertUnits2nm(this.source_units, this.simulationSetup.toWL);
  421. for (let i = 0; i < this.simulationSetup.layers.length; i++) {
  422. // console.log('checking WL conflict layer', i);
  423. if (this.simulationSetup.layers[i].material=='nk') continue;
  424. // console.log(fromWL, ' and ', this.simulationSetup.layers[i].spline_n.xs[0]);
  425. // console.log(toWL, ' and ', this.simulationSetup.layers[i]
  426. // .spline_n.xs[this.simulationSetup.layers[i].spline_n.xs.length-1] );
  427. if (fromWL < this.simulationSetup.layers[i].spline_n.xs[0]) {
  428. this.simulationSetup.fromWL_hasConflict = true;
  429. this.simulationSetup.layers[i].isMaterial_hasConflict = true;
  430. }
  431. if ( toWL > this.simulationSetup.layers[i]
  432. .spline_n.xs[this.simulationSetup.layers[i].spline_n.xs.length-1] ) {
  433. this.simulationSetup.toWL_hasConflict = true;
  434. this.simulationSetup.layers[i].isMaterial_hasConflict = true;
  435. }
  436. }
  437. },
  438. handleResize() {
  439. this.window.width = window.innerWidth;
  440. this.window.height = window.innerHeight*0.8;
  441. },
  442. convertUnits2nm(fromU, val) {
  443. if (fromU === 'nm') return val;
  444. if (fromU === 'mkm') return val*1e3;
  445. if (fromU === 'mm') return val*1e6;
  446. if (fromU === 'cm') return val*1e7;
  447. if (fromU === 'm') return val*1e9;
  448. if (fromU === 'km') return val*1e12;
  449. let c = 299792458; // m/s
  450. let hc = 1239841930.092394328e-15; // m*eV
  451. if (fromU === 'THz') return c/(val*1e12)*1e9;
  452. if (fromU === 'GHz') return c/(val*1e9)*1e9;
  453. if (fromU === 'MHz') return c/(val*1e6)*1e9;
  454. if (fromU === 'kHz') return c/(val*1e3)*1e9;
  455. if (fromU === 'Hz') return c/(val*1e0)*1e9;
  456. if (fromU === 'eV') return hc/(val*1e0)*1e9;
  457. if (fromU === 'meV') return hc/(val/1e3)*1e9;
  458. if (fromU === 'fs') return (val/1e12)/c*1e9;
  459. if (fromU === 'ps') return (val/1e15)/c*1e9;
  460. return undefined;
  461. },
  462. convertUnitsFrom_nm(toU, val) {
  463. if (toU === 'nm') return val;
  464. if (toU === 'mkm') return val/1e3;
  465. if (toU === 'mm') return val/1e6;
  466. if (toU === 'cm') return val/1e7;
  467. if (toU === 'm') return val/1e9;
  468. if (toU === 'km') return val/1e12;
  469. let c = 299792458; // m/s
  470. let hc = 1239841930.092394328e-15; // m*eV
  471. if (toU === 'THz') return c/(val/1e9)/1e12;
  472. if (toU === 'GHz') return c/(val/1e9)/1e9;
  473. if (toU === 'MHz') return c/(val/1e9)/1e6;
  474. if (toU === 'kHz') return c/(val/1e9)/1e3;
  475. if (toU === 'Hz') return c/(val/1e9)/1e0;
  476. if (toU === 'eV') return hc/(val/1e9);
  477. if (toU === 'meV') return hc/(val/1e9)*1e3;
  478. if (toU === 'fs') return (val/1e9)/c*1e12;
  479. if (toU === 'ps') return (val/1e9)/c*1e15;
  480. return undefined;
  481. },
  482. convertUnits(fromU,toU, val) {
  483. if (fromU === toU) return val;
  484. return this.convertUnitsFrom_nm(toU, this.convertUnits2nm(fromU, val));
  485. },
  486. notifyDanger: function(message) {
  487. this.$buefy.notification.open({
  488. duration: 3000,
  489. message: message,
  490. type: 'is-danger',
  491. position: 'is-bottom-left',
  492. });
  493. },
  494. runSimulation: function() {
  495. if (this.simulationSetup.toWL_hasConflict ||
  496. this.simulationSetup.fromWL_hasConflict) {
  497. this.$buefy.notification.open({
  498. duration: 2000,
  499. message: 'Simulation has input conflicts, fixing...',
  500. type: 'is-danger',
  501. position: 'is-bottom-left',
  502. });
  503. }
  504. for (let i = 0; i < this.simulationSetup.layers.length; i++) {
  505. if (this.simulationSetup.layers[i].material=='nk') continue;
  506. let fromWLnm = this.convertUnits2nm(this.source_units, this.simulationSetup.fromWL);
  507. let toWLnm = this.convertUnits2nm(this.source_units, this.simulationSetup.toWL);
  508. let fromWL_material = this.simulationSetup.layers[i].spline_n.xs[0];
  509. if (fromWLnm < fromWL_material ) {
  510. this.simulationSetup.fromWL = this.convertUnitsFrom_nm(
  511. this.source_units, fromWL_material);
  512. }
  513. let toWL_material = this.simulationSetup.layers[i]
  514. .spline_n.xs[this.simulationSetup.layers[i].spline_n.xs.length-1];
  515. if ( toWLnm > toWL_material ) {
  516. this.simulationSetup.toWL = this.convertUnitsFrom_nm(
  517. this.source_units, toWL_material);
  518. }
  519. }
  520. this.$buefy.notification.open({
  521. duration: 200,
  522. message: 'Simulation was started!',
  523. type: 'is-danger',
  524. position: 'is-bottom-left',
  525. });
  526. setTimeout(
  527. () => {
  528. this.runMie();
  529. this.plotResults();
  530. this.$buefy.notification.open({
  531. duration: 3000,
  532. message: 'Finished! '+"It took " + this.ttime + " s.",
  533. type: 'is-success',
  534. position: 'is-bottom-left',
  535. })
  536. ;
  537. }, 20);
  538. },
  539. runMie: function () {
  540. const math = require('mathjs')
  541. this.simulationRuntime.r_units = this.units;
  542. this.simulationRuntime.r_source_units = this.source_units;
  543. this.simulationRuntime.fromWL = math.evaluate(this.simulationSetup.fromWL);
  544. this.simulationRuntime.toWL = math.evaluate(this.simulationSetup.toWL);
  545. this.simulationRuntime.stepWL = this.simulationSetup.stepWL;
  546. this.simulationRuntime.layers =this.simulationSetup.layers; // TODO: not a copy, but we need spline_n.at() method in every layer.
  547. let t0 = performance.now();
  548. let fromWL = parseFloat(this.simulationRuntime.fromWL);
  549. let toWL = parseFloat(this.simulationRuntime.toWL);
  550. let stepWL = parseFloat(this.simulationRuntime.stepWL);
  551. let host = parseFloat(this.simulationSetup.hostIndex);
  552. let Qsca = [], Qabs = [], Qext = [];
  553. let Qsca_n = [[], []], Qabs_n = [[], []];
  554. let WLs = range(fromWL, toWL, stepWL);
  555. this.simulationRuntime.WLs = WLs;
  556. let total_mode_n = this.simulationSetup.total_mode_n;
  557. let mode_n = [];
  558. mode_n = rangeInt(Number(total_mode_n), 1);
  559. this.simulationRuntime.total_mode_n_evaluated = total_mode_n;
  560. let mode_types = range(0, 2);
  561. this.simulationSetup.mode_n = mode_n;
  562. this.simulationSetup.mode_types = mode_types;
  563. mode_types.forEach(function (mode_type) {
  564. mode_n.forEach(function () {
  565. Qsca_n[mode_type].push([]);
  566. Qabs_n[mode_type].push([]);
  567. });
  568. });
  569. // Provide all sizes (germetry and wavelengths) to nmie in same units [nm]
  570. const nmie = this.nmie;
  571. // nmie.ClearTarget();
  572. // nmie.AddTargetLayerReIm( this.convertUnits2nm(this.units, R)*host,
  573. // reN/host, imN/host);
  574. let WLs_nm = range(fromWL, toWL, stepWL);
  575. let WL_points = WLs_nm.length;
  576. for (let i = 0; i < WL_points; i++) {
  577. let WL = this.convertUnits2nm(this.source_units, WLs[i]);
  578. nmie.ClearTarget();
  579. for (let num_layer = 0;
  580. num_layer < this.simulationRuntime.layers.length;
  581. num_layer++) {
  582. let layer = this.simulationRuntime.layers[num_layer];
  583. let R = parseFloat(layer.R);
  584. if (layer.material === 'PEC') {
  585. // TODO: set PEC layer
  586. continue;
  587. }
  588. if (layer.material !== 'nk') {
  589. if ( layer.spline_n === undefined ) {
  590. this.notifyDanger('Failed to load material!');
  591. return;
  592. }
  593. if (layer.spline_n.xs[0] > WL
  594. || layer.spline_n.xs[layer.spline_n.xs.length-1] < WL) {
  595. this.notifyDanger('ERROR!!! Source parameters are out of material spectral range!');
  596. layer.isMaterial_hasConflict = true;
  597. this.simulationSetup.toWL_hasConflict = true;
  598. this.simulationSetup.fromWL_hasConflict = true;
  599. return;
  600. }
  601. layer.reN = layer.spline_n.at(WL);
  602. layer.imN = layer.spline_k.at(WL);
  603. }
  604. let reN = parseFloat(layer.reN);
  605. let imN = parseFloat(layer.imN);
  606. nmie.AddTargetLayerReIm( this.convertUnits2nm(this.units, R)*host,
  607. reN/host, imN/host);
  608. }
  609. nmie.SetModeNmaxAndType(-1, -1);
  610. nmie.SetWavelength(WL);
  611. nmie.RunMieCalculation();
  612. Qsca.push(nmie.GetQsca());
  613. Qabs.push(nmie.GetQabs());
  614. Qext.push(nmie.GetQsca()+nmie.GetQabs());
  615. mode_types.forEach(function (mode_type) {
  616. mode_n.forEach(function (n) {
  617. nmie.SetModeNmaxAndType(n, mode_type);
  618. nmie.RunMieCalculation();
  619. Qsca_n[mode_type][n - 1].push(nmie.GetQsca());
  620. Qabs_n[mode_type][n - 1].push(nmie.GetQabs());
  621. });
  622. });
  623. }
  624. this.simulationRuntime.Qsca = Qsca;
  625. this.simulationRuntime.Qabs = Qabs;
  626. this.simulationRuntime.Qext = Qext;
  627. this.simulationRuntime.Qsca_n = Qsca_n;
  628. this.simulationRuntime.Qabs_n = Qabs_n;
  629. let t1 = performance.now();
  630. this.ttime = ((t1 - t0) / 1000).toFixed(2);
  631. // console.log("It took " + this.ttime + " s.");
  632. this.plotData.pWLs = WLs.slice();
  633. this.plotData.pQsca = Qsca;
  634. this.plotData.pQabs = Qabs;
  635. this.plotData.pQext = Qext;
  636. this.plotData.pQsca_n = Qsca_n;
  637. this.plotData.pQabs_n = Qabs_n;
  638. this.changes++;
  639. },
  640. setIsPlotMode: function () {
  641. let total_mode_n = this.simulationSetup.total_mode_n;
  642. total_mode_n++;
  643. let np1 = Number(total_mode_n);
  644. let mode_n = rangeInt(Number(np1), 0);
  645. let modeE = [];
  646. let modeH = [];
  647. mode_n.forEach(function (n) {
  648. if (n !== 0) {
  649. modeE.push(false);
  650. modeH.push(false);
  651. }
  652. });
  653. let saved_modes = this.simulationSetup.total_mode_n > this.simulationRuntime.total_mode_n_evaluated ?
  654. this.simulationRuntime.total_mode_n_evaluated : this.simulationSetup.total_mode_n;
  655. this.simulationRuntime.total_mode_n_evaluated = saved_modes;
  656. let oldModeE = this.plotSelector.isPlotModeE;
  657. let oldModeH = this.plotSelector.isPlotModeH;
  658. let ii;
  659. for (ii = 0; ii < saved_modes; ++ii) {
  660. if (ii < oldModeE.length) {
  661. modeE[ii] = oldModeE[ii];
  662. modeH[ii] = oldModeH[ii];
  663. }
  664. }
  665. this.plotSelector.isPlotModeE = modeE;
  666. this.plotSelector.isPlotModeH = modeH;
  667. },
  668. setXaxisTitle: function () {
  669. if (this.source_units.endsWith('Hz')) {
  670. this.chart.layout.xaxis.title = "Frequency, " + this.source_units;
  671. } else if (this.source_units.endsWith('eV')) {
  672. this.chart.layout.xaxis.title = "Energy, " + this.source_units;
  673. } else if (this.source_units.endsWith('s')) {
  674. this.chart.layout.xaxis.title = "Period, " + this.source_units;
  675. } else {
  676. this.chart.layout.xaxis.title = "Wavelength, " + this.source_units;
  677. }
  678. },
  679. setModeNNames: function () {
  680. let total_mode_n = this.simulationSetup.total_mode_n;
  681. total_mode_n++;
  682. let np1 = Number(total_mode_n);
  683. let mode_n = rangeInt(Number(np1), 0);
  684. let mode_n_names = [];
  685. mode_n.forEach(function (n) {
  686. if (n === 0) mode_n_names.push({id: 0, name: 'type'});
  687. else if (n === 1) mode_n_names.push({id: 1, name: 'dipole'});
  688. else if (n === 2) mode_n_names.push({id: 2, name: 'quadrupole'});
  689. else if (n === 3) mode_n_names.push({id: 3, name: 'octupole'});
  690. else mode_n_names.push({id: n, name: (Math.pow(2, n)).toString()});
  691. });
  692. this.simulationRuntime.mode_n_names = mode_n_names;
  693. },
  694. setQtotalChart: function () {
  695. let traceQsca, traceQabs, traceQext;
  696. traceQsca = {
  697. x: this.plotData.pWLs,
  698. y: this.plotData.pQsca,
  699. type: 'scatter',
  700. name: 'Qsca'
  701. };
  702. traceQabs = {
  703. x: this.plotData.pWLs,
  704. y: this.plotData.pQabs,
  705. type: 'scatter',
  706. name: 'Qabs'
  707. };
  708. traceQext = {
  709. x: this.plotData.pWLs,
  710. y: this.plotData.pQext,
  711. type: 'scatter',
  712. name: 'Qext'
  713. };
  714. this.chart.traces = [];
  715. if (this.plotSelector.isPlotQsca === true) this.chart.traces.push(traceQsca);
  716. if (this.plotSelector.isPlotQabs === true) this.chart.traces.push(traceQabs);
  717. if (this.plotSelector.isPlotQext === true) this.chart.traces.push(traceQext);
  718. },
  719. plotResults: function () {
  720. this.setQtotalChart();
  721. let mode_type, mode_n;
  722. let mode_n_names = this.simulationRuntime.mode_n_names;
  723. let mode_names = this.simulationRuntime.mode_names;
  724. for (mode_type = 0; mode_type < 2; ++mode_type) {
  725. for (mode_n = 0; mode_n < this.simulationRuntime.total_mode_n_evaluated; ++mode_n) {
  726. let is_mode_plot = mode_type === 0 ? this.plotSelector.isPlotModeE : this.plotSelector.isPlotModeH;
  727. if (is_mode_plot[mode_n] === false) continue;
  728. if (this.plotSelector.isPlotQsca === true) {
  729. let trace_sca = {
  730. x: this.plotData.pWLs,
  731. y: this.plotData.pQsca_n[mode_type][mode_n],
  732. type: 'scatter',
  733. name: 'Qsca ' + mode_names[mode_type] + ' ' + mode_n_names[mode_n + 1].name
  734. };
  735. this.chart.traces.push(trace_sca);
  736. }
  737. if (this.plotSelector.isPlotQabs === true) {
  738. let trace_abs = {
  739. x: this.plotData.pWLs,
  740. y: this.plotData.pQabs_n[mode_type][mode_n],
  741. type: 'scatter',
  742. name: 'Qabs ' + mode_names[mode_type] + ' ' + mode_n_names[mode_n + 1].name
  743. };
  744. this.chart.traces.push(trace_abs);
  745. }
  746. }
  747. }
  748. }
  749. },
  750. };
  751. </script>
  752. <style lang="scss">
  753. // Import Bulma's core
  754. @import "~bulma/sass/utilities/_all";
  755. // https://bulma.io/documentation/customize/variables/
  756. // $body-background-color: $green;
  757. // Set your colors
  758. $primary: #7957d5;
  759. $twitter: #4099FF;
  760. $twitter-invert: findColorInvert($twitter);
  761. // Setup $colors to use as bulma classes (e.g. 'is-twitter')
  762. $colors: (
  763. "white": ($white, $black),
  764. "black": ($black, $white),
  765. "light": ($light, $light-invert),
  766. "dark": ($dark, $dark-invert),
  767. "primary": ($primary, $primary-invert),
  768. "info": ($info, $info-invert),
  769. "success": ($success, $success-invert),
  770. "warning": ($warning, $warning-invert),
  771. "danger": ($danger, $danger-invert),
  772. "twitter": ($twitter, $twitter-invert)
  773. );
  774. // Links
  775. // $link: $primary;
  776. // $link-invert: $primary-invert;
  777. // $link-focus-border: $primary;
  778. // Import Bulma and Buefy styles
  779. @charset "utf-8";
  780. /*! bulma.io v0.7.5 | MIT License | github.com/jgthms/bulma */
  781. @import "~bulma/sass/utilities/_all";
  782. @import "~bulma/sass/base/_all";
  783. /*@import "~bulma/sass/elements/_all";*/
  784. @import "~bulma/sass/elements/box.sass";
  785. @import "~bulma/sass/elements/button.sass";
  786. @import "~bulma/sass/elements/container.sass";
  787. @import "~bulma/sass/elements/content.sass";
  788. /*@import "~bulma/sass/elements/icon.sass";*/
  789. @import "~bulma/sass/elements/image.sass";
  790. @import "~bulma/sass/elements/notification.sass";
  791. @import "~bulma/sass/elements/progress.sass";
  792. @import "~bulma/sass/elements/table.sass";
  793. @import "~bulma/sass/elements/tag.sass";
  794. @import "~bulma/sass/elements/title.sass";
  795. @import "~bulma/sass/elements/other.sass";
  796. @import "~bulma/sass/form/_all";
  797. /*@import "~bulma/sass/components/_all";*/
  798. @import "~bulma/sass/components/dropdown";
  799. @import "~bulma/sass/components/tabs";
  800. @import "~bulma/sass/components/modal";
  801. @import "~bulma/sass/components/message";
  802. @import "~bulma/sass/grid/_all";
  803. /*@import "~bulma/sass/layout/_all";*/
  804. // @import "~bulma";
  805. @import "~buefy/src/scss/buefy";
  806. .add_padding {
  807. padding: 1rem;
  808. }
  809. // Custom styles to build into the design of physics.itmo.ru
  810. // npm run build && ./deploy.sh && xclip -sel c < dist/index.html
  811. .content table {
  812. width: auto;
  813. }
  814. ul {
  815. padding-left: 1.5em;
  816. }
  817. body {
  818. line-height: 1.42857;
  819. }
  820. /*.container {*/
  821. /* width: 750px;*/
  822. /*}*/
  823. h1 {
  824. font-size: 34px;
  825. font-weight: bold;
  826. }
  827. .block-views-blockmessages-header-message-block {
  828. top: 25px;
  829. }
  830. </style>