/home/lindsay/xeolabs/xeogl-next/xeogl/examples/js/models/sceneJSModel.js
API Docs for:

File: /home/lindsay/xeolabs/xeogl-next/xeogl/examples/js/models/sceneJSModel.js

  1. /**
  2. A **SceneJSModel** is a {{#crossLink "Model"}}{{/crossLink}} that
  3. imports content from the JSON-based <a href="http://scenejs.org">SceneJS</a> scene definition format.
  4.  
  5. <a href="../../examples/#importing_scenejs_tronTank"><img src="http://i.giphy.com/l3vR50pFTpEbJTztS.gif"></img></a>
  6.  
  7. ## Overview
  8.  
  9. * A SceneJSModel is a container of {{#crossLink "Component"}}Components{{/crossLink}} that loads itself from a
  10. SceneJS scene definition, given as either a JSON file or a JavaScript object (POJO).
  11. * It begins loading as soon as you either set its {{#crossLink "SceneJSModel/src:property"}}{{/crossLink}}
  12. property to the location of a valid SceneJS JSON file, or set its {{#crossLink "SceneJSModel/data:property"}}{{/crossLink}} property to a
  13. valid POJO.
  14. * You can set these properties to new values at any time, which causes
  15. the SceneJSModel to clear itself and load fresh components.
  16. * Can be configured to do a best-effort conversion of SceneJS Phong materials into xeogl's PBR {{#crossLink "PBRMetalness"}}{{/crossLink}} or {{#crossLink "SpecularMaterials"}}{{/crossLink}}.
  17.  
  18. It inherits these capabilities from its {{#crossLink "Model"}}{{/crossLink}} base class:
  19.  
  20. * Allows you to access and manipulate the components within it.
  21. * Can be transformed within World-space by attaching it to a {{#crossLink "Transform"}}{{/crossLink}}.
  22. * Provides its World-space boundary as a {{#crossLink "Boundary3D"}}{{/crossLink}}.
  23.  
  24. <img src="../../../assets/images/SceneJSModel.png"></img>
  25.  
  26. ## SceneJS Support
  27.  
  28. SceneJSModel was developed to import the [Tron Tank model](../../examples/#importing_scenejs_tronTank). As such,
  29. it only imports a limited subset of the SceneJS scene definition API. <b>Use with caution</b> and be prepared to
  30. fix and contribute missing functionality!
  31.  
  32. SceneJS nodes supported so far:
  33.  
  34. * ````"node"````
  35. * ````"rotate"````
  36. * ````"translate"````
  37. * ````"scale"````
  38. * ````"material"````
  39. * ````"texture"````
  40. * ````"fresnel"````
  41. * ````"flags"````
  42. * ````"geometry"````
  43. * ````"layer"````
  44. * ````"stage"````
  45.  
  46. Unsupported API features include:
  47.  
  48. * Lights
  49. * Cameras
  50. * Shared node cores
  51. * SceneJS plugins
  52.  
  53. ## Examples
  54.  
  55. * [Importing POJO defining geometry with diffuse, specular and normal maps](../../examples/#importing_scenejs_pojo_textures)
  56. * [Importing POJO defining transparent geometry](../../examples/#importing_scenejs_pojo_transparency)
  57. * [Importing JSON file defining geometry with diffuse, specular and normal maps](../../examples/#importing_scenejs_json_textures)
  58. * [Importing JSON file defining the SceneJS Tron Tank](../../examples/#importing_scenejs_tronTank)
  59.  
  60. ## Usage
  61.  
  62. #### Loading a POJO scene definition
  63.  
  64. The simplest way to import SceneJS content is by setting a POJO on the SceneJSModel's {{#crossLink "SceneJSModel/data:property"}}data{{/crossLink}}
  65. property:
  66.  
  67. ````javascript
  68. var pojoModel = new xeogl.SceneJSModel({
  69. id: "myModel",
  70.  
  71. // Our POJO scene definition
  72. data: {
  73. type: "node",
  74. nodes: [
  75. {
  76. type: "rotate",
  77. id: "myRotate",
  78. nodes: [
  79. {
  80. type: "geometry",
  81. id: "boxEntity",
  82. primitive: "triangles",
  83. positions: [
  84. 2, 2, 2, -2, 2, 2, -2, -2, 2, 2, -2, 2, 2, 2, 2, 2, -2,
  85. 2, 2, -2, -2, 2, 2, -2, 2, 2, 2, 2, 2, -2, -2, 2, -2,
  86. -2, 2, 2, -2, 2, 2, -2, 2, -2, -2, -2, -2, -2, -2, 2,
  87. -2, -2, -2, 2, -2, -2, 2, -2, 2, -2, -2, 2, 2, -2, -2,
  88. -2, -2, -2, -2, 2, -2, 2, 2, -2
  89. ],
  90. normals: [
  91. 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 1, 0, 0, 1, 0, 0, 1,
  92. 0, 0, 1, 0, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, -1, 0,
  93. 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, 0, -1, 0, 0, -1, 0, 0,
  94. -1, 0, 0, -1, 0, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1
  95. ],
  96. uv: [
  97. 5, 5, 0, 5, 0, 0, 5, 0, 0, 5, 0, 0, 5, 0, 5, 5,
  98. 5, 0, 5, 5, 0, 5, 0, 0, 5, 5, 0, 5, 0, 0, 5, 0,
  99. 0, 0, 5, 0, 5, 5, 0, 5, 0, 0, 5, 0, 5, 5, 0, 5
  100. ],
  101. indices: [
  102. 0, 1, 2, 0, 2, 3, 4, 5, 6, 4, 6, 7, 8, 9, 10, 8, 10, 11,
  103. 12, 13, 14, 12, 14, 15, 16, 17, 18, 16, 18, 19, 20, 21,
  104. 22, 20, 22, 23
  105. ]
  106. }
  107. ]
  108. }
  109. ]
  110. }
  111. });
  112.  
  113. // Set camera position
  114. var camera = pojoModel.scene.camera;
  115. camera.eye = [0, 0, -25];
  116. camera.look = [0, 0, 0];
  117. camera.up = [0, 1, 0];
  118. ````
  119.  
  120. #### Finding components
  121.  
  122. Our SceneJSModel has now created various xeogl components
  123. within itself, which we can find by their IDs. In this particular example, our POJO has a SceneJS ````"rotate"```` node
  124. with ID ````"myRotate"````. Our SceneJSModel parsed that into a {{#crossLink "Rotate"}}{{/crossLink}} component with
  125. ID ````"myModel.myRotate"````.
  126.  
  127. To see what components our SceneJSModel created, we can drop this expression into the browser's JavaScript
  128. debug console (we're using Chrome here):
  129.  
  130. ````
  131. pojoModel.types;
  132. ````
  133.  
  134. The result is the value of the SceneJSModel's {{#crossLink "Model/types:property"}}types{{/crossLink}} map, which
  135. contains its xeogl components, mapped to their types:
  136.  
  137. <img src="../../../assets/images/screenshots/SceneJSModel_console_tankModel.types.png"></img>
  138.  
  139. Here we've expanded the {{#crossLink "Rotate"}}{{/crossLink}} components, and we can see
  140. our {{#crossLink "Rotate"}}{{/crossLink}}. **Note that its ID is prefixed with the ID of the SceneJSModel.**
  141.  
  142. Let's get that {{#crossLink "Rotate"}}{{/crossLink}} from our SceneJSModel's
  143. {{#crossLink "Model/components:property"}}{{/crossLink}} map and set it spinning:
  144.  
  145. ```` JavaScript
  146. var rotate = pojoModel.components["myModel.myRotate"];
  147.  
  148. pojoModel.scene.on("tick", function() {
  149. rotate.angle += 0.2;
  150. });
  151. ````
  152.  
  153. #### Loading a JSON scene definition
  154.  
  155. As shown in the example below, we can also import a SceneJS scene definition from a JSON file (eg. <a href="../../examples/models/scenejs/tronTank.json">tronTank.json</a>).
  156. Note that we need to wait for the SceneJSModel's {{#crossLink "SceneJSModel/loaded:event"}}{{/crossLink}} event before we
  157. can access its components. In this example we're also showing how a SceneJSModel can be attached to a modeling {{#crossLink "Transform"}}{{/crossLink}}
  158. hierarchy to transform it within World space (see {{#crossLink "Model"}}{{/crossLink}}).
  159.  
  160. ````javascript
  161. // Import SceneJS JSON model
  162. var tankModel = new xeogl.SceneJSModel({
  163. id: "tankModel",
  164.  
  165. // Path to our JSON scene definition file
  166. src: "models/scenejs/tronTank.json",
  167.  
  168. // We can also bolt on a hierarchy of modeling transforms,
  169. // to transform the entire SceneJSModel in World space
  170. transform: new xeogl.Rotate({
  171. xyz: [0, 1, 0],
  172. angle: 0,
  173. parent: new xeogl.Translate({
  174. xyz: [0, 0, 0]
  175. })
  176. })
  177. });
  178.  
  179. // Once our SceneJSModel has loaded, we can access its components
  180. tankModel.on("loaded", function() {
  181.  
  182. tankModel.components["tankModel.gunDir"].angle = gunDir;
  183.  
  184. // Set camera position
  185. var camera = tankModel.scene.camera;
  186. camera.eye = [0, 0, -70];
  187. camera.look = [0, 0, 0];
  188. camera.up = [0, 1, 0];
  189. });
  190. ````
  191.  
  192. #### Converting materials to PBR
  193.  
  194.  
  195. ````javascript
  196. var pbrSpecularTankModel = new xeogl.SceneJSModel({
  197. src: "models/scenejs/tronTank.json",
  198. materialWorkflow: "SpecularMaterial"
  199. });
  200. ````
  201.  
  202. ````javascript
  203. var pbrMetalnessTankModel = new xeogl.SceneJSModel({
  204. src: "models/scenejs/tronTank.json",
  205. materialWorkflow: "MetallicMaterial"
  206. });
  207. ````
  208.  
  209. @class SceneJSModel
  210. @module xeogl
  211. @submodule models
  212. @constructor
  213. @param [scene] {Scene} Parent {{#crossLink "Scene"}}Scene{{/crossLink}} - creates this SceneJSModel in the default
  214. {{#crossLink "Scene"}}Scene{{/crossLink}} when omitted.
  215. @param [cfg] {*} Configs
  216. @param [cfg.id] {String} Optional ID, unique among all components in the parent {{#crossLink "Scene"}}Scene{{/crossLink}},
  217. generated automatically when omitted.
  218. @param [cfg.entityType] {String} Optional entity classification when using within a semantic data model. See the {{#crossLink "Object"}}{{/crossLink}} documentation for usage.
  219. @param [cfg.meta] {String:Object} Optional map of user-defined metadata to attach to this SceneJSModel.
  220. @param [cfg.materialWorkflow] {String} Selects material workflow - "classic" | "pbrMatalness" | "pbrSpecular"
  221. @param [cfg.src] {String} Path to a SceneJS JSON scene description file.
  222. @param [cfg.data] {String} Path to a SceneJS JSON scene description file.
  223. @param [cfg.position=[0,0,0]] {Float32Array} The Model's local 3D position.
  224. @param [cfg.scale=[1,1,1]] {Float32Array} The SceneJSModel's local scale.
  225. @param [cfg.rotation=[0,0,0]] {Float32Array} The SceneJSModel's local rotation, as Euler angles given in degrees.
  226. @param [cfg.matrix=[1,0,0,0,0,1,0,0,0,0,1,0,0,0,0,1] {Float32Array} The SceneJSModel's local transform matrix. Overrides the position, scale and rotation parameters.
  227. @extends Geometry
  228. */
  229. {
  230.  
  231. xeogl.SceneJSModel = class xeoglSceneJSModel extends xeogl.Model {
  232.  
  233. init(cfg) {
  234. super.init(cfg);
  235. this._src = null;
  236. this.materialWorkflow = cfg.materialWorkflow;
  237. this.src = cfg.src;
  238. this.data = cfg.data;
  239. }
  240.  
  241. /**
  242. Selects which xeogl material type to create from each SceneJS Phong material.
  243.  
  244. Causes the SceneJSModel to attempt a best-effort conversion.
  245.  
  246. Update this at any time to reconvert the materials.
  247.  
  248. Fires a {{#crossLink "SceneJSModel/materialWorkFlow:event"}}{{/crossLink}} event on change.
  249.  
  250. @property materialWorkflow
  251. @type {*}
  252. */
  253. set materialWorkflow(value) {
  254.  
  255. value = value || "PhongMaterial";
  256.  
  257. if (value !== "MetallicMaterial" && value !== "SpecularMaterial" && value !== "PhongMaterial") {
  258. this.error("Unsupported value for 'materialWorkflow' - defaulting to 'PhongMaterial'");
  259. value = "PhongMaterial";
  260. }
  261.  
  262. if (this._materialWorkflow === value) {
  263. return;
  264. }
  265.  
  266. this._materialWorkflow = value;
  267.  
  268. //this.destroyAll();
  269. //
  270. //this._src = null;
  271. //
  272. //this._parse(this._materialWorkflow, null, null, null);
  273.  
  274. /**
  275. Fired whenever this SceneJSModel's {{#crossLink "SceneJSModel/materialWorkflow:property"}}{{/crossLink}} property changes.
  276. @event materialWorkflow
  277. @param value The property's new value
  278. */
  279. this.fire("materialWorkflow", this._materialWorkflow);
  280. }
  281.  
  282. get materialWorkFlow() {
  283. return this._materialWorkflow;
  284. }
  285.  
  286. /**
  287. Path to the SceneJS JSON scene description file.
  288.  
  289. Update this at any time to clear and re-import content.
  290.  
  291. Fires a {{#crossLink "SceneJSModel/src:event"}}{{/crossLink}} event on change.
  292.  
  293. @property src
  294. @type String
  295. */
  296. set src(value) {
  297.  
  298. if (!value) {
  299. return;
  300. }
  301.  
  302. if (!xeogl._isString(value)) {
  303. this.error("Value for 'src' should be a string");
  304. return;
  305. }
  306.  
  307. if (value === this._src) { // Already loaded this SceneJSModel
  308.  
  309. /**
  310. Fired whenever this SceneJSModel has finished loading the SceneJS JSON file
  311. specified by {{#crossLink "SceneJSModel/src:property"}}{{/crossLink}}.
  312. @event loaded
  313. */
  314. this.fire("loaded", true, true);
  315.  
  316. return;
  317. }
  318.  
  319. this.destroyAll();
  320.  
  321. this._data = null;
  322.  
  323. this._src = value;
  324.  
  325. // Increment processes represented by loading spinner
  326. // Spinner appears as soon as count is non-zero
  327.  
  328. var spinner = this.scene.canvas.spinner;
  329. spinner.processes++;
  330.  
  331. var self = this;
  332.  
  333. load(this._src, function (node) {
  334.  
  335. var group = self;
  336.  
  337. self._parse(node, group, null, null);
  338.  
  339. // Decrement processes represented by loading spinner
  340. // Spinner disappears if the count is now zero
  341. spinner.processes--;
  342.  
  343. xeogl.scheduleTask(function () {
  344. self.fire("loaded", true);
  345. });
  346. },
  347.  
  348. function (msg) {
  349.  
  350. spinner.processes--;
  351.  
  352. self.error("Failed to load JSON file: " + msg);
  353.  
  354. self.fire("failed", msg);
  355. });
  356.  
  357. /**
  358. Fired whenever this SceneJSModel's {{#crossLink "SceneJSModel/src:property"}}{{/crossLink}} property changes.
  359. @event src
  360. @param value The property's new value
  361. */
  362. this.fire("src", this._src);
  363. }
  364.  
  365. get src() {
  366. return this._src;
  367. }
  368.  
  369. /**
  370. A SceneJS POJO scene definition.
  371.  
  372. Update this at any time to clear and re-import content.
  373.  
  374. Fires a {{#crossLink "SceneJSModel/data:event"}}{{/crossLink}} event on change.
  375.  
  376. @property data
  377. @type {*}
  378. */
  379. set data(value) {
  380.  
  381. if (!value) {
  382. return;
  383. }
  384.  
  385. this.destroyAll();
  386.  
  387. this._src = null;
  388.  
  389. this._data = value;
  390.  
  391. var group = this;
  392.  
  393. this._parse(this._data, group, null, null);
  394.  
  395. var self = this;
  396.  
  397. xeogl.scheduleTask(function () {
  398. self.fire("loaded", true);
  399. });
  400.  
  401. /**
  402. Fired whenever this SceneJSModel's {{#crossLink "SceneJSModel/data:property"}}{{/crossLink}} property changes.
  403. @event data
  404. @param value The property's new value
  405. */
  406. this.fire("data", this._data);
  407. }
  408.  
  409. get data() {
  410. return this._data;
  411. }
  412.  
  413. //---------------------------------------------------------------------------------------------------------------
  414. // A simple recursive descent parser that loads SceneJS JSON into a xeogl.Model.
  415. // This is just the bare essentials to prove the concept - just transforms, diffuse material and geometry.
  416. //---------------------------------------------------------------------------------------------------------------
  417.  
  418. _parse(node,
  419. group,
  420. material,
  421. diffuseMap,
  422. specularMap,
  423. emissiveMap,
  424. normalMap,
  425. alphaMap,
  426. diffuseFresnel,
  427. specularFresnel,
  428. emissiveFresnel,
  429. normalFresnel,
  430. alphaFresnel,
  431. transparent,
  432. backfaces,
  433. layer) {
  434.  
  435. switch (node.type) {
  436.  
  437. case "material":
  438.  
  439. var scenejsBaseColor = node.baseColor;
  440. var scenejsSpecularColor = node.specularColor;
  441. var scenejsSpecular = node.specular;
  442. var scenejsEmit = node.emit;
  443. var diffuse = scenejsBaseColor ? [scenejsBaseColor.r, scenejsBaseColor.g, scenejsBaseColor.b] : null;
  444. var specular = (scenejsSpecular && scenejsSpecularColor) ? [scenejsSpecular * scenejsSpecularColor.r, scenejsSpecular * scenejsSpecularColor.g, scenejsSpecular * scenejsSpecularColor.b] : null;
  445. var emissive = (scenejsEmit && diffuse) ? [scenejsEmit * diffuse[0], scenejsEmit * diffuse[1], scenejsEmit * diffuse[2]] : null;
  446.  
  447. switch (this._materialWorkflow) {
  448. case "MetallicMaterial":
  449. material = {
  450. id: this._createID(node),
  451. type: "xeogl.MetallicMaterial",
  452. baseColor: diffuse,
  453. metallic: 1.0,
  454. roughness: 0.4,
  455. emissive: emissive,
  456. alpha: node.alpha,
  457. alphaMode: "blend"
  458. };
  459. break;
  460.  
  461. case "SpecularMaterial":
  462. material = {
  463. id: this._createID(node),
  464. type: "xeogl.SpecularMaterial",
  465. diffuse: diffuse,
  466. specular: specular,
  467. glossiness: 0.5,
  468. emissive: emissive,
  469. alpha: node.alpha,
  470. alphaMode: "blend"
  471. };
  472. break;
  473.  
  474. default:
  475. material = {
  476. id: this._createID(node),
  477. type: "xeogl.PhongMaterial",
  478. ambient: [.2, .2, .2],
  479. diffuse: diffuse,
  480. specular: specular,
  481. // shininess: node.shine,
  482. emissive: emissive,
  483. alpha: node.alpha,
  484. alphaMode: "blend"
  485. };
  486. }
  487.  
  488. break;
  489.  
  490. case "translate":
  491.  
  492. group = group.addChild(new xeogl.Group(this.scene, {
  493. id: this._createID(node),
  494. position: [node.x, node.y, node.z]
  495. }));
  496.  
  497. this._addComponent(group);
  498.  
  499. break;
  500.  
  501. case "scale":
  502.  
  503. group = group.addChild(new xeogl.Group(this.scene, {
  504. id: this._createID(node),
  505. scale: [node.x, node.y, node.z]
  506. }));
  507.  
  508. this._addComponent(group);
  509.  
  510. break;
  511.  
  512. case "rotate":
  513.  
  514. var newGroup = new xeogl.Group(this.scene, {
  515. id: this._createID(node)
  516. });
  517.  
  518. newGroup.rotate([node.x, node.y, node.z], node.angle);
  519.  
  520. group = group.addChild(newGroup);
  521.  
  522. this._addComponent(group);
  523.  
  524. // var localMatrix = xeogl.math.rotationMat4c(node.angle * xeogl.math.DEGTORAD, node.x, node.y, node.z);
  525. // if (matrix) {
  526. // matrix = xeogl.math.mulMat4(matrix, localMatrix, xeogl.math.mat4());
  527. // } else {
  528. // matrix = localMatrix;
  529. // }
  530.  
  531. break;
  532.  
  533. case "texture":
  534.  
  535. var texture = new xeogl.Texture(this.scene, {
  536. id: this._createID(node),
  537. src: node.src,
  538. wrapS: node.wrapS,
  539. wrapT: node.wrapT,
  540. scale: node.scale ? [node.scale.x || 1, node.scale.y || 1] : undefined,
  541. translate: node.translate ? [node.translate.x || 0, node.translate.y || 1] : undefined,
  542. rotate: node.rotate,
  543. minFilter: node.minFilter,
  544. maxFilter: node.maxFilter,
  545. encoding: "sRGB"
  546. });
  547.  
  548. this._addComponent(texture);
  549.  
  550. switch (node.applyTo) {
  551.  
  552. case "baseColor":
  553. case "color":
  554. diffuseMap = texture;
  555. break;
  556.  
  557. case "specular":
  558. specularMap = texture;
  559. break;
  560.  
  561. case "emit":
  562. emissiveMap = texture;
  563. break;
  564.  
  565. case "alpha":
  566. alphaMap = texture;
  567. break;
  568.  
  569. case "normals":
  570. normalMap = texture;
  571. break;
  572.  
  573. case "shine":
  574. this.warn("Unsupported SceneJS feature - texture applyTo:'shine'");
  575. break;
  576. }
  577.  
  578. break;
  579.  
  580. case "fresnel":
  581.  
  582. var fresnel = new xeogl.Fresnel(this.scene, {
  583. id: this._createID(node)
  584. // TODO
  585. });
  586.  
  587. this._addComponent(fresnel);
  588.  
  589. switch (node.applyTo) {
  590.  
  591. case "baseColor":
  592. case "color":
  593. diffuseFresnel = fresnel;
  594. break;
  595.  
  596. case "specular":
  597. specularFresnel = fresnel;
  598. break;
  599.  
  600. case "emit":
  601. emissiveFresnel = fresnel;
  602. break;
  603.  
  604. case "alpha":
  605. alphaFresnel = fresnel;
  606. break;
  607. }
  608.  
  609. break;
  610.  
  611. case "flags":
  612.  
  613. transparent = node.transparent;
  614. backfaces = node.backfaces;
  615.  
  616. // TODO: pickable and clippable
  617.  
  618. break;
  619.  
  620. case "layer":
  621.  
  622. layer = node.priority;
  623.  
  624. break;
  625.  
  626.  
  627. case "geometry":
  628.  
  629. var geometry = new xeogl.Geometry(this.scene, {
  630. primitive: node.primitive,
  631. positions: node.positions,
  632. normals: node.normals,
  633. uv: node.uv,
  634. indices: node.indices
  635. });
  636.  
  637. this._addComponent(geometry);
  638.  
  639. if (material) {
  640.  
  641. // Set properties on material component
  642.  
  643. material.diffuseMap = diffuseMap;
  644. material.specularMap = specularMap;
  645. material.emissiveMap = emissiveMap;
  646. material.alphaMap = alphaMap;
  647. material.normalMap = normalMap;
  648.  
  649. material.diffuseFresnel = diffuseFresnel;
  650. material.specularFresnel = specularFresnel;
  651. material.emissiveFresnel = emissiveFresnel;
  652. material.alphaFresnel = alphaFresnel;
  653.  
  654. material.alphaMode = transparent ? "blend" : "opaque";
  655.  
  656. material.backfaces = !!backfaces;
  657. }
  658.  
  659. var material2 = this.scene.components[material.id];
  660. if (!material2) {
  661. material2 = new xeogl[material.type.substring(6)](this.scene, material);
  662. }
  663.  
  664. var mesh = new xeogl.Mesh(this.scene, {
  665. id: this._createID(node),
  666. geometry: geometry,
  667. material: material2,
  668. layer: layer
  669. });
  670.  
  671. if (group) {
  672. group.addChild(mesh);
  673. } else {
  674. this.addChild(mesh);
  675. }
  676.  
  677. this._addComponent(mesh);
  678.  
  679. break;
  680. }
  681.  
  682. var nodes = node.nodes;
  683.  
  684. if (nodes) {
  685. for (var i = 0, len = nodes.length; i < len; i++) {
  686. this._parse(
  687. nodes[i],
  688. group,
  689. material,
  690. diffuseMap,
  691. specularMap,
  692. emissiveMap,
  693. normalMap,
  694. alphaMap,
  695. diffuseFresnel,
  696. specularFresnel,
  697. emissiveFresnel,
  698. normalFresnel,
  699. alphaFresnel,
  700. transparent,
  701. backfaces,
  702. layer);
  703. }
  704. }
  705. }
  706.  
  707. _createID(node, type) {
  708. return (node.id !== null && node.id !== undefined) ? ("" + this.id + "." + (type ? type + "." : "") + node.id) : null;
  709. }
  710. };
  711.  
  712. function load(url, ok, error) {
  713. var xhr = new XMLHttpRequest();
  714. xhr.overrideMimeType("application/json");
  715. xhr.addEventListener('load', function (event) {
  716. if (event.target.responseText) {
  717. try {
  718. ok(JSON.parse(event.target.responseText));
  719. } catch (e) {
  720. error('Invalid file [' + url + ']: ' + e);
  721. }
  722. } else {
  723. error('Invalid file [' + url + ']');
  724. }
  725. }, false);
  726. xhr.addEventListener('error', function () {
  727. error('Couldn\'t load URL [' + url + ']');
  728. }, false);
  729. xhr.open('GET', url, true);
  730. xhr.send(null);
  731. }
  732.  
  733. }