js.blend.js 139 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545
  1. (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
  2. 'use strict'
  3. exports.byteLength = byteLength
  4. exports.toByteArray = toByteArray
  5. exports.fromByteArray = fromByteArray
  6. var lookup = []
  7. var revLookup = []
  8. var Arr = typeof Uint8Array !== 'undefined' ? Uint8Array : Array
  9. var code = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/'
  10. for (var i = 0, len = code.length; i < len; ++i) {
  11. lookup[i] = code[i]
  12. revLookup[code.charCodeAt(i)] = i
  13. }
  14. revLookup['-'.charCodeAt(0)] = 62
  15. revLookup['_'.charCodeAt(0)] = 63
  16. function placeHoldersCount (b64) {
  17. var len = b64.length
  18. if (len % 4 > 0) {
  19. throw new Error('Invalid string. Length must be a multiple of 4')
  20. }
  21. // the number of equal signs (place holders)
  22. // if there are two placeholders, than the two characters before it
  23. // represent one byte
  24. // if there is only one, then the three characters before it represent 2 bytes
  25. // this is just a cheap hack to not do indexOf twice
  26. return b64[len - 2] === '=' ? 2 : b64[len - 1] === '=' ? 1 : 0
  27. }
  28. function byteLength (b64) {
  29. // base64 is 4/3 + up to two characters of the original data
  30. return (b64.length * 3 / 4) - placeHoldersCount(b64)
  31. }
  32. function toByteArray (b64) {
  33. var i, l, tmp, placeHolders, arr
  34. var len = b64.length
  35. placeHolders = placeHoldersCount(b64)
  36. arr = new Arr((len * 3 / 4) - placeHolders)
  37. // if there are placeholders, only get up to the last complete 4 chars
  38. l = placeHolders > 0 ? len - 4 : len
  39. var L = 0
  40. for (i = 0; i < l; i += 4) {
  41. tmp = (revLookup[b64.charCodeAt(i)] << 18) | (revLookup[b64.charCodeAt(i + 1)] << 12) | (revLookup[b64.charCodeAt(i + 2)] << 6) | revLookup[b64.charCodeAt(i + 3)]
  42. arr[L++] = (tmp >> 16) & 0xFF
  43. arr[L++] = (tmp >> 8) & 0xFF
  44. arr[L++] = tmp & 0xFF
  45. }
  46. if (placeHolders === 2) {
  47. tmp = (revLookup[b64.charCodeAt(i)] << 2) | (revLookup[b64.charCodeAt(i + 1)] >> 4)
  48. arr[L++] = tmp & 0xFF
  49. } else if (placeHolders === 1) {
  50. tmp = (revLookup[b64.charCodeAt(i)] << 10) | (revLookup[b64.charCodeAt(i + 1)] << 4) | (revLookup[b64.charCodeAt(i + 2)] >> 2)
  51. arr[L++] = (tmp >> 8) & 0xFF
  52. arr[L++] = tmp & 0xFF
  53. }
  54. return arr
  55. }
  56. function tripletToBase64 (num) {
  57. return lookup[num >> 18 & 0x3F] + lookup[num >> 12 & 0x3F] + lookup[num >> 6 & 0x3F] + lookup[num & 0x3F]
  58. }
  59. function encodeChunk (uint8, start, end) {
  60. var tmp
  61. var output = []
  62. for (var i = start; i < end; i += 3) {
  63. tmp = (uint8[i] << 16) + (uint8[i + 1] << 8) + (uint8[i + 2])
  64. output.push(tripletToBase64(tmp))
  65. }
  66. return output.join('')
  67. }
  68. function fromByteArray (uint8) {
  69. var tmp
  70. var len = uint8.length
  71. var extraBytes = len % 3 // if we have 1 byte left, pad 2 bytes
  72. var output = ''
  73. var parts = []
  74. var maxChunkLength = 16383 // must be multiple of 3
  75. // go through the array every three bytes, we'll deal with trailing stuff later
  76. for (var i = 0, len2 = len - extraBytes; i < len2; i += maxChunkLength) {
  77. parts.push(encodeChunk(uint8, i, (i + maxChunkLength) > len2 ? len2 : (i + maxChunkLength)))
  78. }
  79. // pad the end with zeros, but make sure to not forget the extra bytes
  80. if (extraBytes === 1) {
  81. tmp = uint8[len - 1]
  82. output += lookup[tmp >> 2]
  83. output += lookup[(tmp << 4) & 0x3F]
  84. output += '=='
  85. } else if (extraBytes === 2) {
  86. tmp = (uint8[len - 2] << 8) + (uint8[len - 1])
  87. output += lookup[tmp >> 10]
  88. output += lookup[(tmp >> 4) & 0x3F]
  89. output += lookup[(tmp << 2) & 0x3F]
  90. output += '='
  91. }
  92. parts.push(output)
  93. return parts.join('')
  94. }
  95. },{}],2:[function(require,module,exports){
  96. /*jshint esversion: 6 */
  97. const three = require("./threejs/blend_three.js");
  98. const parser = require("./parser/parser.js")();
  99. function loadFile(blender_file, res, rej){
  100. three_module = three(blender_file);
  101. //TODO: Report any errors with ThreeJS before continuing.
  102. res({
  103. file : blender_file,
  104. three : three_module
  105. });
  106. }
  107. /* This represents a parsed blendfile instance if parsing is successful. It will accept a string or a binary data object. Strings must be a valid URI to a blender file. Binary data may be in the form of an ArrayBuffer, TypedArray, or a Blob. Binary data must also contain the binary data of a blender file.*/
  108. JSBLEND = (fileuri_or_filedata, name = "")=>{
  109. const promise = new Promise(
  110. (res, rej) =>{
  111. parser.onParseReady = (blender_file) => {
  112. loadFile(blender_file, res, rej);
  113. };
  114. //If fileuri_or_filedata is a string, attempt to load the file asynchronously
  115. if(typeof fileuri_or_filedata == "string"){
  116. let request = new XMLHttpRequest();
  117. request.open("GET", fileuri_or_filedata, true);
  118. request.responseType = 'blob';
  119. request.onload = () => {
  120. let file = request.response;
  121. parser.loadBlendFromBlob(new Blob([file]), fileuri_or_filedata);
  122. };
  123. request.send();
  124. return;
  125. }
  126. debugger
  127. if(typeof fileuri_or_filedata == "object"){
  128. //Attempt to load from blob or array buffer;
  129. if(fileuri_or_filedata instanceof ArrayBuffer){
  130. parser.loadBlendFromArrayBuffer(fileuri_or_filedata, name);
  131. return;
  132. }
  133. if(fileuri_or_filedata instanceof Blob){
  134. parser.loadBlendFromBlob(fileuri_or_filedata, name);
  135. return;
  136. }
  137. }
  138. //Unknown file type passed -> abort and reject
  139. console.warn("Unsupported file type passed to JSBlend", fileuri_or_filedata);
  140. rej("Unsupported file type passed to JSBlend");
  141. }
  142. );
  143. return promise;
  144. };
  145. },{"./parser/parser.js":3,"./threejs/blend_three.js":4}],3:[function(require,module,exports){
  146. /*jshint esversion: 6 */
  147. const DNA1 = 826363460;
  148. const ENDB = 1111772741;
  149. /* Note: Blender cooridinates treat the Z axis as the verticle and Y as depth. */
  150. module.exports = (function(unzipper) {
  151. //A helper object to identify Blender Object structs by type name.
  152. var blender_types = {
  153. mesh_object: 1,
  154. lamp: 10,
  155. };
  156. //web worker not functional in this version
  157. USE_WEBWORKER = false;
  158. var worker = null,
  159. FR = new FileReader(),
  160. return_object = {
  161. loadBlendFromArrayBuffer: function(array_buffer) {
  162. return_object.ready = false;
  163. if (USE_WEBWORKER) {
  164. worker.postMessage(array_buffer, array_buffer);
  165. } else {
  166. worker.onmessage({
  167. data: array_buffer
  168. });
  169. }
  170. },
  171. loadBlendFromBlob: function(blob) {
  172. FR.onload = function() {
  173. return_object.loadBlendFromArrayBuffer(this.result);
  174. };
  175. FR.readAsArrayBuffer(blob);
  176. },
  177. ready: true,
  178. onParseReady: function() {},
  179. };
  180. worker = new worker_code();
  181. worker.postMessage = function(message) {
  182. return_object.onParseReady(message);
  183. };
  184. function worker_code() {
  185. "use strict";
  186. var data = null,
  187. _data = null,
  188. BIG_ENDIAN = false,
  189. pointer_size = 0,
  190. struct_names = [],
  191. offset = 0,
  192. working_blend_file = null,
  193. current_SDNA_template = null,
  194. templates = {},
  195. finished_objects = [],
  196. FILE = null,
  197. AB = null;
  198. function parseFile(msg) {
  199. var self = this;
  200. if (typeof msg.data == "object") {
  201. // reset global variables
  202. AB = null;
  203. data = null;
  204. BIG_ENDIAN = false;
  205. pointer_size = 0;
  206. struct_names = [];
  207. offset = 0;
  208. working_blend_file = null;
  209. finished_objects = [];
  210. current_SDNA_template = null;
  211. // set data
  212. _data = msg.data;
  213. AB = _data.slice();
  214. data = new DataView(_data);
  215. FILE = new BLENDER_FILE(AB);
  216. //start parsing
  217. readFile();
  218. //export parsed data
  219. self.postMessage(FILE);
  220. }
  221. }
  222. /*
  223. Export object for a parsed __blender_file__.
  224. */
  225. var BLENDER_FILE = function(AB) {
  226. this.AB = AB;
  227. //this.double = new Float64Array(AB);
  228. this.byte = new Uint8Array(AB);
  229. this.dv = new DataView(AB);
  230. this.objects = {};
  231. this.memory_lookup = {},
  232. this.object_array = [];
  233. this.template = null;
  234. };
  235. BLENDER_FILE.prototype = {
  236. addObject: function(obj) {
  237. this.object_array.push(obj);
  238. if (!this.objects[obj.blender_name]) this.objects[obj.blender_name] = [];
  239. this.objects[obj.blender_name].push(obj);
  240. },
  241. primeTypes: function(list_of_dna_names) {
  242. for (var i = 0; i < list_of_dna_names.length; i++) {
  243. //this.objects[list_of_dna_names[i]] = [];
  244. }
  245. },
  246. getPointer: function(offset) {
  247. var pointerLow = this.dv.getUint32(offset, this.template.endianess);
  248. if (this.template.pointer_size > 4) {
  249. var pointerHigh = this.dv.getUint32(offset + 4, this.template.endianess);
  250. if (this.template.endianess) {
  251. return (pointerLow) + "l|h" + pointerHigh;
  252. } else {
  253. return (pointerHigh) + "h|l" + pointerLow;
  254. }
  255. } else {
  256. return pointerLow;
  257. }
  258. }
  259. };
  260. function getDocument(data) {
  261. var obj = readFile(null, data);
  262. }
  263. self.onmessage = parseFile;
  264. this.onmessage = parseFile;
  265. /*
  266. These functions map offsets in the blender __blender_file__ to basic types (byte,short,int,float) through TypedArrays;
  267. This allows the underlying binary data to be changed.
  268. */
  269. function float64Prop(offset, Blender_Array_Length, length) {
  270. return {
  271. get: function() {
  272. return (Blender_Array_Length > 1) ?
  273. new Float64Array(this.__blender_file__.AB, this.__data_address__ + offset, length) :
  274. this.__blender_file__.dv.getFloat64(this.__data_address__ + offset, this.__blender_file__.template.endianess);
  275. },
  276. set: function(float) {
  277. if (Blender_Array_Length > 1) {} else {
  278. this.__blender_file__.dv.setFloat64(this.__data_address__ + offset, float, this.__blender_file__.template.endianess);
  279. }
  280. },
  281. };
  282. }
  283. function floatProp(offset, Blender_Array_Length, length) {
  284. return {
  285. get: function() {
  286. return (Blender_Array_Length > 1) ?
  287. new Float32Array(this.__blender_file__.AB, this.__data_address__ + offset, length) :
  288. this.__blender_file__.dv.getFloat32(this.__data_address__ + offset, this.__blender_file__.template.endianess);
  289. },
  290. set: function(float) {
  291. if (Blender_Array_Length > 1) {} else {
  292. this.__blender_file__.dv.setFloat32(this.__data_address__ + offset, float, this.__blender_file__.template.endianess);
  293. }
  294. },
  295. };
  296. }
  297. function intProp(offset, Blender_Array_Length, length) {
  298. return {
  299. get: function() {
  300. return (Blender_Array_Length > 1) ?
  301. new Int32Array(this.__blender_file__.AB, this.__data_address__ + offset, length) :
  302. this.__blender_file__.dv.getInt32(this.__data_address__ + offset, this.__blender_file__.template.endianess);
  303. },
  304. set: function(int) {
  305. if (Blender_Array_Length > 1) {} else {
  306. this.__blender_file__.dv.setInt32(this.__data_address__ + offset, float, this.__blender_file__.template.endianess);
  307. }
  308. },
  309. };
  310. }
  311. function uIntProp(offset, Blender_Array_Length, length) {
  312. return {
  313. get: function() {
  314. return (Blender_Array_Length > 1) ?
  315. new Uint32Array(this.__blender_file__.AB, this.__data_address__ + offset, length) :
  316. this.__blender_file__.dv.getUint32(this.__data_address__ + offset, this.__blender_file__.template.endianess);
  317. },
  318. set: function(int) {
  319. if (Blender_Array_Length > 1) {} else {
  320. this.__blender_file__.dv.setUint32(this.__data_address__ + offset, float, this.__blender_file__.template.endianess);
  321. }
  322. },
  323. };
  324. }
  325. function shortProp(offset, Blender_Array_Length, length) {
  326. return {
  327. get: function() {
  328. return (Blender_Array_Length > 1) ?
  329. new Int16Array(this.__blender_file__.AB, this.__data_address__ + offset, length) :
  330. this.__blender_file__.dv.getInt16(this.__data_address__ + offset, this.__blender_file__.template.endianess);
  331. },
  332. set: function(float) {
  333. if (Blender_Array_Length > 1) {} else {
  334. this.__blender_file__.dv.setInt16(this.__data_address__ + offset, float, this.__blender_file__.template.endianess);
  335. }
  336. },
  337. };
  338. }
  339. var uShortProp = (offset, Blender_Array_Length, length) => {
  340. return {
  341. get: function() {
  342. return (Blender_Array_Length > 1) ?
  343. new Uint16Array(this.__blender_file__.AB, this.__data_address__ + offset, length) :
  344. this.__blender_file__.dv.getUint16(this.__data_address__ + offset, this.__blender_file__.template.endianess);
  345. },
  346. set: function(float) {
  347. if (Blender_Array_Length > 1) {} else {
  348. this.__blender_file__.dv.setUint16(this.__data_address__ + offset, float, this.__blender_file__.template.endianess);
  349. }
  350. },
  351. };
  352. }
  353. function charProp(offset, Blender_Array_Length, length) {
  354. return {
  355. get: function() {
  356. if (Blender_Array_Length > 1) {
  357. let start = this.__data_address__ + offset;
  358. let end = start;
  359. let buffer_guard = 0;
  360. while (this.__blender_file__.byte[end] != 0 && buffer_guard++ < length) end++;
  361. return toString(this.__blender_file__.AB, start, end)
  362. }
  363. return this.__blender_file__.byte[(this.__data_address__ + offset)];
  364. },
  365. set: function(byte) {
  366. if (Blender_Array_Length > 1) {
  367. var string = byte + "",
  368. i = 0,
  369. l = string.length;
  370. while (i < length) {
  371. if (i < l) {
  372. this.__blender_file__.byte[(this.__data_address__ + offset + i)] = string.charCodeAt(i) | 0;
  373. } else {
  374. this.__blender_file__.byte[(this.__data_address__ + offset + i)] = 0;
  375. }
  376. i++;
  377. }
  378. } else {
  379. this.__blender_file__.byte[(this.__data_address__ + offset)] = byte | 0;
  380. }
  381. }
  382. };
  383. }
  384. function pointerProp2(offset) {
  385. return {
  386. get: function() {
  387. let pointer = this.__blender_file__.getPointer(this.__data_address__ + offset, this.__blender_file__);
  388. var link = this.__blender_file__.memory_lookup[pointer];
  389. var results = [];
  390. if (link) {
  391. var address = link.__data_address__;
  392. let j = 0;
  393. while (true) {
  394. pointer = this.__blender_file__.getPointer(address + j * 8, this.__blender_file__);
  395. let obj = this.__blender_file__.memory_lookup[pointer];
  396. if (!obj) break;
  397. results.push(obj);
  398. j++
  399. }
  400. };
  401. return results;
  402. },
  403. set: function() {}
  404. }
  405. }
  406. function pointerProp(offset, Blender_Array_Length, length) {
  407. return {
  408. get: function() {
  409. if (Blender_Array_Length > 1) {
  410. let array = [];
  411. let j = 0;
  412. let off = offset;
  413. while (j < Blender_Array_Length) {
  414. let pointer = this.__blender_file__.getPointer(this.__data_address__ + off, this.__blender_file__);
  415. array.push(this.__blender_file__.memory_lookup[pointer]);
  416. off += length ///this.__blender_file__.template.pointer_size;
  417. j++;
  418. }
  419. return array;
  420. } else {
  421. let pointer = this.__blender_file__.getPointer(this.__data_address__ + offset, this.__blender_file__);
  422. return this.__blender_file__.memory_lookup[pointer];
  423. }
  424. },
  425. set: function() {}
  426. }
  427. }
  428. function compileProp(obj, name, type, offset, array_size, IS_POINTER, pointer_size, length) {
  429. if (!IS_POINTER) {
  430. switch (type) {
  431. case "double":
  432. Object.defineProperty(obj, name, float64Prop(offset, array_size, length >> 3));
  433. break;
  434. case "float":
  435. Object.defineProperty(obj, name, floatProp(offset, array_size, length >> 2));
  436. break;
  437. case "int":
  438. Object.defineProperty(obj, name, intProp(offset, array_size, length >> 2));
  439. break;
  440. case "short":
  441. case "ushort":
  442. Object.defineProperty(obj, name, shortProp(offset, array_size, length >> 1));
  443. break;
  444. case "char":
  445. case "uchar":
  446. Object.defineProperty(obj, name, charProp(offset, array_size, length));
  447. break;
  448. default:
  449. //compile list to
  450. obj[name] = {};
  451. obj.__list__.push(name, type, length, offset, array_size, IS_POINTER);
  452. }
  453. obj._length += length;
  454. offset += length;
  455. } else {
  456. Object.defineProperty(obj, name, pointerProp(offset, array_size, pointer_size));
  457. offset += pointer_size * array_size;
  458. }
  459. return offset;
  460. }
  461. //Store final DNA structs
  462. var MASTER_SDNA_SCHEMA = function(version) {
  463. this.version = version;
  464. this.SDNA_SET = false;
  465. this.byte_size = 0;
  466. this.struct_index = 0;
  467. this.structs = {};
  468. this.SDNA = {};
  469. this.endianess = false;
  470. };
  471. MASTER_SDNA_SCHEMA.prototype = {
  472. getSDNAStructureConstructor: function(name, struct) {
  473. if (struct) {
  474. var blen_struct = Function("function " + name + "(){}; return " + name)();
  475. blen_struct.prototype = new BLENDER_STRUCTURE();
  476. blen_struct.prototype.blender_name = name;
  477. blen_struct.prototype.__pointers = [];
  478. blen_struct.prototype.__list__ = [];
  479. var offset = 0;
  480. //Create properties of struct
  481. for (var i = 0; i < struct.length; i += 3) {
  482. var _name = struct[i],
  483. n = _name,
  484. type = struct[i + 1],
  485. length = struct[i + 2],
  486. array_length = 0,
  487. match = null,
  488. Blender_Array_Length = 1,
  489. Suparray_match = 1,
  490. PointerToArray = false,
  491. Pointer_Match = 0;
  492. var DNA = this.SDNA[name] = {
  493. constructor: blen_struct
  494. };
  495. let original_name = _name;
  496. //mini type parser
  497. if ((match = _name.match(/(\*?)(\*?)(\w+)(\[(\w*)\])?(\[(\w*)\])?/))) {
  498. //base name
  499. _name = match[3];
  500. //pointer type
  501. if (match[1]) {
  502. Pointer_Match = 10;
  503. blen_struct.prototype.__pointers.push(_name);
  504. }
  505. if (match[2]) {
  506. PointerToArray = true;
  507. }
  508. //arrays
  509. if (match[4]) {
  510. if (match[6]) {
  511. Suparray_match = parseInt(match[5]);
  512. Blender_Array_Length = parseInt(match[7]);
  513. } else {
  514. Blender_Array_Length = parseInt(match[5]);
  515. }
  516. }
  517. array_length = Blender_Array_Length * length;
  518. length = array_length * Suparray_match;
  519. }
  520. DNA[n] = {
  521. type: type,
  522. length: length,
  523. isArray: (Blender_Array_Length > 0),
  524. };
  525. if (PointerToArray) {
  526. Object.defineProperty(blen_struct.prototype, _name, pointerProp2(offset));
  527. offset += pointer_size;
  528. } else if (Suparray_match > 1) {
  529. var array_names = new Array(Suparray_match);
  530. //construct sub_array object that will return the correct structs
  531. for (var j = 0; j < Suparray_match; j++) {
  532. let array_name_ = `__${_name}[${j}]__`;
  533. array_names[j] = array_name_;
  534. offset = compileProp(blen_struct.prototype, array_name_, type, offset, Blender_Array_Length, Pointer_Match, pointer_size, array_length);
  535. }
  536. Object.defineProperty(blen_struct.prototype, _name, {
  537. get: (function(array_names) {
  538. return function() {
  539. var array = [];
  540. for (var i = 0; i < array_names.length; i++) {
  541. array.push(this[array_names[i]])
  542. }
  543. return array;
  544. }
  545. })(array_names)
  546. });
  547. } else {
  548. offset = compileProp(blen_struct.prototype, _name, type, offset, Blender_Array_Length, Pointer_Match, pointer_size, length);
  549. }
  550. }
  551. return this.SDNA[name].constructor;
  552. } else {
  553. if (!this.SDNA[name]) {
  554. return null;
  555. }
  556. return this.SDNA[name].constructor;
  557. }
  558. }
  559. };
  560. var BLENDER_STRUCTURE = function() {
  561. this.__blender_file__ = null;
  562. this.__list__ = null;
  563. this.__super_array_list__ = null;
  564. this.blender_name = "";
  565. this.__pointers = null;
  566. this.address = null;
  567. this.length = 0;
  568. this.__data_address__ = 0;
  569. this.blender_name = "";
  570. this._length = 0;
  571. };
  572. /*
  573. Returns a pre-constructed BLENDER_STRUCTURE or creates a new BLENDER_STRUCTURE to match the DNA struct type
  574. */
  575. var pointer_function = (pointer) => () => {
  576. return FILE.memory_lookup[pointer]
  577. };
  578. function getPointer(offset) {
  579. var pointerLow = data.getUint32(offset, BIG_ENDIAN);
  580. if (pointer_size > 4) {
  581. var pointerHigh = data.getUint32(offset + 4, BIG_ENDIAN);
  582. if (BIG_ENDIAN) {
  583. return (pointerLow) + "" + pointerHigh;
  584. } else {
  585. return (pointerHigh) + "" + pointerLow;
  586. }
  587. } else {
  588. return pointerLow;
  589. }
  590. }
  591. BLENDER_STRUCTURE.prototype = {
  592. setData: function(pointer, _data_offset, data_block_length, BLENDER_FILE) {
  593. if (this.__list__ == null) return this;
  594. BLENDER_FILE.addObject(this);
  595. this.__blender_file__ = BLENDER_FILE;
  596. var struct = this.__list__,
  597. j = 0,
  598. i = 0,
  599. obj, name = "",
  600. type, length, Blender_Array_Length, Pointer_Match, offset, constructor;
  601. this.__data_address__ = _data_offset;
  602. if (struct === null) return this;
  603. for (i = 0; i < struct.length; i += 6) {
  604. obj = null;
  605. name = struct[i];
  606. type = struct[i + 1];
  607. Blender_Array_Length = struct[i + 4];
  608. Pointer_Match = struct[i + 5];
  609. offset = this.__data_address__ + struct[i + 3];
  610. if (Blender_Array_Length > 1) {
  611. this[name] = [];
  612. j = 0;
  613. while (j < Blender_Array_Length) {
  614. if (current_SDNA_template.getSDNAStructureConstructor(type)) {
  615. constructor = current_SDNA_template.getSDNAStructureConstructor(type);
  616. this[name].push((new constructor()).setData(0, offset, offset + length / Blender_Array_Length, BLENDER_FILE));
  617. } else this[name].push(null);
  618. offset += length / Blender_Array_Length;
  619. j++;
  620. }
  621. } else {
  622. if (current_SDNA_template.getSDNAStructureConstructor(type)) {
  623. constructor = current_SDNA_template.getSDNAStructureConstructor(type);
  624. this[name] = (new constructor()).setData(0, offset, length + offset, BLENDER_FILE);
  625. } else this[name] = null;
  626. }
  627. }
  628. //break connection to configuration list
  629. this.__list__ = null;
  630. return this;
  631. },
  632. get aname() {
  633. if (this.id) return this.id.name.slice(2);
  634. else return undefined;
  635. }
  636. };
  637. function toString(buffer, _in, _out) {
  638. return String.fromCharCode.apply(String, new Uint8Array(buffer, _in, _out - _in));
  639. }
  640. //Begin parsing blender __blender_file__
  641. function readFile() {
  642. var count = 0;
  643. var offset2 = 0;
  644. var root = 0;
  645. var i = 0;
  646. var data_offset = 0;
  647. var sdna_index = 0;
  648. var code = "";
  649. var block_length = 0;
  650. var curr_count = 0;
  651. var curr_count2 = 0;
  652. FILE.memory_lookup = {};
  653. struct_names = [];
  654. offset = 0;
  655. // Make sure we have a .blend __blender_file__. All blend files have the first 12bytes
  656. // set with BLENDER-v### in Utf-8
  657. if (toString(_data, offset, 7) !== "BLENDER") return console.warn("File supplied is not a .blend compatible Blender file.");
  658. // otherwise get templete from save version.
  659. offset += 7;
  660. pointer_size = ((toString(_data, offset++, offset)) == "_") ? 4 : 8;
  661. BIG_ENDIAN = toString(_data, offset++, offset) !== "V";
  662. var version = toString(_data, offset, offset + 3);
  663. //create new master template if none exist for current blender version;
  664. if (!templates[version]) {
  665. templates[version] = new MASTER_SDNA_SCHEMA(version);
  666. }
  667. current_SDNA_template = templates[version];
  668. FILE.template = current_SDNA_template;
  669. offset += 3;
  670. //Set SDNA structs if template hasn't been set.
  671. //Todo: Move the following block into the MASTER_SDNA_SCHEMA object.
  672. //*Like so:*/ current_SDNA_template.set(AB);
  673. if (!current_SDNA_template.SDNA_SET) {
  674. current_SDNA_template.endianess = BIG_ENDIAN;
  675. current_SDNA_template.pointer_size = pointer_size;
  676. //find DNA1 data block
  677. offset2 = offset;
  678. while (true) {
  679. sdna_index = data.getInt32(offset2 + pointer_size + 8, BIG_ENDIAN);
  680. code = toString(_data, offset2, offset2 + 4).replace(/\u0000/g, "");
  681. block_length = data.getInt32(offset2 + 4, true);
  682. offset2 += 16 + (pointer_size);
  683. if (code === "DNA1") {
  684. // DNA found; This is the core of the __blender_file__ and contains all the structure for the various data types used in Blender.
  685. count = 0;
  686. var types = [],
  687. fields = [],
  688. names = [],
  689. lengths = [],
  690. name = "",
  691. curr_name = "";
  692. //skip SDNA and NAME identifiers
  693. offset2 += 8;
  694. //Number of structs.
  695. count = data.getInt32(offset2, true);
  696. offset2 += 4;
  697. curr_count = 0;
  698. //Build up list of names for structs
  699. while (curr_count < count) {
  700. curr_name = "";
  701. while (data.getInt8(offset2) !== 0) {
  702. curr_name += toString(_data, offset2, offset2 + 1);
  703. offset2++;
  704. }
  705. names.push(curr_name);
  706. offset2++;
  707. curr_count++;
  708. }
  709. //Adjust for 4byte alignment
  710. if ((offset2 % 4) > 0) offset2 = (4 - (offset2 % 4)) + offset2;
  711. offset2 += 4;
  712. //Number of struct types
  713. count = data.getInt32(offset2, true);
  714. offset2 += 4;
  715. curr_count = 0;
  716. //Build up list of types
  717. while (curr_count < count) {
  718. curr_name = "";
  719. while (data.getInt8(offset2) !== 0) {
  720. curr_name += toString(_data, offset2, offset2 + 1);
  721. offset2++;
  722. }
  723. types.push(curr_name);
  724. offset2++;
  725. curr_count++;
  726. }
  727. //Adjust for 4byte alignment
  728. if ((offset2 % 4) > 0) offset2 = (4 - (offset2 % 4)) + offset2;
  729. offset2 += 4;
  730. curr_count = 0;
  731. //Build up list of byte lengths for types
  732. while (curr_count < count) {
  733. lengths.push(data.getInt16(offset2, BIG_ENDIAN));
  734. offset2 += 2;
  735. curr_count++;
  736. }
  737. //Adjust for 4byte alignment
  738. if ((offset2 % 4) > 0) offset2 = (4 - (offset2 % 4)) + offset2;
  739. offset2 += 4;
  740. //Number of structures
  741. var structure_count = data.getInt32(offset2, BIG_ENDIAN);
  742. offset2 += 4;
  743. curr_count = 0;
  744. //Create constructor objects from list of SDNA structs
  745. while (curr_count < structure_count) {
  746. var struct_name = types[data.getInt16(offset2, BIG_ENDIAN)];
  747. offset2 += 2;
  748. obj = [];
  749. count = data.getInt16(offset2, BIG_ENDIAN);
  750. offset2 += 2;
  751. curr_count2 = 0;
  752. struct_names.push(struct_name);
  753. //Fill an array with name, type, and length for each SDNA struct property
  754. while (curr_count2 < count) {
  755. obj.push(names[data.getInt16(offset2 + 2, BIG_ENDIAN)], types[data.getInt16(offset2, BIG_ENDIAN)], lengths[data.getInt16(offset2, BIG_ENDIAN)]);
  756. offset2 += 4;
  757. curr_count2++;
  758. }
  759. //Create a SDNA constructor by passing [type,name,lenth] array as second argument
  760. current_SDNA_template.getSDNAStructureConstructor(struct_name, obj);
  761. curr_count++;
  762. }
  763. current_SDNA_template.SDNA_SET = true;
  764. current_SDNA_template.SDNA_NAMES = struct_names;
  765. break;
  766. }
  767. offset2 += block_length;
  768. }
  769. }
  770. //parse the rest of the data, starting back at the top.
  771. //TODO: turn into "on-demand" parsing.
  772. while (true) {
  773. if ((offset % 4) > 0) {
  774. offset = (4 - (offset % 4)) + offset;
  775. }
  776. data_offset = offset;
  777. sdna_index = data.getInt32(offset + pointer_size + 8, BIG_ENDIAN);
  778. let code_uint = data.getUint32(offset, BIG_ENDIAN);
  779. offset2 = offset + 16 + (pointer_size);
  780. offset += data.getInt32(offset + 4, true) + 16 + (pointer_size);
  781. if (code_uint === DNA1); //skip - already processed at this point
  782. else if (code_uint === ENDB) break; //end of __blender_file__ found
  783. else {
  784. //Create a Blender object using a constructor template from current_SDNA_template
  785. var data_start = data_offset + pointer_size + 16;
  786. //Get a SDNA constructor by name;
  787. var constructor = current_SDNA_template.getSDNAStructureConstructor(current_SDNA_template.SDNA_NAMES[sdna_index]);
  788. var size = data.getInt32(data_offset + 4, BIG_ENDIAN);
  789. count = data.getInt32(data_offset + 12 + pointer_size, BIG_ENDIAN);
  790. if (count > 0) {
  791. var obj = new constructor();
  792. var length = constructor.prototype._length;
  793. var address = FILE.getPointer(data_offset + 8);
  794. obj.address = address + "";
  795. obj.setData(address, data_start, data_start + size, FILE);
  796. if (count > 1) {
  797. let array = [];
  798. array.push(obj);
  799. for (var u = 1; u < count; u++) {
  800. obj = new constructor();
  801. obj.setData(address, data_start + length * u, data_start + (length * u) + length, FILE);
  802. array.push(obj);
  803. }
  804. FILE.memory_lookup[address] = array;
  805. } else {
  806. FILE.memory_lookup[address] = obj;
  807. }
  808. }
  809. }
  810. }
  811. }
  812. }
  813. return return_object;
  814. });
  815. },{}],4:[function(require,module,exports){
  816. /*jshint esversion: 6 */
  817. const createMaterial = require("./material.js");
  818. const createTexture = require("./texture.js");
  819. const createMesh = require("./mesh.js");
  820. const createLight = require("./light.js");
  821. function loadModel(three_scene, model_name, blender_file, cache) {
  822. var mats = blender_mesh.mat,
  823. materials = [];
  824. for (var i = 0; i < mats.length; i++) {
  825. var material = createThreeJSMaterial(mats[i]);
  826. materials.push(material);
  827. }
  828. }
  829. var blender_types = {
  830. mesh_object: 1,
  831. lamp: 10
  832. };
  833. function loadScene(three_scene, blender_file, cache) {
  834. //build object from blender mesh object
  835. for (let i = 0; i < blender_file.objects.Object.length; i++) {
  836. let obj = blender_file.objects.Object[i];
  837. //Load Lights
  838. if (obj.type == blender_types.lamp) {
  839. let light = createLight(obj, blender_file);
  840. three_scene.add(light);
  841. }
  842. //Load Meshes
  843. if (obj.type == blender_types.mesh_object) {
  844. if (obj.data) {
  845. //get the mesh
  846. var buffered_geometry = createMesh(obj.data, [0, 0, 0]);
  847. var blend_material = obj.data.mat[0];
  848. if(blend_material){
  849. var material = createMaterial(blend_material);
  850. }else{
  851. //create generic material
  852. }
  853. //var geometry = createThreeJSGeometry(obj.data, [0, 0, 0]);
  854. ///*
  855. //create a transform from the mesh object
  856. var mesh = new THREE.Mesh(buffered_geometry, material);
  857. mesh.castShadow = true;
  858. mesh.receiveShadow = true;
  859. three_scene.add(mesh);
  860. mesh.rotateZ(obj.rot[2]);
  861. mesh.rotateY(obj.rot[1]);
  862. mesh.rotateX(obj.rot[0]);
  863. mesh.scale.fromArray(obj.size, 0);
  864. mesh.position.fromArray([obj.loc[0], (obj.loc[2]), (-obj.loc[1])], 0);
  865. //*/
  866. }
  867. }
  868. }
  869. }
  870. module.exports = (blender_file) => {
  871. if (!THREE) {
  872. console.warn("No ThreeJS object detected");
  873. return {};
  874. }
  875. var cache = {};
  876. return {
  877. loadScene: (three_scene) => loadScene(three_scene, blender_file, cache),
  878. loadModel: (model_name) => loadModel(model_name, blender_file, cache)
  879. };
  880. };
  881. },{"./light.js":5,"./material.js":6,"./mesh.js":7,"./texture.js":8}],5:[function(require,module,exports){
  882. /*jshint esversion: 6 */
  883. var blender_light_types = {
  884. point: 0,
  885. sun: 1,
  886. spot: 0,
  887. hemi: 0,
  888. area: 0
  889. };
  890. module.exports = function createThreeJSLamp(blend_lamp) {
  891. let ldata = blend_lamp.data;
  892. let pos_array = [blend_lamp.loc[0], blend_lamp.loc[2], -blend_lamp.loc[1]];
  893. let color = ((ldata.r * 255) << 16) | ((ldata.g * 255) << 8) | ((ldata.b * 255) << 0);
  894. let intesity = ldata.energy;
  895. let distance = 0;
  896. var three_light = null;
  897. switch (ldata.type) {
  898. case blender_light_types.point:
  899. var three_light = new THREE.PointLight(color, intesity, distance);
  900. three_light.position.fromArray(pos_array, 0);
  901. three_light.castShadow = true;
  902. break;
  903. case blender_light_types.sun:
  904. var three_light = new THREE.PointLight(color, intesity, distance);
  905. three_light.position.fromArray(pos_array, 0);
  906. three_light.castShadow = true;
  907. three_light.shadow.mapSize.width = 1024;
  908. three_light.shadow.mapSize.height = 1024;
  909. three_light.shadow.camera.near = 0.01;
  910. three_light.shadow.camera.far = 500;
  911. break;
  912. }
  913. return three_light;
  914. }
  915. },{}],6:[function(require,module,exports){
  916. /*jshint esversion: 6 */
  917. module.exports = (() => {
  918. const createTexture = require("./texture.js");
  919. var texture_mappings = {
  920. diff_color: 1,
  921. normal: 2,
  922. mirror: 8,
  923. diff_intensity: 16,
  924. spec_intensity: 32,
  925. emit: 32,
  926. alpha: 128,
  927. spec_hardness: 256,
  928. ray_mirror: 512,
  929. translucency: 1024,
  930. ambient: 2048,
  931. displacement: 4096,
  932. warp: 8192
  933. };
  934. let blender_specular_types = {
  935. cooktorr: 0,
  936. phong: 1,
  937. blinn: 2,
  938. toon: 3,
  939. wardiso: 4
  940. };
  941. function applyColorMapping(blender_texture, three_texture, material) {
  942. if (blender_texture.mapto & texture_mappings.diff_color) {
  943. material.map = three_texture;
  944. }
  945. }
  946. function applySpecMapping(blender_texture, three_texture, material) {
  947. if (blender_texture.mapto & texture_mappings.spec_color && material.type != "MeshStandardMaterial") {
  948. material.specularMap = three_texture;
  949. }
  950. if (blender_texture.mapto & texture_mappings.spec_intensity && material.type != "MeshStandardMaterial") {
  951. material.roughnessMap = three_texture;
  952. }
  953. }
  954. function applyAlphaMapping(blender_texture, three_texture, material) {
  955. if (blender_texture.mapto & texture_mappings.alpha) {
  956. material.alphaMap = three_texture;
  957. }
  958. }
  959. function applyNormalMapping(blender_texture, three_texture, material) {
  960. if (blender_texture.mapto & texture_mappings.normal) {
  961. material.normalMap = three_texture;
  962. material.normalScale = {
  963. x: blender_texture.norfac,
  964. y: blender_texture.norfac
  965. };
  966. }
  967. }
  968. function applyMirrorMapping(blender_texture, three_texture, material) {
  969. if (blender_texture.mapto & texture_mappings.mirror) {
  970. material.envMap = three_texture;
  971. material.envMapIntensity = blender_texture.mirrfac;
  972. }
  973. }
  974. var blender_texture_coordinates = {
  975. GENERATED : 1,
  976. REFLECTION : 2,
  977. NORMAL:4,
  978. GLOBAL : 8,
  979. UV : 16,
  980. OBJECT : 32,
  981. WINDOW: 1024,
  982. TANGENT:4096,
  983. PARTICLE: 8192,
  984. STRESS:16384
  985. }
  986. var blender_texture_mapping = {
  987. FLAT : 0,
  988. CUBE : 1,
  989. TUBE : 2,
  990. SPHERE : 3
  991. }
  992. function applyTexture(blender_texture, material) {
  993. //extract blender_texture data. Use Only if image has been supplied.
  994. if (blender_texture && blender_texture.tex && blender_texture.tex.ima) {
  995. let three_texture = createTexture(blender_texture.tex.ima);
  996. if(blender_texture.texco == blender_texture_coordinates.REFLECTION){
  997. switch(blender_texture.mapping){
  998. case blender_texture_mapping.FLAT:
  999. three_texture.mapping = THREE.EquirectangularReflectionMapping;
  1000. break;
  1001. case blender_texture_mapping.SPHERE:
  1002. three_texture.mapping = THREE.SphericalReflectionMapping;
  1003. break;
  1004. }
  1005. //three_texture.mapping = THREE.EquirectangularRefractionMapping;
  1006. }
  1007. applyColorMapping(blender_texture, three_texture, material);
  1008. applySpecMapping(blender_texture, three_texture, material);
  1009. applyAlphaMapping(blender_texture, three_texture, material);
  1010. applyNormalMapping(blender_texture, three_texture, material);
  1011. applyMirrorMapping(blender_texture, three_texture, material);
  1012. }
  1013. }
  1014. return function createThreeJSMaterial(blend_mat) {
  1015. var material = null;
  1016. var textures = blend_mat.mtex;
  1017. switch (blend_mat.spec_shader) {
  1018. case blender_specular_types.lambert:
  1019. material = new THREE.MeshLambertMaterial();
  1020. material.color.setRGB(blend_mat.r, blend_mat.g, blend_mat.b);
  1021. break;
  1022. case blender_specular_types.blinn:
  1023. case blender_specular_types.phong:
  1024. material = new THREE.MeshStandardMaterial();
  1025. material.color.setRGB(blend_mat.r, blend_mat.g, blend_mat.b);
  1026. //material.specular.setRGB(blend_mat.specr, blend_mat.specg, blend_mat.specb);
  1027. material.roughness = (1 - (blend_mat.har / 512));
  1028. material.metalness = 1 - blend_mat.ref;
  1029. if(blend_mat.alpha < 0.98){
  1030. material.transparent = true;
  1031. material.opacity = blend_mat.alpha;
  1032. console.log(blend_mat, material)
  1033. }
  1034. break;
  1035. case blender_specular_types.wardiso:
  1036. case blender_specular_types.cooktorr:
  1037. material = new THREE.MeshPhongMaterial();
  1038. material.color.setRGB(blend_mat.r, blend_mat.g, blend_mat.b);
  1039. material.specular.setRGB(blend_mat.specr, blend_mat.specg, blend_mat.specb);
  1040. material.shininess = blend_mat.har / 512;
  1041. material.reflectivity = blend_mat.ref * 100;
  1042. break;
  1043. default:
  1044. material = new THREE.MeshLambertMaterial();
  1045. material.color.setRGB(blend_mat.r, blend_mat.g, blend_mat.b);
  1046. break;
  1047. }
  1048. var at = (texture) => applyTexture(texture, material);
  1049. if (textures && textures.length) textures.map(at);
  1050. return material;
  1051. };
  1052. })();
  1053. },{"./texture.js":8}],7:[function(require,module,exports){
  1054. /*jshint esversion: 6 */
  1055. module.exports = function createThreeJSBufferGeometry(blender_mesh, origin) {
  1056. //get materials
  1057. let pick_material = 0,
  1058. mesh = blender_mesh,
  1059. faces = mesh.mpoly,
  1060. loops = mesh.mloop,
  1061. UV = mesh.mloopuv,
  1062. verts = mesh.mvert;
  1063. var geometry = new THREE.BufferGeometry();
  1064. if (!faces) return geometry;
  1065. var index_count = 0;
  1066. //precalculate the size of the array needed for faces
  1067. var face_indice_count = 0;
  1068. var face_indice_counta = 0;
  1069. for (var i = 0; i < faces.length; i++) {
  1070. var face = faces[i] || faces;
  1071. var len = face.totloop;
  1072. var indexi = 1;
  1073. face_indice_counta += (len * 2 / 3) | 0;
  1074. while (indexi < len) {
  1075. face_indice_count += 3;
  1076. indexi += 2;
  1077. }
  1078. }
  1079. //extract face info and dump into array buffer;
  1080. var face_buffer = new Uint32Array(face_indice_count);
  1081. var uv_buffer = new Float32Array(face_indice_count * 2);
  1082. var normal_buffer = new Float32Array(face_indice_count * 3);
  1083. var verts_array_buff = new Float32Array(face_indice_count * 3);
  1084. for (var i = 0; i < faces.length; i++) {
  1085. var face = faces[i] || faces;
  1086. var len = face.totloop;
  1087. var start = face.loopstart;
  1088. var indexi = 1;
  1089. var offset = 0;
  1090. while (indexi < len) {
  1091. var face_normals = [];
  1092. var face_index_array = [];
  1093. var face_uvs = [];
  1094. let index = 0;
  1095. for (var l = 0; l < 3; l++) {
  1096. //Per Vertice
  1097. if ((indexi - 1) + l < len) {
  1098. index = start + (indexi - 1) + l;
  1099. } else {
  1100. index = start;
  1101. }
  1102. var v = loops[index].v;
  1103. var vert = verts[v];
  1104. face_buffer[index_count] = index_count;
  1105. //get normals, which are 16byte ints, and norm them back into floats.
  1106. verts_array_buff[index_count * 3 + 0] = vert.co[0] + origin[0];
  1107. verts_array_buff[index_count * 3 + 1] = vert.co[2] + origin[2];
  1108. verts_array_buff[index_count * 3 + 2] = -vert.co[1] + -origin[1];
  1109. normal_buffer[index_count * 3 + 0] = vert.no[0];
  1110. normal_buffer[index_count * 3 + 1] = vert.no[2];
  1111. normal_buffer[index_count * 3 + 2] = (-vert.no[1]);
  1112. if (UV) {
  1113. var uv = UV[index].uv;
  1114. uv_buffer[index_count * 2 + 0] = uv[0];
  1115. uv_buffer[index_count * 2 + 1] = uv[1];
  1116. }
  1117. index_count++;
  1118. }
  1119. indexi += 2;
  1120. }
  1121. }
  1122. geometry.addAttribute('position', new THREE.BufferAttribute(verts_array_buff, 3));
  1123. geometry.setIndex(new THREE.BufferAttribute(face_buffer, 1));
  1124. geometry.addAttribute('normal', new THREE.BufferAttribute(normal_buffer, 3));
  1125. geometry.addAttribute('uv', new THREE.BufferAttribute(uv_buffer, 2));
  1126. //geometry.blend_mat = materials[pick_material];
  1127. return geometry;
  1128. };
  1129. function createThreeJSGeometry(blender_mesh, origin) {
  1130. //get materials
  1131. var mats = blender_mesh.mat,
  1132. materials = [];
  1133. for (var i = 0; i < mats.length; i++) {
  1134. var material = createThreeJSMaterial(mats[i]);
  1135. materials.push(material);
  1136. }
  1137. let pick_material = 0,
  1138. mesh = blender_mesh,
  1139. faces = mesh.mpoly,
  1140. loops = mesh.mloop,
  1141. UV = mesh.mloopuv,
  1142. verts = mesh.mvert,
  1143. vert_array = [],
  1144. face_array = [],
  1145. uv_array = [],
  1146. normal_array = [];
  1147. var geometry = new THREE.Geometry();
  1148. if (!faces) return geometry;
  1149. var index_count = 0;
  1150. let verts_array_buff = new Float32Array(verts.length * 3);
  1151. for (var i = 0; i < verts.length; i++) {
  1152. let vert = verts[i];
  1153. vert_array.push(new THREE.Vector3(vert.co[0] + origin[0], vert.co[2] + origin[2], -vert.co[1] - origin[1]));
  1154. }
  1155. for (var i = 0; i < faces.length; i++) {
  1156. var face = faces[i] || faces;
  1157. var len = face.totloop;
  1158. var start = face.loopstart;
  1159. var indexi = 1;
  1160. pick_material = face.mat_nr;
  1161. while (indexi < len) {
  1162. var face_normals = [];
  1163. var face_index_array = [];
  1164. var face_uvs = [];
  1165. let index = 0;
  1166. for (var l = 0; l < 3; l++) {
  1167. //Per Vertice
  1168. if ((indexi - 1) + l < len) {
  1169. index = start + (indexi - 1) + l;
  1170. } else {
  1171. index = start;
  1172. }
  1173. var v = loops[index].v;
  1174. var vert = verts[v];
  1175. face_index_array.push(v);
  1176. index_count++;
  1177. //get normals, which are 16byte ints, and norm them back into floats.
  1178. var
  1179. n1 = vert.no[0],
  1180. n2 = vert.no[2],
  1181. n3 = -vert.no[1];
  1182. var nl = 1;
  1183. Math.sqrt((n1 * n1) + (n2 * n2) + (n3 * n3));
  1184. face_normals.push(new THREE.Vector3(n1 / nl, n2 / nl, n3 / nl));
  1185. if (UV) {
  1186. var uv = UV[index].uv;
  1187. face_uvs.push(new THREE.Vector2(uv[0], uv[1]));
  1188. }
  1189. }
  1190. uv_array.push(face_uvs);
  1191. face_array.push(new THREE.Face3(
  1192. face_index_array[0], face_index_array[1], face_index_array[2],
  1193. face_normals
  1194. ));
  1195. indexi += 2;
  1196. }
  1197. }
  1198. geometry.blend_mat = materials[pick_material];
  1199. geometry.vertices = vert_array;
  1200. geometry.faces = face_array;
  1201. if (uv_array.length > 0) {
  1202. geometry.faceVertexUvs = [uv_array];
  1203. }
  1204. geometry.uvsNeedUpdate = true;
  1205. //Well, using blender file normals does not work. Will need to investigate why normals from the blender file do not provide correct results.
  1206. //For now, have Three calculate normals.
  1207. geometry.computeVertexNormals();
  1208. return geometry;
  1209. };
  1210. },{}],8:[function(require,module,exports){
  1211. /*jshint esversion: 6 */
  1212. let blender_texture_cache = {};
  1213. module.exports = function createThreeJSTexture(image) {
  1214. let base64 = require("base64-js");
  1215. let parsed_blend_file = image.__blender_file__;
  1216. let texture = null;
  1217. let name = image.aname;
  1218. if (image.packedfile) {
  1219. if (blender_texture_cache[name]) {
  1220. texture = blender_texture_cache[name];
  1221. } else {
  1222. //get the extension
  1223. let ext = name.split('.').pop();
  1224. let data = image.packedfile;
  1225. let size = data.size;
  1226. let offset = data.data.__data_address__;
  1227. let raw_data = parsed_blend_file.byte.slice(offset, offset + size);
  1228. let encodedData = base64.fromByteArray(raw_data);
  1229. let dataURI;
  1230. switch (ext) {
  1231. case "png":
  1232. dataURI = "data:image/png;base64," + encodedData;
  1233. break;
  1234. case "jpg":
  1235. dataURI = "data:image/jpeg;base64," + encodedData;
  1236. break;
  1237. }
  1238. let img = new Image();
  1239. img.src = dataURI;
  1240. texture = new THREE.Texture(img);
  1241. img.onload = () => {
  1242. texture.needsUpdate = true;
  1243. };
  1244. blender_texture_cache[name] = texture;
  1245. }
  1246. }
  1247. return texture;
  1248. };
  1249. },{"base64-js":1}]},{},[2])
  1250. //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../AppData/Roaming/npm/node_modules/browserify/node_modules/browser-pack/_prelude.js","node_modules/base64-js/index.js","source/main.js","source/parser/parser.js","source/threejs/blend_three.js","source/threejs/light.js","source/threejs/material.js","source/threejs/mesh.js","source/threejs/texture.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvxBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9MA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","'use strict'\n\nexports.byteLength = byteLength\nexports.toByteArray = toByteArray\nexports.fromByteArray = fromByteArray\n\nvar lookup = []\nvar revLookup = []\nvar Arr = typeof Uint8Array !== 'undefined' ? Uint8Array : Array\n\nvar code = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/'\nfor (var i = 0, len = code.length; i < len; ++i) {\n  lookup[i] = code[i]\n  revLookup[code.charCodeAt(i)] = i\n}\n\nrevLookup['-'.charCodeAt(0)] = 62\nrevLookup['_'.charCodeAt(0)] = 63\n\nfunction placeHoldersCount (b64) {\n  var len = b64.length\n  if (len % 4 > 0) {\n    throw new Error('Invalid string. Length must be a multiple of 4')\n  }\n\n  // the number of equal signs (place holders)\n  // if there are two placeholders, than the two characters before it\n  // represent one byte\n  // if there is only one, then the three characters before it represent 2 bytes\n  // this is just a cheap hack to not do indexOf twice\n  return b64[len - 2] === '=' ? 2 : b64[len - 1] === '=' ? 1 : 0\n}\n\nfunction byteLength (b64) {\n  // base64 is 4/3 + up to two characters of the original data\n  return (b64.length * 3 / 4) - placeHoldersCount(b64)\n}\n\nfunction toByteArray (b64) {\n  var i, l, tmp, placeHolders, arr\n  var len = b64.length\n  placeHolders = placeHoldersCount(b64)\n\n  arr = new Arr((len * 3 / 4) - placeHolders)\n\n  // if there are placeholders, only get up to the last complete 4 chars\n  l = placeHolders > 0 ? len - 4 : len\n\n  var L = 0\n\n  for (i = 0; i < l; i += 4) {\n    tmp = (revLookup[b64.charCodeAt(i)] << 18) | (revLookup[b64.charCodeAt(i + 1)] << 12) | (revLookup[b64.charCodeAt(i + 2)] << 6) | revLookup[b64.charCodeAt(i + 3)]\n    arr[L++] = (tmp >> 16) & 0xFF\n    arr[L++] = (tmp >> 8) & 0xFF\n    arr[L++] = tmp & 0xFF\n  }\n\n  if (placeHolders === 2) {\n    tmp = (revLookup[b64.charCodeAt(i)] << 2) | (revLookup[b64.charCodeAt(i + 1)] >> 4)\n    arr[L++] = tmp & 0xFF\n  } else if (placeHolders === 1) {\n    tmp = (revLookup[b64.charCodeAt(i)] << 10) | (revLookup[b64.charCodeAt(i + 1)] << 4) | (revLookup[b64.charCodeAt(i + 2)] >> 2)\n    arr[L++] = (tmp >> 8) & 0xFF\n    arr[L++] = tmp & 0xFF\n  }\n\n  return arr\n}\n\nfunction tripletToBase64 (num) {\n  return lookup[num >> 18 & 0x3F] + lookup[num >> 12 & 0x3F] + lookup[num >> 6 & 0x3F] + lookup[num & 0x3F]\n}\n\nfunction encodeChunk (uint8, start, end) {\n  var tmp\n  var output = []\n  for (var i = start; i < end; i += 3) {\n    tmp = (uint8[i] << 16) + (uint8[i + 1] << 8) + (uint8[i + 2])\n    output.push(tripletToBase64(tmp))\n  }\n  return output.join('')\n}\n\nfunction fromByteArray (uint8) {\n  var tmp\n  var len = uint8.length\n  var extraBytes = len % 3 // if we have 1 byte left, pad 2 bytes\n  var output = ''\n  var parts = []\n  var maxChunkLength = 16383 // must be multiple of 3\n\n  // go through the array every three bytes, we'll deal with trailing stuff later\n  for (var i = 0, len2 = len - extraBytes; i < len2; i += maxChunkLength) {\n    parts.push(encodeChunk(uint8, i, (i + maxChunkLength) > len2 ? len2 : (i + maxChunkLength)))\n  }\n\n  // pad the end with zeros, but make sure to not forget the extra bytes\n  if (extraBytes === 1) {\n    tmp = uint8[len - 1]\n    output += lookup[tmp >> 2]\n    output += lookup[(tmp << 4) & 0x3F]\n    output += '=='\n  } else if (extraBytes === 2) {\n    tmp = (uint8[len - 2] << 8) + (uint8[len - 1])\n    output += lookup[tmp >> 10]\n    output += lookup[(tmp >> 4) & 0x3F]\n    output += lookup[(tmp << 2) & 0x3F]\n    output += '='\n  }\n\n  parts.push(output)\n\n  return parts.join('')\n}\n","\r\n/*jshint esversion: 6 */\r\n\r\nconst three = require(\"./threejs/blend_three.js\");\r\n\r\nconst parser = require(\"./parser/parser.js\")();\r\n\r\n\r\nfunction loadFile(blender_file, res, rej){\t\r\n\tthree_module = three(blender_file);\r\n\r\n\t//TODO: Report any errors with ThreeJS before continuing.\r\n\t\r\n\tres({\r\n\t\tfile : blender_file,\r\n\t\tthree : three_module\r\n\t});\r\n}\r\n\r\n/* This represents a parsed blendfile instance if parsing is successful. It will accept a string or a binary data object. Strings must be a valid URI to a blender file. Binary data may be in the form of an ArrayBuffer, TypedArray, or a Blob. Binary data must also contain the binary data of a blender file.*/\r\n\r\nJSBLEND = (fileuri_or_filedata, name = \"\")=>{\r\n\r\n\tconst promise = new Promise(\r\n\t\t(res, rej) =>{\r\n\t\t\tparser.onParseReady = (blender_file) => {\r\n\t\t\t\tloadFile(blender_file, res, rej);\r\n\t\t\t};\r\n\r\n\t\t\t//If fileuri_or_filedata is a string, attempt to load the file asynchronously\r\n\t\t\tif(typeof fileuri_or_filedata == \"string\"){\r\n\t\t\t\t\r\n\t\t\t\tlet request = new XMLHttpRequest();\r\n\t\t\t    \r\n\t\t\t    request.open(\"GET\", fileuri_or_filedata, true);\r\n\t\t\t    \r\n\t\t\t    request.responseType = 'blob';\r\n\t\t\t    \r\n\t\t\t    request.onload = () => {\r\n\t\t\t        let file = request.response;\r\n\t\t\t        \r\n\t\t\t        parser.loadBlendFromBlob(new Blob([file]), fileuri_or_filedata);\r\n\t\t\t    };\r\n\t\t\t    \r\n\t\t\t    request.send();\r\n\r\n\t\t\t    return;\r\n\t\t\t}\r\n\t\t\tdebugger\r\n\r\n\t\t\tif(typeof fileuri_or_filedata == \"object\"){\r\n\t\t\t\t//Attempt to load from blob or array buffer;\r\n\t\t\t\tif(fileuri_or_filedata instanceof ArrayBuffer){\r\n\t\t\t\t\tparser.loadBlendFromArrayBuffer(fileuri_or_filedata, name);\r\n\t\t\t\t\treturn;\r\n\t\t\t\t}\r\n\r\n\t\t\t\tif(fileuri_or_filedata instanceof Blob){\r\n\t\t\t\t\tparser.loadBlendFromBlob(fileuri_or_filedata, name);\r\n\t\t\t\t\treturn;\r\n\t\t\t\t}\r\n\t\t\t}\r\n\r\n\t\t\t//Unknown file type passed -> abort and reject\r\n\r\n\t\t\tconsole.warn(\"Unsupported file type passed to JSBlend\", fileuri_or_filedata);\r\n\t\t\t\r\n\t\t\trej(\"Unsupported file type passed to JSBlend\");\r\n\t\t}\r\n\t);\r\n\r\n\treturn promise;\r\n};","/*jshint esversion: 6 */\n\nconst DNA1 = 826363460;\nconst ENDB = 1111772741;\n\n/* Note: Blender cooridinates treat the Z axis as the verticle and  Y as depth. */\nmodule.exports  = (function(unzipper) {\n    //A helper object to identify Blender Object structs by type name. \n    var blender_types = {\n        mesh_object: 1,\n        lamp: 10,\n    };\n\n    //web worker not functional in this version\n    USE_WEBWORKER = false;\n\n    var worker = null,\n\n        FR = new FileReader(),\n\n        return_object = {\n            loadBlendFromArrayBuffer: function(array_buffer) {\n                return_object.ready = false;\n                if (USE_WEBWORKER) {\n                    worker.postMessage(array_buffer, array_buffer);\n                } else {\n                    worker.onmessage({\n                        data: array_buffer\n                    });\n                }\n            },\n            loadBlendFromBlob: function(blob) {\n                FR.onload = function() {\n                    return_object.loadBlendFromArrayBuffer(this.result);\n                };\n                FR.readAsArrayBuffer(blob);\n            },\n            ready: true,\n            onParseReady: function() {},\n        };\n\n    worker = new worker_code();\n\n    worker.postMessage = function(message) {\n        return_object.onParseReady(message);\n    };\n\n    function worker_code() {\n        \"use strict\";\n\n        var data = null,\n            _data = null,\n            BIG_ENDIAN = false,\n            pointer_size = 0,\n            struct_names = [],\n            offset = 0,\n            working_blend_file = null,\n            current_SDNA_template = null,\n            templates = {},\n            finished_objects = [],\n            FILE = null,\n            AB = null;\n\n        function parseFile(msg) {\n            var self = this;\n            if (typeof msg.data == \"object\") {\n                // reset global variables\n                AB = null;\n                data = null;\n                BIG_ENDIAN = false;\n                pointer_size = 0;\n                struct_names = [];\n                offset = 0;\n                working_blend_file = null;\n                finished_objects = [];\n                current_SDNA_template = null;\n\n\n                // set data\n                _data = msg.data;\n\n                AB = _data.slice();\n\n                data = new DataView(_data);\n\n\n                FILE = new BLENDER_FILE(AB);\n\n                //start parsing\n                readFile();\n\n                //export parsed data\n                self.postMessage(FILE);\n            }\n        }\n\n        /*\n            Export object for a parsed __blender_file__.\n        */\n\n        var BLENDER_FILE = function(AB) {\n            this.AB = AB;\n            //this.double = new Float64Array(AB);\n            this.byte = new Uint8Array(AB);\n\n            this.dv = new DataView(AB);\n\n            this.objects = {};\n            this.memory_lookup = {},\n                this.object_array = [];\n\n            this.template = null;\n        };\n\n        BLENDER_FILE.prototype = {\n            addObject: function(obj) {\n                this.object_array.push(obj);\n                if (!this.objects[obj.blender_name]) this.objects[obj.blender_name] = [];\n                this.objects[obj.blender_name].push(obj);\n            },\n            primeTypes: function(list_of_dna_names) {\n                for (var i = 0; i < list_of_dna_names.length; i++) {\n                    //this.objects[list_of_dna_names[i]] = [];\n                }\n            },\n            getPointer: function(offset) {\n                var pointerLow = this.dv.getUint32(offset, this.template.endianess);\n                if (this.template.pointer_size > 4) {\n                    var pointerHigh = this.dv.getUint32(offset + 4, this.template.endianess);\n                    if (this.template.endianess) {\n                        return (pointerLow) + \"l|h\" + pointerHigh;\n                    } else {\n                        return (pointerHigh) + \"h|l\" + pointerLow;\n                    }\n                } else {\n                    return pointerLow;\n                }\n            }\n        };\n\n        function getDocument(data) {\n            var obj = readFile(null, data);\n        }\n\n        self.onmessage = parseFile;\n        this.onmessage = parseFile;\n\n        /*\n            These functions map offsets in the blender __blender_file__ to basic types (byte,short,int,float) through TypedArrays;\n            This allows the underlying binary data to be changed.\n        */\n\n        function float64Prop(offset, Blender_Array_Length, length) {\n            return {\n                get: function() {\n                    return (Blender_Array_Length > 1) ?\n                        new Float64Array(this.__blender_file__.AB, this.__data_address__ + offset, length) :\n                        this.__blender_file__.dv.getFloat64(this.__data_address__ + offset, this.__blender_file__.template.endianess);\n                },\n                set: function(float) {\n                    if (Blender_Array_Length > 1) {} else {\n                        this.__blender_file__.dv.setFloat64(this.__data_address__ + offset, float, this.__blender_file__.template.endianess);\n                    }\n                },\n            };\n        }\n\n        function floatProp(offset, Blender_Array_Length, length) {\n            return {\n                get: function() {\n                    return (Blender_Array_Length > 1) ?\n                        new Float32Array(this.__blender_file__.AB, this.__data_address__ + offset, length) :\n                        this.__blender_file__.dv.getFloat32(this.__data_address__ + offset, this.__blender_file__.template.endianess);\n                },\n                set: function(float) {\n                    if (Blender_Array_Length > 1) {} else {\n                        this.__blender_file__.dv.setFloat32(this.__data_address__ + offset, float, this.__blender_file__.template.endianess);\n                    }\n                },\n            };\n        }\n\n        function intProp(offset, Blender_Array_Length, length) {\n            return {\n                get: function() {\n                    return (Blender_Array_Length > 1) ?\n                        new Int32Array(this.__blender_file__.AB, this.__data_address__ + offset, length) :\n                        this.__blender_file__.dv.getInt32(this.__data_address__ + offset, this.__blender_file__.template.endianess);\n                },\n                set: function(int) {\n                    if (Blender_Array_Length > 1) {} else {\n                        this.__blender_file__.dv.setInt32(this.__data_address__ + offset, float, this.__blender_file__.template.endianess);\n                    }\n                },\n            };\n        }\n\n        function uIntProp(offset, Blender_Array_Length, length) {\n            return {\n                get: function() {\n                    return (Blender_Array_Length > 1) ?\n                        new Uint32Array(this.__blender_file__.AB, this.__data_address__ + offset, length) :\n                        this.__blender_file__.dv.getUint32(this.__data_address__ + offset, this.__blender_file__.template.endianess);\n                },\n                set: function(int) {\n                    if (Blender_Array_Length > 1) {} else {\n                        this.__blender_file__.dv.setUint32(this.__data_address__ + offset, float, this.__blender_file__.template.endianess);\n                    }\n                },\n            };\n        }\n\n        function shortProp(offset, Blender_Array_Length, length) {\n            return {\n                get: function() {\n                    return (Blender_Array_Length > 1) ?\n                        new Int16Array(this.__blender_file__.AB, this.__data_address__ + offset, length) :\n                        this.__blender_file__.dv.getInt16(this.__data_address__ + offset, this.__blender_file__.template.endianess);\n                },\n                set: function(float) {\n                    if (Blender_Array_Length > 1) {} else {\n                        this.__blender_file__.dv.setInt16(this.__data_address__ + offset, float, this.__blender_file__.template.endianess);\n                    }\n                },\n            };\n        }\n\n        var uShortProp = (offset, Blender_Array_Length, length) => {\n            return {\n                get: function() {\n                    return (Blender_Array_Length > 1) ?\n                        new Uint16Array(this.__blender_file__.AB, this.__data_address__ + offset, length) :\n                        this.__blender_file__.dv.getUint16(this.__data_address__ + offset, this.__blender_file__.template.endianess);\n                },\n                set: function(float) {\n                    if (Blender_Array_Length > 1) {} else {\n                        this.__blender_file__.dv.setUint16(this.__data_address__ + offset, float, this.__blender_file__.template.endianess);\n                    }\n                },\n            };\n        }\n\n\n        function charProp(offset, Blender_Array_Length, length) {\n            return {\n                get: function() {\n                    if (Blender_Array_Length > 1) {\n                        let start = this.__data_address__ + offset;\n                        let end = start;\n                        let buffer_guard = 0;\n\n                        while (this.__blender_file__.byte[end] != 0 && buffer_guard++ < length) end++;\n\n                        return toString(this.__blender_file__.AB, start, end)\n                    }\n                    return this.__blender_file__.byte[(this.__data_address__ + offset)];\n                },\n                set: function(byte) {\n                    if (Blender_Array_Length > 1) {\n                        var string = byte + \"\",\n                            i = 0,\n                            l = string.length;\n                        while (i < length) {\n                            if (i < l) {\n                                this.__blender_file__.byte[(this.__data_address__ + offset + i)] = string.charCodeAt(i) | 0;\n                            } else {\n                                this.__blender_file__.byte[(this.__data_address__ + offset + i)] = 0;\n                            }\n                            i++;\n                        }\n                    } else {\n                        this.__blender_file__.byte[(this.__data_address__ + offset)] = byte | 0;\n                    }\n                }\n            };\n        }\n\n        function pointerProp2(offset) {\n            return {\n                get: function() {\n                    let pointer = this.__blender_file__.getPointer(this.__data_address__ + offset, this.__blender_file__);\n                    var link = this.__blender_file__.memory_lookup[pointer];\n\n                    var results = [];\n\n                    if (link) {\n                        var address = link.__data_address__;\n                        let j = 0;\n                        while (true) {\n                            pointer = this.__blender_file__.getPointer(address + j * 8, this.__blender_file__);\n                            let obj = this.__blender_file__.memory_lookup[pointer];\n                            if (!obj) break;\n                            results.push(obj);\n                            j++\n                        }\n\n                    };\n\n                    return results;\n                },\n                set: function() {}\n            }\n        }\n\n        function pointerProp(offset, Blender_Array_Length, length) {\n            return {\n                get: function() {\n                    if (Blender_Array_Length > 1) {\n                        let array = [];\n                        let j = 0;\n                        let off = offset;\n                        while (j < Blender_Array_Length) {\n                            let pointer = this.__blender_file__.getPointer(this.__data_address__ + off, this.__blender_file__);\n\n                            array.push(this.__blender_file__.memory_lookup[pointer]);\n                            off += length ///this.__blender_file__.template.pointer_size;\n                            j++;\n                        }\n\n                        return array;\n                    } else {\n                        let pointer = this.__blender_file__.getPointer(this.__data_address__ + offset, this.__blender_file__);\n                        return this.__blender_file__.memory_lookup[pointer];\n                    }\n                },\n                set: function() {}\n            }\n        }\n\n        function compileProp(obj, name, type, offset, array_size, IS_POINTER, pointer_size, length) {\n\n            if (!IS_POINTER) {\n                switch (type) {\n                    case \"double\":\n                        Object.defineProperty(obj, name, float64Prop(offset, array_size, length >> 3));\n                        break;\n                    case \"float\":\n                        Object.defineProperty(obj, name, floatProp(offset, array_size, length >> 2));\n                        break;\n                    case \"int\":\n                        Object.defineProperty(obj, name, intProp(offset, array_size, length >> 2));\n                        break;\n                    case \"short\":\n                    case \"ushort\":\n                        Object.defineProperty(obj, name, shortProp(offset, array_size, length >> 1));\n                        break;\n                    case \"char\":\n                    case \"uchar\":\n                        Object.defineProperty(obj, name, charProp(offset, array_size, length));\n                        break;\n                    default:\n                        //compile list to \n                        obj[name] = {};\n                        obj.__list__.push(name, type, length, offset, array_size, IS_POINTER);\n                }\n                obj._length += length;\n                offset += length;\n            } else {\n                Object.defineProperty(obj, name, pointerProp(offset, array_size, pointer_size));\n                offset += pointer_size * array_size;\n            }\n\n            return offset;\n        }\n\n        //Store final DNA structs\n        var MASTER_SDNA_SCHEMA = function(version) {\n            this.version = version;\n            this.SDNA_SET = false;\n            this.byte_size = 0;\n            this.struct_index = 0;\n            this.structs = {};\n            this.SDNA = {};\n            this.endianess = false;\n        };\n\n        MASTER_SDNA_SCHEMA.prototype = {\n            getSDNAStructureConstructor: function(name, struct) {\n                if (struct) {\n                    var blen_struct = Function(\"function \" + name + \"(){}; return \" + name)();\n\n                    blen_struct.prototype = new BLENDER_STRUCTURE();\n                    blen_struct.prototype.blender_name = name;\n                    blen_struct.prototype.__pointers = [];\n                    blen_struct.prototype.__list__ = [];\n\n                    var offset = 0;\n                    //Create properties of struct\n                    for (var i = 0; i < struct.length; i += 3) {\n                        var _name = struct[i],\n                            n = _name,\n                            type = struct[i + 1],\n                            length = struct[i + 2],\n                            array_length = 0,\n                            match = null,\n                            Blender_Array_Length = 1,\n                            Suparray_match = 1,\n                            PointerToArray = false,\n                            Pointer_Match = 0;\n                        var DNA = this.SDNA[name] = {\n                            constructor: blen_struct\n                        };\n\n\n                        let original_name = _name;\n\n                        //mini type parser\n                        if ((match = _name.match(/(\\*?)(\\*?)(\\w+)(\\[(\\w*)\\])?(\\[(\\w*)\\])?/))) {\n\n                            //base name\n                            _name = match[3];\n\n                            //pointer type\n                            if (match[1]) {\n                                Pointer_Match = 10;\n                                blen_struct.prototype.__pointers.push(_name);\n                            }\n\n                            if (match[2]) {\n                                PointerToArray = true;\n                            }\n\n                            //arrays\n                            if (match[4]) {\n                                if (match[6]) {\n                                    Suparray_match = parseInt(match[5]);\n                                    Blender_Array_Length = parseInt(match[7]);\n                                } else {\n                                    Blender_Array_Length = parseInt(match[5]);\n                                }\n                            }\n                            array_length = Blender_Array_Length * length;\n                            length = array_length * Suparray_match;\n                        }\n\n                        DNA[n] = {\n                            type: type,\n                            length: length,\n                            isArray: (Blender_Array_Length > 0),\n                        };\n\n                        if (PointerToArray) {\n                            Object.defineProperty(blen_struct.prototype, _name, pointerProp2(offset));\n                            offset += pointer_size;\n                        } else if (Suparray_match > 1) {\n                            var array_names = new Array(Suparray_match);\n\n                            //construct sub_array object that will return the correct structs\n                            for (var j = 0; j < Suparray_match; j++) {\n                                let array_name_ = `__${_name}[${j}]__`;\n                                array_names[j] = array_name_;\n\n                                offset = compileProp(blen_struct.prototype, array_name_, type, offset, Blender_Array_Length, Pointer_Match, pointer_size, array_length);\n                            }\n\n                            Object.defineProperty(blen_struct.prototype, _name, {\n                                get: (function(array_names) {\n                                    return function() {\n                                        var array = [];\n                                        for (var i = 0; i < array_names.length; i++) {\n                                            array.push(this[array_names[i]])\n                                        }\n                                        return array;\n                                    }\n                                })(array_names)\n                            });\n                        } else {\n                            offset = compileProp(blen_struct.prototype, _name, type, offset, Blender_Array_Length, Pointer_Match, pointer_size, length);\n                        }\n                    }\n\n                    return this.SDNA[name].constructor;\n\n                } else {\n                    if (!this.SDNA[name]) {\n                        return null;\n                    }\n                    return this.SDNA[name].constructor;\n                }\n            }\n        };\n\n        var BLENDER_STRUCTURE = function() {\n            this.__blender_file__ = null;\n            this.__list__ = null;\n            this.__super_array_list__ = null;\n            this.blender_name = \"\";\n            this.__pointers = null;\n            this.address = null;\n            this.length = 0;\n            this.__data_address__ = 0;\n            this.blender_name = \"\";\n            this._length = 0;\n        };\n\n\n        /*\n            Returns a pre-constructed BLENDER_STRUCTURE or creates a new BLENDER_STRUCTURE to match the DNA struct type\n        */\n        var pointer_function = (pointer) => () => {\n            return FILE.memory_lookup[pointer]\n        };\n\n        function getPointer(offset) {\n            var pointerLow = data.getUint32(offset, BIG_ENDIAN);\n            if (pointer_size > 4) {\n                var pointerHigh = data.getUint32(offset + 4, BIG_ENDIAN);\n\n                if (BIG_ENDIAN) {\n                    return (pointerLow) + \"\" + pointerHigh;\n                } else {\n                    return (pointerHigh) + \"\" + pointerLow;\n                }\n            } else {\n                return pointerLow;\n            }\n        }\n\n        BLENDER_STRUCTURE.prototype = {\n            setData: function(pointer, _data_offset, data_block_length, BLENDER_FILE) {\n                if (this.__list__ == null) return this;\n                BLENDER_FILE.addObject(this);\n\n                this.__blender_file__ = BLENDER_FILE;\n\n                var struct = this.__list__,\n                    j = 0,\n                    i = 0,\n                    obj, name = \"\",\n                    type, length, Blender_Array_Length, Pointer_Match, offset, constructor;\n\n                this.__data_address__ = _data_offset;\n\n                if (struct === null) return this;\n\n                for (i = 0; i < struct.length; i += 6) {\n                    obj = null;\n                    name = struct[i];\n                    type = struct[i + 1];\n                    Blender_Array_Length = struct[i + 4];\n                    Pointer_Match = struct[i + 5];\n                    offset = this.__data_address__ + struct[i + 3];\n\n                    if (Blender_Array_Length > 1) {\n                        this[name] = [];\n                        j = 0;\n                        while (j < Blender_Array_Length) {\n                            if (current_SDNA_template.getSDNAStructureConstructor(type)) {\n                                constructor = current_SDNA_template.getSDNAStructureConstructor(type);\n                                this[name].push((new constructor()).setData(0, offset, offset + length / Blender_Array_Length, BLENDER_FILE));\n                            } else this[name].push(null);\n                            offset += length / Blender_Array_Length;\n                            j++;\n                        }\n                    } else {\n                        if (current_SDNA_template.getSDNAStructureConstructor(type)) {\n                            constructor = current_SDNA_template.getSDNAStructureConstructor(type);\n                            this[name] = (new constructor()).setData(0, offset, length + offset, BLENDER_FILE);\n                        } else this[name] = null;\n                    }\n                }\n                //break connection to configuration list\n                this.__list__ = null;\n                return this;\n            },\n\n            get aname() {\n                if (this.id) return this.id.name.slice(2);\n                else return undefined;\n            }\n        };\n\n        function toString(buffer, _in, _out) {\n            return String.fromCharCode.apply(String, new Uint8Array(buffer, _in, _out - _in));\n        }\n\n        //Begin parsing blender __blender_file__\n        function readFile() {\n            var count = 0;\n            var offset2 = 0;\n            var root = 0;\n            var i = 0;\n            var data_offset = 0;\n            var sdna_index = 0;\n            var code = \"\";\n            var block_length = 0;\n            var curr_count = 0;\n            var curr_count2 = 0;\n\n            FILE.memory_lookup = {};\n            struct_names = [];\n            offset = 0;\n\n            // Make sure we have a .blend __blender_file__. All blend files have the first 12bytes\n            // set with BLENDER-v### in Utf-8\n            if (toString(_data, offset, 7) !== \"BLENDER\") return console.warn(\"File supplied is not a .blend compatible Blender file.\");\n\n            // otherwise get templete from save version.\n\n            offset += 7;\n            pointer_size = ((toString(_data, offset++, offset)) == \"_\") ? 4 : 8;\n            BIG_ENDIAN = toString(_data, offset++, offset) !== \"V\";\n            var version = toString(_data, offset, offset + 3);\n\n\n            //create new master template if none exist for current blender version;\n            if (!templates[version]) {\n                templates[version] = new MASTER_SDNA_SCHEMA(version);\n            }\n\n            current_SDNA_template = templates[version];\n\n            FILE.template = current_SDNA_template;\n\n            offset += 3;\n\n            //Set SDNA structs if template hasn't been set.\n            //Todo: Move the following block into the MASTER_SDNA_SCHEMA object.\n            //*Like so:*/ current_SDNA_template.set(AB);\n\n            if (!current_SDNA_template.SDNA_SET) {\n                current_SDNA_template.endianess = BIG_ENDIAN;\n                current_SDNA_template.pointer_size = pointer_size;\n                //find DNA1 data block\n                offset2 = offset;\n\n                while (true) {\n                    sdna_index = data.getInt32(offset2 + pointer_size + 8, BIG_ENDIAN);\n                    code = toString(_data, offset2, offset2 + 4).replace(/\\u0000/g, \"\");\n                    block_length = data.getInt32(offset2 + 4, true);\n                    offset2 += 16 + (pointer_size);\n                    if (code === \"DNA1\") {\n                        // DNA found; This is the core of the __blender_file__ and contains all the structure for the various data types used in Blender.\n                        count = 0;\n                        var types = [],\n                            fields = [],\n                            names = [],\n                            lengths = [],\n                            name = \"\",\n                            curr_name = \"\";\n\n                        //skip SDNA and NAME identifiers\n                        offset2 += 8;\n\n                        //Number of structs.\n                        count = data.getInt32(offset2, true);\n                        offset2 += 4;\n\n                        curr_count = 0;\n\n                        //Build up list of names for structs\n                        while (curr_count < count) {\n                            curr_name = \"\";\n                            while (data.getInt8(offset2) !== 0) {\n                                curr_name += toString(_data, offset2, offset2 + 1);\n                                offset2++;\n                            }\n                            names.push(curr_name);\n                            offset2++;\n                            curr_count++;\n                        }\n\n\n                        //Adjust for 4byte alignment\n                        if ((offset2 % 4) > 0) offset2 = (4 - (offset2 % 4)) + offset2;\n                        offset2 += 4;\n\n                        //Number of struct types\n                        count = data.getInt32(offset2, true);\n                        offset2 += 4;\n                        curr_count = 0;\n\n                        //Build up list of types\n                        while (curr_count < count) {\n                            curr_name = \"\";\n                            while (data.getInt8(offset2) !== 0) {\n                                curr_name += toString(_data, offset2, offset2 + 1);\n                                offset2++;\n                            }\n                            types.push(curr_name);\n                            offset2++;\n                            curr_count++;\n                        }\n\n                        //Adjust for 4byte alignment\n                        if ((offset2 % 4) > 0) offset2 = (4 - (offset2 % 4)) + offset2;\n                        offset2 += 4;\n                        curr_count = 0;\n\n                        //Build up list of byte lengths for types\n                        while (curr_count < count) {\n                            lengths.push(data.getInt16(offset2, BIG_ENDIAN));\n                            offset2 += 2;\n                            curr_count++;\n                        }\n\n                        //Adjust for 4byte alignment\n                        if ((offset2 % 4) > 0) offset2 = (4 - (offset2 % 4)) + offset2;\n                        offset2 += 4;\n\n                        //Number of structures\n                        var structure_count = data.getInt32(offset2, BIG_ENDIAN);\n                        offset2 += 4;\n                        curr_count = 0;\n\n                        //Create constructor objects from list of SDNA structs\n                        while (curr_count < structure_count) {\n                            var struct_name = types[data.getInt16(offset2, BIG_ENDIAN)];\n                            offset2 += 2;\n                            obj = [];\n                            count = data.getInt16(offset2, BIG_ENDIAN);\n                            offset2 += 2;\n                            curr_count2 = 0;\n                            struct_names.push(struct_name);\n\n                            //Fill an array with name, type, and length for each SDNA struct property\n                            while (curr_count2 < count) {\n                                obj.push(names[data.getInt16(offset2 + 2, BIG_ENDIAN)], types[data.getInt16(offset2, BIG_ENDIAN)], lengths[data.getInt16(offset2, BIG_ENDIAN)]);\n                                offset2 += 4;\n                                curr_count2++;\n                            }\n\n                            //Create a SDNA constructor by passing [type,name,lenth] array as second argument\n                            current_SDNA_template.getSDNAStructureConstructor(struct_name, obj);\n                            curr_count++;\n                        }\n                        current_SDNA_template.SDNA_SET = true;\n                        current_SDNA_template.SDNA_NAMES = struct_names;\n                        break;\n                    }\n                    offset2 += block_length;\n                }\n            }\n\n            //parse the rest of the data, starting back at the top.\n            //TODO: turn into \"on-demand\" parsing.\n\n            while (true) {\n                if ((offset % 4) > 0) {\n                    offset = (4 - (offset % 4)) + offset;\n                }\n\n                data_offset = offset;\n                sdna_index = data.getInt32(offset + pointer_size + 8, BIG_ENDIAN);\n                let code_uint = data.getUint32(offset, BIG_ENDIAN);\n                offset2 = offset + 16 + (pointer_size);\n                offset += data.getInt32(offset + 4, true) + 16 + (pointer_size);\n\n                if (code_uint === DNA1); //skip - already processed at this point    \n                else if (code_uint === ENDB) break; //end of __blender_file__ found\n                else {\n                    //Create a Blender object using a constructor template from current_SDNA_template\n                    var data_start = data_offset + pointer_size + 16;\n\n                    //Get a SDNA constructor by name;\n                    var constructor = current_SDNA_template.getSDNAStructureConstructor(current_SDNA_template.SDNA_NAMES[sdna_index]);\n\n                    var size = data.getInt32(data_offset + 4, BIG_ENDIAN);\n\n                    count = data.getInt32(data_offset + 12 + pointer_size, BIG_ENDIAN);\n\n                    if (count > 0) {\n                        var obj = new constructor();\n\n                        var length = constructor.prototype._length;\n\n\n                        var address = FILE.getPointer(data_offset + 8);\n\n                        obj.address = address + \"\";\n\n                        obj.setData(address, data_start, data_start + size, FILE);\n\n                        if (count > 1) {\n                            let array = [];\n                            array.push(obj);\n                            for (var u = 1; u < count; u++) {\n                                obj = new constructor();\n                                obj.setData(address, data_start + length * u, data_start + (length * u) + length, FILE);\n                                array.push(obj);\n                            }\n                            FILE.memory_lookup[address] = array;\n                        } else {\n                            FILE.memory_lookup[address] = obj;\n                        }\n                    }\n                }\n            }\n        }\n    }\n    return return_object;\n});","/*jshint esversion: 6 */\r\n\r\nconst createMaterial = require(\"./material.js\");\r\nconst createTexture = require(\"./texture.js\");\r\nconst createMesh = require(\"./mesh.js\");\r\nconst createLight = require(\"./light.js\");\r\n\r\nfunction loadModel(three_scene, model_name, blender_file, cache) {\r\n\tvar mats = blender_mesh.mat,\r\n\t\tmaterials = [];\r\n\tfor (var i = 0; i < mats.length; i++) {\r\n\t\tvar material = createThreeJSMaterial(mats[i]);\r\n\t\tmaterials.push(material);\r\n\t}\r\n}\r\n\r\nvar blender_types = {\r\n\tmesh_object: 1,\r\n\tlamp: 10\r\n};\r\n\r\nfunction loadScene(three_scene, blender_file, cache) {\r\n\t//build object from blender mesh object\r\n\tfor (let i = 0; i < blender_file.objects.Object.length; i++) {\r\n\r\n\t\tlet obj = blender_file.objects.Object[i];\r\n\r\n\t\t//Load Lights\r\n\r\n\t\tif (obj.type == blender_types.lamp) {\r\n\r\n\t\t\tlet light = createLight(obj, blender_file);\r\n\r\n\t\t\tthree_scene.add(light);\r\n\t\t}\r\n\r\n\t\t//Load Meshes\r\n\r\n\t\tif (obj.type == blender_types.mesh_object) {\r\n\t\t\tif (obj.data) {\r\n\t\t\t\t//get the mesh \r\n\t\t\t\tvar buffered_geometry = createMesh(obj.data, [0, 0, 0]);\r\n\t\t\t\t\t\r\n\t\t\t\tvar blend_material = obj.data.mat[0];\r\n\t\t\t\t\r\n\t\t\t\tif(blend_material){\r\n\t\t\t\t\tvar material = createMaterial(blend_material);\r\n\t\t\t\t}else{\r\n\t\t\t\t\t//create generic material\r\n\t\t\t\t}\r\n\r\n\t\t\t\t//var geometry = createThreeJSGeometry(obj.data, [0, 0, 0]);\r\n\t\t\t\t///*\r\n\t\t\t\t//create a transform from the mesh object\r\n\t\t\t\tvar mesh = new THREE.Mesh(buffered_geometry, material);\r\n\r\n\t\t\t\tmesh.castShadow = true;\r\n\t\t\t\tmesh.receiveShadow = true;\r\n\r\n\t\t\t\tthree_scene.add(mesh);\r\n\r\n\t\t\t\tmesh.rotateZ(obj.rot[2]);\r\n\t\t\t\tmesh.rotateY(obj.rot[1]);\r\n\t\t\t\tmesh.rotateX(obj.rot[0]);\r\n\t\t\t\tmesh.scale.fromArray(obj.size, 0);\r\n\t\t\t\tmesh.position.fromArray([obj.loc[0], (obj.loc[2]), (-obj.loc[1])], 0);\r\n\t\t\t\t//*/\r\n\t\t\t}\r\n\t\t}\r\n\t}\r\n}\r\n\r\nmodule.exports = (blender_file) => {\r\n\r\n\tif (!THREE) {\r\n\t\tconsole.warn(\"No ThreeJS object detected\");\r\n\t\treturn {};\r\n\t}\r\n\r\n\tvar cache = {};\r\n\r\n\treturn {\r\n\t\tloadScene: (three_scene) => loadScene(three_scene, blender_file, cache),\r\n\t\tloadModel: (model_name) => loadModel(model_name, blender_file, cache)\r\n\t};\r\n};","/*jshint esversion: 6 */\r\n\r\nvar blender_light_types = {\r\n\tpoint: 0,\r\n\tsun: 1,\r\n\tspot: 0,\r\n\themi: 0,\r\n\tarea: 0\r\n};\r\n\r\nmodule.exports = function createThreeJSLamp(blend_lamp) {\r\n\r\n\tlet ldata = blend_lamp.data;\r\n\r\n\tlet pos_array = [blend_lamp.loc[0], blend_lamp.loc[2], -blend_lamp.loc[1]];\r\n\r\n\tlet color = ((ldata.r * 255) << 16) | ((ldata.g * 255) << 8) | ((ldata.b * 255) << 0);\r\n\tlet intesity = ldata.energy;\r\n\tlet distance = 0;\r\n\r\n\tvar three_light = null;\r\n\r\n\tswitch (ldata.type) {\r\n\t\tcase blender_light_types.point:\r\n\t\t\tvar three_light = new THREE.PointLight(color, intesity, distance);\r\n\t\t\tthree_light.position.fromArray(pos_array, 0);\r\n\t\t\tthree_light.castShadow = true;\r\n\t\t\tbreak;\r\n\t\tcase blender_light_types.sun:\r\n\t\t\tvar three_light = new THREE.PointLight(color, intesity, distance);\r\n\t\t\tthree_light.position.fromArray(pos_array, 0);\r\n\t\t\tthree_light.castShadow = true;\r\n\t\t\tthree_light.shadow.mapSize.width = 1024;\r\n\t\t\tthree_light.shadow.mapSize.height = 1024;\r\n\t\t\tthree_light.shadow.camera.near = 0.01;\r\n\t\t\tthree_light.shadow.camera.far = 500;\r\n\t\t\tbreak;\r\n\t}\r\n\r\n\treturn three_light;\r\n}","/*jshint esversion: 6 */\r\n\r\nmodule.exports = (() => {\r\n    const createTexture = require(\"./texture.js\");\r\n\r\n    var texture_mappings = {\r\n        diff_color: 1,\r\n        normal: 2,\r\n        mirror: 8,\r\n        diff_intensity: 16,\r\n        spec_intensity: 32,\r\n        emit: 32,\r\n        alpha: 128,\r\n        spec_hardness: 256,\r\n        ray_mirror: 512,\r\n        translucency: 1024,\r\n        ambient: 2048,\r\n        displacement: 4096,\r\n        warp: 8192\r\n    };\r\n\r\n    let blender_specular_types = {\r\n        cooktorr: 0,\r\n        phong: 1,\r\n        blinn: 2,\r\n        toon: 3,\r\n        wardiso: 4\r\n    };\r\n\r\n    function applyColorMapping(blender_texture, three_texture, material) {\r\n        if (blender_texture.mapto & texture_mappings.diff_color) {\r\n            material.map = three_texture;\r\n        }\r\n    }\r\n\r\n    function applySpecMapping(blender_texture, three_texture, material) {\r\n        if (blender_texture.mapto & texture_mappings.spec_color && material.type != \"MeshStandardMaterial\") {\r\n            material.specularMap = three_texture;\r\n        }\r\n\r\n        if (blender_texture.mapto & texture_mappings.spec_intensity && material.type != \"MeshStandardMaterial\") {\r\n            material.roughnessMap = three_texture;\r\n        }\r\n    }\r\n\r\n    function applyAlphaMapping(blender_texture, three_texture, material) {\r\n        if (blender_texture.mapto & texture_mappings.alpha) {\r\n            material.alphaMap = three_texture;\r\n        }\r\n    }\r\n\r\n    function applyNormalMapping(blender_texture, three_texture, material) {\r\n        if (blender_texture.mapto & texture_mappings.normal) {\r\n            material.normalMap = three_texture;\r\n            material.normalScale = {\r\n                x: blender_texture.norfac,\r\n                y: blender_texture.norfac\r\n            };\r\n        }\r\n    }\r\n\r\n    function applyMirrorMapping(blender_texture, three_texture, material) {\r\n        if (blender_texture.mapto & texture_mappings.mirror) {\r\n            material.envMap = three_texture;\r\n            material.envMapIntensity = blender_texture.mirrfac;\r\n        }\r\n    }\r\n\r\n    var blender_texture_coordinates = {\r\n        GENERATED : 1,\r\n        REFLECTION : 2,\r\n        NORMAL:4,\r\n        GLOBAL : 8,\r\n        UV : 16,\r\n        OBJECT : 32,\r\n        WINDOW: 1024,\r\n        TANGENT:4096,\r\n        PARTICLE: 8192,\r\n        STRESS:16384\r\n    }\r\n\r\n    var blender_texture_mapping = {\r\n        FLAT : 0,\r\n        CUBE : 1,\r\n        TUBE : 2,\r\n        SPHERE : 3\r\n    }\r\n\r\n    function applyTexture(blender_texture, material) {\r\n        //extract blender_texture data. Use Only if image has been supplied.\r\n        if (blender_texture && blender_texture.tex && blender_texture.tex.ima) {\r\n\r\n            let three_texture = createTexture(blender_texture.tex.ima);\r\n\r\n            if(blender_texture.texco == blender_texture_coordinates.REFLECTION){\r\n                switch(blender_texture.mapping){\r\n                    case blender_texture_mapping.FLAT:\r\n                        three_texture.mapping = THREE.EquirectangularReflectionMapping;\r\n                    break;\r\n                    case blender_texture_mapping.SPHERE:\r\n                        three_texture.mapping = THREE.SphericalReflectionMapping;\r\n                    break;\r\n                }\r\n                 //three_texture.mapping = THREE.EquirectangularRefractionMapping;\r\n            }\r\n            \r\n            applyColorMapping(blender_texture, three_texture, material);\r\n            \r\n            applySpecMapping(blender_texture, three_texture, material);\r\n            \r\n            applyAlphaMapping(blender_texture, three_texture, material);\r\n            \r\n            applyNormalMapping(blender_texture, three_texture, material);\r\n\r\n            applyMirrorMapping(blender_texture, three_texture, material);\r\n        }\r\n    }\r\n\r\n    return function createThreeJSMaterial(blend_mat) {\r\n\r\n        var material = null;\r\n\r\n        var textures = blend_mat.mtex;\r\n\r\n        switch (blend_mat.spec_shader) {\r\n            case blender_specular_types.lambert:\r\n                material = new THREE.MeshLambertMaterial();\r\n                material.color.setRGB(blend_mat.r, blend_mat.g, blend_mat.b);\r\n                break;\r\n            case blender_specular_types.blinn:\r\n            case blender_specular_types.phong:\r\n\r\n                material = new THREE.MeshStandardMaterial();\r\n                material.color.setRGB(blend_mat.r, blend_mat.g, blend_mat.b);\r\n                //material.specular.setRGB(blend_mat.specr, blend_mat.specg, blend_mat.specb);\r\n                material.roughness = (1 - (blend_mat.har / 512));\r\n                material.metalness = 1 - blend_mat.ref;\r\n                if(blend_mat.alpha < 0.98){\r\n                    material.transparent = true;\r\n                    material.opacity = blend_mat.alpha;\r\n                    console.log(blend_mat, material)\r\n                }\r\n                break;\r\n            case blender_specular_types.wardiso:\r\n            case blender_specular_types.cooktorr:\r\n                material = new THREE.MeshPhongMaterial();\r\n                material.color.setRGB(blend_mat.r, blend_mat.g, blend_mat.b);\r\n                material.specular.setRGB(blend_mat.specr, blend_mat.specg, blend_mat.specb);\r\n                material.shininess = blend_mat.har / 512;\r\n                material.reflectivity = blend_mat.ref * 100;\r\n                break;\r\n            default:\r\n                material = new THREE.MeshLambertMaterial();\r\n                material.color.setRGB(blend_mat.r, blend_mat.g, blend_mat.b);\r\n                break;\r\n        }\r\n\r\n        var at = (texture) => applyTexture(texture, material);\r\n\r\n\r\n        if (textures && textures.length) textures.map(at);\r\n\r\n        return material;\r\n    };\r\n})();","/*jshint esversion: 6 */\r\nmodule.exports = function createThreeJSBufferGeometry(blender_mesh, origin) {\r\n    //get materials\r\n    let pick_material = 0,\r\n        mesh = blender_mesh,\r\n        faces = mesh.mpoly,\r\n        loops = mesh.mloop,\r\n        UV = mesh.mloopuv,\r\n        verts = mesh.mvert;\r\n\r\n    var geometry = new THREE.BufferGeometry();\r\n\r\n    if (!faces) return geometry;\r\n\r\n    var index_count = 0;\r\n\r\n    //precalculate the size of the array needed for faces\r\n    var face_indice_count = 0;\r\n    var face_indice_counta = 0;\r\n\r\n    for (var i = 0; i < faces.length; i++) {\r\n        var face = faces[i] || faces;\r\n        var len = face.totloop;\r\n        var indexi = 1;\r\n\r\n        face_indice_counta += (len * 2 / 3) | 0;\r\n\r\n        while (indexi < len) {\r\n            face_indice_count += 3;\r\n            indexi += 2;\r\n        }\r\n    }\r\n\r\n    //extract face info and dump into array buffer;\r\n    var face_buffer = new Uint32Array(face_indice_count);\r\n    var uv_buffer = new Float32Array(face_indice_count * 2);\r\n    var normal_buffer = new Float32Array(face_indice_count * 3);\r\n    var verts_array_buff = new Float32Array(face_indice_count * 3);\r\n\r\n    for (var i = 0; i < faces.length; i++) {\r\n        var face = faces[i] || faces;\r\n        var len = face.totloop;\r\n        var start = face.loopstart;\r\n        var indexi = 1;\r\n        var offset = 0;\r\n\r\n        while (indexi < len) {\r\n            var face_normals = [];\r\n            var face_index_array = [];\r\n            var face_uvs = [];\r\n\r\n            let index = 0;\r\n\r\n            for (var l = 0; l < 3; l++) {\r\n                //Per Vertice \r\n\r\n                if ((indexi - 1) + l < len) {\r\n                    index = start + (indexi - 1) + l;\r\n                } else {\r\n                    index = start;\r\n                }\r\n\r\n                var v = loops[index].v;\r\n                var vert = verts[v];\r\n                face_buffer[index_count] = index_count;\r\n                //get normals, which are 16byte ints, and norm them back into floats.\r\n\r\n                verts_array_buff[index_count * 3 + 0] = vert.co[0] + origin[0];\r\n                verts_array_buff[index_count * 3 + 1] = vert.co[2] + origin[2];\r\n                verts_array_buff[index_count * 3 + 2] = -vert.co[1] + -origin[1];\r\n\r\n                normal_buffer[index_count * 3 + 0] = vert.no[0];\r\n                normal_buffer[index_count * 3 + 1] = vert.no[2];\r\n                normal_buffer[index_count * 3 + 2] = (-vert.no[1]);\r\n\r\n\r\n                if (UV) {\r\n                    var uv = UV[index].uv;\r\n                    uv_buffer[index_count * 2 + 0] = uv[0];\r\n                    uv_buffer[index_count * 2 + 1] = uv[1];\r\n                }\r\n\r\n                index_count++;\r\n            }\r\n\r\n            indexi += 2;\r\n        }\r\n    }\r\n\r\n    geometry.addAttribute('position', new THREE.BufferAttribute(verts_array_buff, 3));\r\n    geometry.setIndex(new THREE.BufferAttribute(face_buffer, 1));\r\n    geometry.addAttribute('normal', new THREE.BufferAttribute(normal_buffer, 3));\r\n    geometry.addAttribute('uv', new THREE.BufferAttribute(uv_buffer, 2));\r\n    //geometry.blend_mat = materials[pick_material];\r\n\r\n    return geometry;\r\n};\r\n\r\nfunction createThreeJSGeometry(blender_mesh, origin) {\r\n    //get materials\r\n    var mats = blender_mesh.mat,\r\n        materials = [];\r\n    for (var i = 0; i < mats.length; i++) {\r\n        var material = createThreeJSMaterial(mats[i]);\r\n        materials.push(material);\r\n    }\r\n\r\n    let pick_material = 0,\r\n        mesh = blender_mesh,\r\n        faces = mesh.mpoly,\r\n        loops = mesh.mloop,\r\n        UV = mesh.mloopuv,\r\n        verts = mesh.mvert,\r\n        vert_array = [],\r\n        face_array = [],\r\n        uv_array = [],\r\n        normal_array = [];\r\n\r\n    var geometry = new THREE.Geometry();\r\n\r\n    if (!faces) return geometry;\r\n\r\n\r\n    var index_count = 0;\r\n\r\n    let verts_array_buff = new Float32Array(verts.length * 3);\r\n\r\n    for (var i = 0; i < verts.length; i++) {\r\n        let vert = verts[i];\r\n        vert_array.push(new THREE.Vector3(vert.co[0] + origin[0], vert.co[2] + origin[2], -vert.co[1] - origin[1]));\r\n    }\r\n\r\n    for (var i = 0; i < faces.length; i++) {\r\n        var face = faces[i] || faces;\r\n        var len = face.totloop;\r\n        var start = face.loopstart;\r\n        var indexi = 1;\r\n\r\n        pick_material = face.mat_nr;\r\n\r\n        while (indexi < len) {\r\n            var face_normals = [];\r\n            var face_index_array = [];\r\n            var face_uvs = [];\r\n\r\n            let index = 0;\r\n\r\n            for (var l = 0; l < 3; l++) {\r\n                //Per Vertice \r\n\r\n                if ((indexi - 1) + l < len) {\r\n                    index = start + (indexi - 1) + l;\r\n                } else {\r\n                    index = start;\r\n                }\r\n\r\n                var v = loops[index].v;\r\n                var vert = verts[v];\r\n\r\n                face_index_array.push(v);\r\n\r\n                index_count++;\r\n\r\n                //get normals, which are 16byte ints, and norm them back into floats.\r\n\r\n                var\r\n                    n1 = vert.no[0],\r\n                    n2 = vert.no[2],\r\n                    n3 = -vert.no[1];\r\n\r\n                var nl = 1;\r\n\r\n                Math.sqrt((n1 * n1) + (n2 * n2) + (n3 * n3));\r\n\r\n                face_normals.push(new THREE.Vector3(n1 / nl, n2 / nl, n3 / nl));\r\n\r\n                if (UV) {\r\n                    var uv = UV[index].uv;\r\n                    face_uvs.push(new THREE.Vector2(uv[0], uv[1]));\r\n                }\r\n            }\r\n            uv_array.push(face_uvs);\r\n            face_array.push(new THREE.Face3(\r\n                face_index_array[0], face_index_array[1], face_index_array[2],\r\n                face_normals\r\n            ));\r\n\r\n            indexi += 2;\r\n        }\r\n    }\r\n    geometry.blend_mat = materials[pick_material];\r\n    geometry.vertices = vert_array;\r\n    geometry.faces = face_array;\r\n    if (uv_array.length > 0) {\r\n        geometry.faceVertexUvs = [uv_array];\r\n    }\r\n\r\n    geometry.uvsNeedUpdate = true;\r\n\r\n    //Well, using blender file normals does not work. Will need to investigate why normals from the blender file do not provide correct results. \r\n    //For now, have Three calculate normals. \r\n\r\n    geometry.computeVertexNormals();\r\n\r\n\r\n    return geometry;\r\n};","/*jshint esversion: 6 */\r\n\r\nlet blender_texture_cache = {};\r\n\r\n\r\nmodule.exports = function createThreeJSTexture(image) {\r\n    let base64 = require(\"base64-js\");\r\n    let parsed_blend_file = image.__blender_file__;\r\n    let texture = null;\r\n    let name = image.aname;\r\n\r\n    if (image.packedfile) {\r\n\r\n        if (blender_texture_cache[name]) {\r\n            texture = blender_texture_cache[name];\r\n        } else {\r\n\r\n            //get the extension\r\n            let ext = name.split('.').pop();\r\n\r\n            let data = image.packedfile;\r\n\r\n            let size = data.size;\r\n\r\n            let offset = data.data.__data_address__;\r\n\r\n            let raw_data = parsed_blend_file.byte.slice(offset, offset + size);\r\n\r\n            let encodedData = base64.fromByteArray(raw_data);\r\n\r\n            let dataURI;\r\n\r\n            switch (ext) {\r\n                case \"png\":\r\n                    dataURI = \"data:image/png;base64,\" + encodedData;\r\n                    break;\r\n                case \"jpg\":\r\n                    dataURI = \"data:image/jpeg;base64,\" + encodedData;\r\n                    break;\r\n            }\r\n\r\n            let img = new Image();\r\n\r\n            img.src = dataURI;\r\n\r\n            texture = new THREE.Texture(img);\r\n\r\n            img.onload = () => {\r\n                texture.needsUpdate = true;\r\n            };\r\n\r\n            blender_texture_cache[name] = texture;\r\n        }\r\n    }\r\n    return texture;\r\n};"]}