Reference Source

src/viewer/scene/model/SceneModelTransform.js

  1. import {math} from "../math/index.js";
  2.  
  3. const angleAxis = math.vec4(4);
  4. const q1 = math.vec4();
  5. const q2 = math.vec4();
  6. const xAxis = math.vec3([1, 0, 0]);
  7. const yAxis = math.vec3([0, 1, 0]);
  8. const zAxis = math.vec3([0, 0, 1]);
  9.  
  10. const veca = math.vec3(3);
  11. const vecb = math.vec3(3);
  12.  
  13. const identityMat = math.identityMat4();
  14.  
  15. /**
  16. * A dynamically-updatable transform within a {@link SceneModel}.
  17. *
  18. * * Can be composed into hierarchies
  19. * * Shared by multiple {@link SceneModelMesh}es
  20. * * Created with {@link SceneModel#createTransform}
  21. * * Stored by ID in {@link SceneModel#transforms}
  22. * * Referenced by {@link SceneModelMesh#transform}
  23. */
  24. export class SceneModelTransform {
  25.  
  26. /**
  27. * @private
  28. */
  29. constructor(cfg) {
  30. this._model = cfg.model;
  31.  
  32. /**
  33. * Unique ID of this SceneModelTransform.
  34. *
  35. * The SceneModelTransform is registered against this ID in {@link SceneModel#transforms}.
  36. */
  37. this.id = cfg.id;
  38.  
  39. this._parentTransform = cfg.parent;
  40. this._childTransforms = [];
  41. this._meshes = [];
  42. this._scale = new Float32Array([1,1,1]);
  43. this._quaternion = math.identityQuaternion(new Float32Array(4));
  44. this._rotation = new Float32Array(3);
  45. this._position = new Float32Array(3);
  46. this._localMatrix = math.identityMat4(new Float32Array(16));
  47. this._worldMatrix = math.identityMat4(new Float32Array(16));
  48. this._localMatrixDirty = true;
  49. this._worldMatrixDirty = true;
  50.  
  51. if (cfg.matrix) {
  52. this.matrix = cfg.matrix;
  53. } else {
  54. this.scale = cfg.scale;
  55. this.position = cfg.position;
  56. if (cfg.quaternion) {
  57. } else {
  58. this.rotation = cfg.rotation;
  59. }
  60. }
  61. if (cfg.parent) {
  62. cfg.parent._addChildTransform(this);
  63. }
  64. }
  65.  
  66. _addChildTransform(childTransform) {
  67. this._childTransforms.push(childTransform);
  68. childTransform._parentTransform = this;
  69. childTransform._transformDirty();
  70. childTransform._setSubtreeAABBsDirty(this);
  71. }
  72.  
  73. _addMesh(mesh) {
  74. this._meshes.push(mesh);
  75. mesh.transform = this;
  76. // childTransform._setWorldMatrixDirty();
  77. // childTransform._setAABBDirty();
  78. }
  79.  
  80. /**
  81. * The optional parent SceneModelTransform.
  82. *
  83. * @type {SceneModelTransform}
  84. */
  85. get parentTransform() {
  86. return this._parentTransform;
  87. }
  88.  
  89. /**
  90. * The {@link SceneModelMesh}es transformed by this SceneModelTransform.
  91. *
  92. * @returns {[]}
  93. */
  94. get meshes() {
  95. return this._meshes;
  96. }
  97.  
  98. /**
  99. * Sets the SceneModelTransform's local translation.
  100. *
  101. * Default value is ````[0,0,0]````.
  102. *
  103. * @type {Number[]}
  104. */
  105. set position(value) {
  106. this._position.set(value || [0, 0, 0]);
  107. this._setLocalMatrixDirty();
  108. this._model.glRedraw();
  109. }
  110.  
  111. /**
  112. * Gets the SceneModelTransform's translation.
  113. *
  114. * Default value is ````[0,0,0]````.
  115. *
  116. * @type {Number[]}
  117. */
  118. get position() {
  119. return this._position;
  120. }
  121.  
  122. /**
  123. * Sets the SceneModelTransform's rotation, as Euler angles given in degrees, for each of the X, Y and Z axis.
  124. *
  125. * Default value is ````[0,0,0]````.
  126. *
  127. * @type {Number[]}
  128. */
  129. set rotation(value) {
  130. this._rotation.set(value || [0, 0, 0]);
  131. math.eulerToQuaternion(this._rotation, "XYZ", this._quaternion);
  132. this._setLocalMatrixDirty();
  133. this._model.glRedraw();
  134. }
  135.  
  136. /**
  137. * Gets the SceneModelTransform's rotation, as Euler angles given in degrees, for each of the X, Y and Z axis.
  138. *
  139. * Default value is ````[0,0,0]````.
  140. *
  141. * @type {Number[]}
  142. */
  143. get rotation() {
  144. return this._rotation;
  145. }
  146.  
  147. /**
  148. * Sets the SceneModelTransform's rotation quaternion.
  149. *
  150. * Default value is ````[0,0,0,1]````.
  151. *
  152. * @type {Number[]}
  153. */
  154. set quaternion(value) {
  155. this._quaternion.set(value || [0, 0, 0, 1]);
  156. math.quaternionToEuler(this._quaternion, "XYZ", this._rotation);
  157. this._setLocalMatrixDirty();
  158. this._model.glRedraw();
  159. }
  160.  
  161. /**
  162. * Gets the SceneModelTransform's rotation quaternion.
  163. *
  164. * Default value is ````[0,0,0,1]````.
  165. *
  166. * @type {Number[]}
  167. */
  168. get quaternion() {
  169. return this._quaternion;
  170. }
  171.  
  172. /**
  173. * Sets the SceneModelTransform's scale.
  174. *
  175. * Default value is ````[1,1,1]````.
  176. *
  177. * @type {Number[]}
  178. */
  179. set scale(value) {
  180. this._scale.set(value || [1, 1, 1]);
  181. this._setLocalMatrixDirty();
  182. this._model.glRedraw();
  183. }
  184.  
  185. /**
  186. * Gets the SceneModelTransform's scale.
  187. *
  188. * Default value is ````[1,1,1]````.
  189. *
  190. * @type {Number[]}
  191. */
  192. get scale() {
  193. return this._scale;
  194. }
  195.  
  196. /**
  197. * Sets the SceneModelTransform's transform matrix.
  198. *
  199. * Default value is ````[1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1]````.
  200. *
  201. * @type {Number[]}
  202. */
  203. set matrix(value) {
  204. if (!this._localMatrix) {
  205. this._localMatrix = math.identityMat4();
  206. }
  207. this._localMatrix.set(value || identityMat);
  208. math.decomposeMat4(this._localMatrix, this._position, this._quaternion, this._scale);
  209. this._localMatrixDirty = false;
  210. this._transformDirty();
  211. this._model.glRedraw();
  212. }
  213.  
  214. /**
  215. * Gets the SceneModelTransform's transform matrix.
  216. *
  217. * Default value is ````[1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1]````.
  218. *
  219. * @type {Number[]}
  220. */
  221. get matrix() {
  222. if (this._localMatrixDirty) {
  223. if (!this._localMatrix) {
  224. this._localMatrix = math.identityMat4();
  225. }
  226. math.composeMat4(this._position, this._quaternion, this._scale, this._localMatrix);
  227. this._localMatrixDirty = false;
  228. }
  229. return this._localMatrix;
  230. }
  231.  
  232. /**
  233. * Gets the SceneModelTransform's World matrix.
  234. *
  235. * @property worldMatrix
  236. * @type {Number[]}
  237. */
  238. get worldMatrix() {
  239. if (this._worldMatrixDirty) {
  240. this._buildWorldMatrix();
  241. }
  242. return this._worldMatrix;
  243. }
  244.  
  245. /**
  246. * Rotates the SceneModelTransform about the given axis by the given increment.
  247. *
  248. * @param {Number[]} axis Local axis about which to rotate.
  249. * @param {Number} angle Angle increment in degrees.
  250. */
  251. rotate(axis, angle) {
  252. angleAxis[0] = axis[0];
  253. angleAxis[1] = axis[1];
  254. angleAxis[2] = axis[2];
  255. angleAxis[3] = angle * math.DEGTORAD;
  256. math.angleAxisToQuaternion(angleAxis, q1);
  257. math.mulQuaternions(this.quaternion, q1, q2);
  258. this.quaternion = q2;
  259. this._setLocalMatrixDirty();
  260. this._model.glRedraw();
  261. return this;
  262. }
  263.  
  264. /**
  265. * Rotates the SceneModelTransform about the given World-space axis by the given increment.
  266. *
  267. * @param {Number[]} axis Local axis about which to rotate.
  268. * @param {Number} angle Angle increment in degrees.
  269. */
  270. rotateOnWorldAxis(axis, angle) {
  271. angleAxis[0] = axis[0];
  272. angleAxis[1] = axis[1];
  273. angleAxis[2] = axis[2];
  274. angleAxis[3] = angle * math.DEGTORAD;
  275. math.angleAxisToQuaternion(angleAxis, q1);
  276. math.mulQuaternions(q1, this.quaternion, q1);
  277. //this.quaternion.premultiply(q1);
  278. return this;
  279. }
  280.  
  281. /**
  282. * Rotates the SceneModelTransform about the local X-axis by the given increment.
  283. *
  284. * @param {Number} angle Angle increment in degrees.
  285. */
  286. rotateX(angle) {
  287. return this.rotate(xAxis, angle);
  288. }
  289.  
  290. /**
  291. * Rotates the SceneModelTransform about the local Y-axis by the given increment.
  292. *
  293. * @param {Number} angle Angle increment in degrees.
  294. */
  295. rotateY(angle) {
  296. return this.rotate(yAxis, angle);
  297. }
  298.  
  299. /**
  300. * Rotates the SceneModelTransform about the local Z-axis by the given increment.
  301. *
  302. * @param {Number} angle Angle increment in degrees.
  303. */
  304. rotateZ(angle) {
  305. return this.rotate(zAxis, angle);
  306. }
  307.  
  308. /**
  309. * Translates the SceneModelTransform along the local axis by the given increment.
  310. *
  311. * @param {Number[]} axis Normalized local space 3D vector along which to translate.
  312. * @param {Number} distance Distance to translate along the vector.
  313. */
  314. translate(axis) {
  315. this._position[0] += axis[0];
  316. this._position[1] += axis[1];
  317. this._position[2] += axis[2];
  318. this._setLocalMatrixDirty();
  319. this._model.glRedraw();
  320. return this;
  321. }
  322.  
  323. /**
  324. * Translates the SceneModelTransform along the local X-axis by the given increment.
  325. *
  326. * @param {Number} distance Distance to translate along the X-axis.
  327. */
  328. translateX(distance) {
  329. this._position[0] += distance;
  330. this._setLocalMatrixDirty();
  331. this._model.glRedraw();
  332. return this;
  333. }
  334.  
  335. /**
  336. * Translates the SceneModelTransform along the local Y-axis by the given increment.
  337. *
  338. * @param {Number} distance Distance to translate along the Y-axis.
  339. */
  340. translateY(distance) {
  341. this._position[1] += distance;
  342. this._setLocalMatrixDirty();
  343. this._model.glRedraw();
  344. return this;
  345. }
  346.  
  347. /**
  348. * Translates the SceneModelTransform along the local Z-axis by the given increment.
  349. *
  350. * @param {Number} distance Distance to translate along the Z-axis.
  351. */
  352. translateZ(distance) {
  353. this._position[2] += distance;
  354. this._setLocalMatrixDirty();
  355. this._model.glRedraw();
  356. return this;
  357. }
  358.  
  359. _setLocalMatrixDirty() {
  360. this._localMatrixDirty = true;
  361. this._transformDirty();
  362. }
  363.  
  364. _transformDirty() {
  365. this._worldMatrixDirty = true;
  366. for (let i = 0, len = this._childTransforms.length; i < len; i++) {
  367. const childTransform = this._childTransforms[i];
  368. childTransform._transformDirty();
  369. if (childTransform._meshes && childTransform._meshes.length > 0) {
  370. const meshes = childTransform._meshes;
  371. for (let j =0, lenj = meshes.length; j < lenj; j++) {
  372. meshes[j]._transformDirty();
  373. }
  374. }
  375. }
  376. if (this._meshes && this._meshes.length > 0) {
  377. const meshes = this._meshes;
  378. for (let j =0, lenj = meshes.length; j < lenj; j++) {
  379. meshes[j]._transformDirty();
  380. }
  381. }
  382. }
  383.  
  384. _buildWorldMatrix() {
  385. const localMatrix = this.matrix;
  386. if (!this._parentTransform) {
  387. for (let i = 0, len = localMatrix.length; i < len; i++) {
  388. this._worldMatrix[i] = localMatrix[i];
  389. }
  390. } else {
  391. math.mulMat4(this._parentTransform.worldMatrix, localMatrix, this._worldMatrix);
  392. }
  393. this._worldMatrixDirty = false;
  394. }
  395.  
  396. _setSubtreeAABBsDirty(sceneTransform) {
  397. sceneTransform._aabbDirty = true;
  398. if (sceneTransform._childTransforms) {
  399. for (let i = 0, len = sceneTransform._childTransforms.length; i < len; i++) {
  400. this._setSubtreeAABBsDirty(sceneTransform._childTransforms[i]);
  401. }
  402. }
  403. }
  404. }