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

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

  1. /**
  2. An **OBJModel** is a {{#crossLink "Model"}}{{/crossLink}} that loads itself from OBJ and MTL files.
  3.  
  4. <a href="../../examples/#importing_obj_conferenceRoom"><img src="../../../assets/images/screenshots/OBJModel.png"></img></a>
  5.  
  6. ## Overview
  7.  
  8. * Begins loading as soon as you set its {{#crossLink "OBJModel/src:property"}}{{/crossLink}} property to the location of an OBJ file.
  9. * Once loaded, contains an {{#crossLink "Mesh"}}{{/crossLink}} for each object. The {{#crossLink "Mesh"}}Meshes{{/crossLink}} can then be independently shown, hidden, colored, transformed etc.
  10. * Set {{#crossLink "OBJModel/src:property"}}{{/crossLink}} to a new file path at any time, to clear the OBJModel and load components from the new file.
  11.  
  12. OBJModel inherits these capabilities from its {{#crossLink "Group"}}{{/crossLink}} base class:
  13.  
  14. * Allows you to access and manipulate the {{#crossLink "Meshes"}}{{/crossLink}} within it.
  15. * Can be transformed as a unit within World-space.
  16. * Can be a child within a parent {{#crossLink "Group"}}{{/crossLink}}.
  17. * Provides its World-space axis-aligned and object-aligned boundaries.
  18.  
  19. ## Examples
  20.  
  21. * [Basic example](../../examples/#importing_obj_people)
  22. * [Models within an object hierarchy](../../examples/#objects_hierarchy_models)
  23.  
  24. ## Usage
  25.  
  26. Let's load the conference room model (shown in the screenshot above):
  27.  
  28. ````javascript
  29. var confRoom = new xeogl.OBJModel({
  30. id: "confRoom",
  31. src: "models/obj/conference/conference.obj"
  32. });
  33. ````
  34.  
  35. Bind a callback to fire when the model has loaded:
  36.  
  37. ````javascript
  38. confRoom.on("loaded", function() {
  39. // OBJModel has loaded!
  40. });
  41. ````
  42.  
  43. That fires immediately if the OBJModel already happens to be loaded. You can also bind a callback to fire if loading fails:
  44.  
  45. ````javascript
  46. confRoom.on("error", function(msg) {
  47. // Error occurred
  48. });
  49. ````
  50.  
  51. To switch to a different OBJ file, simply update {{#crossLink "OBJModel/src:property"}}{{/crossLink}}:
  52.  
  53. ````javascript
  54. confRoom.src = "models/obj/female02/female02.obj";
  55. ````
  56.  
  57. ### Fitting to view
  58.  
  59. ````javascript
  60. var cameraFlight = new xeogl.CameraFlightAnimation();
  61. cameraFlight.flyTo(confRoom);
  62. ````
  63.  
  64. ### Accessing components
  65.  
  66. Let's make everything transparent, except for the conference table and chairs:
  67.  
  68. ````javascript
  69. for (var id in confRoom.meshes) {
  70. var mesh = confRoom.meshes[id];
  71. switch (id) {
  72. case "confRoom#mesh31":
  73. case "confRoom#mesh29":
  74. case "confRoom#mesh30":
  75. break;
  76. default: // Not a chair mesh
  77. mesh.material.alpha = 0.5;
  78. mesh.material.blendMode = "blend"
  79. }
  80. }
  81. ````
  82.  
  83. Note the format of the {{#crossLink "Mesh"}}{{/crossLink}} IDs - an OBJModel prefixes its own ID to the IDs of its components:
  84.  
  85. ````<OBJModel ID>#<OBJ object/group ID>````
  86.  
  87. **Transforms**
  88.  
  89. An OBJModel lets us transform its Meshes as a group:
  90.  
  91. ```` Javascript
  92. var model = new xeogl.OBJModel({
  93. src: "models/obj/conference/conference.obj"
  94. position: [-35, 0, 0],
  95. rotation: [0, 45, 0],
  96. scale: [0.5, 0.5, 0.5]
  97. });
  98.  
  99. model.position = [-20, 0, 0];
  100. ````
  101.  
  102. Let's move the white table top upwards:
  103.  
  104. ````javascript
  105. var tableTop = confRoom.meshes["confRoom#mesh29"];
  106. tableTop.position = [0, 150, 0];
  107. ````
  108.  
  109. ## Examples
  110.  
  111. * [Conference room model](../../examples/#importing_obj_conferenceRoom)
  112. * [Two character models](../../examples/#importing_obj_people)
  113.  
  114. @class OBJModel
  115. @module xeogl
  116. @submodule models
  117. @constructor
  118. @param [scene] {Scene} Parent {{#crossLink "Scene"}}Scene{{/crossLink}} - creates this OBJModel in the default
  119. {{#crossLink "Scene"}}Scene{{/crossLink}} when omitted.
  120. @param [cfg] {*} Configs
  121. @param [cfg.id] {String} Optional ID, unique among all components in the parent {{#crossLink "Scene"}}Scene{{/crossLink}},
  122. generated automatically when omitted.
  123. @param [cfg.entityType] {String} Optional entity classification when using within a semantic data model. See the {{#crossLink "Object"}}{{/crossLink}} documentation for usage.
  124. @param [cfg.meta] {String:Object} Optional map of user-defined metadata to attach to this OBJModel.
  125. @param [cfg.src] {String} Path to an OBJ file. You can set this to a new file path at any time, which will cause the
  126. OBJModel to load components from the new file (after first destroying any components loaded from a previous file path).
  127. @param [cfg.quantizeGeometry=true] {Boolean} When true, quantizes geometry to reduce memory and GPU bus usage.
  128. @param [cfg.combineGeometry=true] {Boolean} When true, combines geometry vertex buffers to improve rendering performance.
  129. @param [cfg.ghosted=false] {Boolean} When true, sets all the OBJModel's Meshes initially ghosted.
  130. @param [cfg.highlighted=false] {Boolean} When true, sets all the OBJModel's Meshes initially highlighted.
  131. @param [cfg.outline=false] {Boolean} When true, sets all the OBJModel's Meshes initially outlined.
  132. @param [cfg.edgeThreshold=2] {Number} When ghosting, this is the threshold angle between normals of adjacent triangles, below which their shared wireframe edge is not drawn.
  133. @param [cfg.transform] {Number|String|Transform} A Local-to-World-space (modelling) {{#crossLink "Transform"}}{{/crossLink}} to attach to this OBJModel.
  134. Must be within the same {{#crossLink "Scene"}}{{/crossLink}} as this STLModel. Internally, the given
  135. {{#crossLink "Transform"}}{{/crossLink}} will be inserted above each top-most {{#crossLink "Transform"}}Transform{{/crossLink}}
  136. that the STLModel attaches to its {{#crossLink "Mesh"}}Meshes{{/crossLink}}.
  137. @param [cfg.splitMeshes=true] {Boolean} When true, creates a separate {{#crossLink "Mesh"}}{{/crossLink}} for each group of faces that share the same vertex colors. Only works with binary STL.|
  138. @param [cfg.position=[0,0,0]] {Float32Array} The STLModel's local 3D position.
  139. @param [cfg.scale=[1,1,1]] {Float32Array} The STLModel's local scale.
  140. @param [cfg.rotation=[0,0,0]] {Float32Array} The STLModel's local rotation, as Euler angles given in degrees.
  141. @param [cfg.matrix=[1,0,0,0,0,1,0,0,0,0,1,0,0,0,0,1] {Float32Array} The STLModel's local transform matrix. Overrides the position, scale and rotation parameters.
  142. @extends Model
  143. */
  144. {
  145.  
  146. xeogl.OBJModel = class xeoglOBJModel extends xeogl.Model {
  147.  
  148.  
  149. init(cfg) {
  150. super.init(cfg);
  151. this._src = null;
  152. this.src = cfg.src;
  153. }
  154.  
  155.  
  156. /**
  157. Path to a Wavefront OBJ file.
  158.  
  159. You can set this to a new file path at any time, which will cause the OBJModel to load components from
  160. the new file (after first destroying any components loaded from a previous file path).
  161.  
  162. Also loads materials from any MTL files referenced in the OBJ.
  163.  
  164. Fires a {{#crossLink "OBJModel/src:event"}}{{/crossLink}} event on change.
  165.  
  166. @property src
  167. @type String
  168. */
  169. set src(value) {
  170.  
  171. if (!value) {
  172. return;
  173. }
  174.  
  175. if (!xeogl._isString(value)) {
  176. this.error("Value for 'src' should be a string");
  177. return;
  178. }
  179.  
  180. if (value === this._src) { // Already loaded this OBJModel
  181.  
  182. /**
  183. Fired whenever this OBJModel has finished loading components from the OBJ file
  184. specified by {{#crossLink "OBJModel/src:property"}}{{/crossLink}}.
  185. @event loaded
  186. */
  187. this.fire("loaded", true, true);
  188.  
  189. return;
  190. }
  191.  
  192. this.destroyAll();
  193.  
  194. this._src = value;
  195.  
  196. xeogl.OBJModel.load(this, this._src);
  197.  
  198. /**
  199. Fired whenever this OBJModel's {{#crossLink "OBJModel/src:property"}}{{/crossLink}} property changes.
  200. @event src
  201. @param value The property's new value
  202. */
  203. this.fire("src", this._src);
  204. }
  205.  
  206. get src() {
  207. return this._src;
  208. }
  209.  
  210.  
  211. /**
  212. * Loads OBJ and MTL from file(s) into a {{#crossLink "Model"}}{{/crossLink}}.
  213. *
  214. * @method load
  215. * @static
  216. * @param {Model} model Model to load into.
  217. * @param {String} src Path to OBJ file.
  218. * @param {Function} [ok] Completion callback.
  219. */
  220. static load(model, src, ok) {
  221.  
  222. var spinner = model.scene.canvas.spinner;
  223. spinner.processes++;
  224.  
  225. loadOBJ(model, src, function (state) {
  226. loadMTLs(model, state, function () {
  227.  
  228. createMeshes(model, state);
  229.  
  230. spinner.processes--;
  231.  
  232. xeogl.scheduleTask(function () {
  233. model.fire("loaded", true);
  234. });
  235.  
  236. if (ok) {
  237. ok();
  238. }
  239. });
  240. });
  241. }
  242.  
  243. /**
  244. * Parses OBJ and MTL text strings into a {{#crossLink "Model"}}{{/crossLink}}.
  245. *
  246. * @method parse
  247. * @static
  248. * @param {Model} model Model to load into.
  249. * @param {String} objText OBJ text string.
  250. * @param {String} [mtlText] MTL text string.
  251. * @param {String} [basePath] Base path for external resources.
  252. */
  253. static parse(model, objText, mtlText, basePath) {
  254. if (!objText) {
  255. this.warn("load() param expected: objText");
  256. return;
  257. }
  258. var state = parseOBJ(model, objText, null);
  259. if (mtlText) {
  260. parseMTL(model, mtlText, basePath);
  261. }
  262. createMeshes(model, state);
  263. model.src = null;
  264. model.fire("loaded", true, true);
  265. }
  266. };
  267.  
  268. //--------------------------------------------------------------------------------------------
  269. // Loads OBJ
  270. //
  271. // Parses OBJ into an intermediate state object. The object will contain geometry data
  272. // and material IDs from which meshes can be created later. The object will also
  273. // contain a list of filenames of the MTL files referenced by the OBJ, is any.
  274. //
  275. // Originally based on the THREE.js OBJ and MTL loaders:
  276. //
  277. // https://github.com/mrdoob/three.js/blob/dev/examples/js/loaders/OBJLoader.js
  278. // https://github.com/mrdoob/three.js/blob/dev/examples/js/loaders/MTLLoader.js
  279. //--------------------------------------------------------------------------------------------
  280.  
  281. var loadOBJ = function (model, url, ok) {
  282.  
  283. loadFile(url, function (text) {
  284. var state = parseOBJ(model, text, url);
  285. ok(state);
  286. },
  287. function (error) {
  288. model.error(error);
  289. });
  290. };
  291.  
  292. var parseOBJ = (function () {
  293.  
  294. const regexp = {
  295. // v float float float
  296. vertex_pattern: /^v\s+([\d|\.|\+|\-|e|E]+)\s+([\d|\.|\+|\-|e|E]+)\s+([\d|\.|\+|\-|e|E]+)/,
  297. // vn float float float
  298. normal_pattern: /^vn\s+([\d|\.|\+|\-|e|E]+)\s+([\d|\.|\+|\-|e|E]+)\s+([\d|\.|\+|\-|e|E]+)/,
  299. // vt float float
  300. uv_pattern: /^vt\s+([\d|\.|\+|\-|e|E]+)\s+([\d|\.|\+|\-|e|E]+)/,
  301. // f vertex vertex vertex
  302. face_vertex: /^f\s+(-?\d+)\s+(-?\d+)\s+(-?\d+)(?:\s+(-?\d+))?/,
  303. // f vertex/uv vertex/uv vertex/uv
  304. face_vertex_uv: /^f\s+(-?\d+)\/(-?\d+)\s+(-?\d+)\/(-?\d+)\s+(-?\d+)\/(-?\d+)(?:\s+(-?\d+)\/(-?\d+))?/,
  305. // f vertex/uv/normal vertex/uv/normal vertex/uv/normal
  306. face_vertex_uv_normal: /^f\s+(-?\d+)\/(-?\d+)\/(-?\d+)\s+(-?\d+)\/(-?\d+)\/(-?\d+)\s+(-?\d+)\/(-?\d+)\/(-?\d+)(?:\s+(-?\d+)\/(-?\d+)\/(-?\d+))?/,
  307. // f vertex//normal vertex//normal vertex//normal
  308. face_vertex_normal: /^f\s+(-?\d+)\/\/(-?\d+)\s+(-?\d+)\/\/(-?\d+)\s+(-?\d+)\/\/(-?\d+)(?:\s+(-?\d+)\/\/(-?\d+))?/,
  309. // o object_name | g group_name
  310. object_pattern: /^[og]\s*(.+)?/,
  311. // s boolean
  312. smoothing_pattern: /^s\s+(\d+|on|off)/,
  313. // mtllib file_reference
  314. material_library_pattern: /^mtllib /,
  315. // usemtl material_name
  316. material_use_pattern: /^usemtl /
  317. };
  318.  
  319. return function (model, text, url) {
  320.  
  321. url = url || ""
  322.  
  323. var state = {
  324. src: url,
  325. basePath: getBasePath(url),
  326. objects: [],
  327. object: {},
  328. positions: [],
  329. normals: [],
  330. uv: [],
  331. materialLibraries: {}
  332. };
  333.  
  334. startObject(state, "", false);
  335.  
  336. // Parts of this parser logic are derived from the THREE.js OBJ loader:
  337. // https://github.com/mrdoob/three.js/blob/dev/examples/js/loaders/OBJLoader.js
  338.  
  339. if (text.indexOf('\r\n') !== -1) {
  340. // This is faster than String.split with regex that splits on both
  341. text = text.replace('\r\n', '\n');
  342. }
  343.  
  344. var lines = text.split('\n');
  345. var line = '', lineFirstChar = '', lineSecondChar = '';
  346. var lineLength = 0;
  347. var result = [];
  348.  
  349. // Faster to just trim left side of the line. Use if available.
  350. var trimLeft = ( typeof ''.trimLeft === 'function' );
  351.  
  352. for (var i = 0, l = lines.length; i < l; i++) {
  353.  
  354. line = lines[i];
  355.  
  356. line = trimLeft ? line.trimLeft() : line.trim();
  357.  
  358. lineLength = line.length;
  359.  
  360. if (lineLength === 0) {
  361. continue;
  362. }
  363.  
  364. lineFirstChar = line.charAt(0);
  365.  
  366. if (lineFirstChar === '#') {
  367. continue;
  368. }
  369.  
  370. if (lineFirstChar === 'v') {
  371.  
  372. lineSecondChar = line.charAt(1);
  373.  
  374. if (lineSecondChar === ' ' && ( result = regexp.vertex_pattern.exec(line) ) !== null) {
  375.  
  376. // 0 1 2 3
  377. // ['v 1.0 2.0 3.0', '1.0', '2.0', '3.0']
  378.  
  379. state.positions.push(
  380. parseFloat(result[1]),
  381. parseFloat(result[2]),
  382. parseFloat(result[3])
  383. );
  384.  
  385. } else if (lineSecondChar === 'n' && ( result = regexp.normal_pattern.exec(line) ) !== null) {
  386.  
  387. // 0 1 2 3
  388. // ['vn 1.0 2.0 3.0', '1.0', '2.0', '3.0']
  389.  
  390. state.normals.push(
  391. parseFloat(result[1]),
  392. parseFloat(result[2]),
  393. parseFloat(result[3])
  394. );
  395.  
  396. } else if (lineSecondChar === 't' && ( result = regexp.uv_pattern.exec(line) ) !== null) {
  397.  
  398. // 0 1 2
  399. // ['vt 0.1 0.2', '0.1', '0.2']
  400.  
  401. state.uv.push(
  402. parseFloat(result[1]),
  403. parseFloat(result[2])
  404. );
  405.  
  406. } else {
  407.  
  408. model.error('Unexpected vertex/normal/uv line: \'' + line + '\'');
  409. return;
  410. }
  411.  
  412. } else if (lineFirstChar === 'f') {
  413.  
  414. if (( result = regexp.face_vertex_uv_normal.exec(line) ) !== null) {
  415.  
  416. // f vertex/uv/normal vertex/uv/normal vertex/uv/normal
  417. // 0 1 2 3 4 5 6 7 8 9 10 11 12
  418. // ['f 1/1/1 2/2/2 3/3/3', '1', '1', '1', '2', '2', '2', '3', '3', '3', undefined, undefined, undefined]
  419.  
  420. addFace(state,
  421. result[1], result[4], result[7], result[10],
  422. result[2], result[5], result[8], result[11],
  423. result[3], result[6], result[9], result[12]
  424. );
  425.  
  426. } else if (( result = regexp.face_vertex_uv.exec(line) ) !== null) {
  427.  
  428. // f vertex/uv vertex/uv vertex/uv
  429. // 0 1 2 3 4 5 6 7 8
  430. // ['f 1/1 2/2 3/3', '1', '1', '2', '2', '3', '3', undefined, undefined]
  431.  
  432. addFace(state,
  433. result[1], result[3], result[5], result[7],
  434. result[2], result[4], result[6], result[8]
  435. );
  436.  
  437. } else if (( result = regexp.face_vertex_normal.exec(line) ) !== null) {
  438.  
  439. // f vertex//normal vertex//normal vertex//normal
  440. // 0 1 2 3 4 5 6 7 8
  441. // ['f 1//1 2//2 3//3', '1', '1', '2', '2', '3', '3', undefined, undefined]
  442.  
  443. addFace(state,
  444. result[1], result[3], result[5], result[7],
  445. undefined, undefined, undefined, undefined,
  446. result[2], result[4], result[6], result[8]
  447. );
  448.  
  449. } else if (( result = regexp.face_vertex.exec(line) ) !== null) {
  450.  
  451. // f vertex vertex vertex
  452. // 0 1 2 3 4
  453. // ['f 1 2 3', '1', '2', '3', undefined]
  454.  
  455. addFace(state, result[1], result[2], result[3], result[4]);
  456. } else {
  457. model.error('Unexpected face line: \'' + line + '\'');
  458. return;
  459. }
  460.  
  461. } else if (lineFirstChar === 'l') {
  462.  
  463. var lineParts = line.substring(1).trim().split(' ');
  464. var lineVertices = [], lineUVs = [];
  465.  
  466. if (line.indexOf('/') === -1) {
  467.  
  468. lineVertices = lineParts;
  469.  
  470. } else {
  471. for (var li = 0, llen = lineParts.length; li < llen; li++) {
  472. var parts = lineParts[li].split('/');
  473. if (parts[0] !== '') {
  474. lineVertices.push(parts[0]);
  475. }
  476. if (parts[1] !== '') {
  477. lineUVs.push(parts[1]);
  478. }
  479. }
  480. }
  481. addLineGeometry(state, lineVertices, lineUVs);
  482.  
  483. } else if (( result = regexp.object_pattern.exec(line) ) !== null) {
  484.  
  485. // o object_name
  486. // or
  487. // g group_name
  488.  
  489. var id = result[0].substr(1).trim();
  490. startObject(state, id, true);
  491.  
  492. } else if (regexp.material_use_pattern.test(line)) {
  493.  
  494. // material
  495.  
  496. var id = line.substring(7).trim();
  497. state.object.material.id = id;
  498.  
  499. } else if (regexp.material_library_pattern.test(line)) {
  500.  
  501. // mtl file
  502.  
  503. state.materialLibraries[line.substring(7).trim()] = true;
  504.  
  505. } else if (( result = regexp.smoothing_pattern.exec(line) ) !== null) {
  506.  
  507. // smooth shading
  508.  
  509. var value = result[1].trim().toLowerCase();
  510. state.object.material.smooth = ( value === '1' || value === 'on' );
  511.  
  512. } else {
  513.  
  514. // Handle null terminated files without exception
  515. if (line === '\0') {
  516. continue;
  517. }
  518.  
  519. model.error('Unexpected line: \'' + line + '\'');
  520. return;
  521. }
  522. }
  523.  
  524. return state;
  525. };
  526.  
  527. function getBasePath(src) {
  528. var n = src.lastIndexOf('/');
  529. return (n === -1) ? src : src.substring(0, n + 1);
  530. }
  531.  
  532. function startObject(state, id, fromDeclaration) {
  533. if (state.object && state.object.fromDeclaration === false) {
  534. state.object.id = id;
  535. state.object.fromDeclaration = ( fromDeclaration !== false );
  536. return;
  537. }
  538. state.object = {
  539. id: id || '',
  540. geometry: {
  541. positions: [],
  542. normals: [],
  543. uv: []
  544. },
  545. material: {
  546. id: '',
  547. smooth: true
  548. },
  549. fromDeclaration: ( fromDeclaration !== false )
  550. };
  551. state.objects.push(state.object);
  552. }
  553.  
  554. function parseVertexIndex(value, len) {
  555. var index = parseInt(value, 10);
  556. return ( index >= 0 ? index - 1 : index + len / 3 ) * 3;
  557. }
  558.  
  559. function parseNormalIndex(value, len) {
  560. var index = parseInt(value, 10);
  561. return ( index >= 0 ? index - 1 : index + len / 3 ) * 3;
  562. }
  563.  
  564. function parseUVIndex(value, len) {
  565. var index = parseInt(value, 10);
  566. return ( index >= 0 ? index - 1 : index + len / 2 ) * 2;
  567. }
  568.  
  569. function addVertex(state, a, b, c) {
  570. var src = state.positions;
  571. var dst = state.object.geometry.positions;
  572. dst.push(src[a + 0]);
  573. dst.push(src[a + 1]);
  574. dst.push(src[a + 2]);
  575. dst.push(src[b + 0]);
  576. dst.push(src[b + 1]);
  577. dst.push(src[b + 2]);
  578. dst.push(src[c + 0]);
  579. dst.push(src[c + 1]);
  580. dst.push(src[c + 2]);
  581. }
  582.  
  583. function addVertexLine(state, a) {
  584. var src = state.positions;
  585. var dst = state.object.geometry.positions;
  586. dst.push(src[a + 0]);
  587. dst.push(src[a + 1]);
  588. dst.push(src[a + 2]);
  589. }
  590.  
  591. function addNormal(state, a, b, c) {
  592. var src = state.normals;
  593. var dst = state.object.geometry.normals;
  594. dst.push(src[a + 0]);
  595. dst.push(src[a + 1]);
  596. dst.push(src[a + 2]);
  597. dst.push(src[b + 0]);
  598. dst.push(src[b + 1]);
  599. dst.push(src[b + 2]);
  600. dst.push(src[c + 0]);
  601. dst.push(src[c + 1]);
  602. dst.push(src[c + 2]);
  603. }
  604.  
  605. function addUV(state, a, b, c) {
  606. var src = state.uv;
  607. var dst = state.object.geometry.uv;
  608. dst.push(src[a + 0]);
  609. dst.push(src[a + 1]);
  610. dst.push(src[b + 0]);
  611. dst.push(src[b + 1]);
  612. dst.push(src[c + 0]);
  613. dst.push(src[c + 1]);
  614. }
  615.  
  616. function addUVLine(state, a) {
  617. var src = state.uv;
  618. var dst = state.object.geometry.uv;
  619. dst.push(src[a + 0]);
  620. dst.push(src[a + 1]);
  621. }
  622.  
  623. function addFace(state, a, b, c, d, ua, ub, uc, ud, na, nb, nc, nd) {
  624. var vLen = state.positions.length;
  625. var ia = parseVertexIndex(a, vLen);
  626. var ib = parseVertexIndex(b, vLen);
  627. var ic = parseVertexIndex(c, vLen);
  628. var id;
  629. if (d === undefined) {
  630. addVertex(state, ia, ib, ic);
  631.  
  632. } else {
  633. id = parseVertexIndex(d, vLen);
  634. addVertex(state, ia, ib, id);
  635. addVertex(state, ib, ic, id);
  636. }
  637.  
  638. if (ua !== undefined) {
  639.  
  640. var uvLen = state.uv.length;
  641.  
  642. ia = parseUVIndex(ua, uvLen);
  643. ib = parseUVIndex(ub, uvLen);
  644. ic = parseUVIndex(uc, uvLen);
  645.  
  646. if (d === undefined) {
  647. addUV(state, ia, ib, ic);
  648.  
  649. } else {
  650. id = parseUVIndex(ud, uvLen);
  651. addUV(state, ia, ib, id);
  652. addUV(state, ib, ic, id);
  653. }
  654. }
  655.  
  656. if (na !== undefined) {
  657.  
  658. // Normals are many times the same. If so, skip function call and parseInt.
  659.  
  660. var nLen = state.normals.length;
  661.  
  662. ia = parseNormalIndex(na, nLen);
  663. ib = na === nb ? ia : parseNormalIndex(nb, nLen);
  664. ic = na === nc ? ia : parseNormalIndex(nc, nLen);
  665.  
  666. if (d === undefined) {
  667. addNormal(state, ia, ib, ic);
  668.  
  669. } else {
  670.  
  671. id = parseNormalIndex(nd, nLen);
  672. addNormal(state, ia, ib, id);
  673. addNormal(state, ib, ic, id);
  674. }
  675. }
  676. }
  677.  
  678. function addLineGeometry(state, positions, uv) {
  679.  
  680. state.object.geometry.type = 'Line';
  681.  
  682. var vLen = state.positions.length;
  683. var uvLen = state.uv.length;
  684.  
  685. for (var vi = 0, l = positions.length; vi < l; vi++) {
  686. addVertexLine(state, parseVertexIndex(positions[vi], vLen));
  687. }
  688.  
  689. for (var uvi = 0, uvl = uv.length; uvi < uvl; uvi++) {
  690. addUVLine(state, parseUVIndex(uv[uvi], uvLen));
  691. }
  692. }
  693. })();
  694.  
  695. //--------------------------------------------------------------------------------------------
  696. // Loads MTL files listed in parsed state
  697. //--------------------------------------------------------------------------------------------
  698.  
  699. function loadMTLs(model, state, ok) {
  700. var basePath = state.basePath;
  701. var srcList = Object.keys(state.materialLibraries);
  702. var numToLoad = srcList.length;
  703. for (var i = 0, len = numToLoad; i < len; i++) {
  704. loadMTL(model, basePath, basePath + srcList[i], function () {
  705. if (--numToLoad === 0) {
  706. ok();
  707. }
  708. });
  709. }
  710. }
  711.  
  712. //--------------------------------------------------------------------------------------------
  713. // Loads an MTL file
  714. //--------------------------------------------------------------------------------------------
  715.  
  716. var loadMTL = function (model, basePath, src, ok) {
  717. loadFile(src, function (text) {
  718. parseMTL(model, text, basePath);
  719. ok();
  720. },
  721. function (error) {
  722. model.error(error);
  723. ok();
  724. });
  725. };
  726.  
  727. var parseMTL = (function () {
  728.  
  729. var delimiter_pattern = /\s+/;
  730.  
  731. return function (model, mtlText, basePath) {
  732.  
  733. var lines = mtlText.split('\n');
  734. var materialCfg = {
  735. id: "Default"
  736. };
  737. var needCreate = false;
  738. var line;
  739. var pos;
  740. var key;
  741. var value;
  742. var alpha;
  743.  
  744. basePath = basePath || "";
  745.  
  746. for (var i = 0; i < lines.length; i++) {
  747.  
  748. line = lines[i].trim();
  749.  
  750. if (line.length === 0 || line.charAt(0) === '#') { // Blank line or comment ignore
  751. continue;
  752. }
  753.  
  754. pos = line.indexOf(' ');
  755.  
  756. key = ( pos >= 0 ) ? line.substring(0, pos) : line;
  757. key = key.toLowerCase();
  758.  
  759. value = ( pos >= 0 ) ? line.substring(pos + 1) : '';
  760. value = value.trim();
  761.  
  762. switch (key.toLowerCase()) {
  763.  
  764. case "newmtl": // New material
  765. //if (needCreate) {
  766. createMaterial(model, materialCfg);
  767. //}
  768. materialCfg = {
  769. id: value
  770. };
  771. needCreate = true;
  772. break;
  773.  
  774. case 'ka':
  775. materialCfg.ambient = parseRGB(value);
  776. break;
  777.  
  778. case 'kd':
  779. materialCfg.diffuse = parseRGB(value);
  780. break;
  781.  
  782. case 'ks':
  783. materialCfg.specular = parseRGB(value);
  784. break;
  785.  
  786. case 'map_kd':
  787. if (!materialCfg.diffuseMap) {
  788. materialCfg.diffuseMap = createTexture(model, basePath, value, "sRGB");
  789. }
  790. break;
  791.  
  792. case 'map_ks':
  793. if (!materialCfg.specularMap) {
  794. materialCfg.specularMap = createTexture(model, basePath, value, "linear");
  795. }
  796. break;
  797.  
  798. case 'map_bump':
  799. case 'bump':
  800. if (!materialCfg.normalMap) {
  801. materialCfg.normalMap = createTexture(model, basePath, value);
  802. }
  803. break;
  804.  
  805. case 'ns':
  806. materialCfg.shininess = parseFloat(value);
  807. break;
  808.  
  809. case 'd':
  810. alpha = parseFloat(value);
  811. if (alpha < 1) {
  812. materialCfg.alpha = alpha;
  813. materialCfg.alphaMode = "blend";
  814. }
  815. break;
  816.  
  817. case 'tr':
  818. alpha = parseFloat(value);
  819. if (alpha > 0) {
  820. materialCfg.alpha = 1 - alpha;
  821. materialCfg.alphaMode = "blend";
  822. }
  823. break;
  824.  
  825. default:
  826. // model.error("Unrecognized token: " + key);
  827. }
  828. }
  829.  
  830. if (needCreate) {
  831. createMaterial(model, materialCfg);
  832. }
  833. };
  834.  
  835. function createTexture(model, basePath, value, encoding) {
  836. var textureCfg = {};
  837. var items = value.split(/\s+/);
  838. var pos = items.indexOf('-bm');
  839. if (pos >= 0) {
  840. //matParams.bumpScale = parseFloat(items[pos + 1]);
  841. items.splice(pos, 2);
  842. }
  843. pos = items.indexOf('-s');
  844. if (pos >= 0) {
  845. textureCfg.scale = [parseFloat(items[pos + 1]), parseFloat(items[pos + 2])];
  846. items.splice(pos, 4); // we expect 3 parameters here!
  847. }
  848. pos = items.indexOf('-o');
  849. if (pos >= 0) {
  850. textureCfg.translate = [parseFloat(items[pos + 1]), parseFloat(items[pos + 2])];
  851. items.splice(pos, 4); // we expect 3 parameters here!
  852. }
  853. textureCfg.src = basePath + items.join(' ').trim();
  854. textureCfg.flipY = true;
  855. textureCfg.encoding = encoding || "linear";
  856. //textureCfg.wrapS = self.wrap;
  857. //textureCfg.wrapT = self.wrap;
  858. var texture = new xeogl.Texture(model, textureCfg);
  859. model._addComponent(texture);
  860. return texture.id;
  861. }
  862.  
  863. function createMaterial(model, materialCfg) {
  864. model._addComponent(new xeogl.PhongMaterial(model, materialCfg));
  865. }
  866.  
  867. function parseRGB(value) {
  868. var ss = value.split(delimiter_pattern, 3);
  869. return [parseFloat(ss[0]), parseFloat(ss[1]), parseFloat(ss[2])];
  870. }
  871.  
  872. })();
  873. //--------------------------------------------------------------------------------------------
  874. // Creates meshes from parsed state
  875. //--------------------------------------------------------------------------------------------
  876.  
  877. var createMeshes = (function () {
  878.  
  879. return function (model, state) {
  880.  
  881. for (var j = 0, k = state.objects.length; j < k; j++) {
  882.  
  883. var object = state.objects[j];
  884. var geometry = object.geometry;
  885. var isLine = ( geometry.type === 'Line' );
  886.  
  887. if (geometry.positions.length === 0) {
  888. // Skip o/g line declarations that did not follow with any faces
  889. continue;
  890. }
  891.  
  892. var geometryCfg = {
  893. primitive: "triangles"
  894. };
  895.  
  896. geometryCfg.positions = geometry.positions;
  897.  
  898. if (geometry.normals.length > 0) {
  899. geometryCfg.normals = geometry.normals;
  900. } else {
  901. geometryCfg.autoVertexNormals = true;
  902. }
  903.  
  904. if (geometry.uv.length > 0) {
  905. geometryCfg.uv = geometry.uv;
  906. }
  907.  
  908. var indices = new Array(geometryCfg.positions.length / 3); // Triangle soup
  909. for (var idx = 0; idx < indices.length; idx++) {
  910. indices[idx] = idx;
  911. }
  912. geometryCfg.indices = indices;
  913.  
  914. var xeoGeometry = new xeogl.Geometry(model, geometryCfg);
  915. model._addComponent(xeoGeometry);
  916.  
  917. var materialId = object.material.id;
  918. var material;
  919. if (materialId && materialId !== "") {
  920. material = model.scene.components[materialId];
  921. if (!material) {
  922. model.error("Material not found: " + materialId);
  923. }
  924. } else {
  925. material = new xeogl.PhongMaterial(model, {
  926. //emissive: [0.6, 0.6, 0.0],
  927. diffuse: [0.6, 0.6, 0.6],
  928. backfaces: true
  929. });
  930. model._addComponent(material);
  931. }
  932.  
  933. // material.emissive = [Math.random(), Math.random(), Math.random()];
  934.  
  935. var mesh = new xeogl.Mesh(model, {
  936. id: model.id + "#" + object.id,
  937. geometry: xeoGeometry,
  938. material: material,
  939. pickable: true
  940. });
  941.  
  942. model.addChild(mesh);
  943. model._addComponent(mesh);
  944. }
  945. };
  946. })();
  947.  
  948. function loadFile(url, ok, err) {
  949. var request = new XMLHttpRequest();
  950. request.open('GET', url, true);
  951. request.addEventListener('load', function (event) {
  952. var response = event.target.response;
  953. if (this.status === 200) {
  954. if (ok) {
  955. ok(response);
  956. }
  957. } else if (this.status === 0) {
  958. // Some browsers return HTTP Status 0 when using non-http protocol
  959. // e.g. 'file://' or 'data://'. Handle as success.
  960. console.warn('loadFile: HTTP Status 0 received.');
  961. if (ok) {
  962. ok(response);
  963. }
  964. } else {
  965. if (err) {
  966. err(event);
  967. }
  968. }
  969. }, false);
  970.  
  971. request.addEventListener('error', function (event) {
  972. if (err) {
  973. err(event);
  974. }
  975. }, false);
  976. request.send(null);
  977. }
  978. }