Source: core/sceneObjs/LineObjMixin.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 geometry from '../geometry.js';
  17. import BaseSceneObj from './BaseSceneObj.js';
  18. /**
  19. * The mixin for the scene objects that are defined by a line segment.
  20. * @template {typeof BaseSceneObj} T
  21. * @param {T} Base
  22. * @returns {T}
  23. */
  24. const LineObjMixin = Base => class extends Base {
  25. move(diffX, diffY) {
  26. // Move the first point
  27. this.p1.x = this.p1.x + diffX;
  28. this.p1.y = this.p1.y + diffY;
  29. // Move the second point
  30. this.p2.x = this.p2.x + diffX;
  31. this.p2.y = this.p2.y + diffY;
  32. }
  33. onConstructMouseDown(mouse, ctrl, shift) {
  34. if (!this.constructionPoint) {
  35. // Initialize the construction stage.
  36. this.constructionPoint = mouse.getPosSnappedToGrid();
  37. this.p1 = this.constructionPoint;
  38. this.p2 = this.constructionPoint;
  39. }
  40. if (shift) {
  41. this.p2 = mouse.getPosSnappedToDirection(this.constructionPoint, [{ x: 1, y: 0 }, { x: 0, y: 1 }, { x: 1, y: 1 }, { x: 1, y: -1 }]);
  42. } else {
  43. this.p2 = mouse.getPosSnappedToGrid();
  44. }
  45. }
  46. onConstructMouseMove(mouse, ctrl, shift) {
  47. if (shift) {
  48. this.p2 = mouse.getPosSnappedToDirection(this.constructionPoint, [{ x: 1, y: 0 }, { x: 0, y: 1 }, { x: 1, y: 1 }, { x: 1, y: -1 }]);
  49. } else {
  50. this.p2 = mouse.getPosSnappedToGrid();
  51. }
  52. this.p1 = ctrl ? geometry.point(2 * this.constructionPoint.x - this.p2.x, 2 * this.constructionPoint.y - this.p2.y) : this.constructionPoint;
  53. }
  54. onConstructMouseUp(mouse, ctrl, shift) {
  55. if (!mouse.snapsOnPoint(this.p1)) {
  56. delete this.constructionPoint;
  57. return {
  58. isDone: true
  59. };
  60. }
  61. }
  62. checkMouseOver(mouse) {
  63. let dragContext = {};
  64. if (mouse.isOnPoint(this.p1) && geometry.distanceSquared(mouse.pos, this.p1) <= geometry.distanceSquared(mouse.pos, this.p2)) {
  65. dragContext.part = 1;
  66. dragContext.targetPoint = geometry.point(this.p1.x, this.p1.y);
  67. return dragContext;
  68. }
  69. if (mouse.isOnPoint(this.p2)) {
  70. dragContext.part = 2;
  71. dragContext.targetPoint = geometry.point(this.p2.x, this.p2.y);
  72. return dragContext;
  73. }
  74. if (mouse.isOnSegment(this)) {
  75. const mousePos = mouse.getPosSnappedToGrid();
  76. dragContext.part = 0;
  77. dragContext.mousePos0 = mousePos; // Mouse position when the user starts dragging
  78. dragContext.mousePos1 = mousePos; // Mouse position at the last moment during dragging
  79. dragContext.snapContext = {};
  80. return dragContext;
  81. }
  82. }
  83. onDrag(mouse, dragContext, ctrl, shift) {
  84. var basePoint;
  85. if (dragContext.part == 1) {
  86. // Dragging the first endpoint Dragging the first endpoint
  87. basePoint = ctrl ? geometry.segmentMidpoint(dragContext.originalObj) : dragContext.originalObj.p2;
  88. this.p1 = shift ? mouse.getPosSnappedToDirection(basePoint, [{ x: 1, y: 0 }, { x: 0, y: 1 }, { x: 1, y: 1 }, { x: 1, y: -1 }, { x: (dragContext.originalObj.p2.x - dragContext.originalObj.p1.x), y: (dragContext.originalObj.p2.y - dragContext.originalObj.p1.y) }]) : mouse.getPosSnappedToGrid();
  89. this.p2 = ctrl ? geometry.point(2 * basePoint.x - this.p1.x, 2 * basePoint.y - this.p1.y) : basePoint;
  90. }
  91. if (dragContext.part == 2) {
  92. // Dragging the second endpoint Dragging the second endpoint
  93. basePoint = ctrl ? geometry.segmentMidpoint(dragContext.originalObj) : dragContext.originalObj.p1;
  94. this.p2 = shift ? mouse.getPosSnappedToDirection(basePoint, [{ x: 1, y: 0 }, { x: 0, y: 1 }, { x: 1, y: 1 }, { x: 1, y: -1 }, { x: (dragContext.originalObj.p2.x - dragContext.originalObj.p1.x), y: (dragContext.originalObj.p2.y - dragContext.originalObj.p1.y) }]) : mouse.getPosSnappedToGrid();
  95. this.p1 = ctrl ? geometry.point(2 * basePoint.x - this.p2.x, 2 * basePoint.y - this.p2.y) : basePoint;
  96. }
  97. if (dragContext.part == 0) {
  98. // Dragging the entire line
  99. if (shift) {
  100. var mousePos = mouse.getPosSnappedToDirection(dragContext.mousePos0, [{ x: 1, y: 0 }, { x: 0, y: 1 }, { x: (dragContext.originalObj.p2.x - dragContext.originalObj.p1.x), y: (dragContext.originalObj.p2.y - dragContext.originalObj.p1.y) }, { x: (dragContext.originalObj.p2.y - dragContext.originalObj.p1.y), y: -(dragContext.originalObj.p2.x - dragContext.originalObj.p1.x) }], dragContext.snapContext);
  101. } else {
  102. var mousePos = mouse.getPosSnappedToGrid();
  103. dragContext.snapContext = {}; // Unlock the dragging direction when the user release the shift key
  104. }
  105. var mouseDiffX = dragContext.mousePos1.x - mousePos.x; // The X difference between the mouse position now and at the previous moment
  106. var mouseDiffY = dragContext.mousePos1.y - mousePos.y; // The Y difference between the mouse position now and at the previous moment The Y difference between the mouse position now and at the previous moment
  107. // Move the first point
  108. this.p1.x = this.p1.x - mouseDiffX;
  109. this.p1.y = this.p1.y - mouseDiffY;
  110. // Move the second point
  111. this.p2.x = this.p2.x - mouseDiffX;
  112. this.p2.y = this.p2.y - mouseDiffY;
  113. // Update the mouse position
  114. dragContext.mousePos1 = mousePos;
  115. }
  116. }
  117. /**
  118. * Check if a ray intersects the line segment.
  119. * In the child class, this can be called from the `checkRayIntersects` method.
  120. * @param {Ray} ray - The ray.
  121. * @returns {Point} The intersection point, or null if there is no intersection.
  122. */
  123. checkRayIntersectsShape(ray) {
  124. var rp_temp = geometry.linesIntersection(geometry.line(ray.p1, ray.p2), geometry.line(this.p1, this.p2));
  125. if (geometry.intersectionIsOnSegment(rp_temp, this) && geometry.intersectionIsOnRay(rp_temp, ray)) {
  126. return rp_temp;
  127. } else {
  128. return null;
  129. }
  130. }
  131. };
  132. export default LineObjMixin;