Source: core/sceneObjs/glass/CircleGlass.js

  1. /*
  2. * Copyright 2024 The Ray Optics Simulation authors and contributors
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. import BaseGlass from '../BaseGlass.js';
  17. import CircleObjMixin from '../CircleObjMixin.js';
  18. import i18next from 'i18next';
  19. import geometry from '../../geometry.js';
  20. /**
  21. * Glass of the shape of a circle.
  22. *
  23. * Tools -> Glass -> Circle
  24. * @class
  25. * @extends BaseGlass
  26. * @memberof sceneObjs
  27. * @property {Point} p1 - The center of the circle.
  28. * @property {Point} p2 - A point on the boundary of the circle.
  29. * @property {number} refIndex - The refractive index of the glass, or the Cauchy coefficient A of the glass if "Simulate Colors" is on.
  30. * @property {number} cauchyB - The Cauchy coefficient B of the glass if "Simulate Colors" is on, in micrometer squared.
  31. */
  32. class CircleGlass extends CircleObjMixin(BaseGlass) {
  33. static type = 'CircleGlass';
  34. static isOptical = true;
  35. static supportsSurfaceMerging = true;
  36. static serializableDefaults = {
  37. p1: null,
  38. p2: null,
  39. refIndex: 1.5,
  40. cauchyB: 0.004
  41. };
  42. populateObjBar(objBar) {
  43. objBar.setTitle(i18next.t('main:meta.parentheses', { main: i18next.t('main:tools.categories.glass'), sub: i18next.t('main:tools.CircleGlass.title') }));
  44. super.populateObjBar(objBar);
  45. }
  46. draw(canvasRenderer, isAboveLight, isHovered) {
  47. const ctx = canvasRenderer.ctx;
  48. const ls = canvasRenderer.lengthScale;
  49. ctx.beginPath();
  50. ctx.arc(this.p1.x, this.p1.y, geometry.segmentLength(this), 0, Math.PI * 2, false);
  51. this.fillGlass(canvasRenderer, isAboveLight, isHovered);
  52. ctx.lineWidth = 1;
  53. ctx.fillStyle = 'red';
  54. ctx.fillRect(this.p1.x - 1.5 * ls, this.p1.y - 1.5 * ls, 3 * ls, 3 * ls);
  55. if (isHovered) {
  56. ctx.fillStyle = 'magenta';
  57. ctx.fillRect(this.p2.x - 1.5 * ls, this.p2.y - 1.5 * ls, 3 * ls, 3 * ls);
  58. }
  59. }
  60. checkRayIntersects(ray) {
  61. if (this.refIndex <= 0) return;
  62. return this.checkRayIntersectsShape(ray);
  63. }
  64. onRayIncident(ray, rayIndex, incidentPoint, surfaceMergingObjs) {
  65. var midpoint = geometry.segmentMidpoint(geometry.line(ray.p1, incidentPoint));
  66. var d = geometry.distanceSquared(this.p1, this.p2) - geometry.distanceSquared(this.p1, midpoint);
  67. if (d > 0) {
  68. // From inside to outside
  69. var n1 = this.getRefIndexAt(incidentPoint, ray);
  70. var normal = { x: this.p1.x - incidentPoint.x, y: this.p1.y - incidentPoint.y };
  71. } else if (d < 0) {
  72. // From outside to inside
  73. var n1 = 1 / this.getRefIndexAt(incidentPoint, ray);
  74. var normal = { x: incidentPoint.x - this.p1.x, y: incidentPoint.y - this.p1.y };
  75. } else {
  76. // Situation that may cause bugs (e.g. incident on an edge point)
  77. // To prevent shooting the ray to a wrong direction, absorb the ray
  78. return {
  79. isAbsorbed: true
  80. };
  81. }
  82. return this.refract(ray, rayIndex, incidentPoint, normal, n1, surfaceMergingObjs, ray.bodyMergingObj);
  83. }
  84. getIncidentType(ray) {
  85. var midpoint = geometry.segmentMidpoint(geometry.line(ray.p1, this.checkRayIntersects(ray)));
  86. var d = geometry.distanceSquared(this.p1, this.p2) - geometry.distanceSquared(this.p1, midpoint);
  87. if (d > 0) {
  88. return 1; // From inside to outside
  89. }
  90. if (d < 0) {
  91. return -1; // From outside to inside
  92. }
  93. return NaN;
  94. }
  95. };
  96. export default CircleGlass;