Source: hws.edu-examples/trackball-rotator.js

  1. /**
  2. * @file
  3. *
  4. * Summary.
  5. *
  6. * <p>The TrackballRotator class implements an <a href="/WebGL/labs/WebGL/extras/doc/Arcball.pdf">ArcBall</a> like interface.</p>
  7. * Create by {@link https://dl.acm.org/profile/81100026146 Ken Shoemake} in 1992,
  8. * it is the de facto <a href="/WebGL/labs/WebGL/extras/doc/shoemake92-arcball.pdf">standard</a>
  9. * for interactive 3D model manipulation and visualization.
  10. * <p>The class defines the following methods for an object rotator of type TrackballRotator:</p>
  11. * <ul>
  12. * <li>{@link TrackballRotator#getViewMatrix}() <br>
  13. * Returns the view transformation matrix as a regular JavaScript
  14. * array of 16 elements, in {@link https://en.wikipedia.org/wiki/Row-_and_column-major_order column-major} order,
  15. * suitable for use with
  16. * {@link https://developer.mozilla.org/en-US/docs/Web/API/WebGLRenderingContext/uniformMatrix gl.uniformMatrix4fv}
  17. * or for further transformation with the {@link https://glmatrix.net glMatrix} library {@link mat4} class.</li>
  18. * <li>{@link TrackballRotator#setViewMatrix}(matrix) <br>
  19. * Sets the view matrix.
  20. * <li>{@link TrackballRotator#setView}(viewDistance, viewpointDirection, viewUp) <br>
  21. * Sets up the view, where the
  22. * parameters are optional and are used in the same way as the corresponding parameters
  23. * in the constructor.</li>
  24. * <li>{@link TrackballRotator#setViewDistance}(viewDistance) <br>
  25. * Sets the distance of the viewer from the origin without
  26. * changing the direction of view. <br>
  27. * The parameter must be a positive number.
  28. * <li>{@link TrackballRotator#getViewDistance}() <br>
  29. * Returns the current value.</li>
  30. * <li>{@link TrackballRotator#setRotationCenter}(vector) <br>
  31. * Sets the center of rotation. <br>
  32. * The parameter must be an array of (at least) three numbers.
  33. * The view is rotated about this point. <br>
  34. * Usually, you want the rotation center to be the point that
  35. * appears at the middle of the canvas, but that is not a requirement. <br>
  36. * The initial value is effectively equal to [0,0,0].</li>
  37. * <li>{@link TrackballRotator#getRotationCenter}() <br>
  38. * Returns the current value.</li>
  39. * </ul>
  40. *
  41. * @since 19/11/2022
  42. * @author David J. Eck and modified by Paulo Roma
  43. * @see <a href="/WebGL/hws.edu-examples/trackball-rotator.js">source</a>
  44. * @see https://math.hws.edu/graphicsbook/source/webgl/cube-with-trackball-rotator.html
  45. * @see https://math.hws.edu/graphicsbook/source/webgl/trackball-rotator.js
  46. * @see https://math.hws.edu/graphicsbook/source/webgl/skybox-and-env-map.html
  47. * @see <img src="/WebGL/lib/arcball4.png" width="256">
  48. */
  49. /**
  50. * <p>An object of type TrackballRotator can be used to implement a
  51. * {@link https://www.xarg.org/2021/07/trackball-rotation-using-quaternions/ trackball}-like mouse rotation
  52. * of a WebGL scene about the origin. </p>
  53. *
  54. * Only the first parameter to the constructor is required.
  55. * When an object is created, mouse event handlers are set up on the canvas to respond to rotation.<br>
  56. * It will also work with a touchscreen.
  57. */
  58. class TrackballRotator {
  59. /**
  60. * <p>Constructor of TrackballRotator.</p>
  61. * @param {HTMLCanvasElement} canvas the HTML canvas element used for WebGL drawing.
  62. * The user will rotate the scene by dragging the mouse on this canvas.
  63. * This parameter is required.
  64. * @param {function} callback if present must be a function, which is called whenever the rotation changes.
  65. * It is typically the function that draws the scene.
  66. * @param {Number} viewDistance if present must be a positive number. Gives the distance of the viewer
  67. * from the origin. If not present, the length is zero, which can be OK for orthographic projection,
  68. * but never for perspective projection.
  69. * @param {Array<Number>} viewpointDirection if present must be an array of three numbers, not all zero.
  70. * The view is from the direction of this vector towards the origin (0,0,0). If not present,
  71. * the value [0,0,10] is used. This is just the initial value for viewpointDirection; it will
  72. * be modified by rotation.
  73. * @param {Array<Number>} viewUp if present must be an array of three numbers. Gives a vector that will
  74. * be seen as pointing upwards in the view. If not present, the value is [0,1,0].
  75. * Cannot be a multiple of viewpointDirection. This is just the initial value for
  76. * viewUp; it will be modified by rotation.
  77. */
  78. constructor(canvas, callback, viewDistance, viewpointDirection, viewUp) {
  79. var unitx = new Array(3);
  80. var unity = new Array(3);
  81. var unitz = new Array(3);
  82. /**
  83. * View distance, that is, the z-coord in eye coordinates.
  84. * @type {Number}
  85. */
  86. var viewZ;
  87. /**
  88. * <p>Center of view and rotation is about this point.</p>
  89. * Default is [0,0,0].
  90. * @type {Array<Number>}
  91. */
  92. var center;
  93. /**
  94. * Set up the view, where the parameters are optional,
  95. * and are used in the same way, <br>
  96. * as the corresponding parameters in the constructor.
  97. * @param {Number} viewDistance distance of the viewer from the origin.
  98. * @param {Array<Number>} viewpointDirection direction of view is from this point towards the origin (0,0,0).
  99. * @param {Array<Number>} viewUp view up vector.
  100. */
  101. this.setView = function (viewDistance, viewpointDirection, viewUp) {
  102. unitz =
  103. viewpointDirection === undefined
  104. ? [0, 0, 10]
  105. : viewpointDirection;
  106. viewUp = viewUp === undefined ? [0, 1, 0] : viewUp;
  107. viewZ = viewDistance;
  108. normalize(unitz, unitz);
  109. copy(unity, unitz);
  110. scale(unity, unity, dot(unitz, viewUp));
  111. subtract(unity, viewUp, unity);
  112. normalize(unity, unity);
  113. cross(unitx, unity, unitz);
  114. };
  115. /**
  116. * Returns the view transformation matrix as a regular JavaScript
  117. * array of 16 elements, in column-major order, <br>
  118. * suitable for use with
  119. * {@link https://developer.mozilla.org/en-US/docs/Web/API/WebGLRenderingContext/uniformMatrix gl.uniformMatrix4fv}
  120. * or, for further transformation, with the glMatrix library {@link https://glmatrix.net/docs/module-mat4.html mat4} class.
  121. * @return {Array<Number>} view matrix.
  122. */
  123. this.getViewMatrix = function () {
  124. var mat = [
  125. unitx[0],
  126. unity[0],
  127. unitz[0],
  128. 0,
  129. unitx[1],
  130. unity[1],
  131. unitz[1],
  132. 0,
  133. unitx[2],
  134. unity[2],
  135. unitz[2],
  136. 0,
  137. 0,
  138. 0,
  139. 0,
  140. 1,
  141. ];
  142. if (center !== undefined) {
  143. // multiply on left by translation by rotationCenter, on right by translation by -rotationCenter
  144. var t0 =
  145. center[0] -
  146. mat[0] * center[0] -
  147. mat[4] * center[1] -
  148. mat[8] * center[2];
  149. var t1 =
  150. center[1] -
  151. mat[1] * center[0] -
  152. mat[5] * center[1] -
  153. mat[9] * center[2];
  154. var t2 =
  155. center[2] -
  156. mat[2] * center[0] -
  157. mat[6] * center[1] -
  158. mat[10] * center[2];
  159. mat[12] = t0;
  160. mat[13] = t1;
  161. mat[14] = t2;
  162. }
  163. if (viewZ !== undefined) {
  164. mat[14] -= viewZ;
  165. }
  166. return mat;
  167. };
  168. /**
  169. * Sets the view matrix.
  170. * @param {Float32Array} matrix view matrix.
  171. */
  172. this.setViewMatrix = function (matrix) {
  173. unitx[0] = matrix[0];
  174. unity[0] = matrix[1];
  175. unitz[0] = matrix[2];
  176. unitx[1] = matrix[4];
  177. unity[1] = matrix[5];
  178. unitz[1] = matrix[6];
  179. unitx[2] = matrix[8];
  180. unity[2] = matrix[9];
  181. unitz[2] = matrix[10];
  182. };
  183. /**
  184. * Returns the viewDistance.
  185. * @return {Number} view distance.
  186. */
  187. this.getViewDistance = function () {
  188. return viewZ;
  189. };
  190. /**
  191. * Sets the distance of the viewer from the origin without
  192. * changing the direction of view. <br>
  193. * The parameter must be a positive number.
  194. * @param {Number} viewDistance view distance.
  195. */
  196. this.setViewDistance = function (viewDistance) {
  197. viewZ = viewDistance;
  198. };
  199. /**
  200. * Returns the current rotation center.
  201. * @returns {Array<Number>} center or [0,0,0], if undefined.
  202. */
  203. this.getRotationCenter = function () {
  204. return center === undefined ? [0, 0, 0] : center;
  205. };
  206. /**
  207. * <p>Sets the center of rotation.</p>
  208. * The parameter must be an array of (at least) three numbers.
  209. * The view is rotated about this point.
  210. * <p>Usually, you want the rotation center to be the point that appears
  211. * at the middle of the canvas, but that is not a requirement.</p>
  212. * The initial value is effectively equal to [0,0,0].
  213. * @param {Array<Number>} rotationCenter center of rotation.
  214. */
  215. this.setRotationCenter = function (rotationCenter) {
  216. center = rotationCenter;
  217. };
  218. this.setView(viewDistance, viewpointDirection, viewUp);
  219. canvas.addEventListener("mousedown", doMouseDown, false);
  220. canvas.addEventListener("touchstart", doTouchStart, false);
  221. function applyTransvection(e1, e2) {
  222. // rotate vector e1 onto e2
  223. function reflectInAxis(axis, source, destination) {
  224. var s =
  225. 2 *
  226. (axis[0] * source[0] +
  227. axis[1] * source[1] +
  228. axis[2] * source[2]);
  229. destination[0] = s * axis[0] - source[0];
  230. destination[1] = s * axis[1] - source[1];
  231. destination[2] = s * axis[2] - source[2];
  232. }
  233. normalize(e1, e1);
  234. normalize(e2, e2);
  235. var e = [0, 0, 0];
  236. add(e, e1, e2);
  237. normalize(e, e);
  238. var temp = [0, 0, 0];
  239. reflectInAxis(e, unitz, temp);
  240. reflectInAxis(e1, temp, unitz);
  241. reflectInAxis(e, unitx, temp);
  242. reflectInAxis(e1, temp, unitx);
  243. reflectInAxis(e, unity, temp);
  244. reflectInAxis(e1, temp, unity);
  245. }
  246. var centerX, centerY, radius2;
  247. var prevx, prevy;
  248. var dragging = false;
  249. var touchStarted = false;
  250. function doMouseDown(evt) {
  251. if (dragging) return;
  252. dragging = true;
  253. centerX = canvas.width / 2;
  254. centerY = canvas.height / 2;
  255. var radius = Math.min(centerX, centerY);
  256. radius2 = radius * radius;
  257. document.addEventListener("mousemove", doMouseDrag, false);
  258. document.addEventListener("mouseup", doMouseUp, false);
  259. var box = canvas.getBoundingClientRect();
  260. prevx = evt.clientX - box.left;
  261. prevy = evt.clientY - box.top;
  262. }
  263. function doMouseDrag(evt) {
  264. if (!dragging) return;
  265. var box = canvas.getBoundingClientRect();
  266. var x = evt.clientX - box.left;
  267. var y = evt.clientY - box.top;
  268. var ray1 = toRay(prevx, prevy);
  269. var ray2 = toRay(x, y);
  270. applyTransvection(ray1, ray2);
  271. prevx = x;
  272. prevy = y;
  273. if (callback) {
  274. callback();
  275. }
  276. }
  277. function doMouseUp(evt) {
  278. if (dragging) {
  279. document.removeEventListener("mousemove", doMouseDrag, false);
  280. document.removeEventListener("mouseup", doMouseUp, false);
  281. dragging = false;
  282. }
  283. }
  284. function doTouchStart(evt) {
  285. if (evt.touches.length != 1) {
  286. doTouchCancel();
  287. return;
  288. }
  289. evt.preventDefault();
  290. var r = canvas.getBoundingClientRect();
  291. prevx = evt.touches[0].clientX - r.left;
  292. prevy = evt.touches[0].clientY - r.top;
  293. canvas.addEventListener("touchmove", doTouchMove, false);
  294. canvas.addEventListener("touchend", doTouchEnd, false);
  295. canvas.addEventListener("touchcancel", doTouchCancel, false);
  296. touchStarted = true;
  297. centerX = canvas.width / 2;
  298. centerY = canvas.height / 2;
  299. var radius = Math.min(centerX, centerY);
  300. radius2 = radius * radius;
  301. }
  302. function doTouchMove(evt) {
  303. if (evt.touches.length != 1 || !touchStarted) {
  304. doTouchCancel();
  305. return;
  306. }
  307. evt.preventDefault();
  308. var r = canvas.getBoundingClientRect();
  309. var x = evt.touches[0].clientX - r.left;
  310. var y = evt.touches[0].clientY - r.top;
  311. var ray1 = toRay(prevx, prevy);
  312. var ray2 = toRay(x, y);
  313. applyTransvection(ray1, ray2);
  314. prevx = x;
  315. prevy = y;
  316. if (callback) {
  317. callback();
  318. }
  319. }
  320. function doTouchEnd(evt) {
  321. doTouchCancel();
  322. }
  323. function doTouchCancel() {
  324. if (touchStarted) {
  325. touchStarted = false;
  326. canvas.removeEventListener("touchmove", doTouchMove, false);
  327. canvas.removeEventListener("touchend", doTouchEnd, false);
  328. canvas.removeEventListener("touchcancel", doTouchCancel, false);
  329. }
  330. }
  331. function toRay(x, y) {
  332. // converts a point (x,y) in pixel coords to a 3D ray by mapping interior of
  333. // a circle in the plane to a hemisphere with that circle as equator.
  334. var dx = x - centerX;
  335. var dy = centerY - y;
  336. var vx = dx * unitx[0] + dy * unity[0]; // The mouse point as a vector in the image plane.
  337. var vy = dx * unitx[1] + dy * unity[1];
  338. var vz = dx * unitx[2] + dy * unity[2];
  339. var dist2 = vx * vx + vy * vy + vz * vz;
  340. if (dist2 > radius2) {
  341. // Map a point ouside the circle to itself
  342. return [vx, vy, vz];
  343. } else {
  344. var z = Math.sqrt(radius2 - dist2);
  345. return [
  346. vx + z * unitz[0],
  347. vy + z * unitz[1],
  348. vz + z * unitz[2],
  349. ];
  350. }
  351. }
  352. function dot(v, w) {
  353. return v[0] * w[0] + v[1] * w[1] + v[2] * w[2];
  354. }
  355. function length(v) {
  356. return Math.sqrt(v[0] * v[0] + v[1] * v[1] + v[2] * v[2]);
  357. }
  358. function normalize(v, w) {
  359. var d = length(w);
  360. v[0] = w[0] / d;
  361. v[1] = w[1] / d;
  362. v[2] = w[2] / d;
  363. }
  364. function copy(v, w) {
  365. v[0] = w[0];
  366. v[1] = w[1];
  367. v[2] = w[2];
  368. }
  369. function add(sum, v, w) {
  370. sum[0] = v[0] + w[0];
  371. sum[1] = v[1] + w[1];
  372. sum[2] = v[2] + w[2];
  373. }
  374. function subtract(dif, v, w) {
  375. dif[0] = v[0] - w[0];
  376. dif[1] = v[1] - w[1];
  377. dif[2] = v[2] - w[2];
  378. }
  379. function scale(ans, v, num) {
  380. ans[0] = v[0] * num;
  381. ans[1] = v[1] * num;
  382. ans[2] = v[2] * num;
  383. }
  384. function cross(c, v, w) {
  385. var x = v[1] * w[2] - v[2] * w[1];
  386. var y = v[2] * w[0] - v[0] * w[2];
  387. var z = v[0] * w[1] - v[1] * w[0];
  388. c[0] = x;
  389. c[1] = y;
  390. c[2] = z;
  391. }
  392. }
  393. }