Reference Source

src/viewer/scene/sectionPlane/SectionPlane.js

  1. import {Component} from '../Component.js';
  2. import {RenderState} from '../webgl/RenderState.js';
  3. import {math} from "../math/math.js";
  4.  
  5. const tempVec3a = math.vec3();
  6. const tempVec3b = math.vec3();
  7. const tempVec4a = math.vec4();
  8. const front = math.vec3([0, 0, -1]);
  9. const back = math.vec3([0, 0, 1]);
  10. const up = math.vec3([0, 1, 0]);
  11.  
  12. /**
  13. * @desc An arbitrarily-aligned World-space clipping plane.
  14. *
  15. * * Slices portions off objects to create cross-section views or reveal interiors.
  16. * * Registered by {@link SectionPlane#id} in {@link Scene#sectionPlanes}.
  17. * * Indicates World-space position in {@link SectionPlane#pos} and orientation in {@link SectionPlane#dir}.
  18. * * Discards elements from the half-space in the direction of {@link SectionPlane#dir}.
  19. * * Can be be enabled or disabled via {@link SectionPlane#active}.
  20. *
  21. * ## Usage
  22. *
  23. * In the example below, we'll create two SectionPlanes to slice a model loaded from glTF. Note that we could also create them
  24. * using a {@link SectionPlanesPlugin}.
  25. *
  26. * ````javascript
  27. * import {Viewer, GLTFLoaderPlugin, SectionPlane} from "xeokit-sdk.es.js";
  28. *
  29. * const viewer = new Viewer({
  30. * canvasId: "myCanvas"
  31. * });
  32. *
  33. * const gltfLoaderPlugin = new GLTFModelsPlugin(viewer, {
  34. * id: "GLTFModels"
  35. * });
  36. *
  37. * const model = gltfLoaderPlugin.load({
  38. * id: "myModel",
  39. * src: "./models/gltf/mygltfmodel.gltf"
  40. * });
  41. *
  42. * // Create a SectionPlane on negative diagonal
  43. * const sectionPlane1 = new SectionPlane(viewer.scene, {
  44. * pos: [1.0, 1.0, 1.0],
  45. * dir: [-1.0, -1.0, -1.0],
  46. * active: true
  47. * }),
  48. *
  49. * // Create a SectionPlane on positive diagonal
  50. * const sectionPlane2 = new SectionPlane(viewer.scene, {
  51. * pos: [-1.0, -1.0, -1.0],
  52. * dir: [1.0, 1.0, 1.0],
  53. * active: true
  54. * });
  55. * ````
  56. */
  57. class SectionPlane extends Component {
  58.  
  59. /**
  60. @private
  61. */
  62. get type() {
  63. return "SectionPlane";
  64. }
  65.  
  66. /**
  67. * @constructor
  68. * @param {Component} [owner] Owner component. When destroyed, the owner will destroy this SectionPlane as well.
  69. * @param {*} [cfg] SectionPlane configuration
  70. * @param {String} [cfg.id] Optional ID, unique among all components in the parent {@link Scene}, generated automatically when omitted.
  71. * @param {Boolean} [cfg.active=true] Indicates whether or not this SectionPlane is active.
  72. * @param {Number[]} [cfg.pos=[0,0,0]] World-space position of the SectionPlane.
  73. * @param {Number[]} [cfg.dir=[0,0,-1]] Vector perpendicular to the plane surface, indicating the SectionPlane plane orientation.
  74. */
  75. constructor(owner, cfg = {}) {
  76.  
  77. super(owner, cfg);
  78.  
  79. this._state = new RenderState({
  80. active: true,
  81. pos: math.vec3(),
  82. quaternion: math.vec4(),
  83. roll: 0,
  84. dir: math.vec3(),
  85. dist: 0
  86. });
  87.  
  88. this.active = cfg.active;
  89. this.pos = cfg.pos;
  90. this.dir = cfg.dir;
  91.  
  92. this.scene._sectionPlaneCreated(this);
  93. }
  94.  
  95. /**
  96. * Sets if this SectionPlane is active or not.
  97. *
  98. * Default value is ````true````.
  99. *
  100. * @param {Boolean} value Set ````true```` to activate else ````false```` to deactivate.
  101. */
  102. set active(value) {
  103. this._state.active = value !== false;
  104. this.glRedraw();
  105. this.fire("active", this._state.active);
  106. this.scene.fire("sectionPlaneUpdated", this);
  107. }
  108.  
  109. /**
  110. * Gets if this SectionPlane is active or not.
  111. *
  112. * Default value is ````true````.
  113. *
  114. * @returns {Boolean} Returns ````true```` if active.
  115. */
  116. get active() {
  117. return this._state.active;
  118. }
  119.  
  120. /**
  121. * Sets the World-space position of this SectionPlane's plane.
  122. *
  123. * Default value is ````[0, 0, 0]````.
  124. *
  125. * @param {Number[]} value New position.
  126. */
  127. set pos(value) {
  128. this._state.pos.set(value || [0, 0, 0]);
  129. this._state.dist = (-math.dotVec3(this._state.pos, this._state.dir));
  130. this.fire("pos", this._state.pos);
  131. this.scene.fire("sectionPlaneUpdated", this);
  132. }
  133.  
  134. /**
  135. * Gets the World-space position of this SectionPlane's plane.
  136. *
  137. * Default value is ````[0, 0, 0]````.
  138. *
  139. * @returns {Number[]} Current position.
  140. */
  141. get pos() {
  142. return this._state.pos;
  143. }
  144.  
  145. /**
  146. * Sets the quaternion of this SectionPlane's plane.
  147. *
  148. * Default value is ````[0, -1, 0, 0]````.
  149. *
  150. * @param {Number[]} value New quaternion.
  151. */
  152. set quaternion(value) {
  153. this._state.quaternion.set(value || [0, 0, 0, -1]);
  154. math.vec3ApplyQuaternion(this._state.quaternion, back, this._state.dir);
  155. const quatUp = math.vec3ApplyQuaternion(this._state.quaternion, up, tempVec3a);
  156. const dirOnlyQ = math.vec3PairToQuaternion(back, this._state.dir, tempVec4a);
  157. const dirOnlyUp = math.vec3ApplyQuaternion(dirOnlyQ, up, tempVec3b);
  158. const angle = Math.acos(Math.min(1, math.dotVec3(quatUp, dirOnlyUp)));
  159. const sign = Math.sign(math.dotVec3(this._state.dir, math.cross3Vec3(quatUp, dirOnlyUp, tempVec3b)));
  160. this._state.roll = sign * angle;
  161. this._onDirUpdated();
  162. }
  163.  
  164. /**
  165. * Gets the quaternion of this SectionPlane's plane.
  166. *
  167. * Default value is ````[0, -1, 0, 0]````.
  168. *
  169. * @returns {Number[]} value Current quaternion.
  170. */
  171. get quaternion() {
  172. return this._state.quaternion;
  173. }
  174.  
  175. /**
  176. * Sets the roll of this SectionPlane's plane.
  177. *
  178. * Default value is ````0````.
  179. *
  180. * @param {Number[]} value New roll.
  181. */
  182. set roll(value) {
  183. this._state.roll = value || 0;
  184. this._onDirRollUpdated();
  185. }
  186.  
  187. /**
  188. * Gets the roll of this SectionPlane's plane.
  189. *
  190. * Default value is ````0````.
  191. *
  192. * @returns {Number[]} value Current roll.
  193. */
  194. get roll() {
  195. return this._state.roll;
  196. }
  197.  
  198. /**
  199. * Sets the direction of this SectionPlane's plane.
  200. *
  201. * Default value is ````[0, 0, -1]````.
  202. *
  203. * @param {Number[]} value New direction.
  204. */
  205. set dir(value) {
  206. this._state.dir.set(value || front);
  207. this._onDirRollUpdated();
  208. }
  209.  
  210. _onDirRollUpdated() {
  211. math.vec3PairToQuaternion(back, this._state.dir, this._state.quaternion);
  212.  
  213. tempVec4a[0] = 0;
  214. tempVec4a[1] = 0;
  215. tempVec4a[2] = -1;
  216. tempVec4a[3] = this._state.roll;
  217. math.angleAxisToQuaternion(tempVec4a, tempVec4a);
  218.  
  219. math.mulQuaternions(this._state.quaternion, tempVec4a, this._state.quaternion);
  220.  
  221. this._onDirUpdated();
  222. }
  223.  
  224. _onDirUpdated() {
  225. this._state.dist = (-math.dotVec3(this._state.pos, this._state.dir));
  226. this.glRedraw();
  227. this.fire("dir", this._state.dir);
  228. this.scene.fire("sectionPlaneUpdated", this);
  229. }
  230.  
  231. /**
  232. * Gets the direction of this SectionPlane's plane.
  233. *
  234. * Default value is ````[0, 0, -1]````.
  235. *
  236. * @returns {Number[]} value Current direction.
  237. */
  238. get dir() {
  239. return this._state.dir;
  240. }
  241.  
  242. /**
  243. * Gets this SectionPlane's distance to the origin of the World-space coordinate system.
  244. *
  245. * This is the dot product of {@link SectionPlane#pos} and {@link SectionPlane#dir} and is automatically re-calculated
  246. * each time either of two properties are updated.
  247. *
  248. * @returns {Number}
  249. */
  250. get dist() {
  251. return this._state.dist;
  252. }
  253.  
  254. /**
  255. * Inverts the direction of {@link SectionPlane#dir}.
  256. */
  257. flipDir() {
  258. math.mulVec3Scalar(this._state.dir, -1.0, this._state.dir);
  259. this.dir = this._state.dir;
  260. }
  261.  
  262. /**
  263. * @destroy
  264. */
  265. destroy() {
  266. this._state.destroy();
  267. this.scene._sectionPlaneDestroyed(this);
  268. super.destroy();
  269. }
  270. }
  271.  
  272. export {SectionPlane};