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

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

  1. /**
  2. An **STLModel** is a {{#crossLink "Model"}}{{/crossLink}} that's loaded from an <a href="https://en.wikipedia.org/wiki/STL_(file_format)">STL</a> file.
  3.  
  4. <a href="../../examples/#importing_stl_shapes"><img src="../../../assets/images/screenshots/STLModel.png"></img></a>
  5.  
  6. ## Overview
  7.  
  8. * An <a href="https://en.wikipedia.org/wiki/STL_(file_format)">STL</a> (“StereoLithography”) file is a triangular representation of a 3-dimensional surface geometry. The surface is
  9. tessellated logically into a series of triangles. Each facet is described by a perpendicular
  10. direction and three points representing the vertices (corners) of the triangle.
  11. * An STLModel is a container of {{#crossLink "Component"}}Components{{/crossLink}} that loads itself from an STL file.
  12. * It begins loading as soon as you set its {{#crossLink "STLModel/src:property"}}{{/crossLink}}
  13. property to the location of a valid STL file.
  14. * You can set {{#crossLink "STLModel/src:property"}}{{/crossLink}} to a new file path at any time, which causes
  15. the STLModel to clear itself and load components from the new file.
  16. * For binary STL, has the option to create a separate {{#crossLink "Mesh"}}{{/crossLink}} for each group of faces
  17. that share the same vertex colors. This allows us to treat STL models as parts assemblies.
  18. * Can be configured to automatically smooth STL models by converting their face-oriented normals to vertex-oriented.
  19.  
  20. STLModel inherits these capabilities from its {{#crossLink "Group"}}{{/crossLink}} base class:
  21.  
  22. * Allows you to access and manipulate the {{#crossLink "Meshes"}}{{/crossLink}} within it.
  23. * Can be transformed as a unit within World-space.
  24. * Can be a child within a parent {{#crossLink "Group"}}{{/crossLink}}.
  25. * Provides its World-space axis-aligned and object-aligned boundaries.
  26.  
  27. ## Examples
  28.  
  29. * [Simple shapes with smoothing](../../examples/#importing_stl_shapes)
  30. * [F1 concept car with smoothing](../../examples/#importing_stl_F1Concept)
  31. * [Models within an object hierarchy](../../examples/#objects_hierarchy_models)
  32.  
  33. ## Usage
  34.  
  35. * [Loading STL](#loading-stl)
  36. * [Parsing STL](#parsing-stl)
  37. * [Options](#options)
  38. * [Smoothing Normals](#smoothing-normals)
  39. * [Finding loaded Meshes](#finding-loaded-meshes)
  40. * [Transforming an STLModel](#transforming-a-gltfmodel)
  41. * [Getting the World-space boundary of an STLModel](#getting-the-world-space-boundary-of-an-stlmodel)
  42. * [Clearing an STLModel](#clearing-a-gltfmodel)
  43. * [Destroying an STLModel](#destroying-a-gltfmodel)
  44.  
  45. ### Loading STL
  46.  
  47. Load an STL file by creating an STLModel:
  48.  
  49. ````javascript
  50. var model = new xeogl.STLModel({
  51. id: "myModel",
  52. src: "models/stl/F1Concept.stl",
  53.  
  54. // Some example loading options (see "Options" below)
  55. smoothNormals: true,
  56. smoothNormalsAngleThreshold: 45
  57. });
  58. ````
  59.  
  60. An STLModel prefixes its own ID to those of its components. The ID is optional, but in this example we're providing our own custom ID.
  61.  
  62. The STLModel begins loading the STL file immediately.
  63.  
  64. To bind a callback to be notified when the file has loaded (which fires immediately if already loaded):
  65.  
  66. ````javascript
  67. model.on("loaded", function() {
  68. // STLModel has loaded!
  69. });
  70. ````
  71.  
  72. You can also bind a callback to fire if loading fails:
  73.  
  74. ````javascript
  75. model.on("error", function(msg) {
  76. // Error occurred
  77. });
  78. ````
  79.  
  80. To switch to a different STL file, you can dynamically update {{#crossLink "STLModel/src:property"}}{{/crossLink}}:
  81.  
  82. ````javascript
  83. model.src = "models/stl/F1Concept.stl"
  84. ````
  85.  
  86. That will apply whatever options were specified to the constructor.
  87.  
  88. ### Parsing STL
  89.  
  90. If we have STL data in memory, then we can parse it directly into an existing STLModel instance using the
  91. static {{#crossLink "STLModel/parse:method"}}{{/crossLink}} method:
  92.  
  93. ````javascript
  94. xeogl.STLModel.parse(model, stlData, {
  95.  
  96. // Some example parsing options (see "Options" below)
  97. smoothNormals: true,
  98. smoothNormalsAngleThreshold: 45,
  99. combineGeometry: true,
  100. quantizeGeometry: true
  101. });
  102. ````
  103.  
  104. That's asynchronous because STL is self-contained and does not need to load any external assets.
  105.  
  106. ### Options
  107.  
  108. The following options may be specified when loading or parsing STL:
  109.  
  110. | Option | Type | Range | Default Value | Description |
  111. |:--------:|:----:|:-----:|:-------------:|:-----:|:-----------:|
  112. | quantizeGeometry | Boolean | | true | When true, quantizes geometry to reduce memory and GPU bus usage (see {{#crossLink "Geometry"}}{{/crossLink}}). |
  113. | combineGeometry | Boolean | | true | When true, internally combines geometry vertex buffers to improve rendering performance (see {{#crossLink "Geometry"}}{{/crossLink}}). |
  114. | smoothNormals | Boolean | | false | When true, automatically converts face-oriented normals to vertex normals for a smooth appearance. See [Smoothing Normals](#smoothing-normals). |
  115. | smoothNormalsAngleThreshold | Number (degrees) | [0..180] | 20 | See [Smoothing Normals](#smoothing-normals). |
  116. | backfaces | Boolean | | true | When true, allows visible backfaces, wherever specified in the STL. When false, ignores backfaces. |
  117. | ghosted | Boolean | | false | When true, ghosts all the model's Meshes (see {{#crossLink "Mesh"}}{{/crossLink}} and {{#crossLink "EmphasisMaterial"}}{{/crossLink}}). |
  118. | outlined | Boolean | | false | When true, outlines all the model's Meshes (see {{#crossLink "Mesh"}}{{/crossLink}} and {{#crossLink "OutlineMaterial"}}{{/crossLink}}). |
  119. | highlighted | Boolean | | false | When true, highlights all the model's Meshes (see {{#crossLink "Mesh"}}{{/crossLink}} and {{#crossLink "EmphasisMaterial"}}{{/crossLink}}). |
  120. | selected | Boolean | | false | When true, renders all the model's Meshes as selected (see {{#crossLink "Mesh"}}{{/crossLink}} and {{#crossLink "EmphasisMaterial"}}{{/crossLink}}). |
  121. | edges | Boolean | | false | When true, emphasizes the edges on all the model's Meshes (see {{#crossLink "Mesh"}}{{/crossLink}} and {{#crossLink "EdgeMaterial"}}{{/crossLink}}). |
  122. | edgeThreshold | Number | [0..180] | 2 | When ghosting, this is the threshold angle between normals of adjacent triangles, below which their shared wireframe edge is not drawn. |
  123. | splitMeshes | Boolean | | true | When true, creates a separate {{#crossLink "Mesh"}}{{/crossLink}} for each group of faces that share the same vertex colors. Only works with binary STL.| |
  124.  
  125. ### Smoothing Normals
  126.  
  127. As mentioned above, providing a ````smoothNormals```` flag to the constructor gives our STLModel a smooth appearance. Triangles in STL
  128. are disjoint, where each triangle has its own separate vertex positions, normals and (optionally) colors. This means that you can
  129. have gaps between triangles. Normals for each triangle are perpendicular to the triangle's surface, which gives the model a faceted appearance by default.
  130.  
  131. The ```smoothNormals``` flag causes the STLModel to recalculate its normals, so that each normal's direction is the average
  132. of the orientations of the triangles adjacent to its vertex. When smoothing, each vertex normal is set to the average of the
  133. orientations of all other triangles that have a vertex at the same position, excluding those triangles whose direction deviates from
  134. the direction of the vertice's triangle by a threshold given in ````smoothNormalsAngleThreshold````. This makes
  135. smoothing robust for hard edges, which you can see on the cylindrical objects in one of the examples:
  136.  
  137. <a href="../../examples/#importing_stl_shapes"><img src="../../../assets/images/screenshots/STLModelHardEdges.png"></img></a>
  138.  
  139. Note how the rim is smooth, yet the there is still a sharp edge adjacent to the flat portions.
  140.  
  141. ### Finding STLModels in Scenes
  142.  
  143. Our STLModel will now be registered by ID on its Scene, so we can now find it like this:
  144.  
  145. ````javascript
  146. var scene = xeogl.getDefaultScene();
  147. model = scene.models["myModel"];
  148. ````
  149.  
  150. That's assuming that we've created the STLModel in the default xeogl Scene, which we're doing in these examples.
  151.  
  152. We can also get all the STLModels in a Scene, using the Scene's {{#crossLink "Scene/types:property"}}{{/crossLink}} map:
  153.  
  154. ````javascript
  155. var scene = xeogl.getDefaultScene();
  156. var stlModels = scene.types["xeogl.STLModel"];
  157.  
  158. model = stlModels["myModel"];
  159. ````
  160.  
  161. ### Finding loaded Meshes
  162.  
  163. Once the STLModel has loaded, its {{#crossLink "Scene"}}{{/crossLink}} will contain various components that represent the elements of the STL file.
  164. We'll now access some of those components by ID, to query and update them programmatically.
  165.  
  166. Let's highlight an {{#crossLink "Object"}}{{/crossLink}} in our STLModel:
  167.  
  168. ````javascript
  169. scene.omponents["myModel#1"].highlighted = true;
  170. ````
  171.  
  172. An STLModel also has an ID map of the components within it. Let's highlight another {{#crossLink "Object"}}{{/crossLink}} via the STLModel:
  173.  
  174. ````javascript
  175. model.components["myModel#2"].highlighted = true;
  176. ````
  177.  
  178. An STLModel also has a map containing just the {{#crossLink "Objects"}}Objects{{/crossLink}}:
  179.  
  180. ````javascript
  181. model.objects["myModel#3"].highlighted = true;
  182. ````
  183.  
  184. TODO: ID format description
  185.  
  186. ### Transforming an STLModel
  187.  
  188. An STLModel lets us transform its Meshes as a group:
  189.  
  190. ```` Javascript
  191. var model = new xeogl.STLModel({
  192. src: "models/stl/F1Concept.stl",
  193. position: [-35, 0, 0],
  194. rotation: [0, 45, 0],
  195. scale: [0.5, 0.5, 0.5]
  196. });
  197.  
  198. model.position = [-20, 0, 0];
  199. ````
  200.  
  201. ### Getting the World-space boundary of an STLModel
  202.  
  203. Get the World-space axis-aligned boundary like this:
  204.  
  205. ```` Javascript
  206. model.on("boundary", function() {
  207. var aabb = model.aabb; // [xmin, ymin,zmin,xmax,ymax, zmax]
  208. //...
  209. });
  210. ````
  211.  
  212. We can also subscribe to changes to that boundary, which will happen whenever
  213.  
  214. * the STLModel's {{#crossLink "Transform"}}{{/crossLink}} is updated,
  215. * components are added or removed, or
  216. * the STLModel is reloaded from a different source,
  217. * the {{#crossLink "Geometry"}}Geometries{{/crossLink}} or {{#crossLink "Transform"}}Transforms{{/crossLink}} of its {{#crossLink "Mesh"}}Meshes{{/crossLink}} are updated.
  218.  
  219. ````javascript
  220. model.on("boundary", function() {
  221. var aabb = model.aabb; // [xmin, ymin,zmin,xmax,ymax, zmax]
  222. });
  223. ````
  224.  
  225. ### Clearing an STLModel
  226.  
  227. ```` Javascript
  228. model.clear();
  229. ````
  230.  
  231. ### Destroying an STLModel
  232.  
  233. ```` Javascript
  234. model.destroy();
  235. ````
  236.  
  237. @class STLModel
  238. @module xeogl
  239. @submodule models
  240. @constructor
  241. @param [scene] {Scene} Parent {{#crossLink "Scene"}}Scene{{/crossLink}} - creates this STLModel in the default
  242. {{#crossLink "Scene"}}Scene{{/crossLink}} when omitted.
  243. @param [cfg] {*} Configs
  244. @param [cfg.id] {String} Optional ID, unique among all components in the parent {{#crossLink "Scene"}}Scene{{/crossLink}},
  245. generated automatically when omitted.
  246. @param [cfg.entityType] {String} Optional entity classification when using within a semantic data model. See the {{#crossLink "Object"}}{{/crossLink}} documentation for usage.
  247. @param [cfg.meta] {String:Object} Optional map of user-defined metadata to attach to this STLModel.
  248. @param [cfg.src] {String} Path to an STL file. You can set this to a new file path at any time, which will cause the
  249. @param [cfg.quantizeGeometry=true] {Boolean} When true, quantizes geometry to reduce memory and GPU bus usage.
  250. @param [cfg.combineGeometry=true] {Boolean} When true, combines geometry vertex buffers to improve rendering performance.
  251. @param [cfg.smoothNormals=false] {Boolean} When true, automatically converts face-oriented normals to vertex normals for a smooth appearance - see <a href="#smoothing-normals">Smoothing Normals</a>.
  252. @param [cfg.smoothNormalsAngleThreshold=20] {Number} See <a href="#smoothing-normals">Smoothing Normals</a>.
  253. @param [cfg.backfaces=false] {Boolean} When true, allows visible backfaces, wherever specified in the STL. When false, ignores backfaces.
  254. @param [cfg.ghosted=false] {Boolean} When true, sets all the Model's Meshes initially ghosted.
  255. @param [cfg.highlighted=false] {Boolean} When true, sets all the Model's Meshes initially highlighted.
  256. @param [cfg.outline=false] {Boolean} When true, sets all the Model's Meshes initially outlined.
  257. @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.
  258. @param [cfg.transform] {Number|String|Transform} A Local-to-World-space (modelling) {{#crossLink "Transform"}}{{/crossLink}} to attach to this STLModel.
  259. Must be within the same {{#crossLink "Scene"}}{{/crossLink}} as this STLModel. Internally, the given
  260. {{#crossLink "Transform"}}{{/crossLink}} will be inserted above each top-most {{#crossLink "Transform"}}Transform{{/crossLink}}
  261. that the STLModel attaches to its {{#crossLink "Mesh"}}Meshes{{/crossLink}}.
  262. @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.|
  263. @param [cfg.position=[0,0,0]] {Float32Array} The STLModel's local 3D position.
  264. @param [cfg.scale=[1,1,1]] {Float32Array} The STLModel's local scale.
  265. @param [cfg.rotation=[0,0,0]] {Float32Array} The STLModel's local rotation, as Euler angles given in degrees.
  266. @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.
  267. @extends Model
  268. */
  269. {
  270. xeogl.STLModel = class xeoglSTLModel extends xeogl.Model {
  271.  
  272. init(cfg) {
  273. super.init(cfg);
  274. this._src = null;
  275. this._options = {
  276. combineGeometry: cfg.combineGeometry !== false,
  277. quantizeGeometry: cfg.quantizeGeometry !== false,
  278. edgeThreshold: cfg.edgeThreshold,
  279. splitMeshes: cfg.splitMeshes,
  280. smoothNormals: cfg.smoothNormals,
  281. smoothNormalsAngleThreshold: cfg.smoothNormalsAngleThreshold
  282. };
  283. this.src = cfg.src;
  284. }
  285.  
  286. /**
  287. Path to an STL file.
  288.  
  289. You can set this to a new file path at any time (except while loading), which will cause the STLModel to load components from
  290. the new file (after first destroying any components loaded from a previous file path).
  291.  
  292. Fires a {{#crossLink "STLModel/loaded:event"}}{{/crossLink}} event when the STL has loaded.
  293.  
  294. @property src
  295. @type String
  296. */
  297. set src(value) {
  298. if (!value) {
  299. return;
  300. }
  301. if (!xeogl._isString(value)) {
  302. this.error("Value for 'src' should be a string");
  303. return;
  304. }
  305. if (value === this._src) { // Already loaded this STLModel
  306.  
  307. /**
  308. Fired whenever this STLModel has finished loading components from the STL file
  309. specified by {{#crossLink "STLModel/src:property"}}{{/crossLink}}.
  310. @event loaded
  311. */
  312. this.fire("loaded", true, true);
  313. return;
  314. }
  315. this.clear();
  316. this._src = value;
  317. xeogl.STLModel.load(this, this._src, this._options);
  318. }
  319.  
  320. get source() {
  321. return this._src;
  322. }
  323.  
  324.  
  325. destroy() {
  326. this.destroyAll();
  327. super.destroy();
  328. }
  329.  
  330.  
  331. /**
  332. * Loads STL from a URL into a {{#crossLink "Model"}}{{/crossLink}}.
  333. *
  334. * @method load
  335. * @static
  336. * @param {Model} model Model to load into.
  337. * @param {String} src Path to STL file.
  338. * @param {Object} options Loading options.
  339. * @param {Function} [ok] Completion callback.
  340. * @param {Function} [error] Error callback.
  341. */
  342. static load(model, src, options, ok, error) {
  343. var spinner = model.scene.canvas.spinner;
  344. spinner.processes++;
  345. load(model, src, options, function () {
  346. spinner.processes--;
  347. xeogl.scheduleTask(function () {
  348. model.fire("loaded", true, true);
  349. });
  350. if (ok) {
  351. ok();
  352. }
  353. },
  354. function (msg) {
  355. spinner.processes--;
  356. model.error(msg);
  357. if (error) {
  358. error(msg);
  359. }
  360. /**
  361. Fired whenever this STLModel fails to load the STL file
  362. specified by {{#crossLink "STLModel/src:property"}}{{/crossLink}}.
  363. @event error
  364. @param msg {String} Description of the error
  365. */
  366. model.fire("error", msg);
  367. });
  368. }
  369.  
  370. /**
  371. * Parses STL into a {{#crossLink "Model"}}{{/crossLink}}.
  372. *
  373. * @method parse
  374. * @static
  375. * @param {Model} model Model to parse into.
  376. * @param {ArrayBuffer} data The STL data.
  377. * @param {Object} [options] Parsing options
  378. * @param {String} [options.basePath] Base path path to find external resources on, if any.
  379. * @param {String} [options.loadBuffer] Callback to load buffer files.
  380. */
  381. static parse(model, data, options) {
  382. options = options || {};
  383. var spinner = model.scene.canvas.spinner;
  384. spinner.processes++;
  385. parse(data, "", options, model, function () {
  386. spinner.processes--;
  387. model.fire("loaded", true, true);
  388. },
  389. function (msg) {
  390. spinner.processes--;
  391. model.error(msg);
  392. model.fire("error", msg);
  393. });
  394. }
  395. };
  396.  
  397. var load = (function () {
  398. function loadData(src, ok, error) {
  399. var request = new XMLHttpRequest();
  400. request.overrideMimeType("application/json");
  401. request.open('GET', src, true);
  402. request.responseType = 'arraybuffer';
  403. request.onreadystatechange = function () {
  404. if (request.readyState == 4 && request.status == "200") {
  405. ok(request.response, this);
  406. }
  407. };
  408. request.send(null);
  409. }
  410.  
  411. return function (model, src, options, ok, error) {
  412. loadData(src, function (data) { // OK
  413. parse(data, model, options);
  414. ok();
  415. },
  416. error);
  417. };
  418. })();
  419.  
  420. function parse(data, model, options) {
  421.  
  422. var entityCount = 0;
  423.  
  424. function isBinary(data) {
  425. var reader = new DataView(data);
  426. var numFaces = reader.getUint32(80, true);
  427. var faceSize = ( 32 / 8 * 3 ) + ( ( 32 / 8 * 3 ) * 3 ) + ( 16 / 8 );
  428. var numExpectedBytes = 80 + ( 32 / 8 ) + ( numFaces * faceSize );
  429. if (numExpectedBytes === reader.byteLength) {
  430. return true;
  431. }
  432. var solid = [115, 111, 108, 105, 100];
  433. for (var i = 0; i < 5; i++) {
  434. if (solid[i] != reader.getUint8(i, false)) {
  435. return true;
  436. }
  437. }
  438. return false;
  439. }
  440.  
  441. function parseBinary(data, model, options) {
  442. var autoVertexNormals = options.autoVertexNormals;
  443. var reader = new DataView(data);
  444. var faces = reader.getUint32(80, true);
  445. var r;
  446. var g;
  447. var b;
  448. var hasColors = false;
  449. var colors;
  450. var defaultR;
  451. var defaultG;
  452. var defaultB;
  453. var lastR = null;
  454. var lastG = null;
  455. var lastB = null;
  456. var newMesh = false;
  457. var alpha;
  458. var indices;
  459. var geometry;
  460. var mesh;
  461. for (var index = 0; index < 80 - 10; index++) {
  462. if (( reader.getUint32(index, false) == 0x434F4C4F /*COLO*/ ) &&
  463. ( reader.getUint8(index + 4) == 0x52 /*'R'*/ ) &&
  464. ( reader.getUint8(index + 5) == 0x3D /*'='*/ )) {
  465. hasColors = true;
  466. colors = [];
  467. defaultR = reader.getUint8(index + 6) / 255;
  468. defaultG = reader.getUint8(index + 7) / 255;
  469. defaultB = reader.getUint8(index + 8) / 255;
  470. alpha = reader.getUint8(index + 9) / 255;
  471. }
  472. }
  473. var material = new xeogl.MetallicMaterial(model, { // Share material with all meshes
  474. roughness: 0.5
  475. });
  476. // var material = new xeogl.PhongMaterial(model, { // Share material with all meshes
  477. // diffuse: [0.4, 0.4, 0.4],
  478. // reflectivity: 1,
  479. // specular: [0.5, 0.5, 1.0]
  480. // });
  481. model._addComponent(material);
  482. var dataOffset = 84;
  483. var faceLength = 12 * 4 + 2;
  484. var positions = [];
  485. var normals = [];
  486. var splitMeshes = options.splitMeshes;
  487. for (var face = 0; face < faces; face++) {
  488. var start = dataOffset + face * faceLength;
  489. var normalX = reader.getFloat32(start, true);
  490. var normalY = reader.getFloat32(start + 4, true);
  491. var normalZ = reader.getFloat32(start + 8, true);
  492. if (hasColors) {
  493. var packedColor = reader.getUint16(start + 48, true);
  494. if (( packedColor & 0x8000 ) === 0) {
  495. r = ( packedColor & 0x1F ) / 31;
  496. g = ( ( packedColor >> 5 ) & 0x1F ) / 31;
  497. b = ( ( packedColor >> 10 ) & 0x1F ) / 31;
  498. } else {
  499. r = defaultR;
  500. g = defaultG;
  501. b = defaultB;
  502. }
  503. if (splitMeshes && r !== lastR || g !== lastG || b !== lastB) {
  504. if (lastR !== null) {
  505. newMesh = true;
  506. }
  507. lastR = r;
  508. lastG = g;
  509. lastB = b;
  510. }
  511. }
  512. for (var i = 1; i <= 3; i++) {
  513. var vertexstart = start + i * 12;
  514. positions.push(reader.getFloat32(vertexstart, true));
  515. positions.push(reader.getFloat32(vertexstart + 4, true));
  516. positions.push(reader.getFloat32(vertexstart + 8, true));
  517. if (!autoVertexNormals) {
  518. normals.push(normalX, normalY, normalZ);
  519. }
  520. if (hasColors) {
  521. colors.push(r, g, b, 1); // TODO: handle alpha
  522. }
  523. }
  524. if (splitMeshes && newMesh) {
  525. addMesh(model, positions, normals, colors, material, options);
  526. positions = [];
  527. normals = [];
  528. colors = colors ? [] : null;
  529. newMesh = false;
  530. }
  531. }
  532. if (positions.length > 0) {
  533. addMesh(model, positions, normals, colors, material, options);
  534. }
  535. }
  536.  
  537. function parseASCII(data, model, options) {
  538. var faceRegex = /facet([\s\S]*?)endfacet/g;
  539. var faceCounter = 0;
  540. var floatRegex = /[\s]+([+-]?(?:\d+.\d+|\d+.|\d+|.\d+)(?:[eE][+-]?\d+)?)/.source;
  541. var vertexRegex = new RegExp('vertex' + floatRegex + floatRegex + floatRegex, 'g');
  542. var normalRegex = new RegExp('normal' + floatRegex + floatRegex + floatRegex, 'g');
  543. var positions = [];
  544. var normals = [];
  545. var colors = null;
  546. var normalx;
  547. var normaly;
  548. var normalz;
  549. var result;
  550. var verticesPerFace;
  551. var normalsPerFace;
  552. var text;
  553. while (( result = faceRegex.exec(data) ) !== null) {
  554. verticesPerFace = 0;
  555. normalsPerFace = 0;
  556. text = result[0];
  557. while (( result = normalRegex.exec(text) ) !== null) {
  558. normalx = parseFloat(result[1]);
  559. normaly = parseFloat(result[2]);
  560. normalz = parseFloat(result[3]);
  561. normalsPerFace++;
  562. }
  563. while (( result = vertexRegex.exec(text) ) !== null) {
  564. positions.push(parseFloat(result[1]), parseFloat(result[2]), parseFloat(result[3]));
  565. normals.push(normalx, normaly, normalz);
  566. verticesPerFace++;
  567. }
  568. if (normalsPerFace !== 1) {
  569. model.error("Error in normal of face " + faceCounter);
  570. }
  571. if (verticesPerFace !== 3) {
  572. model.error("Error in positions of face " + faceCounter);
  573. }
  574. faceCounter++;
  575. }
  576. var material = new xeogl.MetallicMaterial(model, {
  577. roughness: 0.5
  578. });
  579. // var material = new xeogl.PhongMaterial(model, {
  580. // diffuse: [0.4, 0.4, 0.4],
  581. // reflectivity: 1,
  582. // specular: [0.5, 0.5, 1.0]
  583. // });
  584. model._addComponent(material);
  585. addMesh(model, positions, normals, colors, material, options);
  586. }
  587.  
  588. function addMesh(model, positions, normals, colors, material, options) {
  589.  
  590. var indices = new Int32Array(positions.length / 3);
  591. for (var ni = 0, len = indices.length; ni < len; ni++) {
  592. indices[ni] = ni;
  593. }
  594.  
  595. normals = normals && normals.length > 0 ? normals : null;
  596. colors = colors && colors.length > 0 ? colors : null;
  597.  
  598. if (options.smoothNormals) {
  599. xeogl.math.faceToVertexNormals(positions, normals, options);
  600. }
  601.  
  602. var geometry = new xeogl.Geometry(model, {
  603. primitive: "triangles",
  604. positions: positions,
  605. normals: normals,
  606. // autoVertexNormals: !normals,
  607. colors: colors,
  608. indices: indices
  609. });
  610.  
  611. var mesh = new xeogl.Mesh(model, {
  612. id: model.id + "#" + entityCount++,
  613. geometry: geometry,
  614. material: material
  615. });
  616.  
  617. model._addComponent(geometry);
  618. model.addChild(mesh);
  619. }
  620.  
  621. function ensureString(buffer) {
  622. if (typeof buffer !== 'string') {
  623. return decodeText(new Uint8Array(buffer));
  624. }
  625. return buffer;
  626. }
  627.  
  628. function ensureBinary(buffer) {
  629. if (typeof buffer === 'string') {
  630. var arrayBuffer = new Uint8Array(buffer.length);
  631. for (var i = 0; i < buffer.length; i++) {
  632. arrayBuffer[i] = buffer.charCodeAt(i) & 0xff; // implicitly assumes little-endian
  633. }
  634. return arrayBuffer.buffer || arrayBuffer;
  635. } else {
  636. return buffer;
  637. }
  638. }
  639.  
  640. function decodeText(array) {
  641. if (typeof TextDecoder !== 'undefined') {
  642. return new TextDecoder().decode(array);
  643. }
  644. var s = '';
  645. for (var i = 0, il = array.length; i < il; i++) {
  646. s += String.fromCharCode(array[i]); // Implicitly assumes little-endian.
  647. }
  648. return decodeURIComponent(escape(s));
  649. }
  650.  
  651. var binData = ensureBinary(data);
  652.  
  653. return isBinary(binData) ? parseBinary(binData, model, options) : parseASCII(ensureString(data), model, options);
  654.  
  655. }
  656. }
  657.