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

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

  1. /**
  2. A **Canvas** manages a {{#crossLink "Scene"}}Scene{{/crossLink}}'s HTML canvas and its WebGL context.
  3.  
  4. ## Overview
  5.  
  6. * Each {{#crossLink "Scene"}}Scene{{/crossLink}} provides a Canvas as a read-only property on itself.
  7. * When a {{#crossLink "Scene"}}Scene{{/crossLink}} is configured with the ID of
  8. an existing <a href="http://www.w3.org/TR/html5/scripting-1.html#the-canvas-element">HTMLCanvasElement</a>, then
  9. the Canvas will bind to that, otherwise the Canvas will automatically create its own.
  10. * A Canvas will fire a {{#crossLink "Canvas/boundary:event"}}{{/crossLink}} event whenever
  11. the <a href="http://www.w3.org/TR/html5/scripting-1.html#the-canvas-element">HTMLCanvasElement</a> resizes.
  12. * A Canvas is responsible for obtaining a WebGL context from
  13. the <a href="http://www.w3.org/TR/html5/scripting-1.html#the-canvas-element">HTMLCanvasElement</a>.
  14. * A Canvas also fires a {{#crossLink "Canvas/webglContextLost:event"}}{{/crossLink}} event when the WebGL context is
  15. lost, and a {{#crossLink "Canvas/webglContextRestored:event"}}{{/crossLink}} when it is restored again.
  16. * The various components within the parent {{#crossLink "Scene"}}Scene{{/crossLink}} will transparently recover on
  17. the {{#crossLink "Canvas/webglContextRestored:event"}}{{/crossLink}} event.
  18.  
  19. A Canvas also has
  20.  
  21. * a {{#crossLink "Progress"}}{{/crossLink}}, which shows a busy progress when a {{#crossLink "Model"}}{{/crossLink}}
  22. is loading, or when directed by application logic, and
  23.  
  24. ## Examples
  25.  
  26. * [Multiple canvases/scenes in a page](../../examples/#scenes_multipleScenes)
  27. * [Taking canvas snapshots](../../examples/#canvas_snapshot)
  28. * [Transparent canvas with background image](../../examples/#canvas_transparent)
  29. * [Canvas with multiple viewports](../../examples/#canvas_multipleViewports)
  30.  
  31. ## Usage
  32.  
  33. In the example below, we're creating a {{#crossLink "Scene"}}Scene{{/crossLink}} without specifying an HTML canvas element
  34. for it. This causes the {{#crossLink "Scene"}}Scene{{/crossLink}}'s Canvas component to create its own default element
  35. within the page. Then we subscribe to various events fired by that Canvas component.
  36.  
  37. ```` javascript
  38. var scene = new xeogl.Scene();
  39.  
  40. // Get the Canvas off the Scene
  41. // Since we did not configure the Scene with the ID of a DOM canvas element,
  42. // the Canvas will create its own canvas element in the DOM
  43. var canvas = scene.canvas;
  44.  
  45. // Get the WebGL context off the Canvas
  46. var gl = canvas.gl;
  47.  
  48. // Subscribe to Canvas size updates
  49. canvas.on("boundary", function(boundary) {
  50. //...
  51. });
  52.  
  53. // Subscribe to WebGL context loss events on the Canvas
  54. canvas.on("webglContextLost", function() {
  55. //...
  56. });
  57.  
  58. // Subscribe to WebGL context restored events on the Canvas
  59. canvas.on("webglContextRestored", function(gl) {
  60. var newContext = gl;
  61. //...
  62. });
  63. ````
  64.  
  65. When we want to bind the Canvas to an existing HTML canvas element, configure the
  66. {{#crossLink "Scene"}}{{/crossLink}} with the ID of the element, like this:
  67.  
  68. ```` javascript
  69. // Create a Scene, this time configuring it with the
  70. // ID of an existing DOM canvas element
  71. var scene = new xeogl.Scene({
  72. canvasId: "myCanvas"
  73. });
  74.  
  75. // ..and the rest of this example can be the same as the previous example.
  76.  
  77. ````
  78.  
  79. The {{#crossLink "Scene"}}{{/crossLink}} will attempt to get use WebGL 2, or fall back on WebGL 1
  80. if that's absent. If you just want WebGL 1, disable WebGL 2 like so:
  81.  
  82. ```` javascript
  83. var scene = new xeogl.Scene({
  84. canvasId: "myCanvas",
  85. webgl2 : true
  86. });
  87.  
  88. // ..and the rest of this example can be the same as the previous examples.
  89.  
  90. ````
  91.  
  92.  
  93. @class Canvas
  94. @module xeogl
  95. @submodule canvas
  96. @static
  97. @param {Scene} scene Parent scene
  98. @extends Component
  99. */
  100. import {Canvas2Image} from "../libs/canvas2image.js";
  101. import {core} from "../core.js";
  102. import {utils} from '../utils.js';
  103. import {math} from '../math/math.js';
  104. import {stats} from '../stats.js';
  105. import {Component} from '../component.js';
  106. import {Spinner} from './spinner.js';
  107. import {WEBGL_INFO} from '../webglInfo.js';
  108. import {componentClasses} from "../componentClasses.js";
  109.  
  110. const type = "xeogl.Canvas";
  111.  
  112. const WEBGL_CONTEXT_NAMES = [
  113. "webgl",
  114. "experimental-webgl",
  115. "webkit-3d",
  116. "moz-webgl",
  117. "moz-glweb20"
  118. ];
  119.  
  120. class Canvas extends Component {
  121.  
  122. /**
  123. JavaScript class name for this Component.
  124.  
  125. For example: "xeogl.AmbientLight", "xeogl.MetallicMaterial" etc.
  126.  
  127. @property type
  128. @type String
  129. @final
  130. */
  131. get type() {
  132. return type;
  133. }
  134.  
  135. init(cfg) {
  136.  
  137. super.init(cfg);
  138.  
  139. /**
  140. * The HTML canvas. When the {{#crossLink "Viewer"}}{{/crossLink}} was configured with the ID of an existing canvas within the DOM,
  141. * then this property will be that element, otherwise it will be a full-page canvas that this Canvas has
  142. * created by default, with a z-index of -10000.
  143. *
  144. * @property canvas
  145. * @type {HTMLCanvasElement}
  146. * @final
  147. */
  148. this.canvas = null;
  149.  
  150. /**
  151. * The WebGL rendering context.
  152. *
  153. * @property gl
  154. * @type {WebGLRenderingContext}
  155. * @final
  156. */
  157. this.gl = null;
  158.  
  159. /**
  160. * True when WebGL 2 support is enabled.
  161. *
  162. * @property webgl2
  163. * @type {Boolean}
  164. * @final
  165. */
  166. this.webgl2 = false; // Will set true in _initWebGL if WebGL is requested and we succeed in getting it.
  167.  
  168. /**
  169. * Indicates whether this Canvas is transparent.
  170. *
  171. * @property transparent
  172. * @type {Boolean}
  173. * @default {false}
  174. * @final
  175. */
  176. this.transparent = !!cfg.transparent;
  177.  
  178. /**
  179. * Attributes for the WebGL context
  180. *
  181. * @type {{}|*}
  182. */
  183. this.contextAttr = cfg.contextAttr || {};
  184. this.contextAttr.alpha = this.transparent;
  185.  
  186. if (this.contextAttr.preserveDrawingBuffer === undefined || this.contextAttr.preserveDrawingBuffer === null) {
  187. this.contextAttr.preserveDrawingBuffer = true;
  188. }
  189.  
  190. this.contextAttr.stencil = false;
  191. this.contextAttr.antialias = true;
  192. this.contextAttr.premultipliedAlpha = this.contextAttr.premultipliedAlpha !== false;
  193. this.contextAttr.antialias = this.contextAttr.antialias !== false;
  194.  
  195. if (!cfg.canvas) { // Canvas not supplied, create one automatically
  196. this._createCanvas();
  197. } else { // Canvas supplied
  198. if (utils.isString(cfg.canvas)) { // Canvas ID supplied - find the canvas
  199. this.canvas = document.getElementById(cfg.canvas);
  200. if (!this.canvas) { // Canvas not found - create one automatically
  201. this.error("Canvas element not found: " + utils.inQuotes(cfg.canvas) + " - creating default canvas instead.");
  202. this._createCanvas();
  203. }
  204. } else {
  205. this.canvas = cfg.canvas;
  206. }
  207. }
  208.  
  209. if (!this.canvas) {
  210. this.error("Faied to create canvas");
  211. return;
  212. }
  213.  
  214. // If the canvas uses css styles to specify the sizes make sure the basic
  215. // width and height attributes match or the WebGL context will use 300 x 150
  216.  
  217. this.canvas.width = this.canvas.clientWidth;
  218. this.canvas.height = this.canvas.clientHeight;
  219.  
  220. /**
  221. * Boundary of the Canvas in absolute browser window coordinates.
  222. *
  223. * ### Usage:
  224. *
  225. * ````javascript
  226. * var boundary = myScene.canvas.boundary;
  227. *
  228. * var xmin = boundary[0];
  229. * var ymin = boundary[1];
  230. * var width = boundary[2];
  231. * var height = boundary[3];
  232. * ````
  233. *
  234. * @property boundary
  235. * @type {{Array of Number}}
  236. * @final
  237. */
  238. this.boundary = [
  239. this.canvas.offsetLeft, this.canvas.offsetTop,
  240. this.canvas.clientWidth, this.canvas.clientHeight
  241. ];
  242.  
  243. this._createBackground();
  244.  
  245. // Get WebGL context
  246.  
  247. if (cfg.simulateWebGLContextLost) {
  248. if (window.WebGLDebugUtils) {
  249. this.canvas = WebGLDebugUtils.makeLostContextSimulatingCanvas(this.canvas);
  250. } else {
  251. this.error("To simulate context loss, please include WebGLDebugUtils");
  252. }
  253. }
  254.  
  255. this._initWebGL(cfg);
  256.  
  257. // Bind context loss and recovery handlers
  258.  
  259. const self = this;
  260.  
  261. this.canvas.addEventListener("webglcontextlost", this._webglcontextlostListener = function (event) {
  262. console.time("webglcontextrestored");
  263. self.scene._webglContextLost();
  264. /**
  265. * Fired whenever the WebGL context has been lost
  266. * @event webglcontextlost
  267. */
  268. self.fire("webglcontextlost");
  269. event.preventDefault();
  270. },
  271. false);
  272.  
  273. this.canvas.addEventListener("webglcontextrestored", this._webglcontextrestoredListener = function (event) {
  274. self._initWebGL();
  275. if (self.gl) {
  276. self.scene._webglContextRestored(self.gl);
  277. /**
  278. * Fired whenever the WebGL context has been restored again after having previously being lost
  279. * @event webglContextRestored
  280. * @param value The WebGL context object
  281. */
  282. self.fire("webglcontextrestored", self.gl);
  283. event.preventDefault();
  284. }
  285. console.timeEnd("webglcontextrestored");
  286. },
  287. false);
  288.  
  289. // Publish canvas size and position changes on each scene tick
  290.  
  291. let lastWindowWidth = null;
  292. let lastWindowHeight = null;
  293.  
  294. let lastCanvasWidth = null;
  295. let lastCanvasHeight = null;
  296.  
  297. let lastCanvasOffsetLeft = null;
  298. let lastCanvasOffsetTop = null;
  299.  
  300. let lastParent = null;
  301.  
  302. this._tick = this.scene.on("tick", function () {
  303.  
  304. const canvas = self.canvas;
  305.  
  306. const newWindowSize = (window.innerWidth !== lastWindowWidth || window.innerHeight !== lastWindowHeight);
  307. const newCanvasSize = (canvas.clientWidth !== lastCanvasWidth || canvas.clientHeight !== lastCanvasHeight);
  308. const newCanvasPos = (canvas.offsetLeft !== lastCanvasOffsetLeft || canvas.offsetTop !== lastCanvasOffsetTop);
  309.  
  310. const parent = canvas.parentElement;
  311. const newParent = (parent !== lastParent);
  312.  
  313. if (newWindowSize || newCanvasSize || newCanvasPos || newParent) {
  314.  
  315. self._spinner._adjustPosition();
  316.  
  317. if (newCanvasSize || newCanvasPos) {
  318.  
  319. const newWidth = canvas.clientWidth;
  320. const newHeight = canvas.clientHeight;
  321.  
  322. // TODO: Wasteful to re-count pixel size of each canvas on each canvas' resize
  323. if (newCanvasSize) {
  324. let countPixels = 0;
  325. let scene;
  326. for (const sceneId in core.scenes) {
  327. if (core.scenes.hasOwnProperty(sceneId)) {
  328. scene = core.scenes[sceneId];
  329. countPixels += scene.canvas.canvas.clientWidth * scene.canvas.canvas.clientHeight;
  330. }
  331. }
  332. stats.memory.pixels = countPixels;
  333.  
  334. canvas.width = canvas.clientWidth;
  335. canvas.height = canvas.clientHeight;
  336. }
  337.  
  338. const boundary = self.boundary;
  339.  
  340. boundary[0] = canvas.offsetLeft;
  341. boundary[1] = canvas.offsetTop;
  342. boundary[2] = newWidth;
  343. boundary[3] = newHeight;
  344.  
  345. /**
  346. * Fired whenever this Canvas's {{#crossLink "Canvas/boundary:property"}}{{/crossLink}} property changes.
  347. *
  348. * @event boundary
  349. * @param value The property's new value
  350. */
  351. self.fire("boundary", boundary);
  352.  
  353. lastCanvasWidth = newWidth;
  354. lastCanvasHeight = newHeight;
  355. }
  356.  
  357. if (newWindowSize) {
  358. lastWindowWidth = window.innerWidth;
  359. lastWindowHeight = window.innerHeight;
  360. }
  361.  
  362. if (newCanvasPos) {
  363. lastCanvasOffsetLeft = canvas.offsetLeft;
  364. lastCanvasOffsetTop = canvas.offsetTop;
  365. }
  366.  
  367. lastParent = parent;
  368. }
  369. });
  370.  
  371. this.canvas.oncontextmenu = function (e) {
  372. e.preventDefault();
  373. };
  374.  
  375. this._spinner = new Spinner(this.scene, {
  376. canvas: this.canvas
  377. });
  378.  
  379. // Set property, see definition further down
  380. this.backgroundColor = cfg.backgroundColor;
  381. this.backgroundImage = cfg.backgroundImage;
  382. }
  383.  
  384. /**
  385. * Creates a default canvas in the DOM.
  386. * @private
  387. */
  388. _createCanvas() {
  389.  
  390. const canvasId = "xeogl-canvas-" + math.createUUID();
  391. const body = document.getElementsByTagName("body")[0];
  392. const div = document.createElement('div');
  393.  
  394. const style = div.style;
  395. style.height = "100%";
  396. style.width = "100%";
  397. style.padding = "0";
  398. style.margin = "0";
  399. style.background = "rgba(0,0,0,0);";
  400. style.float = "left";
  401. style.left = "0";
  402. style.top = "0";
  403. style.position = "absolute";
  404. style.opacity = "1.0";
  405. style["z-index"] = "-10000";
  406.  
  407. div.innerHTML += '<canvas id="' + canvasId + '" style="width: 100%; height: 100%; float: left; margin: 0; padding: 0;"></canvas>';
  408.  
  409. body.appendChild(div);
  410.  
  411. this.canvas = document.getElementById(canvasId);
  412. }
  413.  
  414. /**
  415. * Creates a image element behind the canvas, for purpose of showing a custom background.
  416. * @private
  417. */
  418. _createBackground() {
  419.  
  420. const div = document.createElement('div');
  421. const style = div.style;
  422. style.padding = "0";
  423. style.margin = "0";
  424. style.background = null;
  425. style.backgroundImage = null;
  426. style.float = "left";
  427. style.left = "0";
  428. style.top = "0";
  429. style.width = "100%";
  430. style.height = "100%";
  431. style.position = "absolute";
  432. style.opacity = 1;
  433. style["z-index"] = "-20000";
  434.  
  435. this.canvas.parentElement.appendChild(div);
  436.  
  437. this._backgroundElement = div;
  438. }
  439.  
  440. _getElementXY(e) {
  441. let x = 0, y = 0;
  442. while (e) {
  443. x += (e.offsetLeft - e.scrollLeft);
  444. y += (e.offsetTop - e.scrollTop);
  445. e = e.offsetParent;
  446. }
  447. return {x: x, y: y};
  448. }
  449.  
  450. /**
  451. * Initialises the WebGL context
  452. * @private
  453. */
  454. _initWebGL(cfg) {
  455.  
  456. // Default context attribute values
  457.  
  458. if (false && cfg.webgl2) {
  459. try {
  460. this.gl = this.canvas.getContext("webgl2", this.contextAttr);
  461. } catch (e) { // Try with next context name
  462. }
  463. if (!this.gl) {
  464. this.warn('Failed to get a WebGL 2 context - defaulting to WebGL 1.');
  465. } else {
  466. this.webgl2 = true;
  467. }
  468. }
  469.  
  470. if (!this.gl) {
  471. for (let i = 0; !this.gl && i < WEBGL_CONTEXT_NAMES.length; i++) {
  472. try {
  473. this.gl = this.canvas.getContext(WEBGL_CONTEXT_NAMES[i], this.contextAttr);
  474. } catch (e) { // Try with next context name
  475. }
  476. }
  477. }
  478.  
  479. if (!this.gl) {
  480.  
  481. this.error('Failed to get a WebGL context');
  482.  
  483. /**
  484. * Fired whenever the canvas failed to get a WebGL context, which probably means that WebGL
  485. * is either unsupported or has been disabled.
  486. * @event webglContextFailed
  487. */
  488. this.fire("webglContextFailed", true, true);
  489. }
  490.  
  491. if (this.gl) {
  492. // Setup extension (if necessary) and hints for fragment shader derivative functions
  493. if (this.webgl2) {
  494. this.gl.hint(this.gl.FRAGMENT_SHADER_DERIVATIVE_HINT, this.gl.FASTEST);
  495. } else if (WEBGL_INFO.SUPPORTED_EXTENSIONS["OES_standard_derivatives"]) {
  496. const ext = this.gl.getExtension("OES_standard_derivatives");
  497. this.gl.hint(ext.FRAGMENT_SHADER_DERIVATIVE_HINT_OES, this.gl.FASTEST);
  498. }
  499. }
  500. }
  501.  
  502. /**
  503. Returns a snapshot of this Canvas as a Base64-encoded image.
  504.  
  505. When a callback is given, this method will capture the snapshot asynchronously, on the next animation frame,
  506. and return it via the callback.
  507.  
  508. When no callback is given, this method captures and returns the snapshot immediately. Note that is only
  509. possible when you have configured the Canvas's {{#crossLink "Scene"}}Scene{{/crossLink}} to preserve the
  510. WebGL drawing buffer, which has a performance overhead.
  511.  
  512. #### Usage:
  513.  
  514. ````javascript
  515. // Get snapshot asynchronously
  516. myScene.canvas.getSnapshot({
  517. width: 500, // Defaults to size of canvas
  518. height: 500,
  519. format: "png" // Options are "jpeg" (default), "png" and "bmp"
  520. }, function(imageDataURL) {
  521. imageElement.src = imageDataURL;
  522. });
  523.  
  524. // Get snapshot synchronously, requires that Scene be
  525. // configured with preserveDrawingBuffer; true
  526. imageElement.src = myScene.canvas.getSnapshot({
  527. width: 500,
  528. height: 500,
  529. format: "png"
  530. });
  531. ````
  532. @method getSnapshot
  533. @param {*} [params] Capture options.
  534. @param {Number} [params.width] Desired width of result in pixels - defaults to width of canvas.
  535. @param {Number} [params.height] Desired height of result in pixels - defaults to height of canvas.
  536. @param {String} [params.format="jpeg"] Desired format; "jpeg", "png" or "bmp".
  537. @param {Function} [ok] Callback to return the image data when taking a snapshot asynchronously.
  538. @returns {String} String-encoded image data when taking the snapshot synchronously. Returns null when the ````ok```` callback is given.
  539. */
  540. getSnapshot(params, ok) {
  541.  
  542. if (!this.canvas) {
  543. this.error("Can't get snapshot - no canvas.");
  544. ok(null);
  545. return;
  546. }
  547.  
  548. if (ok) { // Asynchronous
  549. const self = this;
  550. requestAnimationFrame(function () {
  551. self.scene.render(true); // Force-render a frame
  552. ok(self._getSnapshot(params));
  553. });
  554. } else {
  555. return this._getSnapshot(params);
  556. }
  557. }
  558.  
  559. _getSnapshot(params) {
  560. params = params || {};
  561. const width = params.width || this.canvas.width;
  562. const height = params.height || this.canvas.height;
  563. const format = params.format || "jpeg";
  564. let image;
  565. switch (format) {
  566. case "jpeg":
  567. image = Canvas2Image.saveAsJPEG(this.canvas, false, width, height);
  568. break;
  569. case "png":
  570. image = Canvas2Image.saveAsPNG(this.canvas, true, width, height);
  571. break;
  572. case "bmp":
  573. image = Canvas2Image.saveAsBMP(this.canvas, true, width, height);
  574. break;
  575. default:
  576. this.error("Unsupported snapshot format: '" + format
  577. + "' - supported types are 'jpeg', 'bmp' and 'png' - defaulting to 'jpeg'");
  578. image = Canvas2Image.saveAsJPEG(this.canvas, true, width, height);
  579. }
  580. return image.src;
  581. }
  582.  
  583. /**
  584. Reads colors of pixels from the last rendered frame.
  585.  
  586. <p>Call this method like this:</p>
  587.  
  588. ````JavaScript
  589.  
  590. // Ignore transparent pixels (default is false)
  591. var opaqueOnly = true;
  592.  
  593. var colors = new Float32Array(8);
  594.  
  595. myCanvas.readPixels([ 100, 22, 12, 33 ], colors, 2, opaqueOnly);
  596. ````
  597.  
  598. Then the r,g,b components of the colors will be set to the colors at those pixels.
  599.  
  600. @param {Float32Array} pixels
  601. @param {Float32Array} colors
  602. @param {Number} size
  603. @param {Boolean} opaqueOnly
  604. */
  605. readPixels(pixels, colors, size, opaqueOnly) {
  606. return this.scene._renderer.readPixels(pixels, colors, size, opaqueOnly);
  607. }
  608.  
  609. /**
  610. * Simulates lost WebGL context.
  611. */
  612. loseWebGLContext() {
  613. if (this.canvas.loseContext) {
  614. this.canvas.loseContext();
  615. }
  616. }
  617.  
  618. /**
  619. A background color for the canvas. This is overridden by {{#crossLink "Canvas/backgroundImage:property"}}{{/crossLink}}.
  620.  
  621. You can set this to a new color at any time.
  622.  
  623. @property backgroundColor
  624. @type Float32Array
  625. @default null
  626. */
  627. set backgroundColor(value) {
  628. if (!value) {
  629. this._backgroundColor = null;
  630. } else {
  631. (this._backgroundColor = this._backgroundColor || math.vec4()).set(value || [0, 0, 0, 1]);
  632. if (!this._backgroundImageSrc) {
  633. const rgb = "rgb(" + Math.round(this._backgroundColor[0] * 255) + ", " + Math.round(this._backgroundColor[1] * 255) + "," + Math.round(this._backgroundColor[2] * 255) + ")";
  634. this._backgroundElement.style.background = rgb;
  635. }
  636. }
  637. }
  638.  
  639. get backgroundColor() {
  640. return this._backgroundColor;
  641. }
  642.  
  643. /**
  644. URL of a background image for the canvas. This is overrided by {{#crossLink "Canvas/backgroundColor/property"}}{{/crossLink}}.
  645.  
  646. You can set this to a new file path at any time.
  647.  
  648. @property backgroundImage
  649. @type String
  650. */
  651. set backgroundImage(value) {
  652. if (!value) {
  653. return;
  654. }
  655. if (!utils.isString(value)) {
  656. this.error("Value for 'backgroundImage' should be a string");
  657. return;
  658. }
  659. if (value === this._backgroundImageSrc) { // Already loaded this image
  660. return;
  661. }
  662. this._backgroundElement.style.backgroundImage = "url('" + value + "')";
  663. this._backgroundImageSrc = value;
  664. if (!this._backgroundImageSrc) {
  665. const rgb = "rgb(" + Math.round(this._backgroundColor[0] * 255) + ", " + Math.round(this._backgroundColor[1] * 255) + "," + Math.round(this._backgroundColor[2] * 255) + ")";
  666. this._backgroundElement.style.background = rgb;
  667. }
  668. }
  669.  
  670. get backgroundImage() {
  671. return this._backgroundImageSrc;
  672. }
  673.  
  674. /**
  675. The busy {{#crossLink "Spinner"}}{{/crossLink}} for this Canvas.
  676.  
  677. @property spinner
  678. @type Spinner
  679. @final
  680. */
  681. get spinner() {
  682. return this._spinner;
  683. }
  684.  
  685. destroy() {
  686. this.scene.off(this._tick);
  687. // Memory leak avoidance
  688. this.canvas.removeEventListener("webglcontextlost", this._webglcontextlostListener);
  689. this.canvas.removeEventListener("webglcontextrestored", this._webglcontextrestoredListener);
  690. this.canvas = null;
  691. this.gl = null;
  692. super.destroy();
  693. }
  694. }
  695.  
  696. componentClasses[type] = Canvas;
  697.  
  698. export {Canvas};
  699.