/home/lindsay/xeolabs/xeogl-next/xeogl/examples/js/effects/stereoEffect.js
API Docs for:

File: /home/lindsay/xeolabs/xeogl-next/xeogl/examples/js/effects/stereoEffect.js

  1. /**
  2. A **StereoEffect** sets up a stereo view for its {{#crossLink "Scene"}}Scene{{/crossLink}}.
  3.  
  4. <a href="../../examples/#effects_stereo"><img src="../../assets/images/screenshots/StereoEffect.png"></img></a>
  5.  
  6. ## Overview
  7.  
  8. * Based on technique described in [this article by Paul Bourke](http://paulbourke.net/stereographics/stereorender/).
  9.  
  10. ## Examples
  11.  
  12. * [Stereo view using a StereoEffect](../../examples/#effects_stereo)
  13.  
  14. ## Usage
  15.  
  16. Stereo view of an {{#crossLink "Mesh"}}{{/crossLink}} using the {{#crossLink "Scene"}}Scene{{/crossLink}}'s default {{#crossLink "Camera"}}{{/crossLink}} and {{#crossLink "Viewport"}}{{/crossLink}}:
  17.  
  18. ````javascript
  19. // Both the Mesh and the StereoEffect use their Scene's default Camera and Viewport
  20.  
  21. var mesh = new xeogl.Mesh({
  22. geometry: new xeogl.TorusGeometry()
  23. });
  24.  
  25. var stereoEffect = new xeogl.StereoEffect({
  26. fov: 45, // Default
  27. active: true // Default
  28. });
  29. ````
  30.  
  31. @class StereoEffect
  32. @module xeogl
  33. @submodule effects
  34. @constructor
  35. @param [scene] {Scene} Parent {{#crossLink "Scene"}}Scene{{/crossLink}} - creates this StereoEffect in the default
  36. {{#crossLink "Scene"}}Scene{{/crossLink}} when omitted.
  37. @param [cfg] {*} Configs
  38. @param [cfg.id] {String} Optional ID, unique among all components in the parent {{#crossLink "Scene"}}Scene{{/crossLink}},
  39. generated automatically when omitted.
  40. @param [cfg.meta] {String:Object} Optional map of user-defined metadata to attach to this StereoEffect.
  41. @param [cfg.fov=45] Field-of-view angle in degrees.
  42. @param [cfg.active=true] {Boolean} Whether or not this StereoEffect is active.
  43. @extends Mesh
  44. */
  45.  
  46. xeogl.StereoEffect = class xeoglStereoEffect extends xeogl.Component {
  47.  
  48. init(cfg) {
  49. super.init(cfg);
  50. this.fov = cfg.fov;
  51. this.active = cfg.active !== false;
  52. }
  53.  
  54. /**
  55. * Field-of-view angle in degrees.
  56. *
  57. *
  58. * @property fov
  59. * @type Number
  60. * @default 45
  61. */
  62. set fov(value) {
  63. value = value || 45;
  64. if (this._fov === value) {
  65. return;
  66. }
  67. this._fov = value;
  68. }
  69.  
  70. get fov() {
  71. return this._fov;
  72. }
  73.  
  74. /**
  75. * Flag which indicates whether this StereoEffect is active or not.
  76. *
  77. * @property active
  78. * @type Boolean
  79. * @default true
  80. */
  81. set active(value) {
  82. value = !!value;
  83. if (this._active === value) {
  84. return;
  85. }
  86. this._active = value;
  87. this._active ? this._activate() : this._deactivate();
  88. }
  89.  
  90. get active() {
  91. return this._active;
  92. }
  93.  
  94. _activate() {
  95.  
  96. var scene = this.scene;
  97. var camera = scene.camera;
  98. var viewport = scene.viewport;
  99. var frustum = camera.frustum;
  100. var canvas = scene.canvas;
  101.  
  102. scene.passes = 2; // Two passes per render
  103. scene.clearEachPass = false; // Clear before first pass only
  104. camera.projection = "frustum"; // Camera in frustum projection mode
  105. viewport.autoBoundary = false; // Allow custom viewport boundary
  106.  
  107. var math = xeogl.math;
  108. var eye = math.vec3();
  109. var look = math.vec3();
  110. var up = math.vec3();
  111. var eyeLook = math.vec3();
  112. var eyeVec = math.vec3();
  113. var sepVec = math.vec3();
  114. var leftEye = math.vec3();
  115. var leftLook = math.vec3();
  116. var rightEye = math.vec3();
  117. var rightLook = math.vec3();
  118.  
  119. var self = this;
  120.  
  121. this._onSceneRendering = scene.on("rendering", function (e) {
  122.  
  123. var focalLength = -Math.abs(math.lenVec3(math.subVec3(camera.look, camera.eye, eyeLook)));
  124. var eyeSep = (1 / 30) * focalLength;
  125. var near = 0.1;
  126. var DTOR = 0.0174532925;
  127. var radians = DTOR * self._fov / 2;
  128. var wd2 = near * Math.tan(radians);
  129. var ndfl = near / focalLength;
  130. var canvasBoundary = canvas.boundary;
  131. var canvasWidth = canvasBoundary[2];
  132. var canvasHeight = canvasBoundary[3];
  133. var halfCanvasWidth = Math.round(canvasWidth / 2);
  134. var canvasAspectRatio = canvasWidth / canvasHeight;
  135.  
  136. switch (e.pass) {
  137.  
  138. case 0:
  139.  
  140. eye.set(camera.eye);
  141. look.set(camera.look);
  142. up.set(camera.up);
  143.  
  144. math.subVec3(look, eye, eyeVec);
  145. math.cross3Vec3(up, eyeVec, sepVec);
  146. math.normalizeVec3(sepVec);
  147. math.mulVec3Scalar(sepVec, eyeSep / 2.0);
  148.  
  149. // Find left and right viewpoints
  150.  
  151. math.subVec3(eye, sepVec, leftEye);
  152. math.subVec3(look, sepVec, leftLook);
  153.  
  154. math.addVec3(eye, sepVec, rightEye);
  155. math.addVec3(look, sepVec, rightLook);
  156.  
  157. // Set view transform to left side
  158.  
  159. camera.eye = leftEye;
  160. camera.look = leftLook;
  161.  
  162. // Set projection frustum to left half of view space
  163.  
  164. frustum.left = -canvasAspectRatio * wd2 - 0.5 * eyeSep * ndfl;
  165. frustum.right = canvasAspectRatio * wd2 - 0.5 * eyeSep * ndfl;
  166. frustum.top = wd2 * 2;
  167. frustum.bottom = -wd2 * 2;
  168.  
  169. // Set viewport to left half of canvas
  170.  
  171. viewport.boundary = [0, 0, halfCanvasWidth, canvasHeight];
  172.  
  173. break;
  174.  
  175. case 1:
  176.  
  177. // Set view transform to right side
  178.  
  179. camera.eye = rightEye;
  180. camera.look = rightLook;
  181.  
  182. // Set projection frustum to left half of view space
  183.  
  184. frustum.left = -canvasAspectRatio * wd2 + 0.5 * eyeSep * ndfl;
  185. frustum.right = canvasAspectRatio * wd2 + 0.5 * eyeSep * ndfl;
  186. frustum.top = wd2 * 2;
  187. frustum.bottom = -wd2 * 2;
  188.  
  189. // Set viewport to right half of canvas
  190.  
  191. viewport.boundary = [halfCanvasWidth, 0, halfCanvasWidth, canvasHeight];
  192.  
  193. break;
  194. }
  195. });
  196.  
  197. // Intercept Scene after each render
  198. // After the second pass we'll restore the thispoint
  199.  
  200. this._onSceneRendered = scene.on("rendered", function (e) {
  201. switch (e.pass) {
  202. case 1:
  203. camera.eye = eye;
  204. camera.look = look;
  205. camera.up = up;
  206. break;
  207. }
  208. });
  209. }
  210.  
  211. _deactivate() {
  212. var scene = this.scene;
  213. scene.passes = 1; // Don't need to restore scene.clearEachPass
  214. scene.viewport.autoBoundary = true;
  215. scene.off(this._onSceneRendering);
  216. scene.off(this._onSceneRendered);
  217. }
  218.  
  219. destroy() {
  220. super.destroy();
  221. this.active = false;
  222. }
  223. };
  224.