From 66534c191ac3ddaf36e3bd87a77d93a2e60dec5c Mon Sep 17 00:00:00 2001 From: Aaron Boman Date: Fri, 1 Nov 2019 10:12:24 -0500 Subject: [PATCH] Prepare for v2.0.6 --- dist/layout.d.ts | 10 ++++----- dist/mesh.d.ts | 1 + dist/webgl-obj-loader.js | 40 +++++++++++++++++++++++++++--------- dist/webgl-obj-loader.min.js | 8 ++++---- 4 files changed, 40 insertions(+), 19 deletions(-) diff --git a/dist/layout.d.ts b/dist/layout.d.ts index da3f6f4..57854f7 100644 --- a/dist/layout.d.ts +++ b/dist/layout.d.ts @@ -1,9 +1,9 @@ export declare enum TYPES { - "BYTE" = 1, - "UNSIGNED_BYTE" = 1, - "SHORT" = 2, - "UNSIGNED_SHORT" = 2, - "FLOAT" = 4 + "BYTE" = "BYTE", + "UNSIGNED_BYTE" = "UNSIGNED_BYTE", + "SHORT" = "SHORT", + "UNSIGNED_SHORT" = "UNSIGNED_SHORT", + "FLOAT" = "FLOAT" } export interface AttributeInfo { attribute: Attribute; diff --git a/dist/mesh.d.ts b/dist/mesh.d.ts index abca2ae..b2bd0d9 100644 --- a/dist/mesh.d.ts +++ b/dist/mesh.d.ts @@ -76,5 +76,6 @@ export default class Mesh { */ makeBufferData(layout: Layout): ArrayBufferWithItemSize; makeIndexBufferData(): Uint16ArrayWithItemSize; + makeIndexBufferDataForMaterials(...materialIndices: Array): Uint16ArrayWithItemSize; addMaterialLibrary(mtl: MaterialLibrary): void; } diff --git a/dist/webgl-obj-loader.js b/dist/webgl-obj-loader.js index 07bb35e..b4b2691 100644 --- a/dist/webgl-obj-loader.js +++ b/dist/webgl-obj-loader.js @@ -160,11 +160,11 @@ __webpack_require__.r(__webpack_exports__); /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "Layout", function() { return Layout; }); var TYPES; (function (TYPES) { - TYPES[TYPES["BYTE"] = 1] = "BYTE"; - TYPES[TYPES["UNSIGNED_BYTE"] = 1] = "UNSIGNED_BYTE"; - TYPES[TYPES["SHORT"] = 2] = "SHORT"; - TYPES[TYPES["UNSIGNED_SHORT"] = 2] = "UNSIGNED_SHORT"; - TYPES[TYPES["FLOAT"] = 4] = "FLOAT"; + TYPES["BYTE"] = "BYTE"; + TYPES["UNSIGNED_BYTE"] = "UNSIGNED_BYTE"; + TYPES["SHORT"] = "SHORT"; + TYPES["UNSIGNED_SHORT"] = "UNSIGNED_SHORT"; + TYPES["FLOAT"] = "FLOAT"; })(TYPES || (TYPES = {})); /** * An exception for when two or more of the same attributes are found in the @@ -215,7 +215,21 @@ class Attribute { this.size = size; this.type = type; this.normalized = normalized; - this.sizeOfType = this.type; + switch (type) { + case "BYTE": + case "UNSIGNED_BYTE": + this.sizeOfType = 1; + break; + case "SHORT": + case "UNSIGNED_SHORT": + this.sizeOfType = 2; + break; + case "FLOAT": + this.sizeOfType = 4; + break; + default: + throw new Error(`Unknown gl type: ${type}`); + } this.sizeInBytes = this.sizeOfType * size; } } @@ -1310,7 +1324,7 @@ class Mesh { Think of faces having Vertices which are comprised of the attributes location (v), texture (vt), and normal (vn). */ - const vertex = elements[j].split("/"); + const vertex = triangle[j].split("/"); // it's possible for faces to only specify the vertex // and the normal. In this case, vertex will only have // a length of 2 and not 3 and the normal will be the @@ -1699,6 +1713,12 @@ class Mesh { buffer.numItems = this.indices.length; return buffer; } + makeIndexBufferDataForMaterials(...materialIndices) { + const indices = new Array().concat(...materialIndices.map(mtlIdx => this.indicesPerMaterial[mtlIdx])); + const buffer = new Uint16Array(indices); + buffer.numItems = indices.length; + return buffer; + } addMaterialLibrary(mtl) { for (const name in mtl.materials) { if (!(name in this.materialIndices)) { @@ -1771,7 +1791,7 @@ function downloadMtlTextures(mtl, root) { const material = mtl.materials[materialName]; for (const attr of mapAttributes) { const mapData = material[attr]; - if (!mapData) { + if (!mapData || !mapData.filename) { continue; } const url = root + mapData.filename; @@ -2059,11 +2079,11 @@ function deleteMeshBuffers(gl, mesh) { /*! no static exports found */ /***/ (function(module, exports, __webpack_require__) { -module.exports = __webpack_require__(/*! /Users/aaron/git/webgl-obj-loader/src/index.ts */"./src/index.ts"); +module.exports = __webpack_require__(/*! /home/aaron/google_drive/projects/webgl-obj-loader/src/index.ts */"./src/index.ts"); /***/ }) /******/ }); }); -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack://OBJ/webpack/universalModuleDefinition","webpack://OBJ/webpack/bootstrap","webpack://OBJ/./src/index.ts","webpack://OBJ/./src/layout.ts","webpack://OBJ/./src/material.ts","webpack://OBJ/./src/mesh.ts","webpack://OBJ/./src/utils.ts"],"names":[],"mappings":"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","file":"webgl-obj-loader.js","sourcesContent":["(function webpackUniversalModuleDefinition(root, factory) {\n\tif(typeof exports === 'object' && typeof module === 'object')\n\t\tmodule.exports = factory();\n\telse if(typeof define === 'function' && define.amd)\n\t\tdefine(\"OBJ\", [], factory);\n\telse if(typeof exports === 'object')\n\t\texports[\"OBJ\"] = factory();\n\telse\n\t\troot[\"OBJ\"] = factory();\n})(typeof self !== 'undefined' ? self : this, function() {\nreturn "," \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, { enumerable: true, get: getter });\n \t\t}\n \t};\n\n \t// define __esModule on exports\n \t__webpack_require__.r = function(exports) {\n \t\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n \t\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n \t\t}\n \t\tObject.defineProperty(exports, '__esModule', { value: true });\n \t};\n\n \t// create a fake namespace object\n \t// mode & 1: value is a module id, require it\n \t// mode & 2: merge all properties of value into the ns\n \t// mode & 4: return value when already ns object\n \t// mode & 8|1: behave like require\n \t__webpack_require__.t = function(value, mode) {\n \t\tif(mode & 1) value = __webpack_require__(value);\n \t\tif(mode & 8) return value;\n \t\tif((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;\n \t\tvar ns = Object.create(null);\n \t\t__webpack_require__.r(ns);\n \t\tObject.defineProperty(ns, 'default', { enumerable: true, value: value });\n \t\tif(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));\n \t\treturn ns;\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"/\";\n\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = 0);\n","import Mesh, {\n    MeshOptions,\n    MaterialNameToIndex,\n    IndexToMaterial,\n    ArrayBufferWithItemSize,\n    Uint16ArrayWithItemSize,\n} from \"./mesh\";\nimport { Material, MaterialLibrary, Vec3, UVW, TextureMapData } from \"./material\";\nimport { Layout, TYPES, AttributeInfo, DuplicateAttributeException, Attribute } from \"./layout\";\nimport {\n    downloadModels,\n    downloadMeshes,\n    initMeshBuffers,\n    deleteMeshBuffers,\n    DownloadModelsOptions,\n    MeshMap,\n    NameAndUrls,\n    ExtendedGLBuffer,\n    MeshWithBuffers,\n} from \"./utils\";\n\nconst version = \"2.0.3\";\n\n/**\n * @namespace\n */\nexport {\n    ArrayBufferWithItemSize,\n    Attribute,\n    AttributeInfo,\n    DownloadModelsOptions,\n    DuplicateAttributeException,\n    ExtendedGLBuffer,\n    IndexToMaterial,\n    Layout,\n    Material,\n    MaterialLibrary,\n    MaterialNameToIndex,\n    Mesh,\n    MeshMap,\n    MeshOptions,\n    MeshWithBuffers,\n    NameAndUrls,\n    TextureMapData,\n    TYPES,\n    Uint16ArrayWithItemSize,\n    UVW,\n    Vec3,\n    downloadModels,\n    downloadMeshes,\n    initMeshBuffers,\n    deleteMeshBuffers,\n    version,\n};\n","export enum TYPES {\n    \"BYTE\" = 1,\n    \"UNSIGNED_BYTE\" = 1,\n    \"SHORT\" = 2,\n    \"UNSIGNED_SHORT\" = 2,\n    \"FLOAT\" = 4,\n}\n\nexport interface AttributeInfo {\n    attribute: Attribute;\n    size: Attribute[\"size\"];\n    type: Attribute[\"type\"];\n    normalized: Attribute[\"normalized\"];\n    offset: number;\n    stride: number;\n}\n\n/**\n * An exception for when two or more of the same attributes are found in the\n * same layout.\n * @private\n */\nexport class DuplicateAttributeException extends Error {\n    /**\n     * Create a DuplicateAttributeException\n     * @param {Attribute} attribute - The attribute that was found more than\n     *        once in the {@link Layout}\n     */\n    constructor(attribute: Attribute) {\n        super(`found duplicate attribute: ${attribute.key}`);\n    }\n}\n\n/**\n * Represents how a vertex attribute should be packed into an buffer.\n * @private\n */\nexport class Attribute {\n    public sizeOfType: number;\n    public sizeInBytes: number;\n    /**\n     * Create an attribute. Do not call this directly, use the predefined\n     * constants.\n     * @param {string} key - The name of this attribute as if it were a key in\n     *        an Object. Use the camel case version of the upper snake case\n     *        const name.\n     * @param {number} size - The number of components per vertex attribute.\n     *        Must be 1, 2, 3, or 4.\n     * @param {string} type - The data type of each component for this\n     *        attribute. Possible values:<br/>\n     *        \"BYTE\": signed 8-bit integer, with values in [-128, 127]<br/>\n     *        \"SHORT\": signed 16-bit integer, with values in\n     *            [-32768, 32767]<br/>\n     *        \"UNSIGNED_BYTE\": unsigned 8-bit integer, with values in\n     *            [0, 255]<br/>\n     *        \"UNSIGNED_SHORT\": unsigned 16-bit integer, with values in\n     *            [0, 65535]<br/>\n     *        \"FLOAT\": 32-bit floating point number\n     * @param {boolean} normalized - Whether integer data values should be\n     *        normalized when being casted to a float.<br/>\n     *        If true, signed integers are normalized to [-1, 1].<br/>\n     *        If true, unsigned integers are normalized to [0, 1].<br/>\n     *        For type \"FLOAT\", this parameter has no effect.\n     */\n    constructor(public key: string, public size: number, public type: TYPES, public normalized: boolean = false) {\n        this.sizeOfType = this.type;\n        this.sizeInBytes = this.sizeOfType * size;\n    }\n}\n\n/**\n * A class to represent the memory layout for a vertex attribute array. Used by\n * {@link Mesh}'s TBD(...) method to generate a packed array from mesh data.\n * <p>\n * Layout can sort of be thought of as a C-style struct declaration.\n * {@link Mesh}'s TBD(...) method will use the {@link Layout} instance to\n * pack an array in the given attribute order.\n * <p>\n * Layout also is very helpful when calling a WebGL context's\n * <code>vertexAttribPointer</code> method. If you've created a buffer using\n * a Layout instance, then the same Layout instance can be used to determine\n * the size, type, normalized, stride, and offset parameters for\n * <code>vertexAttribPointer</code>.\n * <p>\n * For example:\n * <pre><code>\n *\n * const index = glctx.getAttribLocation(shaderProgram, \"pos\");\n * glctx.vertexAttribPointer(\n *   layout.position.size,\n *   glctx[layout.position.type],\n *   layout.position.normalized,\n *   layout.position.stride,\n *   layout.position.offset);\n * </code></pre>\n * @see {@link Mesh}\n */\nexport class Layout {\n    // Geometry attributes\n    /**\n     * Attribute layout to pack a vertex's x, y, & z as floats\n     *\n     * @see {@link Layout}\n     */\n    static POSITION = new Attribute(\"position\", 3, TYPES.FLOAT);\n\n    /**\n     * Attribute layout to pack a vertex's normal's x, y, & z as floats\n     *\n     * @see {@link Layout}\n     */\n    static NORMAL = new Attribute(\"normal\", 3, TYPES.FLOAT);\n\n    /**\n     * Attribute layout to pack a vertex's normal's x, y, & z as floats.\n     * <p>\n     * This value will be computed on-the-fly based on the texture coordinates.\n     * If no texture coordinates are available, the generated value will default to\n     * 0, 0, 0.\n     *\n     * @see {@link Layout}\n     */\n    static TANGENT = new Attribute(\"tangent\", 3, TYPES.FLOAT);\n\n    /**\n     * Attribute layout to pack a vertex's normal's bitangent x, y, & z as floats.\n     * <p>\n     * This value will be computed on-the-fly based on the texture coordinates.\n     * If no texture coordinates are available, the generated value will default to\n     * 0, 0, 0.\n     * @see {@link Layout}\n     */\n    static BITANGENT = new Attribute(\"bitangent\", 3, TYPES.FLOAT);\n\n    /**\n     * Attribute layout to pack a vertex's texture coordinates' u & v as floats\n     *\n     * @see {@link Layout}\n     */\n    static UV = new Attribute(\"uv\", 2, TYPES.FLOAT);\n\n    // Material attributes\n\n    /**\n     * Attribute layout to pack an unsigned short to be interpreted as a the index\n     * into a {@link Mesh}'s materials list.\n     * <p>\n     * The intention of this value is to send all of the {@link Mesh}'s materials\n     * into multiple shader uniforms and then reference the current one by this\n     * vertex attribute.\n     * <p>\n     * example glsl code:\n     *\n     * <pre><code>\n     *  // this is bound using MATERIAL_INDEX\n     *  attribute int materialIndex;\n     *\n     *  struct Material {\n     *    vec3 diffuse;\n     *    vec3 specular;\n     *    vec3 specularExponent;\n     *  };\n     *\n     *  uniform Material materials[MAX_MATERIALS];\n     *\n     *  // ...\n     *\n     *  vec3 diffuse = materials[materialIndex];\n     *\n     * </code></pre>\n     * TODO: More description & test to make sure subscripting by attributes even\n     * works for webgl\n     *\n     * @see {@link Layout}\n     */\n    static MATERIAL_INDEX = new Attribute(\"materialIndex\", 1, TYPES.SHORT);\n    static MATERIAL_ENABLED = new Attribute(\"materialEnabled\", 1, TYPES.UNSIGNED_SHORT);\n    static AMBIENT = new Attribute(\"ambient\", 3, TYPES.FLOAT);\n    static DIFFUSE = new Attribute(\"diffuse\", 3, TYPES.FLOAT);\n    static SPECULAR = new Attribute(\"specular\", 3, TYPES.FLOAT);\n    static SPECULAR_EXPONENT = new Attribute(\"specularExponent\", 3, TYPES.FLOAT);\n    static EMISSIVE = new Attribute(\"emissive\", 3, TYPES.FLOAT);\n    static TRANSMISSION_FILTER = new Attribute(\"transmissionFilter\", 3, TYPES.FLOAT);\n    static DISSOLVE = new Attribute(\"dissolve\", 1, TYPES.FLOAT);\n    static ILLUMINATION = new Attribute(\"illumination\", 1, TYPES.UNSIGNED_SHORT);\n    static REFRACTION_INDEX = new Attribute(\"refractionIndex\", 1, TYPES.FLOAT);\n    static SHARPNESS = new Attribute(\"sharpness\", 1, TYPES.FLOAT);\n    static MAP_DIFFUSE = new Attribute(\"mapDiffuse\", 1, TYPES.SHORT);\n    static MAP_AMBIENT = new Attribute(\"mapAmbient\", 1, TYPES.SHORT);\n    static MAP_SPECULAR = new Attribute(\"mapSpecular\", 1, TYPES.SHORT);\n    static MAP_SPECULAR_EXPONENT = new Attribute(\"mapSpecularExponent\", 1, TYPES.SHORT);\n    static MAP_DISSOLVE = new Attribute(\"mapDissolve\", 1, TYPES.SHORT);\n    static ANTI_ALIASING = new Attribute(\"antiAliasing\", 1, TYPES.UNSIGNED_SHORT);\n    static MAP_BUMP = new Attribute(\"mapBump\", 1, TYPES.SHORT);\n    static MAP_DISPLACEMENT = new Attribute(\"mapDisplacement\", 1, TYPES.SHORT);\n    static MAP_DECAL = new Attribute(\"mapDecal\", 1, TYPES.SHORT);\n    static MAP_EMISSIVE = new Attribute(\"mapEmissive\", 1, TYPES.SHORT);\n\n    public stride: number;\n    public attributes: Attribute[];\n    public attributeMap: { [idx: string]: AttributeInfo };\n    /**\n     * Create a Layout object. This constructor will throw if any duplicate\n     * attributes are given.\n     * @param {Array} ...attributes - An ordered list of attributes that\n     *        describe the desired memory layout for each vertex attribute.\n     *        <p>\n     *\n     * @see {@link Mesh}\n     */\n    constructor(...attributes: Attribute[]) {\n        this.attributes = attributes;\n        this.attributeMap = {};\n        let offset = 0;\n        let maxStrideMultiple = 0;\n        for (const attribute of attributes) {\n            if (this.attributeMap[attribute.key]) {\n                throw new DuplicateAttributeException(attribute);\n            }\n            // Add padding to satisfy WebGL's requirement that all\n            // vertexAttribPointer calls have an offset that is a multiple of\n            // the type size.\n            if (offset % attribute.sizeOfType !== 0) {\n                offset += attribute.sizeOfType - (offset % attribute.sizeOfType);\n                console.warn(\"Layout requires padding before \" + attribute.key + \" attribute\");\n            }\n            this.attributeMap[attribute.key] = {\n                attribute: attribute,\n                size: attribute.size,\n                type: attribute.type,\n                normalized: attribute.normalized,\n                offset: offset,\n            } as AttributeInfo;\n            offset += attribute.sizeInBytes;\n            maxStrideMultiple = Math.max(maxStrideMultiple, attribute.sizeOfType);\n        }\n        // Add padding to the end to satisfy WebGL's requirement that all\n        // vertexAttribPointer calls have a stride that is a multiple of the\n        // type size. Because we're putting differently sized attributes into\n        // the same buffer, it must be padded to a multiple of the largest\n        // type size.\n        if (offset % maxStrideMultiple !== 0) {\n            offset += maxStrideMultiple - (offset % maxStrideMultiple);\n            console.warn(\"Layout requires padding at the back\");\n        }\n        this.stride = offset;\n        for (const attribute of attributes) {\n            this.attributeMap[attribute.key].stride = this.stride;\n        }\n    }\n}\n","export type Vec3 = [number, number, number];\n\nexport interface UVW {\n    u: number;\n    v: number;\n    w: number;\n}\n\nexport interface TextureMapData {\n    colorCorrection: boolean;\n    horizontalBlending: boolean;\n    verticalBlending: boolean;\n    boostMipMapSharpness: number;\n    modifyTextureMap: {\n        brightness: number;\n        contrast: number;\n    };\n    offset: UVW;\n    scale: UVW;\n    turbulence: UVW;\n    clamp: boolean;\n    textureResolution: number | null;\n    bumpMultiplier: number;\n    imfChan: string | null;\n    filename: string;\n    reflectionType?: string;\n    texture?: HTMLImageElement;\n}\n\n/**\n * The Material class.\n */\nexport class Material {\n    /**\n     * Constructor\n     * @param {String} name the unique name of the material\n     */\n    // The values for the following attibutes\n    // are an array of R, G, B normalized values.\n    // Ka - Ambient Reflectivity\n    ambient: Vec3 = [0, 0, 0];\n    // Kd - Defuse Reflectivity\n    diffuse: Vec3 = [0, 0, 0];\n    // Ks\n    specular: Vec3 = [0, 0, 0];\n    // Ke\n    emissive: Vec3 = [0, 0, 0];\n    // Tf\n    transmissionFilter: Vec3 = [0, 0, 0];\n    // d\n    dissolve: number = 0;\n    // valid range is between 0 and 1000\n    specularExponent: number = 0;\n    // either d or Tr; valid values are normalized\n    transparency: number = 0;\n    // illum - the enum of the illumination model to use\n    illumination: number = 0;\n    // Ni - Set to \"normal\" (air).\n    refractionIndex: number = 1;\n    // sharpness\n    sharpness: number = 0;\n    // map_Kd\n    mapDiffuse: TextureMapData = emptyTextureOptions();\n    // map_Ka\n    mapAmbient: TextureMapData = emptyTextureOptions();\n    // map_Ks\n    mapSpecular: TextureMapData = emptyTextureOptions();\n    // map_Ns\n    mapSpecularExponent: TextureMapData = emptyTextureOptions();\n    // map_d\n    mapDissolve: TextureMapData = emptyTextureOptions();\n    // map_aat\n    antiAliasing: boolean = false;\n    // map_bump or bump\n    mapBump: TextureMapData = emptyTextureOptions();\n    // disp\n    mapDisplacement: TextureMapData = emptyTextureOptions();\n    // decal\n    mapDecal: TextureMapData = emptyTextureOptions();\n    // map_Ke\n    mapEmissive: TextureMapData = emptyTextureOptions();\n    // refl - when the reflection type is a cube, there will be multiple refl\n    //        statements for each side of the cube. If it's a spherical\n    //        reflection, there should only ever be one.\n    mapReflections: TextureMapData[] = [];\n    constructor(public name: string) {}\n}\n\nconst SENTINEL_MATERIAL = new Material(\"sentinel\");\n\n/**\n * https://en.wikipedia.org/wiki/Wavefront_.obj_file\n * http://paulbourke.net/dataformats/mtl/\n */\nexport class MaterialLibrary {\n    /**\n     * Constructs the Material Parser\n     * @param mtlData the MTL file contents\n     */\n    public currentMaterial: Material = SENTINEL_MATERIAL;\n    public materials: { [k: string]: Material } = {};\n\n    constructor(public data: string) {\n        this.parse();\n    }\n\n    /* eslint-disable camelcase */\n    /* the function names here disobey camelCase conventions\n     to make parsing/routing easier. see the parse function\n     documentation for more information. */\n\n    /**\n     * Creates a new Material object and adds to the registry.\n     * @param tokens the tokens associated with the directive\n     */\n    parse_newmtl(tokens: string[]) {\n        const name = tokens[0];\n        // console.info('Parsing new Material:', name);\n\n        this.currentMaterial = new Material(name);\n        this.materials[name] = this.currentMaterial;\n    }\n\n    /**\n     * See the documenation for parse_Ka below for a better understanding.\n     *\n     * Given a list of possible color tokens, returns an array of R, G, and B\n     * color values.\n     *\n     * @param tokens the tokens associated with the directive\n     * @return {*} a 3 element array containing the R, G, and B values\n     * of the color.\n     */\n    parseColor(tokens: string[]): Vec3 {\n        if (tokens[0] == \"spectral\") {\n            throw new Error(\n                \"The MTL parser does not support spectral curve files. You will \" +\n                    \"need to convert the MTL colors to either RGB or CIEXYZ.\",\n            );\n        }\n\n        if (tokens[0] == \"xyz\") {\n            throw new Error(\n                \"The MTL parser does not currently support XYZ colors. Either convert the \" +\n                    \"XYZ values to RGB or create an issue to add support for XYZ\",\n            );\n        }\n\n        // from my understanding of the spec, RGB values at this point\n        // will either be 3 floats or exactly 1 float, so that's the check\n        // that i'm going to perform here\n        if (tokens.length == 3) {\n            const [x, y, z] = tokens;\n            return [parseFloat(x), parseFloat(y), parseFloat(z)];\n        }\n\n        // Since tokens at this point has a length of 3, we're going to assume\n        // it's exactly 1, skipping the check for 2.\n        const value = parseFloat(tokens[0]);\n        // in this case, all values are equivalent\n        return [value, value, value];\n    }\n\n    /**\n     * Parse the ambient reflectivity\n     *\n     * A Ka directive can take one of three forms:\n     *   - Ka r g b\n     *   - Ka spectral file.rfl\n     *   - Ka xyz x y z\n     * These three forms are mutually exclusive in that only one\n     * declaration can exist per material. It is considered a syntax\n     * error otherwise.\n     *\n     * The \"Ka\" form specifies the ambient reflectivity using RGB values.\n     * The \"g\" and \"b\" values are optional. If only the \"r\" value is\n     * specified, then the \"g\" and \"b\" values are assigned the value of\n     * \"r\". Values are normally in the range 0.0 to 1.0. Values outside\n     * of this range increase or decrease the reflectivity accordingly.\n     *\n     * The \"Ka spectral\" form specifies the ambient reflectivity using a\n     * spectral curve. \"file.rfl\" is the name of the \".rfl\" file containing\n     * the curve data. \"factor\" is an optional argument which is a multiplier\n     * for the values in the .rfl file and defaults to 1.0 if not specified.\n     *\n     * The \"Ka xyz\" form specifies the ambient reflectivity using CIEXYZ values.\n     * \"x y z\" are the values of the CIEXYZ color space. The \"y\" and \"z\" arguments\n     * are optional and take on the value of the \"x\" component if only \"x\" is\n     * specified. The \"x y z\" values are normally in the range of 0.0 to 1.0 and\n     * increase or decrease ambient reflectivity accordingly outside of that\n     * range.\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_Ka(tokens: string[]) {\n        this.currentMaterial.ambient = this.parseColor(tokens);\n    }\n\n    /**\n     * Diffuse Reflectivity\n     *\n     * Similar to the Ka directive. Simply replace \"Ka\" with \"Kd\" and the rules\n     * are the same\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_Kd(tokens: string[]) {\n        this.currentMaterial.diffuse = this.parseColor(tokens);\n    }\n\n    /**\n     * Spectral Reflectivity\n     *\n     * Similar to the Ka directive. Simply replace \"Ks\" with \"Kd\" and the rules\n     * are the same\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_Ks(tokens: string[]) {\n        this.currentMaterial.specular = this.parseColor(tokens);\n    }\n\n    /**\n     * Emissive\n     *\n     * The amount and color of light emitted by the object.\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_Ke(tokens: string[]) {\n        this.currentMaterial.emissive = this.parseColor(tokens);\n    }\n\n    /**\n     * Transmission Filter\n     *\n     * Any light passing through the object is filtered by the transmission\n     * filter, which only allows specific colors to pass through. For example, Tf\n     * 0 1 0 allows all of the green to pass through and filters out all of the\n     * red and blue.\n     *\n     * Similar to the Ka directive. Simply replace \"Ks\" with \"Tf\" and the rules\n     * are the same\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_Tf(tokens: string[]) {\n        this.currentMaterial.transmissionFilter = this.parseColor(tokens);\n    }\n\n    /**\n     * Specifies the dissolve for the current material.\n     *\n     * Statement: d [-halo] `factor`\n     *\n     * Example: \"d 0.5\"\n     *\n     * The factor is the amount this material dissolves into the background. A\n     * factor of 1.0 is fully opaque. This is the default when a new material is\n     * created. A factor of 0.0 is fully dissolved (completely transparent).\n     *\n     * Unlike a real transparent material, the dissolve does not depend upon\n     * material thickness nor does it have any spectral character. Dissolve works\n     * on all illumination models.\n     *\n     * The dissolve statement allows for an optional \"-halo\" flag which indicates\n     * that a dissolve is dependent on the surface orientation relative to the\n     * viewer. For example, a sphere with the following dissolve, \"d -halo 0.0\",\n     * will be fully dissolved at its center and will appear gradually more opaque\n     * toward its edge.\n     *\n     * \"factor\" is the minimum amount of dissolve applied to the material. The\n     * amount of dissolve will vary between 1.0 (fully opaque) and the specified\n     * \"factor\". The formula is:\n     *\n     *    dissolve = 1.0 - (N*v)(1.0-factor)\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_d(tokens: string[]) {\n        // this ignores the -halo option as I can't find any documentation on what\n        // it's supposed to be.\n        this.currentMaterial.dissolve = parseFloat(tokens.pop() || \"0\");\n    }\n\n    /**\n     * The \"illum\" statement specifies the illumination model to use in the\n     * material. Illumination models are mathematical equations that represent\n     * various material lighting and shading effects.\n     *\n     * The illumination number can be a number from 0 to 10. The following are\n     * the list of illumination enumerations and their summaries:\n     * 0. Color on and Ambient off\n     * 1. Color on and Ambient on\n     * 2. Highlight on\n     * 3. Reflection on and Ray trace on\n     * 4. Transparency: Glass on, Reflection: Ray trace on\n     * 5. Reflection: Fresnel on and Ray trace on\n     * 6. Transparency: Refraction on, Reflection: Fresnel off and Ray trace on\n     * 7. Transparency: Refraction on, Reflection: Fresnel on and Ray trace on\n     * 8. Reflection on and Ray trace off\n     * 9. Transparency: Glass on, Reflection: Ray trace off\n     * 10. Casts shadows onto invisible surfaces\n     *\n     * Example: \"illum 2\" to specify the \"Highlight on\" model\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_illum(tokens: string[]) {\n        this.currentMaterial.illumination = parseInt(tokens[0]);\n    }\n\n    /**\n     * Optical Density (AKA Index of Refraction)\n     *\n     * Statement: Ni `index`\n     *\n     * Example: Ni 1.0\n     *\n     * Specifies the optical density for the surface. `index` is the value\n     * for the optical density. The values can range from 0.001 to 10.  A value of\n     * 1.0 means that light does not bend as it passes through an object.\n     * Increasing the optical_density increases the amount of bending. Glass has\n     * an index of refraction of about 1.5. Values of less than 1.0 produce\n     * bizarre results and are not recommended\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_Ni(tokens: string[]) {\n        this.currentMaterial.refractionIndex = parseFloat(tokens[0]);\n    }\n\n    /**\n     * Specifies the specular exponent for the current material. This defines the\n     * focus of the specular highlight.\n     *\n     * Statement: Ns `exponent`\n     *\n     * Example: \"Ns 250\"\n     *\n     * `exponent` is the value for the specular exponent. A high exponent results\n     * in a tight, concentrated highlight. Ns Values normally range from 0 to\n     * 1000.\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_Ns(tokens: string[]) {\n        this.currentMaterial.specularExponent = parseInt(tokens[0]);\n    }\n\n    /**\n     * Specifies the sharpness of the reflections from the local reflection map.\n     *\n     * Statement: sharpness `value`\n     *\n     * Example: \"sharpness 100\"\n     *\n     * If a material does not have a local reflection map defined in its material\n     * defintions, sharpness will apply to the global reflection map defined in\n     * PreView.\n     *\n     * `value` can be a number from 0 to 1000. The default is 60. A high value\n     * results in a clear reflection of objects in the reflection map.\n     *\n     * Tip: sharpness values greater than 100 introduce aliasing effects in\n     * flat surfaces that are viewed at a sharp angle.\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_sharpness(tokens: string[]) {\n        this.currentMaterial.sharpness = parseInt(tokens[0]);\n    }\n\n    /**\n     * Parses the -cc flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -cc flag\n     * @param options the Object of all image options\n     */\n    parse_cc(values: string[], options: TextureMapData) {\n        options.colorCorrection = values[0] == \"on\";\n    }\n\n    /**\n     * Parses the -blendu flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -blendu flag\n     * @param options the Object of all image options\n     */\n    parse_blendu(values: string[], options: TextureMapData) {\n        options.horizontalBlending = values[0] == \"on\";\n    }\n\n    /**\n     * Parses the -blendv flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -blendv flag\n     * @param options the Object of all image options\n     */\n    parse_blendv(values: string[], options: TextureMapData) {\n        options.verticalBlending = values[0] == \"on\";\n    }\n\n    /**\n     * Parses the -boost flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -boost flag\n     * @param options the Object of all image options\n     */\n    parse_boost(values: string[], options: TextureMapData) {\n        options.boostMipMapSharpness = parseFloat(values[0]);\n    }\n\n    /**\n     * Parses the -mm flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -mm flag\n     * @param options the Object of all image options\n     */\n    parse_mm(values: string[], options: TextureMapData) {\n        options.modifyTextureMap.brightness = parseFloat(values[0]);\n        options.modifyTextureMap.contrast = parseFloat(values[1]);\n    }\n\n    /**\n     * Parses and sets the -o, -s, and -t  u, v, and w values\n     *\n     * @param values the values passed to the -o, -s, -t flag\n     * @param {Object} option the Object of either the -o, -s, -t option\n     * @param {Integer} defaultValue the Object of all image options\n     */\n    parse_ost(values: string[], option: UVW, defaultValue: number) {\n        while (values.length < 3) {\n            values.push(defaultValue.toString());\n        }\n\n        option.u = parseFloat(values[0]);\n        option.v = parseFloat(values[1]);\n        option.w = parseFloat(values[2]);\n    }\n\n    /**\n     * Parses the -o flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -o flag\n     * @param options the Object of all image options\n     */\n    parse_o(values: string[], options: TextureMapData) {\n        this.parse_ost(values, options.offset, 0);\n    }\n\n    /**\n     * Parses the -s flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -s flag\n     * @param options the Object of all image options\n     */\n    parse_s(values: string[], options: TextureMapData) {\n        this.parse_ost(values, options.scale, 1);\n    }\n\n    /**\n     * Parses the -t flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -t flag\n     * @param options the Object of all image options\n     */\n    parse_t(values: string[], options: TextureMapData) {\n        this.parse_ost(values, options.turbulence, 0);\n    }\n\n    /**\n     * Parses the -texres flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -texres flag\n     * @param options the Object of all image options\n     */\n    parse_texres(values: string[], options: TextureMapData) {\n        options.textureResolution = parseFloat(values[0]);\n    }\n\n    /**\n     * Parses the -clamp flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -clamp flag\n     * @param options the Object of all image options\n     */\n    parse_clamp(values: string[], options: TextureMapData) {\n        options.clamp = values[0] == \"on\";\n    }\n\n    /**\n     * Parses the -bm flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -bm flag\n     * @param options the Object of all image options\n     */\n    parse_bm(values: string[], options: TextureMapData) {\n        options.bumpMultiplier = parseFloat(values[0]);\n    }\n\n    /**\n     * Parses the -imfchan flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -imfchan flag\n     * @param options the Object of all image options\n     */\n    parse_imfchan(values: string[], options: TextureMapData) {\n        options.imfChan = values[0];\n    }\n\n    /**\n     * This only exists for relection maps and denotes the type of reflection.\n     *\n     * @param values the values passed to the -type flag\n     * @param options the Object of all image options\n     */\n    parse_type(values: string[], options: TextureMapData) {\n        options.reflectionType = values[0];\n    }\n\n    /**\n     * Parses the texture's options and returns an options object with the info\n     *\n     * @param tokens all of the option tokens to pass to the texture\n     * @return {Object} a complete object of objects to apply to the texture\n     */\n    parseOptions(tokens: string[]): TextureMapData {\n        const options = emptyTextureOptions();\n\n        let option;\n        let values;\n        const optionsToValues: { [k: string]: string[] } = {};\n\n        tokens.reverse();\n\n        while (tokens.length) {\n            // token is guaranteed to exists here, hence the explicit \"as\"\n            const token = tokens.pop() as string;\n\n            if (token.startsWith(\"-\")) {\n                option = token.substr(1);\n                optionsToValues[option] = [];\n            } else if (option) {\n                optionsToValues[option].push(token);\n            }\n        }\n\n        for (option in optionsToValues) {\n            if (!optionsToValues.hasOwnProperty(option)) {\n                continue;\n            }\n            values = optionsToValues[option];\n            const optionMethod = (this as any)[`parse_${option}`];\n            if (optionMethod) {\n                optionMethod.bind(this)(values, options);\n            }\n        }\n\n        return options;\n    }\n\n    /**\n     * Parses the given texture map line.\n     *\n     * @param tokens all of the tokens representing the texture\n     * @return a complete object of objects to apply to the texture\n     */\n    parseMap(tokens: string[]): TextureMapData {\n        // according to wikipedia:\n        // (https://en.wikipedia.org/wiki/Wavefront_.obj_file#Vendor_specific_alterations)\n        // there is at least one vendor that places the filename before the options\n        // rather than after (which is to spec). All options start with a '-'\n        // so if the first token doesn't start with a '-', we're going to assume\n        // it's the name of the map file.\n        let optionsString;\n        let filename = \"\";\n        if (!tokens[0].startsWith(\"-\")) {\n            [filename, ...optionsString] = tokens;\n        } else {\n            filename = tokens.pop() as string;\n            optionsString = tokens;\n        }\n\n        const options = this.parseOptions(optionsString);\n        options.filename = filename.replace(/\\\\/g, \"/\");\n\n        return options;\n    }\n\n    /**\n     * Parses the ambient map.\n     *\n     * @param tokens list of tokens for the map_Ka direcive\n     */\n    parse_map_Ka(tokens: string[]) {\n        this.currentMaterial.mapAmbient = this.parseMap(tokens);\n    }\n\n    /**\n     * Parses the diffuse map.\n     *\n     * @param tokens list of tokens for the map_Kd direcive\n     */\n    parse_map_Kd(tokens: string[]) {\n        this.currentMaterial.mapDiffuse = this.parseMap(tokens);\n    }\n\n    /**\n     * Parses the specular map.\n     *\n     * @param tokens list of tokens for the map_Ks direcive\n     */\n    parse_map_Ks(tokens: string[]) {\n        this.currentMaterial.mapSpecular = this.parseMap(tokens);\n    }\n\n    /**\n     * Parses the emissive map.\n     *\n     * @param tokens list of tokens for the map_Ke direcive\n     */\n    parse_map_Ke(tokens: string[]) {\n        this.currentMaterial.mapEmissive = this.parseMap(tokens);\n    }\n\n    /**\n     * Parses the specular exponent map.\n     *\n     * @param tokens list of tokens for the map_Ns direcive\n     */\n    parse_map_Ns(tokens: string[]) {\n        this.currentMaterial.mapSpecularExponent = this.parseMap(tokens);\n    }\n\n    /**\n     * Parses the dissolve map.\n     *\n     * @param tokens list of tokens for the map_d direcive\n     */\n    parse_map_d(tokens: string[]) {\n        this.currentMaterial.mapDissolve = this.parseMap(tokens);\n    }\n\n    /**\n     * Parses the anti-aliasing option.\n     *\n     * @param tokens list of tokens for the map_aat direcive\n     */\n    parse_map_aat(tokens: string[]) {\n        this.currentMaterial.antiAliasing = tokens[0] == \"on\";\n    }\n\n    /**\n     * Parses the bump map.\n     *\n     * @param tokens list of tokens for the map_bump direcive\n     */\n    parse_map_bump(tokens: string[]) {\n        this.currentMaterial.mapBump = this.parseMap(tokens);\n    }\n\n    /**\n     * Parses the bump map.\n     *\n     * @param tokens list of tokens for the bump direcive\n     */\n    parse_bump(tokens: string[]) {\n        this.parse_map_bump(tokens);\n    }\n\n    /**\n     * Parses the disp map.\n     *\n     * @param tokens list of tokens for the disp direcive\n     */\n    parse_disp(tokens: string[]) {\n        this.currentMaterial.mapDisplacement = this.parseMap(tokens);\n    }\n\n    /**\n     * Parses the decal map.\n     *\n     * @param tokens list of tokens for the map_decal direcive\n     */\n    parse_decal(tokens: string[]) {\n        this.currentMaterial.mapDecal = this.parseMap(tokens);\n    }\n\n    /**\n     * Parses the refl map.\n     *\n     * @param tokens list of tokens for the refl direcive\n     */\n    parse_refl(tokens: string[]) {\n        this.currentMaterial.mapReflections.push(this.parseMap(tokens));\n    }\n\n    /**\n     * Parses the MTL file.\n     *\n     * Iterates line by line parsing each MTL directive.\n     *\n     * This function expects the first token in the line\n     * to be a valid MTL directive. That token is then used\n     * to try and run a method on this class. parse_[directive]\n     * E.g., the `newmtl` directive would try to call the method\n     * parse_newmtl. Each parsing function takes in the remaining\n     * list of tokens and updates the currentMaterial class with\n     * the attributes provided.\n     */\n    parse() {\n        const lines = this.data.split(/\\r?\\n/);\n        for (let line of lines) {\n            line = line.trim();\n            if (!line || line.startsWith(\"#\")) {\n                continue;\n            }\n\n            const [directive, ...tokens] = line.split(/\\s/);\n\n            const parseMethod = (this as any)[`parse_${directive}`];\n\n            if (!parseMethod) {\n                console.warn(`Don't know how to parse the directive: \"${directive}\"`);\n                continue;\n            }\n\n            // console.log(`Parsing \"${directive}\" with tokens: ${tokens}`);\n            parseMethod.bind(this)(tokens);\n        }\n\n        // some cleanup. These don't need to be exposed as public data.\n        delete this.data;\n        this.currentMaterial = SENTINEL_MATERIAL;\n    }\n\n    /* eslint-enable camelcase*/\n}\n\nfunction emptyTextureOptions(): TextureMapData {\n    return {\n        colorCorrection: false,\n        horizontalBlending: true,\n        verticalBlending: true,\n        boostMipMapSharpness: 0,\n        modifyTextureMap: {\n            brightness: 0,\n            contrast: 1,\n        },\n        offset: { u: 0, v: 0, w: 0 },\n        scale: { u: 1, v: 1, w: 1 },\n        turbulence: { u: 0, v: 0, w: 0 },\n        clamp: false,\n        textureResolution: null,\n        bumpMultiplier: 1,\n        imfChan: null,\n        filename: \"\",\n    };\n}\n","import { Layout } from \"./layout\";\nimport { Material, MaterialLibrary } from \"./material\";\n\nexport interface MeshOptions {\n    enableWTextureCoord?: boolean;\n    calcTangentsAndBitangents?: boolean;\n    materials?: { [key: string]: Material };\n}\n\ninterface UnpackedAttrs {\n    verts: number[];\n    norms: number[];\n    textures: number[];\n    hashindices: { [k: string]: number };\n    indices: number[][];\n    materialIndices: number[];\n    index: number;\n}\n\nexport interface MaterialNameToIndex {\n    [k: string]: number;\n}\n\nexport interface IndexToMaterial {\n    [k: number]: Material;\n}\n\nexport interface ArrayBufferWithItemSize extends ArrayBuffer {\n    numItems?: number;\n}\n\nexport interface Uint16ArrayWithItemSize extends Uint16Array {\n    numItems?: number;\n}\n\n/**\n * The main Mesh class. The constructor will parse through the OBJ file data\n * and collect the vertex, vertex normal, texture, and face information. This\n * information can then be used later on when creating your VBOs. See\n * OBJ.initMeshBuffers for an example of how to use the newly created Mesh\n */\nexport default class Mesh {\n    public vertices: number[];\n    public vertexNormals: number[];\n    public textures: number[];\n    public indices: number[];\n    public name: string = \"\";\n    public vertexMaterialIndices: number[];\n    public indicesPerMaterial: number[][] = [];\n    public materialNames: string[];\n    public materialIndices: MaterialNameToIndex;\n    public materialsByIndex: IndexToMaterial = {};\n    public tangents: number[] = [];\n    public bitangents: number[] = [];\n    public textureStride: number;\n\n    /**\n     * Create a Mesh\n     * @param {String} objectData - a string representation of an OBJ file with\n     *     newlines preserved.\n     * @param {Object} options - a JS object containing valid options. See class\n     *     documentation for options.\n     * @param {bool} options.enableWTextureCoord - Texture coordinates can have\n     *     an optional \"w\" coordinate after the u and v coordinates. This extra\n     *     value can be used in order to perform fancy transformations on the\n     *     textures themselves. Default is to truncate to only the u an v\n     *     coordinates. Passing true will provide a default value of 0 in the\n     *     event that any or all texture coordinates don't provide a w value.\n     *     Always use the textureStride attribute in order to determine the\n     *     stride length of the texture coordinates when rendering the element\n     *     array.\n     * @param {bool} options.calcTangentsAndBitangents - Calculate the tangents\n     *     and bitangents when loading of the OBJ is completed. This adds two new\n     *     attributes to the Mesh instance: `tangents` and `bitangents`.\n     */\n    constructor(objectData: string, options?: MeshOptions) {\n        options = options || {};\n        options.materials = options.materials || {};\n        options.enableWTextureCoord = !!options.enableWTextureCoord;\n\n        // the list of unique vertex, normal, texture, attributes\n        this.vertexNormals = [];\n        this.textures = [];\n        // the indicies to draw the faces\n        this.indices = [];\n        this.textureStride = options.enableWTextureCoord ? 3 : 2;\n\n        /*\n        The OBJ file format does a sort of compression when saving a model in a\n        program like Blender. There are at least 3 sections (4 including textures)\n        within the file. Each line in a section begins with the same string:\n          * 'v': indicates vertex section\n          * 'vn': indicates vertex normal section\n          * 'f': indicates the faces section\n          * 'vt': indicates vertex texture section (if textures were used on the model)\n        Each of the above sections (except for the faces section) is a list/set of\n        unique vertices.\n\n        Each line of the faces section contains a list of\n        (vertex, [texture], normal) groups.\n\n        **Note:** The following documentation will use a capital \"V\" Vertex to\n        denote the above (vertex, [texture], normal) groups whereas a lowercase\n        \"v\" vertex is used to denote an X, Y, Z coordinate.\n\n        Some examples:\n            // the texture index is optional, both formats are possible for models\n            // without a texture applied\n            f 1/25 18/46 12/31\n            f 1//25 18//46 12//31\n\n            // A 3 vertex face with texture indices\n            f 16/92/11 14/101/22 1/69/1\n\n            // A 4 vertex face\n            f 16/92/11 40/109/40 38/114/38 14/101/22\n\n        The first two lines are examples of a 3 vertex face without a texture applied.\n        The second is an example of a 3 vertex face with a texture applied.\n        The third is an example of a 4 vertex face. Note: a face can contain N\n        number of vertices.\n\n        Each number that appears in one of the groups is a 1-based index\n        corresponding to an item from the other sections (meaning that indexing\n        starts at one and *not* zero).\n\n        For example:\n            `f 16/92/11` is saying to\n              - take the 16th element from the [v] vertex array\n              - take the 92nd element from the [vt] texture array\n              - take the 11th element from the [vn] normal array\n            and together they make a unique vertex.\n        Using all 3+ unique Vertices from the face line will produce a polygon.\n\n        Now, you could just go through the OBJ file and create a new vertex for\n        each face line and WebGL will draw what appears to be the same model.\n        However, vertices will be overlapped and duplicated all over the place.\n\n        Consider a cube in 3D space centered about the origin and each side is\n        2 units long. The front face (with the positive Z-axis pointing towards\n        you) would have a Top Right vertex (looking orthogonal to its normal)\n        mapped at (1,1,1) The right face would have a Top Left vertex (looking\n        orthogonal to its normal) at (1,1,1) and the top face would have a Bottom\n        Right vertex (looking orthogonal to its normal) at (1,1,1). Each face\n        has a vertex at the same coordinates, however, three distinct vertices\n        will be drawn at the same spot.\n\n        To solve the issue of duplicate Vertices (the `(vertex, [texture], normal)`\n        groups), while iterating through the face lines, when a group is encountered\n        the whole group string ('16/92/11') is checked to see if it exists in the\n        packed.hashindices object, and if it doesn't, the indices it specifies\n        are used to look up each attribute in the corresponding attribute arrays\n        already created. The values are then copied to the corresponding unpacked\n        array (flattened to play nice with WebGL's ELEMENT_ARRAY_BUFFER indexing),\n        the group string is added to the hashindices set and the current unpacked\n        index is used as this hashindices value so that the group of elements can\n        be reused. The unpacked index is incremented. If the group string already\n        exists in the hashindices object, its corresponding value is the index of\n        that group and is appended to the unpacked indices array.\n       */\n        const verts = [];\n        const vertNormals = [];\n        const textures = [];\n        const materialNamesByIndex = [];\n        const materialIndicesByName: MaterialNameToIndex = {};\n        // keep track of what material we've seen last\n        let currentMaterialIndex = -1;\n        let currentObjectByMaterialIndex = 0;\n        // unpacking stuff\n        const unpacked: UnpackedAttrs = {\n            verts: [],\n            norms: [],\n            textures: [],\n            hashindices: {},\n            indices: [[]],\n            materialIndices: [],\n            index: 0,\n        };\n\n        const VERTEX_RE = /^v\\s/;\n        const NORMAL_RE = /^vn\\s/;\n        const TEXTURE_RE = /^vt\\s/;\n        const FACE_RE = /^f\\s/;\n        const WHITESPACE_RE = /\\s+/;\n        const USE_MATERIAL_RE = /^usemtl/;\n\n        // array of lines separated by the newline\n        const lines = objectData.split(\"\\n\");\n\n        for (let line of lines) {\n            line = line.trim();\n            if (!line || line.startsWith(\"#\")) {\n                continue;\n            }\n            const elements = line.split(WHITESPACE_RE);\n            elements.shift();\n\n            if (VERTEX_RE.test(line)) {\n                // if this is a vertex\n                verts.push(...elements);\n            } else if (NORMAL_RE.test(line)) {\n                // if this is a vertex normal\n                vertNormals.push(...elements);\n            } else if (TEXTURE_RE.test(line)) {\n                let coords = elements;\n                // by default, the loader will only look at the U and V\n                // coordinates of the vt declaration. So, this truncates the\n                // elements to only those 2 values. If W texture coordinate\n                // support is enabled, then the texture coordinate is\n                // expected to have three values in it.\n                if (elements.length > 2 && !options.enableWTextureCoord) {\n                    coords = elements.slice(0, 2);\n                } else if (elements.length === 2 && options.enableWTextureCoord) {\n                    // If for some reason W texture coordinate support is enabled\n                    // and only the U and V coordinates are given, then we supply\n                    // the default value of 0 so that the stride length is correct\n                    // when the textures are unpacked below.\n                    coords.push(\"0\");\n                }\n                textures.push(...coords);\n            } else if (USE_MATERIAL_RE.test(line)) {\n                const materialName = elements[0];\n\n                // check to see if we've ever seen it before\n                if (!(materialName in materialIndicesByName)) {\n                    // new material we've never seen\n                    materialNamesByIndex.push(materialName);\n                    materialIndicesByName[materialName] = materialNamesByIndex.length - 1;\n                    // push new array into indices\n                    // already contains an array at index zero, don't add\n                    if (materialIndicesByName[materialName] > 0) {\n                        unpacked.indices.push([]);\n                    }\n                }\n                // keep track of the current material index\n                currentMaterialIndex = materialIndicesByName[materialName];\n                // update current index array\n                currentObjectByMaterialIndex = currentMaterialIndex;\n            } else if (FACE_RE.test(line)) {\n                // if this is a face\n                /*\n                split this face into an array of Vertex groups\n                for example:\n                   f 16/92/11 14/101/22 1/69/1\n                becomes:\n                  ['16/92/11', '14/101/22', '1/69/1'];\n                */\n\n                const triangles = triangulate(elements);\n                for (const triangle of triangles) {\n                    for (let j = 0, eleLen = triangle.length; j < eleLen; j++) {\n                        const hash = triangle[j] + \",\" + currentMaterialIndex;\n                        if (hash in unpacked.hashindices) {\n                            unpacked.indices[currentObjectByMaterialIndex].push(unpacked.hashindices[hash]);\n                        } else {\n                            /*\n                        Each element of the face line array is a Vertex which has its\n                        attributes delimited by a forward slash. This will separate\n                        each attribute into another array:\n                            '19/92/11'\n                        becomes:\n                            Vertex = ['19', '92', '11'];\n                        where\n                            Vertex[0] is the vertex index\n                            Vertex[1] is the texture index\n                            Vertex[2] is the normal index\n                         Think of faces having Vertices which are comprised of the\n                         attributes location (v), texture (vt), and normal (vn).\n                         */\n                            const vertex = elements[j].split(\"/\");\n                            // it's possible for faces to only specify the vertex\n                            // and the normal. In this case, vertex will only have\n                            // a length of 2 and not 3 and the normal will be the\n                            // second item in the list with an index of 1.\n                            const normalIndex = vertex.length - 1;\n                            /*\n                         The verts, textures, and vertNormals arrays each contain a\n                         flattend array of coordinates.\n\n                         Because it gets confusing by referring to Vertex and then\n                         vertex (both are different in my descriptions) I will explain\n                         what's going on using the vertexNormals array:\n\n                         vertex[2] will contain the one-based index of the vertexNormals\n                         section (vn). One is subtracted from this index number to play\n                         nice with javascript's zero-based array indexing.\n\n                         Because vertexNormal is a flattened array of x, y, z values,\n                         simple pointer arithmetic is used to skip to the start of the\n                         vertexNormal, then the offset is added to get the correct\n                         component: +0 is x, +1 is y, +2 is z.\n\n                         This same process is repeated for verts and textures.\n                         */\n                            // Vertex position\n                            unpacked.verts.push(+verts[(+vertex[0] - 1) * 3 + 0]);\n                            unpacked.verts.push(+verts[(+vertex[0] - 1) * 3 + 1]);\n                            unpacked.verts.push(+verts[(+vertex[0] - 1) * 3 + 2]);\n                            // Vertex textures\n                            if (textures.length) {\n                                const stride = options.enableWTextureCoord ? 3 : 2;\n                                unpacked.textures.push(+textures[(+vertex[1] - 1) * stride + 0]);\n                                unpacked.textures.push(+textures[(+vertex[1] - 1) * stride + 1]);\n                                if (options.enableWTextureCoord) {\n                                    unpacked.textures.push(+textures[(+vertex[1] - 1) * stride + 2]);\n                                }\n                            }\n                            // Vertex normals\n                            unpacked.norms.push(+vertNormals[(+vertex[normalIndex] - 1) * 3 + 0]);\n                            unpacked.norms.push(+vertNormals[(+vertex[normalIndex] - 1) * 3 + 1]);\n                            unpacked.norms.push(+vertNormals[(+vertex[normalIndex] - 1) * 3 + 2]);\n                            // Vertex material indices\n                            unpacked.materialIndices.push(currentMaterialIndex);\n                            // add the newly created Vertex to the list of indices\n                            unpacked.hashindices[hash] = unpacked.index;\n                            unpacked.indices[currentObjectByMaterialIndex].push(unpacked.hashindices[hash]);\n                            // increment the counter\n                            unpacked.index += 1;\n                        }\n                    }\n                }\n            }\n        }\n        this.vertices = unpacked.verts;\n        this.vertexNormals = unpacked.norms;\n        this.textures = unpacked.textures;\n        this.vertexMaterialIndices = unpacked.materialIndices;\n        this.indices = unpacked.indices[currentObjectByMaterialIndex];\n        this.indicesPerMaterial = unpacked.indices;\n\n        this.materialNames = materialNamesByIndex;\n        this.materialIndices = materialIndicesByName;\n        this.materialsByIndex = {};\n\n        if (options.calcTangentsAndBitangents) {\n            this.calculateTangentsAndBitangents();\n        }\n    }\n\n    /**\n     * Calculates the tangents and bitangents of the mesh that forms an orthogonal basis together with the\n     * normal in the direction of the texture coordinates. These are useful for setting up the TBN matrix\n     * when distorting the normals through normal maps.\n     * Method derived from: http://www.opengl-tutorial.org/intermediate-tutorials/tutorial-13-normal-mapping/\n     *\n     * This method requires the normals and texture coordinates to be parsed and set up correctly.\n     * Adds the tangents and bitangents as members of the class instance.\n     */\n    calculateTangentsAndBitangents() {\n        console.assert(\n            !!(\n                this.vertices &&\n                this.vertices.length &&\n                this.vertexNormals &&\n                this.vertexNormals.length &&\n                this.textures &&\n                this.textures.length\n            ),\n            \"Missing attributes for calculating tangents and bitangents\",\n        );\n\n        const unpacked = {\n            tangents: [...new Array(this.vertices.length)].map(_ => 0),\n            bitangents: [...new Array(this.vertices.length)].map(_ => 0),\n        };\n\n        // Loop through all faces in the whole mesh\n        const indices = this.indices;\n        const vertices = this.vertices;\n        const normals = this.vertexNormals;\n        const uvs = this.textures;\n\n        for (let i = 0; i < indices.length; i += 3) {\n            const i0 = indices[i + 0];\n            const i1 = indices[i + 1];\n            const i2 = indices[i + 2];\n\n            const x_v0 = vertices[i0 * 3 + 0];\n            const y_v0 = vertices[i0 * 3 + 1];\n            const z_v0 = vertices[i0 * 3 + 2];\n\n            const x_uv0 = uvs[i0 * 2 + 0];\n            const y_uv0 = uvs[i0 * 2 + 1];\n\n            const x_v1 = vertices[i1 * 3 + 0];\n            const y_v1 = vertices[i1 * 3 + 1];\n            const z_v1 = vertices[i1 * 3 + 2];\n\n            const x_uv1 = uvs[i1 * 2 + 0];\n            const y_uv1 = uvs[i1 * 2 + 1];\n\n            const x_v2 = vertices[i2 * 3 + 0];\n            const y_v2 = vertices[i2 * 3 + 1];\n            const z_v2 = vertices[i2 * 3 + 2];\n\n            const x_uv2 = uvs[i2 * 2 + 0];\n            const y_uv2 = uvs[i2 * 2 + 1];\n\n            const x_deltaPos1 = x_v1 - x_v0;\n            const y_deltaPos1 = y_v1 - y_v0;\n            const z_deltaPos1 = z_v1 - z_v0;\n\n            const x_deltaPos2 = x_v2 - x_v0;\n            const y_deltaPos2 = y_v2 - y_v0;\n            const z_deltaPos2 = z_v2 - z_v0;\n\n            const x_uvDeltaPos1 = x_uv1 - x_uv0;\n            const y_uvDeltaPos1 = y_uv1 - y_uv0;\n\n            const x_uvDeltaPos2 = x_uv2 - x_uv0;\n            const y_uvDeltaPos2 = y_uv2 - y_uv0;\n\n            const rInv = x_uvDeltaPos1 * y_uvDeltaPos2 - y_uvDeltaPos1 * x_uvDeltaPos2;\n            const r = 1.0 / Math.abs(rInv < 0.0001 ? 1.0 : rInv);\n\n            // Tangent\n            const x_tangent = (x_deltaPos1 * y_uvDeltaPos2 - x_deltaPos2 * y_uvDeltaPos1) * r;\n            const y_tangent = (y_deltaPos1 * y_uvDeltaPos2 - y_deltaPos2 * y_uvDeltaPos1) * r;\n            const z_tangent = (z_deltaPos1 * y_uvDeltaPos2 - z_deltaPos2 * y_uvDeltaPos1) * r;\n\n            // Bitangent\n            const x_bitangent = (x_deltaPos2 * x_uvDeltaPos1 - x_deltaPos1 * x_uvDeltaPos2) * r;\n            const y_bitangent = (y_deltaPos2 * x_uvDeltaPos1 - y_deltaPos1 * x_uvDeltaPos2) * r;\n            const z_bitangent = (z_deltaPos2 * x_uvDeltaPos1 - z_deltaPos1 * x_uvDeltaPos2) * r;\n\n            // Gram-Schmidt orthogonalize\n            //t = glm::normalize(t - n * glm:: dot(n, t));\n            const x_n0 = normals[i0 * 3 + 0];\n            const y_n0 = normals[i0 * 3 + 1];\n            const z_n0 = normals[i0 * 3 + 2];\n\n            const x_n1 = normals[i1 * 3 + 0];\n            const y_n1 = normals[i1 * 3 + 1];\n            const z_n1 = normals[i1 * 3 + 2];\n\n            const x_n2 = normals[i2 * 3 + 0];\n            const y_n2 = normals[i2 * 3 + 1];\n            const z_n2 = normals[i2 * 3 + 2];\n\n            // Tangent\n            const n0_dot_t = x_tangent * x_n0 + y_tangent * y_n0 + z_tangent * z_n0;\n            const n1_dot_t = x_tangent * x_n1 + y_tangent * y_n1 + z_tangent * z_n1;\n            const n2_dot_t = x_tangent * x_n2 + y_tangent * y_n2 + z_tangent * z_n2;\n\n            const x_resTangent0 = x_tangent - x_n0 * n0_dot_t;\n            const y_resTangent0 = y_tangent - y_n0 * n0_dot_t;\n            const z_resTangent0 = z_tangent - z_n0 * n0_dot_t;\n\n            const x_resTangent1 = x_tangent - x_n1 * n1_dot_t;\n            const y_resTangent1 = y_tangent - y_n1 * n1_dot_t;\n            const z_resTangent1 = z_tangent - z_n1 * n1_dot_t;\n\n            const x_resTangent2 = x_tangent - x_n2 * n2_dot_t;\n            const y_resTangent2 = y_tangent - y_n2 * n2_dot_t;\n            const z_resTangent2 = z_tangent - z_n2 * n2_dot_t;\n\n            const magTangent0 = Math.sqrt(\n                x_resTangent0 * x_resTangent0 + y_resTangent0 * y_resTangent0 + z_resTangent0 * z_resTangent0,\n            );\n            const magTangent1 = Math.sqrt(\n                x_resTangent1 * x_resTangent1 + y_resTangent1 * y_resTangent1 + z_resTangent1 * z_resTangent1,\n            );\n            const magTangent2 = Math.sqrt(\n                x_resTangent2 * x_resTangent2 + y_resTangent2 * y_resTangent2 + z_resTangent2 * z_resTangent2,\n            );\n\n            // Bitangent\n            const n0_dot_bt = x_bitangent * x_n0 + y_bitangent * y_n0 + z_bitangent * z_n0;\n            const n1_dot_bt = x_bitangent * x_n1 + y_bitangent * y_n1 + z_bitangent * z_n1;\n            const n2_dot_bt = x_bitangent * x_n2 + y_bitangent * y_n2 + z_bitangent * z_n2;\n\n            const x_resBitangent0 = x_bitangent - x_n0 * n0_dot_bt;\n            const y_resBitangent0 = y_bitangent - y_n0 * n0_dot_bt;\n            const z_resBitangent0 = z_bitangent - z_n0 * n0_dot_bt;\n\n            const x_resBitangent1 = x_bitangent - x_n1 * n1_dot_bt;\n            const y_resBitangent1 = y_bitangent - y_n1 * n1_dot_bt;\n            const z_resBitangent1 = z_bitangent - z_n1 * n1_dot_bt;\n\n            const x_resBitangent2 = x_bitangent - x_n2 * n2_dot_bt;\n            const y_resBitangent2 = y_bitangent - y_n2 * n2_dot_bt;\n            const z_resBitangent2 = z_bitangent - z_n2 * n2_dot_bt;\n\n            const magBitangent0 = Math.sqrt(\n                x_resBitangent0 * x_resBitangent0 +\n                    y_resBitangent0 * y_resBitangent0 +\n                    z_resBitangent0 * z_resBitangent0,\n            );\n            const magBitangent1 = Math.sqrt(\n                x_resBitangent1 * x_resBitangent1 +\n                    y_resBitangent1 * y_resBitangent1 +\n                    z_resBitangent1 * z_resBitangent1,\n            );\n            const magBitangent2 = Math.sqrt(\n                x_resBitangent2 * x_resBitangent2 +\n                    y_resBitangent2 * y_resBitangent2 +\n                    z_resBitangent2 * z_resBitangent2,\n            );\n\n            unpacked.tangents[i0 * 3 + 0] += x_resTangent0 / magTangent0;\n            unpacked.tangents[i0 * 3 + 1] += y_resTangent0 / magTangent0;\n            unpacked.tangents[i0 * 3 + 2] += z_resTangent0 / magTangent0;\n\n            unpacked.tangents[i1 * 3 + 0] += x_resTangent1 / magTangent1;\n            unpacked.tangents[i1 * 3 + 1] += y_resTangent1 / magTangent1;\n            unpacked.tangents[i1 * 3 + 2] += z_resTangent1 / magTangent1;\n\n            unpacked.tangents[i2 * 3 + 0] += x_resTangent2 / magTangent2;\n            unpacked.tangents[i2 * 3 + 1] += y_resTangent2 / magTangent2;\n            unpacked.tangents[i2 * 3 + 2] += z_resTangent2 / magTangent2;\n\n            unpacked.bitangents[i0 * 3 + 0] += x_resBitangent0 / magBitangent0;\n            unpacked.bitangents[i0 * 3 + 1] += y_resBitangent0 / magBitangent0;\n            unpacked.bitangents[i0 * 3 + 2] += z_resBitangent0 / magBitangent0;\n\n            unpacked.bitangents[i1 * 3 + 0] += x_resBitangent1 / magBitangent1;\n            unpacked.bitangents[i1 * 3 + 1] += y_resBitangent1 / magBitangent1;\n            unpacked.bitangents[i1 * 3 + 2] += z_resBitangent1 / magBitangent1;\n\n            unpacked.bitangents[i2 * 3 + 0] += x_resBitangent2 / magBitangent2;\n            unpacked.bitangents[i2 * 3 + 1] += y_resBitangent2 / magBitangent2;\n            unpacked.bitangents[i2 * 3 + 2] += z_resBitangent2 / magBitangent2;\n\n            // TODO: check handedness\n        }\n\n        this.tangents = unpacked.tangents;\n        this.bitangents = unpacked.bitangents;\n    }\n\n    /**\n     * @param layout - A {@link Layout} object that describes the\n     * desired memory layout of the generated buffer\n     * @return The packed array in the ... TODO\n     */\n    makeBufferData(layout: Layout): ArrayBufferWithItemSize {\n        const numItems = this.vertices.length / 3;\n        const buffer: ArrayBufferWithItemSize = new ArrayBuffer(layout.stride * numItems);\n        buffer.numItems = numItems;\n        const dataView = new DataView(buffer);\n        for (let i = 0, vertexOffset = 0; i < numItems; i++) {\n            vertexOffset = i * layout.stride;\n            // copy in the vertex data in the order and format given by the\n            // layout param\n            for (const attribute of layout.attributes) {\n                const offset = vertexOffset + layout.attributeMap[attribute.key].offset;\n                switch (attribute.key) {\n                    case Layout.POSITION.key:\n                        dataView.setFloat32(offset, this.vertices[i * 3], true);\n                        dataView.setFloat32(offset + 4, this.vertices[i * 3 + 1], true);\n                        dataView.setFloat32(offset + 8, this.vertices[i * 3 + 2], true);\n                        break;\n                    case Layout.UV.key:\n                        dataView.setFloat32(offset, this.textures[i * 2], true);\n                        dataView.setFloat32(offset + 4, this.textures[i * 2 + 1], true);\n                        break;\n                    case Layout.NORMAL.key:\n                        dataView.setFloat32(offset, this.vertexNormals[i * 3], true);\n                        dataView.setFloat32(offset + 4, this.vertexNormals[i * 3 + 1], true);\n                        dataView.setFloat32(offset + 8, this.vertexNormals[i * 3 + 2], true);\n                        break;\n                    case Layout.MATERIAL_INDEX.key:\n                        dataView.setInt16(offset, this.vertexMaterialIndices[i], true);\n                        break;\n                    case Layout.AMBIENT.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setFloat32(offset, material.ambient[0], true);\n                        dataView.setFloat32(offset + 4, material.ambient[1], true);\n                        dataView.setFloat32(offset + 8, material.ambient[2], true);\n                        break;\n                    }\n                    case Layout.DIFFUSE.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setFloat32(offset, material.diffuse[0], true);\n                        dataView.setFloat32(offset + 4, material.diffuse[1], true);\n                        dataView.setFloat32(offset + 8, material.diffuse[2], true);\n                        break;\n                    }\n                    case Layout.SPECULAR.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setFloat32(offset, material.specular[0], true);\n                        dataView.setFloat32(offset + 4, material.specular[1], true);\n                        dataView.setFloat32(offset + 8, material.specular[2], true);\n                        break;\n                    }\n                    case Layout.SPECULAR_EXPONENT.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setFloat32(offset, material.specularExponent, true);\n                        break;\n                    }\n                    case Layout.EMISSIVE.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setFloat32(offset, material.emissive[0], true);\n                        dataView.setFloat32(offset + 4, material.emissive[1], true);\n                        dataView.setFloat32(offset + 8, material.emissive[2], true);\n                        break;\n                    }\n                    case Layout.TRANSMISSION_FILTER.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setFloat32(offset, material.transmissionFilter[0], true);\n                        dataView.setFloat32(offset + 4, material.transmissionFilter[1], true);\n                        dataView.setFloat32(offset + 8, material.transmissionFilter[2], true);\n                        break;\n                    }\n                    case Layout.DISSOLVE.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setFloat32(offset, material.dissolve, true);\n                        break;\n                    }\n                    case Layout.ILLUMINATION.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setInt16(offset, material.illumination, true);\n                        break;\n                    }\n                    case Layout.REFRACTION_INDEX.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setFloat32(offset, material.refractionIndex, true);\n                        break;\n                    }\n                    case Layout.SHARPNESS.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setFloat32(offset, material.sharpness, true);\n                        break;\n                    }\n                    case Layout.ANTI_ALIASING.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setInt16(offset, material.antiAliasing ? 1 : 0, true);\n                        break;\n                    }\n                }\n            }\n        }\n        return buffer;\n    }\n\n    makeIndexBufferData(): Uint16ArrayWithItemSize {\n        const buffer: Uint16ArrayWithItemSize = new Uint16Array(this.indices);\n        buffer.numItems = this.indices.length;\n        return buffer;\n    }\n\n    addMaterialLibrary(mtl: MaterialLibrary) {\n        for (const name in mtl.materials) {\n            if (!(name in this.materialIndices)) {\n                // This material is not referenced by the mesh\n                continue;\n            }\n\n            const material = mtl.materials[name];\n\n            // Find the material index for this material\n            const materialIndex = this.materialIndices[material.name];\n\n            // Put the material into the materialsByIndex object at the right\n            // spot as determined when the obj file was parsed\n            this.materialsByIndex[materialIndex] = material;\n        }\n    }\n}\n\nfunction* triangulate(elements: string[]) {\n    if (elements.length <= 3) {\n        yield elements;\n    } else if (elements.length === 4) {\n        yield [elements[0], elements[1], elements[2]];\n        yield [elements[2], elements[3], elements[0]];\n    } else {\n        for (let i = 1; i < elements.length - 1; i++) {\n            yield [elements[0], elements[i], elements[i + 1]];\n        }\n    }\n}\n","import Mesh from \"./mesh\";\nimport { MaterialLibrary, TextureMapData } from \"./material\";\n\nfunction downloadMtlTextures(mtl: MaterialLibrary, root: string) {\n    const mapAttributes = [\n        \"mapDiffuse\",\n        \"mapAmbient\",\n        \"mapSpecular\",\n        \"mapDissolve\",\n        \"mapBump\",\n        \"mapDisplacement\",\n        \"mapDecal\",\n        \"mapEmissive\",\n    ];\n    if (!root.endsWith(\"/\")) {\n        root += \"/\";\n    }\n    const textures = [];\n\n    for (const materialName in mtl.materials) {\n        if (!mtl.materials.hasOwnProperty(materialName)) {\n            continue;\n        }\n        const material = mtl.materials[materialName];\n\n        for (const attr of mapAttributes) {\n            const mapData = (material as any)[attr] as TextureMapData;\n            if (!mapData) {\n                continue;\n            }\n\n            const url = root + mapData.filename;\n            textures.push(\n                fetch(url)\n                    .then(response => {\n                        if (!response.ok) {\n                            throw new Error();\n                        }\n                        return response.blob();\n                    })\n                    .then(function(data) {\n                        const image = new Image();\n                        image.src = URL.createObjectURL(data);\n                        mapData.texture = image;\n                        return new Promise(resolve => (image.onload = resolve));\n                    })\n                    .catch(() => {\n                        console.error(`Unable to download texture: ${url}`);\n                    }),\n            );\n        }\n    }\n\n    return Promise.all(textures);\n}\n\nfunction getMtl(modelOptions: DownloadModelsOptions): string {\n    if (!(typeof modelOptions.mtl === \"string\")) {\n        return modelOptions.obj.replace(/\\.obj$/, \".mtl\");\n    }\n\n    return modelOptions.mtl;\n}\n\nexport interface DownloadModelsOptions {\n    obj: string;\n    mtl?: boolean | string;\n    downloadMtlTextures?: boolean;\n    mtlTextureRoot?: string;\n    name?: string;\n    indicesPerMaterial?: boolean;\n    calcTangentsAndBitangents?: boolean;\n}\n\ntype ModelPromises = [Promise<string>, Promise<Mesh>, undefined | Promise<MaterialLibrary>];\nexport type MeshMap = { [name: string]: Mesh };\n/**\n * Accepts a list of model request objects and returns a Promise that\n * resolves when all models have been downloaded and parsed.\n *\n * The list of model objects follow this interface:\n * {\n *  obj: 'path/to/model.obj',\n *  mtl: true | 'path/to/model.mtl',\n *  downloadMtlTextures: true | false\n *  mtlTextureRoot: '/models/suzanne/maps'\n *  name: 'suzanne'\n * }\n *\n * The `obj` attribute is required and should be the path to the\n * model's .obj file relative to the current repo (absolute URLs are\n * suggested).\n *\n * The `mtl` attribute is optional and can either be a boolean or\n * a path to the model's .mtl file relative to the current URL. If\n * the value is `true`, then the path and basename given for the `obj`\n * attribute is used replacing the .obj suffix for .mtl\n * E.g.: {obj: 'models/foo.obj', mtl: true} would search for 'models/foo.mtl'\n *\n * The `name` attribute is optional and is a human friendly name to be\n * included with the parsed OBJ and MTL files. If not given, the base .obj\n * filename will be used.\n *\n * The `downloadMtlTextures` attribute is a flag for automatically downloading\n * any images found in the MTL file and attaching them to each Material\n * created from that file. For example, if material.mapDiffuse is set (there\n * was data in the MTL file), then material.mapDiffuse.texture will contain\n * the downloaded image. This option defaults to `true`. By default, the MTL's\n * URL will be used to determine the location of the images.\n *\n * The `mtlTextureRoot` attribute is optional and should point to the location\n * on the server that this MTL's texture files are located. The default is to\n * use the MTL file's location.\n *\n * @returns {Promise} the result of downloading the given list of models. The\n * promise will resolve with an object whose keys are the names of the models\n * and the value is its Mesh object. Each Mesh object will automatically\n * have its addMaterialLibrary() method called to set the given MTL data (if given).\n */\nexport function downloadModels(models: DownloadModelsOptions[]): Promise<MeshMap> {\n    const finished = [];\n\n    for (const model of models) {\n        if (!model.obj) {\n            throw new Error(\n                '\"obj\" attribute of model object not set. The .obj file is required to be set ' +\n                    \"in order to use downloadModels()\",\n            );\n        }\n\n        const options = {\n            indicesPerMaterial: !!model.indicesPerMaterial,\n            calcTangentsAndBitangents: !!model.calcTangentsAndBitangents,\n        };\n\n        // if the name is not provided, dervive it from the given OBJ\n        let name = model.name;\n        if (!name) {\n            const parts = model.obj.split(\"/\");\n            name = parts[parts.length - 1].replace(\".obj\", \"\");\n        }\n        const namePromise = Promise.resolve(name);\n\n        const meshPromise = fetch(model.obj)\n            .then(response => response.text())\n            .then(data => {\n                return new Mesh(data, options);\n            });\n\n        let mtlPromise;\n        // Download MaterialLibrary file?\n        if (model.mtl) {\n            const mtl = getMtl(model);\n            mtlPromise = fetch(mtl)\n                .then(response => response.text())\n                .then(\n                    (data: string): Promise<[MaterialLibrary, any]> => {\n                        const material = new MaterialLibrary(data);\n                        if (model.downloadMtlTextures !== false) {\n                            let root = model.mtlTextureRoot;\n                            if (!root) {\n                                // get the directory of the MTL file as default\n                                root = mtl.substr(0, mtl.lastIndexOf(\"/\"));\n                            }\n                            // downloadMtlTextures returns a Promise that\n                            // is resolved once all of the images it\n                            // contains are downloaded. These are then\n                            // attached to the map data objects\n                            return Promise.all([Promise.resolve(material), downloadMtlTextures(material, root)]);\n                        }\n                        return Promise.all([Promise.resolve(material), undefined]);\n                    },\n                )\n                .then((value: [MaterialLibrary, any]) => {\n                    return value[0];\n                });\n        }\n\n        const parsed: ModelPromises = [namePromise, meshPromise, mtlPromise];\n        finished.push(Promise.all<string, Mesh, MaterialLibrary | undefined>(parsed));\n    }\n\n    return Promise.all(finished).then(ms => {\n        // the \"finished\" promise is a list of name, Mesh instance,\n        // and MaterialLibary instance. This unpacks and returns an\n        // object mapping name to Mesh (Mesh points to MTL).\n        const models: MeshMap = {};\n\n        for (const model of ms) {\n            const [name, mesh, mtl] = model;\n            mesh.name = name;\n            if (mtl) {\n                mesh.addMaterialLibrary(mtl);\n            }\n            models[name] = mesh;\n        }\n\n        return models;\n    });\n}\n\nexport interface NameAndUrls {\n    [meshName: string]: string;\n}\n\n/**\n * Takes in an object of `mesh_name`, `'/url/to/OBJ/file'` pairs and a callback\n * function. Each OBJ file will be ajaxed in and automatically converted to\n * an OBJ.Mesh. When all files have successfully downloaded the callback\n * function provided will be called and passed in an object containing\n * the newly created meshes.\n *\n * **Note:** In order to use this function as a way to download meshes, a\n * webserver of some sort must be used.\n *\n * @param {Object} nameAndAttrs an object where the key is the name of the mesh and the value is the url to that mesh's OBJ file\n *\n * @param {Function} completionCallback should contain a function that will take one parameter: an object array where the keys will be the unique object name and the value will be a Mesh object\n *\n * @param {Object} meshes In case other meshes are loaded separately or if a previously declared variable is desired to be used, pass in a (possibly empty) json object of the pattern: { '<mesh_name>': OBJ.Mesh }\n *\n */\nexport function downloadMeshes(\n    nameAndURLs: NameAndUrls,\n    completionCallback: (meshes: MeshMap) => void,\n    meshes: MeshMap,\n) {\n    if (meshes === undefined) {\n        meshes = {};\n    }\n\n    const completed: Promise<[string, Mesh]>[] = [];\n\n    for (const mesh_name in nameAndURLs) {\n        if (!nameAndURLs.hasOwnProperty(mesh_name)) {\n            continue;\n        }\n        const url = nameAndURLs[mesh_name];\n        completed.push(\n            fetch(url)\n                .then(response => response.text())\n                .then(data => {\n                    return [mesh_name, new Mesh(data)] as [string, Mesh];\n                }),\n        );\n    }\n\n    Promise.all(completed).then(ms => {\n        for (const [name, mesh] of ms) {\n            meshes[name] = mesh;\n        }\n\n        return completionCallback(meshes);\n    });\n}\n\nexport interface ExtendedGLBuffer extends WebGLBuffer {\n    itemSize: number;\n    numItems: number;\n}\n\nfunction _buildBuffer(gl: WebGLRenderingContext, type: GLenum, data: number[], itemSize: number): ExtendedGLBuffer {\n    const buffer = gl.createBuffer() as ExtendedGLBuffer;\n    const arrayView = type === gl.ARRAY_BUFFER ? Float32Array : Uint16Array;\n    gl.bindBuffer(type, buffer);\n    gl.bufferData(type, new arrayView(data), gl.STATIC_DRAW);\n    buffer.itemSize = itemSize;\n    buffer.numItems = data.length / itemSize;\n    return buffer;\n}\n\nexport interface MeshWithBuffers extends Mesh {\n    normalBuffer: ExtendedGLBuffer;\n    textureBuffer: ExtendedGLBuffer;\n    vertexBuffer: ExtendedGLBuffer;\n    indexBuffer: ExtendedGLBuffer;\n}\n\n/**\n * Takes in the WebGL context and a Mesh, then creates and appends the buffers\n * to the mesh object as attributes.\n *\n * @param {WebGLRenderingContext} gl the `canvas.getContext('webgl')` context instance\n * @param {Mesh} mesh a single `OBJ.Mesh` instance\n *\n * The newly created mesh attributes are:\n *\n * Attrbute | Description\n * :--- | ---\n * **normalBuffer**       |contains the model&#39;s Vertex Normals\n * normalBuffer.itemSize  |set to 3 items\n * normalBuffer.numItems  |the total number of vertex normals\n * |\n * **textureBuffer**      |contains the model&#39;s Texture Coordinates\n * textureBuffer.itemSize |set to 2 items\n * textureBuffer.numItems |the number of texture coordinates\n * |\n * **vertexBuffer**       |contains the model&#39;s Vertex Position Coordinates (does not include w)\n * vertexBuffer.itemSize  |set to 3 items\n * vertexBuffer.numItems  |the total number of vertices\n * |\n * **indexBuffer**        |contains the indices of the faces\n * indexBuffer.itemSize   |is set to 1\n * indexBuffer.numItems   |the total number of indices\n *\n * A simple example (a lot of steps are missing, so don't copy and paste):\n *\n *     const gl   = canvas.getContext('webgl'),\n *         mesh = OBJ.Mesh(obj_file_data);\n *     // compile the shaders and create a shader program\n *     const shaderProgram = gl.createProgram();\n *     // compilation stuff here\n *     ...\n *     // make sure you have vertex, vertex normal, and texture coordinate\n *     // attributes located in your shaders and attach them to the shader program\n *     shaderProgram.vertexPositionAttribute = gl.getAttribLocation(shaderProgram, \"aVertexPosition\");\n *     gl.enableVertexAttribArray(shaderProgram.vertexPositionAttribute);\n *\n *     shaderProgram.vertexNormalAttribute = gl.getAttribLocation(shaderProgram, \"aVertexNormal\");\n *     gl.enableVertexAttribArray(shaderProgram.vertexNormalAttribute);\n *\n *     shaderProgram.textureCoordAttribute = gl.getAttribLocation(shaderProgram, \"aTextureCoord\");\n *     gl.enableVertexAttribArray(shaderProgram.textureCoordAttribute);\n *\n *     // create and initialize the vertex, vertex normal, and texture coordinate buffers\n *     // and save on to the mesh object\n *     OBJ.initMeshBuffers(gl, mesh);\n *\n *     // now to render the mesh\n *     gl.bindBuffer(gl.ARRAY_BUFFER, mesh.vertexBuffer);\n *     gl.vertexAttribPointer(shaderProgram.vertexPositionAttribute, mesh.vertexBuffer.itemSize, gl.FLOAT, false, 0, 0);\n *     // it's possible that the mesh doesn't contain\n *     // any texture coordinates (e.g. suzanne.obj in the development branch).\n *     // in this case, the texture vertexAttribArray will need to be disabled\n *     // before the call to drawElements\n *     if(!mesh.textures.length){\n *       gl.disableVertexAttribArray(shaderProgram.textureCoordAttribute);\n *     }\n *     else{\n *       // if the texture vertexAttribArray has been previously\n *       // disabled, then it needs to be re-enabled\n *       gl.enableVertexAttribArray(shaderProgram.textureCoordAttribute);\n *       gl.bindBuffer(gl.ARRAY_BUFFER, mesh.textureBuffer);\n *       gl.vertexAttribPointer(shaderProgram.textureCoordAttribute, mesh.textureBuffer.itemSize, gl.FLOAT, false, 0, 0);\n *     }\n *\n *     gl.bindBuffer(gl.ARRAY_BUFFER, mesh.normalBuffer);\n *     gl.vertexAttribPointer(shaderProgram.vertexNormalAttribute, mesh.normalBuffer.itemSize, gl.FLOAT, false, 0, 0);\n *\n *     gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, model.mesh.indexBuffer);\n *     gl.drawElements(gl.TRIANGLES, model.mesh.indexBuffer.numItems, gl.UNSIGNED_SHORT, 0);\n */\nexport function initMeshBuffers(gl: WebGLRenderingContext, mesh: Mesh): MeshWithBuffers {\n    (mesh as MeshWithBuffers).normalBuffer = _buildBuffer(gl, gl.ARRAY_BUFFER, mesh.vertexNormals, 3);\n    (mesh as MeshWithBuffers).textureBuffer = _buildBuffer(gl, gl.ARRAY_BUFFER, mesh.textures, mesh.textureStride);\n    (mesh as MeshWithBuffers).vertexBuffer = _buildBuffer(gl, gl.ARRAY_BUFFER, mesh.vertices, 3);\n    (mesh as MeshWithBuffers).indexBuffer = _buildBuffer(gl, gl.ELEMENT_ARRAY_BUFFER, mesh.indices, 1);\n\n    return mesh as MeshWithBuffers;\n}\n\nexport function deleteMeshBuffers(gl: WebGLRenderingContext, mesh: MeshWithBuffers) {\n    gl.deleteBuffer(mesh.normalBuffer);\n    gl.deleteBuffer(mesh.textureBuffer);\n    gl.deleteBuffer(mesh.vertexBuffer);\n    gl.deleteBuffer(mesh.indexBuffer);\n}\n"],"sourceRoot":""} \ No newline at end of file +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack://OBJ/webpack/universalModuleDefinition","webpack://OBJ/webpack/bootstrap","webpack://OBJ/./src/index.ts","webpack://OBJ/./src/layout.ts","webpack://OBJ/./src/material.ts","webpack://OBJ/./src/mesh.ts","webpack://OBJ/./src/utils.ts"],"names":[],"mappings":"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","file":"webgl-obj-loader.js","sourcesContent":["(function webpackUniversalModuleDefinition(root, factory) {\n\tif(typeof exports === 'object' && typeof module === 'object')\n\t\tmodule.exports = factory();\n\telse if(typeof define === 'function' && define.amd)\n\t\tdefine(\"OBJ\", [], factory);\n\telse if(typeof exports === 'object')\n\t\texports[\"OBJ\"] = factory();\n\telse\n\t\troot[\"OBJ\"] = factory();\n})(typeof self !== 'undefined' ? self : this, function() {\nreturn "," \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, { enumerable: true, get: getter });\n \t\t}\n \t};\n\n \t// define __esModule on exports\n \t__webpack_require__.r = function(exports) {\n \t\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n \t\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n \t\t}\n \t\tObject.defineProperty(exports, '__esModule', { value: true });\n \t};\n\n \t// create a fake namespace object\n \t// mode & 1: value is a module id, require it\n \t// mode & 2: merge all properties of value into the ns\n \t// mode & 4: return value when already ns object\n \t// mode & 8|1: behave like require\n \t__webpack_require__.t = function(value, mode) {\n \t\tif(mode & 1) value = __webpack_require__(value);\n \t\tif(mode & 8) return value;\n \t\tif((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;\n \t\tvar ns = Object.create(null);\n \t\t__webpack_require__.r(ns);\n \t\tObject.defineProperty(ns, 'default', { enumerable: true, value: value });\n \t\tif(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));\n \t\treturn ns;\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"/\";\n\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = 0);\n","import Mesh, {\n    MeshOptions,\n    MaterialNameToIndex,\n    IndexToMaterial,\n    ArrayBufferWithItemSize,\n    Uint16ArrayWithItemSize,\n} from \"./mesh\";\nimport { Material, MaterialLibrary, Vec3, UVW, TextureMapData } from \"./material\";\nimport { Layout, TYPES, AttributeInfo, DuplicateAttributeException, Attribute } from \"./layout\";\nimport {\n    downloadModels,\n    downloadMeshes,\n    initMeshBuffers,\n    deleteMeshBuffers,\n    DownloadModelsOptions,\n    MeshMap,\n    NameAndUrls,\n    ExtendedGLBuffer,\n    MeshWithBuffers,\n} from \"./utils\";\n\nconst version = \"2.0.3\";\n\n/**\n * @namespace\n */\nexport {\n    ArrayBufferWithItemSize,\n    Attribute,\n    AttributeInfo,\n    DownloadModelsOptions,\n    DuplicateAttributeException,\n    ExtendedGLBuffer,\n    IndexToMaterial,\n    Layout,\n    Material,\n    MaterialLibrary,\n    MaterialNameToIndex,\n    Mesh,\n    MeshMap,\n    MeshOptions,\n    MeshWithBuffers,\n    NameAndUrls,\n    TextureMapData,\n    TYPES,\n    Uint16ArrayWithItemSize,\n    UVW,\n    Vec3,\n    downloadModels,\n    downloadMeshes,\n    initMeshBuffers,\n    deleteMeshBuffers,\n    version,\n};\n","export enum TYPES {\n    \"BYTE\" = \"BYTE\",\n    \"UNSIGNED_BYTE\" = \"UNSIGNED_BYTE\",\n    \"SHORT\" = \"SHORT\",\n    \"UNSIGNED_SHORT\" = \"UNSIGNED_SHORT\",\n    \"FLOAT\" = \"FLOAT\",\n}\n\nexport interface AttributeInfo {\n    attribute: Attribute;\n    size: Attribute[\"size\"];\n    type: Attribute[\"type\"];\n    normalized: Attribute[\"normalized\"];\n    offset: number;\n    stride: number;\n}\n\n/**\n * An exception for when two or more of the same attributes are found in the\n * same layout.\n * @private\n */\nexport class DuplicateAttributeException extends Error {\n    /**\n     * Create a DuplicateAttributeException\n     * @param {Attribute} attribute - The attribute that was found more than\n     *        once in the {@link Layout}\n     */\n    constructor(attribute: Attribute) {\n        super(`found duplicate attribute: ${attribute.key}`);\n    }\n}\n\n/**\n * Represents how a vertex attribute should be packed into an buffer.\n * @private\n */\nexport class Attribute {\n    public sizeOfType: number;\n    public sizeInBytes: number;\n    /**\n     * Create an attribute. Do not call this directly, use the predefined\n     * constants.\n     * @param {string} key - The name of this attribute as if it were a key in\n     *        an Object. Use the camel case version of the upper snake case\n     *        const name.\n     * @param {number} size - The number of components per vertex attribute.\n     *        Must be 1, 2, 3, or 4.\n     * @param {string} type - The data type of each component for this\n     *        attribute. Possible values:<br/>\n     *        \"BYTE\": signed 8-bit integer, with values in [-128, 127]<br/>\n     *        \"SHORT\": signed 16-bit integer, with values in\n     *            [-32768, 32767]<br/>\n     *        \"UNSIGNED_BYTE\": unsigned 8-bit integer, with values in\n     *            [0, 255]<br/>\n     *        \"UNSIGNED_SHORT\": unsigned 16-bit integer, with values in\n     *            [0, 65535]<br/>\n     *        \"FLOAT\": 32-bit floating point number\n     * @param {boolean} normalized - Whether integer data values should be\n     *        normalized when being casted to a float.<br/>\n     *        If true, signed integers are normalized to [-1, 1].<br/>\n     *        If true, unsigned integers are normalized to [0, 1].<br/>\n     *        For type \"FLOAT\", this parameter has no effect.\n     */\n    constructor(public key: string, public size: number, public type: TYPES, public normalized: boolean = false) {\n        switch (type) {\n            case \"BYTE\":\n            case \"UNSIGNED_BYTE\":\n                this.sizeOfType = 1;\n                break;\n            case \"SHORT\":\n            case \"UNSIGNED_SHORT\":\n                this.sizeOfType = 2;\n                break;\n            case \"FLOAT\":\n                this.sizeOfType = 4;\n                break;\n            default:\n                throw new Error(`Unknown gl type: ${type}`);\n        }\n        this.sizeInBytes = this.sizeOfType * size;\n    }\n}\n\n/**\n * A class to represent the memory layout for a vertex attribute array. Used by\n * {@link Mesh}'s TBD(...) method to generate a packed array from mesh data.\n * <p>\n * Layout can sort of be thought of as a C-style struct declaration.\n * {@link Mesh}'s TBD(...) method will use the {@link Layout} instance to\n * pack an array in the given attribute order.\n * <p>\n * Layout also is very helpful when calling a WebGL context's\n * <code>vertexAttribPointer</code> method. If you've created a buffer using\n * a Layout instance, then the same Layout instance can be used to determine\n * the size, type, normalized, stride, and offset parameters for\n * <code>vertexAttribPointer</code>.\n * <p>\n * For example:\n * <pre><code>\n *\n * const index = glctx.getAttribLocation(shaderProgram, \"pos\");\n * glctx.vertexAttribPointer(\n *   layout.position.size,\n *   glctx[layout.position.type],\n *   layout.position.normalized,\n *   layout.position.stride,\n *   layout.position.offset);\n * </code></pre>\n * @see {@link Mesh}\n */\nexport class Layout {\n    // Geometry attributes\n    /**\n     * Attribute layout to pack a vertex's x, y, & z as floats\n     *\n     * @see {@link Layout}\n     */\n    static POSITION = new Attribute(\"position\", 3, TYPES.FLOAT);\n\n    /**\n     * Attribute layout to pack a vertex's normal's x, y, & z as floats\n     *\n     * @see {@link Layout}\n     */\n    static NORMAL = new Attribute(\"normal\", 3, TYPES.FLOAT);\n\n    /**\n     * Attribute layout to pack a vertex's normal's x, y, & z as floats.\n     * <p>\n     * This value will be computed on-the-fly based on the texture coordinates.\n     * If no texture coordinates are available, the generated value will default to\n     * 0, 0, 0.\n     *\n     * @see {@link Layout}\n     */\n    static TANGENT = new Attribute(\"tangent\", 3, TYPES.FLOAT);\n\n    /**\n     * Attribute layout to pack a vertex's normal's bitangent x, y, & z as floats.\n     * <p>\n     * This value will be computed on-the-fly based on the texture coordinates.\n     * If no texture coordinates are available, the generated value will default to\n     * 0, 0, 0.\n     * @see {@link Layout}\n     */\n    static BITANGENT = new Attribute(\"bitangent\", 3, TYPES.FLOAT);\n\n    /**\n     * Attribute layout to pack a vertex's texture coordinates' u & v as floats\n     *\n     * @see {@link Layout}\n     */\n    static UV = new Attribute(\"uv\", 2, TYPES.FLOAT);\n\n    // Material attributes\n\n    /**\n     * Attribute layout to pack an unsigned short to be interpreted as a the index\n     * into a {@link Mesh}'s materials list.\n     * <p>\n     * The intention of this value is to send all of the {@link Mesh}'s materials\n     * into multiple shader uniforms and then reference the current one by this\n     * vertex attribute.\n     * <p>\n     * example glsl code:\n     *\n     * <pre><code>\n     *  // this is bound using MATERIAL_INDEX\n     *  attribute int materialIndex;\n     *\n     *  struct Material {\n     *    vec3 diffuse;\n     *    vec3 specular;\n     *    vec3 specularExponent;\n     *  };\n     *\n     *  uniform Material materials[MAX_MATERIALS];\n     *\n     *  // ...\n     *\n     *  vec3 diffuse = materials[materialIndex];\n     *\n     * </code></pre>\n     * TODO: More description & test to make sure subscripting by attributes even\n     * works for webgl\n     *\n     * @see {@link Layout}\n     */\n    static MATERIAL_INDEX = new Attribute(\"materialIndex\", 1, TYPES.SHORT);\n    static MATERIAL_ENABLED = new Attribute(\"materialEnabled\", 1, TYPES.UNSIGNED_SHORT);\n    static AMBIENT = new Attribute(\"ambient\", 3, TYPES.FLOAT);\n    static DIFFUSE = new Attribute(\"diffuse\", 3, TYPES.FLOAT);\n    static SPECULAR = new Attribute(\"specular\", 3, TYPES.FLOAT);\n    static SPECULAR_EXPONENT = new Attribute(\"specularExponent\", 3, TYPES.FLOAT);\n    static EMISSIVE = new Attribute(\"emissive\", 3, TYPES.FLOAT);\n    static TRANSMISSION_FILTER = new Attribute(\"transmissionFilter\", 3, TYPES.FLOAT);\n    static DISSOLVE = new Attribute(\"dissolve\", 1, TYPES.FLOAT);\n    static ILLUMINATION = new Attribute(\"illumination\", 1, TYPES.UNSIGNED_SHORT);\n    static REFRACTION_INDEX = new Attribute(\"refractionIndex\", 1, TYPES.FLOAT);\n    static SHARPNESS = new Attribute(\"sharpness\", 1, TYPES.FLOAT);\n    static MAP_DIFFUSE = new Attribute(\"mapDiffuse\", 1, TYPES.SHORT);\n    static MAP_AMBIENT = new Attribute(\"mapAmbient\", 1, TYPES.SHORT);\n    static MAP_SPECULAR = new Attribute(\"mapSpecular\", 1, TYPES.SHORT);\n    static MAP_SPECULAR_EXPONENT = new Attribute(\"mapSpecularExponent\", 1, TYPES.SHORT);\n    static MAP_DISSOLVE = new Attribute(\"mapDissolve\", 1, TYPES.SHORT);\n    static ANTI_ALIASING = new Attribute(\"antiAliasing\", 1, TYPES.UNSIGNED_SHORT);\n    static MAP_BUMP = new Attribute(\"mapBump\", 1, TYPES.SHORT);\n    static MAP_DISPLACEMENT = new Attribute(\"mapDisplacement\", 1, TYPES.SHORT);\n    static MAP_DECAL = new Attribute(\"mapDecal\", 1, TYPES.SHORT);\n    static MAP_EMISSIVE = new Attribute(\"mapEmissive\", 1, TYPES.SHORT);\n\n    public stride: number;\n    public attributes: Attribute[];\n    public attributeMap: { [idx: string]: AttributeInfo };\n    /**\n     * Create a Layout object. This constructor will throw if any duplicate\n     * attributes are given.\n     * @param {Array} ...attributes - An ordered list of attributes that\n     *        describe the desired memory layout for each vertex attribute.\n     *        <p>\n     *\n     * @see {@link Mesh}\n     */\n    constructor(...attributes: Attribute[]) {\n        this.attributes = attributes;\n        this.attributeMap = {};\n        let offset = 0;\n        let maxStrideMultiple = 0;\n        for (const attribute of attributes) {\n            if (this.attributeMap[attribute.key]) {\n                throw new DuplicateAttributeException(attribute);\n            }\n            // Add padding to satisfy WebGL's requirement that all\n            // vertexAttribPointer calls have an offset that is a multiple of\n            // the type size.\n            if (offset % attribute.sizeOfType !== 0) {\n                offset += attribute.sizeOfType - (offset % attribute.sizeOfType);\n                console.warn(\"Layout requires padding before \" + attribute.key + \" attribute\");\n            }\n            this.attributeMap[attribute.key] = {\n                attribute: attribute,\n                size: attribute.size,\n                type: attribute.type,\n                normalized: attribute.normalized,\n                offset: offset,\n            } as AttributeInfo;\n            offset += attribute.sizeInBytes;\n            maxStrideMultiple = Math.max(maxStrideMultiple, attribute.sizeOfType);\n        }\n        // Add padding to the end to satisfy WebGL's requirement that all\n        // vertexAttribPointer calls have a stride that is a multiple of the\n        // type size. Because we're putting differently sized attributes into\n        // the same buffer, it must be padded to a multiple of the largest\n        // type size.\n        if (offset % maxStrideMultiple !== 0) {\n            offset += maxStrideMultiple - (offset % maxStrideMultiple);\n            console.warn(\"Layout requires padding at the back\");\n        }\n        this.stride = offset;\n        for (const attribute of attributes) {\n            this.attributeMap[attribute.key].stride = this.stride;\n        }\n    }\n}\n","export type Vec3 = [number, number, number];\n\nexport interface UVW {\n    u: number;\n    v: number;\n    w: number;\n}\n\nexport interface TextureMapData {\n    colorCorrection: boolean;\n    horizontalBlending: boolean;\n    verticalBlending: boolean;\n    boostMipMapSharpness: number;\n    modifyTextureMap: {\n        brightness: number;\n        contrast: number;\n    };\n    offset: UVW;\n    scale: UVW;\n    turbulence: UVW;\n    clamp: boolean;\n    textureResolution: number | null;\n    bumpMultiplier: number;\n    imfChan: string | null;\n    filename: string;\n    reflectionType?: string;\n    texture?: HTMLImageElement;\n}\n\n/**\n * The Material class.\n */\nexport class Material {\n    /**\n     * Constructor\n     * @param {String} name the unique name of the material\n     */\n    // The values for the following attibutes\n    // are an array of R, G, B normalized values.\n    // Ka - Ambient Reflectivity\n    ambient: Vec3 = [0, 0, 0];\n    // Kd - Defuse Reflectivity\n    diffuse: Vec3 = [0, 0, 0];\n    // Ks\n    specular: Vec3 = [0, 0, 0];\n    // Ke\n    emissive: Vec3 = [0, 0, 0];\n    // Tf\n    transmissionFilter: Vec3 = [0, 0, 0];\n    // d\n    dissolve: number = 0;\n    // valid range is between 0 and 1000\n    specularExponent: number = 0;\n    // either d or Tr; valid values are normalized\n    transparency: number = 0;\n    // illum - the enum of the illumination model to use\n    illumination: number = 0;\n    // Ni - Set to \"normal\" (air).\n    refractionIndex: number = 1;\n    // sharpness\n    sharpness: number = 0;\n    // map_Kd\n    mapDiffuse: TextureMapData = emptyTextureOptions();\n    // map_Ka\n    mapAmbient: TextureMapData = emptyTextureOptions();\n    // map_Ks\n    mapSpecular: TextureMapData = emptyTextureOptions();\n    // map_Ns\n    mapSpecularExponent: TextureMapData = emptyTextureOptions();\n    // map_d\n    mapDissolve: TextureMapData = emptyTextureOptions();\n    // map_aat\n    antiAliasing: boolean = false;\n    // map_bump or bump\n    mapBump: TextureMapData = emptyTextureOptions();\n    // disp\n    mapDisplacement: TextureMapData = emptyTextureOptions();\n    // decal\n    mapDecal: TextureMapData = emptyTextureOptions();\n    // map_Ke\n    mapEmissive: TextureMapData = emptyTextureOptions();\n    // refl - when the reflection type is a cube, there will be multiple refl\n    //        statements for each side of the cube. If it's a spherical\n    //        reflection, there should only ever be one.\n    mapReflections: TextureMapData[] = [];\n    constructor(public name: string) {}\n}\n\nconst SENTINEL_MATERIAL = new Material(\"sentinel\");\n\n/**\n * https://en.wikipedia.org/wiki/Wavefront_.obj_file\n * http://paulbourke.net/dataformats/mtl/\n */\nexport class MaterialLibrary {\n    /**\n     * Constructs the Material Parser\n     * @param mtlData the MTL file contents\n     */\n    public currentMaterial: Material = SENTINEL_MATERIAL;\n    public materials: { [k: string]: Material } = {};\n\n    constructor(public data: string) {\n        this.parse();\n    }\n\n    /* eslint-disable camelcase */\n    /* the function names here disobey camelCase conventions\n     to make parsing/routing easier. see the parse function\n     documentation for more information. */\n\n    /**\n     * Creates a new Material object and adds to the registry.\n     * @param tokens the tokens associated with the directive\n     */\n    parse_newmtl(tokens: string[]) {\n        const name = tokens[0];\n        // console.info('Parsing new Material:', name);\n\n        this.currentMaterial = new Material(name);\n        this.materials[name] = this.currentMaterial;\n    }\n\n    /**\n     * See the documenation for parse_Ka below for a better understanding.\n     *\n     * Given a list of possible color tokens, returns an array of R, G, and B\n     * color values.\n     *\n     * @param tokens the tokens associated with the directive\n     * @return {*} a 3 element array containing the R, G, and B values\n     * of the color.\n     */\n    parseColor(tokens: string[]): Vec3 {\n        if (tokens[0] == \"spectral\") {\n            throw new Error(\n                \"The MTL parser does not support spectral curve files. You will \" +\n                    \"need to convert the MTL colors to either RGB or CIEXYZ.\",\n            );\n        }\n\n        if (tokens[0] == \"xyz\") {\n            throw new Error(\n                \"The MTL parser does not currently support XYZ colors. Either convert the \" +\n                    \"XYZ values to RGB or create an issue to add support for XYZ\",\n            );\n        }\n\n        // from my understanding of the spec, RGB values at this point\n        // will either be 3 floats or exactly 1 float, so that's the check\n        // that i'm going to perform here\n        if (tokens.length == 3) {\n            const [x, y, z] = tokens;\n            return [parseFloat(x), parseFloat(y), parseFloat(z)];\n        }\n\n        // Since tokens at this point has a length of 3, we're going to assume\n        // it's exactly 1, skipping the check for 2.\n        const value = parseFloat(tokens[0]);\n        // in this case, all values are equivalent\n        return [value, value, value];\n    }\n\n    /**\n     * Parse the ambient reflectivity\n     *\n     * A Ka directive can take one of three forms:\n     *   - Ka r g b\n     *   - Ka spectral file.rfl\n     *   - Ka xyz x y z\n     * These three forms are mutually exclusive in that only one\n     * declaration can exist per material. It is considered a syntax\n     * error otherwise.\n     *\n     * The \"Ka\" form specifies the ambient reflectivity using RGB values.\n     * The \"g\" and \"b\" values are optional. If only the \"r\" value is\n     * specified, then the \"g\" and \"b\" values are assigned the value of\n     * \"r\". Values are normally in the range 0.0 to 1.0. Values outside\n     * of this range increase or decrease the reflectivity accordingly.\n     *\n     * The \"Ka spectral\" form specifies the ambient reflectivity using a\n     * spectral curve. \"file.rfl\" is the name of the \".rfl\" file containing\n     * the curve data. \"factor\" is an optional argument which is a multiplier\n     * for the values in the .rfl file and defaults to 1.0 if not specified.\n     *\n     * The \"Ka xyz\" form specifies the ambient reflectivity using CIEXYZ values.\n     * \"x y z\" are the values of the CIEXYZ color space. The \"y\" and \"z\" arguments\n     * are optional and take on the value of the \"x\" component if only \"x\" is\n     * specified. The \"x y z\" values are normally in the range of 0.0 to 1.0 and\n     * increase or decrease ambient reflectivity accordingly outside of that\n     * range.\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_Ka(tokens: string[]) {\n        this.currentMaterial.ambient = this.parseColor(tokens);\n    }\n\n    /**\n     * Diffuse Reflectivity\n     *\n     * Similar to the Ka directive. Simply replace \"Ka\" with \"Kd\" and the rules\n     * are the same\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_Kd(tokens: string[]) {\n        this.currentMaterial.diffuse = this.parseColor(tokens);\n    }\n\n    /**\n     * Spectral Reflectivity\n     *\n     * Similar to the Ka directive. Simply replace \"Ks\" with \"Kd\" and the rules\n     * are the same\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_Ks(tokens: string[]) {\n        this.currentMaterial.specular = this.parseColor(tokens);\n    }\n\n    /**\n     * Emissive\n     *\n     * The amount and color of light emitted by the object.\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_Ke(tokens: string[]) {\n        this.currentMaterial.emissive = this.parseColor(tokens);\n    }\n\n    /**\n     * Transmission Filter\n     *\n     * Any light passing through the object is filtered by the transmission\n     * filter, which only allows specific colors to pass through. For example, Tf\n     * 0 1 0 allows all of the green to pass through and filters out all of the\n     * red and blue.\n     *\n     * Similar to the Ka directive. Simply replace \"Ks\" with \"Tf\" and the rules\n     * are the same\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_Tf(tokens: string[]) {\n        this.currentMaterial.transmissionFilter = this.parseColor(tokens);\n    }\n\n    /**\n     * Specifies the dissolve for the current material.\n     *\n     * Statement: d [-halo] `factor`\n     *\n     * Example: \"d 0.5\"\n     *\n     * The factor is the amount this material dissolves into the background. A\n     * factor of 1.0 is fully opaque. This is the default when a new material is\n     * created. A factor of 0.0 is fully dissolved (completely transparent).\n     *\n     * Unlike a real transparent material, the dissolve does not depend upon\n     * material thickness nor does it have any spectral character. Dissolve works\n     * on all illumination models.\n     *\n     * The dissolve statement allows for an optional \"-halo\" flag which indicates\n     * that a dissolve is dependent on the surface orientation relative to the\n     * viewer. For example, a sphere with the following dissolve, \"d -halo 0.0\",\n     * will be fully dissolved at its center and will appear gradually more opaque\n     * toward its edge.\n     *\n     * \"factor\" is the minimum amount of dissolve applied to the material. The\n     * amount of dissolve will vary between 1.0 (fully opaque) and the specified\n     * \"factor\". The formula is:\n     *\n     *    dissolve = 1.0 - (N*v)(1.0-factor)\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_d(tokens: string[]) {\n        // this ignores the -halo option as I can't find any documentation on what\n        // it's supposed to be.\n        this.currentMaterial.dissolve = parseFloat(tokens.pop() || \"0\");\n    }\n\n    /**\n     * The \"illum\" statement specifies the illumination model to use in the\n     * material. Illumination models are mathematical equations that represent\n     * various material lighting and shading effects.\n     *\n     * The illumination number can be a number from 0 to 10. The following are\n     * the list of illumination enumerations and their summaries:\n     * 0. Color on and Ambient off\n     * 1. Color on and Ambient on\n     * 2. Highlight on\n     * 3. Reflection on and Ray trace on\n     * 4. Transparency: Glass on, Reflection: Ray trace on\n     * 5. Reflection: Fresnel on and Ray trace on\n     * 6. Transparency: Refraction on, Reflection: Fresnel off and Ray trace on\n     * 7. Transparency: Refraction on, Reflection: Fresnel on and Ray trace on\n     * 8. Reflection on and Ray trace off\n     * 9. Transparency: Glass on, Reflection: Ray trace off\n     * 10. Casts shadows onto invisible surfaces\n     *\n     * Example: \"illum 2\" to specify the \"Highlight on\" model\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_illum(tokens: string[]) {\n        this.currentMaterial.illumination = parseInt(tokens[0]);\n    }\n\n    /**\n     * Optical Density (AKA Index of Refraction)\n     *\n     * Statement: Ni `index`\n     *\n     * Example: Ni 1.0\n     *\n     * Specifies the optical density for the surface. `index` is the value\n     * for the optical density. The values can range from 0.001 to 10.  A value of\n     * 1.0 means that light does not bend as it passes through an object.\n     * Increasing the optical_density increases the amount of bending. Glass has\n     * an index of refraction of about 1.5. Values of less than 1.0 produce\n     * bizarre results and are not recommended\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_Ni(tokens: string[]) {\n        this.currentMaterial.refractionIndex = parseFloat(tokens[0]);\n    }\n\n    /**\n     * Specifies the specular exponent for the current material. This defines the\n     * focus of the specular highlight.\n     *\n     * Statement: Ns `exponent`\n     *\n     * Example: \"Ns 250\"\n     *\n     * `exponent` is the value for the specular exponent. A high exponent results\n     * in a tight, concentrated highlight. Ns Values normally range from 0 to\n     * 1000.\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_Ns(tokens: string[]) {\n        this.currentMaterial.specularExponent = parseInt(tokens[0]);\n    }\n\n    /**\n     * Specifies the sharpness of the reflections from the local reflection map.\n     *\n     * Statement: sharpness `value`\n     *\n     * Example: \"sharpness 100\"\n     *\n     * If a material does not have a local reflection map defined in its material\n     * defintions, sharpness will apply to the global reflection map defined in\n     * PreView.\n     *\n     * `value` can be a number from 0 to 1000. The default is 60. A high value\n     * results in a clear reflection of objects in the reflection map.\n     *\n     * Tip: sharpness values greater than 100 introduce aliasing effects in\n     * flat surfaces that are viewed at a sharp angle.\n     *\n     * @param tokens the tokens associated with the directive\n     */\n    parse_sharpness(tokens: string[]) {\n        this.currentMaterial.sharpness = parseInt(tokens[0]);\n    }\n\n    /**\n     * Parses the -cc flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -cc flag\n     * @param options the Object of all image options\n     */\n    parse_cc(values: string[], options: TextureMapData) {\n        options.colorCorrection = values[0] == \"on\";\n    }\n\n    /**\n     * Parses the -blendu flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -blendu flag\n     * @param options the Object of all image options\n     */\n    parse_blendu(values: string[], options: TextureMapData) {\n        options.horizontalBlending = values[0] == \"on\";\n    }\n\n    /**\n     * Parses the -blendv flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -blendv flag\n     * @param options the Object of all image options\n     */\n    parse_blendv(values: string[], options: TextureMapData) {\n        options.verticalBlending = values[0] == \"on\";\n    }\n\n    /**\n     * Parses the -boost flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -boost flag\n     * @param options the Object of all image options\n     */\n    parse_boost(values: string[], options: TextureMapData) {\n        options.boostMipMapSharpness = parseFloat(values[0]);\n    }\n\n    /**\n     * Parses the -mm flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -mm flag\n     * @param options the Object of all image options\n     */\n    parse_mm(values: string[], options: TextureMapData) {\n        options.modifyTextureMap.brightness = parseFloat(values[0]);\n        options.modifyTextureMap.contrast = parseFloat(values[1]);\n    }\n\n    /**\n     * Parses and sets the -o, -s, and -t  u, v, and w values\n     *\n     * @param values the values passed to the -o, -s, -t flag\n     * @param {Object} option the Object of either the -o, -s, -t option\n     * @param {Integer} defaultValue the Object of all image options\n     */\n    parse_ost(values: string[], option: UVW, defaultValue: number) {\n        while (values.length < 3) {\n            values.push(defaultValue.toString());\n        }\n\n        option.u = parseFloat(values[0]);\n        option.v = parseFloat(values[1]);\n        option.w = parseFloat(values[2]);\n    }\n\n    /**\n     * Parses the -o flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -o flag\n     * @param options the Object of all image options\n     */\n    parse_o(values: string[], options: TextureMapData) {\n        this.parse_ost(values, options.offset, 0);\n    }\n\n    /**\n     * Parses the -s flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -s flag\n     * @param options the Object of all image options\n     */\n    parse_s(values: string[], options: TextureMapData) {\n        this.parse_ost(values, options.scale, 1);\n    }\n\n    /**\n     * Parses the -t flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -t flag\n     * @param options the Object of all image options\n     */\n    parse_t(values: string[], options: TextureMapData) {\n        this.parse_ost(values, options.turbulence, 0);\n    }\n\n    /**\n     * Parses the -texres flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -texres flag\n     * @param options the Object of all image options\n     */\n    parse_texres(values: string[], options: TextureMapData) {\n        options.textureResolution = parseFloat(values[0]);\n    }\n\n    /**\n     * Parses the -clamp flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -clamp flag\n     * @param options the Object of all image options\n     */\n    parse_clamp(values: string[], options: TextureMapData) {\n        options.clamp = values[0] == \"on\";\n    }\n\n    /**\n     * Parses the -bm flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -bm flag\n     * @param options the Object of all image options\n     */\n    parse_bm(values: string[], options: TextureMapData) {\n        options.bumpMultiplier = parseFloat(values[0]);\n    }\n\n    /**\n     * Parses the -imfchan flag and updates the options object with the values.\n     *\n     * @param values the values passed to the -imfchan flag\n     * @param options the Object of all image options\n     */\n    parse_imfchan(values: string[], options: TextureMapData) {\n        options.imfChan = values[0];\n    }\n\n    /**\n     * This only exists for relection maps and denotes the type of reflection.\n     *\n     * @param values the values passed to the -type flag\n     * @param options the Object of all image options\n     */\n    parse_type(values: string[], options: TextureMapData) {\n        options.reflectionType = values[0];\n    }\n\n    /**\n     * Parses the texture's options and returns an options object with the info\n     *\n     * @param tokens all of the option tokens to pass to the texture\n     * @return {Object} a complete object of objects to apply to the texture\n     */\n    parseOptions(tokens: string[]): TextureMapData {\n        const options = emptyTextureOptions();\n\n        let option;\n        let values;\n        const optionsToValues: { [k: string]: string[] } = {};\n\n        tokens.reverse();\n\n        while (tokens.length) {\n            // token is guaranteed to exists here, hence the explicit \"as\"\n            const token = tokens.pop() as string;\n\n            if (token.startsWith(\"-\")) {\n                option = token.substr(1);\n                optionsToValues[option] = [];\n            } else if (option) {\n                optionsToValues[option].push(token);\n            }\n        }\n\n        for (option in optionsToValues) {\n            if (!optionsToValues.hasOwnProperty(option)) {\n                continue;\n            }\n            values = optionsToValues[option];\n            const optionMethod = (this as any)[`parse_${option}`];\n            if (optionMethod) {\n                optionMethod.bind(this)(values, options);\n            }\n        }\n\n        return options;\n    }\n\n    /**\n     * Parses the given texture map line.\n     *\n     * @param tokens all of the tokens representing the texture\n     * @return a complete object of objects to apply to the texture\n     */\n    parseMap(tokens: string[]): TextureMapData {\n        // according to wikipedia:\n        // (https://en.wikipedia.org/wiki/Wavefront_.obj_file#Vendor_specific_alterations)\n        // there is at least one vendor that places the filename before the options\n        // rather than after (which is to spec). All options start with a '-'\n        // so if the first token doesn't start with a '-', we're going to assume\n        // it's the name of the map file.\n        let optionsString;\n        let filename = \"\";\n        if (!tokens[0].startsWith(\"-\")) {\n            [filename, ...optionsString] = tokens;\n        } else {\n            filename = tokens.pop() as string;\n            optionsString = tokens;\n        }\n\n        const options = this.parseOptions(optionsString);\n        options.filename = filename.replace(/\\\\/g, \"/\");\n\n        return options;\n    }\n\n    /**\n     * Parses the ambient map.\n     *\n     * @param tokens list of tokens for the map_Ka direcive\n     */\n    parse_map_Ka(tokens: string[]) {\n        this.currentMaterial.mapAmbient = this.parseMap(tokens);\n    }\n\n    /**\n     * Parses the diffuse map.\n     *\n     * @param tokens list of tokens for the map_Kd direcive\n     */\n    parse_map_Kd(tokens: string[]) {\n        this.currentMaterial.mapDiffuse = this.parseMap(tokens);\n    }\n\n    /**\n     * Parses the specular map.\n     *\n     * @param tokens list of tokens for the map_Ks direcive\n     */\n    parse_map_Ks(tokens: string[]) {\n        this.currentMaterial.mapSpecular = this.parseMap(tokens);\n    }\n\n    /**\n     * Parses the emissive map.\n     *\n     * @param tokens list of tokens for the map_Ke direcive\n     */\n    parse_map_Ke(tokens: string[]) {\n        this.currentMaterial.mapEmissive = this.parseMap(tokens);\n    }\n\n    /**\n     * Parses the specular exponent map.\n     *\n     * @param tokens list of tokens for the map_Ns direcive\n     */\n    parse_map_Ns(tokens: string[]) {\n        this.currentMaterial.mapSpecularExponent = this.parseMap(tokens);\n    }\n\n    /**\n     * Parses the dissolve map.\n     *\n     * @param tokens list of tokens for the map_d direcive\n     */\n    parse_map_d(tokens: string[]) {\n        this.currentMaterial.mapDissolve = this.parseMap(tokens);\n    }\n\n    /**\n     * Parses the anti-aliasing option.\n     *\n     * @param tokens list of tokens for the map_aat direcive\n     */\n    parse_map_aat(tokens: string[]) {\n        this.currentMaterial.antiAliasing = tokens[0] == \"on\";\n    }\n\n    /**\n     * Parses the bump map.\n     *\n     * @param tokens list of tokens for the map_bump direcive\n     */\n    parse_map_bump(tokens: string[]) {\n        this.currentMaterial.mapBump = this.parseMap(tokens);\n    }\n\n    /**\n     * Parses the bump map.\n     *\n     * @param tokens list of tokens for the bump direcive\n     */\n    parse_bump(tokens: string[]) {\n        this.parse_map_bump(tokens);\n    }\n\n    /**\n     * Parses the disp map.\n     *\n     * @param tokens list of tokens for the disp direcive\n     */\n    parse_disp(tokens: string[]) {\n        this.currentMaterial.mapDisplacement = this.parseMap(tokens);\n    }\n\n    /**\n     * Parses the decal map.\n     *\n     * @param tokens list of tokens for the map_decal direcive\n     */\n    parse_decal(tokens: string[]) {\n        this.currentMaterial.mapDecal = this.parseMap(tokens);\n    }\n\n    /**\n     * Parses the refl map.\n     *\n     * @param tokens list of tokens for the refl direcive\n     */\n    parse_refl(tokens: string[]) {\n        this.currentMaterial.mapReflections.push(this.parseMap(tokens));\n    }\n\n    /**\n     * Parses the MTL file.\n     *\n     * Iterates line by line parsing each MTL directive.\n     *\n     * This function expects the first token in the line\n     * to be a valid MTL directive. That token is then used\n     * to try and run a method on this class. parse_[directive]\n     * E.g., the `newmtl` directive would try to call the method\n     * parse_newmtl. Each parsing function takes in the remaining\n     * list of tokens and updates the currentMaterial class with\n     * the attributes provided.\n     */\n    parse() {\n        const lines = this.data.split(/\\r?\\n/);\n        for (let line of lines) {\n            line = line.trim();\n            if (!line || line.startsWith(\"#\")) {\n                continue;\n            }\n\n            const [directive, ...tokens] = line.split(/\\s/);\n\n            const parseMethod = (this as any)[`parse_${directive}`];\n\n            if (!parseMethod) {\n                console.warn(`Don't know how to parse the directive: \"${directive}\"`);\n                continue;\n            }\n\n            // console.log(`Parsing \"${directive}\" with tokens: ${tokens}`);\n            parseMethod.bind(this)(tokens);\n        }\n\n        // some cleanup. These don't need to be exposed as public data.\n        delete this.data;\n        this.currentMaterial = SENTINEL_MATERIAL;\n    }\n\n    /* eslint-enable camelcase*/\n}\n\nfunction emptyTextureOptions(): TextureMapData {\n    return {\n        colorCorrection: false,\n        horizontalBlending: true,\n        verticalBlending: true,\n        boostMipMapSharpness: 0,\n        modifyTextureMap: {\n            brightness: 0,\n            contrast: 1,\n        },\n        offset: { u: 0, v: 0, w: 0 },\n        scale: { u: 1, v: 1, w: 1 },\n        turbulence: { u: 0, v: 0, w: 0 },\n        clamp: false,\n        textureResolution: null,\n        bumpMultiplier: 1,\n        imfChan: null,\n        filename: \"\",\n    };\n}\n","import { Layout } from \"./layout\";\nimport { Material, MaterialLibrary } from \"./material\";\n\nexport interface MeshOptions {\n    enableWTextureCoord?: boolean;\n    calcTangentsAndBitangents?: boolean;\n    materials?: { [key: string]: Material };\n}\n\ninterface UnpackedAttrs {\n    verts: number[];\n    norms: number[];\n    textures: number[];\n    hashindices: { [k: string]: number };\n    indices: number[][];\n    materialIndices: number[];\n    index: number;\n}\n\nexport interface MaterialNameToIndex {\n    [k: string]: number;\n}\n\nexport interface IndexToMaterial {\n    [k: number]: Material;\n}\n\nexport interface ArrayBufferWithItemSize extends ArrayBuffer {\n    numItems?: number;\n}\n\nexport interface Uint16ArrayWithItemSize extends Uint16Array {\n    numItems?: number;\n}\n\n/**\n * The main Mesh class. The constructor will parse through the OBJ file data\n * and collect the vertex, vertex normal, texture, and face information. This\n * information can then be used later on when creating your VBOs. See\n * OBJ.initMeshBuffers for an example of how to use the newly created Mesh\n */\nexport default class Mesh {\n    public vertices: number[];\n    public vertexNormals: number[];\n    public textures: number[];\n    public indices: number[];\n    public name: string = \"\";\n    public vertexMaterialIndices: number[];\n    public indicesPerMaterial: number[][] = [];\n    public materialNames: string[];\n    public materialIndices: MaterialNameToIndex;\n    public materialsByIndex: IndexToMaterial = {};\n    public tangents: number[] = [];\n    public bitangents: number[] = [];\n    public textureStride: number;\n\n    /**\n     * Create a Mesh\n     * @param {String} objectData - a string representation of an OBJ file with\n     *     newlines preserved.\n     * @param {Object} options - a JS object containing valid options. See class\n     *     documentation for options.\n     * @param {bool} options.enableWTextureCoord - Texture coordinates can have\n     *     an optional \"w\" coordinate after the u and v coordinates. This extra\n     *     value can be used in order to perform fancy transformations on the\n     *     textures themselves. Default is to truncate to only the u an v\n     *     coordinates. Passing true will provide a default value of 0 in the\n     *     event that any or all texture coordinates don't provide a w value.\n     *     Always use the textureStride attribute in order to determine the\n     *     stride length of the texture coordinates when rendering the element\n     *     array.\n     * @param {bool} options.calcTangentsAndBitangents - Calculate the tangents\n     *     and bitangents when loading of the OBJ is completed. This adds two new\n     *     attributes to the Mesh instance: `tangents` and `bitangents`.\n     */\n    constructor(objectData: string, options?: MeshOptions) {\n        options = options || {};\n        options.materials = options.materials || {};\n        options.enableWTextureCoord = !!options.enableWTextureCoord;\n\n        // the list of unique vertex, normal, texture, attributes\n        this.vertexNormals = [];\n        this.textures = [];\n        // the indicies to draw the faces\n        this.indices = [];\n        this.textureStride = options.enableWTextureCoord ? 3 : 2;\n\n        /*\n        The OBJ file format does a sort of compression when saving a model in a\n        program like Blender. There are at least 3 sections (4 including textures)\n        within the file. Each line in a section begins with the same string:\n          * 'v': indicates vertex section\n          * 'vn': indicates vertex normal section\n          * 'f': indicates the faces section\n          * 'vt': indicates vertex texture section (if textures were used on the model)\n        Each of the above sections (except for the faces section) is a list/set of\n        unique vertices.\n\n        Each line of the faces section contains a list of\n        (vertex, [texture], normal) groups.\n\n        **Note:** The following documentation will use a capital \"V\" Vertex to\n        denote the above (vertex, [texture], normal) groups whereas a lowercase\n        \"v\" vertex is used to denote an X, Y, Z coordinate.\n\n        Some examples:\n            // the texture index is optional, both formats are possible for models\n            // without a texture applied\n            f 1/25 18/46 12/31\n            f 1//25 18//46 12//31\n\n            // A 3 vertex face with texture indices\n            f 16/92/11 14/101/22 1/69/1\n\n            // A 4 vertex face\n            f 16/92/11 40/109/40 38/114/38 14/101/22\n\n        The first two lines are examples of a 3 vertex face without a texture applied.\n        The second is an example of a 3 vertex face with a texture applied.\n        The third is an example of a 4 vertex face. Note: a face can contain N\n        number of vertices.\n\n        Each number that appears in one of the groups is a 1-based index\n        corresponding to an item from the other sections (meaning that indexing\n        starts at one and *not* zero).\n\n        For example:\n            `f 16/92/11` is saying to\n              - take the 16th element from the [v] vertex array\n              - take the 92nd element from the [vt] texture array\n              - take the 11th element from the [vn] normal array\n            and together they make a unique vertex.\n        Using all 3+ unique Vertices from the face line will produce a polygon.\n\n        Now, you could just go through the OBJ file and create a new vertex for\n        each face line and WebGL will draw what appears to be the same model.\n        However, vertices will be overlapped and duplicated all over the place.\n\n        Consider a cube in 3D space centered about the origin and each side is\n        2 units long. The front face (with the positive Z-axis pointing towards\n        you) would have a Top Right vertex (looking orthogonal to its normal)\n        mapped at (1,1,1) The right face would have a Top Left vertex (looking\n        orthogonal to its normal) at (1,1,1) and the top face would have a Bottom\n        Right vertex (looking orthogonal to its normal) at (1,1,1). Each face\n        has a vertex at the same coordinates, however, three distinct vertices\n        will be drawn at the same spot.\n\n        To solve the issue of duplicate Vertices (the `(vertex, [texture], normal)`\n        groups), while iterating through the face lines, when a group is encountered\n        the whole group string ('16/92/11') is checked to see if it exists in the\n        packed.hashindices object, and if it doesn't, the indices it specifies\n        are used to look up each attribute in the corresponding attribute arrays\n        already created. The values are then copied to the corresponding unpacked\n        array (flattened to play nice with WebGL's ELEMENT_ARRAY_BUFFER indexing),\n        the group string is added to the hashindices set and the current unpacked\n        index is used as this hashindices value so that the group of elements can\n        be reused. The unpacked index is incremented. If the group string already\n        exists in the hashindices object, its corresponding value is the index of\n        that group and is appended to the unpacked indices array.\n       */\n        const verts = [];\n        const vertNormals = [];\n        const textures = [];\n        const materialNamesByIndex = [];\n        const materialIndicesByName: MaterialNameToIndex = {};\n        // keep track of what material we've seen last\n        let currentMaterialIndex = -1;\n        let currentObjectByMaterialIndex = 0;\n        // unpacking stuff\n        const unpacked: UnpackedAttrs = {\n            verts: [],\n            norms: [],\n            textures: [],\n            hashindices: {},\n            indices: [[]],\n            materialIndices: [],\n            index: 0,\n        };\n\n        const VERTEX_RE = /^v\\s/;\n        const NORMAL_RE = /^vn\\s/;\n        const TEXTURE_RE = /^vt\\s/;\n        const FACE_RE = /^f\\s/;\n        const WHITESPACE_RE = /\\s+/;\n        const USE_MATERIAL_RE = /^usemtl/;\n\n        // array of lines separated by the newline\n        const lines = objectData.split(\"\\n\");\n\n        for (let line of lines) {\n            line = line.trim();\n            if (!line || line.startsWith(\"#\")) {\n                continue;\n            }\n            const elements = line.split(WHITESPACE_RE);\n            elements.shift();\n\n            if (VERTEX_RE.test(line)) {\n                // if this is a vertex\n                verts.push(...elements);\n            } else if (NORMAL_RE.test(line)) {\n                // if this is a vertex normal\n                vertNormals.push(...elements);\n            } else if (TEXTURE_RE.test(line)) {\n                let coords = elements;\n                // by default, the loader will only look at the U and V\n                // coordinates of the vt declaration. So, this truncates the\n                // elements to only those 2 values. If W texture coordinate\n                // support is enabled, then the texture coordinate is\n                // expected to have three values in it.\n                if (elements.length > 2 && !options.enableWTextureCoord) {\n                    coords = elements.slice(0, 2);\n                } else if (elements.length === 2 && options.enableWTextureCoord) {\n                    // If for some reason W texture coordinate support is enabled\n                    // and only the U and V coordinates are given, then we supply\n                    // the default value of 0 so that the stride length is correct\n                    // when the textures are unpacked below.\n                    coords.push(\"0\");\n                }\n                textures.push(...coords);\n            } else if (USE_MATERIAL_RE.test(line)) {\n                const materialName = elements[0];\n\n                // check to see if we've ever seen it before\n                if (!(materialName in materialIndicesByName)) {\n                    // new material we've never seen\n                    materialNamesByIndex.push(materialName);\n                    materialIndicesByName[materialName] = materialNamesByIndex.length - 1;\n                    // push new array into indices\n                    // already contains an array at index zero, don't add\n                    if (materialIndicesByName[materialName] > 0) {\n                        unpacked.indices.push([]);\n                    }\n                }\n                // keep track of the current material index\n                currentMaterialIndex = materialIndicesByName[materialName];\n                // update current index array\n                currentObjectByMaterialIndex = currentMaterialIndex;\n            } else if (FACE_RE.test(line)) {\n                // if this is a face\n                /*\n                split this face into an array of Vertex groups\n                for example:\n                   f 16/92/11 14/101/22 1/69/1\n                becomes:\n                  ['16/92/11', '14/101/22', '1/69/1'];\n                */\n\n                const triangles = triangulate(elements);\n                for (const triangle of triangles) {\n                    for (let j = 0, eleLen = triangle.length; j < eleLen; j++) {\n                        const hash = triangle[j] + \",\" + currentMaterialIndex;\n                        if (hash in unpacked.hashindices) {\n                            unpacked.indices[currentObjectByMaterialIndex].push(unpacked.hashindices[hash]);\n                        } else {\n                            /*\n                        Each element of the face line array is a Vertex which has its\n                        attributes delimited by a forward slash. This will separate\n                        each attribute into another array:\n                            '19/92/11'\n                        becomes:\n                            Vertex = ['19', '92', '11'];\n                        where\n                            Vertex[0] is the vertex index\n                            Vertex[1] is the texture index\n                            Vertex[2] is the normal index\n                         Think of faces having Vertices which are comprised of the\n                         attributes location (v), texture (vt), and normal (vn).\n                         */\n                            const vertex = triangle[j].split(\"/\");\n                            // it's possible for faces to only specify the vertex\n                            // and the normal. In this case, vertex will only have\n                            // a length of 2 and not 3 and the normal will be the\n                            // second item in the list with an index of 1.\n                            const normalIndex = vertex.length - 1;\n                            /*\n                         The verts, textures, and vertNormals arrays each contain a\n                         flattend array of coordinates.\n\n                         Because it gets confusing by referring to Vertex and then\n                         vertex (both are different in my descriptions) I will explain\n                         what's going on using the vertexNormals array:\n\n                         vertex[2] will contain the one-based index of the vertexNormals\n                         section (vn). One is subtracted from this index number to play\n                         nice with javascript's zero-based array indexing.\n\n                         Because vertexNormal is a flattened array of x, y, z values,\n                         simple pointer arithmetic is used to skip to the start of the\n                         vertexNormal, then the offset is added to get the correct\n                         component: +0 is x, +1 is y, +2 is z.\n\n                         This same process is repeated for verts and textures.\n                         */\n                            // Vertex position\n                            unpacked.verts.push(+verts[(+vertex[0] - 1) * 3 + 0]);\n                            unpacked.verts.push(+verts[(+vertex[0] - 1) * 3 + 1]);\n                            unpacked.verts.push(+verts[(+vertex[0] - 1) * 3 + 2]);\n                            // Vertex textures\n                            if (textures.length) {\n                                const stride = options.enableWTextureCoord ? 3 : 2;\n                                unpacked.textures.push(+textures[(+vertex[1] - 1) * stride + 0]);\n                                unpacked.textures.push(+textures[(+vertex[1] - 1) * stride + 1]);\n                                if (options.enableWTextureCoord) {\n                                    unpacked.textures.push(+textures[(+vertex[1] - 1) * stride + 2]);\n                                }\n                            }\n                            // Vertex normals\n                            unpacked.norms.push(+vertNormals[(+vertex[normalIndex] - 1) * 3 + 0]);\n                            unpacked.norms.push(+vertNormals[(+vertex[normalIndex] - 1) * 3 + 1]);\n                            unpacked.norms.push(+vertNormals[(+vertex[normalIndex] - 1) * 3 + 2]);\n                            // Vertex material indices\n                            unpacked.materialIndices.push(currentMaterialIndex);\n                            // add the newly created Vertex to the list of indices\n                            unpacked.hashindices[hash] = unpacked.index;\n                            unpacked.indices[currentObjectByMaterialIndex].push(unpacked.hashindices[hash]);\n                            // increment the counter\n                            unpacked.index += 1;\n                        }\n                    }\n                }\n            }\n        }\n        this.vertices = unpacked.verts;\n        this.vertexNormals = unpacked.norms;\n        this.textures = unpacked.textures;\n        this.vertexMaterialIndices = unpacked.materialIndices;\n        this.indices = unpacked.indices[currentObjectByMaterialIndex];\n        this.indicesPerMaterial = unpacked.indices;\n\n        this.materialNames = materialNamesByIndex;\n        this.materialIndices = materialIndicesByName;\n        this.materialsByIndex = {};\n\n        if (options.calcTangentsAndBitangents) {\n            this.calculateTangentsAndBitangents();\n        }\n    }\n\n    /**\n     * Calculates the tangents and bitangents of the mesh that forms an orthogonal basis together with the\n     * normal in the direction of the texture coordinates. These are useful for setting up the TBN matrix\n     * when distorting the normals through normal maps.\n     * Method derived from: http://www.opengl-tutorial.org/intermediate-tutorials/tutorial-13-normal-mapping/\n     *\n     * This method requires the normals and texture coordinates to be parsed and set up correctly.\n     * Adds the tangents and bitangents as members of the class instance.\n     */\n    calculateTangentsAndBitangents() {\n        console.assert(\n            !!(\n                this.vertices &&\n                this.vertices.length &&\n                this.vertexNormals &&\n                this.vertexNormals.length &&\n                this.textures &&\n                this.textures.length\n            ),\n            \"Missing attributes for calculating tangents and bitangents\",\n        );\n\n        const unpacked = {\n            tangents: [...new Array(this.vertices.length)].map(_ => 0),\n            bitangents: [...new Array(this.vertices.length)].map(_ => 0),\n        };\n\n        // Loop through all faces in the whole mesh\n        const indices = this.indices;\n        const vertices = this.vertices;\n        const normals = this.vertexNormals;\n        const uvs = this.textures;\n\n        for (let i = 0; i < indices.length; i += 3) {\n            const i0 = indices[i + 0];\n            const i1 = indices[i + 1];\n            const i2 = indices[i + 2];\n\n            const x_v0 = vertices[i0 * 3 + 0];\n            const y_v0 = vertices[i0 * 3 + 1];\n            const z_v0 = vertices[i0 * 3 + 2];\n\n            const x_uv0 = uvs[i0 * 2 + 0];\n            const y_uv0 = uvs[i0 * 2 + 1];\n\n            const x_v1 = vertices[i1 * 3 + 0];\n            const y_v1 = vertices[i1 * 3 + 1];\n            const z_v1 = vertices[i1 * 3 + 2];\n\n            const x_uv1 = uvs[i1 * 2 + 0];\n            const y_uv1 = uvs[i1 * 2 + 1];\n\n            const x_v2 = vertices[i2 * 3 + 0];\n            const y_v2 = vertices[i2 * 3 + 1];\n            const z_v2 = vertices[i2 * 3 + 2];\n\n            const x_uv2 = uvs[i2 * 2 + 0];\n            const y_uv2 = uvs[i2 * 2 + 1];\n\n            const x_deltaPos1 = x_v1 - x_v0;\n            const y_deltaPos1 = y_v1 - y_v0;\n            const z_deltaPos1 = z_v1 - z_v0;\n\n            const x_deltaPos2 = x_v2 - x_v0;\n            const y_deltaPos2 = y_v2 - y_v0;\n            const z_deltaPos2 = z_v2 - z_v0;\n\n            const x_uvDeltaPos1 = x_uv1 - x_uv0;\n            const y_uvDeltaPos1 = y_uv1 - y_uv0;\n\n            const x_uvDeltaPos2 = x_uv2 - x_uv0;\n            const y_uvDeltaPos2 = y_uv2 - y_uv0;\n\n            const rInv = x_uvDeltaPos1 * y_uvDeltaPos2 - y_uvDeltaPos1 * x_uvDeltaPos2;\n            const r = 1.0 / Math.abs(rInv < 0.0001 ? 1.0 : rInv);\n\n            // Tangent\n            const x_tangent = (x_deltaPos1 * y_uvDeltaPos2 - x_deltaPos2 * y_uvDeltaPos1) * r;\n            const y_tangent = (y_deltaPos1 * y_uvDeltaPos2 - y_deltaPos2 * y_uvDeltaPos1) * r;\n            const z_tangent = (z_deltaPos1 * y_uvDeltaPos2 - z_deltaPos2 * y_uvDeltaPos1) * r;\n\n            // Bitangent\n            const x_bitangent = (x_deltaPos2 * x_uvDeltaPos1 - x_deltaPos1 * x_uvDeltaPos2) * r;\n            const y_bitangent = (y_deltaPos2 * x_uvDeltaPos1 - y_deltaPos1 * x_uvDeltaPos2) * r;\n            const z_bitangent = (z_deltaPos2 * x_uvDeltaPos1 - z_deltaPos1 * x_uvDeltaPos2) * r;\n\n            // Gram-Schmidt orthogonalize\n            //t = glm::normalize(t - n * glm:: dot(n, t));\n            const x_n0 = normals[i0 * 3 + 0];\n            const y_n0 = normals[i0 * 3 + 1];\n            const z_n0 = normals[i0 * 3 + 2];\n\n            const x_n1 = normals[i1 * 3 + 0];\n            const y_n1 = normals[i1 * 3 + 1];\n            const z_n1 = normals[i1 * 3 + 2];\n\n            const x_n2 = normals[i2 * 3 + 0];\n            const y_n2 = normals[i2 * 3 + 1];\n            const z_n2 = normals[i2 * 3 + 2];\n\n            // Tangent\n            const n0_dot_t = x_tangent * x_n0 + y_tangent * y_n0 + z_tangent * z_n0;\n            const n1_dot_t = x_tangent * x_n1 + y_tangent * y_n1 + z_tangent * z_n1;\n            const n2_dot_t = x_tangent * x_n2 + y_tangent * y_n2 + z_tangent * z_n2;\n\n            const x_resTangent0 = x_tangent - x_n0 * n0_dot_t;\n            const y_resTangent0 = y_tangent - y_n0 * n0_dot_t;\n            const z_resTangent0 = z_tangent - z_n0 * n0_dot_t;\n\n            const x_resTangent1 = x_tangent - x_n1 * n1_dot_t;\n            const y_resTangent1 = y_tangent - y_n1 * n1_dot_t;\n            const z_resTangent1 = z_tangent - z_n1 * n1_dot_t;\n\n            const x_resTangent2 = x_tangent - x_n2 * n2_dot_t;\n            const y_resTangent2 = y_tangent - y_n2 * n2_dot_t;\n            const z_resTangent2 = z_tangent - z_n2 * n2_dot_t;\n\n            const magTangent0 = Math.sqrt(\n                x_resTangent0 * x_resTangent0 + y_resTangent0 * y_resTangent0 + z_resTangent0 * z_resTangent0,\n            );\n            const magTangent1 = Math.sqrt(\n                x_resTangent1 * x_resTangent1 + y_resTangent1 * y_resTangent1 + z_resTangent1 * z_resTangent1,\n            );\n            const magTangent2 = Math.sqrt(\n                x_resTangent2 * x_resTangent2 + y_resTangent2 * y_resTangent2 + z_resTangent2 * z_resTangent2,\n            );\n\n            // Bitangent\n            const n0_dot_bt = x_bitangent * x_n0 + y_bitangent * y_n0 + z_bitangent * z_n0;\n            const n1_dot_bt = x_bitangent * x_n1 + y_bitangent * y_n1 + z_bitangent * z_n1;\n            const n2_dot_bt = x_bitangent * x_n2 + y_bitangent * y_n2 + z_bitangent * z_n2;\n\n            const x_resBitangent0 = x_bitangent - x_n0 * n0_dot_bt;\n            const y_resBitangent0 = y_bitangent - y_n0 * n0_dot_bt;\n            const z_resBitangent0 = z_bitangent - z_n0 * n0_dot_bt;\n\n            const x_resBitangent1 = x_bitangent - x_n1 * n1_dot_bt;\n            const y_resBitangent1 = y_bitangent - y_n1 * n1_dot_bt;\n            const z_resBitangent1 = z_bitangent - z_n1 * n1_dot_bt;\n\n            const x_resBitangent2 = x_bitangent - x_n2 * n2_dot_bt;\n            const y_resBitangent2 = y_bitangent - y_n2 * n2_dot_bt;\n            const z_resBitangent2 = z_bitangent - z_n2 * n2_dot_bt;\n\n            const magBitangent0 = Math.sqrt(\n                x_resBitangent0 * x_resBitangent0 +\n                    y_resBitangent0 * y_resBitangent0 +\n                    z_resBitangent0 * z_resBitangent0,\n            );\n            const magBitangent1 = Math.sqrt(\n                x_resBitangent1 * x_resBitangent1 +\n                    y_resBitangent1 * y_resBitangent1 +\n                    z_resBitangent1 * z_resBitangent1,\n            );\n            const magBitangent2 = Math.sqrt(\n                x_resBitangent2 * x_resBitangent2 +\n                    y_resBitangent2 * y_resBitangent2 +\n                    z_resBitangent2 * z_resBitangent2,\n            );\n\n            unpacked.tangents[i0 * 3 + 0] += x_resTangent0 / magTangent0;\n            unpacked.tangents[i0 * 3 + 1] += y_resTangent0 / magTangent0;\n            unpacked.tangents[i0 * 3 + 2] += z_resTangent0 / magTangent0;\n\n            unpacked.tangents[i1 * 3 + 0] += x_resTangent1 / magTangent1;\n            unpacked.tangents[i1 * 3 + 1] += y_resTangent1 / magTangent1;\n            unpacked.tangents[i1 * 3 + 2] += z_resTangent1 / magTangent1;\n\n            unpacked.tangents[i2 * 3 + 0] += x_resTangent2 / magTangent2;\n            unpacked.tangents[i2 * 3 + 1] += y_resTangent2 / magTangent2;\n            unpacked.tangents[i2 * 3 + 2] += z_resTangent2 / magTangent2;\n\n            unpacked.bitangents[i0 * 3 + 0] += x_resBitangent0 / magBitangent0;\n            unpacked.bitangents[i0 * 3 + 1] += y_resBitangent0 / magBitangent0;\n            unpacked.bitangents[i0 * 3 + 2] += z_resBitangent0 / magBitangent0;\n\n            unpacked.bitangents[i1 * 3 + 0] += x_resBitangent1 / magBitangent1;\n            unpacked.bitangents[i1 * 3 + 1] += y_resBitangent1 / magBitangent1;\n            unpacked.bitangents[i1 * 3 + 2] += z_resBitangent1 / magBitangent1;\n\n            unpacked.bitangents[i2 * 3 + 0] += x_resBitangent2 / magBitangent2;\n            unpacked.bitangents[i2 * 3 + 1] += y_resBitangent2 / magBitangent2;\n            unpacked.bitangents[i2 * 3 + 2] += z_resBitangent2 / magBitangent2;\n\n            // TODO: check handedness\n        }\n\n        this.tangents = unpacked.tangents;\n        this.bitangents = unpacked.bitangents;\n    }\n\n    /**\n     * @param layout - A {@link Layout} object that describes the\n     * desired memory layout of the generated buffer\n     * @return The packed array in the ... TODO\n     */\n    makeBufferData(layout: Layout): ArrayBufferWithItemSize {\n        const numItems = this.vertices.length / 3;\n        const buffer: ArrayBufferWithItemSize = new ArrayBuffer(layout.stride * numItems);\n        buffer.numItems = numItems;\n        const dataView = new DataView(buffer);\n        for (let i = 0, vertexOffset = 0; i < numItems; i++) {\n            vertexOffset = i * layout.stride;\n            // copy in the vertex data in the order and format given by the\n            // layout param\n            for (const attribute of layout.attributes) {\n                const offset = vertexOffset + layout.attributeMap[attribute.key].offset;\n                switch (attribute.key) {\n                    case Layout.POSITION.key:\n                        dataView.setFloat32(offset, this.vertices[i * 3], true);\n                        dataView.setFloat32(offset + 4, this.vertices[i * 3 + 1], true);\n                        dataView.setFloat32(offset + 8, this.vertices[i * 3 + 2], true);\n                        break;\n                    case Layout.UV.key:\n                        dataView.setFloat32(offset, this.textures[i * 2], true);\n                        dataView.setFloat32(offset + 4, this.textures[i * 2 + 1], true);\n                        break;\n                    case Layout.NORMAL.key:\n                        dataView.setFloat32(offset, this.vertexNormals[i * 3], true);\n                        dataView.setFloat32(offset + 4, this.vertexNormals[i * 3 + 1], true);\n                        dataView.setFloat32(offset + 8, this.vertexNormals[i * 3 + 2], true);\n                        break;\n                    case Layout.MATERIAL_INDEX.key:\n                        dataView.setInt16(offset, this.vertexMaterialIndices[i], true);\n                        break;\n                    case Layout.AMBIENT.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setFloat32(offset, material.ambient[0], true);\n                        dataView.setFloat32(offset + 4, material.ambient[1], true);\n                        dataView.setFloat32(offset + 8, material.ambient[2], true);\n                        break;\n                    }\n                    case Layout.DIFFUSE.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setFloat32(offset, material.diffuse[0], true);\n                        dataView.setFloat32(offset + 4, material.diffuse[1], true);\n                        dataView.setFloat32(offset + 8, material.diffuse[2], true);\n                        break;\n                    }\n                    case Layout.SPECULAR.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setFloat32(offset, material.specular[0], true);\n                        dataView.setFloat32(offset + 4, material.specular[1], true);\n                        dataView.setFloat32(offset + 8, material.specular[2], true);\n                        break;\n                    }\n                    case Layout.SPECULAR_EXPONENT.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setFloat32(offset, material.specularExponent, true);\n                        break;\n                    }\n                    case Layout.EMISSIVE.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setFloat32(offset, material.emissive[0], true);\n                        dataView.setFloat32(offset + 4, material.emissive[1], true);\n                        dataView.setFloat32(offset + 8, material.emissive[2], true);\n                        break;\n                    }\n                    case Layout.TRANSMISSION_FILTER.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setFloat32(offset, material.transmissionFilter[0], true);\n                        dataView.setFloat32(offset + 4, material.transmissionFilter[1], true);\n                        dataView.setFloat32(offset + 8, material.transmissionFilter[2], true);\n                        break;\n                    }\n                    case Layout.DISSOLVE.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setFloat32(offset, material.dissolve, true);\n                        break;\n                    }\n                    case Layout.ILLUMINATION.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setInt16(offset, material.illumination, true);\n                        break;\n                    }\n                    case Layout.REFRACTION_INDEX.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setFloat32(offset, material.refractionIndex, true);\n                        break;\n                    }\n                    case Layout.SHARPNESS.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setFloat32(offset, material.sharpness, true);\n                        break;\n                    }\n                    case Layout.ANTI_ALIASING.key: {\n                        const materialIndex = this.vertexMaterialIndices[i];\n                        const material = this.materialsByIndex[materialIndex];\n                        if (!material) {\n                            console.warn(\n                                'Material \"' +\n                                    this.materialNames[materialIndex] +\n                                    '\" not found in mesh. Did you forget to call addMaterialLibrary(...)?\"',\n                            );\n                            break;\n                        }\n                        dataView.setInt16(offset, material.antiAliasing ? 1 : 0, true);\n                        break;\n                    }\n                }\n            }\n        }\n        return buffer;\n    }\n\n    makeIndexBufferData(): Uint16ArrayWithItemSize {\n        const buffer: Uint16ArrayWithItemSize = new Uint16Array(this.indices);\n        buffer.numItems = this.indices.length;\n        return buffer;\n    }\n\n    makeIndexBufferDataForMaterials(...materialIndices: Array<number>): Uint16ArrayWithItemSize {\n        const indices: number[] = new Array<number>().concat(\n            ...materialIndices.map(mtlIdx => this.indicesPerMaterial[mtlIdx]),\n        );\n        const buffer: Uint16ArrayWithItemSize = new Uint16Array(indices);\n        buffer.numItems = indices.length;\n        return buffer;\n    }\n\n    addMaterialLibrary(mtl: MaterialLibrary) {\n        for (const name in mtl.materials) {\n            if (!(name in this.materialIndices)) {\n                // This material is not referenced by the mesh\n                continue;\n            }\n\n            const material = mtl.materials[name];\n\n            // Find the material index for this material\n            const materialIndex = this.materialIndices[material.name];\n\n            // Put the material into the materialsByIndex object at the right\n            // spot as determined when the obj file was parsed\n            this.materialsByIndex[materialIndex] = material;\n        }\n    }\n}\n\nfunction* triangulate(elements: string[]) {\n    if (elements.length <= 3) {\n        yield elements;\n    } else if (elements.length === 4) {\n        yield [elements[0], elements[1], elements[2]];\n        yield [elements[2], elements[3], elements[0]];\n    } else {\n        for (let i = 1; i < elements.length - 1; i++) {\n            yield [elements[0], elements[i], elements[i + 1]];\n        }\n    }\n}\n","import Mesh from \"./mesh\";\nimport { MaterialLibrary, TextureMapData } from \"./material\";\n\nfunction downloadMtlTextures(mtl: MaterialLibrary, root: string) {\n    const mapAttributes = [\n        \"mapDiffuse\",\n        \"mapAmbient\",\n        \"mapSpecular\",\n        \"mapDissolve\",\n        \"mapBump\",\n        \"mapDisplacement\",\n        \"mapDecal\",\n        \"mapEmissive\",\n    ];\n    if (!root.endsWith(\"/\")) {\n        root += \"/\";\n    }\n    const textures = [];\n\n    for (const materialName in mtl.materials) {\n        if (!mtl.materials.hasOwnProperty(materialName)) {\n            continue;\n        }\n        const material = mtl.materials[materialName];\n\n        for (const attr of mapAttributes) {\n            const mapData = (material as any)[attr] as TextureMapData;\n            if (!mapData || !mapData.filename) {\n                continue;\n            }\n            const url = root + mapData.filename;\n            textures.push(\n                fetch(url)\n                    .then(response => {\n                        if (!response.ok) {\n                            throw new Error();\n                        }\n                        return response.blob();\n                    })\n                    .then(function(data) {\n                        const image = new Image();\n                        image.src = URL.createObjectURL(data);\n                        mapData.texture = image;\n                        return new Promise(resolve => (image.onload = resolve));\n                    })\n                    .catch(() => {\n                        console.error(`Unable to download texture: ${url}`);\n                    }),\n            );\n        }\n    }\n\n    return Promise.all(textures);\n}\n\nfunction getMtl(modelOptions: DownloadModelsOptions): string {\n    if (!(typeof modelOptions.mtl === \"string\")) {\n        return modelOptions.obj.replace(/\\.obj$/, \".mtl\");\n    }\n\n    return modelOptions.mtl;\n}\n\nexport interface DownloadModelsOptions {\n    obj: string;\n    mtl?: boolean | string;\n    downloadMtlTextures?: boolean;\n    mtlTextureRoot?: string;\n    name?: string;\n    indicesPerMaterial?: boolean;\n    calcTangentsAndBitangents?: boolean;\n}\n\ntype ModelPromises = [Promise<string>, Promise<Mesh>, undefined | Promise<MaterialLibrary>];\nexport type MeshMap = { [name: string]: Mesh };\n/**\n * Accepts a list of model request objects and returns a Promise that\n * resolves when all models have been downloaded and parsed.\n *\n * The list of model objects follow this interface:\n * {\n *  obj: 'path/to/model.obj',\n *  mtl: true | 'path/to/model.mtl',\n *  downloadMtlTextures: true | false\n *  mtlTextureRoot: '/models/suzanne/maps'\n *  name: 'suzanne'\n * }\n *\n * The `obj` attribute is required and should be the path to the\n * model's .obj file relative to the current repo (absolute URLs are\n * suggested).\n *\n * The `mtl` attribute is optional and can either be a boolean or\n * a path to the model's .mtl file relative to the current URL. If\n * the value is `true`, then the path and basename given for the `obj`\n * attribute is used replacing the .obj suffix for .mtl\n * E.g.: {obj: 'models/foo.obj', mtl: true} would search for 'models/foo.mtl'\n *\n * The `name` attribute is optional and is a human friendly name to be\n * included with the parsed OBJ and MTL files. If not given, the base .obj\n * filename will be used.\n *\n * The `downloadMtlTextures` attribute is a flag for automatically downloading\n * any images found in the MTL file and attaching them to each Material\n * created from that file. For example, if material.mapDiffuse is set (there\n * was data in the MTL file), then material.mapDiffuse.texture will contain\n * the downloaded image. This option defaults to `true`. By default, the MTL's\n * URL will be used to determine the location of the images.\n *\n * The `mtlTextureRoot` attribute is optional and should point to the location\n * on the server that this MTL's texture files are located. The default is to\n * use the MTL file's location.\n *\n * @returns {Promise} the result of downloading the given list of models. The\n * promise will resolve with an object whose keys are the names of the models\n * and the value is its Mesh object. Each Mesh object will automatically\n * have its addMaterialLibrary() method called to set the given MTL data (if given).\n */\nexport function downloadModels(models: DownloadModelsOptions[]): Promise<MeshMap> {\n    const finished = [];\n\n    for (const model of models) {\n        if (!model.obj) {\n            throw new Error(\n                '\"obj\" attribute of model object not set. The .obj file is required to be set ' +\n                    \"in order to use downloadModels()\",\n            );\n        }\n\n        const options = {\n            indicesPerMaterial: !!model.indicesPerMaterial,\n            calcTangentsAndBitangents: !!model.calcTangentsAndBitangents,\n        };\n\n        // if the name is not provided, dervive it from the given OBJ\n        let name = model.name;\n        if (!name) {\n            const parts = model.obj.split(\"/\");\n            name = parts[parts.length - 1].replace(\".obj\", \"\");\n        }\n        const namePromise = Promise.resolve(name);\n\n        const meshPromise = fetch(model.obj)\n            .then(response => response.text())\n            .then(data => {\n                return new Mesh(data, options);\n            });\n\n        let mtlPromise;\n        // Download MaterialLibrary file?\n        if (model.mtl) {\n            const mtl = getMtl(model);\n            mtlPromise = fetch(mtl)\n                .then(response => response.text())\n                .then(\n                    (data: string): Promise<[MaterialLibrary, any]> => {\n                        const material = new MaterialLibrary(data);\n                        if (model.downloadMtlTextures !== false) {\n                            let root = model.mtlTextureRoot;\n                            if (!root) {\n                                // get the directory of the MTL file as default\n                                root = mtl.substr(0, mtl.lastIndexOf(\"/\"));\n                            }\n                            // downloadMtlTextures returns a Promise that\n                            // is resolved once all of the images it\n                            // contains are downloaded. These are then\n                            // attached to the map data objects\n                            return Promise.all([Promise.resolve(material), downloadMtlTextures(material, root)]);\n                        }\n                        return Promise.all([Promise.resolve(material), undefined]);\n                    },\n                )\n                .then((value: [MaterialLibrary, any]) => {\n                    return value[0];\n                });\n        }\n\n        const parsed: ModelPromises = [namePromise, meshPromise, mtlPromise];\n        finished.push(Promise.all<string, Mesh, MaterialLibrary | undefined>(parsed));\n    }\n\n    return Promise.all(finished).then(ms => {\n        // the \"finished\" promise is a list of name, Mesh instance,\n        // and MaterialLibary instance. This unpacks and returns an\n        // object mapping name to Mesh (Mesh points to MTL).\n        const models: MeshMap = {};\n\n        for (const model of ms) {\n            const [name, mesh, mtl] = model;\n            mesh.name = name;\n            if (mtl) {\n                mesh.addMaterialLibrary(mtl);\n            }\n            models[name] = mesh;\n        }\n\n        return models;\n    });\n}\n\nexport interface NameAndUrls {\n    [meshName: string]: string;\n}\n\n/**\n * Takes in an object of `mesh_name`, `'/url/to/OBJ/file'` pairs and a callback\n * function. Each OBJ file will be ajaxed in and automatically converted to\n * an OBJ.Mesh. When all files have successfully downloaded the callback\n * function provided will be called and passed in an object containing\n * the newly created meshes.\n *\n * **Note:** In order to use this function as a way to download meshes, a\n * webserver of some sort must be used.\n *\n * @param {Object} nameAndAttrs an object where the key is the name of the mesh and the value is the url to that mesh's OBJ file\n *\n * @param {Function} completionCallback should contain a function that will take one parameter: an object array where the keys will be the unique object name and the value will be a Mesh object\n *\n * @param {Object} meshes In case other meshes are loaded separately or if a previously declared variable is desired to be used, pass in a (possibly empty) json object of the pattern: { '<mesh_name>': OBJ.Mesh }\n *\n */\nexport function downloadMeshes(\n    nameAndURLs: NameAndUrls,\n    completionCallback: (meshes: MeshMap) => void,\n    meshes: MeshMap,\n) {\n    if (meshes === undefined) {\n        meshes = {};\n    }\n\n    const completed: Promise<[string, Mesh]>[] = [];\n\n    for (const mesh_name in nameAndURLs) {\n        if (!nameAndURLs.hasOwnProperty(mesh_name)) {\n            continue;\n        }\n        const url = nameAndURLs[mesh_name];\n        completed.push(\n            fetch(url)\n                .then(response => response.text())\n                .then(data => {\n                    return [mesh_name, new Mesh(data)] as [string, Mesh];\n                }),\n        );\n    }\n\n    Promise.all(completed).then(ms => {\n        for (const [name, mesh] of ms) {\n            meshes[name] = mesh;\n        }\n\n        return completionCallback(meshes);\n    });\n}\n\nexport interface ExtendedGLBuffer extends WebGLBuffer {\n    itemSize: number;\n    numItems: number;\n}\n\nfunction _buildBuffer(gl: WebGLRenderingContext, type: GLenum, data: number[], itemSize: number): ExtendedGLBuffer {\n    const buffer = gl.createBuffer() as ExtendedGLBuffer;\n    const arrayView = type === gl.ARRAY_BUFFER ? Float32Array : Uint16Array;\n    gl.bindBuffer(type, buffer);\n    gl.bufferData(type, new arrayView(data), gl.STATIC_DRAW);\n    buffer.itemSize = itemSize;\n    buffer.numItems = data.length / itemSize;\n    return buffer;\n}\n\nexport interface MeshWithBuffers extends Mesh {\n    normalBuffer: ExtendedGLBuffer;\n    textureBuffer: ExtendedGLBuffer;\n    vertexBuffer: ExtendedGLBuffer;\n    indexBuffer: ExtendedGLBuffer;\n}\n\n/**\n * Takes in the WebGL context and a Mesh, then creates and appends the buffers\n * to the mesh object as attributes.\n *\n * @param {WebGLRenderingContext} gl the `canvas.getContext('webgl')` context instance\n * @param {Mesh} mesh a single `OBJ.Mesh` instance\n *\n * The newly created mesh attributes are:\n *\n * Attrbute | Description\n * :--- | ---\n * **normalBuffer**       |contains the model&#39;s Vertex Normals\n * normalBuffer.itemSize  |set to 3 items\n * normalBuffer.numItems  |the total number of vertex normals\n * |\n * **textureBuffer**      |contains the model&#39;s Texture Coordinates\n * textureBuffer.itemSize |set to 2 items\n * textureBuffer.numItems |the number of texture coordinates\n * |\n * **vertexBuffer**       |contains the model&#39;s Vertex Position Coordinates (does not include w)\n * vertexBuffer.itemSize  |set to 3 items\n * vertexBuffer.numItems  |the total number of vertices\n * |\n * **indexBuffer**        |contains the indices of the faces\n * indexBuffer.itemSize   |is set to 1\n * indexBuffer.numItems   |the total number of indices\n *\n * A simple example (a lot of steps are missing, so don't copy and paste):\n *\n *     const gl   = canvas.getContext('webgl'),\n *         mesh = OBJ.Mesh(obj_file_data);\n *     // compile the shaders and create a shader program\n *     const shaderProgram = gl.createProgram();\n *     // compilation stuff here\n *     ...\n *     // make sure you have vertex, vertex normal, and texture coordinate\n *     // attributes located in your shaders and attach them to the shader program\n *     shaderProgram.vertexPositionAttribute = gl.getAttribLocation(shaderProgram, \"aVertexPosition\");\n *     gl.enableVertexAttribArray(shaderProgram.vertexPositionAttribute);\n *\n *     shaderProgram.vertexNormalAttribute = gl.getAttribLocation(shaderProgram, \"aVertexNormal\");\n *     gl.enableVertexAttribArray(shaderProgram.vertexNormalAttribute);\n *\n *     shaderProgram.textureCoordAttribute = gl.getAttribLocation(shaderProgram, \"aTextureCoord\");\n *     gl.enableVertexAttribArray(shaderProgram.textureCoordAttribute);\n *\n *     // create and initialize the vertex, vertex normal, and texture coordinate buffers\n *     // and save on to the mesh object\n *     OBJ.initMeshBuffers(gl, mesh);\n *\n *     // now to render the mesh\n *     gl.bindBuffer(gl.ARRAY_BUFFER, mesh.vertexBuffer);\n *     gl.vertexAttribPointer(shaderProgram.vertexPositionAttribute, mesh.vertexBuffer.itemSize, gl.FLOAT, false, 0, 0);\n *     // it's possible that the mesh doesn't contain\n *     // any texture coordinates (e.g. suzanne.obj in the development branch).\n *     // in this case, the texture vertexAttribArray will need to be disabled\n *     // before the call to drawElements\n *     if(!mesh.textures.length){\n *       gl.disableVertexAttribArray(shaderProgram.textureCoordAttribute);\n *     }\n *     else{\n *       // if the texture vertexAttribArray has been previously\n *       // disabled, then it needs to be re-enabled\n *       gl.enableVertexAttribArray(shaderProgram.textureCoordAttribute);\n *       gl.bindBuffer(gl.ARRAY_BUFFER, mesh.textureBuffer);\n *       gl.vertexAttribPointer(shaderProgram.textureCoordAttribute, mesh.textureBuffer.itemSize, gl.FLOAT, false, 0, 0);\n *     }\n *\n *     gl.bindBuffer(gl.ARRAY_BUFFER, mesh.normalBuffer);\n *     gl.vertexAttribPointer(shaderProgram.vertexNormalAttribute, mesh.normalBuffer.itemSize, gl.FLOAT, false, 0, 0);\n *\n *     gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, model.mesh.indexBuffer);\n *     gl.drawElements(gl.TRIANGLES, model.mesh.indexBuffer.numItems, gl.UNSIGNED_SHORT, 0);\n */\nexport function initMeshBuffers(gl: WebGLRenderingContext, mesh: Mesh): MeshWithBuffers {\n    (mesh as MeshWithBuffers).normalBuffer = _buildBuffer(gl, gl.ARRAY_BUFFER, mesh.vertexNormals, 3);\n    (mesh as MeshWithBuffers).textureBuffer = _buildBuffer(gl, gl.ARRAY_BUFFER, mesh.textures, mesh.textureStride);\n    (mesh as MeshWithBuffers).vertexBuffer = _buildBuffer(gl, gl.ARRAY_BUFFER, mesh.vertices, 3);\n    (mesh as MeshWithBuffers).indexBuffer = _buildBuffer(gl, gl.ELEMENT_ARRAY_BUFFER, mesh.indices, 1);\n\n    return mesh as MeshWithBuffers;\n}\n\nexport function deleteMeshBuffers(gl: WebGLRenderingContext, mesh: MeshWithBuffers) {\n    gl.deleteBuffer(mesh.normalBuffer);\n    gl.deleteBuffer(mesh.textureBuffer);\n    gl.deleteBuffer(mesh.vertexBuffer);\n    gl.deleteBuffer(mesh.indexBuffer);\n}\n"],"sourceRoot":""} \ No newline at end of file diff --git a/dist/webgl-obj-loader.min.js b/dist/webgl-obj-loader.min.js index 89ecf53..0a5122e 100644 --- a/dist/webgl-obj-loader.min.js +++ b/dist/webgl-obj-loader.min.js @@ -6,7 +6,7 @@ /*!***********************!*\ !*** ./src/layout.ts ***! \***********************/ -/*! exports provided: TYPES, DuplicateAttributeException, Attribute, Layout */function(module,__webpack_exports__,__webpack_require__){"use strict";eval('__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "TYPES", function() { return TYPES; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "DuplicateAttributeException", function() { return DuplicateAttributeException; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "Attribute", function() { return Attribute; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "Layout", function() { return Layout; });\nvar TYPES;\n(function (TYPES) {\n TYPES[TYPES["BYTE"] = 1] = "BYTE";\n TYPES[TYPES["UNSIGNED_BYTE"] = 1] = "UNSIGNED_BYTE";\n TYPES[TYPES["SHORT"] = 2] = "SHORT";\n TYPES[TYPES["UNSIGNED_SHORT"] = 2] = "UNSIGNED_SHORT";\n TYPES[TYPES["FLOAT"] = 4] = "FLOAT";\n})(TYPES || (TYPES = {}));\n/**\n * An exception for when two or more of the same attributes are found in the\n * same layout.\n * @private\n */\nclass DuplicateAttributeException extends Error {\n /**\n * Create a DuplicateAttributeException\n * @param {Attribute} attribute - The attribute that was found more than\n * once in the {@link Layout}\n */\n constructor(attribute) {\n super(`found duplicate attribute: ${attribute.key}`);\n }\n}\n/**\n * Represents how a vertex attribute should be packed into an buffer.\n * @private\n */\nclass Attribute {\n /**\n * Create an attribute. Do not call this directly, use the predefined\n * constants.\n * @param {string} key - The name of this attribute as if it were a key in\n * an Object. Use the camel case version of the upper snake case\n * const name.\n * @param {number} size - The number of components per vertex attribute.\n * Must be 1, 2, 3, or 4.\n * @param {string} type - The data type of each component for this\n * attribute. Possible values:
\n * "BYTE": signed 8-bit integer, with values in [-128, 127]
\n * "SHORT": signed 16-bit integer, with values in\n * [-32768, 32767]
\n * "UNSIGNED_BYTE": unsigned 8-bit integer, with values in\n * [0, 255]
\n * "UNSIGNED_SHORT": unsigned 16-bit integer, with values in\n * [0, 65535]
\n * "FLOAT": 32-bit floating point number\n * @param {boolean} normalized - Whether integer data values should be\n * normalized when being casted to a float.
\n * If true, signed integers are normalized to [-1, 1].
\n * If true, unsigned integers are normalized to [0, 1].
\n * For type "FLOAT", this parameter has no effect.\n */\n constructor(key, size, type, normalized = false) {\n this.key = key;\n this.size = size;\n this.type = type;\n this.normalized = normalized;\n this.sizeOfType = this.type;\n this.sizeInBytes = this.sizeOfType * size;\n }\n}\n/**\n * A class to represent the memory layout for a vertex attribute array. Used by\n * {@link Mesh}\'s TBD(...) method to generate a packed array from mesh data.\n *

\n * Layout can sort of be thought of as a C-style struct declaration.\n * {@link Mesh}\'s TBD(...) method will use the {@link Layout} instance to\n * pack an array in the given attribute order.\n *

\n * Layout also is very helpful when calling a WebGL context\'s\n * vertexAttribPointer method. If you\'ve created a buffer using\n * a Layout instance, then the same Layout instance can be used to determine\n * the size, type, normalized, stride, and offset parameters for\n * vertexAttribPointer.\n *

\n * For example:\n *

\n *\n * const index = glctx.getAttribLocation(shaderProgram, "pos");\n * glctx.vertexAttribPointer(\n *   layout.position.size,\n *   glctx[layout.position.type],\n *   layout.position.normalized,\n *   layout.position.stride,\n *   layout.position.offset);\n * 
\n * @see {@link Mesh}\n */\nclass Layout {\n /**\n * Create a Layout object. This constructor will throw if any duplicate\n * attributes are given.\n * @param {Array} ...attributes - An ordered list of attributes that\n * describe the desired memory layout for each vertex attribute.\n *

\n *\n * @see {@link Mesh}\n */\n constructor(...attributes) {\n this.attributes = attributes;\n this.attributeMap = {};\n let offset = 0;\n let maxStrideMultiple = 0;\n for (const attribute of attributes) {\n if (this.attributeMap[attribute.key]) {\n throw new DuplicateAttributeException(attribute);\n }\n // Add padding to satisfy WebGL\'s requirement that all\n // vertexAttribPointer calls have an offset that is a multiple of\n // the type size.\n if (offset % attribute.sizeOfType !== 0) {\n offset += attribute.sizeOfType - (offset % attribute.sizeOfType);\n console.warn("Layout requires padding before " + attribute.key + " attribute");\n }\n this.attributeMap[attribute.key] = {\n attribute: attribute,\n size: attribute.size,\n type: attribute.type,\n normalized: attribute.normalized,\n offset: offset,\n };\n offset += attribute.sizeInBytes;\n maxStrideMultiple = Math.max(maxStrideMultiple, attribute.sizeOfType);\n }\n // Add padding to the end to satisfy WebGL\'s requirement that all\n // vertexAttribPointer calls have a stride that is a multiple of the\n // type size. Because we\'re putting differently sized attributes into\n // the same buffer, it must be padded to a multiple of the largest\n // type size.\n if (offset % maxStrideMultiple !== 0) {\n offset += maxStrideMultiple - (offset % maxStrideMultiple);\n console.warn("Layout requires padding at the back");\n }\n this.stride = offset;\n for (const attribute of attributes) {\n this.attributeMap[attribute.key].stride = this.stride;\n }\n }\n}\n// Geometry attributes\n/**\n * Attribute layout to pack a vertex\'s x, y, & z as floats\n *\n * @see {@link Layout}\n */\nLayout.POSITION = new Attribute("position", 3, TYPES.FLOAT);\n/**\n * Attribute layout to pack a vertex\'s normal\'s x, y, & z as floats\n *\n * @see {@link Layout}\n */\nLayout.NORMAL = new Attribute("normal", 3, TYPES.FLOAT);\n/**\n * Attribute layout to pack a vertex\'s normal\'s x, y, & z as floats.\n *

\n * This value will be computed on-the-fly based on the texture coordinates.\n * If no texture coordinates are available, the generated value will default to\n * 0, 0, 0.\n *\n * @see {@link Layout}\n */\nLayout.TANGENT = new Attribute("tangent", 3, TYPES.FLOAT);\n/**\n * Attribute layout to pack a vertex\'s normal\'s bitangent x, y, & z as floats.\n *

\n * This value will be computed on-the-fly based on the texture coordinates.\n * If no texture coordinates are available, the generated value will default to\n * 0, 0, 0.\n * @see {@link Layout}\n */\nLayout.BITANGENT = new Attribute("bitangent", 3, TYPES.FLOAT);\n/**\n * Attribute layout to pack a vertex\'s texture coordinates\' u & v as floats\n *\n * @see {@link Layout}\n */\nLayout.UV = new Attribute("uv", 2, TYPES.FLOAT);\n// Material attributes\n/**\n * Attribute layout to pack an unsigned short to be interpreted as a the index\n * into a {@link Mesh}\'s materials list.\n *

\n * The intention of this value is to send all of the {@link Mesh}\'s materials\n * into multiple shader uniforms and then reference the current one by this\n * vertex attribute.\n *

\n * example glsl code:\n *\n *

\n *  // this is bound using MATERIAL_INDEX\n *  attribute int materialIndex;\n *\n *  struct Material {\n *    vec3 diffuse;\n *    vec3 specular;\n *    vec3 specularExponent;\n *  };\n *\n *  uniform Material materials[MAX_MATERIALS];\n *\n *  // ...\n *\n *  vec3 diffuse = materials[materialIndex];\n *\n * 
\n * TODO: More description & test to make sure subscripting by attributes even\n * works for webgl\n *\n * @see {@link Layout}\n */\nLayout.MATERIAL_INDEX = new Attribute("materialIndex", 1, TYPES.SHORT);\nLayout.MATERIAL_ENABLED = new Attribute("materialEnabled", 1, TYPES.UNSIGNED_SHORT);\nLayout.AMBIENT = new Attribute("ambient", 3, TYPES.FLOAT);\nLayout.DIFFUSE = new Attribute("diffuse", 3, TYPES.FLOAT);\nLayout.SPECULAR = new Attribute("specular", 3, TYPES.FLOAT);\nLayout.SPECULAR_EXPONENT = new Attribute("specularExponent", 3, TYPES.FLOAT);\nLayout.EMISSIVE = new Attribute("emissive", 3, TYPES.FLOAT);\nLayout.TRANSMISSION_FILTER = new Attribute("transmissionFilter", 3, TYPES.FLOAT);\nLayout.DISSOLVE = new Attribute("dissolve", 1, TYPES.FLOAT);\nLayout.ILLUMINATION = new Attribute("illumination", 1, TYPES.UNSIGNED_SHORT);\nLayout.REFRACTION_INDEX = new Attribute("refractionIndex", 1, TYPES.FLOAT);\nLayout.SHARPNESS = new Attribute("sharpness", 1, TYPES.FLOAT);\nLayout.MAP_DIFFUSE = new Attribute("mapDiffuse", 1, TYPES.SHORT);\nLayout.MAP_AMBIENT = new Attribute("mapAmbient", 1, TYPES.SHORT);\nLayout.MAP_SPECULAR = new Attribute("mapSpecular", 1, TYPES.SHORT);\nLayout.MAP_SPECULAR_EXPONENT = new Attribute("mapSpecularExponent", 1, TYPES.SHORT);\nLayout.MAP_DISSOLVE = new Attribute("mapDissolve", 1, TYPES.SHORT);\nLayout.ANTI_ALIASING = new Attribute("antiAliasing", 1, TYPES.UNSIGNED_SHORT);\nLayout.MAP_BUMP = new Attribute("mapBump", 1, TYPES.SHORT);\nLayout.MAP_DISPLACEMENT = new Attribute("mapDisplacement", 1, TYPES.SHORT);\nLayout.MAP_DECAL = new Attribute("mapDecal", 1, TYPES.SHORT);\nLayout.MAP_EMISSIVE = new Attribute("mapEmissive", 1, TYPES.SHORT);\n\n\n//# sourceURL=webpack://OBJ/./src/layout.ts?')},"./src/material.ts": +/*! exports provided: TYPES, DuplicateAttributeException, Attribute, Layout */function(module,__webpack_exports__,__webpack_require__){"use strict";eval('__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "TYPES", function() { return TYPES; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "DuplicateAttributeException", function() { return DuplicateAttributeException; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "Attribute", function() { return Attribute; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "Layout", function() { return Layout; });\nvar TYPES;\n(function (TYPES) {\n TYPES["BYTE"] = "BYTE";\n TYPES["UNSIGNED_BYTE"] = "UNSIGNED_BYTE";\n TYPES["SHORT"] = "SHORT";\n TYPES["UNSIGNED_SHORT"] = "UNSIGNED_SHORT";\n TYPES["FLOAT"] = "FLOAT";\n})(TYPES || (TYPES = {}));\n/**\n * An exception for when two or more of the same attributes are found in the\n * same layout.\n * @private\n */\nclass DuplicateAttributeException extends Error {\n /**\n * Create a DuplicateAttributeException\n * @param {Attribute} attribute - The attribute that was found more than\n * once in the {@link Layout}\n */\n constructor(attribute) {\n super(`found duplicate attribute: ${attribute.key}`);\n }\n}\n/**\n * Represents how a vertex attribute should be packed into an buffer.\n * @private\n */\nclass Attribute {\n /**\n * Create an attribute. Do not call this directly, use the predefined\n * constants.\n * @param {string} key - The name of this attribute as if it were a key in\n * an Object. Use the camel case version of the upper snake case\n * const name.\n * @param {number} size - The number of components per vertex attribute.\n * Must be 1, 2, 3, or 4.\n * @param {string} type - The data type of each component for this\n * attribute. Possible values:
\n * "BYTE": signed 8-bit integer, with values in [-128, 127]
\n * "SHORT": signed 16-bit integer, with values in\n * [-32768, 32767]
\n * "UNSIGNED_BYTE": unsigned 8-bit integer, with values in\n * [0, 255]
\n * "UNSIGNED_SHORT": unsigned 16-bit integer, with values in\n * [0, 65535]
\n * "FLOAT": 32-bit floating point number\n * @param {boolean} normalized - Whether integer data values should be\n * normalized when being casted to a float.
\n * If true, signed integers are normalized to [-1, 1].
\n * If true, unsigned integers are normalized to [0, 1].
\n * For type "FLOAT", this parameter has no effect.\n */\n constructor(key, size, type, normalized = false) {\n this.key = key;\n this.size = size;\n this.type = type;\n this.normalized = normalized;\n switch (type) {\n case "BYTE":\n case "UNSIGNED_BYTE":\n this.sizeOfType = 1;\n break;\n case "SHORT":\n case "UNSIGNED_SHORT":\n this.sizeOfType = 2;\n break;\n case "FLOAT":\n this.sizeOfType = 4;\n break;\n default:\n throw new Error(`Unknown gl type: ${type}`);\n }\n this.sizeInBytes = this.sizeOfType * size;\n }\n}\n/**\n * A class to represent the memory layout for a vertex attribute array. Used by\n * {@link Mesh}\'s TBD(...) method to generate a packed array from mesh data.\n *

\n * Layout can sort of be thought of as a C-style struct declaration.\n * {@link Mesh}\'s TBD(...) method will use the {@link Layout} instance to\n * pack an array in the given attribute order.\n *

\n * Layout also is very helpful when calling a WebGL context\'s\n * vertexAttribPointer method. If you\'ve created a buffer using\n * a Layout instance, then the same Layout instance can be used to determine\n * the size, type, normalized, stride, and offset parameters for\n * vertexAttribPointer.\n *

\n * For example:\n *

\n *\n * const index = glctx.getAttribLocation(shaderProgram, "pos");\n * glctx.vertexAttribPointer(\n *   layout.position.size,\n *   glctx[layout.position.type],\n *   layout.position.normalized,\n *   layout.position.stride,\n *   layout.position.offset);\n * 
\n * @see {@link Mesh}\n */\nclass Layout {\n /**\n * Create a Layout object. This constructor will throw if any duplicate\n * attributes are given.\n * @param {Array} ...attributes - An ordered list of attributes that\n * describe the desired memory layout for each vertex attribute.\n *

\n *\n * @see {@link Mesh}\n */\n constructor(...attributes) {\n this.attributes = attributes;\n this.attributeMap = {};\n let offset = 0;\n let maxStrideMultiple = 0;\n for (const attribute of attributes) {\n if (this.attributeMap[attribute.key]) {\n throw new DuplicateAttributeException(attribute);\n }\n // Add padding to satisfy WebGL\'s requirement that all\n // vertexAttribPointer calls have an offset that is a multiple of\n // the type size.\n if (offset % attribute.sizeOfType !== 0) {\n offset += attribute.sizeOfType - (offset % attribute.sizeOfType);\n console.warn("Layout requires padding before " + attribute.key + " attribute");\n }\n this.attributeMap[attribute.key] = {\n attribute: attribute,\n size: attribute.size,\n type: attribute.type,\n normalized: attribute.normalized,\n offset: offset,\n };\n offset += attribute.sizeInBytes;\n maxStrideMultiple = Math.max(maxStrideMultiple, attribute.sizeOfType);\n }\n // Add padding to the end to satisfy WebGL\'s requirement that all\n // vertexAttribPointer calls have a stride that is a multiple of the\n // type size. Because we\'re putting differently sized attributes into\n // the same buffer, it must be padded to a multiple of the largest\n // type size.\n if (offset % maxStrideMultiple !== 0) {\n offset += maxStrideMultiple - (offset % maxStrideMultiple);\n console.warn("Layout requires padding at the back");\n }\n this.stride = offset;\n for (const attribute of attributes) {\n this.attributeMap[attribute.key].stride = this.stride;\n }\n }\n}\n// Geometry attributes\n/**\n * Attribute layout to pack a vertex\'s x, y, & z as floats\n *\n * @see {@link Layout}\n */\nLayout.POSITION = new Attribute("position", 3, TYPES.FLOAT);\n/**\n * Attribute layout to pack a vertex\'s normal\'s x, y, & z as floats\n *\n * @see {@link Layout}\n */\nLayout.NORMAL = new Attribute("normal", 3, TYPES.FLOAT);\n/**\n * Attribute layout to pack a vertex\'s normal\'s x, y, & z as floats.\n *

\n * This value will be computed on-the-fly based on the texture coordinates.\n * If no texture coordinates are available, the generated value will default to\n * 0, 0, 0.\n *\n * @see {@link Layout}\n */\nLayout.TANGENT = new Attribute("tangent", 3, TYPES.FLOAT);\n/**\n * Attribute layout to pack a vertex\'s normal\'s bitangent x, y, & z as floats.\n *

\n * This value will be computed on-the-fly based on the texture coordinates.\n * If no texture coordinates are available, the generated value will default to\n * 0, 0, 0.\n * @see {@link Layout}\n */\nLayout.BITANGENT = new Attribute("bitangent", 3, TYPES.FLOAT);\n/**\n * Attribute layout to pack a vertex\'s texture coordinates\' u & v as floats\n *\n * @see {@link Layout}\n */\nLayout.UV = new Attribute("uv", 2, TYPES.FLOAT);\n// Material attributes\n/**\n * Attribute layout to pack an unsigned short to be interpreted as a the index\n * into a {@link Mesh}\'s materials list.\n *

\n * The intention of this value is to send all of the {@link Mesh}\'s materials\n * into multiple shader uniforms and then reference the current one by this\n * vertex attribute.\n *

\n * example glsl code:\n *\n *

\n *  // this is bound using MATERIAL_INDEX\n *  attribute int materialIndex;\n *\n *  struct Material {\n *    vec3 diffuse;\n *    vec3 specular;\n *    vec3 specularExponent;\n *  };\n *\n *  uniform Material materials[MAX_MATERIALS];\n *\n *  // ...\n *\n *  vec3 diffuse = materials[materialIndex];\n *\n * 
\n * TODO: More description & test to make sure subscripting by attributes even\n * works for webgl\n *\n * @see {@link Layout}\n */\nLayout.MATERIAL_INDEX = new Attribute("materialIndex", 1, TYPES.SHORT);\nLayout.MATERIAL_ENABLED = new Attribute("materialEnabled", 1, TYPES.UNSIGNED_SHORT);\nLayout.AMBIENT = new Attribute("ambient", 3, TYPES.FLOAT);\nLayout.DIFFUSE = new Attribute("diffuse", 3, TYPES.FLOAT);\nLayout.SPECULAR = new Attribute("specular", 3, TYPES.FLOAT);\nLayout.SPECULAR_EXPONENT = new Attribute("specularExponent", 3, TYPES.FLOAT);\nLayout.EMISSIVE = new Attribute("emissive", 3, TYPES.FLOAT);\nLayout.TRANSMISSION_FILTER = new Attribute("transmissionFilter", 3, TYPES.FLOAT);\nLayout.DISSOLVE = new Attribute("dissolve", 1, TYPES.FLOAT);\nLayout.ILLUMINATION = new Attribute("illumination", 1, TYPES.UNSIGNED_SHORT);\nLayout.REFRACTION_INDEX = new Attribute("refractionIndex", 1, TYPES.FLOAT);\nLayout.SHARPNESS = new Attribute("sharpness", 1, TYPES.FLOAT);\nLayout.MAP_DIFFUSE = new Attribute("mapDiffuse", 1, TYPES.SHORT);\nLayout.MAP_AMBIENT = new Attribute("mapAmbient", 1, TYPES.SHORT);\nLayout.MAP_SPECULAR = new Attribute("mapSpecular", 1, TYPES.SHORT);\nLayout.MAP_SPECULAR_EXPONENT = new Attribute("mapSpecularExponent", 1, TYPES.SHORT);\nLayout.MAP_DISSOLVE = new Attribute("mapDissolve", 1, TYPES.SHORT);\nLayout.ANTI_ALIASING = new Attribute("antiAliasing", 1, TYPES.UNSIGNED_SHORT);\nLayout.MAP_BUMP = new Attribute("mapBump", 1, TYPES.SHORT);\nLayout.MAP_DISPLACEMENT = new Attribute("mapDisplacement", 1, TYPES.SHORT);\nLayout.MAP_DECAL = new Attribute("mapDecal", 1, TYPES.SHORT);\nLayout.MAP_EMISSIVE = new Attribute("mapEmissive", 1, TYPES.SHORT);\n\n\n//# sourceURL=webpack://OBJ/./src/layout.ts?')},"./src/material.ts": /*!*************************!*\ !*** ./src/material.ts ***! \*************************/ @@ -14,12 +14,12 @@ /*!*********************!*\ !*** ./src/mesh.ts ***! \*********************/ -/*! exports provided: default */function(module,__webpack_exports__,__webpack_require__){"use strict";eval('__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "default", function() { return Mesh; });\n/* harmony import */ var _layout__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./layout */ "./src/layout.ts");\n\n/**\n * The main Mesh class. The constructor will parse through the OBJ file data\n * and collect the vertex, vertex normal, texture, and face information. This\n * information can then be used later on when creating your VBOs. See\n * OBJ.initMeshBuffers for an example of how to use the newly created Mesh\n */\nclass Mesh {\n /**\n * Create a Mesh\n * @param {String} objectData - a string representation of an OBJ file with\n * newlines preserved.\n * @param {Object} options - a JS object containing valid options. See class\n * documentation for options.\n * @param {bool} options.enableWTextureCoord - Texture coordinates can have\n * an optional "w" coordinate after the u and v coordinates. This extra\n * value can be used in order to perform fancy transformations on the\n * textures themselves. Default is to truncate to only the u an v\n * coordinates. Passing true will provide a default value of 0 in the\n * event that any or all texture coordinates don\'t provide a w value.\n * Always use the textureStride attribute in order to determine the\n * stride length of the texture coordinates when rendering the element\n * array.\n * @param {bool} options.calcTangentsAndBitangents - Calculate the tangents\n * and bitangents when loading of the OBJ is completed. This adds two new\n * attributes to the Mesh instance: `tangents` and `bitangents`.\n */\n constructor(objectData, options) {\n this.name = "";\n this.indicesPerMaterial = [];\n this.materialsByIndex = {};\n this.tangents = [];\n this.bitangents = [];\n options = options || {};\n options.materials = options.materials || {};\n options.enableWTextureCoord = !!options.enableWTextureCoord;\n // the list of unique vertex, normal, texture, attributes\n this.vertexNormals = [];\n this.textures = [];\n // the indicies to draw the faces\n this.indices = [];\n this.textureStride = options.enableWTextureCoord ? 3 : 2;\n /*\n The OBJ file format does a sort of compression when saving a model in a\n program like Blender. There are at least 3 sections (4 including textures)\n within the file. Each line in a section begins with the same string:\n * \'v\': indicates vertex section\n * \'vn\': indicates vertex normal section\n * \'f\': indicates the faces section\n * \'vt\': indicates vertex texture section (if textures were used on the model)\n Each of the above sections (except for the faces section) is a list/set of\n unique vertices.\n\n Each line of the faces section contains a list of\n (vertex, [texture], normal) groups.\n\n **Note:** The following documentation will use a capital "V" Vertex to\n denote the above (vertex, [texture], normal) groups whereas a lowercase\n "v" vertex is used to denote an X, Y, Z coordinate.\n\n Some examples:\n // the texture index is optional, both formats are possible for models\n // without a texture applied\n f 1/25 18/46 12/31\n f 1//25 18//46 12//31\n\n // A 3 vertex face with texture indices\n f 16/92/11 14/101/22 1/69/1\n\n // A 4 vertex face\n f 16/92/11 40/109/40 38/114/38 14/101/22\n\n The first two lines are examples of a 3 vertex face without a texture applied.\n The second is an example of a 3 vertex face with a texture applied.\n The third is an example of a 4 vertex face. Note: a face can contain N\n number of vertices.\n\n Each number that appears in one of the groups is a 1-based index\n corresponding to an item from the other sections (meaning that indexing\n starts at one and *not* zero).\n\n For example:\n `f 16/92/11` is saying to\n - take the 16th element from the [v] vertex array\n - take the 92nd element from the [vt] texture array\n - take the 11th element from the [vn] normal array\n and together they make a unique vertex.\n Using all 3+ unique Vertices from the face line will produce a polygon.\n\n Now, you could just go through the OBJ file and create a new vertex for\n each face line and WebGL will draw what appears to be the same model.\n However, vertices will be overlapped and duplicated all over the place.\n\n Consider a cube in 3D space centered about the origin and each side is\n 2 units long. The front face (with the positive Z-axis pointing towards\n you) would have a Top Right vertex (looking orthogonal to its normal)\n mapped at (1,1,1) The right face would have a Top Left vertex (looking\n orthogonal to its normal) at (1,1,1) and the top face would have a Bottom\n Right vertex (looking orthogonal to its normal) at (1,1,1). Each face\n has a vertex at the same coordinates, however, three distinct vertices\n will be drawn at the same spot.\n\n To solve the issue of duplicate Vertices (the `(vertex, [texture], normal)`\n groups), while iterating through the face lines, when a group is encountered\n the whole group string (\'16/92/11\') is checked to see if it exists in the\n packed.hashindices object, and if it doesn\'t, the indices it specifies\n are used to look up each attribute in the corresponding attribute arrays\n already created. The values are then copied to the corresponding unpacked\n array (flattened to play nice with WebGL\'s ELEMENT_ARRAY_BUFFER indexing),\n the group string is added to the hashindices set and the current unpacked\n index is used as this hashindices value so that the group of elements can\n be reused. The unpacked index is incremented. If the group string already\n exists in the hashindices object, its corresponding value is the index of\n that group and is appended to the unpacked indices array.\n */\n const verts = [];\n const vertNormals = [];\n const textures = [];\n const materialNamesByIndex = [];\n const materialIndicesByName = {};\n // keep track of what material we\'ve seen last\n let currentMaterialIndex = -1;\n let currentObjectByMaterialIndex = 0;\n // unpacking stuff\n const unpacked = {\n verts: [],\n norms: [],\n textures: [],\n hashindices: {},\n indices: [[]],\n materialIndices: [],\n index: 0,\n };\n const VERTEX_RE = /^v\\s/;\n const NORMAL_RE = /^vn\\s/;\n const TEXTURE_RE = /^vt\\s/;\n const FACE_RE = /^f\\s/;\n const WHITESPACE_RE = /\\s+/;\n const USE_MATERIAL_RE = /^usemtl/;\n // array of lines separated by the newline\n const lines = objectData.split("\\n");\n for (let line of lines) {\n line = line.trim();\n if (!line || line.startsWith("#")) {\n continue;\n }\n const elements = line.split(WHITESPACE_RE);\n elements.shift();\n if (VERTEX_RE.test(line)) {\n // if this is a vertex\n verts.push(...elements);\n }\n else if (NORMAL_RE.test(line)) {\n // if this is a vertex normal\n vertNormals.push(...elements);\n }\n else if (TEXTURE_RE.test(line)) {\n let coords = elements;\n // by default, the loader will only look at the U and V\n // coordinates of the vt declaration. So, this truncates the\n // elements to only those 2 values. If W texture coordinate\n // support is enabled, then the texture coordinate is\n // expected to have three values in it.\n if (elements.length > 2 && !options.enableWTextureCoord) {\n coords = elements.slice(0, 2);\n }\n else if (elements.length === 2 && options.enableWTextureCoord) {\n // If for some reason W texture coordinate support is enabled\n // and only the U and V coordinates are given, then we supply\n // the default value of 0 so that the stride length is correct\n // when the textures are unpacked below.\n coords.push("0");\n }\n textures.push(...coords);\n }\n else if (USE_MATERIAL_RE.test(line)) {\n const materialName = elements[0];\n // check to see if we\'ve ever seen it before\n if (!(materialName in materialIndicesByName)) {\n // new material we\'ve never seen\n materialNamesByIndex.push(materialName);\n materialIndicesByName[materialName] = materialNamesByIndex.length - 1;\n // push new array into indices\n // already contains an array at index zero, don\'t add\n if (materialIndicesByName[materialName] > 0) {\n unpacked.indices.push([]);\n }\n }\n // keep track of the current material index\n currentMaterialIndex = materialIndicesByName[materialName];\n // update current index array\n currentObjectByMaterialIndex = currentMaterialIndex;\n }\n else if (FACE_RE.test(line)) {\n // if this is a face\n /*\n split this face into an array of Vertex groups\n for example:\n f 16/92/11 14/101/22 1/69/1\n becomes:\n [\'16/92/11\', \'14/101/22\', \'1/69/1\'];\n */\n const triangles = triangulate(elements);\n for (const triangle of triangles) {\n for (let j = 0, eleLen = triangle.length; j < eleLen; j++) {\n const hash = triangle[j] + "," + currentMaterialIndex;\n if (hash in unpacked.hashindices) {\n unpacked.indices[currentObjectByMaterialIndex].push(unpacked.hashindices[hash]);\n }\n else {\n /*\n Each element of the face line array is a Vertex which has its\n attributes delimited by a forward slash. This will separate\n each attribute into another array:\n \'19/92/11\'\n becomes:\n Vertex = [\'19\', \'92\', \'11\'];\n where\n Vertex[0] is the vertex index\n Vertex[1] is the texture index\n Vertex[2] is the normal index\n Think of faces having Vertices which are comprised of the\n attributes location (v), texture (vt), and normal (vn).\n */\n const vertex = elements[j].split("/");\n // it\'s possible for faces to only specify the vertex\n // and the normal. In this case, vertex will only have\n // a length of 2 and not 3 and the normal will be the\n // second item in the list with an index of 1.\n const normalIndex = vertex.length - 1;\n /*\n The verts, textures, and vertNormals arrays each contain a\n flattend array of coordinates.\n\n Because it gets confusing by referring to Vertex and then\n vertex (both are different in my descriptions) I will explain\n what\'s going on using the vertexNormals array:\n\n vertex[2] will contain the one-based index of the vertexNormals\n section (vn). One is subtracted from this index number to play\n nice with javascript\'s zero-based array indexing.\n\n Because vertexNormal is a flattened array of x, y, z values,\n simple pointer arithmetic is used to skip to the start of the\n vertexNormal, then the offset is added to get the correct\n component: +0 is x, +1 is y, +2 is z.\n\n This same process is repeated for verts and textures.\n */\n // Vertex position\n unpacked.verts.push(+verts[(+vertex[0] - 1) * 3 + 0]);\n unpacked.verts.push(+verts[(+vertex[0] - 1) * 3 + 1]);\n unpacked.verts.push(+verts[(+vertex[0] - 1) * 3 + 2]);\n // Vertex textures\n if (textures.length) {\n const stride = options.enableWTextureCoord ? 3 : 2;\n unpacked.textures.push(+textures[(+vertex[1] - 1) * stride + 0]);\n unpacked.textures.push(+textures[(+vertex[1] - 1) * stride + 1]);\n if (options.enableWTextureCoord) {\n unpacked.textures.push(+textures[(+vertex[1] - 1) * stride + 2]);\n }\n }\n // Vertex normals\n unpacked.norms.push(+vertNormals[(+vertex[normalIndex] - 1) * 3 + 0]);\n unpacked.norms.push(+vertNormals[(+vertex[normalIndex] - 1) * 3 + 1]);\n unpacked.norms.push(+vertNormals[(+vertex[normalIndex] - 1) * 3 + 2]);\n // Vertex material indices\n unpacked.materialIndices.push(currentMaterialIndex);\n // add the newly created Vertex to the list of indices\n unpacked.hashindices[hash] = unpacked.index;\n unpacked.indices[currentObjectByMaterialIndex].push(unpacked.hashindices[hash]);\n // increment the counter\n unpacked.index += 1;\n }\n }\n }\n }\n }\n this.vertices = unpacked.verts;\n this.vertexNormals = unpacked.norms;\n this.textures = unpacked.textures;\n this.vertexMaterialIndices = unpacked.materialIndices;\n this.indices = unpacked.indices[currentObjectByMaterialIndex];\n this.indicesPerMaterial = unpacked.indices;\n this.materialNames = materialNamesByIndex;\n this.materialIndices = materialIndicesByName;\n this.materialsByIndex = {};\n if (options.calcTangentsAndBitangents) {\n this.calculateTangentsAndBitangents();\n }\n }\n /**\n * Calculates the tangents and bitangents of the mesh that forms an orthogonal basis together with the\n * normal in the direction of the texture coordinates. These are useful for setting up the TBN matrix\n * when distorting the normals through normal maps.\n * Method derived from: http://www.opengl-tutorial.org/intermediate-tutorials/tutorial-13-normal-mapping/\n *\n * This method requires the normals and texture coordinates to be parsed and set up correctly.\n * Adds the tangents and bitangents as members of the class instance.\n */\n calculateTangentsAndBitangents() {\n console.assert(!!(this.vertices &&\n this.vertices.length &&\n this.vertexNormals &&\n this.vertexNormals.length &&\n this.textures &&\n this.textures.length), "Missing attributes for calculating tangents and bitangents");\n const unpacked = {\n tangents: [...new Array(this.vertices.length)].map(_ => 0),\n bitangents: [...new Array(this.vertices.length)].map(_ => 0),\n };\n // Loop through all faces in the whole mesh\n const indices = this.indices;\n const vertices = this.vertices;\n const normals = this.vertexNormals;\n const uvs = this.textures;\n for (let i = 0; i < indices.length; i += 3) {\n const i0 = indices[i + 0];\n const i1 = indices[i + 1];\n const i2 = indices[i + 2];\n const x_v0 = vertices[i0 * 3 + 0];\n const y_v0 = vertices[i0 * 3 + 1];\n const z_v0 = vertices[i0 * 3 + 2];\n const x_uv0 = uvs[i0 * 2 + 0];\n const y_uv0 = uvs[i0 * 2 + 1];\n const x_v1 = vertices[i1 * 3 + 0];\n const y_v1 = vertices[i1 * 3 + 1];\n const z_v1 = vertices[i1 * 3 + 2];\n const x_uv1 = uvs[i1 * 2 + 0];\n const y_uv1 = uvs[i1 * 2 + 1];\n const x_v2 = vertices[i2 * 3 + 0];\n const y_v2 = vertices[i2 * 3 + 1];\n const z_v2 = vertices[i2 * 3 + 2];\n const x_uv2 = uvs[i2 * 2 + 0];\n const y_uv2 = uvs[i2 * 2 + 1];\n const x_deltaPos1 = x_v1 - x_v0;\n const y_deltaPos1 = y_v1 - y_v0;\n const z_deltaPos1 = z_v1 - z_v0;\n const x_deltaPos2 = x_v2 - x_v0;\n const y_deltaPos2 = y_v2 - y_v0;\n const z_deltaPos2 = z_v2 - z_v0;\n const x_uvDeltaPos1 = x_uv1 - x_uv0;\n const y_uvDeltaPos1 = y_uv1 - y_uv0;\n const x_uvDeltaPos2 = x_uv2 - x_uv0;\n const y_uvDeltaPos2 = y_uv2 - y_uv0;\n const rInv = x_uvDeltaPos1 * y_uvDeltaPos2 - y_uvDeltaPos1 * x_uvDeltaPos2;\n const r = 1.0 / Math.abs(rInv < 0.0001 ? 1.0 : rInv);\n // Tangent\n const x_tangent = (x_deltaPos1 * y_uvDeltaPos2 - x_deltaPos2 * y_uvDeltaPos1) * r;\n const y_tangent = (y_deltaPos1 * y_uvDeltaPos2 - y_deltaPos2 * y_uvDeltaPos1) * r;\n const z_tangent = (z_deltaPos1 * y_uvDeltaPos2 - z_deltaPos2 * y_uvDeltaPos1) * r;\n // Bitangent\n const x_bitangent = (x_deltaPos2 * x_uvDeltaPos1 - x_deltaPos1 * x_uvDeltaPos2) * r;\n const y_bitangent = (y_deltaPos2 * x_uvDeltaPos1 - y_deltaPos1 * x_uvDeltaPos2) * r;\n const z_bitangent = (z_deltaPos2 * x_uvDeltaPos1 - z_deltaPos1 * x_uvDeltaPos2) * r;\n // Gram-Schmidt orthogonalize\n //t = glm::normalize(t - n * glm:: dot(n, t));\n const x_n0 = normals[i0 * 3 + 0];\n const y_n0 = normals[i0 * 3 + 1];\n const z_n0 = normals[i0 * 3 + 2];\n const x_n1 = normals[i1 * 3 + 0];\n const y_n1 = normals[i1 * 3 + 1];\n const z_n1 = normals[i1 * 3 + 2];\n const x_n2 = normals[i2 * 3 + 0];\n const y_n2 = normals[i2 * 3 + 1];\n const z_n2 = normals[i2 * 3 + 2];\n // Tangent\n const n0_dot_t = x_tangent * x_n0 + y_tangent * y_n0 + z_tangent * z_n0;\n const n1_dot_t = x_tangent * x_n1 + y_tangent * y_n1 + z_tangent * z_n1;\n const n2_dot_t = x_tangent * x_n2 + y_tangent * y_n2 + z_tangent * z_n2;\n const x_resTangent0 = x_tangent - x_n0 * n0_dot_t;\n const y_resTangent0 = y_tangent - y_n0 * n0_dot_t;\n const z_resTangent0 = z_tangent - z_n0 * n0_dot_t;\n const x_resTangent1 = x_tangent - x_n1 * n1_dot_t;\n const y_resTangent1 = y_tangent - y_n1 * n1_dot_t;\n const z_resTangent1 = z_tangent - z_n1 * n1_dot_t;\n const x_resTangent2 = x_tangent - x_n2 * n2_dot_t;\n const y_resTangent2 = y_tangent - y_n2 * n2_dot_t;\n const z_resTangent2 = z_tangent - z_n2 * n2_dot_t;\n const magTangent0 = Math.sqrt(x_resTangent0 * x_resTangent0 + y_resTangent0 * y_resTangent0 + z_resTangent0 * z_resTangent0);\n const magTangent1 = Math.sqrt(x_resTangent1 * x_resTangent1 + y_resTangent1 * y_resTangent1 + z_resTangent1 * z_resTangent1);\n const magTangent2 = Math.sqrt(x_resTangent2 * x_resTangent2 + y_resTangent2 * y_resTangent2 + z_resTangent2 * z_resTangent2);\n // Bitangent\n const n0_dot_bt = x_bitangent * x_n0 + y_bitangent * y_n0 + z_bitangent * z_n0;\n const n1_dot_bt = x_bitangent * x_n1 + y_bitangent * y_n1 + z_bitangent * z_n1;\n const n2_dot_bt = x_bitangent * x_n2 + y_bitangent * y_n2 + z_bitangent * z_n2;\n const x_resBitangent0 = x_bitangent - x_n0 * n0_dot_bt;\n const y_resBitangent0 = y_bitangent - y_n0 * n0_dot_bt;\n const z_resBitangent0 = z_bitangent - z_n0 * n0_dot_bt;\n const x_resBitangent1 = x_bitangent - x_n1 * n1_dot_bt;\n const y_resBitangent1 = y_bitangent - y_n1 * n1_dot_bt;\n const z_resBitangent1 = z_bitangent - z_n1 * n1_dot_bt;\n const x_resBitangent2 = x_bitangent - x_n2 * n2_dot_bt;\n const y_resBitangent2 = y_bitangent - y_n2 * n2_dot_bt;\n const z_resBitangent2 = z_bitangent - z_n2 * n2_dot_bt;\n const magBitangent0 = Math.sqrt(x_resBitangent0 * x_resBitangent0 +\n y_resBitangent0 * y_resBitangent0 +\n z_resBitangent0 * z_resBitangent0);\n const magBitangent1 = Math.sqrt(x_resBitangent1 * x_resBitangent1 +\n y_resBitangent1 * y_resBitangent1 +\n z_resBitangent1 * z_resBitangent1);\n const magBitangent2 = Math.sqrt(x_resBitangent2 * x_resBitangent2 +\n y_resBitangent2 * y_resBitangent2 +\n z_resBitangent2 * z_resBitangent2);\n unpacked.tangents[i0 * 3 + 0] += x_resTangent0 / magTangent0;\n unpacked.tangents[i0 * 3 + 1] += y_resTangent0 / magTangent0;\n unpacked.tangents[i0 * 3 + 2] += z_resTangent0 / magTangent0;\n unpacked.tangents[i1 * 3 + 0] += x_resTangent1 / magTangent1;\n unpacked.tangents[i1 * 3 + 1] += y_resTangent1 / magTangent1;\n unpacked.tangents[i1 * 3 + 2] += z_resTangent1 / magTangent1;\n unpacked.tangents[i2 * 3 + 0] += x_resTangent2 / magTangent2;\n unpacked.tangents[i2 * 3 + 1] += y_resTangent2 / magTangent2;\n unpacked.tangents[i2 * 3 + 2] += z_resTangent2 / magTangent2;\n unpacked.bitangents[i0 * 3 + 0] += x_resBitangent0 / magBitangent0;\n unpacked.bitangents[i0 * 3 + 1] += y_resBitangent0 / magBitangent0;\n unpacked.bitangents[i0 * 3 + 2] += z_resBitangent0 / magBitangent0;\n unpacked.bitangents[i1 * 3 + 0] += x_resBitangent1 / magBitangent1;\n unpacked.bitangents[i1 * 3 + 1] += y_resBitangent1 / magBitangent1;\n unpacked.bitangents[i1 * 3 + 2] += z_resBitangent1 / magBitangent1;\n unpacked.bitangents[i2 * 3 + 0] += x_resBitangent2 / magBitangent2;\n unpacked.bitangents[i2 * 3 + 1] += y_resBitangent2 / magBitangent2;\n unpacked.bitangents[i2 * 3 + 2] += z_resBitangent2 / magBitangent2;\n // TODO: check handedness\n }\n this.tangents = unpacked.tangents;\n this.bitangents = unpacked.bitangents;\n }\n /**\n * @param layout - A {@link Layout} object that describes the\n * desired memory layout of the generated buffer\n * @return The packed array in the ... TODO\n */\n makeBufferData(layout) {\n const numItems = this.vertices.length / 3;\n const buffer = new ArrayBuffer(layout.stride * numItems);\n buffer.numItems = numItems;\n const dataView = new DataView(buffer);\n for (let i = 0, vertexOffset = 0; i < numItems; i++) {\n vertexOffset = i * layout.stride;\n // copy in the vertex data in the order and format given by the\n // layout param\n for (const attribute of layout.attributes) {\n const offset = vertexOffset + layout.attributeMap[attribute.key].offset;\n switch (attribute.key) {\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].POSITION.key:\n dataView.setFloat32(offset, this.vertices[i * 3], true);\n dataView.setFloat32(offset + 4, this.vertices[i * 3 + 1], true);\n dataView.setFloat32(offset + 8, this.vertices[i * 3 + 2], true);\n break;\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].UV.key:\n dataView.setFloat32(offset, this.textures[i * 2], true);\n dataView.setFloat32(offset + 4, this.textures[i * 2 + 1], true);\n break;\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].NORMAL.key:\n dataView.setFloat32(offset, this.vertexNormals[i * 3], true);\n dataView.setFloat32(offset + 4, this.vertexNormals[i * 3 + 1], true);\n dataView.setFloat32(offset + 8, this.vertexNormals[i * 3 + 2], true);\n break;\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].MATERIAL_INDEX.key:\n dataView.setInt16(offset, this.vertexMaterialIndices[i], true);\n break;\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].AMBIENT.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setFloat32(offset, material.ambient[0], true);\n dataView.setFloat32(offset + 4, material.ambient[1], true);\n dataView.setFloat32(offset + 8, material.ambient[2], true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].DIFFUSE.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setFloat32(offset, material.diffuse[0], true);\n dataView.setFloat32(offset + 4, material.diffuse[1], true);\n dataView.setFloat32(offset + 8, material.diffuse[2], true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].SPECULAR.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setFloat32(offset, material.specular[0], true);\n dataView.setFloat32(offset + 4, material.specular[1], true);\n dataView.setFloat32(offset + 8, material.specular[2], true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].SPECULAR_EXPONENT.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setFloat32(offset, material.specularExponent, true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].EMISSIVE.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setFloat32(offset, material.emissive[0], true);\n dataView.setFloat32(offset + 4, material.emissive[1], true);\n dataView.setFloat32(offset + 8, material.emissive[2], true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].TRANSMISSION_FILTER.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setFloat32(offset, material.transmissionFilter[0], true);\n dataView.setFloat32(offset + 4, material.transmissionFilter[1], true);\n dataView.setFloat32(offset + 8, material.transmissionFilter[2], true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].DISSOLVE.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setFloat32(offset, material.dissolve, true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].ILLUMINATION.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setInt16(offset, material.illumination, true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].REFRACTION_INDEX.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setFloat32(offset, material.refractionIndex, true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].SHARPNESS.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setFloat32(offset, material.sharpness, true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].ANTI_ALIASING.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setInt16(offset, material.antiAliasing ? 1 : 0, true);\n break;\n }\n }\n }\n }\n return buffer;\n }\n makeIndexBufferData() {\n const buffer = new Uint16Array(this.indices);\n buffer.numItems = this.indices.length;\n return buffer;\n }\n addMaterialLibrary(mtl) {\n for (const name in mtl.materials) {\n if (!(name in this.materialIndices)) {\n // This material is not referenced by the mesh\n continue;\n }\n const material = mtl.materials[name];\n // Find the material index for this material\n const materialIndex = this.materialIndices[material.name];\n // Put the material into the materialsByIndex object at the right\n // spot as determined when the obj file was parsed\n this.materialsByIndex[materialIndex] = material;\n }\n }\n}\nfunction* triangulate(elements) {\n if (elements.length <= 3) {\n yield elements;\n }\n else if (elements.length === 4) {\n yield [elements[0], elements[1], elements[2]];\n yield [elements[2], elements[3], elements[0]];\n }\n else {\n for (let i = 1; i < elements.length - 1; i++) {\n yield [elements[0], elements[i], elements[i + 1]];\n }\n }\n}\n\n\n//# sourceURL=webpack://OBJ/./src/mesh.ts?')},"./src/utils.ts": +/*! exports provided: default */function(module,__webpack_exports__,__webpack_require__){"use strict";eval('__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "default", function() { return Mesh; });\n/* harmony import */ var _layout__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./layout */ "./src/layout.ts");\n\n/**\n * The main Mesh class. The constructor will parse through the OBJ file data\n * and collect the vertex, vertex normal, texture, and face information. This\n * information can then be used later on when creating your VBOs. See\n * OBJ.initMeshBuffers for an example of how to use the newly created Mesh\n */\nclass Mesh {\n /**\n * Create a Mesh\n * @param {String} objectData - a string representation of an OBJ file with\n * newlines preserved.\n * @param {Object} options - a JS object containing valid options. See class\n * documentation for options.\n * @param {bool} options.enableWTextureCoord - Texture coordinates can have\n * an optional "w" coordinate after the u and v coordinates. This extra\n * value can be used in order to perform fancy transformations on the\n * textures themselves. Default is to truncate to only the u an v\n * coordinates. Passing true will provide a default value of 0 in the\n * event that any or all texture coordinates don\'t provide a w value.\n * Always use the textureStride attribute in order to determine the\n * stride length of the texture coordinates when rendering the element\n * array.\n * @param {bool} options.calcTangentsAndBitangents - Calculate the tangents\n * and bitangents when loading of the OBJ is completed. This adds two new\n * attributes to the Mesh instance: `tangents` and `bitangents`.\n */\n constructor(objectData, options) {\n this.name = "";\n this.indicesPerMaterial = [];\n this.materialsByIndex = {};\n this.tangents = [];\n this.bitangents = [];\n options = options || {};\n options.materials = options.materials || {};\n options.enableWTextureCoord = !!options.enableWTextureCoord;\n // the list of unique vertex, normal, texture, attributes\n this.vertexNormals = [];\n this.textures = [];\n // the indicies to draw the faces\n this.indices = [];\n this.textureStride = options.enableWTextureCoord ? 3 : 2;\n /*\n The OBJ file format does a sort of compression when saving a model in a\n program like Blender. There are at least 3 sections (4 including textures)\n within the file. Each line in a section begins with the same string:\n * \'v\': indicates vertex section\n * \'vn\': indicates vertex normal section\n * \'f\': indicates the faces section\n * \'vt\': indicates vertex texture section (if textures were used on the model)\n Each of the above sections (except for the faces section) is a list/set of\n unique vertices.\n\n Each line of the faces section contains a list of\n (vertex, [texture], normal) groups.\n\n **Note:** The following documentation will use a capital "V" Vertex to\n denote the above (vertex, [texture], normal) groups whereas a lowercase\n "v" vertex is used to denote an X, Y, Z coordinate.\n\n Some examples:\n // the texture index is optional, both formats are possible for models\n // without a texture applied\n f 1/25 18/46 12/31\n f 1//25 18//46 12//31\n\n // A 3 vertex face with texture indices\n f 16/92/11 14/101/22 1/69/1\n\n // A 4 vertex face\n f 16/92/11 40/109/40 38/114/38 14/101/22\n\n The first two lines are examples of a 3 vertex face without a texture applied.\n The second is an example of a 3 vertex face with a texture applied.\n The third is an example of a 4 vertex face. Note: a face can contain N\n number of vertices.\n\n Each number that appears in one of the groups is a 1-based index\n corresponding to an item from the other sections (meaning that indexing\n starts at one and *not* zero).\n\n For example:\n `f 16/92/11` is saying to\n - take the 16th element from the [v] vertex array\n - take the 92nd element from the [vt] texture array\n - take the 11th element from the [vn] normal array\n and together they make a unique vertex.\n Using all 3+ unique Vertices from the face line will produce a polygon.\n\n Now, you could just go through the OBJ file and create a new vertex for\n each face line and WebGL will draw what appears to be the same model.\n However, vertices will be overlapped and duplicated all over the place.\n\n Consider a cube in 3D space centered about the origin and each side is\n 2 units long. The front face (with the positive Z-axis pointing towards\n you) would have a Top Right vertex (looking orthogonal to its normal)\n mapped at (1,1,1) The right face would have a Top Left vertex (looking\n orthogonal to its normal) at (1,1,1) and the top face would have a Bottom\n Right vertex (looking orthogonal to its normal) at (1,1,1). Each face\n has a vertex at the same coordinates, however, three distinct vertices\n will be drawn at the same spot.\n\n To solve the issue of duplicate Vertices (the `(vertex, [texture], normal)`\n groups), while iterating through the face lines, when a group is encountered\n the whole group string (\'16/92/11\') is checked to see if it exists in the\n packed.hashindices object, and if it doesn\'t, the indices it specifies\n are used to look up each attribute in the corresponding attribute arrays\n already created. The values are then copied to the corresponding unpacked\n array (flattened to play nice with WebGL\'s ELEMENT_ARRAY_BUFFER indexing),\n the group string is added to the hashindices set and the current unpacked\n index is used as this hashindices value so that the group of elements can\n be reused. The unpacked index is incremented. If the group string already\n exists in the hashindices object, its corresponding value is the index of\n that group and is appended to the unpacked indices array.\n */\n const verts = [];\n const vertNormals = [];\n const textures = [];\n const materialNamesByIndex = [];\n const materialIndicesByName = {};\n // keep track of what material we\'ve seen last\n let currentMaterialIndex = -1;\n let currentObjectByMaterialIndex = 0;\n // unpacking stuff\n const unpacked = {\n verts: [],\n norms: [],\n textures: [],\n hashindices: {},\n indices: [[]],\n materialIndices: [],\n index: 0,\n };\n const VERTEX_RE = /^v\\s/;\n const NORMAL_RE = /^vn\\s/;\n const TEXTURE_RE = /^vt\\s/;\n const FACE_RE = /^f\\s/;\n const WHITESPACE_RE = /\\s+/;\n const USE_MATERIAL_RE = /^usemtl/;\n // array of lines separated by the newline\n const lines = objectData.split("\\n");\n for (let line of lines) {\n line = line.trim();\n if (!line || line.startsWith("#")) {\n continue;\n }\n const elements = line.split(WHITESPACE_RE);\n elements.shift();\n if (VERTEX_RE.test(line)) {\n // if this is a vertex\n verts.push(...elements);\n }\n else if (NORMAL_RE.test(line)) {\n // if this is a vertex normal\n vertNormals.push(...elements);\n }\n else if (TEXTURE_RE.test(line)) {\n let coords = elements;\n // by default, the loader will only look at the U and V\n // coordinates of the vt declaration. So, this truncates the\n // elements to only those 2 values. If W texture coordinate\n // support is enabled, then the texture coordinate is\n // expected to have three values in it.\n if (elements.length > 2 && !options.enableWTextureCoord) {\n coords = elements.slice(0, 2);\n }\n else if (elements.length === 2 && options.enableWTextureCoord) {\n // If for some reason W texture coordinate support is enabled\n // and only the U and V coordinates are given, then we supply\n // the default value of 0 so that the stride length is correct\n // when the textures are unpacked below.\n coords.push("0");\n }\n textures.push(...coords);\n }\n else if (USE_MATERIAL_RE.test(line)) {\n const materialName = elements[0];\n // check to see if we\'ve ever seen it before\n if (!(materialName in materialIndicesByName)) {\n // new material we\'ve never seen\n materialNamesByIndex.push(materialName);\n materialIndicesByName[materialName] = materialNamesByIndex.length - 1;\n // push new array into indices\n // already contains an array at index zero, don\'t add\n if (materialIndicesByName[materialName] > 0) {\n unpacked.indices.push([]);\n }\n }\n // keep track of the current material index\n currentMaterialIndex = materialIndicesByName[materialName];\n // update current index array\n currentObjectByMaterialIndex = currentMaterialIndex;\n }\n else if (FACE_RE.test(line)) {\n // if this is a face\n /*\n split this face into an array of Vertex groups\n for example:\n f 16/92/11 14/101/22 1/69/1\n becomes:\n [\'16/92/11\', \'14/101/22\', \'1/69/1\'];\n */\n const triangles = triangulate(elements);\n for (const triangle of triangles) {\n for (let j = 0, eleLen = triangle.length; j < eleLen; j++) {\n const hash = triangle[j] + "," + currentMaterialIndex;\n if (hash in unpacked.hashindices) {\n unpacked.indices[currentObjectByMaterialIndex].push(unpacked.hashindices[hash]);\n }\n else {\n /*\n Each element of the face line array is a Vertex which has its\n attributes delimited by a forward slash. This will separate\n each attribute into another array:\n \'19/92/11\'\n becomes:\n Vertex = [\'19\', \'92\', \'11\'];\n where\n Vertex[0] is the vertex index\n Vertex[1] is the texture index\n Vertex[2] is the normal index\n Think of faces having Vertices which are comprised of the\n attributes location (v), texture (vt), and normal (vn).\n */\n const vertex = triangle[j].split("/");\n // it\'s possible for faces to only specify the vertex\n // and the normal. In this case, vertex will only have\n // a length of 2 and not 3 and the normal will be the\n // second item in the list with an index of 1.\n const normalIndex = vertex.length - 1;\n /*\n The verts, textures, and vertNormals arrays each contain a\n flattend array of coordinates.\n\n Because it gets confusing by referring to Vertex and then\n vertex (both are different in my descriptions) I will explain\n what\'s going on using the vertexNormals array:\n\n vertex[2] will contain the one-based index of the vertexNormals\n section (vn). One is subtracted from this index number to play\n nice with javascript\'s zero-based array indexing.\n\n Because vertexNormal is a flattened array of x, y, z values,\n simple pointer arithmetic is used to skip to the start of the\n vertexNormal, then the offset is added to get the correct\n component: +0 is x, +1 is y, +2 is z.\n\n This same process is repeated for verts and textures.\n */\n // Vertex position\n unpacked.verts.push(+verts[(+vertex[0] - 1) * 3 + 0]);\n unpacked.verts.push(+verts[(+vertex[0] - 1) * 3 + 1]);\n unpacked.verts.push(+verts[(+vertex[0] - 1) * 3 + 2]);\n // Vertex textures\n if (textures.length) {\n const stride = options.enableWTextureCoord ? 3 : 2;\n unpacked.textures.push(+textures[(+vertex[1] - 1) * stride + 0]);\n unpacked.textures.push(+textures[(+vertex[1] - 1) * stride + 1]);\n if (options.enableWTextureCoord) {\n unpacked.textures.push(+textures[(+vertex[1] - 1) * stride + 2]);\n }\n }\n // Vertex normals\n unpacked.norms.push(+vertNormals[(+vertex[normalIndex] - 1) * 3 + 0]);\n unpacked.norms.push(+vertNormals[(+vertex[normalIndex] - 1) * 3 + 1]);\n unpacked.norms.push(+vertNormals[(+vertex[normalIndex] - 1) * 3 + 2]);\n // Vertex material indices\n unpacked.materialIndices.push(currentMaterialIndex);\n // add the newly created Vertex to the list of indices\n unpacked.hashindices[hash] = unpacked.index;\n unpacked.indices[currentObjectByMaterialIndex].push(unpacked.hashindices[hash]);\n // increment the counter\n unpacked.index += 1;\n }\n }\n }\n }\n }\n this.vertices = unpacked.verts;\n this.vertexNormals = unpacked.norms;\n this.textures = unpacked.textures;\n this.vertexMaterialIndices = unpacked.materialIndices;\n this.indices = unpacked.indices[currentObjectByMaterialIndex];\n this.indicesPerMaterial = unpacked.indices;\n this.materialNames = materialNamesByIndex;\n this.materialIndices = materialIndicesByName;\n this.materialsByIndex = {};\n if (options.calcTangentsAndBitangents) {\n this.calculateTangentsAndBitangents();\n }\n }\n /**\n * Calculates the tangents and bitangents of the mesh that forms an orthogonal basis together with the\n * normal in the direction of the texture coordinates. These are useful for setting up the TBN matrix\n * when distorting the normals through normal maps.\n * Method derived from: http://www.opengl-tutorial.org/intermediate-tutorials/tutorial-13-normal-mapping/\n *\n * This method requires the normals and texture coordinates to be parsed and set up correctly.\n * Adds the tangents and bitangents as members of the class instance.\n */\n calculateTangentsAndBitangents() {\n console.assert(!!(this.vertices &&\n this.vertices.length &&\n this.vertexNormals &&\n this.vertexNormals.length &&\n this.textures &&\n this.textures.length), "Missing attributes for calculating tangents and bitangents");\n const unpacked = {\n tangents: [...new Array(this.vertices.length)].map(_ => 0),\n bitangents: [...new Array(this.vertices.length)].map(_ => 0),\n };\n // Loop through all faces in the whole mesh\n const indices = this.indices;\n const vertices = this.vertices;\n const normals = this.vertexNormals;\n const uvs = this.textures;\n for (let i = 0; i < indices.length; i += 3) {\n const i0 = indices[i + 0];\n const i1 = indices[i + 1];\n const i2 = indices[i + 2];\n const x_v0 = vertices[i0 * 3 + 0];\n const y_v0 = vertices[i0 * 3 + 1];\n const z_v0 = vertices[i0 * 3 + 2];\n const x_uv0 = uvs[i0 * 2 + 0];\n const y_uv0 = uvs[i0 * 2 + 1];\n const x_v1 = vertices[i1 * 3 + 0];\n const y_v1 = vertices[i1 * 3 + 1];\n const z_v1 = vertices[i1 * 3 + 2];\n const x_uv1 = uvs[i1 * 2 + 0];\n const y_uv1 = uvs[i1 * 2 + 1];\n const x_v2 = vertices[i2 * 3 + 0];\n const y_v2 = vertices[i2 * 3 + 1];\n const z_v2 = vertices[i2 * 3 + 2];\n const x_uv2 = uvs[i2 * 2 + 0];\n const y_uv2 = uvs[i2 * 2 + 1];\n const x_deltaPos1 = x_v1 - x_v0;\n const y_deltaPos1 = y_v1 - y_v0;\n const z_deltaPos1 = z_v1 - z_v0;\n const x_deltaPos2 = x_v2 - x_v0;\n const y_deltaPos2 = y_v2 - y_v0;\n const z_deltaPos2 = z_v2 - z_v0;\n const x_uvDeltaPos1 = x_uv1 - x_uv0;\n const y_uvDeltaPos1 = y_uv1 - y_uv0;\n const x_uvDeltaPos2 = x_uv2 - x_uv0;\n const y_uvDeltaPos2 = y_uv2 - y_uv0;\n const rInv = x_uvDeltaPos1 * y_uvDeltaPos2 - y_uvDeltaPos1 * x_uvDeltaPos2;\n const r = 1.0 / Math.abs(rInv < 0.0001 ? 1.0 : rInv);\n // Tangent\n const x_tangent = (x_deltaPos1 * y_uvDeltaPos2 - x_deltaPos2 * y_uvDeltaPos1) * r;\n const y_tangent = (y_deltaPos1 * y_uvDeltaPos2 - y_deltaPos2 * y_uvDeltaPos1) * r;\n const z_tangent = (z_deltaPos1 * y_uvDeltaPos2 - z_deltaPos2 * y_uvDeltaPos1) * r;\n // Bitangent\n const x_bitangent = (x_deltaPos2 * x_uvDeltaPos1 - x_deltaPos1 * x_uvDeltaPos2) * r;\n const y_bitangent = (y_deltaPos2 * x_uvDeltaPos1 - y_deltaPos1 * x_uvDeltaPos2) * r;\n const z_bitangent = (z_deltaPos2 * x_uvDeltaPos1 - z_deltaPos1 * x_uvDeltaPos2) * r;\n // Gram-Schmidt orthogonalize\n //t = glm::normalize(t - n * glm:: dot(n, t));\n const x_n0 = normals[i0 * 3 + 0];\n const y_n0 = normals[i0 * 3 + 1];\n const z_n0 = normals[i0 * 3 + 2];\n const x_n1 = normals[i1 * 3 + 0];\n const y_n1 = normals[i1 * 3 + 1];\n const z_n1 = normals[i1 * 3 + 2];\n const x_n2 = normals[i2 * 3 + 0];\n const y_n2 = normals[i2 * 3 + 1];\n const z_n2 = normals[i2 * 3 + 2];\n // Tangent\n const n0_dot_t = x_tangent * x_n0 + y_tangent * y_n0 + z_tangent * z_n0;\n const n1_dot_t = x_tangent * x_n1 + y_tangent * y_n1 + z_tangent * z_n1;\n const n2_dot_t = x_tangent * x_n2 + y_tangent * y_n2 + z_tangent * z_n2;\n const x_resTangent0 = x_tangent - x_n0 * n0_dot_t;\n const y_resTangent0 = y_tangent - y_n0 * n0_dot_t;\n const z_resTangent0 = z_tangent - z_n0 * n0_dot_t;\n const x_resTangent1 = x_tangent - x_n1 * n1_dot_t;\n const y_resTangent1 = y_tangent - y_n1 * n1_dot_t;\n const z_resTangent1 = z_tangent - z_n1 * n1_dot_t;\n const x_resTangent2 = x_tangent - x_n2 * n2_dot_t;\n const y_resTangent2 = y_tangent - y_n2 * n2_dot_t;\n const z_resTangent2 = z_tangent - z_n2 * n2_dot_t;\n const magTangent0 = Math.sqrt(x_resTangent0 * x_resTangent0 + y_resTangent0 * y_resTangent0 + z_resTangent0 * z_resTangent0);\n const magTangent1 = Math.sqrt(x_resTangent1 * x_resTangent1 + y_resTangent1 * y_resTangent1 + z_resTangent1 * z_resTangent1);\n const magTangent2 = Math.sqrt(x_resTangent2 * x_resTangent2 + y_resTangent2 * y_resTangent2 + z_resTangent2 * z_resTangent2);\n // Bitangent\n const n0_dot_bt = x_bitangent * x_n0 + y_bitangent * y_n0 + z_bitangent * z_n0;\n const n1_dot_bt = x_bitangent * x_n1 + y_bitangent * y_n1 + z_bitangent * z_n1;\n const n2_dot_bt = x_bitangent * x_n2 + y_bitangent * y_n2 + z_bitangent * z_n2;\n const x_resBitangent0 = x_bitangent - x_n0 * n0_dot_bt;\n const y_resBitangent0 = y_bitangent - y_n0 * n0_dot_bt;\n const z_resBitangent0 = z_bitangent - z_n0 * n0_dot_bt;\n const x_resBitangent1 = x_bitangent - x_n1 * n1_dot_bt;\n const y_resBitangent1 = y_bitangent - y_n1 * n1_dot_bt;\n const z_resBitangent1 = z_bitangent - z_n1 * n1_dot_bt;\n const x_resBitangent2 = x_bitangent - x_n2 * n2_dot_bt;\n const y_resBitangent2 = y_bitangent - y_n2 * n2_dot_bt;\n const z_resBitangent2 = z_bitangent - z_n2 * n2_dot_bt;\n const magBitangent0 = Math.sqrt(x_resBitangent0 * x_resBitangent0 +\n y_resBitangent0 * y_resBitangent0 +\n z_resBitangent0 * z_resBitangent0);\n const magBitangent1 = Math.sqrt(x_resBitangent1 * x_resBitangent1 +\n y_resBitangent1 * y_resBitangent1 +\n z_resBitangent1 * z_resBitangent1);\n const magBitangent2 = Math.sqrt(x_resBitangent2 * x_resBitangent2 +\n y_resBitangent2 * y_resBitangent2 +\n z_resBitangent2 * z_resBitangent2);\n unpacked.tangents[i0 * 3 + 0] += x_resTangent0 / magTangent0;\n unpacked.tangents[i0 * 3 + 1] += y_resTangent0 / magTangent0;\n unpacked.tangents[i0 * 3 + 2] += z_resTangent0 / magTangent0;\n unpacked.tangents[i1 * 3 + 0] += x_resTangent1 / magTangent1;\n unpacked.tangents[i1 * 3 + 1] += y_resTangent1 / magTangent1;\n unpacked.tangents[i1 * 3 + 2] += z_resTangent1 / magTangent1;\n unpacked.tangents[i2 * 3 + 0] += x_resTangent2 / magTangent2;\n unpacked.tangents[i2 * 3 + 1] += y_resTangent2 / magTangent2;\n unpacked.tangents[i2 * 3 + 2] += z_resTangent2 / magTangent2;\n unpacked.bitangents[i0 * 3 + 0] += x_resBitangent0 / magBitangent0;\n unpacked.bitangents[i0 * 3 + 1] += y_resBitangent0 / magBitangent0;\n unpacked.bitangents[i0 * 3 + 2] += z_resBitangent0 / magBitangent0;\n unpacked.bitangents[i1 * 3 + 0] += x_resBitangent1 / magBitangent1;\n unpacked.bitangents[i1 * 3 + 1] += y_resBitangent1 / magBitangent1;\n unpacked.bitangents[i1 * 3 + 2] += z_resBitangent1 / magBitangent1;\n unpacked.bitangents[i2 * 3 + 0] += x_resBitangent2 / magBitangent2;\n unpacked.bitangents[i2 * 3 + 1] += y_resBitangent2 / magBitangent2;\n unpacked.bitangents[i2 * 3 + 2] += z_resBitangent2 / magBitangent2;\n // TODO: check handedness\n }\n this.tangents = unpacked.tangents;\n this.bitangents = unpacked.bitangents;\n }\n /**\n * @param layout - A {@link Layout} object that describes the\n * desired memory layout of the generated buffer\n * @return The packed array in the ... TODO\n */\n makeBufferData(layout) {\n const numItems = this.vertices.length / 3;\n const buffer = new ArrayBuffer(layout.stride * numItems);\n buffer.numItems = numItems;\n const dataView = new DataView(buffer);\n for (let i = 0, vertexOffset = 0; i < numItems; i++) {\n vertexOffset = i * layout.stride;\n // copy in the vertex data in the order and format given by the\n // layout param\n for (const attribute of layout.attributes) {\n const offset = vertexOffset + layout.attributeMap[attribute.key].offset;\n switch (attribute.key) {\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].POSITION.key:\n dataView.setFloat32(offset, this.vertices[i * 3], true);\n dataView.setFloat32(offset + 4, this.vertices[i * 3 + 1], true);\n dataView.setFloat32(offset + 8, this.vertices[i * 3 + 2], true);\n break;\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].UV.key:\n dataView.setFloat32(offset, this.textures[i * 2], true);\n dataView.setFloat32(offset + 4, this.textures[i * 2 + 1], true);\n break;\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].NORMAL.key:\n dataView.setFloat32(offset, this.vertexNormals[i * 3], true);\n dataView.setFloat32(offset + 4, this.vertexNormals[i * 3 + 1], true);\n dataView.setFloat32(offset + 8, this.vertexNormals[i * 3 + 2], true);\n break;\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].MATERIAL_INDEX.key:\n dataView.setInt16(offset, this.vertexMaterialIndices[i], true);\n break;\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].AMBIENT.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setFloat32(offset, material.ambient[0], true);\n dataView.setFloat32(offset + 4, material.ambient[1], true);\n dataView.setFloat32(offset + 8, material.ambient[2], true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].DIFFUSE.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setFloat32(offset, material.diffuse[0], true);\n dataView.setFloat32(offset + 4, material.diffuse[1], true);\n dataView.setFloat32(offset + 8, material.diffuse[2], true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].SPECULAR.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setFloat32(offset, material.specular[0], true);\n dataView.setFloat32(offset + 4, material.specular[1], true);\n dataView.setFloat32(offset + 8, material.specular[2], true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].SPECULAR_EXPONENT.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setFloat32(offset, material.specularExponent, true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].EMISSIVE.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setFloat32(offset, material.emissive[0], true);\n dataView.setFloat32(offset + 4, material.emissive[1], true);\n dataView.setFloat32(offset + 8, material.emissive[2], true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].TRANSMISSION_FILTER.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setFloat32(offset, material.transmissionFilter[0], true);\n dataView.setFloat32(offset + 4, material.transmissionFilter[1], true);\n dataView.setFloat32(offset + 8, material.transmissionFilter[2], true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].DISSOLVE.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setFloat32(offset, material.dissolve, true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].ILLUMINATION.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setInt16(offset, material.illumination, true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].REFRACTION_INDEX.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setFloat32(offset, material.refractionIndex, true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].SHARPNESS.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setFloat32(offset, material.sharpness, true);\n break;\n }\n case _layout__WEBPACK_IMPORTED_MODULE_0__["Layout"].ANTI_ALIASING.key: {\n const materialIndex = this.vertexMaterialIndices[i];\n const material = this.materialsByIndex[materialIndex];\n if (!material) {\n console.warn(\'Material "\' +\n this.materialNames[materialIndex] +\n \'" not found in mesh. Did you forget to call addMaterialLibrary(...)?"\');\n break;\n }\n dataView.setInt16(offset, material.antiAliasing ? 1 : 0, true);\n break;\n }\n }\n }\n }\n return buffer;\n }\n makeIndexBufferData() {\n const buffer = new Uint16Array(this.indices);\n buffer.numItems = this.indices.length;\n return buffer;\n }\n makeIndexBufferDataForMaterials(...materialIndices) {\n const indices = new Array().concat(...materialIndices.map(mtlIdx => this.indicesPerMaterial[mtlIdx]));\n const buffer = new Uint16Array(indices);\n buffer.numItems = indices.length;\n return buffer;\n }\n addMaterialLibrary(mtl) {\n for (const name in mtl.materials) {\n if (!(name in this.materialIndices)) {\n // This material is not referenced by the mesh\n continue;\n }\n const material = mtl.materials[name];\n // Find the material index for this material\n const materialIndex = this.materialIndices[material.name];\n // Put the material into the materialsByIndex object at the right\n // spot as determined when the obj file was parsed\n this.materialsByIndex[materialIndex] = material;\n }\n }\n}\nfunction* triangulate(elements) {\n if (elements.length <= 3) {\n yield elements;\n }\n else if (elements.length === 4) {\n yield [elements[0], elements[1], elements[2]];\n yield [elements[2], elements[3], elements[0]];\n }\n else {\n for (let i = 1; i < elements.length - 1; i++) {\n yield [elements[0], elements[i], elements[i + 1]];\n }\n }\n}\n\n\n//# sourceURL=webpack://OBJ/./src/mesh.ts?')},"./src/utils.ts": /*!**********************!*\ !*** ./src/utils.ts ***! \**********************/ -/*! exports provided: downloadModels, downloadMeshes, initMeshBuffers, deleteMeshBuffers */function(module,__webpack_exports__,__webpack_require__){"use strict";eval('__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "downloadModels", function() { return downloadModels; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "downloadMeshes", function() { return downloadMeshes; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "initMeshBuffers", function() { return initMeshBuffers; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "deleteMeshBuffers", function() { return deleteMeshBuffers; });\n/* harmony import */ var _mesh__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./mesh */ "./src/mesh.ts");\n/* harmony import */ var _material__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./material */ "./src/material.ts");\n\n\nfunction downloadMtlTextures(mtl, root) {\n const mapAttributes = [\n "mapDiffuse",\n "mapAmbient",\n "mapSpecular",\n "mapDissolve",\n "mapBump",\n "mapDisplacement",\n "mapDecal",\n "mapEmissive",\n ];\n if (!root.endsWith("/")) {\n root += "/";\n }\n const textures = [];\n for (const materialName in mtl.materials) {\n if (!mtl.materials.hasOwnProperty(materialName)) {\n continue;\n }\n const material = mtl.materials[materialName];\n for (const attr of mapAttributes) {\n const mapData = material[attr];\n if (!mapData) {\n continue;\n }\n const url = root + mapData.filename;\n textures.push(fetch(url)\n .then(response => {\n if (!response.ok) {\n throw new Error();\n }\n return response.blob();\n })\n .then(function (data) {\n const image = new Image();\n image.src = URL.createObjectURL(data);\n mapData.texture = image;\n return new Promise(resolve => (image.onload = resolve));\n })\n .catch(() => {\n console.error(`Unable to download texture: ${url}`);\n }));\n }\n }\n return Promise.all(textures);\n}\nfunction getMtl(modelOptions) {\n if (!(typeof modelOptions.mtl === "string")) {\n return modelOptions.obj.replace(/\\.obj$/, ".mtl");\n }\n return modelOptions.mtl;\n}\n/**\n * Accepts a list of model request objects and returns a Promise that\n * resolves when all models have been downloaded and parsed.\n *\n * The list of model objects follow this interface:\n * {\n * obj: \'path/to/model.obj\',\n * mtl: true | \'path/to/model.mtl\',\n * downloadMtlTextures: true | false\n * mtlTextureRoot: \'/models/suzanne/maps\'\n * name: \'suzanne\'\n * }\n *\n * The `obj` attribute is required and should be the path to the\n * model\'s .obj file relative to the current repo (absolute URLs are\n * suggested).\n *\n * The `mtl` attribute is optional and can either be a boolean or\n * a path to the model\'s .mtl file relative to the current URL. If\n * the value is `true`, then the path and basename given for the `obj`\n * attribute is used replacing the .obj suffix for .mtl\n * E.g.: {obj: \'models/foo.obj\', mtl: true} would search for \'models/foo.mtl\'\n *\n * The `name` attribute is optional and is a human friendly name to be\n * included with the parsed OBJ and MTL files. If not given, the base .obj\n * filename will be used.\n *\n * The `downloadMtlTextures` attribute is a flag for automatically downloading\n * any images found in the MTL file and attaching them to each Material\n * created from that file. For example, if material.mapDiffuse is set (there\n * was data in the MTL file), then material.mapDiffuse.texture will contain\n * the downloaded image. This option defaults to `true`. By default, the MTL\'s\n * URL will be used to determine the location of the images.\n *\n * The `mtlTextureRoot` attribute is optional and should point to the location\n * on the server that this MTL\'s texture files are located. The default is to\n * use the MTL file\'s location.\n *\n * @returns {Promise} the result of downloading the given list of models. The\n * promise will resolve with an object whose keys are the names of the models\n * and the value is its Mesh object. Each Mesh object will automatically\n * have its addMaterialLibrary() method called to set the given MTL data (if given).\n */\nfunction downloadModels(models) {\n const finished = [];\n for (const model of models) {\n if (!model.obj) {\n throw new Error(\'"obj" attribute of model object not set. The .obj file is required to be set \' +\n "in order to use downloadModels()");\n }\n const options = {\n indicesPerMaterial: !!model.indicesPerMaterial,\n calcTangentsAndBitangents: !!model.calcTangentsAndBitangents,\n };\n // if the name is not provided, dervive it from the given OBJ\n let name = model.name;\n if (!name) {\n const parts = model.obj.split("/");\n name = parts[parts.length - 1].replace(".obj", "");\n }\n const namePromise = Promise.resolve(name);\n const meshPromise = fetch(model.obj)\n .then(response => response.text())\n .then(data => {\n return new _mesh__WEBPACK_IMPORTED_MODULE_0__["default"](data, options);\n });\n let mtlPromise;\n // Download MaterialLibrary file?\n if (model.mtl) {\n const mtl = getMtl(model);\n mtlPromise = fetch(mtl)\n .then(response => response.text())\n .then((data) => {\n const material = new _material__WEBPACK_IMPORTED_MODULE_1__["MaterialLibrary"](data);\n if (model.downloadMtlTextures !== false) {\n let root = model.mtlTextureRoot;\n if (!root) {\n // get the directory of the MTL file as default\n root = mtl.substr(0, mtl.lastIndexOf("/"));\n }\n // downloadMtlTextures returns a Promise that\n // is resolved once all of the images it\n // contains are downloaded. These are then\n // attached to the map data objects\n return Promise.all([Promise.resolve(material), downloadMtlTextures(material, root)]);\n }\n return Promise.all([Promise.resolve(material), undefined]);\n })\n .then((value) => {\n return value[0];\n });\n }\n const parsed = [namePromise, meshPromise, mtlPromise];\n finished.push(Promise.all(parsed));\n }\n return Promise.all(finished).then(ms => {\n // the "finished" promise is a list of name, Mesh instance,\n // and MaterialLibary instance. This unpacks and returns an\n // object mapping name to Mesh (Mesh points to MTL).\n const models = {};\n for (const model of ms) {\n const [name, mesh, mtl] = model;\n mesh.name = name;\n if (mtl) {\n mesh.addMaterialLibrary(mtl);\n }\n models[name] = mesh;\n }\n return models;\n });\n}\n/**\n * Takes in an object of `mesh_name`, `\'/url/to/OBJ/file\'` pairs and a callback\n * function. Each OBJ file will be ajaxed in and automatically converted to\n * an OBJ.Mesh. When all files have successfully downloaded the callback\n * function provided will be called and passed in an object containing\n * the newly created meshes.\n *\n * **Note:** In order to use this function as a way to download meshes, a\n * webserver of some sort must be used.\n *\n * @param {Object} nameAndAttrs an object where the key is the name of the mesh and the value is the url to that mesh\'s OBJ file\n *\n * @param {Function} completionCallback should contain a function that will take one parameter: an object array where the keys will be the unique object name and the value will be a Mesh object\n *\n * @param {Object} meshes In case other meshes are loaded separately or if a previously declared variable is desired to be used, pass in a (possibly empty) json object of the pattern: { \'\': OBJ.Mesh }\n *\n */\nfunction downloadMeshes(nameAndURLs, completionCallback, meshes) {\n if (meshes === undefined) {\n meshes = {};\n }\n const completed = [];\n for (const mesh_name in nameAndURLs) {\n if (!nameAndURLs.hasOwnProperty(mesh_name)) {\n continue;\n }\n const url = nameAndURLs[mesh_name];\n completed.push(fetch(url)\n .then(response => response.text())\n .then(data => {\n return [mesh_name, new _mesh__WEBPACK_IMPORTED_MODULE_0__["default"](data)];\n }));\n }\n Promise.all(completed).then(ms => {\n for (const [name, mesh] of ms) {\n meshes[name] = mesh;\n }\n return completionCallback(meshes);\n });\n}\nfunction _buildBuffer(gl, type, data, itemSize) {\n const buffer = gl.createBuffer();\n const arrayView = type === gl.ARRAY_BUFFER ? Float32Array : Uint16Array;\n gl.bindBuffer(type, buffer);\n gl.bufferData(type, new arrayView(data), gl.STATIC_DRAW);\n buffer.itemSize = itemSize;\n buffer.numItems = data.length / itemSize;\n return buffer;\n}\n/**\n * Takes in the WebGL context and a Mesh, then creates and appends the buffers\n * to the mesh object as attributes.\n *\n * @param {WebGLRenderingContext} gl the `canvas.getContext(\'webgl\')` context instance\n * @param {Mesh} mesh a single `OBJ.Mesh` instance\n *\n * The newly created mesh attributes are:\n *\n * Attrbute | Description\n * :--- | ---\n * **normalBuffer** |contains the model's Vertex Normals\n * normalBuffer.itemSize |set to 3 items\n * normalBuffer.numItems |the total number of vertex normals\n * |\n * **textureBuffer** |contains the model's Texture Coordinates\n * textureBuffer.itemSize |set to 2 items\n * textureBuffer.numItems |the number of texture coordinates\n * |\n * **vertexBuffer** |contains the model's Vertex Position Coordinates (does not include w)\n * vertexBuffer.itemSize |set to 3 items\n * vertexBuffer.numItems |the total number of vertices\n * |\n * **indexBuffer** |contains the indices of the faces\n * indexBuffer.itemSize |is set to 1\n * indexBuffer.numItems |the total number of indices\n *\n * A simple example (a lot of steps are missing, so don\'t copy and paste):\n *\n * const gl = canvas.getContext(\'webgl\'),\n * mesh = OBJ.Mesh(obj_file_data);\n * // compile the shaders and create a shader program\n * const shaderProgram = gl.createProgram();\n * // compilation stuff here\n * ...\n * // make sure you have vertex, vertex normal, and texture coordinate\n * // attributes located in your shaders and attach them to the shader program\n * shaderProgram.vertexPositionAttribute = gl.getAttribLocation(shaderProgram, "aVertexPosition");\n * gl.enableVertexAttribArray(shaderProgram.vertexPositionAttribute);\n *\n * shaderProgram.vertexNormalAttribute = gl.getAttribLocation(shaderProgram, "aVertexNormal");\n * gl.enableVertexAttribArray(shaderProgram.vertexNormalAttribute);\n *\n * shaderProgram.textureCoordAttribute = gl.getAttribLocation(shaderProgram, "aTextureCoord");\n * gl.enableVertexAttribArray(shaderProgram.textureCoordAttribute);\n *\n * // create and initialize the vertex, vertex normal, and texture coordinate buffers\n * // and save on to the mesh object\n * OBJ.initMeshBuffers(gl, mesh);\n *\n * // now to render the mesh\n * gl.bindBuffer(gl.ARRAY_BUFFER, mesh.vertexBuffer);\n * gl.vertexAttribPointer(shaderProgram.vertexPositionAttribute, mesh.vertexBuffer.itemSize, gl.FLOAT, false, 0, 0);\n * // it\'s possible that the mesh doesn\'t contain\n * // any texture coordinates (e.g. suzanne.obj in the development branch).\n * // in this case, the texture vertexAttribArray will need to be disabled\n * // before the call to drawElements\n * if(!mesh.textures.length){\n * gl.disableVertexAttribArray(shaderProgram.textureCoordAttribute);\n * }\n * else{\n * // if the texture vertexAttribArray has been previously\n * // disabled, then it needs to be re-enabled\n * gl.enableVertexAttribArray(shaderProgram.textureCoordAttribute);\n * gl.bindBuffer(gl.ARRAY_BUFFER, mesh.textureBuffer);\n * gl.vertexAttribPointer(shaderProgram.textureCoordAttribute, mesh.textureBuffer.itemSize, gl.FLOAT, false, 0, 0);\n * }\n *\n * gl.bindBuffer(gl.ARRAY_BUFFER, mesh.normalBuffer);\n * gl.vertexAttribPointer(shaderProgram.vertexNormalAttribute, mesh.normalBuffer.itemSize, gl.FLOAT, false, 0, 0);\n *\n * gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, model.mesh.indexBuffer);\n * gl.drawElements(gl.TRIANGLES, model.mesh.indexBuffer.numItems, gl.UNSIGNED_SHORT, 0);\n */\nfunction initMeshBuffers(gl, mesh) {\n mesh.normalBuffer = _buildBuffer(gl, gl.ARRAY_BUFFER, mesh.vertexNormals, 3);\n mesh.textureBuffer = _buildBuffer(gl, gl.ARRAY_BUFFER, mesh.textures, mesh.textureStride);\n mesh.vertexBuffer = _buildBuffer(gl, gl.ARRAY_BUFFER, mesh.vertices, 3);\n mesh.indexBuffer = _buildBuffer(gl, gl.ELEMENT_ARRAY_BUFFER, mesh.indices, 1);\n return mesh;\n}\nfunction deleteMeshBuffers(gl, mesh) {\n gl.deleteBuffer(mesh.normalBuffer);\n gl.deleteBuffer(mesh.textureBuffer);\n gl.deleteBuffer(mesh.vertexBuffer);\n gl.deleteBuffer(mesh.indexBuffer);\n}\n\n\n//# sourceURL=webpack://OBJ/./src/utils.ts?')},0: +/*! exports provided: downloadModels, downloadMeshes, initMeshBuffers, deleteMeshBuffers */function(module,__webpack_exports__,__webpack_require__){"use strict";eval('__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "downloadModels", function() { return downloadModels; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "downloadMeshes", function() { return downloadMeshes; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "initMeshBuffers", function() { return initMeshBuffers; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "deleteMeshBuffers", function() { return deleteMeshBuffers; });\n/* harmony import */ var _mesh__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./mesh */ "./src/mesh.ts");\n/* harmony import */ var _material__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./material */ "./src/material.ts");\n\n\nfunction downloadMtlTextures(mtl, root) {\n const mapAttributes = [\n "mapDiffuse",\n "mapAmbient",\n "mapSpecular",\n "mapDissolve",\n "mapBump",\n "mapDisplacement",\n "mapDecal",\n "mapEmissive",\n ];\n if (!root.endsWith("/")) {\n root += "/";\n }\n const textures = [];\n for (const materialName in mtl.materials) {\n if (!mtl.materials.hasOwnProperty(materialName)) {\n continue;\n }\n const material = mtl.materials[materialName];\n for (const attr of mapAttributes) {\n const mapData = material[attr];\n if (!mapData || !mapData.filename) {\n continue;\n }\n const url = root + mapData.filename;\n textures.push(fetch(url)\n .then(response => {\n if (!response.ok) {\n throw new Error();\n }\n return response.blob();\n })\n .then(function (data) {\n const image = new Image();\n image.src = URL.createObjectURL(data);\n mapData.texture = image;\n return new Promise(resolve => (image.onload = resolve));\n })\n .catch(() => {\n console.error(`Unable to download texture: ${url}`);\n }));\n }\n }\n return Promise.all(textures);\n}\nfunction getMtl(modelOptions) {\n if (!(typeof modelOptions.mtl === "string")) {\n return modelOptions.obj.replace(/\\.obj$/, ".mtl");\n }\n return modelOptions.mtl;\n}\n/**\n * Accepts a list of model request objects and returns a Promise that\n * resolves when all models have been downloaded and parsed.\n *\n * The list of model objects follow this interface:\n * {\n * obj: \'path/to/model.obj\',\n * mtl: true | \'path/to/model.mtl\',\n * downloadMtlTextures: true | false\n * mtlTextureRoot: \'/models/suzanne/maps\'\n * name: \'suzanne\'\n * }\n *\n * The `obj` attribute is required and should be the path to the\n * model\'s .obj file relative to the current repo (absolute URLs are\n * suggested).\n *\n * The `mtl` attribute is optional and can either be a boolean or\n * a path to the model\'s .mtl file relative to the current URL. If\n * the value is `true`, then the path and basename given for the `obj`\n * attribute is used replacing the .obj suffix for .mtl\n * E.g.: {obj: \'models/foo.obj\', mtl: true} would search for \'models/foo.mtl\'\n *\n * The `name` attribute is optional and is a human friendly name to be\n * included with the parsed OBJ and MTL files. If not given, the base .obj\n * filename will be used.\n *\n * The `downloadMtlTextures` attribute is a flag for automatically downloading\n * any images found in the MTL file and attaching them to each Material\n * created from that file. For example, if material.mapDiffuse is set (there\n * was data in the MTL file), then material.mapDiffuse.texture will contain\n * the downloaded image. This option defaults to `true`. By default, the MTL\'s\n * URL will be used to determine the location of the images.\n *\n * The `mtlTextureRoot` attribute is optional and should point to the location\n * on the server that this MTL\'s texture files are located. The default is to\n * use the MTL file\'s location.\n *\n * @returns {Promise} the result of downloading the given list of models. The\n * promise will resolve with an object whose keys are the names of the models\n * and the value is its Mesh object. Each Mesh object will automatically\n * have its addMaterialLibrary() method called to set the given MTL data (if given).\n */\nfunction downloadModels(models) {\n const finished = [];\n for (const model of models) {\n if (!model.obj) {\n throw new Error(\'"obj" attribute of model object not set. The .obj file is required to be set \' +\n "in order to use downloadModels()");\n }\n const options = {\n indicesPerMaterial: !!model.indicesPerMaterial,\n calcTangentsAndBitangents: !!model.calcTangentsAndBitangents,\n };\n // if the name is not provided, dervive it from the given OBJ\n let name = model.name;\n if (!name) {\n const parts = model.obj.split("/");\n name = parts[parts.length - 1].replace(".obj", "");\n }\n const namePromise = Promise.resolve(name);\n const meshPromise = fetch(model.obj)\n .then(response => response.text())\n .then(data => {\n return new _mesh__WEBPACK_IMPORTED_MODULE_0__["default"](data, options);\n });\n let mtlPromise;\n // Download MaterialLibrary file?\n if (model.mtl) {\n const mtl = getMtl(model);\n mtlPromise = fetch(mtl)\n .then(response => response.text())\n .then((data) => {\n const material = new _material__WEBPACK_IMPORTED_MODULE_1__["MaterialLibrary"](data);\n if (model.downloadMtlTextures !== false) {\n let root = model.mtlTextureRoot;\n if (!root) {\n // get the directory of the MTL file as default\n root = mtl.substr(0, mtl.lastIndexOf("/"));\n }\n // downloadMtlTextures returns a Promise that\n // is resolved once all of the images it\n // contains are downloaded. These are then\n // attached to the map data objects\n return Promise.all([Promise.resolve(material), downloadMtlTextures(material, root)]);\n }\n return Promise.all([Promise.resolve(material), undefined]);\n })\n .then((value) => {\n return value[0];\n });\n }\n const parsed = [namePromise, meshPromise, mtlPromise];\n finished.push(Promise.all(parsed));\n }\n return Promise.all(finished).then(ms => {\n // the "finished" promise is a list of name, Mesh instance,\n // and MaterialLibary instance. This unpacks and returns an\n // object mapping name to Mesh (Mesh points to MTL).\n const models = {};\n for (const model of ms) {\n const [name, mesh, mtl] = model;\n mesh.name = name;\n if (mtl) {\n mesh.addMaterialLibrary(mtl);\n }\n models[name] = mesh;\n }\n return models;\n });\n}\n/**\n * Takes in an object of `mesh_name`, `\'/url/to/OBJ/file\'` pairs and a callback\n * function. Each OBJ file will be ajaxed in and automatically converted to\n * an OBJ.Mesh. When all files have successfully downloaded the callback\n * function provided will be called and passed in an object containing\n * the newly created meshes.\n *\n * **Note:** In order to use this function as a way to download meshes, a\n * webserver of some sort must be used.\n *\n * @param {Object} nameAndAttrs an object where the key is the name of the mesh and the value is the url to that mesh\'s OBJ file\n *\n * @param {Function} completionCallback should contain a function that will take one parameter: an object array where the keys will be the unique object name and the value will be a Mesh object\n *\n * @param {Object} meshes In case other meshes are loaded separately or if a previously declared variable is desired to be used, pass in a (possibly empty) json object of the pattern: { \'\': OBJ.Mesh }\n *\n */\nfunction downloadMeshes(nameAndURLs, completionCallback, meshes) {\n if (meshes === undefined) {\n meshes = {};\n }\n const completed = [];\n for (const mesh_name in nameAndURLs) {\n if (!nameAndURLs.hasOwnProperty(mesh_name)) {\n continue;\n }\n const url = nameAndURLs[mesh_name];\n completed.push(fetch(url)\n .then(response => response.text())\n .then(data => {\n return [mesh_name, new _mesh__WEBPACK_IMPORTED_MODULE_0__["default"](data)];\n }));\n }\n Promise.all(completed).then(ms => {\n for (const [name, mesh] of ms) {\n meshes[name] = mesh;\n }\n return completionCallback(meshes);\n });\n}\nfunction _buildBuffer(gl, type, data, itemSize) {\n const buffer = gl.createBuffer();\n const arrayView = type === gl.ARRAY_BUFFER ? Float32Array : Uint16Array;\n gl.bindBuffer(type, buffer);\n gl.bufferData(type, new arrayView(data), gl.STATIC_DRAW);\n buffer.itemSize = itemSize;\n buffer.numItems = data.length / itemSize;\n return buffer;\n}\n/**\n * Takes in the WebGL context and a Mesh, then creates and appends the buffers\n * to the mesh object as attributes.\n *\n * @param {WebGLRenderingContext} gl the `canvas.getContext(\'webgl\')` context instance\n * @param {Mesh} mesh a single `OBJ.Mesh` instance\n *\n * The newly created mesh attributes are:\n *\n * Attrbute | Description\n * :--- | ---\n * **normalBuffer** |contains the model's Vertex Normals\n * normalBuffer.itemSize |set to 3 items\n * normalBuffer.numItems |the total number of vertex normals\n * |\n * **textureBuffer** |contains the model's Texture Coordinates\n * textureBuffer.itemSize |set to 2 items\n * textureBuffer.numItems |the number of texture coordinates\n * |\n * **vertexBuffer** |contains the model's Vertex Position Coordinates (does not include w)\n * vertexBuffer.itemSize |set to 3 items\n * vertexBuffer.numItems |the total number of vertices\n * |\n * **indexBuffer** |contains the indices of the faces\n * indexBuffer.itemSize |is set to 1\n * indexBuffer.numItems |the total number of indices\n *\n * A simple example (a lot of steps are missing, so don\'t copy and paste):\n *\n * const gl = canvas.getContext(\'webgl\'),\n * mesh = OBJ.Mesh(obj_file_data);\n * // compile the shaders and create a shader program\n * const shaderProgram = gl.createProgram();\n * // compilation stuff here\n * ...\n * // make sure you have vertex, vertex normal, and texture coordinate\n * // attributes located in your shaders and attach them to the shader program\n * shaderProgram.vertexPositionAttribute = gl.getAttribLocation(shaderProgram, "aVertexPosition");\n * gl.enableVertexAttribArray(shaderProgram.vertexPositionAttribute);\n *\n * shaderProgram.vertexNormalAttribute = gl.getAttribLocation(shaderProgram, "aVertexNormal");\n * gl.enableVertexAttribArray(shaderProgram.vertexNormalAttribute);\n *\n * shaderProgram.textureCoordAttribute = gl.getAttribLocation(shaderProgram, "aTextureCoord");\n * gl.enableVertexAttribArray(shaderProgram.textureCoordAttribute);\n *\n * // create and initialize the vertex, vertex normal, and texture coordinate buffers\n * // and save on to the mesh object\n * OBJ.initMeshBuffers(gl, mesh);\n *\n * // now to render the mesh\n * gl.bindBuffer(gl.ARRAY_BUFFER, mesh.vertexBuffer);\n * gl.vertexAttribPointer(shaderProgram.vertexPositionAttribute, mesh.vertexBuffer.itemSize, gl.FLOAT, false, 0, 0);\n * // it\'s possible that the mesh doesn\'t contain\n * // any texture coordinates (e.g. suzanne.obj in the development branch).\n * // in this case, the texture vertexAttribArray will need to be disabled\n * // before the call to drawElements\n * if(!mesh.textures.length){\n * gl.disableVertexAttribArray(shaderProgram.textureCoordAttribute);\n * }\n * else{\n * // if the texture vertexAttribArray has been previously\n * // disabled, then it needs to be re-enabled\n * gl.enableVertexAttribArray(shaderProgram.textureCoordAttribute);\n * gl.bindBuffer(gl.ARRAY_BUFFER, mesh.textureBuffer);\n * gl.vertexAttribPointer(shaderProgram.textureCoordAttribute, mesh.textureBuffer.itemSize, gl.FLOAT, false, 0, 0);\n * }\n *\n * gl.bindBuffer(gl.ARRAY_BUFFER, mesh.normalBuffer);\n * gl.vertexAttribPointer(shaderProgram.vertexNormalAttribute, mesh.normalBuffer.itemSize, gl.FLOAT, false, 0, 0);\n *\n * gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, model.mesh.indexBuffer);\n * gl.drawElements(gl.TRIANGLES, model.mesh.indexBuffer.numItems, gl.UNSIGNED_SHORT, 0);\n */\nfunction initMeshBuffers(gl, mesh) {\n mesh.normalBuffer = _buildBuffer(gl, gl.ARRAY_BUFFER, mesh.vertexNormals, 3);\n mesh.textureBuffer = _buildBuffer(gl, gl.ARRAY_BUFFER, mesh.textures, mesh.textureStride);\n mesh.vertexBuffer = _buildBuffer(gl, gl.ARRAY_BUFFER, mesh.vertices, 3);\n mesh.indexBuffer = _buildBuffer(gl, gl.ELEMENT_ARRAY_BUFFER, mesh.indices, 1);\n return mesh;\n}\nfunction deleteMeshBuffers(gl, mesh) {\n gl.deleteBuffer(mesh.normalBuffer);\n gl.deleteBuffer(mesh.textureBuffer);\n gl.deleteBuffer(mesh.vertexBuffer);\n gl.deleteBuffer(mesh.indexBuffer);\n}\n\n\n//# sourceURL=webpack://OBJ/./src/utils.ts?')},0: /*!****************************!*\ !*** multi ./src/index.ts ***! \****************************/ -/*! no static exports found */function(module,exports,__webpack_require__){eval('module.exports = __webpack_require__(/*! /Users/aaron/git/webgl-obj-loader/src/index.ts */"./src/index.ts");\n\n\n//# sourceURL=webpack://OBJ/multi_./src/index.ts?')}})})); \ No newline at end of file +/*! no static exports found */function(module,exports,__webpack_require__){eval('module.exports = __webpack_require__(/*! /home/aaron/google_drive/projects/webgl-obj-loader/src/index.ts */"./src/index.ts");\n\n\n//# sourceURL=webpack://OBJ/multi_./src/index.ts?')}})})); \ No newline at end of file