/home/lindsay/xeolabs/xeogl-next/xeogl/src/lighting/pointLight.js
API Docs for:

File: /home/lindsay/xeolabs/xeogl-next/xeogl/src/lighting/pointLight.js

  1. /**
  2. A **PointLight** defines a positional light source that originates from a single point and spreads outward in all directions,
  3. to illuminate {{#crossLink "Mesh"}}Meshes{{/crossLink}}.
  4.  
  5. <a href="../../examples/#lights_point_world_normalMap"><img src="http://i.giphy.com/3o6ZsZoFGIOJ2nlmN2.gif"></img></a>
  6.  
  7. ## Overview
  8.  
  9. * PointLights have a position, but no direction.
  10. * PointLights may be defined in either **World** or **View** coordinate space. When in World-space, their positions
  11. are relative to the World coordinate system, and will appear to move as the {{#crossLink "Camera"}}{{/crossLink}} moves.
  12. When in View-space, their positions are relative to the View coordinate system, and will behave as if fixed to the viewer's
  13. head as the {{#crossLink "Camera"}}{{/crossLink}} moves.
  14. * PointLights have {{#crossLink "PointLight/constantAttenuation:property"}}{{/crossLink}}, {{#crossLink "PointLight/linearAttenuation:property"}}{{/crossLink}} and
  15. {{#crossLink "PointLight/quadraticAttenuation:property"}}{{/crossLink}} factors, which indicate how their intensity attenuates over distance.
  16. * {{#crossLink "AmbientLight"}}{{/crossLink}}, {{#crossLink "DirLight"}}{{/crossLink}},
  17. {{#crossLink "SpotLight"}}{{/crossLink}} and {{#crossLink "PointLight"}}{{/crossLink}} instances are registered by ID
  18. on {{#crossLink "Scene/lights:property"}}Scene#lights{{/crossLink}} for convenient access.
  19.  
  20. ## Examples
  21.  
  22. * [View-space positional three-point lighting](../../examples/#lights_point_view_threePoint)
  23. * [World-space positional three-point lighting](../../examples/#lights_point_world_threePoint)
  24. * [World-space point light and normal map](../../examples/#lights_point_world_normalMap)
  25.  
  26. ## Usage
  27.  
  28. In the example below we'll customize the default Scene's light sources, defining an AmbientLight and a couple of
  29. PointLights, then create a Phong-shaded box mesh.
  30.  
  31. ````javascript
  32. new xeogl.AmbientLight({
  33. color: [0.8, 0.8, 0.8],
  34. intensity: 0.5
  35. });
  36.  
  37. new xeogl.PointLight({
  38. pos: [-100, 0, 100],
  39. color: [0.3, 0.3, 0.5],
  40. intensity: .7
  41. constantAttenuation: 0,
  42. linearAttenuation: 0,
  43. quadraticAttenuation: 0,
  44. space: "view"
  45. });
  46.  
  47. new xeogl.PointLight({
  48. pos: [0, 100, 100],
  49. color: [0.5, 0.7, 0.5],
  50. intensity: 1
  51. constantAttenuation: 0,
  52. linearAttenuation: 0,
  53. quadraticAttenuation: 0,
  54. space: "view"
  55. });
  56.  
  57. // Create box mesh
  58. new xeogl.Mesh({
  59. material: new xeogl.PhongMaterial({
  60. ambient: [0.5, 0.5, 0.5],
  61. diffuse: [1,0.3,0.3]
  62. }),
  63. geometry: new xeogl.BoxGeometry()
  64. });
  65. ````
  66.  
  67.  
  68. @class PointLight
  69. @module xeogl
  70. @submodule lighting
  71. @constructor
  72. @extends Component
  73. @param [owner] {Component} Owner component. When destroyed, the owner will destroy this component as well. Creates this component within the default {{#crossLink "Scene"}}{{/crossLink}} when omitted.
  74. @param [cfg] {*} The PointLight configuration
  75. @param [cfg.id] {String} Optional ID, unique among all components in the parent {{#crossLink "Scene"}}Scene{{/crossLink}}, generated automatically when omitted.
  76. @param [cfg.meta] {String:Object} Optional map of user-defined metadata to attach to this PointLight.
  77. @param [cfg.pos=[ 1.0, 1.0, 1.0 ]] {Float32Array} Position, in either World or View space, depending on the value of the **space** parameter.
  78. @param [cfg.color=[0.7, 0.7, 0.8 ]] {Float32Array} Color of this PointLight.
  79. @param [cfg.intensity=1.0] {Number} Intensity of this PointLight, as a factor in range ````[0..1]````.
  80. @param [cfg.constantAttenuation=0] {Number} Constant attenuation factor.
  81. @param [cfg.linearAttenuation=0] {Number} Linear attenuation factor.
  82. @param [cfg.quadraticAttenuation=0] {Number} Quadratic attenuation factor.
  83. @param [cfg.space="view"] {String} The coordinate system this PointLight is defined in - "view" or "world".
  84. @param [cfg.shadow=false] {Boolean} Flag which indicates if this PointLight casts a shadow.
  85. */
  86. import {Component} from '../component.js';
  87. import {State} from '../renderer/state.js';
  88. import {RenderBuffer} from '../renderer/renderBuffer.js';
  89. import {math} from '../math/math.js';
  90. import {componentClasses} from "./../componentClasses.js";
  91.  
  92. const type = "xeogl.PointLight";
  93.  
  94. class PointLight extends Component {
  95.  
  96. /**
  97. JavaScript class name for this Component.
  98.  
  99. For example: "xeogl.AmbientLight", "xeogl.MetallicMaterial" etc.
  100.  
  101. @property type
  102. @type String
  103. @final
  104. */
  105. get type() {
  106. return type;
  107. }
  108.  
  109. init(cfg) {
  110.  
  111. super.init(cfg);
  112.  
  113. const self = this;
  114.  
  115. this._shadowRenderBuf = null;
  116. this._shadowViewMatrix = null;
  117. this._shadowProjMatrix = null;
  118. this._shadowViewMatrixDirty = true;
  119. this._shadowProjMatrixDirty = true;
  120.  
  121. this._state = new State({
  122. type: "point",
  123. pos: math.vec3([1.0, 1.0, 1.0]),
  124. color: math.vec3([0.7, 0.7, 0.8]),
  125. intensity: 1.0, attenuation: [0.0, 0.0, 0.0],
  126. space: cfg.space || "view",
  127. shadow: false,
  128. shadowDirty: true,
  129.  
  130. getShadowViewMatrix: (function () {
  131. const look = math.vec3([0, 0, 0]);
  132. const up = math.vec3([0, 1, 0]);
  133. return function () {
  134. if (self._shadowViewMatrixDirty) {
  135. if (!self._shadowViewMatrix) {
  136. self._shadowViewMatrix = math.identityMat4();
  137. }
  138. math.lookAtMat4v(self._state.pos, look, up, self._shadowViewMatrix);
  139. self._shadowViewMatrixDirty = false;
  140. }
  141. return self._shadowViewMatrix;
  142. };
  143. })(),
  144.  
  145. getShadowProjMatrix: function () {
  146. if (self._shadowProjMatrixDirty) { // TODO: Set when canvas resizes
  147. if (!self._shadowProjMatrix) {
  148. self._shadowProjMatrix = math.identityMat4();
  149. }
  150. const canvas = self.scene.canvas.canvas;
  151. math.perspectiveMat4(70 * (Math.PI / 180.0), canvas.clientWidth / canvas.clientHeight, 0.1, 500.0, self._shadowProjMatrix);
  152. self._shadowProjMatrixDirty = false;
  153. }
  154. return self._shadowProjMatrix;
  155. },
  156.  
  157. getShadowRenderBuf: function () {
  158. if (!self._shadowRenderBuf) {
  159. self._shadowRenderBuf = new RenderBuffer(self.scene.canvas.canvas, self.scene.canvas.gl, {size: [1024, 1024]});
  160. }
  161. return self._shadowRenderBuf;
  162. }
  163. });
  164.  
  165. this.pos = cfg.pos;
  166. this.color = cfg.color;
  167. this.intensity = cfg.intensity;
  168. this.constantAttenuation = cfg.constantAttenuation;
  169. this.linearAttenuation = cfg.linearAttenuation;
  170. this.quadraticAttenuation = cfg.quadraticAttenuation;
  171. this.shadow = cfg.shadow;
  172.  
  173. this.scene._lightCreated(this);
  174. }
  175.  
  176.  
  177. /**
  178. The position of this PointLight.
  179.  
  180. This will be either World- or View-space, depending on the value of {{#crossLink "PointLight/space:property"}}{{/crossLink}}.
  181.  
  182. @property pos
  183. @default [1.0, 1.0, 1.0]
  184. @type Array(Number)
  185. */
  186. set pos(value) {
  187. this._state.pos.set(value || [1.0, 1.0, 1.0]);
  188. this._shadowViewMatrixDirty = true;
  189. this._renderer.imageDirty();
  190. }
  191.  
  192. get pos() {
  193. return this._state.pos;
  194. }
  195.  
  196. /**
  197. The color of this PointLight.
  198.  
  199. @property color
  200. @default [0.7, 0.7, 0.8]
  201. @type Float32Array
  202. */
  203. set color(value) {
  204. this._state.color.set(value || [0.7, 0.7, 0.8]);
  205. this._renderer.imageDirty();
  206. }
  207.  
  208. get color() {
  209. return this._state.color;
  210. }
  211.  
  212. /**
  213. The intensity of this PointLight.
  214.  
  215. @property intensity
  216. @default 1.0
  217. @type Number
  218. */
  219. set intensity(value) {
  220. value = value !== undefined ? value : 1.0;
  221. this._state.intensity = value;
  222. this._renderer.imageDirty();
  223. }
  224.  
  225. get intensity() {
  226. return this._state.intensity;
  227. }
  228.  
  229. /**
  230. The constant attenuation factor for this PointLight.
  231.  
  232. @property constantAttenuation
  233. @default 0
  234. @type Number
  235. */
  236. set constantAttenuation(value) {
  237. this._state.attenuation[0] = value || 0.0;
  238. this._renderer.imageDirty();
  239. }
  240.  
  241. get constantAttenuation() {
  242. return this._state.attenuation[0];
  243. }
  244.  
  245. /**
  246. The linear attenuation factor for this PointLight.
  247.  
  248. @property linearAttenuation
  249. @default 0
  250. @type Number
  251. */
  252. set linearAttenuation(value) {
  253. this._state.attenuation[1] = value || 0.0;
  254. this._renderer.imageDirty();
  255. }
  256.  
  257. get linearAttenuation() {
  258. return this._state.attenuation[1];
  259. }
  260.  
  261. /**
  262. The quadratic attenuation factor for this Pointlight.
  263.  
  264. @property quadraticAttenuation
  265. @default 0
  266. @type Number
  267. */
  268. set quadraticAttenuation(value) {
  269. this._state.attenuation[2] = value || 0.0;
  270. this._renderer.imageDirty();
  271. }
  272.  
  273. get quadraticAttenuation() {
  274. return this._state.attenuation[2];
  275. }
  276.  
  277. /**
  278. Flag which indicates if this PointLight casts a shadow.
  279.  
  280. @property shadow
  281. @default false
  282. @type Boolean
  283. */
  284. set shadow(value) {
  285. value = !!value;
  286. if (this._state.shadow === value) {
  287. return;
  288. }
  289. this._state.shadow = value;
  290. this._shadowViewMatrixDirty = true;
  291. this._renderer.imageDirty();
  292. }
  293.  
  294. get shadow() {
  295. return this._state.shadow;
  296. }
  297.  
  298. destroy() {
  299. super.destroy();
  300. this._state.destroy();
  301. if (this._shadowRenderBuf) {
  302. this._shadowRenderBuf.destroy();
  303. }
  304. this.scene._lightDestroyed(this);
  305. }
  306. }
  307.  
  308. componentClasses[type] = PointLight;
  309.  
  310. export {PointLight};
  311.