/**
 * Minified by jsDelivr using Terser v5.39.0.
 * Original file: /npm/@zeainc/zea-engine@4.17.0/dist/index.esm.mjs
 *
 * Do NOT use SRI with dynamically generated files! More information: https://www.jsdelivr.com/using-sri-with-dynamic-files
 */
var version="4.17.0";class LibsRegistry{registry;version;constructor(A){this.version=A,this.registry={}}registerLib(A){const g=A.name,I=A.version;this.registry[g]&&console.warn("Library already registered:"+g),this.registry[g]=I,console.log(`Registered lib '${g}' v${I}`)}listLibs(){return this.registry}}function isIOSDevice(){return!!/iPad|iPhone|iPod/.test(navigator.platform)||navigator.maxTouchPoints&&navigator.maxTouchPoints>2&&/MacIntel/.test(navigator.platform)}function isMobileDevice(){return navigator.maxTouchPoints&&navigator.maxTouchPoints>0&&/Android|iPhone|iPad|iPod/i.test(navigator.userAgent)}function getBrowserDesc(){const A=navigator.userAgent;let g,I,C,e="",i=""+parseFloat(navigator.appVersion);return navigator.brave?(e="Brave",I=A.indexOf("Chrome"),i=A.substring(I+7,A.indexOf(" ",I+7))):-1!=(I=A.indexOf("Opera"))?(e="Opera",i=A.substring(I+6),-1!=(I=A.indexOf("Version"))&&(i=A.substring(I+8))):-1!=(I=A.indexOf("MSIE"))?(e="Microsoft Internet Explorer",i=A.substring(I+5)):-1!=(I=A.indexOf("Edge"))?(e="Edge",i=A.substring(I+4)):-1!=(I=A.indexOf("Chrome"))?(e="Chrome",i=A.substring(I+7,A.indexOf(" ",I+7))):-1!=(I=A.indexOf("Safari"))?(e="Safari",i=A.substring(I+7),-1!=(I=A.indexOf("Version"))&&(i=A.substring(I+8))):-1!=(I=A.indexOf("Firefox"))?(e="Firefox",i=A.substring(I+8)):(g=A.lastIndexOf(" ")+1)<(I=A.lastIndexOf("/"))&&(e=A.substring(g,I),i=A.substring(I+1)),-1!=(C=i.indexOf(";"))&&(i=i.substring(0,C)),-1!=(C=i.indexOf(" "))&&(i=i.substring(0,C)),{browserName:e,fullVersion:i}}function getGPUDesc(){let A,g;try{A=document.createElement("canvas").getContext("webgl")}catch(A){}if(!A)return{vendor:"Unknown",renderer:"Unknown",gpuVendor:"Unknown",maxTextureSize:0,supportsWebGL:!1,supportsWebGL2:!1};try{g=document.createElement("canvas").getContext("webgl2")}catch(A){}const I=A.getExtension("WEBGL_debug_renderer_info");if(!I)return console.warn("Unable to determine GPU Info:"),{vendor:"Unknown",renderer:"Unknown",gpuVendor:"Unknown",maxTextureSize:0,supportsWebGL:null!=A,supportsWebGL2:null!=g};const C=A.getParameter(I.UNMASKED_VENDOR_WEBGL),e=A.getParameter(I.UNMASKED_RENDERER_WEBGL),i=A.getParameter(A.MAX_TEXTURE_SIZE);let t;return e.match(/NVIDIA/i)?t="NVidia":e.match(/AMD/i)||e.match(/Radeon/i)?t="AMD":e.match(/Intel/i)?t="Intel":e.match(/Mali/i)?t="ARM":e.match(/Apple/i)?t="Apple":e.match(/Adreno/i)?t="Adreno":e.match(/Swiftshader/i)?(t="Google",console.warn("Hardware rendering is disabled or not working on your system. Falling back to the Swiftshader. Expect poor performance:",e)):console.warn("Unable to determine GPU vendor:",e),{vendor:C,renderer:e,gpuVendor:t,maxTextureSize:i,supportsWebGL:!0,supportsWebGL2:null!=g}}const getOS=()=>{const{userAgent:A}=window.navigator;let g=null;return/macintosh/i.test(A)?g="macOS":/ios/i.test(A)?g="iOS":/win/i.test(A)?g="Windows":/android/i.test(A)?g="Android":/linux/i.test(A)&&(g="Linux"),g},SystemDesc=function(){if(!globalThis.navigator||/node/i.test(globalThis.navigator.userAgent))return{OS:"Node",isMobileDevice:!1,isIOSDevice:!1,browserName:"Node",webGLSupported:!1,deviceCategory:"High",hardwareConcurrency:4};const A=isMobileDevice(),g=getBrowserDesc(),I=getGPUDesc();let C="Low";if(I.supportsWebGL)if(A)C="Low";else{const A=I.renderer.replace(/[()]/g,"").split(" ");if("NVidia"==I.gpuVendor){const g=A.indexOf("GTX");if(-1!=g){const I=A[g+1];if(I.endsWith("M")){C=parseInt(I.substring(0,I.length-2))>=900?"Medium":"Low"}else{C=parseInt(I)>=1030?"High":"Medium"}}else C=A.includes("RTX")||A.includes("TITAN")||A.includes("Quadro")?"High":"Low"}else if("AMD"==I.gpuVendor){const g=A.indexOf("Radeon");if(-1!=g){const I=A.indexOf("RX");if(-1!=I)if("Vega"==A[I+1])C="High";else{const g=A[I+1];let e;g.endsWith("X")?(e=parseInt(g.substring(0,g.length-2)),C="High"):e=parseInt(g),C=e>=480?"High":"Medium"}else if("Pro"==A[g+1]){C=parseInt(A[I+1])>=450?"Medium":"Low"}else if("Sky"==A[g+1]){C=parseInt(A[I+1])>=700?"Medium":"Low"}else C="Low"}else C=A.includes("FirePro")||A.includes("Quadro")?"High":"Low"}else("Adreno"==I.gpuVendor||"Intel"==I.gpuVendor||"Google"==I.gpuVendor)&&(C="Low")}let e=globalThis.navigator.hardwareConcurrency;return e||(e=A?4:6),{OS:getOS(),isMobileDevice:A,isIOSDevice:isIOSDevice(),browserName:g.browserName,fullVersion:g.fullVersion,webGLSupported:I.supportsWebGL,gpuDesc:I,deviceCategory:C,hardwareConcurrency:e}}();globalThis.ZeaSystemDesc||(globalThis.ZeaSystemDesc=SystemDesc);let registeredClasses={},classNames={},classDefinitions=[];class Registry{static register(A,g){if(A in registeredClasses)return void console.warn(`There's a class registered with '${A}' name. Second registration failed.`);const I=classDefinitions.length;classDefinitions.push(g),classNames[I]=A,registeredClasses[A]=I}static getClassDefinition(A){if(!(A in registeredClasses))throw new Error(`${A} class is not registered`);return classDefinitions[registeredClasses[A]]}static getClassName(A){const g=classDefinitions.indexOf(A);if(g>=0&&classNames[g])return classNames[g];throw new Error("class is not registered")}static constructClass(A){const g=classDefinitions[registeredClasses[A]];if(!g)throw new Error(`${A} class is not registered`);return new g}static flush(){registeredClasses={},classNames={},classDefinitions=[]}}class StringFunctions{static replaceAll(A,g,I){return A.replace(new RegExp(g,"g"),I)}static stringifyJSONWithFixedPrecision(A,g=0,I=5){return JSON.stringify(A,((A,g)=>g&&g.toFixed?Number(g.toFixed(I)):g),g)}static hashStr(A){let g,I,C,e=0;if(0===A.length)return e;for(g=0,C=A.length;g<C;g++)I=A.charCodeAt(g),e=(e<<5)-e+I,e|=0;return Math.abs(e)}}class Vec2{x;y;constructor(A=0,g=0){this.x=A,this.y=g}set(A,g){this.x=A,this.y=g}setFromOther(A){this.x=A.x,this.y=A.y}isEqual(A){return this.x==A.x&&this.y==A.y}notEqual(A){return this.x!=A.x&&this.y!=A.y}approxEqual(A,g=Number.EPSILON){return Math.abs(this.x-A.x)<g&&Math.abs(this.y-A.y)<g}add(A){return new Vec2(this.x+A.x,this.y+A.y)}addInPlace(A){this.x+=A.x,this.y+=A.y}subtract(A){return new Vec2(this.x-A.x,this.y-A.y)}subtractInPlace(A){return this.x-=A.x,this.y-=A.y,this}scale(A){return new Vec2(this.x*A,this.y*A)}scaleInPlace(A){this.x*=A,this.y*=A}invert(){return new Vec2(1/this.x,1/this.y)}invertInPlace(){return this.x=1/this.x,this.y=1/this.y,this}multiply(A){return new Vec2(this.x*A.x,this.y*A.y)}multiplyInPlace(A){this.x*=A.x,this.y*=A.y}lengthSquared(){const A=this.x,g=this.y;return A*A+g*g}length(){return Math.sqrt(this.lengthSquared())}distanceTo(A){const g=this.x-A.x,I=this.y-A.y;return Math.sqrt(g*g+I*I)}normalize(){const A=this.x,g=this.y;let I=A*A+g*g;return I<Number.EPSILON?new Vec2:(I=1/Math.sqrt(I),new Vec2(A*I,g*I))}normalizeInPlace(){const A=this.x,g=this.y;let I=A*A+g*g;I<Number.EPSILON||(I=1/Math.sqrt(I),this.set(A*I,g*I))}dot(A){return this.x*A.x+this.y*A.y}cross(A){return this.x*A.y-this.y*A.x}angleTo(A){const g=this.normalize().dot(A.normalize());return g>1?0:g<-1?Math.PI:Math.acos(g)}signedAngleTo(A){const g=this.angleTo(A);return this.cross(A)<0?-g:g}rotate(A){const g=Math.cos(A),I=Math.sin(A);return new Vec2(this.x*g-this.y*I,this.x*I+this.y*g)}lerp(A,g){const I=this.x,C=this.y;return new Vec2(I+g*(A.x-I),C+g*(A.y-C))}setRandomDir(A=1){const g=2*Math.random()*Math.PI;return this.x=Math.cos(g)*A,this.y=Math.sin(g)*A,this}setRandom(A=1){return this.x=Math.random()*A,this.y=Math.random()*A,this}clone(){return new Vec2(this.x,this.y)}asArray(){return[this.x,this.y]}fromArray(A){this.x=A[0],this.y=A[1]}toString(){return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON())}toJSON(){return{x:this.x,y:this.y}}fromJSON(A){this.x=A.x,this.y=A.y}readBinary(A){this.x=A.loadFloat32(),this.y=A.loadFloat32()}static intersectionOfLines(A,g,I,C){const e=(A.x-g.x)*(I.y-C.y)-(A.y-g.y)*(I.x-C.x);if(0==e)return null;const i=A.x*g.y-A.y*g.x,t=I.x*C.y-I.y*C.x,o=I.x-C.x,a=A.x-g.x,B=I.y-C.y,s=A.y-g.y;return new Vec2((i*o-a*t)/e,(i*B-s*t)/e)}isValid(){for(const A of this.asArray())if(A==1/0||isNaN(A))return!1;return!0}}class Vec3{x;y;z;constructor(A=0,g=0,I=0){this.x=A,this.y=g,this.z=I}get xy(){return new Vec2(this.x,this.y)}get yz(){return new Vec2(this.y,this.z)}set(A,g,I){this.x=A,this.y=void 0!==g?g:A,this.z=void 0!==I?I:A}setFromOther(A){this.x=A.x,this.y=A.y,this.z=A.z}isNull(){return Math.abs(this.x)<Number.EPSILON&&Math.abs(this.y)<Number.EPSILON&&Math.abs(this.z)<Number.EPSILON}is111(){return Math.abs(1-this.x)<Number.EPSILON&&Math.abs(1-this.y)<Number.EPSILON&&Math.abs(1-this.z)<Number.EPSILON}isEqual(A){return this.x==A.x&&this.y==A.y&&this.z==A.z}notEqual(A){return this.x!=A.x&&this.y!=A.y&&this.z!=A.z}approxEqual(A,g=Number.EPSILON){return Math.abs(this.x-A.x)<g&&Math.abs(this.y-A.y)<g&&Math.abs(this.z-A.z)<g}add(A){return new Vec3(this.x+A.x,this.y+A.y,this.z+A.z)}addInPlace(A){this.x+=A.x,this.y+=A.y,this.z+=A.z}subtract(A){return new Vec3(this.x-A.x,this.y-A.y,this.z-A.z)}subtractInPlace(A){this.x-=A.x,this.y-=A.y,this.z-=A.z}multiply(A){return new Vec3(this.x*A.x,this.y*A.y,this.z*A.z)}multiplyInPlace(A){this.x*=A.x,this.y*=A.y,this.z*=A.z}divide(A){return new Vec3(this.x/A.x,this.y/A.y,this.z/A.z)}divideInPlace(A){this.x/=A.x,this.y/=A.y,this.z/=A.z}scale(A){return new Vec3(this.x*A,this.y*A,this.z*A)}scaleInPlace(A){this.x*=A,this.y*=A,this.z*=A}negate(){return new Vec3(-this.x,-this.y,-this.z)}inverse(){return new Vec3(1/this.x,1/this.y,1/this.z)}lengthSquared(){const A=this.x,g=this.y,I=this.z;return A*A+g*g+I*I}length(){return Math.sqrt(this.lengthSquared())}distanceTo(A){const g=this.x-A.x,I=this.y-A.y,C=this.z-A.z;return Math.sqrt(g*g+I*I+C*C)}normalize(){let A=this.x*this.x+this.y*this.y+this.z*this.z;return A<Number.EPSILON?new Vec3:(A=1/Math.sqrt(A),new Vec3(this.x*A,this.y*A,this.z*A))}normalizeInPlace(){let A=this.x*this.x+this.y*this.y+this.z*this.z;if(A<Number.EPSILON)return;A=Math.sqrt(A);const g=1/A;return this.x*=g,this.y*=g,this.z*=g,A}resize(A){const g=this.x*this.x+this.y*this.y+this.z*this.z;if(g<Number.EPSILON)return;const I=A/Math.sqrt(g);return new Vec3(this.x*I,this.y*I,this.z*I)}resizeInPlace(A){const g=this.x*this.x+this.y*this.y+this.z*this.z;if(g<Number.EPSILON)return;const I=A/Math.sqrt(g);this.x*=I,this.y*=I,this.z*=I}dot(A){return this.x*A.x+this.y*A.y+this.z*A.z}cross(A){const g=this.x,I=this.y,C=this.z,e=A.x,i=A.y,t=A.z;return new Vec3(I*t-C*i,C*e-g*t,g*i-I*e)}angleTo(A){const g=this.dot(A);return g>1?0:Math.acos(g)}lerp(A,g){const I=this.x,C=this.y,e=this.z;return new Vec3(I+g*(A.x-I),C+g*(A.y-C),e+g*(A.z-e))}abs(){return new Vec3(Math.abs(this.x),Math.abs(this.y),Math.abs(this.z))}setRandomDir(A=1){const g=2*Math.random()*Math.PI,I=2*Math.random()-1,C=Math.sqrt(1-I*I)*A;return this.x=Math.cos(g)*C,this.y=Math.sin(g)*C,this.z=I*A,this}setRandom(A=1){return this.x=(Math.random()-.5)*A,this.y=(Math.random()-.5)*A,this.z=(Math.random()-.5)*A,this}clone(){return new Vec3(this.x,this.y,this.z)}asArray(){return[this.x,this.y,this.z]}fromArray(A){this.x=A[0],this.y=A[1],this.z=A[1]}toString(){return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON())}toJSON(){return{x:this.x,y:this.y,z:this.z}}fromJSON(A){this.x=A.x,this.y=A.y,this.z=A.z}readBinary(A){this.x=A.loadFloat32(),this.y=A.loadFloat32(),this.z=A.loadFloat32()}isValid(){for(const A of this.asArray())if(A==1/0||isNaN(A))return!1;return!0}}class Vec4{x;y;z;w;constructor(A=0,g=0,I=0,C=0){this.x=A,this.y=g,this.z=I,this.w=C}get xyz(){return new Vec3(this.x,this.y,this.z)}set(A,g,I,C){this.x=A,this.y=g,this.z=I,this.w=C}setFromOther(A){this.x=A.x,this.y=A.y,this.z=A.z,this.w=A.w}isEqual(A){return this.x==A.x&&this.y==A.y&&this.z==A.z&&this.w==A.w}notEqual(A){return this.x!=A.x&&this.y!=A.y&&this.z!=A.z&&this.w!=A.w}approxEqual(A,g=Number.EPSILON){return Math.abs(this.x-A.x)<g&&Math.abs(this.y-A.y)<g&&Math.abs(this.z-A.z)<g&&Math.abs(this.w-A.w)<g}add(A){return new Vec4(this.x+A.x,this.y+A.y,this.z+A.z,this.w+A.w)}addInPlace(A){this.x+=A.x,this.y+=A.y,this.z+=A.z,this.w+=A.w}subtract(A){return new Vec4(this.x-A.x,this.y-A.y,this.z-A.z,this.w-A.w)}subtractInPlace(A){this.x-=A.x,this.y-=A.y,this.z-=A.z,this.w-=A.w}multiply(A){return new Vec4(this.x*A.x,this.y*A.y,this.z*A.z,this.w*A.w)}multiplyInPlace(A){this.x*=A.x,this.y*=A.y,this.z*=A.z,this.w*=A.w}divide(A){return new Vec4(this.x/A.x,this.y/A.y,this.z/A.z,this.w/A.w)}divideInPlace(A){this.x/=A.x,this.y/=A.y,this.z/=A.z,this.w/=A.w}scale(A){return new Vec4(this.x*A,this.y*A,this.z*A,this.w*A)}scaleInPlace(A){this.set(this.x*A,this.y*A,this.z*A,this.w*A)}length(){const A=this.x,g=this.y,I=this.z,C=this.w;return Math.sqrt(A*A+g*g+I*I+C*C)}lengthSquared(){const A=this.x,g=this.y,I=this.z,C=this.w;return A*A+g*g+I*I+C*C}normalize(){const A=this.x,g=this.y,I=this.z,C=this.w;let e=A*A+g*g+I*I+C*C;return e<Number.EPSILON?new Vec4:(e=1/Math.sqrt(e),new Vec4(A*e,g*e,I*e))}normalizeInPlace(){const A=this.x,g=this.y,I=this.z,C=this.w;let e=A*A+g*g+I*I+C*C;e<Number.EPSILON||(e=1/Math.sqrt(e),this.set(A*e,g*e,I*e,C*e))}dot(A){return this.x*A.x+this.y*A.y+this.z*A.z+this.w*A.w}cross(A){const g=this.x,I=this.y,C=this.z,e=this.w,i=A.x,t=A.y,o=A.z,a=A.w;return new Vec4(I*o-C*t,C*a-e*o,e*i-g*a,g*t-I*i)}angleTo(A){const g=this.normalize(),I=A.normalize(),C=g.dot(I);return C>1?0:Math.acos(C)}lerp(A,g){const I=this.x,C=this.y,e=this.z,i=this.w;return new Vec4(I+g*(A.x-I),C+g*(A.y-C),e+g*(A.z-e),i+g*(A.w-i))}clone(){return new Vec4(this.x,this.y,this.z,this.w)}toVec3(){return new Vec3(this.x,this.y,this.z)}asArray(){return[this.x,this.y,this.z,this.w]}fromArray(A){this.x=A[0],this.y=A[1],this.z=A[1],this.w=A[1]}toString(){return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON())}toJSON(){return{x:this.x,y:this.y,z:this.z,w:this.w}}fromJSON(A){this.x=A.x,this.y=A.y,this.z=A.z,this.w=A.w}readBinary(A){this.x=A.loadFloat32(),this.y=A.loadFloat32(),this.z=A.loadFloat32(),this.w=A.loadFloat32()}isValid(){for(const A of this.asArray())if(A==1/0||isNaN(A))return!1;return!0}}class RGBA{r=0;g=0;b=0;a=255;constructor(A=0,g=0,I=0,C=255){"string"==typeof A?A.startsWith("#")?this.setFromHex(A):this.setFromCSSColorName(A):(this.r=A,this.g=g,this.b=I,this.a=C)}set(A,g,I,C=255){this.r=A,this.g=g,this.b=I,this.a=C}setFromOther(A){this.r=A.r,this.g=A.g,this.b=A.b,this.a=A.a}setFromArray(A){this.r=A[0],this.g=A[1],this.b=A[2],this.a=4==A.length?A[3]:1}setFromHex(A){const g=function(A){const g=/^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(A);return g?{r:parseInt(g[1],16),g:parseInt(g[2],16),b:parseInt(g[3],16)}:null}(A);g?this.set(g.r,g.g,g.b):console.warn("Invalid hex code:"+A)}setFromCSSColorName(A){if(A.startsWith("#"))this.setFromHex(A);else{const g={aliceblue:"#f0f8ff",antiquewhite:"#faebd7",aqua:"#00ffff",aquamarine:"#7fffd4",azure:"#f0ffff",beige:"#f5f5dc",bisque:"#ffe4c4",black:"#000000",blanchedalmond:"#ffebcd",blue:"#0000ff",blueviolet:"#8a2be2",brown:"#a52a2a",burlywood:"#deb887",cadetblue:"#5f9ea0",chartreuse:"#7fff00",chocolate:"#d2691e",coral:"#ff7f50",cornflowerblue:"#6495ed",cornsilk:"#fff8dc",crimson:"#dc143c",cyan:"#00ffff",darkblue:"#00008b",darkcyan:"#008b8b",darkgoldenrod:"#b8860b",darkgray:"#a9a9a9",darkgreen:"#006400",darkkhaki:"#bdb76b",darkmagenta:"#8b008b",darkolivegreen:"#556b2f",darkorange:"#ff8c00",darkorchid:"#9932cc",darkred:"#8b0000",darksalmon:"#e9967a",darkseagreen:"#8fbc8f",darkslateblue:"#483d8b",darkslategray:"#2f4f4f",darkturquoise:"#00ced1",darkviolet:"#9400d3",deeppink:"#ff1493",deepskyblue:"#00bfff",dimgray:"#696969",dodgerblue:"#1e90ff",firebrick:"#b22222",floralwhite:"#fffaf0",forestgreen:"#228b22",fuchsia:"#ff00ff",gainsboro:"#dcdcdc",ghostwhite:"#f8f8ff",gold:"#ffd700",goldenrod:"#daa520",gray:"#808080",green:"#008000",greenyellow:"#adff2f",honeydew:"#f0fff0",hotpink:"#ff69b4","indianred ":"#cd5c5c",indigo:"#4b0082",ivory:"#fffff0",khaki:"#f0e68c",lavender:"#e6e6fa",lavenderblush:"#fff0f5",lawngreen:"#7cfc00",lemonchiffon:"#fffacd",lightblue:"#add8e6",lightcoral:"#f08080",lightcyan:"#e0ffff",lightgoldenrodyellow:"#fafad2",lightgrey:"#d3d3d3",lightgreen:"#90ee90",lightpink:"#ffb6c1",lightsalmon:"#ffa07a",lightseagreen:"#20b2aa",lightskyblue:"#87cefa",lightslategray:"#778899",lightsteelblue:"#b0c4de",lightyellow:"#ffffe0",lime:"#00ff00",limegreen:"#32cd32",linen:"#faf0e6",magenta:"#ff00ff",maroon:"#800000",mediumaquamarine:"#66cdaa",mediumblue:"#0000cd",mediumorchid:"#ba55d3",mediumpurple:"#9370d8",mediumseagreen:"#3cb371",mediumslateblue:"#7b68ee",mediumspringgreen:"#00fa9a",mediumturquoise:"#48d1cc",mediumvioletred:"#c71585",midnightblue:"#191970",mintcream:"#f5fffa",mistyrose:"#ffe4e1",moccasin:"#ffe4b5",navajowhite:"#ffdead",navy:"#000080",oldlace:"#fdf5e6",olive:"#808000",olivedrab:"#6b8e23",orange:"#ffa500",orangered:"#ff4500",orchid:"#da70d6",palegoldenrod:"#eee8aa",palegreen:"#98fb98",paleturquoise:"#afeeee",palevioletred:"#d87093",papayawhip:"#ffefd5",peachpuff:"#ffdab9",peru:"#cd853f",pink:"#ffc0cb",plum:"#dda0dd",powderblue:"#b0e0e6",purple:"#800080",rebeccapurple:"#663399",red:"#ff0000",rosybrown:"#bc8f8f",royalblue:"#4169e1",saddlebrown:"#8b4513",salmon:"#fa8072",sandybrown:"#f4a460",seagreen:"#2e8b57",seashell:"#fff5ee",sienna:"#a0522d",silver:"#c0c0c0",skyblue:"#87ceeb",slateblue:"#6a5acd",slategray:"#708090",snow:"#fffafa",springgreen:"#00ff7f",steelblue:"#4682b4",tan:"#d2b48c",teal:"#008080",thistle:"#d8bfd8",tomato:"#ff6347",turquoise:"#40e0d0",violet:"#ee82ee",wheat:"#f5deb3",white:"#ffffff",whitesmoke:"#f5f5f5",yellow:"#ffff00",yellowgreen:"#9acd32"}[A.toLowerCase()];g&&this.setFromHex(g)}}toHex(){function A(A){const g=A.toString(16);return 1==g.length?"0"+g:g}return"#"+A(this.r)+A(this.g)+A(this.b)}equal(A){return this.r==A.r&&this.g==A.g&&this.b==A.b&&this.a==A.a}notEquals(A){return this.r!=A.r&&this.g!=A.g&&this.b!=A.b&&this.a!=A.a}approxEqual(A,g=Number.EPSILON){return Math.abs(this.r-A.r)<g&&Math.abs(this.g-A.g)<g&&Math.abs(this.b-A.b)<g&&Math.abs(this.a-A.a)<g}add(A){return new RGBA(this.r+A.r,this.g+A.g,this.b+A.b,this.a+A.a)}subtract(A){return new RGBA(this.r-A.r,this.g-A.g,this.b-A.b,this.a-A.a)}scale(A){return new RGBA(this.r*A,this.g*A,this.b*A,this.a*A)}scaleInPlace(A){this.r*=A,this.g*=A,this.b*=A,this.a*=A}applyGamma(A){this.set(Math.pow(this.r,A),Math.pow(this.g,A),Math.pow(this.b,A),this.a)}toLinear(A=2.2){return new RGBA(Math.pow(this.r,A),Math.pow(this.g,A),Math.pow(this.b,A),this.a)}toGamma(A=2.2){return new RGBA(Math.pow(this.r,1/A),Math.pow(this.g,1/A),Math.pow(this.b,1/A),this.a)}luminance(){return.2126*this.r+.7152*this.g+.0722*this.b}lerp(A,g){const I=this.r,C=this.g,e=this.b,i=this.a;return new RGBA(I+g*(A.r-I),C+g*(A.g-C),e+g*(A.b-e),i+g*(A.a-i))}static random(A=0,g=!1){return A>0?new RGBA(A+Math.random()*(1-A),A+Math.random()*(1-A),A+Math.random()*(1-A),g?A+Math.random()*(1-A):1):A<0?new RGBA(Math.random()*(1+A),Math.random()*(1+A),Math.random()*(1+A),g?Math.random()*(1+A):1):new RGBA(Math.random(),Math.random(),Math.random(),g?Math.random():1)}clone(){return new RGBA(this.r,this.g,this.b,this.a)}asArray(){return[this.r,this.g,this.b,this.a]}toJSON(){return{r:this.r,g:this.g,b:this.b,a:this.a}}fromJSON(A){this.r=A.r,this.g=A.g,this.b=A.b,this.a=A.a}toCSSString(){return"rgba("+Math.round(255*this.r)+", "+Math.round(255*this.g)+", "+Math.round(255*this.b)+", "+this.a+")"}}let counter=0;class BaseClass{__id;constructor(){this.__id=++counter}getId(){return this.__id}getClassName(){return Registry.getClassName(Object.getPrototypeOf(this).constructor)}}class BaseEvent{constructor(){}}class EventEmitter extends BaseClass{listeners={};constructor(){super()}on(A,g){if(!g)throw new Error("Missing listener.");this.listeners[A]||(this.listeners[A]=[]);const I=this.listeners[A];if(I.includes(g))throw new Error(`Listener "${g.name}" already connected to event "${A}".`);const C=I.length;return I[C]=g,C}once(A,g){const I=C=>{this.off(A,I),g(C)};return this.on(A,I)}off(A,g){if(null==g)throw new Error("Missing callback function (listener).");const I=this.listeners[A]||[];if("number"==typeof g){return void(I[g]=null)}const C=g;I.forEach(((A,g)=>{A===C&&(I[g]=null)}))}removeListenerById(A,g){this.off(A,g)}emit(A,g=new BaseEvent){(this.listeners[A]||[]).forEach((A=>{if(A)try{A(g)}catch(A){console.warn(A)}}))}}class ResizedEvent extends BaseEvent{width;height;constructor(A,g){super(),this.width=A,this.height=g}}class GrowingPacker extends EventEmitter{root;freeNodes=[];constructor(A=0,g=0){super(),this.root={x:0,y:0,w:A,h:g,a:A*g},this.freeNodes.push(this.root)}fit(A){if(0==A.length)return;let g=!1;if(this.root.w<A[0].w&&(this.root.w=A[0].w,g=!0),this.root.h<A[0].h&&(this.root.h=A[0].h,g=!0),g){const A=new ResizedEvent(this.root.w,this.root.h);this.emit("resized",A)}A.forEach((A=>{A.fit=this.__addBlock(A)}))}__addBlock(A){const g=this.findNode(A.w,A.h);return g?this.splitNode(g,A.w,A.h):this.growNode(A.w,A.h)}addBlock(A){let g=!1;if(this.root.w<A.w&&(this.root.w=A.w,g=!0),this.root.h<A.h&&(this.root.h=A.h,g=!0),g){const A=new ResizedEvent(this.root.w,this.root.h);this.emit("resized",A)}const I=this.findNode(A.w,A.h);return I?this.splitNode(I,A.w,A.h):this.growNode(A.w,A.h)}findNode(A,g){const I=this.freeNodes.findIndex((I=>A<=I.w&&g<=I.h));return I>=0?this.freeNodes.splice(I,1)[0]:null}splitNode(A,g,I){return A.used=!0,A.w-g<A.h-I?(A.h-I>0&&(A.down={x:A.x,y:A.y+I,w:A.w,h:A.h-I,a:A.w*(A.h-I)},this.freeNodes.push(A.down)),A.w-g>0&&(A.right={x:A.x+g,y:A.y,w:A.w-g,h:I,a:(A.w-g)*I},this.freeNodes.push(A.right))):(A.w-g>0&&(A.right={x:A.x+g,y:A.y,w:A.w-g,h:A.h,a:(A.w-g)*A.h},this.freeNodes.push(A.right)),A.h-I>0&&(A.down={x:A.x,y:A.y+I,w:g,h:A.h-I,a:g*(A.h-I)},this.freeNodes.push(A.down))),this.freeNodes.sort(((A,g)=>A.a-g.a)),A}growNode(A,g){const I=A<=this.root.w,C=g<=this.root.h,e=C&&this.root.h>=this.root.w+A,i=I&&this.root.w>=this.root.h+g;return e?this.growRight(A,g):i?this.growDown(A,g):C?this.growRight(A,g):I?this.growDown(A,g):null}growRight(A,g){this.root={used:!0,x:0,y:0,w:this.root.w+A,h:this.root.h,a:(this.root.w+A)*this.root.h,down:this.root,right:{x:this.root.w,y:0,w:A,h:this.root.h,a:A*this.root.h}},this.freeNodes.push(this.root.right),this.freeNodes.sort(((A,g)=>A.a-g.a));const I=this.findNode(A,g);let C;I&&(C=this.splitNode(I,A,g));const e=new ResizedEvent(this.root.w,this.root.h);return this.emit("resized",e),C}growDown(A,g){this.root={used:!0,x:0,y:0,w:this.root.w,h:this.root.h+g,a:this.root.w*(this.root.h+g),down:{x:0,y:this.root.h,w:this.root.w,h:g,a:this.root.w*g},right:this.root},this.freeNodes.push(this.root.down),this.freeNodes.sort(((A,g)=>A.a-g.a));const I=this.findNode(A,g);let C;I&&(C=this.splitNode(I,A,g));const e=new ResizedEvent(this.root.w,this.root.h);return this.emit("resized",e),C}}const UInt8=0,SInt8=1,UInt16=2,SInt16=3,UInt32=4,SInt32=5,Float32=6;class MathFunctions{static radToDeg(A){return A/(Math.PI/180)}static degToRad(A){return A*(Math.PI/180)}static isNumeric(A){return!isNaN(parseFloat(A))&&isFinite(A)}static randomInt(A,g){return A=Math.ceil(A),g=Math.floor(g),Math.floor(Math.random()*(g-A))+A}static lerp(A,g,I){return A+I*(g-A)}static clamp(A,g,I){return Math.min(Math.max(A,g),I)}static nearestPow2(A){return Math.pow(2,Math.round(Math.log(A)/Math.log(2)))}static nearestPow10(A){return Math.pow(10,Math.round(Math.log10(A)/Math.log10(10)))}static nextPow2(A){if(0==this.fract(Math.log2(A)))return A;let g=0;for(;A>0;)g++,A>>=1;return 1<<g}static fract(A){return 0==A?0:A<0?A>-1?-A:-A%Math.floor(-A):A<1?A:A%Math.floor(A)}static remap(A,g,I,C,e){return C+(A-g)/(I-g)*(e-C)}static smoothStep(A,g,I){const C=this.clamp((I-A)/(g-A),0,1);return C*C*(3-2*C)}static linStep(A,g,I){return this.clamp((I-A)/(g-A),0,1)}static decode16BitFloatFrom2xUInt8(A){const g=A[0],I=(120&g)>>3;let C=0==I?0:2048;const e=C+((7&g)<<8)+A[1];C=0==I?1:0;return(128&g?1:-1)*e*Math.pow(2,I+C-16)}static encode16BitFloatInto2xUInt8(A){const g=new Uint8Array(2),I=A>=0?128:0;A=Math.abs(A);let C,e=15,i=1024;for(let g=15;g>0;g--)A<i&&(i/=2,e--);C=0==e?A/i/2:(A-i)/i;const t=Math.round(2048*C),o=t/256,a=t-256*o;return g[0]=I+8*e+o,g[1]=a,A>=2048&&(g[0]=255),g}static encode16BitFloat(A){const g=new Float32Array(1);g[0]=A;return(A=>{let g=A>>16&32768,I=A>>12&2047;const C=A>>23&255;return C<103?g:C>142?(g|=31744,g|=(255==C?0:1)&&8388607&A,g):C<113?(I|=2048,g|=(I>>114-C)+(I>>113-C&1),g):(g|=C-112<<10|I>>1,g+=1&I,g)})(new Int32Array(g.buffer)[0])}static decode16BitFloat(A){const g=(32768&A)>>15,I=(31744&A)>>10,C=1023&A;return 0==I?(g?-1:1)*Math.pow(2,-14)*(C/Math.pow(2,10)):31==I?C?NaN:1/0*(g?-1:1):(g?-1:1)*Math.pow(2,I-15)*(1+C/Math.pow(2,10))}static convertFloat32ArrayToUInt16Array(A){const g=new Uint16Array(A.length),I=new Int32Array(A.buffer),C=A=>{let g=A>>16&32768,I=A>>12&2047;const C=A>>23&255;return C<103?g:C>142?(g|=31744,g|=(255==C?0:1)&&8388607&A,g):C<113?(I|=2048,g|=(I>>114-C)+(I>>113-C&1),g):(g|=C-112<<10|I>>1,g+=1&I,g)};for(let e=0;e<A.length;e++)g[e]=C(I[e]);return g}}class Allocation1D{start;size;constructor(A=0,g=0){this.start=A,this.size=g}}class Allocator1D extends EventEmitter{freeList=[];allocations=[];allocationsMap={};allocatedSpace=0;reservedSpace=0;freeSpace=0;constructor(){super()}getAllocation(A){return this.allocations[this.allocationsMap[A]]}allocate(A,g){if(null!=this.allocationsMap[A]){const I=this.allocationsMap[A],C=this.allocations[I];if(g==C.size)return C;if(g<C.size){const A=C.size-g;return this.addBlock(I+1,new Allocation1D(C.start+g,A)),this.freeBlock(I+1),C.size=g,C}{const e=I+1;if(this.freeList.includes(e)&&C.size+this.allocations[e].size>=g){const A=this.allocations[e];if(C.size+A.size==g)return C.size+=A.size,this.freeSpace-=A.size,this.freeList.splice(this.freeList.indexOf(e),1),this.removeBlock(e),C;{const I=g-C.size;return C.size+=I,this.freeSpace-=I,A.start+=I,A.size-=I,C}}delete this.allocationsMap[A],C.start+C.size==this.allocatedSpace?(this.removeBlock(I),this.allocatedSpace-=C.size):this.freeBlock(I)}}let I=-1;for(let A=0;A<this.freeList.length;A++){const C=this.freeList[A],e=this.allocations[C];if(e.size==g){I=C;break}e.size>g&&(I=C)}if(-1!=I){const C=this.allocations[I];if(this.freeSpace-=C.size,this.freeList.splice(this.freeList.indexOf(I),1),C.size>g){const A=C.size-g;this.addBlock(I+1,new Allocation1D(C.start+g,A)),this.freeBlock(I+1),this.allocations[I].size=g}this.allocationsMap[A]=I}else{const I=this.allocatedSpace,C=this.allocations.length;this.allocatedSpace+=g;const e=MathFunctions.nextPow2(this.allocatedSpace);e>this.reservedSpace&&(this.reservedSpace=e,this.emit("resized",{reservedSpace:this.reservedSpace})),this.allocations.push(new Allocation1D(I,g)),this.allocationsMap[A]=C}return this.allocations[this.allocationsMap[A]]}addBlock(A,g){this.allocations.splice(A,0,g);for(const g in this.allocationsMap)this.allocationsMap[g]>=A&&this.allocationsMap[g]++;for(let g=0;g<this.freeList.length;g++)this.freeList[g]>=A&&this.freeList[g]++}removeBlock(A){this.allocations.splice(A,1);for(const g in this.allocationsMap)this.allocationsMap[g]>A&&this.allocationsMap[g]--;for(let g=0;g<this.freeList.length;g++)this.freeList[g]>A&&this.freeList[g]--}freeBlock(A){const g=this.allocations[A];this.freeSpace+=g.size;const I=A-1;if(this.freeList.includes(I)){return this.allocations[I].size+=g.size,void this.removeBlock(A)}const C=A+1;if(this.freeList.includes(C)){const I=this.allocations[C];return I.start-=g.size,I.size+=g.size,void this.removeBlock(A)}this.freeList.push(A),this.freeList.sort(((A,g)=>this.allocations[A].size-this.allocations[g].size))}deallocate(A){const g=this.allocationsMap[A];if(null==g)throw new Error(`allocation ${A} does not exist.`);this.freeBlock(g),delete this.allocationsMap[A]}getFragmentation(){return this.freeSpace/this.allocatedSpace}defragment(){}verifyConsistency(){if(Object.keys(this.allocationsMap).length+this.freeList.length!=this.allocations.length)throw new Error("number of blocks does not match the number of allocations");for(const A in this.allocationsMap){const g=this.allocationsMap[A];if(this.freeList.includes(g))throw new Error("block of used memory is also on the free list")}let A=0;for(let g=0;g<this.allocations.length;g++){const I=this.allocations[g];if(I.start!=A)throw"blocks of memory are not sequential";A+=I.size}if(A!=this.allocatedSpace)throw`allocated size: ${this.allocatedSpace}  does not match allocated blocks: ${A}`;if(this.reservedSpace<this.allocatedSpace)throw`reserved space: ${this.reservedSpace} is less than allocated space: ${this.allocatedSpace}`}}class IntersectionData{screenPos;pointerRay;intersectionPos;geomData;geomItem;componentId;componentIds=[];dist;constructor(A,g,I,C,e){this.screenPos=A,this.pointerRay=g,this.intersectionPos=I,this.geomData=C,this.geomItem=e.geomItem,this.componentId=e.componentId,this.dist=e.dist}}class ChildAddedEvent extends BaseEvent{index;childItem;constructor(A,g){super(),this.index=A,this.childItem=g}}class ControllerAddedEvent extends BaseEvent{controller;constructor(A){super(),this.controller=A}}class CountChangedEvent extends BaseEvent{change;count;constructor(A,g){super(),this.change=A,this.count=g}}class EnvMapAssignedEvent extends BaseEvent{envMap;constructor(A){super(),this.envMap=A}}class IndexEvent extends BaseEvent{index;constructor(A){super(),this.index=A}}class ZeaUIEvent extends BaseEvent{viewport;propagating=!0;constructor(){super()}}class ZeaKeyboardEvent extends ZeaUIEvent{sourceEvent;propagating=!0;altKey;code;ctrlKey;isComposing;key;location;metaKey;repeat;shiftKey;which;constructor(A){super(),A&&(this.sourceEvent=A,this.altKey=A.altKey,this.code=A.code,this.ctrlKey=A.ctrlKey,this.isComposing=A.isComposing,this.key=A.key,this.location=A.location,this.metaKey=A.metaKey,this.repeat=A.repeat,this.shiftKey=A.shiftKey)}stopPropagation(){this.propagating=!1,this.sourceEvent&&this.sourceEvent.stopPropagation()}preventDefault(){this.sourceEvent&&this.sourceEvent.preventDefault()}}class KeyboardEvent extends ZeaKeyboardEvent{}class NameChangedEvent extends BaseEvent{oldName;newName;constructor(A,g){super(),this.oldName=A,this.newName=g}}class OpacityStateChangedEvent extends BaseEvent{isOpaque;isOpaqueStateChanged;constructor(A,g){super(),this.isOpaque=A,this.isOpaqueStateChanged=g}}class ParameterAddedEvent extends BaseEvent{name;constructor(A){super(),this.name=A}}class ParameterRemovedEvent extends BaseEvent{name;constructor(A){super(),this.name=A}}class RangeLoadedEvent extends BaseEvent{range;constructor(A){super(),this.range=A}}class SceneSetEvent extends BaseEvent{scene;constructor(A){super(),this.scene=A}}class SelectabilityChangedEvent extends BaseEvent{value;constructor(A){super(),this.value=A}}class SelectedEvent extends BaseEvent{selected;constructor(A){super(),this.selected=A}}class ShaderNameChangedEvent extends BaseEvent{shaderName;constructor(A){super(),this.shaderName=A}}class StateChangedEvent extends BaseEvent{state;constructor(A){super(),this.state=A}}class StreamFileParsedEvent extends BaseEvent{geomFileID;geomCount;constructor(A,g){super(),this.geomFileID=A,this.geomCount=g}}class TexturedChangedEvent extends BaseEvent{isTextured;param;constructor(A,g){super(),this.isTextured=A,this.param=g}}class ViewChangedEvent extends BaseEvent{interfaceType;viewXfo;viewport;constructor(A,g){super(),this.interfaceType=A,this.viewXfo=g}}class ProgressEvent extends BaseEvent{percent;constructor(A){super(),this.percent=A}}const POINTER_TYPES={mouse:"mouse",touch:"touch",xr:"xr"};let capturedItem$1=null;class ZeaPointerEvent extends ZeaUIEvent{pointerType;pointerRay;pointerPos;detail;intersectionData;leftGeometry;constructor(A){super(),this.pointerType=A}stopPropagation(){this.propagating=!1}setCapture(A){capturedItem$1=A}getCapture(){return capturedItem$1}releaseCapture(){capturedItem$1=null}}const captureItems=[];class XRControllerEvent extends ZeaPointerEvent{controller;button;buttonPressed=0;constructor(A,g,I,C){super(POINTER_TYPES.xr),this.viewport=A,this.controller=g,this.button=I,this.buttonPressed=C}stopPropagation(){this.propagating=!1}setCapture(A){captureItems[this.controller.id]=A}getCapture(){return captureItems[this.controller.id]}releaseCapture(){captureItems[this.controller.id]=null}}let capturedItem=null;class XRPointerEvent extends ZeaPointerEvent{xfo;xrSelectEvent;hitTestResults;constructor(A,g,I,C){super(POINTER_TYPES.xr),this.viewport=A,this.xfo=g,this.xrSelectEvent=I,this.hitTestResults=C}stopPropagation(){this.propagating=!1}setCapture(A){capturedItem=A}getCapture(){return capturedItem}releaseCapture(){capturedItem=null}}class XRPoseEvent extends ZeaPointerEvent{viewXfo;controllers=[];constructor(A,g,I=[]){super(POINTER_TYPES.xr),this.viewport=A,this.viewXfo=g,I.forEach((A=>{this.controllers.push(A)}))}}class XRViewChangedEvent extends ViewChangedEvent{hmd="";controllers=[];xrviewport;constructor(A){super("VR",A)}}class XrViewportEvent extends BaseEvent{xrViewport;constructor(A){super(),this.xrViewport=A}}class ZeaMouseEvent extends ZeaPointerEvent{button;clientX;clientY;rendererX;rendererY;altKey;metaKey;ctrlKey;shiftKey;sourceEvent;constructor(A,g){super(POINTER_TYPES.mouse),this.sourceEvent=A,this.button=A.button,this.clientX=A.clientX,this.clientY=A.clientY,this.rendererX=this.clientX-g.left,this.rendererY=this.clientY-g.top,this.altKey=A.altKey,this.metaKey=A.metaKey,this.ctrlKey=A.ctrlKey,this.shiftKey=A.shiftKey}stopPropagation(){super.stopPropagation(),this.sourceEvent&&this.sourceEvent.stopPropagation()}preventDefault(){this.sourceEvent&&this.sourceEvent.preventDefault()}}class Ray{dir;start;constructor(A,g){this.start=A instanceof Vec3?A:new Vec3,this.dir=g instanceof Vec3?g:new Vec3}closestPoint(A){const g=A.subtract(this.start).dot(this.dir);if(g<Number.EPSILON)return 0;const I=this.dir.dot(this.dir);return I<Number.EPSILON?0:g/I}closestPointOnLineSegment(A,g){const I=this.dir,C=g.subtract(A),e=C.length();C.normalizeInPlace();const i=this.start.subtract(A),t=I.dot(I),o=I.dot(C),a=C.dot(C),B=I.dot(i),s=C.dot(i);if(0==t&&0==a)return[this.start.distanceTo(A),0];if(0==t)return[0,0];if(0==a)return[this.closestPoint(A),0];const l=t*a-o*o;let n,d;return l<.001?(n=0,d=o>a?B/o:s/a):(n=(o*s-a*B)/l,d=(t*s-o*B)/l),[n,MathFunctions.clamp(d/e,0,1)]}pointAtDist(A){return this.start.add(this.dir.scale(A))}intersectRayVector(A){const g=this.dir,I=A.dir,C=this.start.subtract(A.start),e=g.dot(g),i=g.dot(I),t=I.dot(I),o=g.dot(C),a=I.dot(C);if(0==e&&0==t)return;if(0==e){return[0,A.closestPoint(this.start)]}if(0==t){return[this.closestPoint(A.start),0]}const B=e*t-i*i;let s,l;return B<.001?(s=0,l=i>t?o/i:a/t):(s=(i*a-t*o)/B,l=(e*a-i*o)/B),[s,l]}intersectRayPlane(A){const g=this.start.subtract(A.start),I=A.dir.dot(this.dir),C=-A.dir.dot(g);if(Math.abs(I)<Number.PRECISION)return-1;const e=C/I;return e<-Number.PRECISION?-1:e}intersectRayBox3(A,g=0){const I=new Vec3(1/this.dir.x,1/this.dir.y,1/this.dir.z),C=[];C[0]=I.x<0?1:0,C[1]=I.y<0?1:0,C[2]=I.z<0?1:0;const e=[];if(g>0){const I=A.diagonal();I.normalizeInPlace(),I.scaleInPlace(g),e[0]=A.p0.subtract(I),e[1]=A.p1.add(I)}else e[0]=A.p0,e[1]=A.p1;let i=(e[C[0]].x-this.start.x)*I.x,t=(e[1-C[0]].x-this.start.x)*I.x;const o=(e[C[1]].y-this.start.y)*I.y,a=(e[1-C[1]].y-this.start.y)*I.y;if(i>a||o>t)return!1;o>i&&(i=o),a<t&&(t=a);const B=(e[C[2]].z-this.start.z)*I.z,s=(e[1-C[2]].z-this.start.z)*I.z;return!(i>s||B>t)&&(B>i&&(i=B),s<t&&(t=s),!0)}clone(){return new Ray(this.start.clone(),this.dir.clone())}toJSON(){return{start:this.start.toJSON(),dir:this.dir.toJSON()}}fromJSON(A){this.start.fromJSON(A.start),this.dir.fromJSON(A.dir)}toString(){return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON())}}class Touch{identifier;clientX=0;clientY=0;screenX=0;screenY=0;pageX=0;pageY=0;radiusX=0;radiusY=0;rotationAngle=0;force=0;altitudeAngle=0;azimuthAngle=0;touchType="direct";rendererX;rendererY;touchPos;touchRay;constructor(A,g){this.identifier=A.identifier,this.clientX=A.clientX,this.clientY=A.clientY,this.screenX=A.screenX,this.screenY=A.screenY,this.pageX=A.pageX,this.pageY=A.pageY,this.radiusX=A.radiusX,this.radiusY=A.radiusY,this.rotationAngle=A.rotationAngle,this.force=A.force,this.rendererX=this.clientX-g.left,this.rendererY=this.clientY-g.top,this.touchPos=new Vec2(this.rendererX,this.rendererY),this.touchRay=new Ray}}class ZeaTouchEvent extends ZeaPointerEvent{touches=[];changedTouches=[];targetTouches=[];altKey=!1;metaKey=!1;ctrlKey=!1;shiftKey=!1;sourceEvent;constructor(A,g){super(POINTER_TYPES.touch),this.sourceEvent=A,this.sourceEvent.stopPropagation(),this.altKey=A.altKey,this.metaKey=A.metaKey,this.ctrlKey=A.ctrlKey,this.shiftKey=A.shiftKey;for(let I=0;I<A.touches.length;I++)this.touches.push(new Touch(A.touches[I],g));if(A.changedTouches)for(let I=0;I<A.changedTouches.length;I++)this.changedTouches.push(new Touch(A.changedTouches[I],g));if(A.targetTouches)for(let I=0;I<A.targetTouches.length;I++)this.targetTouches.push(new Touch(A.targetTouches[I],g))}stopPropagation(){super.stopPropagation(),this.sourceEvent&&this.sourceEvent.stopPropagation()}preventDefault(){}}class ZeaWheelEvent extends ZeaMouseEvent{wheelDelta;deltaMode;deltaX;deltaY;deltaZ;constructor(A,g){super(A,g),this.wheelDelta=A.wheelDelta,this.deltaMode=A.deltaMode,this.deltaX=A.deltaX,this.deltaY=A.deltaY,this.deltaZ=A.deltaZ}}class Color{r=0;g=0;b=0;a=255;constructor(A=0,g=0,I=0,C=1){"string"==typeof A?A.startsWith("#")?this.setFromHex(A):this.setFromCSSColorName(A):(this.r=A,this.g=g,this.b=I,this.a=C)}set(A,g,I,C=1){this.r=A,this.g=g,this.b=I,this.a=C}setFromOther(A){this.r=A.r,this.g=A.g,this.b=A.b,this.a=A.a}getAsRGBArray(){return[255*this.r,255*this.g,255*this.b]}getAsRGBDict(){return{r:255*this.r,g:255*this.g,b:255*this.b}}setFromRGB(A,g,I,C){this.r=A/255,this.g=g/255,this.b=I/255,this.a=C?C/255:1}setFromRGBDict(A){this.r=A.r/255,this.g=A.g/255,this.b=A.b/255,this.a=4==A.a?A.a/255:1}setFromHex(A){const g=function(A){const g=/^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(A);return g?{r:parseInt(g[1],16),g:parseInt(g[2],16),b:parseInt(g[3],16)}:null}(A);g?this.setFromRGB(g.r,g.g,g.b):console.warn("Invalid hex code:"+A)}setFromCSSColorName(A){if(A.startsWith("#"))this.setFromHex(A);else{const g={aliceblue:"#f0f8ff",antiquewhite:"#faebd7",aqua:"#00ffff",aquamarine:"#7fffd4",azure:"#f0ffff",beige:"#f5f5dc",bisque:"#ffe4c4",black:"#000000",blanchedalmond:"#ffebcd",blue:"#0000ff",blueviolet:"#8a2be2",brown:"#a52a2a",burlywood:"#deb887",cadetblue:"#5f9ea0",chartreuse:"#7fff00",chocolate:"#d2691e",coral:"#ff7f50",cornflowerblue:"#6495ed",cornsilk:"#fff8dc",crimson:"#dc143c",cyan:"#00ffff",darkblue:"#00008b",darkcyan:"#008b8b",darkgoldenrod:"#b8860b",darkgray:"#a9a9a9",darkgreen:"#006400",darkkhaki:"#bdb76b",darkmagenta:"#8b008b",darkolivegreen:"#556b2f",darkorange:"#ff8c00",darkorchid:"#9932cc",darkred:"#8b0000",darksalmon:"#e9967a",darkseagreen:"#8fbc8f",darkslateblue:"#483d8b",darkslategray:"#2f4f4f",darkturquoise:"#00ced1",darkviolet:"#9400d3",deeppink:"#ff1493",deepskyblue:"#00bfff",dimgray:"#696969",dodgerblue:"#1e90ff",firebrick:"#b22222",floralwhite:"#fffaf0",forestgreen:"#228b22",fuchsia:"#ff00ff",gainsboro:"#dcdcdc",ghostwhite:"#f8f8ff",gold:"#ffd700",goldenrod:"#daa520",gray:"#808080",green:"#008000",greenyellow:"#adff2f",honeydew:"#f0fff0",hotpink:"#ff69b4","indianred ":"#cd5c5c",indigo:"#4b0082",ivory:"#fffff0",khaki:"#f0e68c",lavender:"#e6e6fa",lavenderblush:"#fff0f5",lawngreen:"#7cfc00",lemonchiffon:"#fffacd",lightblue:"#add8e6",lightcoral:"#f08080",lightcyan:"#e0ffff",lightgoldenrodyellow:"#fafad2",lightgrey:"#d3d3d3",lightgreen:"#90ee90",lightpink:"#ffb6c1",lightsalmon:"#ffa07a",lightseagreen:"#20b2aa",lightskyblue:"#87cefa",lightslategray:"#778899",lightsteelblue:"#b0c4de",lightyellow:"#ffffe0",lime:"#00ff00",limegreen:"#32cd32",linen:"#faf0e6",magenta:"#ff00ff",maroon:"#800000",mediumaquamarine:"#66cdaa",mediumblue:"#0000cd",mediumorchid:"#ba55d3",mediumpurple:"#9370d8",mediumseagreen:"#3cb371",mediumslateblue:"#7b68ee",mediumspringgreen:"#00fa9a",mediumturquoise:"#48d1cc",mediumvioletred:"#c71585",midnightblue:"#191970",mintcream:"#f5fffa",mistyrose:"#ffe4e1",moccasin:"#ffe4b5",navajowhite:"#ffdead",navy:"#000080",oldlace:"#fdf5e6",olive:"#808000",olivedrab:"#6b8e23",orange:"#ffa500",orangered:"#ff4500",orchid:"#da70d6",palegoldenrod:"#eee8aa",palegreen:"#98fb98",paleturquoise:"#afeeee",palevioletred:"#d87093",papayawhip:"#ffefd5",peachpuff:"#ffdab9",peru:"#cd853f",pink:"#ffc0cb",plum:"#dda0dd",powderblue:"#b0e0e6",purple:"#800080",rebeccapurple:"#663399",red:"#ff0000",rosybrown:"#bc8f8f",royalblue:"#4169e1",saddlebrown:"#8b4513",salmon:"#fa8072",sandybrown:"#f4a460",seagreen:"#2e8b57",seashell:"#fff5ee",sienna:"#a0522d",silver:"#c0c0c0",skyblue:"#87ceeb",slateblue:"#6a5acd",slategray:"#708090",snow:"#fffafa",springgreen:"#00ff7f",steelblue:"#4682b4",tan:"#d2b48c",teal:"#008080",thistle:"#d8bfd8",tomato:"#ff6347",turquoise:"#40e0d0",violet:"#ee82ee",wheat:"#f5deb3",white:"#ffffff",whitesmoke:"#f5f5f5",yellow:"#ffff00",yellowgreen:"#9acd32"}[A.toLowerCase()];g&&this.setFromHex(g)}}toHex(){function A(A){const g=Math.round(255*A).toString(16);return 1==g.length?"0"+g:g}return"#"+A(this.r)+A(this.g)+A(this.b)}isEqual(A){return this.r==A.r&&this.g==A.g&&this.b==A.b&&this.a==A.a}notEquals(A){return this.r!=A.r&&this.g!=A.g&&this.b!=A.b&&this.a!=A.a}approxEqual(A,g=Number.EPSILON){return Math.abs(this.r-A.r)<g&&Math.abs(this.g-A.g)<g&&Math.abs(this.b-A.b)<g&&Math.abs(this.a-A.a)<g}add(A){return new Color(this.r+A.r,this.g+A.g,this.b+A.b,this.a+A.a)}addInPlace(A){this.r+=A.r,this.g+=A.g,this.b+=A.b,this.a+=A.a}subtract(A){return new Color(this.r-A.r,this.g-A.g,this.b-A.b,this.a-A.a)}scale(A){return new Color(this.r*A,this.g*A,this.b*A,this.a*A)}scaleInPlace(A){this.r*=A,this.g*=A,this.b*=A,this.a*=A}applyGamma(A){this.set(Math.pow(this.r,A),Math.pow(this.g,A),Math.pow(this.b,A),this.a)}toLinear(A=2.2){return new Color(Math.pow(this.r,A),Math.pow(this.g,A),Math.pow(this.b,A),this.a)}toGamma(A=2.2){return new Color(Math.pow(this.r,1/A),Math.pow(this.g,1/A),Math.pow(this.b,1/A),this.a)}luminance(){return.2126*this.r+.7152*this.g+.0722*this.b}lerp(A,g){const I=this.r,C=this.g,e=this.b,i=this.a;return new Color(I+g*(A.r-I),C+g*(A.g-C),e+g*(A.b-e),i+g*(A.a-i))}static random(A=0,g=!1){return A>0?new Color(A+Math.random()*(1-A),A+Math.random()*(1-A),A+Math.random()*(1-A),g?A+Math.random()*(1-A):1):A<0?new Color(Math.random()*(1+A),Math.random()*(1+A),Math.random()*(1+A),g?Math.random()*(1+A):1):new Color(Math.random(),Math.random(),Math.random(),g?Math.random():1)}clone(){return new Color(this.r,this.g,this.b,this.a)}asArray(){return[this.r,this.g,this.b,this.a]}fromArray(A){this.r=A[0]/255,this.g=A[1]/255,this.b=A[2]/255,this.a=4==A.length?A[3]/255:1}toJSON(){return{r:this.r,g:this.g,b:this.b,a:this.a}}fromJSON(A){this.r=A.r,this.g=A.g,this.b=A.b,this.a=A.a}readBinary(A){this.r=A.loadFloat32(),this.g=A.loadFloat32(),this.b=A.loadFloat32(),this.a=A.loadFloat32()}toCSSString(){return"rgba("+Math.round(255*this.r)+", "+Math.round(255*this.g)+", "+Math.round(255*this.b)+", "+this.a+")"}toString(){return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON())}}var EulerAnglesAxisOrder;!function(A){A[A.XYZ=0]="XYZ",A[A.YZX=1]="YZX",A[A.ZXY=2]="ZXY",A[A.XZY=3]="XZY",A[A.ZYX=4]="ZYX",A[A.YXZ=5]="YXZ"}(EulerAnglesAxisOrder||(EulerAnglesAxisOrder={}));class EulerAngles{x;y;z;order;constructor(A=0,g=0,I=0,C=0){if(this.x=A,this.y=g,this.z=I,"number"!=typeof C||isNaN(C))switch(C){case"XYZ":this.order=0;break;case"YZX":this.order=1;break;case"ZXY":this.order=2;break;case"XZY":this.order=3;break;case"ZYX":this.order=4;break;case"YXZ":this.order=5;break;default:throw new Error("Invalid Euler Angles Order:"+C)}else this.order=C}set(A,g,I){this.x=A,this.y=g,this.z=I}toString(){return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON())}toJSON(){return{x:this.x,y:this.y,z:this.z,order:this.order}}fromJSON(A){this.x=A.x,this.y=A.y,this.z=A.z,this.order=A.order}}class Mat3{m00=1;m01=0;m02=0;m10=0;m11=1;m12=0;m20=0;m21=0;m22=1;constructor(A=1,g=0,I=0,C=0,e=1,i=0,t=0,o=0,a=1){A instanceof Vec3&&g instanceof Vec3&&I instanceof Vec3?this.set(A.x,A.y,A.z,g.x,g.y,g.z,I.x,I.y,I.z):this.set(A,g,I,C,e,i,t,o,a)}get xAxis(){return new Vec3(this.m00,this.m01,this.m02)}set xAxis(A){this.xAxis.set(A.x,A.y,A.z)}get yAxis(){return new Vec3(this.m10,this.m11,this.m12)}set yAxis(A){this.yAxis.set(A.x,A.y,A.z)}get zAxis(){return new Vec3(this.m20,this.m21,this.m22)}set zAxis(A){this.zAxis.set(A.x,A.y,A.z)}set(A=1,g=0,I=0,C=0,e=1,i=0,t=0,o=0,a=1){this.m00=A,this.m01=g,this.m02=I,this.m10=C,this.m11=e,this.m12=i,this.m20=t,this.m21=o,this.m22=a}setIdentity(){this.set()}setFromMat(A){this.m00=A.m00,this.m01=A.m01,this.m02=A.m02,this.m10=A.m10,this.m11=A.m11,this.m12=A.m12,this.m20=A.m20,this.m21=A.m21,this.m22=A.m22}setFromDirectionAndUpvector(A,g){const I=A,C=I.length();if(C<Number.EPSILON)return void this.setIdentity();I.scaleInPlace(1/C);const e=g.cross(I),i=e.length();i>Number.EPSILON&&e.scaleInPlace(1/i);const t=I.cross(e),o=t.length();o>Number.EPSILON&&t.scaleInPlace(1/o),this.set(e.x,e.y,e.z,t.x,t.y,t.z,I.x,I.y,I.z)}inverse(){const A=this.m00,g=this.m01,I=this.m02,C=this.m10,e=this.m11,i=this.m12,t=this.m20,o=this.m21,a=this.m22,B=a*e-i*o,s=-a*C+i*t,l=o*C-e*t;let n=A*B+g*s+I*l;return n?(n=1/n,new Mat3(B*n,(-a*g+I*o)*n,(i*g-I*e)*n,s*n,(a*A-I*t)*n,(-i*A+I*C)*n,l*n,(-o*A+g*t)*n,(e*A-g*C)*n)):(console.warn("Unable to invert Mat3"),new Mat3)}invertInPlace(){const A=this.m00,g=this.m01,I=this.m02,C=this.m10,e=this.m11,i=this.m12,t=this.m20,o=this.m21,a=this.m22,B=a*e-i*o,s=-a*C+i*t,l=o*C-e*t;let n=A*B+g*s+I*l;return n?(n=1/n,this.set(B*n,(-a*g+I*o)*n,(i*g-I*e)*n,s*n,(a*A-I*t)*n,(-i*A+I*C)*n,l*n,(-o*A+g*t)*n,(e*A-g*C)*n),!0):(console.warn("Unable to invert Mat3"),!1)}transpose(){return new Mat3(this.m00,this.m10,this.m20,this.m01,this.m11,this.m21,this.m02,this.m12,this.m22)}transposeInPlace(){const A=this.m01,g=this.m02,I=this.m12;this.m01=this.m10,this.m02=this.m20,this.m10=A,this.m12=this.m21,this.m20=g,this.m21=I}transformVec3(A){return new Vec3(this.m00*A.x+this.m01*A.y+this.m02*A.z,this.m10*A.x+this.m11*A.y+this.m12*A.z,this.m20*A.x+this.m21*A.y+this.m22*A.z)}clone(){return new Mat3(this.m00,this.m01,this.m02,this.m10,this.m11,this.m12,this.m20,this.m21,this.m22)}readBinary(A){const g=A.loadFloat32Array(9);this.fromArray(g)}toJSON(){return this.asArray()}fromJSON(A){this.fromArray(A)}toString(){return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON())}asArray(){return[this.m00,this.m01,this.m02,this.m10,this.m11,this.m12,this.m20,this.m21,this.m22]}fromArray(A){this.m00=A[0],this.m01=A[1],this.m02=A[2],this.m10=A[3],this.m11=A[4],this.m12=A[5],this.m20=A[6],this.m21=A[7],this.m22=A[8]}}class Mat4{m00;m01;m02;m03;m10;m11;m12;m13;m20;m21;m22;m23;m30;m31;m32;m33;constructor(A=1,g=0,I=0,C=0,e=0,i=1,t=0,o=0,a=0,B=0,s=1,l=0,n=0,d=0,r=0,c=1){this.m00=A,this.m01=g,this.m02=I,this.m03=C,this.m10=e,this.m11=i,this.m12=t,this.m13=o,this.m20=a,this.m21=B,this.m22=s,this.m23=l,this.m30=n,this.m31=d,this.m32=r,this.m33=c}get xAxis(){return new Vec3(this.m00,this.m01,this.m02)}set xAxis(A){this.xAxis.set(A.x,A.y,A.z)}get yAxis(){return new Vec3(this.m10,this.m11,this.m12)}set yAxis(A){this.yAxis.set(A.x,A.y,A.z)}get zAxis(){return new Vec3(this.m20,this.m21,this.m22)}set zAxis(A){this.zAxis.set(A.x,A.y,A.z)}get translation(){return new Vec3(this.m30,this.m31,this.m32)}set translation(A){this.m30=A.x,this.m31=A.y,this.m32=A.z}set(A=1,g=0,I=0,C=0,e=0,i=1,t=0,o=0,a=0,B=0,s=1,l=0,n=0,d=0,r=0,c=1){this.m00=A,this.m01=g,this.m02=I,this.m03=C,this.m10=e,this.m11=i,this.m12=t,this.m13=o,this.m20=a,this.m21=B,this.m22=s,this.m23=l,this.m30=n,this.m31=d,this.m32=r,this.m33=c}setIdentity(){this.set()}setFromMat4(A){this.m00=A.m00,this.m01=A.m01,this.m02=A.m02,this.m03=A.m03,this.m10=A.m10,this.m11=A.m11,this.m12=A.m12,this.m13=A.m13,this.m20=A.m20,this.m21=A.m21,this.m22=A.m22,this.m23=A.m23,this.m30=A.m30,this.m31=A.m31,this.m32=A.m32,this.m33=A.m33}toMat3(){return new Mat3(this.m00,this.m01,this.m02,this.m10,this.m11,this.m12,this.m20,this.m21,this.m22)}transposeInPlace(){const A=this.m01,g=this.m02,I=this.m03,C=this.m12,e=this.m13,i=this.m23;this.m01=this.m10,this.m02=this.m20,this.m03=this.m30,this.m10=A,this.m12=this.m21,this.m13=this.m31,this.m20=g,this.m21=C,this.m23=this.m32,this.m30=I,this.m31=e,this.m32=i}transpose(){return new Mat4(this.m00,this.m10,this.m20,this.m30,this.m01,this.m11,this.m21,this.m31,this.m02,this.m12,this.m22,this.m32,this.m03,this.m13,this.m23,this.m33)}inverse(){const A=this.m00,g=this.m01,I=this.m02,C=this.m03,e=this.m10,i=this.m11,t=this.m12,o=this.m13,a=this.m20,B=this.m21,s=this.m22,l=this.m23,n=this.m30,d=this.m31,r=this.m32,c=this.m33,h=A*i-g*e,G=A*t-I*e,m=A*o-C*e,b=g*t-I*i,Z=g*o-C*i,u=I*o-C*t,y=a*d-B*n,E=a*r-s*n,Q=a*c-l*n,V=B*r-s*d,R=B*c-l*d,W=s*c-l*r;let p=h*W-G*R+m*V+b*Q-Z*E+u*y;return p?(p=1/p,new Mat4((i*W-t*R+o*V)*p,(I*R-g*W-C*V)*p,(d*u-r*Z+c*b)*p,(s*Z-B*u-l*b)*p,(t*Q-e*W-o*E)*p,(A*W-I*Q+C*E)*p,(r*m-n*u-c*G)*p,(a*u-s*m+l*G)*p,(e*R-i*Q+o*y)*p,(g*Q-A*R-C*y)*p,(n*Z-d*m+c*h)*p,(B*m-a*Z-l*h)*p,(i*E-e*V-t*y)*p,(A*V-g*E+I*y)*p,(d*G-n*b-r*h)*p,(a*b-B*G+s*h)*p)):(console.warn("Unable to invert Mat4"),this)}invertInPlace(){const A=this.m00,g=this.m01,I=this.m02,C=this.m03,e=this.m10,i=this.m11,t=this.m12,o=this.m13,a=this.m20,B=this.m21,s=this.m22,l=this.m23,n=this.m30,d=this.m31,r=this.m32,c=this.m33,h=A*i-g*e,G=A*t-I*e,m=A*o-C*e,b=g*t-I*i,Z=g*o-C*i,u=I*o-C*t,y=a*d-B*n,E=a*r-s*n,Q=a*c-l*n,V=B*r-s*d,R=B*c-l*d,W=s*c-l*r;let p=h*W-G*R+m*V+b*Q-Z*E+u*y;return p?(p=1/p,this.set((i*W-t*R+o*V)*p,(I*R-g*W-C*V)*p,(d*u-r*Z+c*b)*p,(s*Z-B*u-l*b)*p,(t*Q-e*W-o*E)*p,(A*W-I*Q+C*E)*p,(r*m-n*u-c*G)*p,(a*u-s*m+l*G)*p,(e*R-i*Q+o*y)*p,(g*Q-A*R-C*y)*p,(n*Z-d*m+c*h)*p,(B*m-a*Z-l*h)*p,(i*E-e*V-t*y)*p,(A*V-g*E+I*y)*p,(d*G-n*b-r*h)*p,(a*b-B*G+s*h)*p),!0):(console.warn("Unable to invert Mat4"),!1)}setInverse(A){const g=A.m00,I=A.m01,C=A.m02,e=A.m03,i=A.m10,t=A.m11,o=A.m12,a=A.m13,B=A.m20,s=A.m21,l=A.m22,n=A.m23,d=A.m30,r=A.m31,c=A.m32,h=A.m33,G=g*t-I*i,m=g*o-C*i,b=g*a-e*i,Z=I*o-C*t,u=I*a-e*t,y=C*a-e*o,E=B*r-s*d,Q=B*c-l*d,V=B*h-n*d,R=s*c-l*r,W=s*h-n*r,p=l*h-n*c;let S=G*p-m*W+b*R+Z*V-u*Q+y*E;if(!S)throw new Error("Unable to invert Mat4");S=1/S,this.set((t*p-o*W+a*R)*S,(C*W-I*p-e*R)*S,(r*y-c*u+h*Z)*S,(l*u-s*y-n*Z)*S,(o*V-i*p-a*Q)*S,(g*p-C*V+e*Q)*S,(c*b-d*y-h*m)*S,(B*y-l*b+n*m)*S,(i*W-t*V+a*E)*S,(I*V-g*W-e*E)*S,(d*u-r*b+h*G)*S,(s*b-B*u-n*G)*S,(t*Q-i*R-o*E)*S,(g*R-I*Q+C*E)*S,(r*m-d*Z-c*G)*S,(B*Z-s*m+l*G)*S)}multiply(A){const g=this.m00,I=this.m01,C=this.m02,e=this.m03,i=this.m10,t=this.m11,o=this.m12,a=this.m13,B=this.m20,s=this.m21,l=this.m22,n=this.m23,d=this.m30,r=this.m31,c=this.m32,h=this.m33,G=A.asArray();let m=G[0],b=G[1],Z=G[2],u=G[3];const y=new Mat4;return y.m00=m*g+b*i+Z*B+u*d,y.m01=m*I+b*t+Z*s+u*r,y.m02=m*C+b*o+Z*l+u*c,y.m03=m*e+b*a+Z*n+u*h,m=G[4],b=G[5],Z=G[6],u=G[7],y.m10=m*g+b*i+Z*B+u*d,y.m11=m*I+b*t+Z*s+u*r,y.m12=m*C+b*o+Z*l+u*c,y.m13=m*e+b*a+Z*n+u*h,m=G[8],b=G[9],Z=G[10],u=G[11],y.m20=m*g+b*i+Z*B+u*d,y.m21=m*I+b*t+Z*s+u*r,y.m22=m*C+b*o+Z*l+u*c,y.m23=m*e+b*a+Z*n+u*h,m=G[12],b=G[13],Z=G[14],u=G[15],y.m30=m*g+b*i+Z*B+u*d,y.m31=m*I+b*t+Z*s+u*r,y.m32=m*C+b*o+Z*l+u*c,y.m33=m*e+b*a+Z*n+u*h,y}multiplyInPlace(A){const g=this.asArray(),I=g[0],C=g[1],e=g[2],i=g[3],t=g[4],o=g[5],a=g[6],B=g[7],s=g[8],l=g[9],n=g[10],d=g[11],r=g[12],c=g[13],h=g[14],G=g[15],m=A.asArray();let b=m[0],Z=m[1],u=m[2],y=m[3];return this.m00=b*I+Z*t+u*s+y*r,this.m01=b*C+Z*o+u*l+y*c,this.m02=b*e+Z*a+u*n+y*h,this.m03=b*i+Z*B+u*d+y*G,b=m[4],Z=m[5],u=m[6],y=m[7],this.m10=b*I+Z*t+u*s+y*r,this.m11=b*C+Z*o+u*l+y*c,this.m12=b*e+Z*a+u*n+y*h,this.m13=b*i+Z*B+u*d+y*G,b=m[8],Z=m[9],u=m[10],y=m[11],this.m20=b*I+Z*t+u*s+y*r,this.m21=b*C+Z*o+u*l+y*c,this.m22=b*e+Z*a+u*n+y*h,this.m23=b*i+Z*B+u*d+y*G,b=m[12],Z=m[13],u=m[14],y=m[15],this.m30=b*I+Z*t+u*s+y*r,this.m31=b*C+Z*o+u*l+y*c,this.m32=b*e+Z*a+u*n+y*h,this.m33=b*i+Z*B+u*d+y*G,this}postMultiplyInPlace(A){const g=A.asArray(),I=g[0],C=g[1],e=g[2],i=g[3],t=g[4],o=g[5],a=g[6],B=g[7],s=g[8],l=g[9],n=g[10],d=g[11],r=g[12],c=g[13],h=g[14],G=g[15],m=this.asArray();let b=m[0],Z=m[1],u=m[2],y=m[3];return this.m00=b*I+Z*t+u*s+y*r,this.m01=b*C+Z*o+u*l+y*c,this.m02=b*e+Z*a+u*n+y*h,this.m03=b*i+Z*B+u*d+y*G,b=m[4],Z=m[5],u=m[6],y=m[7],this.m10=b*I+Z*t+u*s+y*r,this.m11=b*C+Z*o+u*l+y*c,this.m12=b*e+Z*a+u*n+y*h,this.m13=b*i+Z*B+u*d+y*G,b=m[8],Z=m[9],u=m[10],y=m[11],this.m20=b*I+Z*t+u*s+y*r,this.m21=b*C+Z*o+u*l+y*c,this.m22=b*e+Z*a+u*n+y*h,this.m23=b*i+Z*B+u*d+y*G,b=m[12],Z=m[13],u=m[14],y=m[15],this.m30=b*I+Z*t+u*s+y*r,this.m31=b*C+Z*o+u*l+y*c,this.m32=b*e+Z*a+u*n+y*h,this.m33=b*i+Z*B+u*d+y*G,this}translateInPlace(A){const g=this.asArray(),I=A.x,C=A.y,e=A.z;return g[12]=g[0]*I+g[4]*C+g[8]*e+g[12],g[13]=g[1]*I+g[5]*C+g[9]*e+g[13],g[14]=g[2]*I+g[6]*C+g[10]*e+g[14],g[15]=g[3]*I+g[7]*C+g[11]*e+g[15],this}setLookAt(A,g,I){const C=A.subtract(g),e=C.length();if(e<Number.EPSILON)return void this.setIdentity();C.scaleInPlace(1/e);const i=I.cross(C),t=i.length();t>Number.EPSILON&&i.scaleInPlace(1/t);const o=C.cross(i),a=o.length();a>Number.EPSILON&&o.scaleInPlace(1/a),this.set(i.x,i.y,i.z,0,o.x,o.y,o.z,0,C.x,C.y,C.z,0,A.x,A.y,A.z,1)}setRotation(A,g){const I=A.length();if(Math.abs(I)<Number.EPSILON)return null;const C=A.x/I,e=A.y/I,i=A.z/I,t=Math.sin(g),o=Math.cos(g),a=1-o,B=this.asArray();return B[0]=C*C*a+o,B[1]=e*C*a+i*t,B[2]=i*C*a-e*t,B[3]=0,B[4]=C*e*a-i*t,B[5]=e*e*a+o,B[6]=i*e*a+C*t,B[7]=0,B[8]=C*i*a+e*t,B[9]=e*i*a-C*t,B[10]=i*i*a+o,B[11]=0,B[12]=0,B[13]=0,B[14]=0,B[15]=1,this}setXRotation(A){const g=Math.sin(A),I=Math.cos(A),C=this.asArray();return C[0]=1,C[1]=0,C[2]=0,C[3]=0,C[4]=0,C[5]=I,C[6]=g,C[7]=0,C[8]=0,C[9]=-g,C[10]=I,C[11]=0,C[12]=0,C[13]=0,C[14]=0,C[15]=1,this}setYRotation(A){const g=Math.sin(A),I=Math.cos(A),C=this.asArray();return C[0]=I,C[1]=0,C[2]=-g,C[3]=0,C[4]=0,C[5]=1,C[6]=0,C[7]=0,C[8]=g,C[9]=0,C[10]=I,C[11]=0,C[12]=0,C[13]=0,C[14]=0,C[15]=1,this}setZRotation(A){const g=Math.sin(A),I=Math.cos(A),C=this.asArray();return C[0]=I,C[1]=g,C[2]=0,C[3]=0,C[4]=-g,C[5]=I,C[6]=0,C[7]=0,C[8]=0,C[9]=0,C[10]=1,C[11]=0,C[12]=0,C[13]=0,C[14]=0,C[15]=1,this}transformVec4(A){const g=this.asArray(),I=A.x,C=A.y,e=A.z,i=A.w;return new Vec4(g[0]*I+g[4]*C+g[8]*e+g[12]*i,g[1]*I+g[5]*C+g[9]*e+g[13]*i,g[2]*I+g[6]*C+g[10]*e+g[14]*i,g[3]*I+g[7]*C+g[11]*e+g[15]*i)}transformVec3(A){const g=this.asArray(),I=A.x,C=A.y,e=A.z;return new Vec3(g[0]*I+g[4]*C+g[8]*e+g[12],g[1]*I+g[5]*C+g[9]*e+g[13],g[2]*I+g[6]*C+g[10]*e+g[14])}rotateVec3(A){const g=this.asArray(),I=A.x,C=A.y,e=A.z;return new Vec3(g[0]*I+g[4]*C+g[8]*e,g[1]*I+g[5]*C+g[9]*e,g[2]*I+g[6]*C+g[10]*e)}setPerspectiveMatrix(A,g,I,C){const e=Math.tan(.5*Math.PI-.5*A),i=1/(I-C);this.set(e/g,0,0,0,0,e,0,0,0,0,(I+C)*i,-1,0,0,I*C*i*2,0)}setOrthographicMatrix(A,g,I,C,e,i){const t=1/(A-g),o=1/(I-C),a=1/(e-i);this.set(-2*t,0,0,0,0,-2*o,0,0,0,0,2*a,0,(A+g)*t,(C+I)*o,(i+e)*a,1)}setScale(A,g,I){A instanceof Vec3?this.set(A.x,0,0,0,0,A.y,0,0,0,0,A.z,0,0,0,0,1):this.set(A,0,0,0,0,g,0,0,0,0,I,0,0,0,0,1)}setFromMat3x4Array(A){this.set(A[0],A[1],A[2],0,A[3],A[4],A[5],0,A[6],A[7],A[8],0,A[9],A[10],A[11],1)}clone(){return new Mat4(this.m00,this.m01,this.m02,this.m03,this.m10,this.m11,this.m12,this.m13,this.m20,this.m21,this.m22,this.m23,this.m30,this.m31,this.m32,this.m33)}toString(){return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON())}toJSON(){return this.asArray()}fromJSON(A){this.fromArray(A)}readBinary(A){this.fromArray(A.loadFloat32Array(16))}asArray(){return[this.m00,this.m01,this.m02,this.m03,this.m10,this.m11,this.m12,this.m13,this.m20,this.m21,this.m22,this.m23,this.m30,this.m31,this.m32,this.m33]}fromArray(A){this.m00=A[0],this.m01=A[1],this.m02=A[2],this.m03=A[3],this.m10=A[4],this.m11=A[5],this.m12=A[6],this.m13=A[7],this.m20=A[8],this.m21=A[9],this.m22=A[10],this.m23=A[11],this.m30=A[12],this.m31=A[13],this.m32=A[14],this.m33=A[15]}}class Quat{x;y;z;w;constructor(A=0,g=0,I=0,C=1){this.x=A,this.y=g,this.z=I,this.w=C}set(A,g,I,C){this.x=A,this.y=g,this.z=I,this.w=C}setFromOther(A){this.x=A.x,this.y=A.y,this.z=A.z,this.w=A.w}setFromEulerAngles(A){const g=new Vec3;switch(A.order){case EulerAnglesAxisOrder.XYZ:g.set(A.x,-A.y,A.z);break;case EulerAnglesAxisOrder.YZX:g.set(A.y,-A.z,A.x);break;case EulerAnglesAxisOrder.ZXY:g.set(A.z,-A.x,A.y);break;case EulerAnglesAxisOrder.XZY:g.set(A.x,A.z,A.y);break;case EulerAnglesAxisOrder.ZYX:g.set(A.z,A.y,A.x);break;case EulerAnglesAxisOrder.YXZ:g.set(A.y,A.x,A.z);break;default:throw new Error(`Invalid EulerAngles order: ${A.order}`)}const I=.5*g.x,C=.5*g.y,e=.5*g.z,i=Math.cos(I),t=Math.cos(C),o=Math.cos(e),a=Math.sin(I),B=Math.sin(C),s=Math.sin(e),l=i*o,n=i*s,d=a*o,r=a*s,c=t*d-B*n,h=t*r+B*l,G=t*n-B*d;switch(this.w=t*l+B*r,A.order){case 0:this.x=c,this.y=-h,this.z=G;break;case 1:this.x=G,this.y=c,this.z=-h;break;case 2:this.x=-h,this.y=G,this.z=c;break;case 3:this.x=c,this.y=G,this.z=h;break;case 4:this.x=G,this.y=h,this.z=c;break;case 5:this.x=h,this.y=c,this.z=G;break;default:throw new Error(`Invalid EulerAngles order: ${A.order}`)}}toEulerAngles(A){const g=new Vec3;switch(A){case EulerAnglesAxisOrder.XYZ:case"XYZ":g.set(this.z,this.x,this.y);break;case EulerAnglesAxisOrder.YZX:case"YZX":g.set(this.x,this.y,this.z);break;case EulerAnglesAxisOrder.ZXY:case"ZXY":g.set(this.y,this.z,this.x);break;case EulerAnglesAxisOrder.XZY:case"XZY":g.set(this.y,-this.x,this.z);break;case EulerAnglesAxisOrder.ZYX:case"ZYX":g.set(this.x,-this.z,this.y);break;case EulerAnglesAxisOrder.YXZ:case"YXZ":g.set(this.z,-this.y,this.x);break;default:throw new Error("Invalid rotation order:"+A)}const I=new Vec3,C=g.x*g.y+g.z*this.w;if(C>.49999)I.y=2*Math.atan2(g.x,this.w),I.z=.5*Math.PI,I.x=0;else if(C<-.49999)I.y=-2*Math.atan2(g.x,this.w),I.z=-.5*Math.PI,I.x=0;else{const A=g.x*g.x,e=g.y*g.y,i=g.z*g.z;I.y=Math.atan2(2*g.y*this.w-2*g.x*g.z,1-2*e-2*i),I.z=Math.asin(2*C),I.x=Math.atan2(2*g.x*this.w-2*g.y*g.z,1-2*A-2*i)}switch(A){case EulerAnglesAxisOrder.XYZ:case"XYZ":return new EulerAngles(I.y,I.z,I.x,A);case EulerAnglesAxisOrder.YZX:case"YZX":return new EulerAngles(I.x,I.y,I.z,A);case EulerAnglesAxisOrder.ZXY:case"ZXY":return new EulerAngles(I.z,I.x,I.y,A);case EulerAnglesAxisOrder.XZY:case"XZY":return new EulerAngles(-I.y,I.x,I.z,A);case EulerAnglesAxisOrder.ZYX:case"ZYX":return new EulerAngles(I.x,I.z,-I.y,A);case EulerAnglesAxisOrder.YXZ:case"YXZ":return new EulerAngles(I.z,-I.y,I.x,A)}}setFromAxisAndAngle(A,g){const I=g/2,C=A.normalize().scale(Math.sin(I));this.set(C.x,C.y,C.z,Math.cos(I))}setFromDirectionAndUpvector(A,g){const I=new Mat3;I.setFromDirectionAndUpvector(A,g),this.setFromMat3(I)}setFrom2Vectors(A,g){const I=A.cross(g),C=A.dot(g),e=Math.sqrt(2*(1+C));this.set(I.x/e,I.y/e,I.z/e,e/2),this.normalizeInPlace()}setFromMat3(A){const g=A.asArray(),I=g[0]+g[4]+g[8];let C;if(I>0)C=Math.sqrt(I+1),this.w=.5*C,C=.5/C,this.x=(g[5]-g[7])*C,this.y=(g[6]-g[2])*C,this.z=(g[1]-g[3])*C;else{let A=0;g[4]>g[0]&&(A=1),g[8]>g[3*A+A]&&(A=2);const I=(A+1)%3,e=(A+2)%3;C=Math.sqrt(g[3*A+A]-g[3*I+I]-g[3*e+e]+1);const i=[0,0,0,0];i[A]=.5*C,C=.5/C,i[3]=(g[3*I+e]-g[3*e+I])*C,i[I]=(g[3*I+A]+g[3*A+I])*C,i[e]=(g[3*e+A]+g[3*A+e])*C,this.fromArray(i)}this.normalizeInPlace()}setFromMat4(A){const g=A.asArray(),I=g[0]+g[5]+g[10];let C;if(I>0)C=Math.sqrt(I+1),this.w=.5*C,C=.5/C,this.x=(g[6]-g[9])*C,this.y=(g[8]-g[2])*C,this.z=(g[1]-g[4])*C;else{let A=0;g[5]>g[0]&&(A=1),g[10]>g[4*A+A]&&(A=2);const I=(A+1)%3,e=(A+2)%3;C=Math.sqrt(g[4*A+A]-g[4*I+I]-g[4*e+e]+1);const i=[0,0,0,0];i[A]=.5*C,C=.5/C,i[3]=(g[4*I+e]-g[4*e+I])*C,i[I]=(g[4*I+A]+g[4*A+I])*C,i[e]=(g[4*e+A]+g[4*A+e])*C,this.fromArray(i)}this.normalizeInPlace()}isIdentity(){return this.getAngle()<Number.EPSILON}getAngle(){return 2*Math.acos(this.w)}isEqual(A){return this.x==A.x&&this.y==A.y&&this.z==A.z&&this.w==A.w}notEquals(A){return this.x!=A.x&&this.y!=A.y&&this.z!=A.z&&this.w!=A.w}approxEqual(A,g=Number.EPSILON){return Math.abs(this.x-A.x)<g&&Math.abs(this.y-A.y)<g&&Math.abs(this.z-A.z)<g&&Math.abs(this.w-A.w)<g}add(A){return new Quat(this.x+A.x,this.y+A.y,this.z+A.z,this.w+A.w)}addInPlace(A){this.x+=A.x,this.y+=A.y,this.z+=A.z,this.w+=A.w}subtract(A){return new Quat(this.x-A.x,this.y-A.y,this.z-A.z,this.w-A.w)}scale(A){return new Quat(this.x*A,this.y*A,this.z*A,this.w*A)}scaleInPlace(A){this.x*=A,this.y*=A,this.z*=A,this.w*=A}length(){const A=this.x,g=this.y,I=this.z,C=this.w;return Math.sqrt(A*A+g*g+I*I+C*C)}lengthSquared(){const A=this.x,g=this.y,I=this.z,C=this.w;return A*A+g*g+I*I+C*C}normalize(){const A=this.x,g=this.y,I=this.z,C=this.w;let e=A*A+g*g+I*I+C*C;return e<Number.EPSILON?new Quat:(e=1/Math.sqrt(e),new Quat(A*e,g*e,I*e,C*e))}normalizeInPlace(){const A=this.x,g=this.y,I=this.z,C=this.w;let e=A*A+g*g+I*I+C*C;e<Number.EPSILON||(e=1/Math.sqrt(e),this.set(A*e,g*e,I*e,C*e))}dot(A){return this.x*A.x+this.y*A.y+this.z*A.z+this.w*A.w}cross(A){const g=this.x,I=this.y,C=this.z,e=this.w,i=A.x,t=A.y,o=A.z,a=A.w;return new Quat(I*o-C*t,C*a-e*o,e*i-g*a,g*t-I*i)}conjugate(){return new Quat(-this.x,-this.y,-this.z,this.w)}inverse(){return this.conjugate()}alignWith(A){this.dot(A)<0&&this.set(-this.x,-this.y,-this.z,-this.w)}multiply(A){const g=this.x,I=this.y,C=this.z,e=this.w,i=A.x,t=A.y,o=A.z,a=A.w;return new Quat(g*a+e*i+I*o-C*t,I*a+e*t+C*i-g*o,C*a+e*o+g*t-I*i,e*a-g*i-I*t-C*o)}multiplyInPlace(A){const g=this.x,I=this.y,C=this.z,e=this.w,i=A.x,t=A.y,o=A.z,a=A.w;this.set(g*a+e*i+I*o-C*t,I*a+e*t+C*i-g*o,C*a+e*o+g*t-I*i,e*a-g*i-I*t-C*o)}rotateVec3(A){const g=new Quat(A.x,A.y,A.z,0),I=this.multiply(g).multiply(this.conjugate());return new Vec3(I.x,I.y,I.z)}rotateX(A){A*=.5;const g=this.x,I=this.y,C=this.z,e=this.w,i=Math.sin(A),t=Math.cos(A);this.x=g*t+e*i,this.y=I*t+C*i,this.z=C*t-I*i,this.w=e*t-g*i}rotateY(A){A*=.5;const g=this.x,I=this.y,C=this.z,e=this.w,i=Math.sin(A),t=Math.cos(A);this.x=g*t-C*i,this.y=I*t+e*i,this.z=C*t+g*i,this.w=e*t-I*i}rotateZ(A){A*=.5;const g=this.x,I=this.y,C=this.z,e=this.w,i=Math.sin(A),t=Math.cos(A);this.x=g*t+I*i,this.y=I*t-g*i,this.z=C*t+e*i,this.w=e*t-C*i}toMat3(){const A=this.x,g=this.y,I=this.z,C=this.w,e=A+A,i=g+g,t=I+I,o=A*e,a=g*e,B=g*i,s=I*e,l=I*i,n=I*t,d=C*e,r=C*i,c=C*t,h=new Mat3;return h.m00=1-B-n,h.m10=a-c,h.m20=s+r,h.m01=a+c,h.m11=1-o-n,h.m21=l-d,h.m02=s-r,h.m12=l+d,h.m22=1-o-B,h}getXaxis(){const A=this.x*this.y,g=this.x*this.z,I=this.y*this.y,C=this.y*this.w,e=this.z*this.z,i=this.z*this.w;return new Vec3(1-2*(e+I),2*(A+i),2*(g-C))}getYaxis(){const A=this.x*this.x,g=this.x*this.y,I=this.x*this.w,C=this.y*this.z,e=this.z*this.z,i=this.z*this.w;return new Vec3(2*(g-i),1-2*(e+A),2*(C+I))}getZaxis(){const A=this.x*this.x,g=this.x*this.z,I=this.x*this.w,C=this.y*this.y,e=this.y*this.z,i=this.y*this.w;return new Vec3(2*(i+g),2*(e-I),1-2*(C+A))}mirror(A){switch(A){case 0:case 0:default:return new Quat(this.z,this.w,this.x,this.y);case 1:return new Quat(-this.w,this.z,this.y,-this.x);case 2:return new Quat(this.x,this.y,this.z,-this.w)}}toMat4(){const A=this.x,g=this.y,I=this.z,C=this.w,e=A+A,i=g+g,t=I+I,o=A*e,a=g*e,B=g*i,s=I*e,l=I*i,n=I*t,d=C*e,r=C*i,c=C*t,h=new Mat4;return h.m00=1-B-n,h.m10=a-c,h.m20=s+r,h.m01=a+c,h.m11=1-o-n,h.m21=l-d,h.m02=s-r,h.m12=l+d,h.m22=1-o-B,h}lerp(A,g){const I=new Quat(this.x+g*(A.x-this.x),this.y+g*(A.y-this.y),this.z+g*(A.z-this.z),this.w+g*(A.w-this.w));return I.normalizeInPlace(),I}slerp(A,g){const I=this.dot(A);if(I>.999)return this;const C=Math.acos(I),e=Math.sin(C),i=Math.sin(g*C),t=Math.sin((1-g)*C)/e,o=i/e,a=new Quat(t*this.x+o*A.x,t*this.y+o*A.y,t*this.z+o*A.z,t*this.w+o*A.w);return a.normalizeInPlace(),a}clone(){return new Quat(this.x,this.y,this.z,this.w)}asArray(){return[this.x,this.y,this.z,this.w]}fromArray(A){this.x=A[0],this.y=A[1],this.z=A[2],this.w=A[3]}toString(){return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON())}toJSON(){return{x:this.x,y:this.y,z:this.z,w:this.w}}fromJSON(A){this.x=A.x,this.y=A.y,this.z=A.z,this.w=A.w,this.normalizeInPlace()}readBinary(A){this.x=A.loadFloat32(),this.y=A.loadFloat32(),this.z=A.loadFloat32(),this.w=A.loadFloat32()}}class Xfo{tr;ori;sc;constructor(A=new Vec3,g=new Quat,I=new Vec3(1,1,1)){this.tr=A,this.ori=g,this.sc=I}set(A,g,I){this.tr=A,this.ori=g,I instanceof Vec3&&(this.sc=I)}setFromOther(A){this.tr=A.tr,this.ori=A.ori,this.sc=A.sc}isIdentity(){return this.tr.isNull()&&this.ori.isIdentity()&&this.sc.is111()}isEqual(A){return this.tr.isEqual(A.tr)&&this.ori.isEqual(A.ori)&&this.sc.isEqual(A.sc)}approxEqual(A,g=Number.EPSILON){return(!A.tr||this.tr.approxEqual(A.tr,g))&&(!A.ori||this.ori.approxEqual(A.ori,g))&&(!A.sc||this.sc.approxEqual(A.sc,g))}setLookAt(A,g,I){const C=A.subtract(g);if(C.length()<Number.EPSILON)throw new Error("Invalid dir");this.ori.setFromDirectionAndUpvector(C,I),this.tr=A}multiply(A){let g=this.sc;if(Math.abs(this.sc.y-this.sc.x)>.001||Math.abs(this.sc.z-this.sc.x)>.001||Math.abs(this.sc.z-this.sc.y)>.001){const I=this.toMat4(),C=A.ori.toMat4(),e=I.multiply(C);g=new Vec3(e.xAxis.length(),e.yAxis.length(),e.zAxis.length())}return new Xfo(this.tr.add(this.ori.rotateVec3(this.sc.multiply(A.tr))),this.ori.multiply(A.ori),g.multiply(A.sc))}inverse(){const A=new Xfo;if(A.ori=this.ori.inverse(),Math.abs(this.sc.y-this.sc.x)>.001||Math.abs(this.sc.z-this.sc.x)>.001||Math.abs(this.sc.z-this.sc.y)>.001){const g=this.toMat4().inverse();A.sc=new Vec3(g.xAxis.length(),g.yAxis.length(),g.zAxis.length())}else A.sc=this.sc.inverse();return A.tr=A.ori.rotateVec3(this.tr.negate().multiply(A.sc)),A}transformVec3(A){return this.tr.add(this.ori.rotateVec3(this.sc.multiply(A)))}lerp(A,g){return new Xfo(this.tr.lerp(A.tr,g),this.ori.slerp(A.ori,g),this.sc.lerp(A.sc,g))}toMat4(){const A=new Mat4(this.sc.x,0,0,0,0,this.sc.y,0,0,0,0,this.sc.z,0,0,0,0,1),g=this.ori.toMat4(),I=new Mat4;return I.translation=this.tr,I.multiply(g).multiply(A)}setFromMat4(A){this.tr=A.translation,this.ori.setFromMat4(A),this.sc.set(A.xAxis.length(),A.yAxis.length(),A.zAxis.length())}clone(){return new Xfo(this.tr.clone(),this.ori.clone(),this.sc.clone())}toJSON(){return{tr:this.tr.toJSON(),ori:this.ori.toJSON(),sc:this.sc.toJSON()}}fromJSON(A){this.tr.fromJSON(A.tr),this.ori.fromJSON(A.ori),A.sc&&this.sc.fromJSON(A.sc)}readBinary(A){this.tr.readBinary(A),this.ori.readBinary(A),this.sc.readBinary(A)}toString(){return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON())}}class Box2{p0;p1;constructor(A,g){this.p0=A instanceof Vec2?A:new Vec2(Number.POSITIVE_INFINITY,Number.POSITIVE_INFINITY),this.p1=g instanceof Vec2?g:new Vec2(Number.NEGATIVE_INFINITY,Number.NEGATIVE_INFINITY)}set(A,g){this.p0=A,this.p1=g}reset(){this.p0.x=Number.POSITIVE_INFINITY,this.p1.x=Number.NEGATIVE_INFINITY,this.p0.y=Number.POSITIVE_INFINITY,this.p1.y=Number.NEGATIVE_INFINITY}isValid(){return this.p0.x!=Number.POSITIVE_INFINITY&&this.p1.x!=Number.NEGATIVE_INFINITY&&this.p0.y!=Number.POSITIVE_INFINITY&&this.p1.y!=Number.NEGATIVE_INFINITY}addPoint(A){(this.p0.x==Number.POSITIVE_INFINITY||A.x<this.p0.x)&&(this.p0.x=A.x),(this.p0.y==Number.POSITIVE_INFINITY||A.y<this.p0.y)&&(this.p0.y=A.y),(this.p1.y==Number.NEGATIVE_INFINITY||A.x>this.p1.x)&&(this.p1.x=A.x),(this.p1.y==Number.NEGATIVE_INFINITY||A.y>this.p1.y)&&(this.p1.y=A.y)}size(){return this.p1.distanceTo(this.p0)}diagonal(){return this.p1.subtract(this.p0)}center(){const A=this.p1.subtract(this.p0);return A.scaleInPlace(.5),A.addInPlace(this.p0),A}clone(){return new Box2(this.p0.clone(),this.p1.clone())}toJSON(){return{p0:this.p0.toJSON(),p1:this.p1.toJSON()}}fromJSON(A){const g={x:MathFunctions.isNumeric(A.p0.x)?A.p0.x:Number.POSITIVE_INFINITY,y:MathFunctions.isNumeric(A.p0.y)?A.p0.y:Number.POSITIVE_INFINITY},I={x:MathFunctions.isNumeric(A.p1.x)?A.p1.x:Number.NEGATIVE_INFINITY,y:MathFunctions.isNumeric(A.p1.y)?A.p1.y:Number.NEGATIVE_INFINITY};this.p0.fromJSON(g),this.p1.fromJSON(I)}toString(){return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON())}}class SphereType{pos;radius;constructor(A=new Vec3,g=0){this.pos=A instanceof Vec3?A:new Vec3,this.radius=g}clone(){return new SphereType(this.pos.clone(),this.radius)}intersectsBox(A){return A.intersectsSphere(this)}toJSON(){return{pos:this.pos.toJSON(),radius:this.radius}}toString(){return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON())}}class Box3{p0;p1;constructor(A=new Vec3(Number.POSITIVE_INFINITY,Number.POSITIVE_INFINITY,Number.POSITIVE_INFINITY),g=new Vec3(Number.NEGATIVE_INFINITY,Number.NEGATIVE_INFINITY,Number.NEGATIVE_INFINITY)){this.p0=A,this.p1=g}get min(){return this.p0}get max(){return this.p1}set(A,g){this.p0=A,this.p1=g}reset(){this.p0.x=Number.POSITIVE_INFINITY,this.p0.y=Number.POSITIVE_INFINITY,this.p0.z=Number.POSITIVE_INFINITY,this.p1.x=Number.NEGATIVE_INFINITY,this.p1.y=Number.NEGATIVE_INFINITY,this.p1.z=Number.NEGATIVE_INFINITY}isValid(){return this.p0.x!=Number.POSITIVE_INFINITY&&this.p1.x!=Number.NEGATIVE_INFINITY&&this.p0.y!=Number.POSITIVE_INFINITY&&this.p1.y!=Number.NEGATIVE_INFINITY&&this.p0.z!=Number.POSITIVE_INFINITY&&this.p1.z!=Number.NEGATIVE_INFINITY}addPoint(A){A.x!=Number.POSITIVE_INFINITY&&A.x!=Number.NEGATIVE_INFINITY&&(A.x<this.p0.x&&(this.p0.x=A.x),A.x>this.p1.x&&(this.p1.x=A.x)),A.y!=Number.POSITIVE_INFINITY&&A.y!=Number.NEGATIVE_INFINITY&&(A.y<this.p0.y&&(this.p0.y=A.y),A.y>this.p1.y&&(this.p1.y=A.y)),A.z!=Number.POSITIVE_INFINITY&&A.z!=Number.NEGATIVE_INFINITY&&(A.z<this.p0.z&&(this.p0.z=A.z),A.z>this.p1.z&&(this.p1.z=A.z))}addBox3(A,g){g?(this.addPoint(g.transformVec3(A.p0)),this.addPoint(g.transformVec3(new Vec3(A.p0.x,A.p0.y,A.p1.z))),this.addPoint(g.transformVec3(new Vec3(A.p0.x,A.p1.y,A.p0.z))),this.addPoint(g.transformVec3(new Vec3(A.p1.x,A.p0.y,A.p0.z))),this.addPoint(g.transformVec3(new Vec3(A.p0.x,A.p1.y,A.p1.z))),this.addPoint(g.transformVec3(new Vec3(A.p1.x,A.p0.y,A.p1.z))),this.addPoint(g.transformVec3(new Vec3(A.p1.x,A.p1.y,A.p0.z))),this.addPoint(g.transformVec3(A.p1))):(this.addPoint(A.p0),this.addPoint(A.p1))}size(){return this.p1.distanceTo(this.p0)}diagonal(){return this.p1.subtract(this.p0)}center(){const A=this.p1.subtract(this.p0);return A.scaleInPlace(.5),A.addInPlace(this.p0),A}toMat4(){const A=this.p1.x-this.p0.x,g=this.p1.y-this.p0.y,I=this.p1.z-this.p0.z;return new Mat4(A,0,0,0,0,g,0,0,0,0,I,0,this.p0.x,this.p0.y,this.p0.z,1)}getBoundingSphere(){return new SphereType(this.center(),.5*this.diagonal().length())}intersectsBox(A){return!(A.max.x<this.min.x||A.min.x>this.max.x||A.max.y<this.min.y||A.min.y>this.max.y||A.max.z<this.min.z||A.min.z>this.max.z)}intersectsSphere(A){return(new Vec3).distanceTo(A.pos)<=A.radius*A.radius}intersectsPlane(A){let g,I;return A.normal.x>0?(g=A.normal.x*this.min.x,I=A.normal.x*this.max.x):(g=A.normal.x*this.max.x,I=A.normal.x*this.min.x),A.normal.y>0?(g+=A.normal.y*this.min.y,I+=A.normal.y*this.max.y):(g+=A.normal.y*this.max.y,I+=A.normal.y*this.min.y),A.normal.z>0?(g+=A.normal.z*this.min.z,I+=A.normal.z*this.max.z):(g+=A.normal.z*this.max.z,I+=A.normal.z*this.min.z),g<=-A.w&&I>=-A.w}clone(){return new Box3(this.p0.clone(),this.p1.clone())}toJSON(){return{p0:this.p0.toJSON(),p1:this.p1.toJSON()}}fromJSON(A){const g={x:MathFunctions.isNumeric(A.p0.x)?A.p0.x:Number.POSITIVE_INFINITY,y:MathFunctions.isNumeric(A.p0.y)?A.p0.y:Number.POSITIVE_INFINITY,z:MathFunctions.isNumeric(A.p0.z)?A.p0.z:Number.POSITIVE_INFINITY},I={x:MathFunctions.isNumeric(A.p1.x)?A.p1.x:Number.NEGATIVE_INFINITY,y:MathFunctions.isNumeric(A.p1.y)?A.p1.y:Number.NEGATIVE_INFINITY,z:MathFunctions.isNumeric(A.p1.z)?A.p1.z:Number.NEGATIVE_INFINITY};this.p0.fromJSON(g),this.p1.fromJSON(I)}toString(){return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON())}}class PlaneType{normal;w;constructor(A,g=0){this.normal=A instanceof Vec3?A:new Vec3,this.w=g}set(A,g,I,C){this.normal.set(A,g,I),this.w=C}divideScalar(A){this.normal.scaleInPlace(1/A),this.w/=A}distanceToPoint(A){return A.dot(this.normal)+this.w}normalizeInPlace(){const A=1/this.normal.length();this.normal.scaleInPlace(A),this.w*=A}clone(){return new PlaneType(this.normal.clone(),this.w)}toJSON(){return{normal:this.normal.toJSON(),w:this.w}}fromJSON(A){this.normal.fromJSON(A.normal),this.w=A.w}toString(){return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON())}}class Frustum{planes;constructor(A,g,I,C,e,i){this.planes=[A||new PlaneType,g||new PlaneType,I||new PlaneType,C||new PlaneType,e||new PlaneType,i||new PlaneType]}setFromMatrix(A){const g=A,I=this.planes;I[0].set(g.m03-g.m00,g.m13-g.m10,g.m23-g.m20,g.m33-g.m30),I[1].set(g.m03+g.m00,g.m13+g.m10,g.m23+g.m20,g.m33+g.m30),I[2].set(g.m03+g.m01,g.m13+g.m11,g.m23+g.m21,g.m33+g.m31),I[3].set(g.m03-g.m01,g.m13-g.m11,g.m23-g.m21,g.m33-g.m31),I[4].set(g.m03-g.m02,g.m13-g.m12,g.m23-g.m22,g.m33-g.m32),I[5].set(g.m03+g.m02,g.m13+g.m12,g.m23+g.m22,g.m33+g.m32),I.forEach((A=>A.normalizeInPlace()))}intersectsBox(A){const g=new Vec3,I=this.planes,{min:C,max:e}=A;for(let A=0;A<6;A++){const i=I[A];if(g.x=i.normal.x>0?e.x:C.x,g.y=i.normal.y>0?e.y:C.y,g.z=i.normal.z>0?e.z:C.z,i.distanceToPoint(g)<0)return!1}return!0}toJSON(){return{p0:this.planes[0].toJSON(),p1:this.planes[1].toJSON(),p2:this.planes[2].toJSON(),p3:this.planes[3].toJSON(),p4:this.planes[4].toJSON(),p5:this.planes[5].toJSON()}}fromJSON(A){this.planes[0].fromJSON(A.p0),this.planes[1].fromJSON(A.p1),this.planes[2].fromJSON(A.p2),this.planes[3].fromJSON(A.p3),this.planes[4].fromJSON(A.p4),this.planes[5].fromJSON(A.p5)}toString(){return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON())}}class RefCounted extends EventEmitter{#A;constructor(){if(super(),"RefCounted"==this.constructor.name)throw new Error("RefCounted should not be instantiated directly.");this.#A=[]}get refs(){return this.#A}numRefs(){return this.#A.length}addRef(A){if(!A)throw new Error("Error in RefCounted.addRef: Must provide a referer");return this.#A.push(A),!0}removeRef(A){if(!A)throw new Error("Error in RefCounted.removeRef: Must provide a referer");const g=this.#A.indexOf(A);if(-1==g)throw new Error("Error in RefCounted.removeRef: referer not found in refs list.");this.#A.splice(g,1),0==this.#A.length&&this.destroy()}destroy(){this.emit("destructing")}}let numBaseItems=0;class BaseItem extends EventEmitter{#g;#I=void 0;selected=!1;constructor(A=""){super(),this.#g=A,numBaseItems++}static getNumBaseItems(){return numBaseItems}get name(){return this.#g}set name(A){if(this.#g!=A){const g=this.#g;this.#g=A;const I=new NameChangedEvent(g,A);this.emit("nameChanged",I)}}get path(){return this.getPath()}get ownerItem(){return this.#I}getName(){return this.#g}setName(A){this.name=A}getPath(){return null==this.#I?[this.#g]:[...this.#I.getPath(),this.#g]}resolvePath(A,g=0){if(0==g&&("."!=A[0]&&A[0]!=this.#g||g++),".."==A[g]){if(this.#I)return this.#I.resolvePath(A,g+1);throw Error("this.#ownerItem is undefined")}if(g==A.length)return this}getOwner(){return this.#I}setOwner(A){this.#I=A}isSelected(){return this.selected}setSelected(A){this.selected=A;let g=new SelectedEvent(this.selected);this.emit("selectedChanged",g)}toJSON(A){return{type:this.getClassName(),name:this.#g}}fromJSON(A,g){A.name&&(this.#g=A.name)}readBinary(A,g){A.loadStr(),this.setName(A.loadStr())}toString(A){return JSON.stringify(this.toJSON(),null,2)}clone(A){throw new Error(this.constructor.name+" does not implement its clone method")}copyFrom(A,g){this.setName(A.getName())}}class ParameterOwner extends BaseItem{paramEventListenerIDs={};paramMapping={};params=[];deprecatedParamMapping={};constructor(A){super(A)}getNumParameters(){return this.params.length}get numParameters(){return this.params.length}getParameters(){return this.params}getParameterIndex(A){return this.paramMapping[A]}getParameterByIndex(A){return this.params[A]}hasParameter(A){return A in this.paramMapping}addParameterDeprecationMapping(A,g){this.deprecatedParamMapping[A]=g}getParameter(A){let g=this.paramMapping[A];if(null==g){const I=this.deprecatedParamMapping[A];if(!I)return null;console.warn(`Parameter name ${A} is now deprecated. Please use ${I} instead.`),g=this.paramMapping[I]}return this.params[g]}parameterValueChanged(A){this.emit("parameterValueChanged",A)}addParameter(A){return this.insertParameter(A,this.params.length)}insertParameter(A,g){const I=A.getName();null!=this.paramMapping[I]&&(console.warn("Replacing Parameter:"+I),this.removeParameter(I)),A.setOwner(this),this.paramEventListenerIDs[I]=A.on("valueChanged",(g=>{const I={param:A};for(const A in g)I[A]=g[A];this.parameterValueChanged(I)})),this.params.splice(g,0,A);for(let A=g;A<this.params.length;A++)this.paramMapping[this.params[A].getName()]=A;const C=new ParameterAddedEvent(I);return this.emit("parameterAdded",C),A}removeParameter(A){if(null==this.paramMapping[A])throw new Error("Unable to remove Parameter:"+A);const g=this.paramMapping[A];this.params[this.paramMapping[A]].off("valueChanged",this.paramEventListenerIDs[A]),this.params.splice(g,1),delete this.paramMapping[A];for(let A=g;A<this.params.length;A++)this.paramMapping[this.params[A].getName()]=A;const I=new ParameterRemovedEvent(A);this.emit("parameterRemoved",I)}replaceParameter(A){const g=A.getName();if(null==this.paramMapping[g])throw new Error("Unable to replace Parameter:"+g);const I=this.paramMapping[g];return this.removeParameter(g),this.insertParameter(A,I),A}resolvePath(A,g=0){if(0==g&&("."!=A[0]&&A[0]!=this.name||g++),".."==A[g]){if(this.ownerItem)return this.ownerItem.resolvePath(A,g+1);throw Error("this.ownerItem is undefined")}if(g==A.length)return this;const I=this.getParameter(A[g]);if(I)return g<A.length?I.resolvePath(A,g+1):I;throw new Error(`Unable to resolve path : [${A.toString()}] after: ${this.getName()} \nNo child or parameter called : "${A[g]}"`)}toJSON(A){const g=super.toJSON(A),I={};let C=0;for(const g of this.params){if(g.isDrivenByOperator())continue;const e=g.toJSON(A);e&&(I[g.getName()]=e,C++)}return C>0&&(g.params=I),g}fromJSON(A,g){if(super.fromJSON(A,g),A.params)for(const I in A.params){const C=A.params[I];if(C.paramPath)g?.resolvePath(C.paramPath,(A=>{this.replaceParameter(A)}),(()=>{console.warn("Unable to resolve shared parameter:"+C.paramPath)}));else{let A=this.getParameter(I);if(!A&&C.type&&C.name){if(A=Registry.constructClass(C.type),!A){console.error("Unable to construct prop:"+C.name+" of type:"+C.type);continue}A.setName(C.name),this.addParameter(A)}A&&A.fromJSON(C,g)}}}readBinary(A,g){super.readBinary(A,g),this.readBinaryParams(A,g)}readBinaryParams(A,g){if(g?.versions["zea-engine"].compare([0,0,3])>=0){const I=A.loadUInt32();for(let C=0;C<I;C++){const I=A.loadStr(),C=A.loadStr();let e=this.getParameter(C);if(!e){if(e=Registry.constructClass(I),!e){console.error("Unable to construct prop:"+C+" of type:"+I);continue}e.setName(C),this.addParameter(e)}e.readBinary(A,g)}}}clone(A){throw new Error(this.constructor.name+" does not implement its clone method")}copyFrom(A,g){if(!(A instanceof ParameterOwner))throw new Error("cannot copy from src");super.copyFrom(A,g);let I=A.getNumParameters();for(;I--;){const g=A.getParameterByIndex(I),C=this.getParameter(g.getName());C?C.copyFrom(g):this.addParameter(g.clone())}}}const getFileFolder=function(A){return A.substring(0,A.lastIndexOf("/"))+"/"},loadFile=function(A,g,I,C,e=void 0){try{const i=new XMLHttpRequest;i.responseType=g,i.addEventListener("timeout",(g=>{throw new Error("The request for "+A+" timed out.")})),i.addEventListener("error",(g=>{throw new Error("The request for "+A+": xhr.readyState:"+i.readyState)})),i.addEventListener("abort",(g=>{throw new Error("The request for "+A+": xhr.readyState:"+i.readyState)})),i.addEventListener("progress",(A=>{e&&e(A.total,A.loaded)})),i.addEventListener("loadend",(A=>{200==i.status?I(i):C(i.statusText)})),i.open("GET",A,!0),i.send()}catch(A){C(A)}},loadTextfile=function(A,g,I=void 0,C=void 0){loadFile(A,"text",(A=>{g(A.responseText)}),(g=>{if(null==I)throw new Error("Unable to XHR File:"+A);I(g)}),C)},loadJSONfile=function(A,g,I=void 0,C=void 0){loadFile(A,"json",(A=>{g(A.response,A)}),(g=>{if(null==I)throw new Error("Unable to XHR File:"+A);I(g)}),C)},loadXMLfile=function(A,g,I=void 0,C=void 0){loadFile(A,"document",(A=>{g(A.responseXML)}),(g=>{if(null==I)throw new Error("Unable to XHR File:"+A);I(g)}),C)},loadBinfile=function(A,g,I=void 0,C=void 0){loadFile(A,"arraybuffer",(A=>{g(A.response)}),(g=>{if(null==I)throw new Error("Unable to XHR File:"+A);I(g)}),C)};var unpackBase64Str="AGFzbQEAAAABrgIlYAF/AX5gA39/fwF/YAN/fn8AYAR/f39/AX9gAn9/AX9gAAF/YAJ/fwBgA39/fwBgAABgAX8Bf2AEf39/fwBgBn9/f39/fwBgBX9/f39/AGACf38BfGAFf39/f38Bf2AGf39/f39/AX9gCX9/f39/f39/fwF/YAh/f39/f35/fwF/YAJ+fgF/YAF/AGADf398AGAJf39/f39/f39/AGAKf39/f39/f39/fwBgAn9+AGAHf39/f39/fwF/YAp/f39/f39/f39/AX9gB39/f39/f38AYAt/f39/f39/f39/fwBgDX9/f39/f39/f39/f38AYAh/f39/f39/fwBgA39+fwF/YAN+f38Bf2ACfn8Bf2AGf3x/f39/AX9gAnx/AXxgA39/fwF8YAR/f398AAKID1cDZW52Bm1lbW9yeQIAgAIDZW52BXRhYmxlAXABvAK8AgNlbnYJdGFibGVCYXNlA38AA2Vudg5EWU5BTUlDVE9QX1BUUgN/AANlbnYIU1RBQ0tUT1ADfwADZW52BWFib3J0ABMDZW52DWVubGFyZ2VNZW1vcnkABQNlbnYOZ2V0VG90YWxNZW1vcnkABQNlbnYXYWJvcnRPbkNhbm5vdEdyb3dNZW1vcnkABQNlbnYIaW52b2tlX2kACQNlbnYJaW52b2tlX2lpAAQDZW52Cmludm9rZV9paWkAAQNlbnYLaW52b2tlX2lpaWkAAwNlbnYMaW52b2tlX2lpaWlpAA4DZW52Dmludm9rZV9paWlpaWlpABgDZW52EWludm9rZV9paWlpaWlpaWlpABkDZW52CGludm9rZV92ABMDZW52CWludm9rZV92aQAGA2VudgppbnZva2VfdmlpAAcDZW52C2ludm9rZV92aWlpAAoDZW52DGludm9rZV92aWlpaQAMA2Vudg1pbnZva2VfdmlpaWlpAAsDZW52Dmludm9rZV92aWlpaWlpABoDZW52EWludm9rZV92aWlpaWlpaWlpABYDZW52Emludm9rZV92aWlpaWlpaWlpaQAbA2VudhlfX19jeGFfYWxsb2NhdGVfZXhjZXB0aW9uAAkDZW52El9fX2N4YV9iZWdpbl9jYXRjaAAJA2VudhBfX19jeGFfZW5kX2NhdGNoAAgDZW52HF9fX2N4YV9maW5kX21hdGNoaW5nX2NhdGNoXzIABQNlbnYcX19fY3hhX2ZpbmRfbWF0Y2hpbmdfY2F0Y2hfMwAJA2VudhxfX19jeGFfZmluZF9tYXRjaGluZ19jYXRjaF80AAQDZW52FV9fX2N4YV9mcmVlX2V4Y2VwdGlvbgATA2VudgxfX19jeGFfdGhyb3cABwNlbnYHX19fbG9jawATA2VudgtfX19tYXBfZmlsZQAEA2VudhJfX19yZXN1bWVFeGNlcHRpb24AEwNlbnYLX19fc2V0RXJyTm8AEwNlbnYNX19fc3lzY2FsbDE0MAAEA2Vudg1fX19zeXNjYWxsMTQ1AAQDZW52DV9fX3N5c2NhbGwxNDYABANlbnYNX19fc3lzY2FsbDE4MwAEA2Vudg1fX19zeXNjYWxsMTk4AAQDZW52DF9fX3N5c2NhbGwyMAAEA2VudgtfX19zeXNjYWxsNgAEA2VudgxfX19zeXNjYWxsNjAABANlbnYMX19fc3lzY2FsbDgzAAQDZW52DF9fX3N5c2NhbGw5MQAEA2VudglfX191bmxvY2sAEwNlbnYeX19lbWJpbmRfZmluYWxpemVfdmFsdWVfb2JqZWN0ABMDZW52Fl9fZW1iaW5kX3JlZ2lzdGVyX2Jvb2wADANlbnYXX19lbWJpbmRfcmVnaXN0ZXJfY2xhc3MAHANlbnYjX19lbWJpbmRfcmVnaXN0ZXJfY2xhc3NfY29uc3RydWN0b3IACwNlbnYgX19lbWJpbmRfcmVnaXN0ZXJfY2xhc3NfZnVuY3Rpb24AHQNlbnYXX19lbWJpbmRfcmVnaXN0ZXJfZW12YWwABgNlbnYXX19lbWJpbmRfcmVnaXN0ZXJfZmxvYXQABwNlbnYZX19lbWJpbmRfcmVnaXN0ZXJfaW50ZWdlcgAMA2Vudh1fX2VtYmluZF9yZWdpc3Rlcl9tZW1vcnlfdmlldwAHA2VudhxfX2VtYmluZF9yZWdpc3Rlcl9zdGRfc3RyaW5nAAYDZW52HV9fZW1iaW5kX3JlZ2lzdGVyX3N0ZF93c3RyaW5nAAcDZW52Hl9fZW1iaW5kX3JlZ2lzdGVyX3ZhbHVlX29iamVjdAALA2VudiRfX2VtYmluZF9yZWdpc3Rlcl92YWx1ZV9vYmplY3RfZmllbGQAFgNlbnYWX19lbWJpbmRfcmVnaXN0ZXJfdm9pZAAGA2VudgZfYWJvcnQACANlbnYWX2Vtc2NyaXB0ZW5fbWVtY3B5X2JpZwABA2VudgdfZ2V0ZW52AAkDZW52CV9nZXRncm5hbQAJA2VudglfZ2V0cHduYW0ACQNlbnYIX2pzQ2xvc2UAEwNlbnYJX2pzQ3JlYXRlAAkDZW52B19qc09wZW4ACQNlbnYHX2pzUmVhZAABA2VudgdfanNTZWVrAAEDZW52CF9qc1dyaXRlAAEDZW52E19sbHZtX2VoX3R5cGVpZF9mb3IACQNlbnYKX2xvY2FsdGltZQAJA2VudgdfbWt0aW1lAAkDZW52FF9wdGhyZWFkX2dldHNwZWNpZmljAAkDZW52E19wdGhyZWFkX2tleV9jcmVhdGUABANlbnYNX3B0aHJlYWRfb25jZQAEA2VudhRfcHRocmVhZF9zZXRzcGVjaWZpYwAEA2VudgVfdGltZQAJA2VudhBpbnZva2VfaWlpaWlpamlpABkDZW52Cmludm9rZV9pamoADgNlbnYJaW52b2tlX2ppAAQDZW52Cmludm9rZV92aWoACgNlbnYLaW52b2tlX3ZpamkADANlbnYHX2pzVGVsbAAJA4QEggQTAQEGEwQJCRMGEwEBCQcAEwcGBwwJCgEEAQkGCQETCQkEBAQHBAETAQEJBxMHAwkGBhMIBgkTARMEBBMTAQEBIAkJExMGEwEDCQIEEwcEExMKBgEJDAsIBAwEBAEEBAcJBwkDChMGEAERDAAFCwwKBwMBBwEEBgEJAQEOAQkEABMAAQYHBBMTAQkTEwYTBhMJCgYMCgEABwcGBAcTBAQEAhcKDwQHBwoBExMTBQEDBAcJDgQECQEBAwQJASIJAQQJBAkGCQEBBwQHBBMBCQ4JBgMBEwYTAQwBBwcHBhMXBhMHBhMTEwkPBAkTEwQGFR0MBgYGBwkTCQkJAQEJAQAJBAoTCQkJEwYGBBMEAAIBARMMCgQOGRYVCBQSERIGEA4BDRsWGgsMCiQHBhMZGA8OAwEEIwkBCgwLBwETCQkIEwYICgwLCgwLAQgEBh0BAQkEBh0EBgYGCQMEBAEEAQEBAQEEBAQJAwMHBAEhIB8DAwQBBAQBBAkBBQEBCQQJEwgECQEEHhMDFA0FCgUHBBMFAQcEBg4MBRMJEwcIAwYPExMTBgcHBgMEAQYBBgMJCQYTBAYTBhMEBgQTARMTBh4GFw8BBwkDBgQDBwcBBAQGBgYBCQkGEwYGAwcTEwoTBhMTBwoGEwcGBwYTBwcTFQcHExMTBgkGBgQLEwQJBAMBEwgJBhoFfwEjAQt/ASMCC38BQQALfwFBAAt/AUEACweXBSUYX19HTE9CQUxfX3N1Yl9JX2JpbmRfY3BwANEDGl9fR0xPQkFMX19zdWJfSV9icmlkZ2VfY3BwAO0DF19fR0xPQkFMX19zdWJfSV9jcmNfY3BwANIEGl9fR0xPQkFMX19zdWJfSV9nbG9iYWxfY3BwAOoCEF9fX2N4YV9jYW5fY2F0Y2gAhwMWX19fY3hhX2lzX3BvaW50ZXJfdHlwZQCGAxFfX19lcnJub19sb2NhdGlvbgDKAw5fX19nZXRUeXBlTmFtZQDPAwVfZnJlZQBSB19tYWxsb2MAbAtkeW5DYWxsX2RpaQCFAwlkeW5DYWxsX2kAnwIKZHluQ2FsbF9paQCEAwtkeW5DYWxsX2lpaQCDAwxkeW5DYWxsX2lpaWkAggMNZHluQ2FsbF9paWlpaQCBAw5keW5DYWxsX2lpaWlpaQCAAw9keW5DYWxsX2lpaWlpaWkA/wISZHluQ2FsbF9paWlpaWlpaWlpAP4CEWR5bkNhbGxfaWlpaWlpamlpAOcCC2R5bkNhbGxfaWpqAOYCCmR5bkNhbGxfamkA5QIJZHluQ2FsbF92AP0CCmR5bkNhbGxfdmkA/AILZHluQ2FsbF92aWkA+wIMZHluQ2FsbF92aWlkAPoCDGR5bkNhbGxfdmlpaQD5Ag1keW5DYWxsX3ZpaWlpAPgCDmR5bkNhbGxfdmlpaWlpAPcCD2R5bkNhbGxfdmlpaWlpaQD2AhJkeW5DYWxsX3ZpaWlpaWlpaWkA9QITZHluQ2FsbF92aWlpaWlpaWlpaQD0AgtkeW5DYWxsX3ZpagDkAgxkeW5DYWxsX3ZpamkA4wILc2V0VGVtcFJldDAA3AIIc2V0VGhyZXcA7wIKc3RhY2tBbGxvYwDTBAnKBAEAIwALvALzAtoDvgHoA+ED3QPbA/4BvgG+AYEBzQOPA44DX8YEfKkD0wOlBM0EuwKmAcoCywK4ApYC0QKWArIBcVjqA58CxwKBAYEBgQGBAYEBgQGBAaoB8QHdAu8BvgJqc5kE0gLbAs4E8AHMBLoCyQTSA9UD5APXAd8DnAKaAtcBnAKaAtcBuwOi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DiAAQThqIQ8gAEGQAWohECAAQfAAaiELIABBzABqIREgAEEIaiESIABBBGohDAJAAkADQCAERQRAIAMhAgwCCyANKAIAIQcgACwAAARAIAEgEigCACAMKAIAEFMaIAwoAgAhAiAMQQA2AgAFIAopAwAiEyAErVMhCCATpyECIAgEfyACBSAEIgILBEAgDiwAAARAIAksAAAEQCACIAIgBmpBD3FrIgNBAEoEQCADIQILIAhFBEAgBCECCwsLIAcoAgRFDQQgByAFIAIgBygCACgCDEEfcUHKAGoRAQAhAiAHQbCnAWohCCAPKAIAIgMEfyADBSAIC0GJwQBqLAAABEAgECAFIAIQqwILBSADIQILCyALIAspAwAgAqwiE3w3AwAgAiAGaiEGIAogCikDACATfSITNwMAIBNCAFEgDiwAAEEAR3FFDQEgAgRAIAZBD3FFIAksAABFcg0CCyAFIAJqIQUgBCACayEEIAcgAEEBIBEoAgAQtQEEQCACIQMMAQsLIABBAToAUkF/DwsgDSgCACIDBEAgCykDACADQYi8A2opAwB8IRQgACwAKARAIAApA1ghFSAAKQOIASITQgBRBEAgFSETBSAAKQOAASAUfCEUCyADQazzAGooAgAhAyATIBRTBH9B5AAFIBNCAFEEf0EABSAUQuQAfiATf6cLCyEFIANBzIQDaiwAAEUEQCAFIABByABqIgMoAgBHBEAgAyAFNgIACwsLCyACQX9GBEBBfyEGBSAJLAAABEAgACgCRCABIAYQ6gELCxCFASAGDwtBfwtiAQF/IABFBEBBASEACwJAAkACQANAIAAQbCIBDQJBsLwDQbC8AygCACIBNgIAIAFFDQEgAUEHcUGOAWoRCAAMAAsAC0EEEBQiAEHEJTYCACAAQdgKQQoQGwwBCyABDwtBAAtTAQF/IwQhACMEQeAgaiQEIABBgCBqIgMgAjYCACAAQYAIIAEgAxCAAhogAEGQIGoiAUEANgJEIAFBATYCSCABQUBrQQE2AgAgASAANgIAIAAkBAt6AgV/AX4gAEEYaiIDKAIAIgEgACgCFCIETwRAQgAPCyAAKAIAIQUgASEAAkADQCADIABBAWoiATYCACAFIABqLQAAIgBB/wBxrSACrYYgBnwhBiAAQYABcUUNASACQQdqIQIgASAESQRAIAEhAAwBBUIAIQYL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IAQgAGogAy0AAHM6AAAgAEEBaiIAQYAERw0ACyACQfwDagUgAgshAyAFIQAgA0EANgIAIAEsAIAEBH8gACABQYAEEFMaIwQhASMEQRBqJARBFCABECUhAyABJAQgA0HLAGohBEEAIQEDQCAAIAFqIgMgBCABaiADLQAAczoAACABQQFqIgFBgARHDQALIABB/ANqBSAAC0EANgIAIAIgABBzIQEgAkEAOgAAIAJBADoAASACQQA6AAIgAkEAOgADIAJBADoABCACQQA6AAUgAkEAOgAGIAJBADoAByACQQA6AAggAkEAOgAJIAJBADoACiACQQA6AAsgAkEAOgAMIAJBADoADSACQQA6AA4gAkEAOgAPIAJBADoAECACQQA6ABEgAkEAOgASIAJBADoAEyACQQA6ABQgAkEAOgAVIAJBADoAFiACQQA6ABcgAkEAOgAYIAJBADoAGSACQQA6ABogAkEAOgAbIAJBADoAHCACQQA6AB0gAkEAOgAeIAJBADoAHyACQQA6ACAgAkEAOgAhIAJBADoAIiACQQA6ACMgAkEAOgAkIAJBADoAJSACQQA6ACYgAkEAOgAnIAJBADoAKCACQQA6ACkgAkEAOgAqIAJBADoAKyACQQA6ACwgAkEAOgAtIAJBADoALiACQQA6AC8gAkEAOgAwIAJBADoAMSACQQA6ADIgAkEAOgAzIAJBADoANCACQQA6ADUgAkEAOgA2IAJBADoANyACQQA6ADggAkEAOgA5IAJBADoAOiACQQA6ADsgAkEAOgA8IAJBADoAPSACQQA6AD4gAkEAOgA/IAJBQGtBADoAACACQQA6AEEgAkEAOgBCIAJBADoAQyACQQA6AEQgAkEAOgBFIAJBADoARiACQQA6AEcgAkEAOgBIIAJBADoASSACQQA6AEogAkEAOgBLIAJBADoATCACQQA6AE0gAkEAOgBOIAJBADoATyACQQA6AFAgAkEAOgBRIAJBADoAUiACQQA6AFMgAkEAOgBUIAJBADoAVSACQQA6AFYgAkEAOgBXIAJBADoAWCACQQA6AFkgAkEAOgBaIAJBADoAWyACQQA6AFwgAkEAOgBdIAJBADoAXiACQQA6AF8gAkEAOgBgIAJBADoAYSACQQA6AGIgAkEAOgBjIAJBADoAZCACQQA6AGUgAkEAOgBmIAJBADoAZyACQQA6AGggAkEAOgBpIAJBADoAaiACQQA6AGsgAkEAOgBsIAJBADoAbSACQQA6AG4gAkEAOgBvIAJBADoAcCACQQA6AHEgAkEAOgByIAJBADoAcyACQQA6AHQgAkEAOgB1IAJBADoAdiACQQA6AHcgAkEAOgB4IAJBADoAeSACQQA6AHogAkEAOgB7IAJBADoAfCACQQA6AH0gAkEAOgB+IAJBADoAfyAAQQA6AAAgAEEAOgABIABBADoAAiAAQQA6AAMgAEEAOgAEIABBADoABSAAQQA6AAYgAEEAOgAHIABBADoACCAAQQA6AAkgAEEAOgAKIABBADoACyAAQQA6AAwgAEEAOgANIABBADoADiAAQQA6AA8gAEEAOgAQIABBADoAESAAQQA6ABIgAEEAOgATIABBADoAFCAAQQA6ABUgAEEAOgAWIABBADoAFyAAQQA6ABggAEEAOgAZIABBADoAGiAAQQA6ABsgAEEAOgAcIABBADoAHSAAQQA6AB4gAEEAOgAfIABBADoAICAAQQA6ACEgAEEAOgAiIABBADoAIyAAQQA6ACQgAEEAOgAlIABBADoAJiAAQQA6ACcgAEEAOgAoIABBADoAKSAAQQA6ACogAEEAOgArIABBADoALCAAQQA6AC0gAEEAOgAuIABBADoALyAAQQA6ADAgAEEAOgAxIABBADoAMiAAQQA6ADMgAEEAOgA0IABBADoANSAAQQA6ADYgAEEAOgA3IABBADoAOCAAQQA6ADkgAEEAOgA6IABBADoAOyAAQQA6ADwgAEEAOgA9IABBADoAPiAAQQA6AD8gAEFAa0EAOgAAIABBADoAQSAAQQA6AEIgAEEAOgBDIABBADoARCAAQQA6AEUgAEEAOgBGIABBADoARyAAQQA6AEggAEEAOgBJIABBADoASiAAQQA6AEsgAEEAOgBMIABBADoATSAAQQA6AE4gAEEAOgBPIABBADoAUCAAQQA6AFEgAEEAOgBSIABBADoAUyAAQQA6AFQgAEEAOgBVIABBADoAViAAQQA6AFcgAEEAOgBYIABBADoAWSAAQQA6AFogAEEAOgBbIABBADoAXCAAQQA6AF0gAEEAOgBeIABBADoAXyAAQQA6AGAgAEEAOgBhIABBADoAYiAAQQA6AGMgAEEAOgBkIABBADoAZSAAQQA6AGYgAEEAOgBnIABBADoAaCAAQQA6AGkgAEEAOgBqIABBADoAayAAQQA6AGwgAEEAOgBtIABBADoAbiAAQQA6AG8gAEEAOgBwIABBADoAcSAAQQA6AHIgAEEAOgBzIABBADoAdCAAQQA6AHUgAEEAOgB2IABBADoAdyAAQQA6AHggAEEAOgB5IABBADoAeiAAQQA6AHsgAEEAOgB8IABBADoAfSAAQQA6AH4gAEEAOgB/IAAkBCABRQslAQF/IAIgABBZa0F/aiIDQQBMBEAgAA8LIAAgASADELIDGiAACxYAIABBAEHAywQQVBogAEGowAJqEFwLbQEEfyAAKAIUIABBGGoiBCgCACIFayIDIAJJIgYEfyADBSACIgMLBEAgASAAKAIAIAVqIAMQUxoLIAZFBEAgBCAEKAIAIANqNgIAIAMPCyABIANqQQAgAiADaxBUGiAEIAQoAgAgA2o2AgAgAwueAwEIfyMEIQQjBEEgaiQEIARBCGohBSABQQA2AgAgBEIANwMAIARBEGoiAyAANgIAAkACQAJAAkACQAJAIAEgAyACIAQQuQNBf2sOAgABAgsMAgsgACwAAA0BQQEhAAwCC0EBIQAMAQsgAkEBSwRAQQAhAwJAAkADQCAAIAhqIgYsAABFDQEgBUIANwMAIAEgA0ECdGoiCSAGQawjKAIAKAIABH9BBAVBAQsgBRCKAkF/RgR/IAYsAAAiB0EATg0DIApFBEAgCUH+/wM2AgAgA0EBaiIDIAJPDQQgBiwAACEHCyABIANBAnRqIAdB/wFxQYDAA3I2AgBBASEKIAhBAWoFIAVCADcDACAGQawjKAIAKAIABH9BBAVBAQsgBRCIAiIHQQFKBH8gBwVBAQsgCGoLIQggA0EBaiIDIAJJDQBBACEADAULAAsgASADQQJ0akEANgIAQQEhAAwDC0EAIQAFQQAhAAwBCwwBCyACRQRAIAQkBCAAQf8BcUEARw8LCyABIAJBf2pBAnRqQQA2AgAgBCQEIABB/wFxQQBHC/EBAgN/AX4jBCEBIwRB0ABqJAQgAEGtvANqLAAABEAgASQEQQAPCyAAQYi8A2oiAyAAIAAoAgAoAhRBB3FBhgFqEQAAIgQ3AwACQAJAAkACQAJAAkAgAEGYvANqKAIAQQFrDgMAAQIDCyAAENcCIQIMAwsgABDWAiECDAILIAAQ1QIhAgwBCwwBCyACBEAgAykDACEEBSABJARBAA8LCyAAQZC8A2opAwAgBFUEQCABJAQgAg8LIAFBADYCRCABQRg2AkggAUFAa0EBNgIAIAEgAEEYajYCACAAQay8A2pBAToAAEGs9QJBAxBkIAEkBEEAC9cBAgZ/AX4jBCEDIwRBoAJqJAQgA0EgaiEGIAMhBSAAQSBqIgMpAwAhCSADIAkgAq18NwMAIAJFBEAgBUEgEFsgBkGAAhBbIAUkBA8LIABBKGohByAAQSxqIQggCadBP3EhAwNAIAJBwAAgA2siBEsEfyAEBSACIgQLQcAARgRAIAcgATYCAAUgByAINgIAIABBLGogA2ogASAEEFMaCyAEIANqIgNBwABGBEAgABDhAUEAIQMLIAEgBGohASACIARrIgINAAsgBUEgEFsgBkGAAhBbIAUkBAt4AQF/IABB9A02AgAgAEGo8wBqLAAABEAgAEGs8wBqKAIAIgEEQCABEHkgARBSCwsgAEGQ7gJqKAIAIgEEQCABEFILIABB0OcBaigCACIBBEAgARBSCyAAQdjzAGoQtQIgAEHw8QBqEIQBIABBnMAAahCOASAAEHALBgBBEBAACwgAQQUQAEEACwgAQQIQAEEACzABAX8jBCECIwRBEGokBCACIAE2AgBB8CAoAgAiASAAIAIQzgEaQQogARCrAxoQOQujAgEEfyABRQRAIABBvM0DakEANgIAIABByM0DakEANgIAIABBnM0DaiIEKAIAIgEEQCAAQZjNA2ohBQNAIAUoAgAgAkECdGooAgAiAwRAIAMoAjwiAQRAIAEQUgsgAygCLCIBBEAgARBSCyADKAIUIgEEQCABEFILIAMQUiAEKAIAIQELIAJBAWoiAiABSQ0ACwsgBEEANgIACyAAQazNA2oiAygCACICRQRAIANBADYCAA8LIABBqM0DaiEEQQAhASACIQADQCAEKAIAIAFBAnRqKAIAIgIEQCACKAI8IgAEQCAAEFILIAIoAiwiAARAIAAQUgsgAigCFCIABEAgABBSCyACEFIgAygCACEACyABQQFqIgEgAEkNAAsgA0EANgIAC0gBAX8gAEFAaygCACIBBEAgARCOASABEFILIAAoAkQiAQRAIAEQjgEgARBSCyAAQagBahCNASAAQZwBahCNASAAQZABahCNAQsYAEG39QIsAABFBEAPC0Gs9QJB/wEQ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TBogACABEKwDIgALQQEBfwJAIAAQWSICQX9KBEAgACACQQJ0aiECA0AgAigCACABRg0CIAJBfGoiAiAATw0AQQAhAgsFQQAhAgsLIAILZAEDfwJAIAIEQCABIQMgACEBA0AgAygCACIFRQ0CIANBBGohAyABQQRqIQQgASAFNgIAIAJBf2oiAgRAIAQhAQwBBSAEIQFBACECCwsFIAAhAUEAIQILCyABQQAgAhCwAxogAAvCBAEKfyMEIQYjBEGAQGskBCAAIQcDQAJAA0AgACIEKAIAIQgCQAJAA0ACQAJAAkACQCAIDjAAAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgECCyAHIQAMBwsMAQsMAgsgBEEEaiIAKAIAIghBLkYNAiAAIQQMAAsACyAEQQRqIQAMAQsgBCgCCEEuRw0ACyAEQRBqIQggBCgCDEEvRgRAIAghBwsMAQsLAkACQANAAkACQAJAAkACQCAAKAIAIgcOXQACAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAQILDAMLIAAoAgRB3ABGBEAgAEEIakHcABB0IgIEQCACQQRqQdwAEHQiA0EEaiECIAMEfyACBSAAIgILIQMFIAAiAyECCwUgACIDIQILQQ0hBQwBCyAAIgohCSAHIQsLA0AgBUENRgRAQQAhBSADIQogAiEJIAMoAgAhCwsCQAJAAkAgC0Euaw4CAAECCyAKQQRqIQMgCSECQQ0hBQwCCyAKQQRqIgIhA0ENIQUMAQsLIAkgAEYNAiAJIQAMAQsLDAELIAAoAgBBLkYEQCAAKAIEQS5GBEAgAEEIaiICKAIARQRAIAIhAAsLCwsgAUUEQCAGJAQgAA8LIAYgAEGAEBBvGiABIAYQahogBiQEIAALEwAgAAR/IAAgAUEAEJEBBUEACwvaAwMBfwF+AXwCQCABQRRNBEACQAJAAkACQAJAAkACQAJAAkACQAJAIAFBCWsOCgABAgMEBQYHCAkKCyACKAIAQQNqQXxxIgEoAgAhAyACIAFBBGo2AgAgACADNgIADAsLIAIoAgBBA2pBfHEiASgCACEDIAIgAUEE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ACEEQQAhAwNAIAAgDCADrXxBACAAKAIAKAIQQQNxQbgCahECACAIKAIAQQFGBEAgBiAJNgIACyAHLAAABEAgB0EAOgAACyAGKAIAIAEgAiADayIFQYAESQR/IAUFQYAECxBBIgVBf0YEf0GABAUgBQsgBGohBCADQYAEaiIDIAJJDQALIAQPC0Gs9QIgAxC6BEF/DwsgBAuMAgEDfyABKAIAIQIgAEIANwIAIABBADYCCCACQe////8DSwRAIAAQigELIAFBBGohBAJAIAJBAkkEQCAAIAI6AAsgAgRAIAAhAwUgACACQQJ0akEANgIADwsFIAJBBGpBfHEiAUH/////A00EQCAAIAFBAnQQXyIDNgIAIAAgAUGAgICAeHI2AgggACACNgIEDAILQQgQFCEAQQAkBUEXIABBzO4AEA0jBSEBQQAkBSABQQFxBEAQFyEBIAAQGiABEB4FIABB7CU2AgAgAEH4CkEMEBsLCwtBACQFQR0gAyAEIAIQBxojBSEAQQAkBSAAQQFxBEBBABAYIgAQWgUgAyACQQJ0akEANgIACwsPACABIAAoAgBqIAI2AgALDQAgASAAKAIAaigCAAvQBgECfyAAKAIAIgEEQCABEFIgAEEANgIACyAAQQRqIgEoAgAiAgRAIAIQUiABQQA2AgALIABBCGoiASgCACICBEAgAhBSIAFBADYCAAsgAEEMaiIBKAIAIgIEQCACEFIgAUEANgIACyAAQRBqIgEoAgAiAgRAIAIQUiABQQA2AgALIABBFGoiASgCACICBEAgAhBSIAFBADYCAAsgAEEYaiIBKAIAIgIEQCACEFIgAUEANgIACyAAQRxqIgEoAgAiAgRAIAIQUiABQQA2AgALIABBIGoiASgCACICBEAgAhBSIAFBADYCAAsgAEEkaiIBKAIAIgIEQCACEFIgAUEANgIACyAAQShqIgEoAgAiAgRAIAIQUiABQQA2AgALIABBLGoiASgCACICBEAgAhBSIAFBADYCAAsgAEEwaiIBKAIAIgIEQCACEFIgAUEANgIACyAAQTRqIgEoAgAiAgRAIAIQUiABQQA2AgALIABBOGoiASgCACICBEAgAhBSIAFBADYCAAsgAEE8aiIBKAIAIgIEQCACEFIgAUEANgIACyAAQUBrIgEoAgAiAgRAIAIQUiABQQA2AgALIABBxABqIgEoAgAiAgRAIAIQUiABQQA2AgALIABByABqIgEoAgAiAgRAIAIQUiABQQA2AgALIABBzABqIgEoAgAiAgRAIAIQUiABQQA2AgALIABB0ABqIgEoAgAiAgRAIAIQUiABQQA2AgALIABB1ABqIgEoAgAiAgRAIAIQUiABQQA2AgALIABB2ABqIgEoAgAiAgRAIAIQUiABQQA2AgALIABB3ABqIgEoAgAiAgRAIAIQUiABQQA2AgALIABB4ABqIgEoAgAiAgRAIAIQUiABQQA2AgALIABB5ABqIgEoAgAiAgRAIAIQUiABQQA2AgALIABB6ABqIgEoAgAiAgRAIAIQUiABQQA2AgALIABB7ABqIgEoAgAiAgRAIAIQUiABQQA2AgALIABB8ABqIgEoAgAiAgRAIAIQUiABQQA2AgALIABB9ABqIgEoAgAiAgRAIAIQUiABQQA2AgALIABB+ABqIgEoAgAiAgRAIAIQUiABQQA2AgALIABB/ABqIgAoAgAiAUUEQA8LIAEQUiAAQQA2AgAL6AoBGn8jBCECIwRBEGokBCACQQhqIRIgAiEJIABB4ABqIgooAgAgAEHkAGoiDCgCACICayAAQdDNA2oiDSgCACIDcSETIABBNGohBwJAAkAgAEE4aiILKAIARQ0AIABBHGohDiAAQRhqIRQgAEEgaiEXIABBFGohDyAAQczNA2ohGCAAQaiYAWohFSAAQaSWAWohECAAQaiWAWohESAAQbqYAWohGSAAQcCYAWohFiATIQECQAJAAkACQANAAkAgBygCACIGIAVBBHRqIhosAABBCEcEQCAGIAVBBHRqKAIEIQQgBiAFQQR0akENaiIILAAABEAgBCAMKAIAayADcSATSw0CIAhBADoAAAwCCyAGIAVBBHRqKAIIIQYgAyAEIAJrcSABSQRAIAQgAkcEQCAAIAIgBBCzASAEIQIgCigCACAEayANKAIAIgNxIQELIAYgAUsNBCAGBEAgDigCACEBIBQgBjYCACABIAZJBEAgFygCACICQQBHIAYgAktxBH8gCSACNgIAQaz1AkHYGyAJEGBBrPUCEFYgDigCACEBIBQoAgAFIAYLIQIgDygCACACIAFBIGogAUECdmoiAUsEfyACBSABIgILEFciAUUEQEGs9QIQVgsgDyABNgIAIA4gAjYCAAsgDygCACEBAkAgAyAGIARqcSIDQX9qIARJBEAgGCgCACAEayEIIBUsAABFBEAgASAQKAIAIARqIAgQUxogASAIaiAQKAIAIAMQUxoMAgsgCARAQQAhAgNAIAEgAmogESACIARqEHUsAAA6AAAgAkEBaiICIAhHDQALIANFDQILIAEgCGohBEEAIQIDQCAEIAJqIBEgAhB1LAAAOgAAIAJBAWoiAiADRw0ACwUgFSwAAARAQQAhAgNAIAEgAmogESACIARqEHUsAAA6AAAgAkEBaiICIAZHDQALBSABIBAoAgAgBGogBhBTGgsLCyAAIAEgBiAaEPgDIQIgBygCACAFQQR0akEIOgAAIAIEQCAAKAIAIAIgBhBlCyAZQQE6AAAgFiAWKQMAIAatfDcDACAKKAIAIAMiAmsgDSgCACIDcSEBCwsLCyAFQQFqIgUgCygCACIESQ0AQQAhASAEIQMMAgsACyAMIAI2AgAgBSALKAIAIgNJBEAgBygCACEBA0AgASAFQQR0aiwAAEEIRwRAIAEgBUEEdGpBADoADQsgBUEBaiIFIANJDQBBASEGDAMLAAVBASEBCwsgAwRAIAEhBgwBBSABRQ0DCwwBC0EAIQVBACEBQQEhBANAIARFBEAgBygCACIDIAEgBWtBBHRqIgQgAyABQQR0aiIDKQIANwIAIAQgAykCCDcCCCALKAIAIQMLIAUgBygCACABQQR0aiwAAEEIRmoiBUUhBCABQQFqIgEgA0kNAAsgBEUEQCAAQTxqIgQoAgAhASALIAMgBWsiAzYCACABIANJBEAgAEFAaygCACIFQQBHIAMgBUtxBEAgEiAFNgIAQaz1AkHYGyASEGBBrPUCEFYgBCgCACEBIAsoAgAhAwsgBygCACADIAFBIGogAUECdmoiAUsEfyADBSABIgMLQQR0EFciAUUEQEGs9QIQVgsgByABNgIAIAQgAzYCACAGDQIMAwsLIAZFDQELIAooAgAhAgwBCyAAIAIgCigCABCzASAMIAooAgAiAjYCAAsgDSgCACEDIABBoJYBaiIFIABBzM0DaigCACIAQYCAgAJJBH8gAAVBgICAAgsgAmogA3EiATYCACAMKAIAIQAgASACRwRAIAAgAkYEQCAJJAQPCyADIAAgAmtxIAMgASACa3FPBEAgCSQEDwsLIAUgADYCACAJJAQLQQECfyAAQRBqIgQoAgAiAyAAKAIETwRAQQAPCyAEIANBAWogACgCACADQQJ0aiIAEFlqNgIAIAEgACACEG8aQQELvQwBEH8CQCMEIQYjBEGwA2okBCAGQZQDaiEJAkAgAEEEaiIFKAIAIgQgAEHoAGoiCCgCACIBQWdqSgRAIAEgBGsiAkEASA0CIABBBGohAyAEQYCAAUoEQCACQQBKBEAgACgCECIBIAEgBGogAhBdGgsgA0EANgIAIAggAjYCAAUgASECCyAAKAIAIAAoAhAgAmpBgIACIAJrEF4hAiAIKAIAIgQgAmohASACQQBKBEAgCCABNgIAIAAgAUFiajYCbAwCCyAAIARBYmo2AmwgAkF/Rg0CCwsgBUEAIABBCGoiCygCAGtBB3EQVSAAQcjMA2ohAiAFEFgiAUGAgAJxBEAgAkEBNgIAIABBwLECaiAAIABBsMkDahCMBCEAIAYkBCAADwsgAkEANgIAIABBuLECakEANgIAIABBvLECakEANgIAIAFBgIABcUUEQCAAQbTJA2pBAEGUAxBUGgsgBUECEFVBACECA0AgBRBYQQx2IQEgBUEEEFUCQAJAIAFB/wFxQQ9GBEAgBRBYQQx2Qf8BcSEEIAVBBBBVIAQEQCAEQQJqIQEgCSACakEAQQBBfiAEayIEIAJBbGoiA0sEfyAEBSADC2sQVBoDQCABQX9qIQQgAkEBaiIDQRRJIAFBAUpxBEAgBCEBIAMhAgwBCwsFQQ8hAQwCCwUgAUH/AXEhAQwBCwwBCyAJIAJqIAE6AAALIAJBAWoiAkEUSQ0AC0EAIAkgAEG0+ABqIg1BFBBoIABBBGohDiAAQRBqIQkgAEHsAGohDCAAQbj5AGohDyAGQX9qIRBBACECQQAhBAJAAkADQAJAAkACQAJAA0AgBSgCACIHIAgoAgAiA0F7akoEQCADIAdrIgFBAEgNCCAHQYCAAUoEQCABQQBKBEAgCSgCACIDIAMgB2ogARBdGgsgDkEANgIAIAggATYCAAUgAyEBCyAAKAIAIAkoAgAgAWpBgIACIAFrEF4hASAIKAIAIgcgAWohAyABQQBKBEAgCCADNgIAIAwgA0FiajYCAAUgDCAHQWJqNgIAIAFBf0YNCQsLIAkoAgAiASAFKAIAIgNBAWpqLQAAQQh0IAEgA2otAABBEHRyIAEgA0ECamotAAByQQggCygCACIKa3ZB/v8DcSIHIABBuPgAaiAPKAIAIgFBAnRqKAIASQR/IAUgCiAAQbz5AGogB0EQIAFrdiIBai0AAGoiB0EDdiADajYCACALIAdBB3E2AgAgAEG8gQFqIAFBAXRqBQNAAkAgAUEBaiIBQQ9PBEBBDyEBDAELIAcgAEG4+ABqIAFBAnRqKAIATw0BCwsgBSABIApqIgpBA3YgA2o2AgAgCyAKQQdxNgIAIABBvJEBaiAHIAAgAUECdGpBtPgAaigCAGtBECABa3YgAEH4+ABqIAFBAnRqKAIAaiIBIA0oAgBPBH9BAAUgAQtBAXRqCyIBLgEAIgFB//8DcUEQSA0BIAFB//8DcUESTg0DIAUQWCEDIAFBEEYEfyAFQQMQVSADQQ12QQNqBSAFQQcQVSADQQl2QQtqCyEBIARFDQAMAgsACyAGIAJqIABBtMkDaiACai0AACABQf//A3FqQQ9xOgAAIAJBAWohAgwCCyACQZQDTg0DIBAgAmosAAAhBwNAIAFBf2ohAyAGIAJqIAc6AAAgAkEBaiEEIAJBkwNIIAFBAUpxBEAgBCECIAMhAQwBBSAEIQILCwwBCyAFEFghBCABQRJGBH8gBUEDEFUgBEENdkEDagUgBUEHEFUgBEEJdkELagshASACQZQDTg0CIAYgAmpBAEEAIAJB7HxqIgRBfiABayABQX9zIgNBfkoEfyADBUF+C2siA0sEfyAEBSADC2sQVBoDQCABQX9qIQMgAkEBaiEEIAJBkwNIIAFBAUpxBEAgAyEBIAQhAgwBBSAEIQILCwsgAkEASiEEIAJBlANIDQALCyAAQczMA2pBAToAACAFKAIAIAgoAgBKDQFBACAGIABBhAFqQasCEGhBACAGQasCaiAAQfAeakE8EGhBACAGQecCaiAAQdw8akEREGhBACAGQfgCaiAAQcjaAGpBHBBoIABBtMkDaiAGQZQDEFMaIAYkBEEBDwtBACEAIAYkBEEADwsgBiQEQQALpAsBB38gAEEEaiIDEFghBgJ/IABB0K4BaiIHKAIAIgFB/+sBSwR/QYD+AyAGQfD/A3EiBEsEfyADQQgQVUEIIQFBAAVBCCEBA0AgAUEBaiEBIAJBAWoiBUECdEGwGGooAgAgBE0EQCAFIQIMAQsLIAMgARBVIAJBAnRBsBhqKAIACyECIAQgAmtBECABa3YhAiABQQJ0QcgYagUgAUH/uwFLBEBBgBAgBkHw/wNxIgRLBH8gA0EGEFVBBiEBQQAFQQYhAQNAIAFBAWohASACQQFqIgVBAnRB/BhqKAIAIARNBEAgBSECDAELCyADIAEQVSACQQJ0QfwYaigCAAshAiAEIAJrQRAgAWt2IQIgAUECdEGYGWoMAgsgAUH/6wBLBEBBgCAgBkHw/wNxIgRLBH8gA0EFEFVBBSEBQQAFQQUhAQNAIAFBAWohASACQQFqIgVBAnRBsBZqKAIAIARNBEAgBSECDAELCyADIAEQVSACQQJ0QbAWaigCAAshAiAEIAJrQRAgAWt2IQIgAUECdEHQFmoMAgsgBkHw/wNxIQQgAUH/G0sEfyAEQYDAACAESwR/IANBBRBVQQUhAUEABUEFIQEDQCABQQFqIQEgAkEBaiIFQQJ0QYQXaigCACAETQRAIAUhAgwBCwsgAyABEFUgAkECdEGEF2ooAgALIgJrQRAgAWt2IQIgAUECdEGkF2oFIARBgIACIARLBH8gA0EEEFVBBCEBQQAFQQQhAQNAIAFBAWohASACQQFqIgVBAnRB2BdqKAIAIARNBEAgBSECDAELCyADIAEQVSACQQJ0QdgXaigCAAsiAmtBECABa3YhAiABQQJ0QfwXagsLCyIBKAIAIAJqQf8BcSEBIABB7K4BaiIFKAIABEAgBkH/H0sgAUVxBH9BgAIiAQUgAQtBf2ohAiABBEAgAiEBBSADEFghBCADQQEQVSAEQYCAAnEEQCAFQQA2AgAgAEHorgFqQQA2AgAPCyADQQEQVUGAICADEFhB8P8DcSIGSwR/IANBBRBVQQUhAUEABUEFIQFBACECA0AgAUEBaiEBIAJBAWoiBUECdEGwFmooAgAgBk0EQCAFIQIMAQsLIAMgARBVIAJBAnRBsBZqKAIACyECIAYgAmtBECABa3YgAUECdEHQFmooAgBqQQV0IAMQWEELdnIhBiADQQUQVSAAQbCYAWoiASABKQMAIARBDnZBAXFBA2oiAa19NwMAIABBpJYBaiEDIABB4ABqIgUoAgAhAiAAQdDNA2oiBCgCACEAA0AgAygCACIHIAJqIAcgAiAGayAAcWosAAA6AAAgBSAFKAIAQQFqIAQoAgAiAHEiAjYCACABQX9qIgENAAsPCwUgAEHorgFqIgMoAgAhAiADIAJBAWo2AgAgAkEPSgRAIABB9K4BaigCAEUEQCAFQQE2AgALCwsgByAHKAIAIAFqIgIgAkEIdms2AgAgAEH4rgFqIgIoAgBBEGohBSACIAU2AgAgBUH/AUsEQCACQZABNgIAIABB/K4BaiICIAIoAgBBAXY2AgALIABBypgBaiABQQF0aiIFLwEAQQh2IQIgAEGklgFqKAIAIQMgAEHgAGoiBCgCACEBIAQgAUEBajYCACADIAFqIAI6AAAgAEGwmAFqIgEgASkDAEJ/fDcDACAAQcqoAWogBS4BACIDQf//A3EiAkH/AXFqIgQsAAAhASAEIAFBAWo6AAAgA0EBakEQdEEQdUH+AXFBoQFKBEAgAEHKmAFqIQMgAEHKqAFqIQQDQEEAIAMgBBCfASAAQcqoAWogBS4BACIGQf//A3EiAkH/AXFqIgcsAAAhASAHIAFBAWo6AAAgBkEBakEQdEEQdUH+AXFBoQFKDQALCyAFIABBypgBaiABQf8BcUEBdGoiAC4BADsBACAAIAJBAWo7AQAL7wIBCH9BgCAgAEEEaiIEEFhB8P8DcSIDSwR/IARBBRBVQQUhAUEABUEFIQEDQCABQQFqIQEgAkEBaiIFQQJ0QbAWaigCACADTQRAIAUhAgwBCwsgBCABEFUgAkECdEGwFmooAgALIQIgAyACa0EQIAFrdiABQQJ0QdAWaigCAGoiAUH/AUsEQA8LIABBzK4BaiIEIABByqQBaiABQQF0aiIFLgEAIgNB//8DcSICQQh2NgIAIABByqwBaiACQf8BcWoiBiwAACEBIAYgAUEBajoAACADQQFqQRB0QRB1Qf8BcUUEQCAAQcqkAWohAyAAQcqsAWohBgNAQQAgAyAGEJ8BIAQgBS4BACIHQf//A3EiAkEIdjYCACAAQcqsAWogAkH/AXFqIggsAAAhASAIIAFBAWo6AAAgB0EBakEQdEEQdUH/AXFFDQALCyAFIABByqQBaiABQf8BcUEBdGoiAC4BADsBACAAIAJBAWo7AQAL6QgCEH8BfiMEIQYjBEEQaiQEIAAvAQAhAiABQdQMaiINKAIAIgMgAEEIaiIKKAIARwRAIAMpAgAhEgNAIAMgA0F4aiIDKQIANwIAIAMgEjcCACADIAooAgBHDQALCyACQf//A3EiDEF/aiEOIANBAWoiAiACLQAAQQRqOgAAIABBBGoiBy4BACECIAcgAkH//wNxQQRqOwEAIAFB4AxqKAIAQQBHIg8gA0EBaiIFLQAAIghqQQF2IQQgBSAEOgAAIAcgBEH/AXE7AQAgAyAMQQN0aiEQIA4hCSACQQRqQRB0QRB1Qf//A3EgCGshBSADIQIDQCACQQhqIQQgBSACQQlqIggtAAAiC2shBSAIIAsgD2pBAXYiCzoAACAHIAsgBy8BAGo7AQAgCCwAACILQf8BcSACLQABSgRAIAQsAAAhESAGIARBAmoiAigBADYBACAGIAIuAQQ7AQQgBCECA0ACQCACIAJBeGoiCCkCADcCACAIIAooAgBGDQAgC0H/AXEgAkFxai0AAEoEQCAIIQIMAgsLCyAIIBE6AAAgAkF5aiALOgAAIAhBAmoiAiAGKAEANgEAIAIgBi4BBDsBBAsgCUF/aiIJBEAgBCECDAELCyAQQXlqLAAABEAgAC4BACEABUEAIQIgAyAOQQN0aiEDA0AgAkEBaiECIANBeGohBCADQXlqLAAARQRAIAQhAwwBCwsgAiAFaiEDIAAgAC8BACACayIEQf//A3EiAjsBACAEQf//A3FBAUYEQCAKKAIAIgQsAAAhBSAELAABIQIgBiAEQQJqIgkoAQA2AQAgBiAJLgEEOwEEA0AgAkH/AXEiAiACQQF2a0H/AXEhAiADQQF1IgNBAUoNAAsgBCABQciWAWogASAMQQFqQQF2akG5lQFqLQAAQQJ0aiIBKAIANgIAIAEgBDYCACANIAc2AgAgByAFOgAAIAcgAjoAASAAQQZqIgAgBigBADYBACAAIAYuAQQ7AQQgBiQEDwUgAyEFIAIhAAsLIAcgBSAFQQF2ayAHLwEAajsBACAMQQFqQQF2IgIgAEH//wNxQQFqQQF2IgNGBEAgCigCACEABSAKKAIAIQAgASACakG5lQFqLAAAIgJB/wFxIQQgASADakG5lQFqLAAAIglB/wFxIQUgAiAJRwRAIAFByJYBaiAFQQJ0aiIJKAIAIgIEQCAJIAIoAgA2AgAgAiAAIANBBHQQUxogACEDIAIhAAUgAUGUlQFqIARqLQAAIAFBlJUBaiAFai0AACIDayECIAAgA0EEdGohAyACIAFBlJUBaiABIAJqQbmVAWotAAAiBGotAABHBEAgAyABQciWAWogBEF/aiIEQQJ0aiIFKAIANgIAIAUgAzYCACADIAFBlJUBaiAEai0AACIEQQR0aiEDIAIgBGshAgsgASACakG5lQFqLQAAIQQLIAMgAUHIlgFqIARBAnRqIgEoAgA2AgAgASADNgIACyAKIAA2AgALIA0gADYCACAGJAQLuxIBCX8gAEHwDGpBAEGAAhBUGiAAQZCVAWoiBhCOBCAAQewMaiIIIABB5AxqIgUoAgAiA0F/cyIBQXNKBH8gAQVBcws2AgAgAEHElgFqIgcoAgAiASAAQcCWAWoiAigCAEYEQCAAQciWAWoiBCgCACIBBEAgBCABKAIANgIABSAGQQAQoAEhASAFKAIAIQMLBSAHIAFBcGoiATYCAAsgAEHQDGogATYCACAAQcgMaiIFIAE2AgAgAUEANgIMIABB4AxqIAM2AgAgAUGAAjsBACABQYECOwEEIABByJYBaiAAQbmWAWotAAAiA0ECdGoiBCgCACIBBEAgBCABKAIANgIABSACIAIoAgAiASAAQZSVAWogA2oiCS0AAEEEdGoiBDYCACAEIAcoAgBLBEAgAiAEIAktAABBBHRrNgIAIAYgAxCgASEBCwsgBSgCACABNgIIIABB1AxqIAE2AgAgAEHoDGogCCgCADYCACAAQfEUakEAOgAAIAUoAgBBCGohA0EAIQEDQCADKAIAIAFBA3RqIAE6AAAgAygCACABQQN0akEBOgABIAMoAgAgAUEDdGpBADYCBCABQQFqIgFBgAJHDQBBACEBCwNAIABB9BRqIAFBB3RqQYCAAUHd+QAgAUECaiIDbmtB//8DcSICOwEAIAAgAUEHdGpBhBVqIAI7AQAgACABQQd0akGUFWogAjsBACAAIAFBB3RqQaQVaiACOwEAIAAgAUEHdGpBtBVqIAI7AQAgACABQQd0akHEFWogAjsBACAAIAFBB3RqQdQVaiACOwEAIAAgAUEHdGpB5BVqIAI7AQAgACABQQd0akH2FGpBgIABQb8+IANua0H//wNxIgI7AQAgACABQQd0akGGFWogAjsBACAAIAFBB3RqQZYVaiACOwEAIAAgAUEHdGpBphVqIAI7AQAgACABQQd0akG2FWogAjsBACAAIAFBB3RqQcYVaiACOwEAIAAgAUEHdGpB1hVqIAI7AQAgACABQQd0akHmFWogAjsBACAAIAFBB3RqQfgUakGAgAFBv7MBIANua0H//wNxIgI7AQAgACABQQd0akGIFWogAjsBACAAIAFBB3RqQZgVaiACOwEAIAAgAUEHdGpBqBVqIAI7AQAgACABQQd0akG4FWogAjsBACAAIAFBB3RqQcgVaiACOwEAIAAgAUEHdGpB2BVqIAI7AQAgACABQQd0akHoFWogAjsBACAAIAFBB3RqQfoUakGAgAFB85EBIANua0H//wNxIgI7AQAgACABQQd0akGKFWogAjsBACAAIAFBB3RqQZoVaiACOwEAIAAgAUEHdGpBqhVqIAI7AQAgACABQQd0akG6FWogAjsBACAAIAFBB3RqQcoVaiACOwEAIAAgAUEHdGpB2hVqIAI7AQAgACABQQd0akHqFWogAjsBACAAIAFBB3RqQfwUakGAgAFBockBIANua0H//wNxIgI7AQAgACABQQd0akGMFWogAjsBACAAIAFBB3RqQZwVaiACOwEAIAAgAUEHdGpBrBVqIAI7AQAgACABQQd0akG8FWogAjsBACAAIAFBB3RqQcwVaiACOwEAIAAgAUEHdGpB3BVqIAI7AQAgACABQQd0akHsFWogAjsBACAAIAFBB3RqQf4UakGAgAFBvLUBIANua0H//wNxIgI7AQAgACABQQd0akGOFWogAjsBACAAIAFBB3RqQZ4VaiACOwEAIAAgAUEHdGpBrhVqIAI7AQAgACABQQd0akG+FWogAjsBACAAIAFBB3RqQc4VaiACOwEAIAAgAUEHdGpB3hVqIAI7AQAgACABQQd0akHuFWogAjsBACAAIAFBB3RqQYAVakGAgAFBsswBIANua0H//wNxIgI7AQAgACABQQd0akGQFWogAjsBACAAIAFBB3RqQaAVaiACOwEAIAAgAUEHdGpBsBVqIAI7AQAgACABQQd0akHAFWogAjsBACAAIAFBB3RqQdAVaiACOwEAIAAgAUEHdGpB4BVqIAI7AQAgACABQQd0akHwFWogAjsBACAAIAFBB3RqQYIVakGAgAFB0cABIANua0H//wNxIgM7AQAgACABQQd0akGSFWogAzsBACAAIAFBB3RqQaIVaiADOwEAIAAgAUEHdGpBshVqIAM7AQAgACABQQd0akHCFWogAzsBACAAIAFBB3RqQdIVaiADOwEAIAAgAUEHdGpB4hVqIAM7AQAgACABQQd0akHyFWogAzsBACABQQFqIgFBgAFHDQBB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QEEAEBgQWgUgBhAeCwsgByAZNwMAIAMgGjcDACACIAo2AgAgACgCACgCECECQQAkBSACIAAgF6cgF0IgiKdBABBQIwUhAkEAJAUgAkEBcQRAQQAQGBBaBSABIQ8LCyAPLAAABEAgAEGhvANqLAAARQRAIAUkBEEBDwsLIABB5LwDaiAAQRhqEGoaIAUkBEEBDwsgBSQEQQALBgBBFxAACwYAQRYQAAsGAEEREAALCABBBxAAQQALLQEBfyAAQTsQrQEiAgRAIAJBBGoQyQIhACABBEAgAkEANgIACwVBACEACyAAC9YBAQJ/IwQhAyMEQYAQaiQEAkAgAARAAkACQAJAAkAgACgCAA4wAAICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIBAgsMBAsgAUEANgIADAELIANBgBAQtgNFBEAgA0EAOgAACyADIAEgAhB7GiABEFkiBARAIARBAWogAkkgASAEQX9qQQJ0aigCAEEvR3EEQCABQfwMEIYCGgsLCyABIAAgAhB4GiADJAQPCwsgAkUEQCADJAQPCyABQQA2AgAgAyQEC78BAQJ/AkAgAkF/aiIEBEBBACECA0ACQAJAAkAgACACQQJ0aigCACIDDl0AAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgECCwwEC0EvIQMLIAEgAkECdGogAzYCACACQQFqIgIgBEkNAAsFQQAhAgsLIAEgAkECdGpBADYCAAtWAQN/IAAoAgQiBUEIdSEEIAVBAXEEQCACKAIAIARqKAIAIQQLIAAoAgAiACgCACgCHCEGIAAgASACIARqIAVBAnEEfyADBUECCyAGQQdxQZgCahEKAAsHACAAIAFGCwsAIAAQ/AEgABBSCxIAIABB2CU2AgAgAEEEahCNAwuDAQEBfyMEIQEjBEEQaiQEQQAkBSAAEAsjBSEAQQAkBSAAQQFxRQRAQQAkBUEZQbvyACABEA1BACQFC0EAEBgiABAVGkEAJAVBGUHj8gAgAUEIahANQQAkBUEAEBghAUEAJAVBBBALIwUhAEEAJAUgAEEBcQRAQQAQGCIAEFoFIAEQWgsLOAEC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IAQQyAEgBSgCACEACwJAIABBqIACahBZIgYEQCACIAYgAhBZIgFJBH8gBgUgAQtBAnRqIQIDQAJAAkACQAJAAkAgAigCAA4wAAICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIBAgsMAwsMAQsMBAsgAkEEaiECDAELCyADQQA2AgAPCwsCQAJAIABByMsEaigCACIBQdgARiAAQeCEA2ooAgBBBEZxBEBBACEBBSABQcUARgRAQQAhAQwCBUEAIQELCyAFKAIAQeCEA2ooAgBBAUYNACADIAIgBBB4GgwBCyADIAIQmwEgBBB4GgsgAygCABCTASEADwv4AQEDfyMEIQMjBEHwFmokBCADQSRqIQQgAyAAKAIAIgU2AgACQAJAAkACQCAFQQFrDgMAAQIDCyADIAAoAgQ2AgQMAgsgAyAAKAIEQX9zNgIEDAELIAQgACgCCBCmBCAEIANBBGoQqQQLIAIEQCADIAIQvwILIAMoAgAiAkUEQCADJARBAQ8LIAEoAgAiAEUEQCADJARBAQ8LIAJBAUYgAEEBRnFFBEAgAkECRiAAQQJGcUUEQCACQQNGIABBA0ZxRQRAIAMkBEEADwsgA0EEaiABQQRqQSAQa0UhACADJAQgAA8LCyADKAIEIAEoAgRGIQAgAyQEIAALcgEBfyAAKAIAIgNBAUYEQCAAQQRqIgMgAygCAEH//wNxIAEgAhDQBEH//wNxNgIAIAAoAgAhAwsgA0ECRgRAIABBBGoiAyADKAIAIAEgAhCZATYCACAAKAIAIQMLIANBA0cEQA8LIAAoAgggASACEKsEC74FAQJ/IABBCGoiBCgCACICRQRAQQBBzBYQXyICIgNrQT9xIANqIQMgAiADNgLwASACIANBgAFqNgL0ASACIANBoAFqNgL4ASACIANBqAFqNgL8ASACQQAgAkGIAmoiA2tBP3EgA2oiAzYC+AMgAiADQYABajYC/AMgAiADQaABajYCgAQgAiADQagBajYChAQgAkEAIAJBkARqIgNrQT9xIANqIgM2AoAGIAIgA0GAAWo2AoQGIAIgA0GgAWo2AogGIAIgA0GoAWo2AowGIAJBiAhqQQAgAkGYBmoiA2tBP3EgA2oiAzYCACACQYwIaiADQYABajYCACACQZAIaiADQaABajYCACAC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AUGg0ABqIAIEfyADBSAECzYAACABQZjQAGogAEHstwdqIgIoAgA2AABBACQFQQUgAEG4twdqEAUhAyMFIQRBACQFIARBAXFFBEAgAUGc0ABqIAM2AAAgAUGk0ABqIABBovcGai0AAEEwajYAACABQajQAGogAEGk9wZqKAIANgAAIAFBtNAAakEANgAAIAFBuNAAakEANgAAIAFBvNAAaiAAQey4B2ooAgBBCnY2AAACQAJAAkACQCAAQei3B2ooAgBBAWsOAwAAAQILIAFBwNAAakEBNgAADAILIAFBwNAAakECNgAAIAFBxNAAaiIDIAIpAAA3AAAgAyACKQAINwAIIAMgAikAEDcAECADIAIpABg3ABgMAQsgAUHA0ABqQQA2AAALIAFB5NAAaiAAQfi4B2ooAgAiAjYAACACBEAgAUHo0ABqKAAAIgIEQCABQezQAGooAAAiA0F/akGfjQZJBEBBACQFQRAgAiAAQfy4B2ogAxAHGiMFIQJBACQFIAJBAXENCQsLCyABQfDQAGogAEH8+AdqLQAANgAAQQAPCwsLCwsLC0GACBAYIQEjB0GACBBERwRAIAEQHgsgARAVIQECQCAAQajLBGooAgAiAEUEQAJAAkACQAJAAkACQAJAAkACQCABKAIADgwHCAABCAIDCAUECAYIC0ESIQAMCQtBDCEADAgLQRMhAAwHC0EPIQAMBgtBECEADAULQQshAAwEC0EYIQAMAwtBACEADAILQRUhAAsLEBYgAAvEAwEEfyMEIQIjBEFAayQEIAJBIGohAyAAKAIAIgRBAkYEQCADIABBBGoiBSgCACIEOgAAIAMgBEEIdjoAASADIARBEHY6AAIgAyAEQRh2OgADIAFBICADQQQgAhCnASAFQQA2AgAgBSACLQAfQRh0IAItAB5BEHQgAi0AHUEIdCACLQAbQRh0IAItABpBEHQgAi0AGUEIdCACLQAXQRh0IAItABZBEHQgAi0AFUEIdCACLQATQRh0IAItABJBEHQgAi0AEUEIdCACLQAPQRh0IAItAA5BEHQgAi0ADUEIdCACLQALQRh0IAItAApBEHQgAi0ACUEIdCACLQAHQRh0IAItAAZBEHQgAi0ABUEIdCACLQADQRh0IAItAAJBEHQgAi0AAUEIdCACLQAAcnJyIAItAARzc3NzIAItAAhzc3NzIAIt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IBwgAa18NwMAAkACQAJAAkACQAJAAkACQAJAAkAgFCgCACIBQQFrDnkAAQEIAggICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAMFBwYECAtBACQFQRsgAEG4pgFqIgEQDCMFIQJBACQFIAJBAXENDSABIAcpAgA3AgAgASAHKQIINwIIIAEgBygCEDYCEEEAJAVBCiAEEAUhASMFIQJBACQFIAJBAXENDSAAQcymAWoiAyABOwEAQQAkBUEMIAQQBSECIwUhAUEAJAUgAUEBcQ0NIABB0KYBaiACNgIAIABBnbwDaiAAQcCmAWooAgAiAUEBcToAACAAQZy8A2ogAUEDdkEBcToAACAAQZ+8A2ogAUECdkEBcToAACAAQaO8A2ogAUEGdkEBcToAACAIIAFBB3ZBAXE6AAAgAEGgvANqIAIEf0EBBSADLgEAQQBHC0EBcToAACAAQdSmAWogAUEBdkEBcToAACAAQaG8A2ogAUEIdkEBcToAACAAQaK8A2ogAUEEdkEBcToAAAwICyAAQbCnAWohAyAAQfCtAmohAkEAJAVBCCABQQJGIg8EfyADBSACIgMLQQAQDSMFIQFBACQFIAFBAXFFBEAgAyAHKQIANwIAIAMgBykCCDcCCCADIAcoAhA2AhAgA0GIwQBqIANBCGoiCSgCACIBQQFxOgAAIANBicEAaiABQQF2QQFxOgAAIANBi8EAaiIGIAFBAnZBAXE6AAAgA0GQwQBqIAFBCnZBAXE6AAAgA0HgwQBqIA8Ef0EAIQUgAUEEdkEBcQUgAUEQcUEARyEFQQALOgAAIANB6sEAaiAFQQFxOgAAIANB4cEAaiIMIAFB4AFxQeABRiICOgAAIANB5MEAaiACBH9BAAVBgIAEIAFBBXZBB3F0CzYCACADQeLBAGoiGSABQQN2QQFxOgAAIANB48EAaiABQQt2QQFxOgAAQQAkBUEMIAQQBSEBIwUhAkEAJAUgAkEBcUUEQCADQRRqIhogATYCAEEAJAVBDCAEEAUhAiMFIQFBACQFAkAgAUEBcUUEQEEAJAVBCyAEEAUhASMFIQVBACQFIAVBAXFF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AGoiBxAMIwUhAkEAJAUCQCACQQFxBEAQFyECBSAAQdDnAWoiAkIANwIAIAJCADcCCCACQgA3AhAgAkIANwIYIAJCADcCICAAQZDuAmoiAkIANwIAIAJCADcCCCACQgA3AhAgAkIANwIYIAJCADcCICAAQazzAGoiCEEANgIAIABBqPMAaiABRSICOgAAAkAgAgRAQQAkBUEEQcDLBBAFIQMjBSEBQQAkBSABQQFxBEAQFyECBUEAJAVBGiADEAwjBSEBQQAkBSABQQFxRQRAIAMhAQwDCxAXIQIgAxBSIABBkO4CaigCACIDBEAgAxBSCwsgAEHQ5wFqKAIAIgMEQCADEFILIAcQtQIMAwsLIAggATYCACAAIAFBnYUDaiwAADoAFSAAQZi8A2pBAjYCACAAQai8A2pBADYCACAGQgA3AwAgAEGtvANqQQA6AAAgAEGsvANqQQA6AAAgAEHI8wBqQQA2AgAgAEGIvANqIgFCADcDACABQgA3AwggAEGcvANqIgFCADcCACABQQA6AAggAEGw8wBqQn83AwAgAEG48wBqQX82AgAgAEGwrQJqIgFCADcDACABQgA3AwggAUIANwMQIAFCADcDGCAAQcC8A2pBADYCACAAQeS8A2pBADYCACAAQb68A2pBADoAACAAQdDzAGpBADoAACAAQbimAWoiAUIANwMAIAFCADcDCCABQgA3AxAgAUIANwMYIAFCADcDICABQgA3AyggAUIANwMwIAFCADcDOCABQUBrQgA3AwAgAUIANwNIIAFCADcDUCABQgA3A1ggAUIANwNgIAFCADcDaCABQQA2AnAgAEHIvANqIgBCADcDACAAQgA3AwggAEIANwMQIABBADoAGA8LCyAFEIQBCyAEEI4BIAAQcCACEB4LjwgBC38jBCECIwRBMGokBCACQRBqIQogAkEIaiEHIAIiCUEYaiIFQgA3AgAgBUIANwIIQQAkBUEUIAAgBUEAEAchAiMFIQNBACQFAkAgA0EBcUUEQCACBH8gBUEEaiIGKAIAIgtBAWohAiAGIAI2AgAgAiAFQQhqIggoAgAiBEsEfyAFKAIMIgNBAEcgAiADS3EEf0EAJAUgCSADNgIAQQJBrPUCQdgbIAkQDiMFIQNBACQFIANBAXENBEEAJAVBFEGs9QIQDCMFIQNBACQFIANBAXENBCAIKAIA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IABBwJgBaiIaQgA3AwAgAEHoAGoiA0IANwMAIANCADcDCCADQgA3AxAgA0EANgIYIABB8ABqIgZBfzYCACABRQRAIABBwK4CakEANgIAIABBxK4CakEANgIAIABBuK4CakEANgIAIABBvK4CakEBNgIAIABByK4CakEAQfACEFQaIABBhK8BakEAQbT/ABBUGiAAQczMA2pBADoAACAAQbTJA2pBAEGUAxBUGiAAQbDJA2pBAjYCACAAQcjMA2pBADYCAAsgACABEIMBIAMoAgAiAiAFKAIAIghrIgFBAEgNASAGIABB+ABqIhIoAgAgCGsgBigCAGo2AgAgCEGAgAFKBEAgAUEASgRAIAAoAhAiAiACIAhqIAEQXRoLIAVBADYCACADIAE2AgAFIAIhAQsgAUGAgAJGBEBBACEBQYCAAiECBSAAKAIAIAAoAhAgAWpBgIACIAFrEF4hASADKAIAIgggAWohAiABQQBKBEAgAyACNgIABSAIIQILCyAAQewAaiIOIAJBYmoiAjYCACASIAUoAgAiCDYCACAIQX9qIAYoAgAiDGohCCAMQX9HBEAgDiACIAhIBH8gAgUgCAs2AgALIAFBf0YNASAAIABBBGoiDCAAQfAAaiITEKQCRQ0BIAAgDCATIABBhAFqIhsQowJFDQECQCADIQggBSEXIAYhGAsLIAQhDyAAQdDNA2ohECAAQeAAaiEJIABBBGohCiAAQaCWAWohICAAQRBqIRQgAEGIAmohISAAQaiYAWohFSAAQaSWAWohIiAAQaiWAWohDSAAQfQfaiEjIABBzABqIRwgAEHQAGohJCAAQcgAaiEdIABBxABqIRYgAEHYAGohGSAAQeA9aiElIABB3DxqISYgAEHwHmohJyAAQczbAGohKCAAQcjaAGohKSAAQYQBaiEqIABBsJgBaiErIABB8ABqISwgAEH0AGohLSAAQYABaiEuAkACQAJAA0ACQCAJIAkoAgAgECgCACIEcSICNgIAIAooAgAiASAOKAIATgRAA0ACQCABICwoAgAiAyASKAIAIgVqIgJIBEAgASACQX9qRw0BIAcoAgAgLSgCAEgNAQsgLiwAAA0DIAAgDCATEKQCRQ0GIAAgDCATIBsQowJFDQYgCigCACEBDAELCyAIKAIAIgQgAWsiAkEASA0BIBggBSABayADajYCACAB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NwIIIAJCADcCECACQgA3AhggAkEANgIgIABBoJYBaiAAQczNA2ooAgAiAkGAgIACSQR/IAIFQYCAgAILIABB0M0DaigCAHE2AgALIABBADYCOCAAQQhqIghBADYCACAAQQRqIgdBADYCACAAQcCYAWoiGkIANwMAIABB6ABqIgJCADcDACACQgA3AwggAkIANwMQIAJBADYCGCAAQX82AnAgAUUEQCAAQcCuAmpBADYCACAAQcSuAmpBADYCACAAQbiuAmpBADYCACAAQbyuAmpBATYCACAAQciuAmpBAEHwAhBUGiAAQYSvAWpBAEG0/wAQVBogAEHMzANqQQA6AAAgAEG0yQNqQQBBlAMQVBogAEGwyQNqQQI2AgAgAEHIzANqQQA2AgALIAAgARCDASACKAIAIgogBygCACIJayIDQQBIBEAPCyAJQYCAAUoEQCADQQBKBEAgACgCECIKIAogCWogAxBdGgsgB0EANgIAIAIgAzYCAAUgCiEDCyAAKAIAIABBEGoiDigCACADakGAgAIgA2sQXiEDIAIoAgAiCSADaiEKIANBAEoEQCACIAo2AgAgACAKQWJqNgJsBSAAIAlBYmo2AmwgA0F/RgRADwsLIAEEQCAAQczMA2osAAAEQCACIQkgACEKDAMLCyAAENsBBEAgAiEJIAAhCgUPCwsLIABB0M0DaiERIABB4ABqIQsgAEHsAGohFyAAQeQAaiEfIABByMwDaiESIABBwLECaiEMIABBsMkDaiEgIABBpJYBaiETIABBiAJqISEgAEH0H2ohIiAAQcwAaiEUIABB0ABqIRsgAEHIAGohFSAAQcQAaiEPIABB2ABqIRYgAEG8sQJqIRggAEG4sQJqIRkgAEHgPWohIyAAQdw8aiEkIABB8B5qISUgAEHM2wBqISYgAEHI2gBqIScgAEHMzQNqISggAEHMzANqIRwgAEGEAWohKSAAQbCYAWohKkEAIQECQAJAAkACQAJAAkACQAJAA0ACQCALIAsoAgAgESgCAHE2AgAgBygCACIEIBcoAgBKBEAgCSgCACIDIARrIgJBAEgNASAEQYCAAUoEQCACQQBKBEAgDigCACIDIAMgBGogAhBdGgsgB0EANgIAIAkgAjYCAAUgAyECCyAKKAIAIA4oAgAgAmpBgIACIAJrEF4hAiAJKAIAIgQgAmohAyAC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wQBqEJsECwwBCyABQcytAmosAABFDQEgASAAQRBqQQAgDRC1AQRAIAEgAUGIvANqKQMAQQAgASgCACgCEEEDcUG4AmoRAgAgBiQEQQEPBUGs9QJBARBkDAILAAsgASABQZC8A2opAwBBACABKAIAKAIQQQNxQbgCahECACAGJARBAQ8LIAYkBEEAC5oBAQJ/IABCADcDACAAQRBqIgMQtwIgACABNgIIIABBADYC3AEgAEHgwQBqQQA2AgAgAEEANgLMAUEAJAVBBEHYzQMQBSEBIwUhAkEAJAUgAkEBcQRAEBchAiADEIQBIAIQHgtBACQFQQcgASADEA0jBSECQQAkBSACQQFxRQRAIAAgATYCyAEPCxAXIQAgARBSIAMQhAEgABAeC4IDAgd/AX4gAEGQMmohBkGAgAQgAEGUMmoiAygCACICa0GAAkkEQCAAKAIQIgUgBSACaiAGKAIAIAJrIgIQUxogA0EANgIAIAYgAjYCACAAEOIBGiADKAIAIQILIAJBB2ogBigCAEsEQEEADwsgASAAQRBqIgcoAgAgAmpBBxDpASADIAMoAgBBB2o2AgAgARCmASEIIAFBBBC6AiICQX1qIAEQYSIJp2ohBCAJQgBRIAJFIARBAEhycgRAIABB6DFqQQA6AABBAA8LIAYoAgAgAygCACICayEFAkAgBEEASgRAIAEgBygCACACaiAFIARJBH8gBQUgBAsiAhDpASADIAMoAgAgAmo2AgAgBCACayICQQBKBEADQAJAIANBADYCACAGQQA2AgAgABDiAUUEQEEAIQAMAQsgASAHKAIAIAMoAgBqIAUgAkkEfyAFBSACCyIEEOkBIAMgAygCACAEajYCACACIARrIgJBAEoNAQwECwtBAA8LCwsgCCABELgCRgtOAQF/AkACQAJAIAJBmLwDaigCAEECaw4CAAECCyMEIQQjBEGAEGokBCACQcynAWooAgAaIAQkBEEADwsgACADIAJBsKcBahCdBA8LQQALRwAgAEHJiQNqLAAABEAPCyAAQZCFA2osAABFBEAPCyABQZi8A2ooAgBBAkcEQA8LIAFBkK4CakHMERBzBEAPCyABIAIQoQQLOAAgAEHJiQNqLAAABEAPCyABQeitAmouAQBBgQJHBEAPCyAAQZCFA2osAABFBEAPCyABIAIQogQLrQEBAn8jBCED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IABBpA5qIAJBgAFqNgIAIABBqA5qIAJBoAFqNgIAIABBrA5qIAJBqAFqNgIAIABBqBBqQQAgAEG4DmoiAmtBP3EgAmoiAjYCACAAQawQaiACQYABajYCACAAQbAQaiACQaABajYCACAAQbQQaiACQagBajYCAAUgAiABKALwAUGwARBTGiAAIAEoAoACNgKAAiAAIAEsAIQCOgCEAiAAQQAgAEGIAmoiAmtBP3EgAmoiAjYC+AMgACACQYABajYC/AMgACACQaABajYCgAQgACACQagBajYChAQgAiABKAL4A0GwARBTGiAAIAEoAogENgKIBCAAIAEsAIwEOgCMBCAAQQAgAEGQBGoiAmtBP3EgAmoiAjYCgAYgACACQYABajYChAYgACACQaABajYCiAYgACACQagBajYCjAYgAiABKAKABkGwARBTGiAAIAEoApAGNgKQBiAAIAEsAJQGOgCUBiAAQYgIakEAIABBmAZqIgJrQT9xIAJqIgI2AgAgAEGMCGogAkGAAWo2AgAgAEGQCGogAkGgAWo2AgAgAEGUCGogAkGoAWo2AgAgAiABQYgIaigCAEGwARBTGiAAQZgIaiABQZgIaigCADYCACAAQZwIaiABQZwIaiwAADoAACAAQZAKakEAIABBoAhqIgJrQT9xIAJqIgI2AgAgAEGUCmogAkGAAWo2AgAgAEGYCmogAkGgAWo2AgAgAEGcCmogAkGoAWo2AgAgAiABQZAKaigCAEGwARBTGiAAQaAKaiABQaAKaigCADYCACAAQaQKaiABQaQKaiwAADoAACAAQZgMakEAIABBqApqIgJrQT9xIAJqIgI2AgAgAEGcDGogAkGAAWo2AgAgAEGgDGogAkGgAWo2AgAgAEGkDGogAkGoAWo2AgAgAiABQZgMaigCAEGwARBTGiAAQagMaiABQagMaigCADYCACAAQawMaiABQawMaiwAADoAACAAQaAOakEAIABBsAxqIgJrQT9xIAJqIgI2AgAgAEGkDmogAkGAAWo2AgAgAEGoDmogAkGgAWo2AgAgAEGsDmogAkGoAWo2AgAgAiABQaAOaigCAEGwARBTGiAAQbAOaiABQbAOaigCADYCACAAQbQOaiABQbQOaiwAADoAACAAQagQakEAIABBuA5qIgJrQT9xIAJqIgI2AgAgAEGsEGogAkGAAWo2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ABggACACKQAgNwAgIAAgAikAKDcAKCAAIAIpADA3ADAgACACKQA4NwA4IAQgBCgCAEFAaiIANgIAIAMgBmohAyAFIAZrIgUNAAsMAQsgAiADIAUQUxogBCAEKAIAIAVqIgA2AgALQSAhBSAKQYABaiEDAkACQANAIAcoAgAgAGohAiAFQYABIABrIgZNDQEgAiADIAYQUxogBCAEKAIAIAZqNgIAIAkoAgAiACgCACECIAAgAkFAazYCACAAQQRqIgAgACgCACACQb9/S2o2AgAgCCAHKAIAEG0gBygCACIAQUBrIQIgACACKQAANwAAIAAgAikACDcACCAAIAIpABA3ABAgACACKQAYNwAYIAAgAikAIDcAICAAIAIpACg3ACggACACKQAwNwAwIAAgAikAODcAOCAEIAQoAgBBQGoiADYCACADIAZqIQMgBSAGayIFDQALDAELIAIgAyAFEFMaIAQgBCgCACAFaiIANgIAC0EgIQUgCkGgAWohAwJAAkADQCAHKAIAIABqIQIgBUGAASAAayIGTQ0BIAIgAyAGEFMaIAQgBCgCACAGajYCACAJKAIAIgAoAgAhAiAAIAJBQGs2AgAgAEEEaiIAIAAoAgAgAkG/f0tqNgIAIAggBygCABBtIAcoAgAiAEFAayECIAAgAikAADcAACAAIAIpAAg3AAggACACKQAQNwAQIAAgAikAGDcAGCAAIAIpACA3ACAgACACKQAoNwAoIAAgAikAMDcAMCAAIAIpADg3ADggBCAEKAIAQUBqIgA2AgAgAyAGaiEDIAUgBmsiBQ0ACwwBCyACIAMgBRBTGiAEIAQoAgAgBWoiADYCAAtBICEFIApBwAFqIQMCQAJAA0AgBygCACAAaiECIAVBgAEgAGsiBk0NASACIAMgBhBTGiAEIAQoAgAgBmo2AgAgCSgCACIAKAIAIQIgACACQUBrNgIAIABBBGoiACAAKAIAIAJBv39LajYCACAIIAcoAgAQbSAHKAIAIgBBQGshAiAAIAIpAAA3AAAgACACKQAINwAIIAAgAikAEDcAECAAIAIpABg3ABggACACKQAgNwAgIAAgAikAKDcAKCAAIAIpADA3ADAgACACKQA4NwA4IAQgBCgCAEFAaiIANgIAIAMgBmohAyAFIAZrIgUNAAsMAQsgAiADIAUQUxogBCAEKAIAIAVqIgA2AgALQSAhBSAKQeABaiEKAkACQANAAkAgBygCACAAaiEDIAVBgAEgAGsiAk0NACADIAogAhBTGiAEIAQoAgAgAmo2AgAgCSgCACIAKAIAIQMgACADQUBrNgIAIABBBGoiACAAKAIAIANBv39LajYCACAIIAcoAgAQbSAHKAIAIgBBQGshAyAAIAMpAAA3AAAgACADKQAINwAIIAAgAykAEDcAECAAIAMpABg3ABggACADKQAgNwAgIAAgAykAKDcAKCAAIAMpADA3ADAgACADKQA4NwA4IAQgBCgCAEFAaiIANgIAIAUgAmsiBUUNAiAKIAJqIQoMAQsLDAELIAggARDgASALJAQPCyADIAogBRBTGiAEIAQoAgAgBWo2AgAgCCABEOABIAskBAuwAgIEfwN+IABB6DFqIgYsAABFBEBBAA8LIABBqDJqIQUgAEGcMmohByAAQbAyaiEEA0AgBSkDACAHKAIArXwgBCkDAFgEQCAAEKQEDQELCyAGLAAARQRAIABBuDJqLAAARQRAQQAPCyAAKAIAIAQpAwBBABCcAUEADwsgBCkDACIIIAUpAwAiCVoEQCAIIAKtIgp8IAkgBygCAK18WARAIAEgAEGYMmooAgAgCCAJfadqIAIQUxogAyACNgIAIAQgBCkDACAKfDcDACAAQbgyakEBOgAAQQEPCwsgAEG4MmoiBSwAAARAIAAoAgAgCEEAEJwBIAVBADoAAAsgACgCACABIAIQ1AEiAEEASAR/IAZBADoAAEEABSADIAA2AgAgBCAEKQMAIACsfDcDAEEBCwv2BgEPf0GABCAAQcgWaiIOKAIAIgdrIQYgB0UgBiACS3IEfyABIQkgAgUgAEHIEmogB2ogASAGEFMaA0AgACADQYgCbGohDCAAIANBiAJsakHwAWohCCAAIANBiAJsakH4AWohDUHAACEKIABByBJqIANBBnRqIQsgACADQYgCbGpBgAJqIgcoAgAhBAJAAkADQCAIKAIAIARqIQUgCkGAASAEayIJTQ0BIAUgCyAJEFMaIAcgBygCACAJajYCACANKAIAIgQoAgAhBSAEIAVBQGs2AgAgBEEEaiIEIAQoAgAgBUG/f0tqNgIAIAwgCCgCABBtIAgoAgAiBEFAayEFIAQgBSkAADcAACAEIAUpAAg3AAggBCAFKQAQNwAQIAQgBSkAGDcAGCAEIAUpACA3ACAgBCAFKQAoNwAoIAQgBSkAMDcAMCAEIAUpADg3ADggByAHKAIAQUBqIgQ2AgAgCyAJaiELIAogCWsiCg0ACwwBCyAFIAsgChBTGiAHIAcoAgAgCmo2AgALIANBAWoiA0EIRw0ACyABIAZqIQlBACEHIAIgBmsLIgtB/wNLIQ9BACEEA0AgACAEQYgCbGohECAPBEAgACAEQYgCbGpB8AFqIQ0gACAEQYgCbGpB+AFqIREgCSAEQQZ0aiECIAshCiAAIARBiAJsakGAAmoiCCgCACEDA0BBwAAhBSACIQECQAJAA0AgDSgCACADaiEGIAVBgAEgA2siDE0NASAGIAEgDBBTGiAIIAgoAgAgDGo2AgAgESgCACIDKAIAIQYgAyAGQUBrNgIAIANBBGoiAyADKAIAIAZBv39LajYCACAQIA0oAgAQbSANKAIAIgNBQGshBiADIAYpAAA3AAAgAyAGKQAINwAIIAMgBikAEDcAECADIAYpABg3ABggAyAGKQAgNwAgIAMgBikAKDcAKCADIAYpADA3ADAgAyAGKQA4NwA4IAggCCgCAEFAaiIDNgIAIAEgDGohASAFIAxrIgUNACADIQELDAELIAYgASAFEFMaIAggCCgCACAFaiIBNgIACyACQYAEaiECIApBgHxqIgpB/wNLBEAgASEDDAELCwsgBEEBaiIEQQhJDQALIAtB/wNxIgFFBEAgDiABIAdqNgIADwsgAEHIEmogB2ogCSALIAFraiABEFMaIA4gASAHajYCAAu5CgEDfyAAQcAQakEAQfABEFQaIABBwBJqQQA2AgAgAEHEEmoiA0EAOgAAIABByBJqQQBBhAQQVBogAEG0EmoiAigCACIBQawRKQIANwIAIAFBtBEpAgA3AgggAUG8ESkCADcCECABQcQRKQIANwIYIAIoAgAiASABKAIAQaCAoBBzNgIAIAFBDGoiASABKAIAQYCAhIACczYCACAAQQBB8AEQVBogAEEANgKAAiAAQQA6AIQCIABB9AFqIgIoAgAiAUGsESkCADcCACABQbQRKQIANwIIIAFBvBEpAgA3AhAgAUHEESkCADcCGCACKAIAIgEgASgCAEGggKAQczYCACABQQxqIgEgASgCAEGAgICAAnM2AgAgAEGIAmpBAEHwARBUGiAAQQA2AogEIABBADoAjAQgAEH8A2oiAigCACIBQawRKQIANwIAIAFBtBEpAgA3AgggAUG8ESkCADcCECABQcQRKQIANwIYIAIoAgAiASABKAIAQaCAoBBzNgIAIAFBCGoiAiACKAIAQQFzNgIAIAFBDGoiASABKAIAQYCAgIACczYCACAAQZAEakEAQfABEFQaIABBADYCkAYgAEEAOgCUBiAAQYQGaiICKAIAIgFBrBEpAgA3AgAgAUG0ESkCADcCCCABQbwRKQIANwIQIAFBxBEpAgA3AhggAigCACIBIAEoAgBBoICgEHM2AgAgAUEIaiICIAIoAgBBAnM2AgAgAUEMaiIBIAEoAgBBgICAgAJzNgIAIABBmAZqQQBB8AEQVBogAEGYCGpBADYCACAAQZwIakEAOgAAIABBjAhqIgIoAgAiAUGsESkCADcCACABQbQRKQIANwIIIAFBvBEpAgA3AhAgAUHEESkCADcCGCACKAIAIgEgASgCAEGggKAQczYCACABQQhqIgIgAigCAEEDczYCACABQQxqIgEgASgCAEGAgICAAnM2AgAgAEGgCGpBAEHwARBUGiAAQaAKakEANgIAIABBpApqQQA6AAAgAEGUCmoiAigCACIBQawRKQIANwIAIAFBtBEpAgA3AgggAUG8ESkCADcCECABQcQRKQIANwIYIAIoAgAiASABKAIAQaCAoBBzNgIAIAFBCGoiAiACKAIAQQRzNgIAIAFBDGoiASABKAIAQYCAgIACczYCACAAQagKakEAQfABEFQaIABBqAxqQQA2AgAgAEGsDGpBADoAACAAQZwMaiICKAIAIgFBrBEpAgA3AgAgAUG0ESkCADcCCCABQbwRKQIANwIQIAFBxBEpAgA3AhggAigCACIBIAEoAgBBoICgEHM2AgAgAUEIaiICIAIoAgBBBXM2AgAgAUEMaiIBIAEoAgBBgICAgAJzNgIAIABBsAxqQQBB8AEQVBogAEGwDmpBADYCACAAQbQOakEAOgAAIABBpA5qIgIoAgAiAUGsESkCADcCACABQbQRKQIANwIIIAFBvBEpAgA3AhAgAUHEESkCADcCGCACKAIAIgEgASgCAEGggKAQczYCACABQQhqIgIgAigCAEEGczYCACABQQxqIgEgASgCAEGAgICAAnM2AgAgAEG4DmpBAEHwARBUGiAAQbgQakEANgIA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BgYGaWkAdgBSYXJBcmNoaXZlAHZpAG9wZW4AaWlpaWlpAGdldEZpbGVIZWFkZXIAaWlpAHJlYWRGaWxlAGlpaWkAU3RhdGUAaQBlcnJDb2RlAHZpaWkAZXJyVHlwZQBBcmNIZWFkZXIAc3RhdGUAY29tbWVudABmbGFncwBBcmNGaWxlSGVhZGVyAG5hbWUAcGFja1NpemUAZGlpAHZpaWQAdW5wU2l6ZQBob3N0T1MAY3JjAHRpbWUAdW5wVmVyAG1ldGhvZABmaWxlQXR0cgAxM0FyY0ZpbGVIZWFkZXIATlN0M19fMjEyYmFzaWNfc3RyaW5nSXdOU18xMWNoYXJfdHJhaXRzSXdFRU5TXzlhbGxvY2F0b3JJd0VFRUUATlN0M19fMjIxX19iYXNpY19zdHJpbmdfY29tbW9uSUxiMUVFRQA5QXJjSGVhZGVyAE5TdDNfXzIxMmJhc2ljX3N0cmluZ0ljTlNfMTFjaGFyX3RyYWl0c0ljRUVOU185YWxsb2NhdG9ySWNFRUVFADVTdGF0ZQBFUlJfUFJPQ0VTUwBFUlJfUkVBRABFUlJfT1BFTgBQSzEwUmFyQXJjaGl2ZQBQMTBSYXJBcmNoaXZlADEwUmFyQXJjaGl2ZQBTRVQARU5EAENVUgA0RmlsZQB2b2lkAGJvb2wAY2hhcgBzaWduZWQgY2hhcgB1bnNpZ25lZCBjaGFyAHNob3J0AHVuc2lnbmVkIHNob3J0AGludAB1bnNpZ25lZCBpbnQAbG9uZwB1bnNpZ25lZCBsb25nAGZsb2F0AGRvdWJsZQBzdGQ6OnN0cmluZwBzdGQ6OmJhc2ljX3N0cmluZzx1bnNpZ25lZCBjaGFyPgBzdGQ6OndzdHJpbmcAZW1zY3JpcHRlbjo6dmFsAGVtc2NyaXB0ZW46Om1lbW9yeV92aWV3PGNoYXI+AGVtc2NyaXB0ZW46Om1lbW9yeV92aWV3PHNpZ25lZCBjaGFyPgBlbXNjcmlwdGVuOjptZW1vcnlfdmlldzx1bnNpZ25lZCBjaGFyPgBlbXNjcmlwdGVuOjptZW1vcnlfdmlldzxzaG9ydD4AZW1zY3JpcHRlbjo6bWVtb3J5X3ZpZXc8dW5zaWduZWQgc2hvcnQ+AGVtc2NyaXB0ZW46Om1lbW9yeV92aWV3PGludD4AZW1zY3JpcHRlbjo6bWVtb3J5X3ZpZXc8dW5zaWduZWQgaW50PgBlbXNjcmlwdGVuOjptZW1vcnlfdmlldzxsb25nPgBlbXNjcmlwdGVuOjptZW1vcnlfdmlldzx1bnNpZ25lZCBsb25nPgBlbXNjcmlwdGVuOjptZW1vcnlfdmlldzxpbnQ4X3Q+AGVtc2NyaXB0ZW46Om1lbW9yeV92aWV3PHVpbnQ4X3Q+AGVtc2NyaXB0ZW46Om1lbW9yeV92aWV3PGludDE2X3Q+AGVtc2NyaXB0ZW46Om1lbW9yeV92aWV3PHVpbnQxNl90PgBlbXNjcmlwdGVuOjptZW1vcnlfdmlldzxpbnQzMl90PgBlbXNjcmlwdGVuOjptZW1vcnlfdmlldzx1aW50MzJfdD4AZW1zY3JpcHRlbjo6bWVtb3J5X3ZpZXc8ZmxvYXQ+AGVtc2NyaXB0ZW46Om1lbW9yeV92aWV3PGRvdWJsZT4AZW1zY3JpcHRlbjo6bWVtb3J5X3ZpZXc8bG9uZyBkb3VibGU+AE4xMGVtc2NyaXB0ZW4xMW1lbW9yeV92aWV3SWVFRQBOMTBlbXNjcmlwdGVuMTFtZW1vcnlfdmlld0lkRUUATjEwZW1zY3JpcHRlbjExbWVtb3J5X3ZpZXdJZkVFAE4xMGVtc2NyaXB0ZW4xMW1lbW9yeV92aWV3SW1FRQBOMTBlbXNjcmlwdGVuMTFtZW1vcnlfdmlld0lsRUUATjEwZW1zY3JpcHRlbjExbWVtb3J5X3ZpZXdJakVFAE4xMGVtc2NyaXB0ZW4xMW1lbW9yeV92aWV3SWlFRQBOMTBlbXNjcmlwdGVuMTFtZW1vcnlfdmlld0l0RUUATjEwZW1zY3JpcHRlbjExbWVtb3J5X3ZpZXdJc0VFAE4xMGVtc2NyaXB0ZW4xMW1lbW9yeV92aWV3SWhFRQBOMTBlbXNjcmlwdGVuMTFtZW1vcnlfdmlld0lhRUUATjEwZW1zY3JpcHRlbjExbWVtb3J5X3ZpZXdJY0VFAE4xMGVtc2NyaXB0ZW4zdmFsRQBOU3QzX18yMTJiYXNpY19zdHJpbmdJaE5TXzExY2hhcl90cmFpdHNJaEVFTlNfOWFsbG9jYXRvckloRUVFRQASERMUFRYXGBkaGxwdHh8gIREiIyQRJSYnKCkqKywRLS4vEBAwEBAQEBAQEDEyMxA0NRAQERERERERERERERERERERERERERERERERETYRERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERERE3ERERETgROTo7PD0+ERERERERERERERERERERERERERERERERERERERERERERERERERERERERET8QEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBARQEERQkNERUZHSEkQEBBKS0xNThAQEE9QEBAQEFEQEBAQEBAQEBARERFSUxAQEBAQEBAQEBAQEREREVQQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAREVUQEBAQVhAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBBXEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBBYWVpbEBAQEBAQEBAQEBAQEBAQEBAQEBAQEFwQEBAQEBAQEBAQEBAQEBAQEABBusMACyD//////////////////////////////////////////wBB4sMACzb+//8H/v//BwAAAAAABCAE//9/////f//////////////////////////////////D/wMAH1AAQaLEAAukASAAAAAAAN88QNf///v///////////+///////////////////////8D/P////////////////////////8A/v///38C/v////8AAAAAAP+/tgD///8HBwAAAP8H//////////7/w////////////////+8f/uH/nwAA////////AOD///////////////8DAP//////BzAE/////P8fAAD///8BAEHOxQAL5AH9HwAAAAAAAPAD/3//////////7//f4f/P//7+7p/5///9xeOfWYCwz/8DAO6H+f///W3DhxkCXsD/PwDuv/v///3t478bAQDP/wAA7p/5///97eOfGcCwz/8CAOzHPdYYx//Dxx2BAMD/AADu3/3///3v498dYAPP/wAA7N/9///97+PfHWBAz/8GAOzf/f/////n312AAM//APzs/3/8///7L3+AX/8AAAwA/v////9//wc/IP8DAAAAAJYl8P6u7P87XyD/8wAAAAABAAAA/wMAAP/+////H/7/A////v///x8AQbrHAAuVAv///////3/5/wP//+fB//9/QP8z/////78g///////3////////////PX89//////89/////z1/Pf9//////////z3//////////4cAAAAA//8AAP////////////8fAP7//////////////////////////////////////////////////////////5////7//wf////////////HAQD/3w8A//8PAP//DwD/3w0A////////z///AYAQ/wMAAAAA/wP//////////////wD//////wf//////////z8A////H/8P/wHA/////z8fAP//////D////wP/AwAAAAD///8P/////////3/+/x8A/wP/A4AAQdrJAAsw////////7//vD/8DAAAAAP//////8////////7//AwD///////8/AP/j//////8/AEGXygALG95vAP///////////////////////////////wBBusoACyD//z8//////z8//6r///8/////////31/cH88P/x/cHwBB6MoACwYCgAAA/x8AQfrKAAsShPwvPlC9//PgQwAA//////8BAEGwywALUMD///////8DAAD//////3///////3//////////////////////H3gMAP////+/IP////////+AAAD//38Af39/f39/f3//////AAAAAACAAEGazAALV+AAAAD+Az4f/v///////////3/g/v/////////////34P////8//v////////////9/AAD///8HAAAAAAAA////////////////////////////////PwBB+swACzL/////////////////////////////////HwAAAAAAAP//////////////////////HwBBtM0ACzz//////z//H////w8AAP//////f/CP////gP////////////8AAAAAgP/8////////////////eQ8A/wcAQfnNAAtn/7v3////AAAA////////DwD//////////w8A/wMAAPwI//////8H/////wcA////H/////////f/AID/AwAAAAD///////9/AP8//wP//38E/////////38FAAA4//88AH5+fgB/fwBB8s4AC4gB//////8H/wP//////////////////////////w8A//9/+P//////D/////////////////8//////////////////wMAAAAAfwD44P/9f1/b/////////////////wMAAAD4////////////////PwAA///////////8////////AAAAAAD/DwBBiNAAC1Lf/////////////////////x8AAP8D/v//B/7//wfA/////////////3/8/PwcAAAAAP/v//9///+3/z//PwAAAAD///////////////////8HAEHi0AALB////////x8AQYrRAAtG////H////////wEAAAAAAP///38AAP///wcAAAAAAAD///8//////w//PgAAAAAA/////////////////////////z//AwBB2tEACws//f////+/kf//PwBB+tEACwj//z8A////AwBBitIACwj/////////wABBmtIACxBv8O/+//8PAAAAAAD///8fAEG60gALD////////z8A//8/AP//BwBB2tIACwr///////////8BAEH60gALKP//////////PwAAAMD/AAD8////////AQAA////Af8D////////x/8AQarTAAsM//////////8eAP8DAEHK0wALHv///////z8A/wMAAAAAAAD/////////////////fwBB+tMACw3///////////////8HAEGa1AALBv//////fwBButQACwj/////////AQBB2tQACxT//////////x8A//////9/AAD4/wBB+tQACwEDAEGa1QALmAH/////////////3///////////32Te/+vv/////////7/n39////97X/z9//////////////////////////////////////////////////////8//////f//9/////f//9/////f//9/////f/////3////9///3z////////+////+W/vcKhOqWqpb3917/+/8P7vv/DwBButYAC+sPVCEiGQ0BAgMRSxwMEAQLHRIeJ2hub3BxYiAFBg8TFBUaCBYHKCQXGAkKDhsfJSODgn0mKis8PT4/Q0dKTVhZWltcXV5fYGFjZGVmZ2lqa2xyc3R5ent8AElsbGVnYWwgYnl0ZSBzZXF1ZW5jZQBEb21haW4gZXJyb3IAUmVzdWx0IG5vdCByZXByZXNlbnRhYmxlAE5vdCBhIHR0eQBQZXJtaXNzaW9uIGRlbmllZABPcGVyYXRpb24gbm90IHBlcm1pdHRlZABObyBzdWNoIGZpbGUgb3IgZGlyZWN0b3J5AE5vIHN1Y2ggcHJvY2VzcwBGaWxlIGV4aXN0cwBWYWx1ZSB0b28gbGFyZ2UgZm9yIGRhdGEgdHlwZQBObyBzcGFjZSBsZWZ0IG9uIGRldmljZQBPdXQgb2YgbWVtb3J5AFJlc291cmNlIGJ1c3kASW50ZXJydXB0ZWQgc3lzdGVtIGNhbGwAUmVzb3VyY2UgdGVtcG9yYXJpbHkgdW5hdmFpbGFibGUASW52YWxpZCBzZWVrAENyb3NzLWRldmljZSBsaW5rAFJlYWQtb25seSBmaWxlIHN5c3RlbQBEaXJlY3Rvcnkgbm90IGVtcHR5AENvbm5lY3Rpb24gcmVzZXQgYnkgcGVlcgBPcGVyYXRpb24gdGltZWQgb3V0AENvbm5lY3Rpb24gcmVmdXNlZABIb3N0IGlzIGRvd24ASG9zdCBpcyB1bnJlYWNoYWJsZQBBZGRyZXNzIGluIHVzZQBCcm9rZW4gcGlwZQBJL08gZXJyb3IATm8gc3VjaCBkZXZpY2Ugb3IgYWRkcmVzcwBCbG9jayBkZXZpY2UgcmVxdWlyZWQATm8gc3VjaCBkZXZpY2UATm90IGEgZGlyZWN0b3J5AElzIGEgZGlyZWN0b3J5AFRleHQgZmlsZSBidXN5AEV4ZWMgZm9ybWF0IGVycm9yAEludmFsaWQgYXJndW1lbnQAQXJndW1lbnQgbGlzdCB0b28gbG9uZwBTeW1ib2xpYyBsaW5rIGxvb3AARmlsZW5hbWUgdG9vIGxvbmcAVG9vIG1hbnkgb3BlbiBmaWxlcyBpbiBzeXN0ZW0ATm8gZmlsZSBkZXNjcmlwdG9ycyBhdmFpbGFibGUAQmFkIGZpbGUgZGVzY3JpcHRvcgBObyBjaGlsZCBwcm9jZXNzAEJhZCBhZGRyZXNzAEZpbGUgdG9vIGxhcmdlAFRvbyBtYW55IGxpbmtzAE5vIGxvY2tzIGF2YWlsYWJsZQBSZXNvdXJjZSBkZWFkbG9jayB3b3VsZCBvY2N1cgBTdGF0ZSBub3QgcmVjb3ZlcmFibGUAUHJldmlvdXMgb3duZXIgZGllZABPcGVyYXRpb24gY2FuY2VsZWQARnVuY3Rpb24gbm90IGltcGxlbWVudGVkAE5vIG1lc3NhZ2Ugb2YgZGVzaXJlZCB0eXBlAElkZW50aWZpZXIgcmVtb3ZlZABEZXZpY2Ugbm90IGEgc3RyZWFtAE5vIGRhdGEgYXZhaWxhYmxlAERldmljZSB0aW1lb3V0AE91dCBvZiBzdHJlYW1zIHJlc291cmNlcwBMaW5rIGhhcyBiZWVuIHNldmVyZWQAUHJvdG9jb2wgZXJyb3IAQmFkIG1lc3NhZ2UARmlsZSBkZXNjcmlwdG9yIGluIGJhZCBzdGF0ZQBOb3QgYSBzb2NrZXQARGVzdGluYXRpb24gYWRkcmVzcyByZXF1aXJlZABNZXNzYWdlIHRvbyBsYXJnZQBQcm90b2NvbCB3cm9uZyB0eXBlIGZvciBzb2NrZXQAUHJvdG9jb2wgbm90IGF2YWlsYWJsZQBQcm90b2NvbCBub3Qgc3VwcG9ydGVkAFNvY2tldCB0eXBlIG5vdCBzdXBwb3J0ZWQATm90IHN1cHBvcnRlZABQcm90b2NvbCBmYW1pbHkgbm90IHN1cHBvcnRlZABBZGRyZXNzIGZhbWlseSBub3Qgc3VwcG9ydGVkIGJ5IHByb3RvY29sAEFkZHJlc3Mgbm90IGF2YWlsYWJsZQBOZXR3b3JrIGlzIGRvd24ATmV0d29yayB1bnJlYWNoYWJsZQBDb25uZWN0aW9uIHJlc2V0IGJ5IG5ldHdvcmsAQ29ubmVjdGlvbiBhYm9ydGVkAE5vIGJ1ZmZlciBzcGFjZSBhdmFpbGFibGUAU29ja2V0IGlzIGNvbm5lY3RlZABTb2NrZXQgbm90IGNvbm5lY3RlZABDYW5ub3Qgc2VuZCBhZnRlciBzb2NrZXQgc2h1dGRvd24AT3BlcmF0aW9uIGFscmVhZHkgaW4gcHJvZ3Jlc3MAT3BlcmF0aW9uIGluIHByb2dyZXNzAFN0YWxlIGZpbGUgaGFuZGxlAFJlbW90ZSBJL08gZXJyb3IAUXVvdGEgZXhjZWVkZWQATm8gbWVkaXVtIGZvdW5kAFdyb25nIG1lZGl1bSB0eXBlAE5vIGVycm9yIGluZm9ybWF0aW9uAABMQ19BTEwATENfQ1RZUEUAAAAATENfTlVNRVJJQwAATENfVElNRQAAAAAATENfQ09MTEFURQAATENfTU9ORVRBUlkATENfTUVTU0FHRVMATEFORwBDLlVURi04AFBPU0lYAE1VU0xfTE9DUEFUSAARAAoAERERAAAAAAUAAAAAAAAJAAAAAAsAQa3mAAshEQAPChEREQMKBwABEwkLCwAACQYLAAALAAYRAAAAERERAEHe5gALAQsAQefmAAsYEQAKChEREQAKAAACAAkLAAAACQALAAALAEGY5wALAQwAQaTnAAsVDAAAAAAMAAAAAAkMAAAAAAAMAAAMAEHS5wALAQ4AQd7nAAsVDQAAAAQNAAAAAAkOAAAAAAAOAAAOAEGM6AALARAAQZjoAAseDwAAAAAPAAAAAAkQAAAAAAAQAAAQAAASAAAAEhISAEHP6AALDhIAAAASEhIAAAAAAAAJAEGA6QALAQsAQYzpAAsVCgAAAAAKAAAAAAkLAAAAAAALAAALAEG66QALAQwAQcbpAAtdDAAAAAAMAAAAAAkMAAAAAAAMAAAMAAAtKyAgIDBYMHgALTBYKzBYIDBYLTB4KzB4IDB4AGluZgBJTkYAbmFuAE5BTgAwMTIzNDU2Nzg5QUJDREVGLgBDLlVURi04AEGs6gALGkMAEQAKABEREQAAAAAFAAAAAAAACQAAAAALAEHO6gALGhEADwoREREDCgcAARMJCwsAAAkGCwAACwAGAEH/6gALAQsAQYrrAAsWCgoAAAAACgAAAgAJCwAAAAkACwAACwBBuesACwEMAEHF6wALFQwAAAAADAAAAAAJDAAAAAAADAAADABB8+sACwEOAEH/6wALFQ0AAAAEDQAAAAAJDgAAAAAADgAADgBBrewACwEQAEG57AALHg8AAAAADwAAAAAJEAAAAAAAEAAAEAAAEgAAABISEgBB8OwACw4SAAAAEhISAAAAAAAACQBBoe0ACwELAEGt7QALFQoAAAAACgAAAAAJCwAAAAAACwAACwBB2+0ACwEMAEHn7QALvAcMAAAAAAwAAAAACQwAAAAAAAwAAAwAACUqcwAobnVsbCkAIwArAC0AIAAwAEwAAGpMTEwAagAAAAAAamoAAAAAagAAaiUlJXMlcyVzJXMlcyouKiVjJWMAYmFzaWNfc3RyaW5nAGFsbG9jYXRvcjxUPjo6YWxsb2NhdGUoc2l6ZV90IG4pICduJyBleGNlZWRzIG1heGltdW0gc3VwcG9ydGVkIHNpemUAdGVybWluYXRpbmcgd2l0aCAlcyBleGNlcHRpb24gb2YgdHlwZSAlczogJXMAdGVybWluYXRpbmcgd2l0aCAlcyBleGNlcHRpb24gb2YgdHlwZSAlcwB0ZXJtaW5hdGluZyB3aXRoICVzIGZvcmVpZ24gZXhjZXB0aW9uAHRlcm1pbmF0aW5nAHVuY2F1Z2h0AFN0OWV4Y2VwdGlvbgBOMTBfX2N4eGFiaXYxMTZfX3NoaW1fdHlwZV9pbmZvRQBTdDl0eXBlX2luZm8ATjEwX19jeHhhYml2MTIwX19zaV9jbGFzc190eXBlX2luZm9FAE4xMF9fY3h4YWJpdjExN19fY2xhc3NfdHlwZV9pbmZvRQBwdGhyZWFkX29uY2UgZmFpbHVyZSBpbiBfX2N4YV9nZXRfZ2xvYmFsc19mYXN0KCkAY2Fubm90IGNyZWF0ZSBwdGhyZWFkIGtleSBmb3IgX19jeGFfZ2V0X2dsb2JhbHMoKQBjYW5ub3QgemVybyBvdXQgdGhyZWFkIHZhbHVlIGZvciBfX2N4YV9nZXRfZ2xvYmFscygpAHRlcm1pbmF0ZV9oYW5kbGVyIHVuZXhwZWN0ZWRseSByZXR1cm5lZAB0ZXJtaW5hdGVfaGFuZGxlciB1bmV4cGVjdGVkbHkgdGhyZXcgYW4gZXhjZXB0aW9uAHN0ZDo6YmFkX2FsbG9jAFN0OWJhZF9hbGxvYwBTdDExbG9naWNfZXJyb3IAU3QxMmxlbmd0aF9lcnJvcgBOMTBfX2N4eGFiaXYxMTlfX3BvaW50ZXJfdHlwZV9pbmZvRQBOMTBfX2N4eGFiaXYxMTdfX3BiYXNlX3R5cGVfaW5mb0UATjEwX19jeHhhYml2MTIzX19mdW5kYW1lbnRhbF90eXBlX2luZm9FAHYARG4AYgBjAGgAYQBzAHQAaQBqAGwAbQBmAGQATjEwX19jeHhhYml2MTE2X19lbnVtX3R5cGVfaW5mb0UATjEwX19jeHhhYml2MTIxX192bWlfY2xhc3NfdHlwZV9pbmZvRQ==";class ResourceLoaderPlugin{getType(){return"none"}async loadFile(A){return null}}const Module={},WorkerScope={};!function(A){A.unpackBridge=function(A){return function(A){var g={};function I(C){if(g[C])return g[C].exports;var e=g[C]={i:C,l:!1,exports:{}};return A[C].call(e.exports,e,e.exports,I),e.l=!0,e.exports}return I.m=A,I.c=g,I.d=function(A,g,C){I.o(A,g)||Object.defineProperty(A,g,{enumerable:!0,get:C})},I.r=function(A){"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(A,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(A,"__esModule",{value:!0})},I.t=function(A,g){if(1&g&&(A=I(A)),8&g)return A;if(4&g&&"object"==typeof A&&A&&A.__esModule)return A;var C=Object.create(null);if(I.r(C),Object.defineProperty(C,"default",{enumerable:!0,value:A}),2&g&&"string"!=typeof A)for(var e in A)I.d(C,e,function(g){return A[g]}.bind(null,e));return C},I.n=function(A){var g=A&&A.__esModule?function(){return A.default}:function(){return A};return I.d(g,"a",g),g},I.o=function(A,g){return Object.prototype.hasOwnProperty.call(A,g)},I.p="",I(I.s=2)}([function(A,g,I){Object.defineProperty(g,"__esModule",{value:!0});const C=I(1),e={0:"ERAR_SUCCESS",10:"ERAR_END_ARCHIVE",11:"ERAR_NO_MEMORY",12:"ERAR_BAD_DATA",13:"ERAR_BAD_ARCHIVE",14:"ERAR_UNKNOWN_FORMAT",15:"ERAR_EOPEN",16:"ERAR_ECREATE",17:"ERAR_ECLOSE",18:"ERAR_EREAD",19:"ERAR_EWRITE",20:"ERAR_SMALL_BUF",21:"ERAR_UNKNOWN",22:"ERAR_MISSING_PASSWORD",23:"ERAR_EREFERENCE",24:"ERAR_BAD_PASSWORD"},i={0:"Success",11:"Not enough memory",12:"Archive header or data are damaged",13:"File is not RAR archive",14:"Unknown archive format",15:"File open error",16:"File create error",17:"File close error",18:"File read error",19:"File write error",20:"Buffer for archive comment is too small, comment truncated",21:"Unknown error",22:"Password for encrypted file or header is not specified",23:"Cannot open file source for reference record",24:"Wrong password is specified"};class t{constructor(A=""){this._password=A,this._archive=null}getFileList(){let A,[g,I]=this.openArc(!0);if("SUCCESS"!==g.state)A=[g,null];else{let g,C,e=[];for(;[g,C]=this.processNextFile((()=>!0)),"SUCCESS"===g.state;)e.push(C.fileHeader);A="ERAR_END_ARCHIVE"!==g.reason?[g,null]:[{state:"SUCCESS"},{arcHeader:I,fileHeaders:e}]}return this.closeArc(),A}extractAll(){let A,[g,I]=this.openArc(!1);if("SUCCESS"!==g.state)A=[g,null];else{let g,C,e=[];for(;[g,C]=this.processNextFile((()=>!1)),"SUCCESS"===g.state;)e.push(C);A="ERAR_END_ARCHIVE"!==g.reason?[g,null]:[{state:"SUCCESS"},{arcHeader:I,files:e}]}return this.closeArc(),A}extractFiles(A,g){let I,[C,e]=this.openArc(!1,g),i={};for(let g=0;g<A.length;++g)i[A[g]]=g;if("SUCCESS"!==C.state)I=[C,null];else{let g,C,t=Array(A.length).fill(null),o=0;for(;;){let I=!1,e=null;if([g,C]=this.processNextFile((A=>A in i?(e=i[A],!1):(I=!0,!0))),"SUCCESS"!==g.state)break;if(!I&&(t[e]=C,++o===A.length)){g.reason="ERAR_END_ARCHIVE";break}}I="ERAR_END_ARCHIVE"!==g.reason?[g,null]:[{state:"SUCCESS"},{arcHeader:e,files:t}]}return this.closeArc(),I}fileCreated(A){}close(A){this._lastFileContent=this.closeFile(A)}openArc(A,g){C.Ext.current=this,this._archive=new unpack.RarArchive;let I,e=this._archive.open(this._filePath,g||this._password,A);return I=0!==e.state.errCode?[this.getFailInfo(e.state.errCode,e.state.errType),null]:[{state:"SUCCESS"},{comment:e.comment,flags:{volume:!!(1&e.flags),lock:!!(4&e.flags),solid:!!(8&e.flags),authInfo:!!(32&e.flags),recoveryRecord:!!(64&e.flags),headerEncrypted:!!(128&e.flags)}}],C.Ext.current=null,I}processNextFile(A){let g;C.Ext.current=this;let I=this._archive.getFileHeader(),e=[{state:"SUCCESS"},null];if(0===I.state.errCode){let g=A(I.name);this._lastFileContent=null;let C=this._archive.readFile(g);0===C.errCode||g||(e[0]=this.getFailInfo(C.errCode,C.errType),22===C.errCode?C=this._archive.readFile(!0):C.errCode=0),0===C.errCode?e[1]=this._lastFileContent:(I.state.errCode=C.errCode,I.state.errType=C.errType),this._lastFileContent=null}return g=0!==I.state.errCode?[this.getFailInfo(I.state.errCode,I.state.errType),null]:[{state:"SUCCESS"},{fileHeader:{name:I.name,flags:{encrypted:!!(4&I.flags),solid:!!(16&I.flags),directory:!!(32&I.flags)},packSize:I.packSize,unpSize:I.unpSize,crc:I.crc,time:function(A){const g=[5,6,5,5,4,7];let I=[];for(let C of g)I.push(A&(1<<C)-1),A>>=C;let C=A=>A<10?"0"+A:""+A;return`${1980+(I=I.reverse())[0]}-${C(I[1])}-${C(I[2])}T${C(I[3])}:${C(I[4])}:${C(2*I[5])}.000`}(I.time),unpVer:`${Math.floor(I.unpVer/10)}.${I.unpVer%10}`,method:function(A){return{48:"Storing",49:"Fastest",50:"Fast",51:"Normal",52:"Good",53:"Best"}[A]||"Unknown"}(I.method)},extract:e}],C.Ext.current=null,g}closeArc(){C.Ext.current=this,this._archive.delete(),C.Ext.current=null,this._archive=null}getFailInfo(A,g){return{state:"FAIL",reason:e[A],msg:i[A]}}}t._current=null,g.Extractor=t},function(A,g,I){Object.defineProperty(g,"__esModule",{value:!0}),g.Ext={current:null}},function(A,g,I){Object.defineProperty(g,"__esModule",{value:!0}),function(A){for(var I in A)g.hasOwnProperty(I)||(g[I]=A[I])}(I(3));var C=I(1);g.Ext=C.Ext},function(A,g,I){Object.defineProperty(g,"__esModule",{value:!0});const C=I(4),e=I(6);g.createExtractorFromData=function(A,g=""){return new C.DataExtractor(A,g)},g.createExtractorFromFile=function(A,g="",I=""){return new e.FileExtractor(A,g,I)}},function(A,g,I){Object.defineProperty(g,"__esModule",{value:!0});const C=I(5),e=I(0);g.DataExtractor=class extends e.Extractor{constructor(A,g){super(g),this.dataFiles={},this.dataFileMap={},this.currentFd=1;let I={file:new C.DataFile(new Uint8Array(A)),fd:this.currentFd++};this._filePath="_defaultUnrarJS_.rar",this.dataFiles[this._filePath]=I,this.dataFileMap[I.fd]=this._filePath}open(A){let g=this.dataFiles[A];return g?g.fd:0}create(A){let g=this.currentFd++;return this.dataFiles[A]={file:new C.DataFile,fd:this.currentFd++},this.dataFileMap[g]=A,g}closeFile(A){let g=this.dataFiles[this.dataFileMap[A]];if(!g)return null;let I=g.file.readAll();return 1!==A?(delete this.dataFiles[this.dataFileMap[A]],delete this.dataFileMap[A]):g.file.seek(0,"SET"),I}read(A,g,I){let C=this.dataFiles[this.dataFileMap[A]];if(!C)return-1;let e=C.file.read(I);return null===e?-1:(unpack.HEAPU8.set(e,g),e.byteLength)}write(A,g,I){let C=this.dataFiles[this.dataFileMap[A]];return!!C&&(C.file.write(unpack.HEAPU8.slice(g,g+I)),!0)}tell(A){let g=this.dataFiles[this.dataFileMap[A]];return g?g.file.tell():-1}seek(A,g,I){let C=this.dataFiles[this.dataFileMap[A]];return!!C&&C.file.seek(g,I)}}},function(A,g,I){Object.defineProperty(g,"__esModule",{value:!0}),g.DataFile=class{constructor(A){this.buffers=[],this.pos=0,this.size=0,A&&(this.buffers.push(A),this.size=A.byteLength,this.pos=0)}read(A){if(this.flatten(),A+this.pos>this.size)return null;let g=this.pos;return this.pos+=A,this.buffers[0].slice(g,this.pos)}readAll(){return this.flatten(),this.buffers[0]}write(A){return this.buffers.push(A),this.size+=A.byteLength,this.pos+=A.byteLength,!0}tell(){return this.pos}seek(A,g){let I=this.pos;return"SET"===g?I=A:"CUR"===g?I+=A:I=this.size-A,!(I<0||I>this.size||(this.pos=I,0))}flatten(){if(this.buffers.length<=1)return;let A=new Uint8Array(this.size),g=0;for(let I of this.buffers)A.set(I,g),g+=I.byteLength;this.buffers=[A]}}},function(A,g,I){(function(A){Object.defineProperty(g,"__esModule",{value:!0});const C=I(12),e=I(13),i=I(0);g.FileExtractor=class extends i.Extractor{constructor(A,g,I){super(I),this._filePath=A,this.fileMap={},this._target=g}open(A){let g=C.openSync(A,"r");return this.fileMap[g]={size:C.fstatSync(g).size,pos:0,name:A},g}create(A){let g=e.join(this._target,A);e.parse(g).dir.split("/").reduce(((A,g)=>(A+=g+"/",C.existsSync(A)||C.mkdirSync(A),A)),"");let I=C.openSync(g,"w");return this.fileMap[I]={size:0,pos:0,name:A},I}closeFile(A){return delete this.fileMap[A],C.closeSync(A),null}read(g,I,e){let i=this.fileMap[g],t=new A(e),o=C.readSync(g,t,0,e,i.pos);return unpack.HEAPU8.set(t,I),i.pos+=o,o}write(g,I,e){let i=this.fileMap[g],t=C.writeSync(g,new A(unpack.HEAPU8.subarray(I,I+e)),0,e);return i.pos+=t,i.size+=t,t===e}tell(A){return this.fileMap[A].pos}seek(A,g,I){let C=this.fileMap[A],e=C.pos;return"SET"===I?e=0:"END"===I&&(e=C.size),!((e+=g)<0||e>C.size||(C.pos=e,0))}}}).call(this,I(7).Buffer)},function(A,g,I){(function(A){
/*!
         * The buffer module from node.js, for the browser.
         *
         * @author   Feross Aboukhadijeh <feross@feross.org> <http://feross.org>
         * @license  MIT
         */
var C=I(9),e=I(10),i=I(11);function t(){return a.TYPED_ARRAY_SUPPORT?2147483647:1073741823}function o(A,g){if(t()<g)throw new RangeError("Invalid typed array length");return a.TYPED_ARRAY_SUPPORT?(A=new Uint8Array(g)).__proto__=a.prototype:(null===A&&(A=new a(g)),A.length=g),A}function a(A,g,I){if(!(a.TYPED_ARRAY_SUPPORT||this instanceof a))return new a(A,g,I);if("number"==typeof A){if("string"==typeof g)throw new Error("If encoding is specified then the first argument must be a string");return l(this,A)}return B(this,A,g,I)}function B(A,g,I,C){if("number"==typeof g)throw new TypeError('"value" argument must not be a number');return"undefined"!=typeof ArrayBuffer&&g instanceof ArrayBuffer?function(A,g,I,C){if(g.byteLength,I<0||g.byteLength<I)throw new RangeError("'offset' is out of bounds");if(g.byteLength<I+(C||0))throw new RangeError("'length' is out of bounds");return g=void 0===I&&void 0===C?new Uint8Array(g):void 0===C?new Uint8Array(g,I):new Uint8Array(g,I,C),a.TYPED_ARRAY_SUPPORT?(A=g).__proto__=a.prototype:A=n(A,g),A}(A,g,I,C):"string"==typeof g?function(A,g,I){if("string"==typeof I&&""!==I||(I="utf8"),!a.isEncoding(I))throw new TypeError('"encoding" must be a valid string encoding');var C=0|r(g,I),e=(A=o(A,C)).write(g,I);return e!==C&&(A=A.slice(0,e)),A}(A,g,I):function(A,g){if(a.isBuffer(g)){var I=0|d(g.length);return 0===(A=o(A,I)).length||g.copy(A,0,0,I),A}if(g){if("undefined"!=typeof ArrayBuffer&&g.buffer instanceof ArrayBuffer||"length"in g)return"number"!=typeof g.length||function(A){return A!=A}(g.length)?o(A,0):n(A,g);if("Buffer"===g.type&&i(g.data))return n(A,g.data)}throw new TypeError("First argument must be a string, Buffer, ArrayBuffer, Array, or array-like object.")}(A,g)}function s(A){if("number"!=typeof A)throw new TypeError('"size" argument must be a number');if(A<0)throw new RangeError('"size" argument must not be negative')}function l(A,g){if(s(g),A=o(A,g<0?0:0|d(g)),!a.TYPED_ARRAY_SUPPORT)for(var I=0;I<g;++I)A[I]=0;return A}function n(A,g){var I=g.length<0?0:0|d(g.length);A=o(A,I);for(var C=0;C<I;C+=1)A[C]=255&g[C];return A}function d(A){if(A>=t())throw new RangeError("Attempt to allocate Buffer larger than maximum size: 0x"+t().toString(16)+" bytes");return 0|A}function r(A,g){if(a.isBuffer(A))return A.length;if("undefined"!=typeof ArrayBuffer&&"function"==typeof ArrayBuffer.isView&&(ArrayBuffer.isView(A)||A instanceof ArrayBuffer))return A.byteLength;"string"!=typeof A&&(A=""+A);var I=A.length;if(0===I)return 0;for(var C=!1;;)switch(g){case"ascii":case"latin1":case"binary":return I;case"utf8":case"utf-8":case void 0:return L(A).length;case"ucs2":case"ucs-2":case"utf16le":case"utf-16le":return 2*I;case"hex":return I>>>1;case"base64":return M(A).length;default:if(C)return L(A).length;g=(""+g).toLowerCase(),C=!0}}function c(A,g,I){var C=A[g];A[g]=A[I],A[I]=C}function h(A,g,I,C,e){if(0===A.length)return-1;if("string"==typeof I?(C=I,I=0):I>2147483647?I=2147483647:I<-2147483648&&(I=-2147483648),I=+I,isNaN(I)&&(I=e?0:A.length-1),I<0&&(I=A.length+I),I>=A.length){if(e)return-1;I=A.length-1}else if(I<0){if(!e)return-1;I=0}if("string"==typeof g&&(g=a.from(g,C)),a.isBuffer(g))return 0===g.length?-1:G(A,g,I,C,e);if("number"==typeof g)return g&=255,a.TYPED_ARRAY_SUPPORT&&"function"==typeof Uint8Array.prototype.indexOf?e?Uint8Array.prototype.indexOf.call(A,g,I):Uint8Array.prototype.lastIndexOf.call(A,g,I):G(A,[g],I,C,e);throw new TypeError("val must be string, number or Buffer")}function G(A,g,I,C,e){var i,t=1,o=A.length,a=g.length;if(void 0!==C&&("ucs2"===(C=String(C).toLowerCase())||"ucs-2"===C||"utf16le"===C||"utf-16le"===C)){if(A.length<2||g.length<2)return-1;t=2,o/=2,a/=2,I/=2}function B(A,g){return 1===t?A[g]:A.readUInt16BE(g*t)}if(e){var s=-1;for(i=I;i<o;i++)if(B(A,i)===B(g,-1===s?0:i-s)){if(-1===s&&(s=i),i-s+1===a)return s*t}else-1!==s&&(i-=i-s),s=-1}else for(I+a>o&&(I=o-a),i=I;i>=0;i--){for(var l=!0,n=0;n<a;n++)if(B(A,i+n)!==B(g,n)){l=!1;break}if(l)return i}return-1}function m(A,g,I,C){I=Number(I)||0;var e=A.length-I;C?(C=Number(C))>e&&(C=e):C=e;var i=g.length;if(i%2!=0)throw new TypeError("Invalid hex string");C>i/2&&(C=i/2);for(var t=0;t<C;++t){var o=parseInt(g.substr(2*t,2),16);if(isNaN(o))return t;A[I+t]=o}return t}function b(A,g,I,C){return U(L(g,A.length-I),A,I,C)}function Z(A,g,I,C){return U(function(A){for(var g=[],I=0;I<A.length;++I)g.push(255&A.charCodeAt(I));return g}(g),A,I,C)}function u(A,g,I,C){return Z(A,g,I,C)}function y(A,g,I,C){return U(M(g),A,I,C)}function E(A,g,I,C){return U(function(A,g){for(var I,C,e,i=[],t=0;t<A.length&&!((g-=2)<0);++t)C=(I=A.charCodeAt(t))>>8,e=I%256,i.push(e),i.push(C);return i}(g,A.length-I),A,I,C)}function Q(A,g,I){return 0===g&&I===A.length?C.fromByteArray(A):C.fromByteArray(A.slice(g,I))}function V(A,g,I){I=Math.min(A.length,I);for(var C=[],e=g;e<I;){var i,t,o,a,B=A[e],s=null,l=B>239?4:B>223?3:B>191?2:1;if(e+l<=I)switch(l){case 1:B<128&&(s=B);break;case 2:128==(192&(i=A[e+1]))&&(a=(31&B)<<6|63&i)>127&&(s=a);break;case 3:i=A[e+1],t=A[e+2],128==(192&i)&&128==(192&t)&&(a=(15&B)<<12|(63&i)<<6|63&t)>2047&&(a<55296||a>57343)&&(s=a);break;case 4:i=A[e+1],t=A[e+2],o=A[e+3],128==(192&i)&&128==(192&t)&&128==(192&o)&&(a=(15&B)<<18|(63&i)<<12|(63&t)<<6|63&o)>65535&&a<1114112&&(s=a)}null===s?(s=65533,l=1):s>65535&&(s-=65536,C.push(s>>>10&1023|55296),s=56320|1023&s),C.push(s),e+=l}return function(A){var g=A.length;if(g<=R)return String.fromCharCode.apply(String,A);for(var I="",C=0;C<g;)I+=String.fromCharCode.apply(String,A.slice(C,C+=R));return I}(C)}g.Buffer=a,g.SlowBuffer=function(A){return+A!=A&&(A=0),a.alloc(+A)},g.INSPECT_MAX_BYTES=50,a.TYPED_ARRAY_SUPPORT=void 0!==A.TYPED_ARRAY_SUPPORT?A.TYPED_ARRAY_SUPPORT:function(){try{var A=new Uint8Array(1);return A.__proto__={__proto__:Uint8Array.prototype,foo:function(){return 42}},42===A.foo()&&"function"==typeof A.subarray&&0===A.subarray(1,1).byteLength}catch(A){return!1}}(),g.kMaxLength=t(),a.poolSize=8192,a._augment=function(A){return A.__proto__=a.prototype,A},a.from=function(A,g,I){return B(null,A,g,I)},a.TYPED_ARRAY_SUPPORT&&(a.prototype.__proto__=Uint8Array.prototype,a.__proto__=Uint8Array,"undefined"!=typeof Symbol&&Symbol.species&&a[Symbol.species]===a&&Object.defineProperty(a,Symbol.species,{value:null,configurable:!0})),a.alloc=function(A,g,I){return function(A,g,I,C){return s(g),g<=0?o(A,g):void 0!==I?"string"==typeof C?o(A,g).fill(I,C):o(A,g).fill(I):o(A,g)}(null,A,g,I)},a.allocUnsafe=function(A){return l(null,A)},a.allocUnsafeSlow=function(A){return l(null,A)},a.isBuffer=function(A){return!(null==A||!A._isBuffer)},a.compare=function(A,g){if(!a.isBuffer(A)||!a.isBuffer(g))throw new TypeError("Arguments must be Buffers");if(A===g)return 0;for(var I=A.length,C=g.length,e=0,i=Math.min(I,C);e<i;++e)if(A[e]!==g[e]){I=A[e],C=g[e];break}return I<C?-1:C<I?1:0},a.isEncoding=function(A){switch(String(A).toLowerCase()){case"hex":case"utf8":case"utf-8":case"ascii":case"latin1":case"binary":case"base64":case"ucs2":case"ucs-2":case"utf16le":case"utf-16le":return!0;default:return!1}},a.concat=function(A,g){if(!i(A))throw new TypeError('"list" argument must be an Array of Buffers');if(0===A.length)return a.alloc(0);var I;if(void 0===g)for(g=0,I=0;I<A.length;++I)g+=A[I].length;var C=a.allocUnsafe(g),e=0;for(I=0;I<A.length;++I){var t=A[I];if(!a.isBuffer(t))throw new TypeError('"list" argument must be an Array of Buffers');t.copy(C,e),e+=t.length}return C},a.byteLength=r,a.prototype._isBuffer=!0,a.prototype.swap16=function(){var A=this.length;if(A%2!=0)throw new RangeError("Buffer size must be a multiple of 16-bits");for(var g=0;g<A;g+=2)c(this,g,g+1);return this},a.prototype.swap32=function(){var A=this.length;if(A%4!=0)throw new RangeError("Buffer size must be a multiple of 32-bits");for(var g=0;g<A;g+=4)c(this,g,g+3),c(this,g+1,g+2);return this},a.prototype.swap64=function(){var A=this.length;if(A%8!=0)throw new RangeError("Buffer size must be a multiple of 64-bits");for(var g=0;g<A;g+=8)c(this,g,g+7),c(this,g+1,g+6),c(this,g+2,g+5),c(this,g+3,g+4);return this},a.prototype.toString=function(){var A=0|this.length;return 0===A?"":0===arguments.length?V(this,0,A):function(A,g,I){var C=!1;if((void 0===g||g<0)&&(g=0),g>this.length)return"";if((void 0===I||I>this.length)&&(I=this.length),I<=0)return"";if((I>>>=0)<=(g>>>=0))return"";for(A||(A="utf8");;)switch(A){case"hex":return S(this,g,I);case"utf8":case"utf-8":return V(this,g,I);case"ascii":return W(this,g,I);case"latin1":case"binary":return p(this,g,I);case"base64":return Q(this,g,I);case"ucs2":case"ucs-2":case"utf16le":case"utf-16le":return F(this,g,I);default:if(C)throw new TypeError("Unknown encoding: "+A);A=(A+"").toLowerCase(),C=!0}}.apply(this,arguments)},a.prototype.equals=function(A){if(!a.isBuffer(A))throw new TypeError("Argument must be a Buffer");return this===A||0===a.compare(this,A)},a.prototype.inspect=function(){var A="",I=g.INSPECT_MAX_BYTES;return this.length>0&&(A=this.toString("hex",0,I).match(/.{2}/g).join(" "),this.length>I&&(A+=" ... ")),"<Buffer "+A+">"},a.prototype.compare=function(A,g,I,C,e){if(!a.isBuffer(A))throw new TypeError("Argument must be a Buffer");if(void 0===g&&(g=0),void 0===I&&(I=A?A.length:0),void 0===C&&(C=0),void 0===e&&(e=this.length),g<0||I>A.length||C<0||e>this.length)throw new RangeError("out of range index");if(C>=e&&g>=I)return 0;if(C>=e)return-1;if(g>=I)return 1;if(this===A)return 0;for(var i=(e>>>=0)-(C>>>=0),t=(I>>>=0)-(g>>>=0),o=Math.min(i,t),B=this.slice(C,e),s=A.slice(g,I),l=0;l<o;++l)if(B[l]!==s[l]){i=B[l],t=s[l];break}return i<t?-1:t<i?1:0},a.prototype.includes=function(A,g,I){return-1!==this.indexOf(A,g,I)},a.prototype.indexOf=function(A,g,I){return h(this,A,g,I,!0)},a.prototype.lastIndexOf=function(A,g,I){return h(this,A,g,I,!1)},a.prototype.write=function(A,g,I,C){if(void 0===g)C="utf8",I=this.length,g=0;else if(void 0===I&&"string"==typeof g)C=g,I=this.length,g=0;else{if(!isFinite(g))throw new Error("Buffer.write(string, encoding, offset[, length]) is no longer supported");g|=0,isFinite(I)?(I|=0,void 0===C&&(C="utf8")):(C=I,I=void 0)}var e=this.length-g;if((void 0===I||I>e)&&(I=e),A.length>0&&(I<0||g<0)||g>this.length)throw new RangeError("Attempt to write outside buffer bounds");C||(C="utf8");for(var i=!1;;)switch(C){case"hex":return m(this,A,g,I);case"utf8":case"utf-8":return b(this,A,g,I);case"ascii":return Z(this,A,g,I);case"latin1":case"binary":return u(this,A,g,I);case"base64":return y(this,A,g,I);case"ucs2":case"ucs-2":case"utf16le":case"utf-16le":return E(this,A,g,I);default:if(i)throw new TypeError("Unknown encoding: "+C);C=(""+C).toLowerCase(),i=!0}},a.prototype.toJSON=function(){return{type:"Buffer",data:Array.prototype.slice.call(this._arr||this,0)}};var R=4096;function W(A,g,I){var C="";I=Math.min(A.length,I);for(var e=g;e<I;++e)C+=String.fromCharCode(127&A[e]);return C}function p(A,g,I){var C="";I=Math.min(A.length,I);for(var e=g;e<I;++e)C+=String.fromCharCode(A[e]);return C}function S(A,g,I){var C=A.length;(!g||g<0)&&(g=0),(!I||I<0||I>C)&&(I=C);for(var e="",i=g;i<I;++i)e+=f(A[i]);return e}function F(A,g,I){for(var C=A.slice(g,I),e="",i=0;i<C.length;i+=2)e+=String.fromCharCode(C[i]+256*C[i+1]);return e}function w(A,g,I){if(A%1!=0||A<0)throw new RangeError("offset is not uint");if(A+g>I)throw new RangeError("Trying to access beyond buffer length")}function H(A,g,I,C,e,i){if(!a.isBuffer(A))throw new TypeError('"buffer" argument must be a Buffer instance');if(g>e||g<i)throw new RangeError('"value" argument is out of bounds');if(I+C>A.length)throw new RangeError("Index out of range")}function X(A,g,I,C){g<0&&(g=65535+g+1);for(var e=0,i=Math.min(A.length-I,2);e<i;++e)A[I+e]=(g&255<<8*(C?e:1-e))>>>8*(C?e:1-e)}function Y(A,g,I,C){g<0&&(g=4294967295+g+1);for(var e=0,i=Math.min(A.length-I,4);e<i;++e)A[I+e]=g>>>8*(C?e:3-e)&255}function K(A,g,I,C,e,i){if(I+C>A.length)throw new RangeError("Index out of range");if(I<0)throw new RangeError("Index out of range")}function N(A,g,I,C,i){return i||K(A,0,I,4),e.write(A,g,I,C,23,4),I+4}function v(A,g,I,C,i){return i||K(A,0,I,8),e.write(A,g,I,C,52,8),I+8}a.prototype.slice=function(A,g){var I,C=this.length;if((A=~~A)<0?(A+=C)<0&&(A=0):A>C&&(A=C),(g=void 0===g?C:~~g)<0?(g+=C)<0&&(g=0):g>C&&(g=C),g<A&&(g=A),a.TYPED_ARRAY_SUPPORT)(I=this.subarray(A,g)).__proto__=a.prototype;else{var e=g-A;I=new a(e,void 0);for(var i=0;i<e;++i)I[i]=this[i+A]}return I},a.prototype.readUIntLE=function(A,g,I){A|=0,g|=0,I||w(A,g,this.length);for(var C=this[A],e=1,i=0;++i<g&&(e*=256);)C+=this[A+i]*e;return C},a.prototype.readUIntBE=function(A,g,I){A|=0,g|=0,I||w(A,g,this.length);for(var C=this[A+--g],e=1;g>0&&(e*=256);)C+=this[A+--g]*e;return C},a.prototype.readUInt8=function(A,g){return g||w(A,1,this.length),this[A]},a.prototype.readUInt16LE=function(A,g){return g||w(A,2,this.length),this[A]|this[A+1]<<8},a.prototype.readUInt16BE=function(A,g){return g||w(A,2,this.length),this[A]<<8|this[A+1]},a.prototype.readUInt32LE=function(A,g){return g||w(A,4,this.length),(this[A]|this[A+1]<<8|this[A+2]<<16)+16777216*this[A+3]},a.prototype.readUInt32BE=function(A,g){return g||w(A,4,this.length),16777216*this[A]+(this[A+1]<<16|this[A+2]<<8|this[A+3])},a.prototype.readIntLE=function(A,g,I){A|=0,g|=0,I||w(A,g,this.length);for(var C=this[A],e=1,i=0;++i<g&&(e*=256);)C+=this[A+i]*e;return C>=(e*=128)&&(C-=Math.pow(2,8*g)),C},a.prototype.readIntBE=function(A,g,I){A|=0,g|=0,I||w(A,g,this.length);for(var C=g,e=1,i=this[A+--C];C>0&&(e*=256);)i+=this[A+--C]*e;return i>=(e*=128)&&(i-=Math.pow(2,8*g)),i},a.prototype.readInt8=function(A,g){return g||w(A,1,this.length),128&this[A]?-1*(255-this[A]+1):this[A]},a.prototype.readInt16LE=function(A,g){g||w(A,2,this.length);var I=this[A]|this[A+1]<<8;return 32768&I?4294901760|I:I},a.prototype.readInt16BE=function(A,g){g||w(A,2,this.length);var I=this[A+1]|this[A]<<8;return 32768&I?4294901760|I:I},a.prototype.readInt32LE=function(A,g){return g||w(A,4,this.length),this[A]|this[A+1]<<8|this[A+2]<<16|this[A+3]<<24},a.prototype.readInt32BE=function(A,g){return g||w(A,4,this.length),this[A]<<24|this[A+1]<<16|this[A+2]<<8|this[A+3]},a.prototype.readFloatLE=function(A,g){return g||w(A,4,this.length),e.read(this,A,!0,23,4)},a.prototype.readFloatBE=function(A,g){return g||w(A,4,this.length),e.read(this,A,!1,23,4)},a.prototype.readDoubleLE=function(A,g){return g||w(A,8,this.length),e.read(this,A,!0,52,8)},a.prototype.readDoubleBE=function(A,g){return g||w(A,8,this.length),e.read(this,A,!1,52,8)},a.prototype.writeUIntLE=function(A,g,I,C){A=+A,g|=0,I|=0,C||H(this,A,g,I,Math.pow(2,8*I)-1,0);var e=1,i=0;for(this[g]=255&A;++i<I&&(e*=256);)this[g+i]=A/e&255;return g+I},a.prototype.writeUIntBE=function(A,g,I,C){A=+A,g|=0,I|=0,C||H(this,A,g,I,Math.pow(2,8*I)-1,0);var e=I-1,i=1;for(this[g+e]=255&A;--e>=0&&(i*=256);)this[g+e]=A/i&255;return g+I},a.prototype.writeUInt8=function(A,g,I){return A=+A,g|=0,I||H(this,A,g,1,255,0),a.TYPED_ARRAY_SUPPORT||(A=Math.floor(A)),this[g]=255&A,g+1},a.prototype.writeUInt16LE=function(A,g,I){return A=+A,g|=0,I||H(this,A,g,2,65535,0),a.TYPED_ARRAY_SUPPORT?(this[g]=255&A,this[g+1]=A>>>8):X(this,A,g,!0),g+2},a.prototype.writeUInt16BE=function(A,g,I){return A=+A,g|=0,I||H(this,A,g,2,65535,0),a.TYPED_ARRAY_SUPPORT?(this[g]=A>>>8,this[g+1]=255&A):X(this,A,g,!1),g+2},a.prototype.writeUInt32LE=function(A,g,I){return A=+A,g|=0,I||H(this,A,g,4,4294967295,0),a.TYPED_ARRAY_SUPPORT?(this[g+3]=A>>>24,this[g+2]=A>>>16,this[g+1]=A>>>8,this[g]=255&A):Y(this,A,g,!0),g+4},a.prototype.writeUInt32BE=function(A,g,I){return A=+A,g|=0,I||H(this,A,g,4,4294967295,0),a.TYPED_ARRAY_SUPPORT?(this[g]=A>>>24,this[g+1]=A>>>16,this[g+2]=A>>>8,this[g+3]=255&A):Y(this,A,g,!1),g+4},a.prototype.writeIntLE=function(A,g,I,C){if(A=+A,g|=0,!C){var e=Math.pow(2,8*I-1);H(this,A,g,I,e-1,-e)}var i=0,t=1,o=0;for(this[g]=255&A;++i<I&&(t*=256);)A<0&&0===o&&0!==this[g+i-1]&&(o=1),this[g+i]=(A/t|0)-o&255;return g+I},a.prototype.writeIntBE=function(A,g,I,C){if(A=+A,g|=0,!C){var e=Math.pow(2,8*I-1);H(this,A,g,I,e-1,-e)}var i=I-1,t=1,o=0;for(this[g+i]=255&A;--i>=0&&(t*=256);)A<0&&0===o&&0!==this[g+i+1]&&(o=1),this[g+i]=(A/t|0)-o&255;return g+I},a.prototype.writeInt8=function(A,g,I){return A=+A,g|=0,I||H(this,A,g,1,127,-128),a.TYPED_ARRAY_SUPPORT||(A=Math.floor(A)),A<0&&(A=255+A+1),this[g]=255&A,g+1},a.prototype.writeInt16LE=function(A,g,I){return A=+A,g|=0,I||H(this,A,g,2,32767,-32768),a.TYPED_ARRAY_SUPPORT?(this[g]=255&A,this[g+1]=A>>>8):X(this,A,g,!0),g+2},a.prototype.writeInt16BE=function(A,g,I){return A=+A,g|=0,I||H(this,A,g,2,32767,-32768),a.TYPED_ARRAY_SUPPORT?(this[g]=A>>>8,this[g+1]=255&A):X(this,A,g,!1),g+2},a.prototype.writeInt32LE=function(A,g,I){return A=+A,g|=0,I||H(this,A,g,4,2147483647,-2147483648),a.TYPED_ARRAY_SUPPORT?(this[g]=255&A,this[g+1]=A>>>8,this[g+2]=A>>>16,this[g+3]=A>>>24):Y(this,A,g,!0),g+4},a.prototype.writeInt32BE=function(A,g,I){return A=+A,g|=0,I||H(this,A,g,4,2147483647,-2147483648),A<0&&(A=4294967295+A+1),a.TYPED_ARRAY_SUPPORT?(this[g]=A>>>24,this[g+1]=A>>>16,this[g+2]=A>>>8,this[g+3]=255&A):Y(this,A,g,!1),g+4},a.prototype.writeFloatLE=function(A,g,I){return N(this,A,g,!0,I)},a.prototype.writeFloatBE=function(A,g,I){return N(this,A,g,!1,I)},a.prototype.writeDoubleLE=function(A,g,I){return v(this,A,g,!0,I)},a.prototype.writeDoubleBE=function(A,g,I){return v(this,A,g,!1,I)},a.prototype.copy=function(A,g,I,C){if(I||(I=0),C||0===C||(C=this.length),g>=A.length&&(g=A.length),g||(g=0),C>0&&C<I&&(C=I),C===I)return 0;if(0===A.length||0===this.length)return 0;if(g<0)throw new RangeError("targetStart out of bounds");if(I<0||I>=this.length)throw new RangeError("sourceStart out of bounds");if(C<0)throw new RangeError("sourceEnd out of bounds");C>this.length&&(C=this.length),A.length-g<C-I&&(C=A.length-g+I);var e,i=C-I;if(this===A&&I<g&&g<C)for(e=i-1;e>=0;--e)A[e+g]=this[e+I];else if(i<1e3||!a.TYPED_ARRAY_SUPPORT)for(e=0;e<i;++e)A[e+g]=this[e+I];else Uint8Array.prototype.set.call(A,this.subarray(I,I+i),g);return i},a.prototype.fill=function(A,g,I,C){if("string"==typeof A){if("string"==typeof g?(C=g,g=0,I=this.length):"string"==typeof I&&(C=I,I=this.length),1===A.length){var e=A.charCodeAt(0);e<256&&(A=e)}if(void 0!==C&&"string"!=typeof C)throw new TypeError("encoding must be a string");if("string"==typeof C&&!a.isEncoding(C))throw new TypeError("Unknown encoding: "+C)}else"number"==typeof A&&(A&=255);if(g<0||this.length<g||this.length<I)throw new RangeError("Out of range index");if(I<=g)return this;var i;if(g>>>=0,I=void 0===I?this.length:I>>>0,A||(A=0),"number"==typeof A)for(i=g;i<I;++i)this[i]=A;else{var t=a.isBuffer(A)?A:L(new a(A,C).toString()),o=t.length;for(i=0;i<I-g;++i)this[i+g]=t[i%o]}return this};var J=/[^+\/0-9A-Za-z-_]/g;function f(A){return A<16?"0"+A.toString(16):A.toString(16)}function L(A,g){var I;g=g||1/0;for(var C=A.length,e=null,i=[],t=0;t<C;++t){if((I=A.charCodeAt(t))>55295&&I<57344){if(!e){if(I>56319){(g-=3)>-1&&i.push(239,191,189);continue}if(t+1===C){(g-=3)>-1&&i.push(239,191,189);continue}e=I;continue}if(I<56320){(g-=3)>-1&&i.push(239,191,189),e=I;continue}I=65536+(e-55296<<10|I-56320)}else e&&(g-=3)>-1&&i.push(239,191,189);if(e=null,I<128){if((g-=1)<0)break;i.push(I)}else if(I<2048){if((g-=2)<0)break;i.push(I>>6|192,63&I|128)}else if(I<65536){if((g-=3)<0)break;i.push(I>>12|224,I>>6&63|128,63&I|128)}else{if(!(I<1114112))throw new Error("Invalid code point");if((g-=4)<0)break;i.push(I>>18|240,I>>12&63|128,I>>6&63|128,63&I|128)}}return i}function M(A){return C.toByteArray(function(A){if((A=function(A){return A.trim?A.trim():A.replace(/^\s+|\s+$/g,"")}(A).replace(J,"")).length<2)return"";for(;A.length%4!=0;)A+="=";return A}(A))}function U(A,g,I,C){for(var e=0;e<C&&!(e+I>=g.length||e>=A.length);++e)g[e+I]=A[e];return e}}).call(this,I(8))},function(A,g){var I;I=function(){return this}();try{I=I||Function("return this")()||(0,eval)("this")}catch(A){"object"==typeof window&&(I=window)}A.exports=I},function(A,g,I){g.byteLength=function(A){var g=B(A),I=g[0],C=g[1];return 3*(I+C)/4-C},g.toByteArray=function(A){for(var g,I=B(A),C=I[0],t=I[1],o=new i(function(A,g,I){return 3*(g+I)/4-I}(0,C,t)),a=0,s=t>0?C-4:C,l=0;l<s;l+=4)g=e[A.charCodeAt(l)]<<18|e[A.charCodeAt(l+1)]<<12|e[A.charCodeAt(l+2)]<<6|e[A.charCodeAt(l+3)],o[a++]=g>>16&255,o[a++]=g>>8&255,o[a++]=255&g;return 2===t&&(g=e[A.charCodeAt(l)]<<2|e[A.charCodeAt(l+1)]>>4,o[a++]=255&g),1===t&&(g=e[A.charCodeAt(l)]<<10|e[A.charCodeAt(l+1)]<<4|e[A.charCodeAt(l+2)]>>2,o[a++]=g>>8&255,o[a++]=255&g),o},g.fromByteArray=function(A){for(var g,I=A.length,e=I%3,i=[],t=0,o=I-e;t<o;t+=16383)i.push(l(A,t,t+16383>o?o:t+16383));return 1===e?(g=A[I-1],i.push(C[g>>2]+C[g<<4&63]+"==")):2===e&&(g=(A[I-2]<<8)+A[I-1],i.push(C[g>>10]+C[g>>4&63]+C[g<<2&63]+"=")),i.join("")};for(var C=[],e=[],i="undefined"!=typeof Uint8Array?Uint8Array:Array,t="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/",o=0,a=t.length;o<a;++o)C[o]=t[o],e[t.charCodeAt(o)]=o;function B(A){var g=A.length;if(g%4>0)throw new Error("Invalid string. Length must be a multiple of 4");var I=A.indexOf("=");return-1===I&&(I=g),[I,I===g?0:4-I%4]}function s(A){return C[A>>18&63]+C[A>>12&63]+C[A>>6&63]+C[63&A]}function l(A,g,I){for(var C,e=[],i=g;i<I;i+=3)C=(A[i]<<16&16711680)+(A[i+1]<<8&65280)+(255&A[i+2]),e.push(s(C));return e.join("")}e["-".charCodeAt(0)]=62,e["_".charCodeAt(0)]=63},function(A,g){g.read=function(A,g,I,C,e){var i,t,o=8*e-C-1,a=(1<<o)-1,B=a>>1,s=-7,l=I?e-1:0,n=I?-1:1,d=A[g+l];for(l+=n,i=d&(1<<-s)-1,d>>=-s,s+=o;s>0;i=256*i+A[g+l],l+=n,s-=8);for(t=i&(1<<-s)-1,i>>=-s,s+=C;s>0;t=256*t+A[g+l],l+=n,s-=8);if(0===i)i=1-B;else{if(i===a)return t?NaN:1/0*(d?-1:1);t+=Math.pow(2,C),i-=B}return(d?-1:1)*t*Math.pow(2,i-C)},g.write=function(A,g,I,C,e,i){var t,o,a,B=8*i-e-1,s=(1<<B)-1,l=s>>1,n=23===e?Math.pow(2,-24)-Math.pow(2,-77):0,d=C?0:i-1,r=C?1:-1,c=g<0||0===g&&1/g<0?1:0;for(g=Math.abs(g),isNaN(g)||g===1/0?(o=isNaN(g)?1:0,t=s):(t=Math.floor(Math.log(g)/Math.LN2),g*(a=Math.pow(2,-t))<1&&(t--,a*=2),(g+=t+l>=1?n/a:n*Math.pow(2,1-l))*a>=2&&(t++,a/=2),t+l>=s?(o=0,t=s):t+l>=1?(o=(g*a-1)*Math.pow(2,e),t+=l):(o=g*Math.pow(2,l-1)*Math.pow(2,e),t=0));e>=8;A[I+d]=255&o,d+=r,o/=256,e-=8);for(t=t<<e|o,B+=e;B>0;A[I+d]=255&t,d+=r,t/=256,B-=8);A[I+d-r]|=128*c}},function(A,g){var I={}.toString;A.exports=Array.isArray||function(A){return"[object Array]"==I.call(A)}},function(g,I){g.exports=A},function(A,g,I){(function(A){function I(A,g){for(var I=0,C=A.length-1;C>=0;C--){var e=A[C];"."===e?A.splice(C,1):".."===e?(A.splice(C,1),I++):I&&(A.splice(C,1),I--)}if(g)for(;I--;I)A.unshift("..");return A}var C=/^(\/?|)([\s\S]*?)((?:\.{1,2}|[^\/]+?|)(\.[^.\/]*|))(?:[\/]*)$/,e=function(A){return C.exec(A).slice(1)};function i(A,g){if(A.filter)return A.filter(g);for(var I=[],C=0;C<A.length;C++)g(A[C],C,A)&&I.push(A[C]);return I}g.resolve=function(){for(var g="",C=!1,e=arguments.length-1;e>=-1&&!C;e--){var t=e>=0?arguments[e]:A.cwd();if("string"!=typeof t)throw new TypeError("Arguments to path.resolve must be strings");t&&(g=t+"/"+g,C="/"===t.charAt(0))}return g=I(i(g.split("/"),(function(A){return!!A})),!C).join("/"),(C?"/":"")+g||"."},g.normalize=function(A){var C=g.isAbsolute(A),e="/"===t(A,-1);return(A=I(i(A.split("/"),(function(A){return!!A})),!C).join("/"))||C||(A="."),A&&e&&(A+="/"),(C?"/":"")+A},g.isAbsolute=function(A){return"/"===A.charAt(0)},g.join=function(){var A=Array.prototype.slice.call(arguments,0);return g.normalize(i(A,(function(A,g){if("string"!=typeof A)throw new TypeError("Arguments to path.join must be strings");return A})).join("/"))},g.relative=function(A,I){function C(A){for(var g=0;g<A.length&&""===A[g];g++);for(var I=A.length-1;I>=0&&""===A[I];I--);return g>I?[]:A.slice(g,I-g+1)}A=g.resolve(A).substr(1),I=g.resolve(I).substr(1);for(var e=C(A.split("/")),i=C(I.split("/")),t=Math.min(e.length,i.length),o=t,a=0;a<t;a++)if(e[a]!==i[a]){o=a;break}var B=[];for(a=o;a<e.length;a++)B.push("..");return(B=B.concat(i.slice(o))).join("/")},g.sep="/",g.delimiter=":",g.dirname=function(A){var g=e(A),I=g[0],C=g[1];return I||C?(C&&(C=C.substr(0,C.length-1)),I+C):"."},g.basename=function(A,g){var I=e(A)[2];return g&&I.substr(-1*g.length)===g&&(I=I.substr(0,I.length-g.length)),I},g.extname=function(A){return e(A)[3]};var t="b"==="ab".substr(-1)?function(A,g,I){return A.substr(g,I)}:function(A,g,I){return g<0&&(g=A.length+g),A.substr(g,I)}}).call(this,I(14))},function(A,g){var I,C,e=A.exports={};function i(){throw new Error("setTimeout has not been defined")}function t(){throw new Error("clearTimeout has not been defined")}function o(A){if(I===setTimeout)return setTimeout(A,0);if((I===i||!I)&&setTimeout)return I=setTimeout,setTimeout(A,0);try{return I(A,0)}catch(g){try{return I.call(null,A,0)}catch(g){return I.call(this,A,0)}}}!function(){try{I="function"==typeof setTimeout?setTimeout:i}catch(A){I=i}try{C="function"==typeof clearTimeout?clearTimeout:t}catch(A){C=t}}();var a,B=[],s=!1,l=-1;function n(){s&&a&&(s=!1,a.length?B=a.concat(B):l=-1,B.length&&d())}function d(){if(!s){var A=o(n);s=!0;for(var g=B.length;g;){for(a=B,B=[];++l<g;)a&&a[l].run();l=-1,g=B.length}a=null,s=!1,function(A){if(C===clearTimeout)return clearTimeout(A);if((C===t||!C)&&clearTimeout)return C=clearTimeout,clearTimeout(A);try{C(A)}catch(g){try{return C.call(null,A)}catch(g){return C.call(this,A)}}}(A)}}function r(A,g){this.fun=A,this.array=g}function c(){}e.nextTick=function(A){var g=new Array(arguments.length-1);if(arguments.length>1)for(var I=1;I<arguments.length;I++)g[I-1]=arguments[I];B.push(new r(A,g)),1!==B.length||s||o(d)},r.prototype.run=function(){this.fun.apply(null,this.array)},e.title="browser",e.browser=!0,e.env={},e.argv=[],e.version="",e.versions={},e.on=c,e.addListener=c,e.once=c,e.off=c,e.removeListener=c,e.removeAllListeners=c,e.emit=c,e.prependListener=c,e.prependOnceListener=c,e.listeners=function(A){return[]},e.binding=function(A){throw new Error("process.binding is not supported")},e.cwd=function(){return"/"},e.chdir=function(A){throw new Error("process.chdir is not supported")},e.umask=function(){return 0}}])}(A.fs)}(WorkerScope);const unpackBridge=WorkerScope.unpackBridge;let unpack;var initunpack=function(A){const g=Module,I="omit";var C,e=unpackBridge.Ext,i={open:function(){return e.current.open.apply(e.current,arguments)},close:function(){return e.current.close.apply(e.current,arguments)},read:function(){return e.current.read.apply(e.current,arguments)},write:function(){return e.current.write.apply(e.current,arguments)},tell:function(){return e.current.tell.apply(e.current,arguments)},seek:function(){return e.current.seek.apply(e.current,arguments)},create:function(){return e.current.create.apply(e.current,arguments)}},t={};for(C in Module)Module.hasOwnProperty(C)&&(t[C]=Module[C]);Module.wasmBinary=A,Module.arguments=[],Module.thisProgram="./this.program",Module.quit=function(A,g){throw g},Module.preRun=[],Module.postRun=[];var o=!1,a=!1,B=!1;if(Module.ENVIRONMENT){if("WEB"===Module.ENVIRONMENT)o=!0;else if("WORKER"===Module.ENVIRONMENT)a=!0;else if("NODE"===Module.ENVIRONMENT)B=!0;else if("SHELL"!==Module.ENVIRONMENT)throw new Error("Module['ENVIRONMENT'] value is not valid. must be one of: WEB|WORKER|NODE|SHELL.")}else o="object"==typeof window,a="function"==typeof importScripts,B="object"==typeof process&&"function"==typeof require&&!o&&!a;for(C in(o||a)&&(Module.read=function(A){var g=new XMLHttpRequest;return g.open("GET",A,!1),g.send(null),g.responseText},a&&(Module.readBinary=function(A){var g=new XMLHttpRequest;return g.open("GET",A,!1),g.responseType="arraybuffer",g.send(null),new Uint8Array(g.response)}),Module.readAsync=function(A,g,I){var C=new XMLHttpRequest;C.open("GET",A,!0),C.responseType="arraybuffer",C.onload=function(){200==C.status||0==C.status&&C.response?g(C.response):I()},C.onerror=I,C.send(null)},Module.setWindowTitle=function(A){document.title=A}),Module.print="undefined"!=typeof console?console.log.bind(console):"undefined"!=typeof print?print:null,Module.printErr="undefined"!=typeof printErr?printErr:"undefined"!=typeof console&&console.warn.bind(console)||Module.print,Module.print=Module.print,Module.printErr=Module.printErr,t)t.hasOwnProperty(C)&&(Module[C]=t[C]);t=void 0;function s(A){c(!N);var g=K;return K=K+A+15&-16,g}function l(A){c(L);var g=w[L>>2],I=g+A+15&-16;if((w[L>>2]=I,I>=O)&&!z())return w[L>>2]=g,0;return g}function n(A,g){return g||(g=16),A=Math.ceil(A/g)*g}function d(A){switch(A){case"i1":case"i8":return 1;case"i16":return 2;case"i32":case"float":return 4;case"i64":case"double":return 8;default:if("*"===A[A.length-1])return 4;if("i"===A[0]){var g=parseInt(A.substr(1));return c(g%8==0),g/8}return 0}}new Array(0);var r=0;function c(A,g){A||tI("Assertion failed: "+g)}function h(A,g,I,C){switch("*"===(I=I||"i8").charAt(I.length-1)&&(I="i32"),I){case"i1":case"i8":W[A|0]=g;break;case"i16":S[A>>1]=g;break;case"i32":w[A>>2]=g;break;case"i64":tempI64=[g>>>0,(tempDouble=g,+eA(tempDouble)>=1?tempDouble>0?(0|oA(+tA(tempDouble/4294967296),4294967295))>>>0:~~+iA((tempDouble-+(~~tempDouble>>>0))/4294967296)>>>0:0)],w[A>>2]=tempI64[0],w[A+4>>2]=tempI64[1];break;case"float":X[A>>2]=g;break;case"double":Y[A>>3]=g;break;default:tI("invalid type for setValue: "+I)}}function G(A,g,I,C){var e,i;"number"==typeof A?(e=!0,i=A):(e=!1,i=A.length);var t,o="string"==typeof g?g:null;if(t=4==I?C:["function"==typeof gI?gI:s,CI,s,l][void 0===I?2:I](Math.max(i,o?1:g.length)),e){var a;for(C=t,c(!(3&t)),a=t+(-4&i);C<a;C+=4)w[C>>2]=0;for(a=t+i;C<a;)W[C++|0]=0;return t}if("i8"===o)return A.subarray||A.slice?p.set(A,t):p.set(new Uint8Array(A),t),t;for(var B,n,r,G=0;G<i;){var m=A[G];0!==(B=o||g[G])?("i64"==B&&(B="i32"),h(t+G,m,B),r!==B&&(n=d(B),r=B),G+=n):G++}return t}function m(A,g){if(0===g||!A)return"";for(var I,C=0,e=0;C|=I=p[A+e|0],(0!=I||g)&&(e++,!g||e!=g););g||(g=e);var i="";if(C<128){for(var t,o=1024;g>0;)t=String.fromCharCode.apply(String,p.subarray(A,A+Math.min(g,o))),i=i?i+t:t,A+=o,g-=o;return i}return u(A)}var b="undefined"!=typeof TextDecoder?new TextDecoder("utf8"):void 0;function Z(A,g){for(var I=g;A[I];)++I;if(I-g>16&&A.subarray&&b)return b.decode(A.subarray(g,I));for(var C,e,i,t,o,a="";;){if(!(C=A[g++]))return a;if(128&C)if(e=63&A[g++],192!=(224&C))if(i=63&A[g++],224==(240&C)?C=(15&C)<<12|e<<6|i:(t=63&A[g++],240==(248&C)?C=(7&C)<<18|e<<12|i<<6|t:(o=63&A[g++],C=248==(252&C)?(3&C)<<24|e<<18|i<<12|t<<6|o:(1&C)<<30|e<<24|i<<18|t<<12|o<<6|63&A[g++])),C<65536)a+=String.fromCharCode(C);else{var B=C-65536;a+=String.fromCharCode(55296|B>>10,56320|1023&B)}else a+=String.fromCharCode((31&C)<<6|e);else a+=String.fromCharCode(C)}}function u(A){return Z(p,A)}function y(A,g,I,C){if(!(C>0))return 0;for(var e=I,i=I+C-1,t=0;t<A.length;++t){var o=A.charCodeAt(t);if(o>=55296&&o<=57343&&(o=65536+((1023&o)<<10)|1023&A.charCodeAt(++t)),o<=127){if(I>=i)break;g[I++]=o}else if(o<=2047){if(I+1>=i)break;g[I++]=192|o>>6,g[I++]=128|63&o}else if(o<=65535){if(I+2>=i)break;g[I++]=224|o>>12,g[I++]=128|o>>6&63,g[I++]=128|63&o}else if(o<=2097151){if(I+3>=i)break;g[I++]=240|o>>18,g[I++]=128|o>>12&63,g[I++]=128|o>>6&63,g[I++]=128|63&o}else if(o<=67108863){if(I+4>=i)break;g[I++]=248|o>>24,g[I++]=128|o>>18&63,g[I++]=128|o>>12&63,g[I++]=128|o>>6&63,g[I++]=128|63&o}else{if(I+5>=i)break;g[I++]=252|o>>30,g[I++]=128|o>>24&63,g[I++]=128|o>>18&63,g[I++]=128|o>>12&63,g[I++]=128|o>>6&63,g[I++]=128|63&o}}return g[I]=0,I-e}function E(A,g,I){return y(A,p,g,I)}function Q(A){for(var g=0,I=0;I<A.length;++I){var C=A.charCodeAt(I);C>=55296&&C<=57343&&(C=65536+((1023&C)<<10)|1023&A.charCodeAt(++I)),C<=127?++g:g+=C<=2047?2:C<=65535?3:C<=2097151?4:C<=67108863?5:6}return g}function V(A){for(var g=0,I="";;){var C=w[A+4*g>>2];if(0==C)return I;if(++g,C>=65536){var e=C-65536;I+=String.fromCharCode(55296|e>>10,56320|1023&e)}else I+=String.fromCharCode(C)}}function R(){var A=function(){var A=new Error;if(!A.stack){try{throw new Error(0)}catch(g){A=g}if(!A.stack)return"(no stack trace available)"}return A.stack.toString()}();return Module.extraStackTrace&&(A+="\n"+Module.extraStackTrace()),function(A){return A.replace(/__Z[\w\d_]+/g,(function(A){return A==A?A:A+" ["+A+"]"}))}(A)}"undefined"!=typeof TextDecoder&&new TextDecoder("utf-16le");var W,p,S,F,w,H,X,Y,K,N,v,J,f,L,M=65536,U=16777216,k=16777216;function D(A,g){return A%g>0&&(A+=g-A%g),A}function x(g){Module.buffer=A=g}function T(){Module.HEAP8=W=new Int8Array(A),Module.HEAP16=S=new Int16Array(A),Module.HEAP32=w=new Int32Array(A),Module.HEAPU8=p=new Uint8Array(A),Module.HEAPU16=F=new Uint16Array(A),Module.HEAPU32=H=new Uint32Array(A),Module.HEAPF32=X=new Float32Array(A),Module.HEAPF64=Y=new Float64Array(A)}function z(){var A=Module.usingWasm?M:U,g=2147483648-A;if(w[L>>2]>g)return!1;var I=O;for(O=Math.max(O,k);O<w[L>>2];)O=O<=536870912?D(2*O,A):Math.min(D((3*O+2147483648)/4,A),g);var C=Module.reallocBuffer(O);return C&&C.byteLength==O?(x(C),T(),!0):(O=I,!1)}K=J=L=0,N=!1,Module.reallocBuffer||(Module.reallocBuffer=function(g){var I;try{if(ArrayBuffer.transfer)I=ArrayBuffer.transfer(A,g);else{var C=W;I=new ArrayBuffer(g),new Int8Array(I).set(C)}}catch(A){return!1}return!!$g(I)&&I});try{Function.prototype.call.bind(Object.getOwnPropertyDescriptor(ArrayBuffer.prototype,"byteLength").get)(new ArrayBuffer(4))}catch(A){0}var P=Module.TOTAL_STACK||5242880,O=Module.TOTAL_MEMORY||16777216;if(O<P&&Module.printErr("TOTAL_MEMORY should be larger than TOTAL_STACK, was "+O+"! (TOTAL_STACK="+P+")"),Module.buffer?A=Module.buffer:("object"==typeof WebAssembly&&"function"==typeof WebAssembly.Memory?(Module.wasmMemory=new WebAssembly.Memory({initial:O/M}),A=Module.wasmMemory.buffer):A=new ArrayBuffer(O),Module.buffer=A),T(),w[0]=1668509029,S[1]=25459,115!==p[2]||99!==p[3])throw"Runtime error: expected the system to be little-endian!";function j(A){for(;A.length>0;){var g=A.shift();if("function"!=typeof g){var I=g.func;"number"==typeof I?void 0===g.arg?Module.dynCall_v(I):Module.dynCall_vi(I,g.arg):I(void 0===g.arg?null:g.arg)}else g()}}var q=[],_=[],$=[],AA=[],gA=[],IA=!1;function CA(A,g,I){for(var C=0;C<A.length;++C)W[g++|0]=A.charCodeAt(C);I||(W[g|0]=0)}var eA=Math.abs,iA=Math.ceil,tA=Math.floor,oA=Math.min,aA=0,BA=null;function sA(A){aA++,Module.monitorRunDependencies&&Module.monitorRunDependencies(aA)}function lA(A){if(aA--,Module.monitorRunDependencies&&Module.monitorRunDependencies(aA),0==aA&&BA){var g=BA;BA=null,g()}}Module.preloadedImages={},Module.preloadedAudios={};var nA="data:application/octet-stream;base64,";function dA(A){return String.prototype.startsWith?A.startsWith(nA):0===A.indexOf(nA)}!function(){var A="unpack.wast",g="unpack.temp.asm.js";"function"==typeof Module.locateFile&&(dA(A)||(A=Module.locateFile(A)),dA(wasmBinaryFile)||(wasmBinaryFile=Module.locateFile(wasmBinaryFile)),dA(g)||(g=Module.locateFile(g)));const C={asm2wasm:{"f64-rem":function(A,g){return A%g},debugger:function(){}},parent:Module};var e=null;function i(){try{if(Module.wasmBinary)return new Uint8Array(Module.wasmBinary);if(Module.readBinary)return Module.readBinary(wasmBinaryFile);throw"on the web, we need the wasm binary to be preloaded and set on Module['wasmBinary']. emcc.py will do that for you when generating HTML (but not JS)"}catch(A){tI(A)}}function t(A,g,t){if("object"!=typeof WebAssembly)return Module.printErr("no native wasm support detected"),!1;if(!(Module.wasmMemory instanceof WebAssembly.Memory))return Module.printErr("no native wasm Memory in use"),!1;function B(A,g){(e=A.exports).memory&&function(A){var g=Module.buffer;A.byteLength<g.byteLength&&Module.printErr("the new buffer in mergeMemory is smaller than the previous one. in native wasm, we should grow memory here");var I=new Int8Array(g);new Int8Array(A).set(I),x(A),T()}(e.memory),Module.asm=e,Module.usingWasm=!0,lA()}if(g.memory=Module.wasmMemory,C.global={NaN:NaN,Infinity:1/0},C["global.Math"]=Math,C.env=g,sA(),Module.instantiateWasm)try{return Module.instantiateWasm(C,B)}catch(A){return Module.printErr("Module.instantiateWasm callback failed with error: "+A),!1}function s(A){B(A.instance,A.module)}function l(A){(Module.wasmBinary||!o&&!a||"function"!=typeof fetch?new Promise((function(A,g){A(i())})):fetch(wasmBinaryFile,{credentials:I}).then((function(A){if(!A.ok)throw"failed to load wasm binary file at '"+wasmBinaryFile+"'";return A.arrayBuffer()})).catch((function(){return i()}))).then((function(A){return WebAssembly.instantiate(A,C)})).then(A).catch((function(A){Module.printErr("failed to asynchronously prepare wasm: "+A),tI(A)}))}return Module.wasmBinary||"function"!=typeof WebAssembly.instantiateStreaming||dA(wasmBinaryFile)||"function"!=typeof fetch?l(s):WebAssembly.instantiateStreaming(fetch(wasmBinaryFile,{credentials:I}),C).then(s).catch((function(A){Module.printErr("wasm streaming compile failed: "+A),Module.printErr("falling back to ArrayBuffer instantiation"),l(s)})),{}}Module.asmPreload=Module.asm;var B=Module.reallocBuffer;Module.reallocBuffer=function(A){return"asmjs"===s?B(A):function(A){A=D(A,Module.usingWasm?M:U);var g=Module.buffer.byteLength;if(Module.usingWasm)try{return-1!==Module.wasmMemory.grow((A-g)/65536)?Module.buffer=Module.wasmMemory.buffer:null}catch(A){return null}}(A)};var s="";Module.asm=function(A,g,I){if(!g.table){var C=Module.wasmTableSize;void 0===C&&(C=1024);var e=Module.wasmMaxTableSize;"object"==typeof WebAssembly&&"function"==typeof WebAssembly.Table?g.table=void 0!==e?new WebAssembly.Table({initial:C,maximum:e,element:"anyfunc"}):new WebAssembly.Table({initial:C,element:"anyfunc"}):g.table=new Array(C),Module.wasmTable=g.table}var i;return g.memoryBase||(g.memoryBase=Module.STATIC_BASE),g.tableBase||(g.tableBase=0),(i=t(0,g))||tI("no binaryen method succeeded. consider enabling more options, like interpreting, if you want that: https://github.com/kripken/emscripten/wiki/WebAssembly#binaryen-methods"),i}}(),K=67984,_.push({func:function(){qg()}},{func:function(){jg()}},{func:function(){Og()}},{func:function(){Pg()}});function rA(){return!!rA.uncaught_exception}Module.STATIC_BASE=1024,Module.STATIC_BUMP=66960,K+=16;var cA={last:0,caught:[],infos:{},deAdjust:function(A){if(!A||cA.infos[A])return A;for(var g in cA.infos){var I=+g;if(cA.infos[I].adjusted===A)return I}return A},addRef:function(A){A&&cA.infos[A].refcount++},decRef:function(A){if(A){var g=cA.infos[A];c(g.refcount>0),g.refcount--,0!==g.refcount||g.rethrown||(g.destructor&&Module.dynCall_vi(g.destructor,A),delete cA.infos[A],hA(A))}},clearRef:function(A){A&&(cA.infos[A].refcount=0)}};function hA(A){try{return AI(A)}catch(A){}}function GA(){var A=cA.last;if(!A)return 0|(II(0),0);var g=cA.infos[A],I=g.type;if(!I)return 0|(II(0),A);var C=Array.prototype.slice.call(arguments);Module.___cxa_is_pointer_type(I),GA.buffer||(GA.buffer=gI(4)),w[GA.buffer>>2]=A,A=GA.buffer;for(var e=0;e<C.length;e++)if(C[e]&&Module.___cxa_can_catch(C[e],I,A))return A=w[A>>2],g.adjusted=A,0|(II(C[e]),A);return A=w[A>>2],0|(II(I),A)}var mA={EPERM:1,ENOENT:2,ESRCH:3,EINTR:4,EIO:5,ENXIO:6,E2BIG:7,ENOEXEC:8,EBADF:9,ECHILD:10,EAGAIN:11,EWOULDBLOCK:11,ENOMEM:12,EACCES:13,EFAULT:14,ENOTBLK:15,EBUSY:16,EEXIST:17,EXDEV:18,ENODEV:19,ENOTDIR:20,EISDIR:21,EINVAL:22,ENFILE:23,EMFILE:24,ENOTTY:25,ETXTBSY:26,EFBIG:27,ENOSPC:28,ESPIPE:29,EROFS:30,EMLINK:31,EPIPE:32,EDOM:33,ERANGE:34,ENOMSG:42,EIDRM:43,ECHRNG:44,EL2NSYNC:45,EL3HLT:46,EL3RST:47,ELNRNG:48,EUNATCH:49,ENOCSI:50,EL2HLT:51,EDEADLK:35,ENOLCK:37,EBADE:52,EBADR:53,EXFULL:54,ENOANO:55,EBADRQC:56,EBADSLT:57,EDEADLOCK:35,EBFONT:59,ENOSTR:60,ENODATA:61,ETIME:62,ENOSR:63,ENONET:64,ENOPKG:65,EREMOTE:66,ENOLINK:67,EADV:68,ESRMNT:69,ECOMM:70,EPROTO:71,EMULTIHOP:72,EDOTDOT:73,EBADMSG:74,ENOTUNIQ:76,EBADFD:77,EREMCHG:78,ELIBACC:79,ELIBBAD:80,ELIBSCN:81,ELIBMAX:82,ELIBEXEC:83,ENOSYS:38,ENOTEMPTY:39,ENAMETOOLONG:36,ELOOP:40,EOPNOTSUPP:95,EPFNOSUPPORT:96,ECONNRESET:104,ENOBUFS:105,EAFNOSUPPORT:97,EPROTOTYPE:91,ENOTSOCK:88,ENOPROTOOPT:92,ESHUTDOWN:108,ECONNREFUSED:111,EADDRINUSE:98,ECONNABORTED:103,ENETUNREACH:101,ENETDOWN:100,ETIMEDOUT:110,EHOSTDOWN:112,EHOSTUNREACH:113,EINPROGRESS:115,EALREADY:114,EDESTADDRREQ:89,EMSGSIZE:90,EPROTONOSUPPORT:93,ESOCKTNOSUPPORT:94,EADDRNOTAVAIL:99,ENETRESET:102,EISCONN:106,ENOTCONN:107,ETOOMANYREFS:109,EUSERS:87,EDQUOT:122,ESTALE:116,ENOTSUP:95,ENOMEDIUM:123,EILSEQ:84,EOVERFLOW:75,ECANCELED:125,ENOTRECOVERABLE:131,EOWNERDEAD:130,ESTRPIPE:86};function bA(A){return Module.___errno_location&&(w[Module.___errno_location()>>2]=A),A}var ZA={0:"Success",1:"Not super-user",2:"No such file or directory",3:"No such process",4:"Interrupted system call",5:"I/O error",6:"No such device or address",7:"Arg list too long",8:"Exec format error",9:"Bad file number",10:"No children",11:"No more processes",12:"Not enough core",13:"Permission denied",14:"Bad address",15:"Block device required",16:"Mount device busy",17:"File exists",18:"Cross-device link",19:"No such device",20:"Not a directory",21:"Is a directory",22:"Invalid argument",23:"Too many open files in system",24:"Too many open files",25:"Not a typewriter",26:"Text file busy",27:"File too large",28:"No space left on device",29:"Illegal seek",30:"Read only file system",31:"Too many links",32:"Broken pipe",33:"Math arg out of domain of func",34:"Math result not representable",35:"File locking deadlock error",36:"File or path name too long",37:"No record locks available",38:"Function not implemented",39:"Directory not empty",40:"Too many symbolic links",42:"No message of desired type",43:"Identifier removed",44:"Channel number out of range",45:"Level 2 not synchronized",46:"Level 3 halted",47:"Level 3 reset",48:"Link number out of range",49:"Protocol driver not attached",50:"No CSI structure available",51:"Level 2 halted",52:"Invalid exchange",53:"Invalid request descriptor",54:"Exchange full",55:"No anode",56:"Invalid request code",57:"Invalid slot",59:"Bad font file fmt",60:"Device not a stream",61:"No data (for no delay io)",62:"Timer expired",63:"Out of streams resources",64:"Machine is not on the network",65:"Package not installed",66:"The object is remote",67:"The link has been severed",68:"Advertise error",69:"Srmount error",70:"Communication error on send",71:"Protocol error",72:"Multihop attempted",73:"Cross mount point (not really error)",74:"Trying to read unreadable message",75:"Value too large for defined data type",76:"Given log. name not unique",77:"f.d. invalid for this operation",78:"Remote address changed",79:"Can   access a needed shared lib",80:"Accessing a corrupted shared lib",81:".lib section in a.out corrupted",82:"Attempting to link in too many libs",83:"Attempting to exec a shared library",84:"Illegal byte sequence",86:"Streams pipe error",87:"Too many users",88:"Socket operation on non-socket",89:"Destination address required",90:"Message too long",91:"Protocol wrong type for socket",92:"Protocol not available",93:"Unknown protocol",94:"Socket type not supported",95:"Not supported",96:"Protocol family not supported",97:"Address family not supported by protocol family",98:"Address already in use",99:"Address not available",100:"Network interface is not configured",101:"Network is unreachable",102:"Connection reset by network",103:"Connection aborted",104:"Connection reset by peer",105:"No buffer space available",106:"Socket is already connected",107:"Socket is not connected",108:"Can't send after socket shutdown",109:"Too many references",110:"Connection timed out",111:"Connection refused",112:"Host is down",113:"Host is unreachable",114:"Socket already connected",115:"Connection already in progress",116:"Stale file handle",122:"Quota exceeded",123:"No medium (in tape drive)",125:"Operation canceled",130:"Previous owner died",131:"State not recoverable"},uA={splitPath:function(A){return/^(\/?|)([\s\S]*?)((?:\.{1,2}|[^\/]+?|)(\.[^.\/]*|))(?:[\/]*)$/.exec(A).slice(1)},normalizeArray:function(A,g){for(var I=0,C=A.length-1;C>=0;C--){var e=A[C];"."===e?A.splice(C,1):".."===e?(A.splice(C,1),I++):I&&(A.splice(C,1),I--)}if(g)for(;I;I--)A.unshift("..");return A},normalize:function(A){var g="/"===A.charAt(0),I="/"===A.substr(-1);return(A=uA.normalizeArray(A.split("/").filter((function(A){return!!A})),!g).join("/"))||g||(A="."),A&&I&&(A+="/"),(g?"/":"")+A},dirname:function(A){var g=uA.splitPath(A),I=g[0],C=g[1];return I||C?(C&&(C=C.substr(0,C.length-1)),I+C):"."},basename:function(A){if("/"===A)return"/";var g=A.lastIndexOf("/");return-1===g?A:A.substr(g+1)},extname:function(A){return uA.splitPath(A)[3]},join:function(){var A=Array.prototype.slice.call(arguments,0);return uA.normalize(A.join("/"))},join2:function(A,g){return uA.normalize(A+"/"+g)},resolve:function(){for(var A="",g=!1,I=arguments.length-1;I>=-1&&!g;I--){var C=I>=0?arguments[I]:WA.cwd();if("string"!=typeof C)throw new TypeError("Arguments to path.resolve must be strings");if(!C)return"";A=C+"/"+A,g="/"===C.charAt(0)}return(g?"/":"")+(A=uA.normalizeArray(A.split("/").filter((function(A){return!!A})),!g).join("/"))||"."},relative:function(A,g){function I(A){for(var g=0;g<A.length&&""===A[g];g++);for(var I=A.length-1;I>=0&&""===A[I];I--);return g>I?[]:A.slice(g,I-g+1)}A=uA.resolve(A).substr(1),g=uA.resolve(g).substr(1);for(var C=I(A.split("/")),e=I(g.split("/")),i=Math.min(C.length,e.length),t=i,o=0;o<i;o++)if(C[o]!==e[o]){t=o;break}var a=[];for(o=t;o<C.length;o++)a.push("..");return(a=a.concat(e.slice(t))).join("/")}},yA={ttys:[],init:function(){},shutdown:function(){},register:function(A,g){yA.ttys[A]={input:[],output:[],ops:g},WA.registerDevice(A,yA.stream_ops)},stream_ops:{open:function(A){var g=yA.ttys[A.node.rdev];if(!g)throw new WA.ErrnoError(mA.ENODEV);A.tty=g,A.seekable=!1},close:function(A){A.tty.ops.flush(A.tty)},flush:function(A){A.tty.ops.flush(A.tty)},read:function(A,g,I,C,e){if(!A.tty||!A.tty.ops.get_char)throw new WA.ErrnoError(mA.ENXIO);for(var i=0,t=0;t<C;t++){var o;try{o=A.tty.ops.get_char(A.tty)}catch(A){throw new WA.ErrnoError(mA.EIO)}if(void 0===o&&0===i)throw new WA.ErrnoError(mA.EAGAIN);if(null==o)break;i++,g[I+t]=o}return i&&(A.node.timestamp=Date.now()),i},write:function(A,g,I,C,e){if(!A.tty||!A.tty.ops.put_char)throw new WA.ErrnoError(mA.ENXIO);for(var i=0;i<C;i++)try{A.tty.ops.put_char(A.tty,g[I+i])}catch(A){throw new WA.ErrnoError(mA.EIO)}return C&&(A.node.timestamp=Date.now()),i}},default_tty_ops:{get_char:function(A){if(!A.input.length){var g=null;if(B){var I=new Buffer(256),C=0,e="win32"!=process.platform,i=process.stdin.fd;if(e){var t=!1;try{i=fs.openSync("/dev/stdin","r"),t=!0}catch(A){}}try{C=fs.readSync(i,I,0,256,null)}catch(A){if(-1==A.toString().indexOf("EOF"))throw A;C=0}t&&fs.closeSync(i),g=C>0?I.slice(0,C).toString("utf-8"):null}else"undefined"!=typeof window&&"function"==typeof window.prompt?null!==(g=window.prompt("Input: "))&&(g+="\n"):"function"==typeof readline&&null!==(g=readline())&&(g+="\n");if(!g)return null;A.input=Tg(g,!0)}return A.input.shift()},put_char:function(A,g){null===g||10===g?(Module.print(Z(A.output,0)),A.output=[]):0!=g&&A.output.push(g)},flush:function(A){A.output&&A.output.length>0&&(Module.print(Z(A.output,0)),A.output=[])}},default_tty1_ops:{put_char:function(A,g){null===g||10===g?(Module.printErr(Z(A.output,0)),A.output=[]):0!=g&&A.output.push(g)},flush:function(A){A.output&&A.output.length>0&&(Module.printErr(Z(A.output,0)),A.output=[])}}},EA={ops_table:null,mount:function(A){return EA.createNode(null,"/",16895,0)},createNode:function(A,g,I,C){if(WA.isBlkdev(I)||WA.isFIFO(I))throw new WA.ErrnoError(mA.EPERM);EA.ops_table||(EA.ops_table={dir:{node:{getattr:EA.node_ops.getattr,setattr:EA.node_ops.setattr,lookup:EA.node_ops.lookup,mknod:EA.node_ops.mknod,rename:EA.node_ops.rename,unlink:EA.node_ops.unlink,rmdir:EA.node_ops.rmdir,readdir:EA.node_ops.readdir,symlink:EA.node_ops.symlink},stream:{llseek:EA.stream_ops.llseek}},file:{node:{getattr:EA.node_ops.getattr,setattr:EA.node_ops.setattr},stream:{llseek:EA.stream_ops.llseek,read:EA.stream_ops.read,write:EA.stream_ops.write,allocate:EA.stream_ops.allocate,mmap:EA.stream_ops.mmap,msync:EA.stream_ops.msync}},link:{node:{getattr:EA.node_ops.getattr,setattr:EA.node_ops.setattr,readlink:EA.node_ops.readlink},stream:{}},chrdev:{node:{getattr:EA.node_ops.getattr,setattr:EA.node_ops.setattr},stream:WA.chrdev_stream_ops}});var e=WA.createNode(A,g,I,C);return WA.isDir(e.mode)?(e.node_ops=EA.ops_table.dir.node,e.stream_ops=EA.ops_table.dir.stream,e.contents={}):WA.isFile(e.mode)?(e.node_ops=EA.ops_table.file.node,e.stream_ops=EA.ops_table.file.stream,e.usedBytes=0,e.contents=null):WA.isLink(e.mode)?(e.node_ops=EA.ops_table.link.node,e.stream_ops=EA.ops_table.link.stream):WA.isChrdev(e.mode)&&(e.node_ops=EA.ops_table.chrdev.node,e.stream_ops=EA.ops_table.chrdev.stream),e.timestamp=Date.now(),A&&(A.contents[g]=e),e},getFileDataAsRegularArray:function(A){if(A.contents&&A.contents.subarray){for(var g=[],I=0;I<A.usedBytes;++I)g.push(A.contents[I]);return g}return A.contents},getFileDataAsTypedArray:function(A){return A.contents?A.contents.subarray?A.contents.subarray(0,A.usedBytes):new Uint8Array(A.contents):new Uint8Array},expandFileStorage:function(A,g){if(A.contents&&A.contents.subarray&&g>A.contents.length&&(A.contents=EA.getFileDataAsRegularArray(A),A.usedBytes=A.contents.length),!A.contents||A.contents.subarray){var I=A.contents?A.contents.length:0;if(I>=g)return;g=Math.max(g,I*(I<1048576?2:1.125)|0),0!=I&&(g=Math.max(g,256));var C=A.contents;return A.contents=new Uint8Array(g),void(A.usedBytes>0&&A.contents.set(C.subarray(0,A.usedBytes),0))}for(!A.contents&&g>0&&(A.contents=[]);A.contents.length<g;)A.contents.push(0)},resizeFileStorage:function(A,g){if(A.usedBytes!=g){if(0==g)return A.contents=null,void(A.usedBytes=0);if(!A.contents||A.contents.subarray){var I=A.contents;return A.contents=new Uint8Array(new ArrayBuffer(g)),I&&A.contents.set(I.subarray(0,Math.min(g,A.usedBytes))),void(A.usedBytes=g)}if(A.contents||(A.contents=[]),A.contents.length>g)A.contents.length=g;else for(;A.contents.length<g;)A.contents.push(0);A.usedBytes=g}},node_ops:{getattr:function(A){var g={};return g.dev=WA.isChrdev(A.mode)?A.id:1,g.ino=A.id,g.mode=A.mode,g.nlink=1,g.uid=0,g.gid=0,g.rdev=A.rdev,WA.isDir(A.mode)?g.size=4096:WA.isFile(A.mode)?g.size=A.usedBytes:WA.isLink(A.mode)?g.size=A.link.length:g.size=0,g.atime=new Date(A.timestamp),g.mtime=new Date(A.timestamp),g.ctime=new Date(A.timestamp),g.blksize=4096,g.blocks=Math.ceil(g.size/g.blksize),g},setattr:function(A,g){void 0!==g.mode&&(A.mode=g.mode),void 0!==g.timestamp&&(A.timestamp=g.timestamp),void 0!==g.size&&EA.resizeFileStorage(A,g.size)},lookup:function(A,g){throw WA.genericErrors[mA.ENOENT]},mknod:function(A,g,I,C){return EA.createNode(A,g,I,C)},rename:function(A,g,I){if(WA.isDir(A.mode)){var C;try{C=WA.lookupNode(g,I)}catch(A){}if(C)for(var e in C.contents)throw new WA.ErrnoError(mA.ENOTEMPTY)}delete A.parent.contents[A.name],A.name=I,g.contents[I]=A,A.parent=g},unlink:function(A,g){delete A.contents[g]},rmdir:function(A,g){var I=WA.lookupNode(A,g);for(var C in I.contents)throw new WA.ErrnoError(mA.ENOTEMPTY);delete A.contents[g]},readdir:function(A){var g=[".",".."];for(var I in A.contents)A.contents.hasOwnProperty(I)&&g.push(I);return g},symlink:function(A,g,I){var C=EA.createNode(A,g,41471,0);return C.link=I,C},readlink:function(A){if(!WA.isLink(A.mode))throw new WA.ErrnoError(mA.EINVAL);return A.link}},stream_ops:{read:function(A,g,I,C,e){var i=A.node.contents;if(e>=A.node.usedBytes)return 0;var t=Math.min(A.node.usedBytes-e,C);if(c(t>=0),t>8&&i.subarray)g.set(i.subarray(e,e+t),I);else for(var o=0;o<t;o++)g[I+o]=i[e+o];return t},write:function(A,g,I,C,e,i){if(!C)return 0;var t=A.node;if(t.timestamp=Date.now(),g.subarray&&(!t.contents||t.contents.subarray)){if(i)return t.contents=g.subarray(I,I+C),t.usedBytes=C,C;if(0===t.usedBytes&&0===e)return t.contents=new Uint8Array(g.subarray(I,I+C)),t.usedBytes=C,C;if(e+C<=t.usedBytes)return t.contents.set(g.subarray(I,I+C),e),C}if(EA.expandFileStorage(t,e+C),t.contents.subarray&&g.subarray)t.contents.set(g.subarray(I,I+C),e);else for(var o=0;o<C;o++)t.contents[e+o]=g[I+o];return t.usedBytes=Math.max(t.usedBytes,e+C),C},llseek:function(A,g,I){var C=g;if(1===I?C+=A.position:2===I&&WA.isFile(A.node.mode)&&(C+=A.node.usedBytes),C<0)throw new WA.ErrnoError(mA.EINVAL);return C},allocate:function(A,g,I){EA.expandFileStorage(A.node,g+I),A.node.usedBytes=Math.max(A.node.usedBytes,g+I)},mmap:function(A,g,I,C,e,i,t){if(!WA.isFile(A.node.mode))throw new WA.ErrnoError(mA.ENODEV);var o,a,B=A.node.contents;if(2&t||B.buffer!==g&&B.buffer!==g.buffer){if((e>0||e+C<A.node.usedBytes)&&(B=B.subarray?B.subarray(e,e+C):Array.prototype.slice.call(B,e,e+C)),a=!0,!(o=gI(C)))throw new WA.ErrnoError(mA.ENOMEM);g.set(B,o)}else a=!1,o=B.byteOffset;return{ptr:o,allocated:a}},msync:function(A,g,I,C,e){if(!WA.isFile(A.node.mode))throw new WA.ErrnoError(mA.ENODEV);return 2&e||EA.stream_ops.write(A,g,0,C,I,!1),0}}},QA={dbs:{},indexedDB:function(){if("undefined"!=typeof indexedDB)return indexedDB;var A=null;return"object"==typeof window&&(A=window.indexedDB||window.mozIndexedDB||window.webkitIndexedDB||window.msIndexedDB),c(A,"IDBFS used, but indexedDB not supported"),A},DB_VERSION:21,DB_STORE_NAME:"FILE_DATA",mount:function(A){return EA.mount.apply(null,arguments)},syncfs:function(A,g,I){QA.getLocalSet(A,(function(C,e){if(C)return I(C);QA.getRemoteSet(A,(function(A,C){if(A)return I(A);var i=g?C:e,t=g?e:C;QA.reconcile(i,t,I)}))}))},getDB:function(A,g){var I,C=QA.dbs[A];if(C)return g(null,C);try{I=QA.indexedDB().open(A,QA.DB_VERSION)}catch(A){return g(A)}if(!I)return g("Unable to connect to IndexedDB");I.onupgradeneeded=function(A){var g,I=A.target.result,C=A.target.transaction;(g=I.objectStoreNames.contains(QA.DB_STORE_NAME)?C.objectStore(QA.DB_STORE_NAME):I.createObjectStore(QA.DB_STORE_NAME)).indexNames.contains("timestamp")||g.createIndex("timestamp","timestamp",{unique:!1})},I.onsuccess=function(){C=I.result,QA.dbs[A]=C,g(null,C)},I.onerror=function(A){g(this.error),A.preventDefault()}},getLocalSet:function(A,g){var I={};function C(A){return"."!==A&&".."!==A}function e(A){return function(g){return uA.join2(A,g)}}for(var i=WA.readdir(A.mountpoint).filter(C).map(e(A.mountpoint));i.length;){var t,o=i.pop();try{t=WA.stat(o)}catch(A){return g(A)}WA.isDir(t.mode)&&i.push.apply(i,WA.readdir(o).filter(C).map(e(o))),I[o]={timestamp:t.mtime}}return g(null,{type:"local",entries:I})},getRemoteSet:function(A,g){var I={};QA.getDB(A.mountpoint,(function(A,C){if(A)return g(A);try{var e=C.transaction([QA.DB_STORE_NAME],"readonly");e.onerror=function(A){g(this.error),A.preventDefault()},e.objectStore(QA.DB_STORE_NAME).index("timestamp").openKeyCursor().onsuccess=function(A){var e=A.target.result;if(!e)return g(null,{type:"remote",db:C,entries:I});I[e.primaryKey]={timestamp:e.key},e.continue()}}catch(A){return g(A)}}))},loadLocalEntry:function(A,g){var I,C;try{C=WA.lookupPath(A).node,I=WA.stat(A)}catch(A){return g(A)}return WA.isDir(I.mode)?g(null,{timestamp:I.mtime,mode:I.mode}):WA.isFile(I.mode)?(C.contents=EA.getFileDataAsTypedArray(C),g(null,{timestamp:I.mtime,mode:I.mode,contents:C.contents})):g(new Error("node type not supported"))},storeLocalEntry:function(A,g,I){try{if(WA.isDir(g.mode))WA.mkdir(A,g.mode);else{if(!WA.isFile(g.mode))return I(new Error("node type not supported"));WA.writeFile(A,g.contents,{canOwn:!0})}WA.chmod(A,g.mode),WA.utime(A,g.timestamp,g.timestamp)}catch(A){return I(A)}I(null)},removeLocalEntry:function(A,g){try{WA.lookupPath(A);var I=WA.stat(A);WA.isDir(I.mode)?WA.rmdir(A):WA.isFile(I.mode)&&WA.unlink(A)}catch(A){return g(A)}g(null)},loadRemoteEntry:function(A,g,I){var C=A.get(g);C.onsuccess=function(A){I(null,A.target.result)},C.onerror=function(A){I(this.error),A.preventDefault()}},storeRemoteEntry:function(A,g,I,C){var e=A.put(I,g);e.onsuccess=function(){C(null)},e.onerror=function(A){C(this.error),A.preventDefault()}},removeRemoteEntry:function(A,g,I){var C=A.delete(g);C.onsuccess=function(){I(null)},C.onerror=function(A){I(this.error),A.preventDefault()}},reconcile:function(A,g,I){var C=0,e=[];Object.keys(A.entries).forEach((function(I){var i=A.entries[I],t=g.entries[I];(!t||i.timestamp>t.timestamp)&&(e.push(I),C++)}));var i=[];if(Object.keys(g.entries).forEach((function(I){g.entries[I],A.entries[I]||(i.push(I),C++)})),!C)return I(null);var t=0,o=("remote"===A.type?A.db:g.db).transaction([QA.DB_STORE_NAME],"readwrite"),a=o.objectStore(QA.DB_STORE_NAME);function B(A){return A?B.errored?void 0:(B.errored=!0,I(A)):++t>=C?I(null):void 0}o.onerror=function(A){B(this.error),A.preventDefault()},e.sort().forEach((function(A){"local"===g.type?QA.loadRemoteEntry(a,A,(function(g,I){if(g)return B(g);QA.storeLocalEntry(A,I,B)})):QA.loadLocalEntry(A,(function(g,I){if(g)return B(g);QA.storeRemoteEntry(a,A,I,B)}))})),i.sort().reverse().forEach((function(A){"local"===g.type?QA.removeLocalEntry(A,B):QA.removeRemoteEntry(a,A,B)}))}},VA={isWindows:!1,staticInit:function(){VA.isWindows=!!process.platform.match(/^win/);var A=process.binding("constants");A.fs&&(A=A.fs),VA.flagsForNodeMap={1024:A.O_APPEND,64:A.O_CREAT,128:A.O_EXCL,0:A.O_RDONLY,2:A.O_RDWR,4096:A.O_SYNC,512:A.O_TRUNC,1:A.O_WRONLY}},bufferFrom:function(A){return Buffer.alloc?Buffer.from(A):new Buffer(A)},mount:function(A){return c(B),VA.createNode(null,"/",VA.getMode(A.opts.root),0)},createNode:function(A,g,I,C){if(!WA.isDir(I)&&!WA.isFile(I)&&!WA.isLink(I))throw new WA.ErrnoError(mA.EINVAL);var e=WA.createNode(A,g,I);return e.node_ops=VA.node_ops,e.stream_ops=VA.stream_ops,e},getMode:function(A){var g;try{g=fs.lstatSync(A),VA.isWindows&&(g.mode=g.mode|(292&g.mode)>>2)}catch(A){if(!A.code)throw A;throw new WA.ErrnoError(mA[A.code])}return g.mode},realPath:function(A){for(var g=[];A.parent!==A;)g.push(A.name),A=A.parent;return g.push(A.mount.opts.root),g.reverse(),uA.join.apply(null,g)},flagsForNode:function(A){A&=-2097153,A&=-2049,A&=-32769,A&=-524289;var g=0;for(var I in VA.flagsForNodeMap)A&I&&(g|=VA.flagsForNodeMap[I],A^=I);if(A)throw new WA.ErrnoError(mA.EINVAL);return g},node_ops:{getattr:function(A){var g,I=VA.realPath(A);try{g=fs.lstatSync(I)}catch(A){if(!A.code)throw A;throw new WA.ErrnoError(mA[A.code])}return VA.isWindows&&!g.blksize&&(g.blksize=4096),VA.isWindows&&!g.blocks&&(g.blocks=(g.size+g.blksize-1)/g.blksize|0),{dev:g.dev,ino:g.ino,mode:g.mode,nlink:g.nlink,uid:g.uid,gid:g.gid,rdev:g.rdev,size:g.size,atime:g.atime,mtime:g.mtime,ctime:g.ctime,blksize:g.blksize,blocks:g.blocks}},setattr:function(A,g){var I=VA.realPath(A);try{if(void 0!==g.mode&&(fs.chmodSync(I,g.mode),A.mode=g.mode),void 0!==g.timestamp){var C=new Date(g.timestamp);fs.utimesSync(I,C,C)}void 0!==g.size&&fs.truncateSync(I,g.size)}catch(A){if(!A.code)throw A;throw new WA.ErrnoError(mA[A.code])}},lookup:function(A,g){var I=uA.join2(VA.realPath(A),g),C=VA.getMode(I);return VA.createNode(A,g,C)},mknod:function(A,g,I,C){var e=VA.createNode(A,g,I,C),i=VA.realPath(e);try{WA.isDir(e.mode)?fs.mkdirSync(i,e.mode):fs.writeFileSync(i,"",{mode:e.mode})}catch(A){if(!A.code)throw A;throw new WA.ErrnoError(mA[A.code])}return e},rename:function(A,g,I){var C=VA.realPath(A),e=uA.join2(VA.realPath(g),I);try{fs.renameSync(C,e)}catch(A){if(!A.code)throw A;throw new WA.ErrnoError(mA[A.code])}},unlink:function(A,g){var I=uA.join2(VA.realPath(A),g);try{fs.unlinkSync(I)}catch(A){if(!A.code)throw A;throw new WA.ErrnoError(mA[A.code])}},rmdir:function(A,g){var I=uA.join2(VA.realPath(A),g);try{fs.rmdirSync(I)}catch(A){if(!A.code)throw A;throw new WA.ErrnoError(mA[A.code])}},readdir:function(A){var g=VA.realPath(A);try{return fs.readdirSync(g)}catch(A){if(!A.code)throw A;throw new WA.ErrnoError(mA[A.code])}},symlink:function(A,g,I){var C=uA.join2(VA.realPath(A),g);try{fs.symlinkSync(I,C)}catch(A){if(!A.code)throw A;throw new WA.ErrnoError(mA[A.code])}},readlink:function(A){var g=VA.realPath(A);try{return g=fs.readlinkSync(g),g=NODEJS_PATH.relative(NODEJS_PATH.resolve(A.mount.opts.root),g)}catch(A){if(!A.code)throw A;throw new WA.ErrnoError(mA[A.code])}}},stream_ops:{open:function(A){var g=VA.realPath(A.node);try{WA.isFile(A.node.mode)&&(A.nfd=fs.openSync(g,VA.flagsForNode(A.flags)))}catch(A){if(!A.code)throw A;throw new WA.ErrnoError(mA[A.code])}},close:function(A){try{WA.isFile(A.node.mode)&&A.nfd&&fs.closeSync(A.nfd)}catch(A){if(!A.code)throw A;throw new WA.ErrnoError(mA[A.code])}},read:function(A,g,I,C,e){if(0===C)return 0;try{return fs.readSync(A.nfd,VA.bufferFrom(g.buffer),I,C,e)}catch(A){throw new WA.ErrnoError(mA[A.code])}},write:function(A,g,I,C,e){try{return fs.writeSync(A.nfd,VA.bufferFrom(g.buffer),I,C,e)}catch(A){throw new WA.ErrnoError(mA[A.code])}},llseek:function(A,g,I){var C=g;if(1===I)C+=A.position;else if(2===I&&WA.isFile(A.node.mode))try{C+=fs.fstatSync(A.nfd).size}catch(A){throw new WA.ErrnoError(mA[A.code])}if(C<0)throw new WA.ErrnoError(mA.EINVAL);return C}}},RA={DIR_MODE:16895,FILE_MODE:33279,reader:null,mount:function(A){c(a),RA.reader||(RA.reader=new FileReaderSync);var g=RA.createNode(null,"/",RA.DIR_MODE,0),I={};function C(A){for(var C=A.split("/"),e=g,i=0;i<C.length-1;i++){var t=C.slice(0,i+1).join("/");I[t]||(I[t]=RA.createNode(e,C[i],RA.DIR_MODE,0)),e=I[t]}return e}function e(A){var g=A.split("/");return g[g.length-1]}return Array.prototype.forEach.call(A.opts.files||[],(function(A){RA.createNode(C(A.name),e(A.name),RA.FILE_MODE,0,A,A.lastModifiedDate)})),(A.opts.blobs||[]).forEach((function(A){RA.createNode(C(A.name),e(A.name),RA.FILE_MODE,0,A.data)})),(A.opts.packages||[]).forEach((function(A){A.metadata.files.forEach((function(g){var I=g.filename.substr(1);RA.createNode(C(I),e(I),RA.FILE_MODE,0,A.blob.slice(g.start,g.end))}))})),g},createNode:function(A,g,I,C,e,i){var t=WA.createNode(A,g,I);return t.mode=I,t.node_ops=RA.node_ops,t.stream_ops=RA.stream_ops,t.timestamp=(i||new Date).getTime(),c(RA.FILE_MODE!==RA.DIR_MODE),I===RA.FILE_MODE?(t.size=e.size,t.contents=e):(t.size=4096,t.contents={}),A&&(A.contents[g]=t),t},node_ops:{getattr:function(A){return{dev:1,ino:void 0,mode:A.mode,nlink:1,uid:0,gid:0,rdev:void 0,size:A.size,atime:new Date(A.timestamp),mtime:new Date(A.timestamp),ctime:new Date(A.timestamp),blksize:4096,blocks:Math.ceil(A.size/4096)}},setattr:function(A,g){void 0!==g.mode&&(A.mode=g.mode),void 0!==g.timestamp&&(A.timestamp=g.timestamp)},lookup:function(A,g){throw new WA.ErrnoError(mA.ENOENT)},mknod:function(A,g,I,C){throw new WA.ErrnoError(mA.EPERM)},rename:function(A,g,I){throw new WA.ErrnoError(mA.EPERM)},unlink:function(A,g){throw new WA.ErrnoError(mA.EPERM)},rmdir:function(A,g){throw new WA.ErrnoError(mA.EPERM)},readdir:function(A){var g=[".",".."];for(var I in A.contents)A.contents.hasOwnProperty(I)&&g.push(I);return g},symlink:function(A,g,I){throw new WA.ErrnoError(mA.EPERM)},readlink:function(A){throw new WA.ErrnoError(mA.EPERM)}},stream_ops:{read:function(A,g,I,C,e){if(e>=A.node.size)return 0;var i=A.node.contents.slice(e,e+C),t=RA.reader.readAsArrayBuffer(i);return g.set(new Uint8Array(t),I),i.size},write:function(A,g,I,C,e){throw new WA.ErrnoError(mA.EIO)},llseek:function(A,g,I){var C=g;if(1===I?C+=A.position:2===I&&WA.isFile(A.node.mode)&&(C+=A.node.size),C<0)throw new WA.ErrnoError(mA.EINVAL);return C}}};K+=16,K+=16;var WA={root:null,mounts:[],devices:{},streams:[],nextInode:1,nameTable:null,currentPath:"/",initialized:!1,ignorePermissions:!0,trackingDelegate:{},tracking:{openFlags:{READ:1,WRITE:2}},ErrnoError:null,genericErrors:{},filesystems:null,syncFSRequests:0,handleFSError:function(A){if(!(A instanceof WA.ErrnoError))throw A+" : "+R();return bA(A.errno)},lookupPath:function(A,g){if(g=g||{},!(A=uA.resolve(WA.cwd(),A)))return{path:"",node:null};var I={follow_mount:!0,recurse_count:0};for(var C in I)void 0===g[C]&&(g[C]=I[C]);if(g.recurse_count>8)throw new WA.ErrnoError(mA.ELOOP);for(var e=uA.normalizeArray(A.split("/").filter((function(A){return!!A})),!1),i=WA.root,t="/",o=0;o<e.length;o++){var a=o===e.length-1;if(a&&g.parent)break;if(i=WA.lookupNode(i,e[o]),t=uA.join2(t,e[o]),WA.isMountpoint(i)&&(!a||a&&g.follow_mount)&&(i=i.mounted.root),!a||g.follow)for(var B=0;WA.isLink(i.mode);){var s=WA.readlink(t);if(t=uA.resolve(uA.dirname(t),s),i=WA.lookupPath(t,{recurse_count:g.recurse_count}).node,B++>40)throw new WA.ErrnoError(mA.ELOOP)}}return{path:t,node:i}},getPath:function(A){for(var g;;){if(WA.isRoot(A)){var I=A.mount.mountpoint;return g?"/"!==I[I.length-1]?I+"/"+g:I+g:I}g=g?A.name+"/"+g:A.name,A=A.parent}},hashName:function(A,g){for(var I=0,C=0;C<g.length;C++)I=(I<<5)-I+g.charCodeAt(C)|0;return(A+I>>>0)%WA.nameTable.length},hashAddNode:function(A){var g=WA.hashName(A.parent.id,A.name);A.name_next=WA.nameTable[g],WA.nameTable[g]=A},hashRemoveNode:function(A){var g=WA.hashName(A.parent.id,A.name);if(WA.nameTable[g]===A)WA.nameTable[g]=A.name_next;else for(var I=WA.nameTable[g];I;){if(I.name_next===A){I.name_next=A.name_next;break}I=I.name_next}},lookupNode:function(A,g){var I=WA.mayLookup(A);if(I)throw new WA.ErrnoError(I,A);for(var C=WA.hashName(A.id,g),e=WA.nameTable[C];e;e=e.name_next){var i=e.name;if(e.parent.id===A.id&&i===g)return e}return WA.lookup(A,g)},createNode:function(A,g,I,C){if(!WA.FSNode){WA.FSNode=function(A,g,I,C){A||(A=this),this.parent=A,this.mount=A.mount,this.mounted=null,this.id=WA.nextInode++,this.name=g,this.mode=I,this.node_ops={},this.stream_ops={},this.rdev=C},WA.FSNode.prototype={};var e=365,i=146;Object.defineProperties(WA.FSNode.prototype,{read:{get:function(){return(this.mode&e)===e},set:function(A){A?this.mode|=e:this.mode&=-366}},write:{get:function(){return(this.mode&i)===i},set:function(A){A?this.mode|=i:this.mode&=-147}},isFolder:{get:function(){return WA.isDir(this.mode)}},isDevice:{get:function(){return WA.isChrdev(this.mode)}}})}var t=new WA.FSNode(A,g,I,C);return WA.hashAddNode(t),t},destroyNode:function(A){WA.hashRemoveNode(A)},isRoot:function(A){return A===A.parent},isMountpoint:function(A){return!!A.mounted},isFile:function(A){return 32768==(61440&A)},isDir:function(A){return 16384==(61440&A)},isLink:function(A){return 40960==(61440&A)},isChrdev:function(A){return 8192==(61440&A)},isBlkdev:function(A){return 24576==(61440&A)},isFIFO:function(A){return 4096==(61440&A)},isSocket:function(A){return!(49152&~A)},flagModes:{r:0,rs:1052672,"r+":2,w:577,wx:705,xw:705,"w+":578,"wx+":706,"xw+":706,a:1089,ax:1217,xa:1217,"a+":1090,"ax+":1218,"xa+":1218},modeStringToFlags:function(A){var g=WA.flagModes[A];if(void 0===g)throw new Error("Unknown file open mode: "+A);return g},flagsToPermissionString:function(A){var g=["r","w","rw"][3&A];return 512&A&&(g+="w"),g},nodePermissions:function(A,g){return WA.ignorePermissions||(-1===g.indexOf("r")||292&A.mode)&&(-1===g.indexOf("w")||146&A.mode)&&(-1===g.indexOf("x")||73&A.mode)?0:mA.EACCES},mayLookup:function(A){var g=WA.nodePermissions(A,"x");return g||(A.node_ops.lookup?0:mA.EACCES)},mayCreate:function(A,g){try{WA.lookupNode(A,g);return mA.EEXIST}catch(A){}return WA.nodePermissions(A,"wx")},mayDelete:function(A,g,I){var C;try{C=WA.lookupNode(A,g)}catch(A){return A.errno}var e=WA.nodePermissions(A,"wx");if(e)return e;if(I){if(!WA.isDir(C.mode))return mA.ENOTDIR;if(WA.isRoot(C)||WA.getPath(C)===WA.cwd())return mA.EBUSY}else if(WA.isDir(C.mode))return mA.EISDIR;return 0},mayOpen:function(A,g){return A?WA.isLink(A.mode)?mA.ELOOP:WA.isDir(A.mode)&&("r"!==WA.flagsToPermissionString(g)||512&g)?mA.EISDIR:WA.nodePermissions(A,WA.flagsToPermissionString(g)):mA.ENOENT},MAX_OPEN_FDS:4096,nextfd:function(A,g){A=A||0,g=g||WA.MAX_OPEN_FDS;for(var I=A;I<=g;I++)if(!WA.streams[I])return I;throw new WA.ErrnoError(mA.EMFILE)},getStream:function(A){return WA.streams[A]},createStream:function(A,g,I){WA.FSStream||(WA.FSStream=function(){},WA.FSStream.prototype={},Object.defineProperties(WA.FSStream.prototype,{object:{get:function(){return this.node},set:function(A){this.node=A}},isRead:{get:function(){return 1!=(2097155&this.flags)}},isWrite:{get:function(){return!!(2097155&this.flags)}},isAppend:{get:function(){return 1024&this.flags}}}));var C=new WA.FSStream;for(var e in A)C[e]=A[e];A=C;var i=WA.nextfd(g,I);return A.fd=i,WA.streams[i]=A,A},closeStream:function(A){WA.streams[A]=null},chrdev_stream_ops:{open:function(A){var g=WA.getDevice(A.node.rdev);A.stream_ops=g.stream_ops,A.stream_ops.open&&A.stream_ops.open(A)},llseek:function(){throw new WA.ErrnoError(mA.ESPIPE)}},major:function(A){return A>>8},minor:function(A){return 255&A},makedev:function(A,g){return A<<8|g},registerDevice:function(A,g){WA.devices[A]={stream_ops:g}},getDevice:function(A){return WA.devices[A]},getMounts:function(A){for(var g=[],I=[A];I.length;){var C=I.pop();g.push(C),I.push.apply(I,C.mounts)}return g},syncfs:function(A,g){"function"==typeof A&&(g=A,A=!1),WA.syncFSRequests++,WA.syncFSRequests>1&&console.log("warning: "+WA.syncFSRequests+" FS.syncfs operations in flight at once, probably just doing extra work");var I=WA.getMounts(WA.root.mount),C=0;function e(A){return c(WA.syncFSRequests>0),WA.syncFSRequests--,g(A)}function i(A){if(A)return i.errored?void 0:(i.errored=!0,e(A));++C>=I.length&&e(null)}I.forEach((function(g){if(!g.type.syncfs)return i(null);g.type.syncfs(g,A,i)}))},mount:function(A,g,I){var C,e="/"===I,i=!I;if(e&&WA.root)throw new WA.ErrnoError(mA.EBUSY);if(!e&&!i){var t=WA.lookupPath(I,{follow_mount:!1});if(I=t.path,C=t.node,WA.isMountpoint(C))throw new WA.ErrnoError(mA.EBUSY);if(!WA.isDir(C.mode))throw new WA.ErrnoError(mA.ENOTDIR)}var o={type:A,opts:g,mountpoint:I,mounts:[]},a=A.mount(o);return a.mount=o,o.root=a,e?WA.root=a:C&&(C.mounted=o,C.mount&&C.mount.mounts.push(o)),a},unmount:function(A){var g=WA.lookupPath(A,{follow_mount:!1});if(!WA.isMountpoint(g.node))throw new WA.ErrnoError(mA.EINVAL);var I=g.node,C=I.mounted,e=WA.getMounts(C);Object.keys(WA.nameTable).forEach((function(A){for(var g=WA.nameTable[A];g;){var I=g.name_next;-1!==e.indexOf(g.mount)&&WA.destroyNode(g),g=I}})),I.mounted=null;var i=I.mount.mounts.indexOf(C);c(-1!==i),I.mount.mounts.splice(i,1)},lookup:function(A,g){return A.node_ops.lookup(A,g)},mknod:function(A,g,I){var C=WA.lookupPath(A,{parent:!0}).node,e=uA.basename(A);if(!e||"."===e||".."===e)throw new WA.ErrnoError(mA.EINVAL);var i=WA.mayCreate(C,e);if(i)throw new WA.ErrnoError(i);if(!C.node_ops.mknod)throw new WA.ErrnoError(mA.EPERM);return C.node_ops.mknod(C,e,g,I)},create:function(A,g){return g=void 0!==g?g:438,g&=4095,g|=32768,WA.mknod(A,g,0)},mkdir:function(A,g){return g=void 0!==g?g:511,g&=1023,g|=16384,WA.mknod(A,g,0)},mkdirTree:function(A,g){for(var I=A.split("/"),C="",e=0;e<I.length;++e)if(I[e]){C+="/"+I[e];try{WA.mkdir(C,g)}catch(A){if(A.errno!=mA.EEXIST)throw A}}},mkdev:function(A,g,I){return void 0===I&&(I=g,g=438),g|=8192,WA.mknod(A,g,I)},symlink:function(A,g){if(!uA.resolve(A))throw new WA.ErrnoError(mA.ENOENT);var I=WA.lookupPath(g,{parent:!0}).node;if(!I)throw new WA.ErrnoError(mA.ENOENT);var C=uA.basename(g),e=WA.mayCreate(I,C);if(e)throw new WA.ErrnoError(e);if(!I.node_ops.symlink)throw new WA.ErrnoError(mA.EPERM);return I.node_ops.symlink(I,C,A)},rename:function(A,g){var I,C,e=uA.dirname(A),i=uA.dirname(g),t=uA.basename(A),o=uA.basename(g);try{I=WA.lookupPath(A,{parent:!0}).node,C=WA.lookupPath(g,{parent:!0}).node}catch(A){throw new WA.ErrnoError(mA.EBUSY)}if(!I||!C)throw new WA.ErrnoError(mA.ENOENT);if(I.mount!==C.mount)throw new WA.ErrnoError(mA.EXDEV);var a,B=WA.lookupNode(I,t),s=uA.relative(A,i);if("."!==s.charAt(0))throw new WA.ErrnoError(mA.EINVAL);if("."!==(s=uA.relative(g,e)).charAt(0))throw new WA.ErrnoError(mA.ENOTEMPTY);try{a=WA.lookupNode(C,o)}catch(A){}if(B!==a){var l=WA.isDir(B.mode),n=WA.mayDelete(I,t,l);if(n)throw new WA.ErrnoError(n);if(n=a?WA.mayDelete(C,o,l):WA.mayCreate(C,o))throw new WA.ErrnoError(n);if(!I.node_ops.rename)throw new WA.ErrnoError(mA.EPERM);if(WA.isMountpoint(B)||a&&WA.isMountpoint(a))throw new WA.ErrnoError(mA.EBUSY);if(C!==I&&(n=WA.nodePermissions(I,"w")))throw new WA.ErrnoError(n);try{WA.trackingDelegate.willMovePath&&WA.trackingDelegate.willMovePath(A,g)}catch(I){console.log("FS.trackingDelegate['willMovePath']('"+A+"', '"+g+"') threw an exception: "+I.message)}WA.hashRemoveNode(B);try{I.node_ops.rename(B,C,o)}catch(A){throw A}finally{WA.hashAddNode(B)}try{WA.trackingDelegate.onMovePath&&WA.trackingDelegate.onMovePath(A,g)}catch(I){console.log("FS.trackingDelegate['onMovePath']('"+A+"', '"+g+"') threw an exception: "+I.message)}}},rmdir:function(A){var g=WA.lookupPath(A,{parent:!0}).node,I=uA.basename(A),C=WA.lookupNode(g,I),e=WA.mayDelete(g,I,!0);if(e)throw new WA.ErrnoError(e);if(!g.node_ops.rmdir)throw new WA.ErrnoError(mA.EPERM);if(WA.isMountpoint(C))throw new WA.ErrnoError(mA.EBUSY);try{WA.trackingDelegate.willDeletePath&&WA.trackingDelegate.willDeletePath(A)}catch(g){console.log("FS.trackingDelegate['willDeletePath']('"+A+"') threw an exception: "+g.message)}g.node_ops.rmdir(g,I),WA.destroyNode(C);try{WA.trackingDelegate.onDeletePath&&WA.trackingDelegate.onDeletePath(A)}catch(g){console.log("FS.trackingDelegate['onDeletePath']('"+A+"') threw an exception: "+g.message)}},readdir:function(A){var g=WA.lookupPath(A,{follow:!0}).node;if(!g.node_ops.readdir)throw new WA.ErrnoError(mA.ENOTDIR);return g.node_ops.readdir(g)},unlink:function(A){var g=WA.lookupPath(A,{parent:!0}).node,I=uA.basename(A),C=WA.lookupNode(g,I),e=WA.mayDelete(g,I,!1);if(e)throw new WA.ErrnoError(e);if(!g.node_ops.unlink)throw new WA.ErrnoError(mA.EPERM);if(WA.isMountpoint(C))throw new WA.ErrnoError(mA.EBUSY);try{WA.trackingDelegate.willDeletePath&&WA.trackingDelegate.willDeletePath(A)}catch(g){console.log("FS.trackingDelegate['willDeletePath']('"+A+"') threw an exception: "+g.message)}g.node_ops.unlink(g,I),WA.destroyNode(C);try{WA.trackingDelegate.onDeletePath&&WA.trackingDelegate.onDeletePath(A)}catch(g){console.log("FS.trackingDelegate['onDeletePath']('"+A+"') threw an exception: "+g.message)}},readlink:function(A){var g=WA.lookupPath(A).node;if(!g)throw new WA.ErrnoError(mA.ENOENT);if(!g.node_ops.readlink)throw new WA.ErrnoError(mA.EINVAL);return uA.resolve(WA.getPath(g.parent),g.node_ops.readlink(g))},stat:function(A,g){var I=WA.lookupPath(A,{follow:!g}).node;if(!I)throw new WA.ErrnoError(mA.ENOENT);if(!I.node_ops.getattr)throw new WA.ErrnoError(mA.EPERM);return I.node_ops.getattr(I)},lstat:function(A){return WA.stat(A,!0)},chmod:function(A,g,I){var C;"string"==typeof A?C=WA.lookupPath(A,{follow:!I}).node:C=A;if(!C.node_ops.setattr)throw new WA.ErrnoError(mA.EPERM);C.node_ops.setattr(C,{mode:4095&g|-4096&C.mode,timestamp:Date.now()})},lchmod:function(A,g){WA.chmod(A,g,!0)},fchmod:function(A,g){var I=WA.getStream(A);if(!I)throw new WA.ErrnoError(mA.EBADF);WA.chmod(I.node,g)},chown:function(A,g,I,C){var e;"string"==typeof A?e=WA.lookupPath(A,{follow:!C}).node:e=A;if(!e.node_ops.setattr)throw new WA.ErrnoError(mA.EPERM);e.node_ops.setattr(e,{timestamp:Date.now()})},lchown:function(A,g,I){WA.chown(A,g,I,!0)},fchown:function(A,g,I){var C=WA.getStream(A);if(!C)throw new WA.ErrnoError(mA.EBADF);WA.chown(C.node,g,I)},truncate:function(A,g){if(g<0)throw new WA.ErrnoError(mA.EINVAL);var I;"string"==typeof A?I=WA.lookupPath(A,{follow:!0}).node:I=A;if(!I.node_ops.setattr)throw new WA.ErrnoError(mA.EPERM);if(WA.isDir(I.mode))throw new WA.ErrnoError(mA.EISDIR);if(!WA.isFile(I.mode))throw new WA.ErrnoError(mA.EINVAL);var C=WA.nodePermissions(I,"w");if(C)throw new WA.ErrnoError(C);I.node_ops.setattr(I,{size:g,timestamp:Date.now()})},ftruncate:function(A,g){var I=WA.getStream(A);if(!I)throw new WA.ErrnoError(mA.EBADF);if(!(2097155&I.flags))throw new WA.ErrnoError(mA.EINVAL);WA.truncate(I.node,g)},utime:function(A,g,I){var C=WA.lookupPath(A,{follow:!0}).node;C.node_ops.setattr(C,{timestamp:Math.max(g,I)})},open:function(A,g,I,C,e){if(""===A)throw new WA.ErrnoError(mA.ENOENT);var i;if(I=void 0===I?438:I,I=64&(g="string"==typeof g?WA.modeStringToFlags(g):g)?4095&I|32768:0,"object"==typeof A)i=A;else{A=uA.normalize(A);try{i=WA.lookupPath(A,{follow:!(131072&g)}).node}catch(A){}}var t=!1;if(64&g)if(i){if(128&g)throw new WA.ErrnoError(mA.EEXIST)}else i=WA.mknod(A,I,0),t=!0;if(!i)throw new WA.ErrnoError(mA.ENOENT);if(WA.isChrdev(i.mode)&&(g&=-513),65536&g&&!WA.isDir(i.mode))throw new WA.ErrnoError(mA.ENOTDIR);if(!t){var o=WA.mayOpen(i,g);if(o)throw new WA.ErrnoError(o)}512&g&&WA.truncate(i,0),g&=-641;var a=WA.createStream({node:i,path:WA.getPath(i),flags:g,seekable:!0,position:0,stream_ops:i.stream_ops,ungotten:[],error:!1},C,e);a.stream_ops.open&&a.stream_ops.open(a),!Module.logReadFiles||1&g||(WA.readFiles||(WA.readFiles={}),A in WA.readFiles||(WA.readFiles[A]=1,Module.printErr("read file: "+A)));try{if(WA.trackingDelegate.onOpenFile){var B=0;1!=(2097155&g)&&(B|=WA.tracking.openFlags.READ),2097155&g&&(B|=WA.tracking.openFlags.WRITE),WA.trackingDelegate.onOpenFile(A,B)}}catch(g){console.log("FS.trackingDelegate['onOpenFile']('"+A+"', flags) threw an exception: "+g.message)}return a},close:function(A){if(WA.isClosed(A))throw new WA.ErrnoError(mA.EBADF);A.getdents&&(A.getdents=null);try{A.stream_ops.close&&A.stream_ops.close(A)}catch(A){throw A}finally{WA.closeStream(A.fd)}A.fd=null},isClosed:function(A){return null===A.fd},llseek:function(A,g,I){if(WA.isClosed(A))throw new WA.ErrnoError(mA.EBADF);if(!A.seekable||!A.stream_ops.llseek)throw new WA.ErrnoError(mA.ESPIPE);return A.position=A.stream_ops.llseek(A,g,I),A.ungotten=[],A.position},read:function(A,g,I,C,e){if(C<0||e<0)throw new WA.ErrnoError(mA.EINVAL);if(WA.isClosed(A))throw new WA.ErrnoError(mA.EBADF);if(1==(2097155&A.flags))throw new WA.ErrnoError(mA.EBADF);if(WA.isDir(A.node.mode))throw new WA.ErrnoError(mA.EISDIR);if(!A.stream_ops.read)throw new WA.ErrnoError(mA.EINVAL);var i=void 0!==e;if(i){if(!A.seekable)throw new WA.ErrnoError(mA.ESPIPE)}else e=A.position;var t=A.stream_ops.read(A,g,I,C,e);return i||(A.position+=t),t},write:function(A,g,I,C,e,i){if(C<0||e<0)throw new WA.ErrnoError(mA.EINVAL);if(WA.isClosed(A))throw new WA.ErrnoError(mA.EBADF);if(!(2097155&A.flags))throw new WA.ErrnoError(mA.EBADF);if(WA.isDir(A.node.mode))throw new WA.ErrnoError(mA.EISDIR);if(!A.stream_ops.write)throw new WA.ErrnoError(mA.EINVAL);1024&A.flags&&WA.llseek(A,0,2);var t=void 0!==e;if(t){if(!A.seekable)throw new WA.ErrnoError(mA.ESPIPE)}else e=A.position;var o=A.stream_ops.write(A,g,I,C,e,i);t||(A.position+=o);try{A.path&&WA.trackingDelegate.onWriteToFile&&WA.trackingDelegate.onWriteToFile(A.path)}catch(A){console.log("FS.trackingDelegate['onWriteToFile']('"+path+"') threw an exception: "+A.message)}return o},allocate:function(A,g,I){if(WA.isClosed(A))throw new WA.ErrnoError(mA.EBADF);if(g<0||I<=0)throw new WA.ErrnoError(mA.EINVAL);if(!(2097155&A.flags))throw new WA.ErrnoError(mA.EBADF);if(!WA.isFile(A.node.mode)&&!WA.isDir(A.node.mode))throw new WA.ErrnoError(mA.ENODEV);if(!A.stream_ops.allocate)throw new WA.ErrnoError(mA.EOPNOTSUPP);A.stream_ops.allocate(A,g,I)},mmap:function(A,g,I,C,e,i,t){if(1==(2097155&A.flags))throw new WA.ErrnoError(mA.EACCES);if(!A.stream_ops.mmap)throw new WA.ErrnoError(mA.ENODEV);return A.stream_ops.mmap(A,g,I,C,e,i,t)},msync:function(A,g,I,C,e){return A&&A.stream_ops.msync?A.stream_ops.msync(A,g,I,C,e):0},munmap:function(A){return 0},ioctl:function(A,g,I){if(!A.stream_ops.ioctl)throw new WA.ErrnoError(mA.ENOTTY);return A.stream_ops.ioctl(A,g,I)},readFile:function(A,g){if((g=g||{}).flags=g.flags||"r",g.encoding=g.encoding||"binary","utf8"!==g.encoding&&"binary"!==g.encoding)throw new Error('Invalid encoding type "'+g.encoding+'"');var I,C=WA.open(A,g.flags),e=WA.stat(A).size,i=new Uint8Array(e);return WA.read(C,i,0,e,0),"utf8"===g.encoding?I=Z(i,0):"binary"===g.encoding&&(I=i),WA.close(C),I},writeFile:function(A,g,I){(I=I||{}).flags=I.flags||"w";var C=WA.open(A,I.flags,I.mode);if("string"==typeof g){var e=new Uint8Array(Q(g)+1),i=y(g,e,0,e.length);WA.write(C,e,0,i,void 0,I.canOwn)}else{if(!ArrayBuffer.isView(g))throw new Error("Unsupported data type");WA.write(C,g,0,g.byteLength,void 0,I.canOwn)}WA.close(C)},cwd:function(){return WA.currentPath},chdir:function(A){var g=WA.lookupPath(A,{follow:!0});if(null===g.node)throw new WA.ErrnoError(mA.ENOENT);if(!WA.isDir(g.node.mode))throw new WA.ErrnoError(mA.ENOTDIR);var I=WA.nodePermissions(g.node,"x");if(I)throw new WA.ErrnoError(I);WA.currentPath=g.path},createDefaultDirectories:function(){WA.mkdir("/tmp"),WA.mkdir("/home"),WA.mkdir("/home/web_user")},createDefaultDevices:function(){var A;if(WA.mkdir("/dev"),WA.registerDevice(WA.makedev(1,3),{read:function(){return 0},write:function(A,g,I,C,e){return C}}),WA.mkdev("/dev/null",WA.makedev(1,3)),yA.register(WA.makedev(5,0),yA.default_tty_ops),yA.register(WA.makedev(6,0),yA.default_tty1_ops),WA.mkdev("/dev/tty",WA.makedev(5,0)),WA.mkdev("/dev/tty1",WA.makedev(6,0)),"undefined"!=typeof crypto){var g=new Uint8Array(1);A=function(){return crypto.getRandomValues(g),g[0]}}else A=B?function(){return require("crypto").randomBytes(1)[0]}:function(){return 256*Math.random()|0};WA.createDevice("/dev","random",A),WA.createDevice("/dev","urandom",A),WA.mkdir("/dev/shm"),WA.mkdir("/dev/shm/tmp")},createSpecialDirectories:function(){WA.mkdir("/proc"),WA.mkdir("/proc/self"),WA.mkdir("/proc/self/fd"),WA.mount({mount:function(){var A=WA.createNode("/proc/self","fd",16895,73);return A.node_ops={lookup:function(A,g){var I=+g,C=WA.getStream(I);if(!C)throw new WA.ErrnoError(mA.EBADF);var e={parent:null,mount:{mountpoint:"fake"},node_ops:{readlink:function(){return C.path}}};return e.parent=e,e}},A}},{},"/proc/self/fd")},createStandardStreams:function(){Module.stdin?WA.createDevice("/dev","stdin",Module.stdin):WA.symlink("/dev/tty","/dev/stdin"),Module.stdout?WA.createDevice("/dev","stdout",null,Module.stdout):WA.symlink("/dev/tty","/dev/stdout"),Module.stderr?WA.createDevice("/dev","stderr",null,Module.stderr):WA.symlink("/dev/tty1","/dev/stderr");var A=WA.open("/dev/stdin","r");c(0===A.fd,"invalid handle for stdin ("+A.fd+")");var g=WA.open("/dev/stdout","w");c(1===g.fd,"invalid handle for stdout ("+g.fd+")");var I=WA.open("/dev/stderr","w");c(2===I.fd,"invalid handle for stderr ("+I.fd+")")},ensureErrnoError:function(){WA.ErrnoError||(WA.ErrnoError=function(A,g){this.node=g,this.setErrno=function(A){for(var g in this.errno=A,mA)if(mA[g]===A){this.code=g;break}},this.setErrno(A),this.message=ZA[A],this.stack&&Object.defineProperty(this,"stack",{value:(new Error).stack,writable:!0})},WA.ErrnoError.prototype=new Error,WA.ErrnoError.prototype.constructor=WA.ErrnoError,[mA.ENOENT].forEach((function(A){WA.genericErrors[A]=new WA.ErrnoError(A),WA.genericErrors[A].stack="<generic error, no stack>"})))},staticInit:function(){WA.ensureErrnoError(),WA.nameTable=new Array(4096),WA.mount(EA,{},"/"),WA.createDefaultDirectories(),WA.createDefaultDevices(),WA.createSpecialDirectories(),WA.filesystems={MEMFS:EA,IDBFS:QA,NODEFS:VA,WORKERFS:RA}},init:function(A,g,I){c(!WA.init.initialized,"FS.init was previously called. If you want to initialize later with custom parameters, remove any earlier calls (note that one is automatically added to the generated code)"),WA.init.initialized=!0,WA.ensureErrnoError(),Module.stdin=A||Module.stdin,Module.stdout=g||Module.stdout,Module.stderr=I||Module.stderr,WA.createStandardStreams()},quit:function(){WA.init.initialized=!1;var A=Module._fflush;A&&A(0);for(var g=0;g<WA.streams.length;g++){var I=WA.streams[g];I&&WA.close(I)}},getMode:function(A,g){var I=0;return A&&(I|=365),g&&(I|=146),I},joinPath:function(A,g){var I=uA.join.apply(null,A);return g&&"/"==I[0]&&(I=I.substr(1)),I},absolutePath:function(A,g){return uA.resolve(g,A)},standardizePath:function(A){return uA.normalize(A)},findObject:function(A,g){var I=WA.analyzePath(A,g);return I.exists?I.object:(bA(I.error),null)},analyzePath:function(A,g){try{A=(C=WA.lookupPath(A,{follow:!g})).path}catch(A){}var I={isRoot:!1,exists:!1,error:0,name:null,path:null,object:null,parentExists:!1,parentPath:null,parentObject:null};try{var C=WA.lookupPath(A,{parent:!0});I.parentExists=!0,I.parentPath=C.path,I.parentObject=C.node,I.name=uA.basename(A),C=WA.lookupPath(A,{follow:!g}),I.exists=!0,I.path=C.path,I.object=C.node,I.name=C.node.name,I.isRoot="/"===C.path}catch(A){I.error=A.errno}return I},createFolder:function(A,g,I,C){var e=uA.join2("string"==typeof A?A:WA.getPath(A),g),i=WA.getMode(I,C);return WA.mkdir(e,i)},createPath:function(A,g,I,C){A="string"==typeof A?A:WA.getPath(A);for(var e=g.split("/").reverse();e.length;){var i=e.pop();if(i){var t=uA.join2(A,i);try{WA.mkdir(t)}catch(A){}A=t}}return t},createFile:function(A,g,I,C,e){var i=uA.join2("string"==typeof A?A:WA.getPath(A),g),t=WA.getMode(C,e);return WA.create(i,t)},createDataFile:function(A,g,I,C,e,i){var t=g?uA.join2("string"==typeof A?A:WA.getPath(A),g):A,o=WA.getMode(C,e),a=WA.create(t,o);if(I){if("string"==typeof I){for(var B=new Array(I.length),s=0,l=I.length;s<l;++s)B[s]=I.charCodeAt(s);I=B}WA.chmod(a,146|o);var n=WA.open(a,"w");WA.write(n,I,0,I.length,0,i),WA.close(n),WA.chmod(a,o)}return a},createDevice:function(A,g,I,C){var e=uA.join2("string"==typeof A?A:WA.getPath(A),g),i=WA.getMode(!!I,!!C);WA.createDevice.major||(WA.createDevice.major=64);var t=WA.makedev(WA.createDevice.major++,0);return WA.registerDevice(t,{open:function(A){A.seekable=!1},close:function(A){C&&C.buffer&&C.buffer.length&&C(10)},read:function(A,g,C,e,i){for(var t=0,o=0;o<e;o++){var a;try{a=I()}catch(A){throw new WA.ErrnoError(mA.EIO)}if(void 0===a&&0===t)throw new WA.ErrnoError(mA.EAGAIN);if(null==a)break;t++,g[C+o]=a}return t&&(A.node.timestamp=Date.now()),t},write:function(A,g,I,e,i){for(var t=0;t<e;t++)try{C(g[I+t])}catch(A){throw new WA.ErrnoError(mA.EIO)}return e&&(A.node.timestamp=Date.now()),t}}),WA.mkdev(e,i,t)},createLink:function(A,g,I,C,e){var i=uA.join2("string"==typeof A?A:WA.getPath(A),g);return WA.symlink(I,i)},forceLoadFile:function(A){if(A.isDevice||A.isFolder||A.link||A.contents)return!0;var g=!0;if("undefined"!=typeof XMLHttpRequest)throw new Error("Lazy loading should have been performed (contents set) in createLazyFile, but it was not. Lazy loading only works in web workers. Use --embed-file or --preload-file in emcc on the main thread.");if(!Module.read)throw new Error("Cannot load without read() or XMLHttpRequest.");try{A.contents=Tg(Module.read(A.url),!0),A.usedBytes=A.contents.length}catch(A){g=!1}return g||bA(mA.EIO),g},createLazyFile:function(A,g,I,C,e){function i(){this.lengthKnown=!1,this.chunks=[]}if(i.prototype.get=function(A){if(!(A>this.length-1||A<0)){var g=A%this.chunkSize,I=A/this.chunkSize|0;return this.getter(I)[g]}},i.prototype.setDataGetter=function(A){this.getter=A},i.prototype.cacheLength=function(){var A=new XMLHttpRequest;if(A.open("HEAD",I,!1),A.send(null),!(A.status>=200&&A.status<300||304===A.status))throw new Error("Couldn't load "+I+". Status: "+A.status);var g,C=Number(A.getResponseHeader("Content-length")),e=(g=A.getResponseHeader("Accept-Ranges"))&&"bytes"===g,i=(g=A.getResponseHeader("Content-Encoding"))&&"gzip"===g,t=1048576;e||(t=C);var o=this;o.setDataGetter((function(A){var g=A*t,e=(A+1)*t-1;if(e=Math.min(e,C-1),void 0===o.chunks[A]&&(o.chunks[A]=function(A,g){if(A>g)throw new Error("invalid range ("+A+", "+g+") or no bytes requested!");if(g>C-1)throw new Error("only "+C+" bytes available! programmer error!");var e=new XMLHttpRequest;if(e.open("GET",I,!1),C!==t&&e.setRequestHeader("Range","bytes="+A+"-"+g),"undefined"!=typeof Uint8Array&&(e.responseType="arraybuffer"),e.overrideMimeType&&e.overrideMimeType("text/plain; charset=x-user-defined"),e.send(null),!(e.status>=200&&e.status<300||304===e.status))throw new Error("Couldn't load "+I+". Status: "+e.status);return void 0!==e.response?new Uint8Array(e.response||[]):Tg(e.responseText||"",!0)}(g,e)),void 0===o.chunks[A])throw new Error("doXHR failed!");return o.chunks[A]})),!i&&C||(t=C=1,C=this.getter(0).length,t=C,console.log("LazyFiles on gzip forces download of the whole file when length is accessed")),this._length=C,this._chunkSize=t,this.lengthKnown=!0},"undefined"!=typeof XMLHttpRequest){if(!a)throw"Cannot do synchronous binary XHRs outside webworkers in modern browsers. Use --embed-file or --preload-file in emcc";var t=new i;Object.defineProperties(t,{length:{get:function(){return this.lengthKnown||this.cacheLength(),this._length}},chunkSize:{get:function(){return this.lengthKnown||this.cacheLength(),this._chunkSize}}});var o={isDevice:!1,contents:t}}else o={isDevice:!1,url:I};var B=WA.createFile(A,g,o,C,e);o.contents?B.contents=o.contents:o.url&&(B.contents=null,B.url=o.url),Object.defineProperties(B,{usedBytes:{get:function(){return this.contents.length}}});var s={};return Object.keys(B.stream_ops).forEach((function(A){var g=B.stream_ops[A];s[A]=function(){if(!WA.forceLoadFile(B))throw new WA.ErrnoError(mA.EIO);return g.apply(null,arguments)}})),s.read=function(A,g,I,C,e){if(!WA.forceLoadFile(B))throw new WA.ErrnoError(mA.EIO);var i=A.node.contents;if(e>=i.length)return 0;var t=Math.min(i.length-e,C);if(c(t>=0),i.slice)for(var o=0;o<t;o++)g[I+o]=i[e+o];else for(o=0;o<t;o++)g[I+o]=i.get(e+o);return t},B.stream_ops=s,B},createPreloadedFile:function(A,g,I,C,e,i,t,o,a,B){Browser.init();var s=g?uA.resolve(uA.join2(A,g)):A;function l(I){function l(I){B&&B(),o||WA.createDataFile(A,g,I,C,e,a),i&&i(),lA()}var n=!1;Module.preloadPlugins.forEach((function(A){n||A.canHandle(s)&&(A.handle(I,s,l,(function(){t&&t(),lA()})),n=!0)})),n||l(I)}sA(),"string"==typeof I?Browser.asyncLoad(I,(function(A){l(A)}),t):l(I)},indexedDB:function(){return window.indexedDB||window.mozIndexedDB||window.webkitIndexedDB||window.msIndexedDB},DB_NAME:function(){return"EM_FS_"+window.location.pathname},DB_VERSION:20,DB_STORE_NAME:"FILE_DATA",saveFilesToDB:function(A,g,I){g=g||function(){},I=I||function(){};var C=WA.indexedDB();try{var e=C.open(WA.DB_NAME(),WA.DB_VERSION)}catch(A){return I(A)}e.onupgradeneeded=function(){console.log("creating db"),e.result.createObjectStore(WA.DB_STORE_NAME)},e.onsuccess=function(){var C=e.result.transaction([WA.DB_STORE_NAME],"readwrite"),i=C.objectStore(WA.DB_STORE_NAME),t=0,o=0,a=A.length;function B(){0==o?g():I()}A.forEach((function(A){var g=i.put(WA.analyzePath(A).object.contents,A);g.onsuccess=function(){++t+o==a&&B()},g.onerror=function(){o++,t+o==a&&B()}})),C.onerror=I},e.onerror=I},loadFilesFromDB:function(A,g,I){g=g||function(){},I=I||function(){};var C=WA.indexedDB();try{var e=C.open(WA.DB_NAME(),WA.DB_VERSION)}catch(A){return I(A)}e.onupgradeneeded=I,e.onsuccess=function(){var C=e.result;try{var i=C.transaction([WA.DB_STORE_NAME],"readonly")}catch(A){return void I(A)}var t=i.objectStore(WA.DB_STORE_NAME),o=0,a=0,B=A.length;function s(){0==a?g():I()}A.forEach((function(A){var g=t.get(A);g.onsuccess=function(){WA.analyzePath(A).exists&&WA.unlink(A),WA.createDataFile(uA.dirname(A),uA.basename(A),g.result,!0,!0,!0),++o+a==B&&s()},g.onerror=function(){a++,o+a==B&&s()}})),i.onerror=I},e.onerror=I}},pA={DEFAULT_POLLMASK:5,mappings:{},umask:511,calculateAt:function(A,g){if("/"!==g[0]){var I;if(-100===A)I=WA.cwd();else{var C=WA.getStream(A);if(!C)throw new WA.ErrnoError(mA.EBADF);I=C.path}g=uA.join2(I,g)}return g},doStat:function(A,g,I){try{var C=A(g)}catch(A){if(A&&A.node&&uA.normalize(g)!==uA.normalize(WA.getPath(A.node)))return-mA.ENOTDIR;throw A}return w[I>>2]=C.dev,w[I+4>>2]=0,w[I+8>>2]=C.ino,w[I+12>>2]=C.mode,w[I+16>>2]=C.nlink,w[I+20>>2]=C.uid,w[I+24>>2]=C.gid,w[I+28>>2]=C.rdev,w[I+32>>2]=0,w[I+36>>2]=C.size,w[I+40>>2]=4096,w[I+44>>2]=C.blocks,w[I+48>>2]=C.atime.getTime()/1e3|0,w[I+52>>2]=0,w[I+56>>2]=C.mtime.getTime()/1e3|0,w[I+60>>2]=0,w[I+64>>2]=C.ctime.getTime()/1e3|0,w[I+68>>2]=0,w[I+72>>2]=C.ino,0},doMsync:function(A,g,I,C){var e=new Uint8Array(p.subarray(A,A+I));WA.msync(g,e,0,I,C)},doMkdir:function(A,g){return"/"===(A=uA.normalize(A))[A.length-1]&&(A=A.substr(0,A.length-1)),WA.mkdir(A,g,0),0},doMknod:function(A,g,I){switch(61440&g){case 32768:case 8192:case 24576:case 4096:case 49152:break;default:return-mA.EINVAL}return WA.mknod(A,g,I),0},doReadlink:function(A,g,I){if(I<=0)return-mA.EINVAL;var C=WA.readlink(A),e=Math.min(I,Q(C)),i=W[g+e];return E(C,g,I+1),W[g+e]=i,e},doAccess:function(A,g){if(-8&g)return-mA.EINVAL;var I;I=WA.lookupPath(A,{follow:!0}).node;var C="";return 4&g&&(C+="r"),2&g&&(C+="w"),1&g&&(C+="x"),C&&WA.nodePermissions(I,C)?-mA.EACCES:0},doDup:function(A,g,I){var C=WA.getStream(I);return C&&WA.close(C),WA.open(A,g,0,I,I).fd},doReadv:function(A,g,I,C){for(var e=0,i=0;i<I;i++){var t=w[g+8*i>>2],o=w[g+(8*i+4)>>2],a=WA.read(A,W,t,o,C);if(a<0)return-1;if(e+=a,a<o)break}return e},doWritev:function(A,g,I,C){for(var e=0,i=0;i<I;i++){var t=w[g+8*i>>2],o=w[g+(8*i+4)>>2],a=WA.write(A,W,t,o,C);if(a<0)return-1;e+=a}return e},varargs:0,get:function(A){return pA.varargs+=4,w[pA.varargs-4>>2]},getStr:function(){return m(pA.get())},getStreamFromFD:function(){var A=WA.getStream(pA.get());if(!A)throw new WA.ErrnoError(mA.EBADF);return A},getSocketFromFD:function(){var A=SOCKFS.getSocket(pA.get());if(!A)throw new WA.ErrnoError(mA.EBADF);return A},getSocketAddress:function(A){var g=pA.get(),I=pA.get();if(A&&0===g)return null;var C=__read_sockaddr(g,I);if(C.errno)throw new WA.ErrnoError(C.errno);return C.addr=DNS.lookup_addr(C.addr)||C.addr,C},get64:function(){var A=pA.get(),g=pA.get();return c(A>=0?0===g:-1===g),A},getZero:function(){c(0===pA.get())}};var SA=42;var FA={};function wA(A){for(;A.length;){var g=A.pop();A.pop()(g)}}function HA(A){return this.fromWireType(H[A>>2])}var XA={},YA={},KA={};function NA(A){if(void 0===A)return"_unknown";var g=(A=A.replace(/[^a-zA-Z0-9_]/g,"$")).charCodeAt(0);return g>=48&&g<=57?"_"+A:A}function vA(A,g){return A=NA(A),new Function("body","return function "+A+'() {\n    "use strict";    return body.apply(this, arguments);\n};\n')(g)}function JA(A,g){var I=vA(g,(function(A){this.name=g,this.message=A;var I=new Error(A).stack;void 0!==I&&(this.stack=this.toString()+"\n"+I.replace(/^Error(:[^\n]*)?\n/,""))}));return I.prototype=Object.create(A.prototype),I.prototype.constructor=I,I.prototype.toString=function(){return void 0===this.message?this.name:this.name+": "+this.message},I}var fA=void 0;function LA(A){throw new fA(A)}function MA(A,g,I){function C(g){var C=I(g);C.length!==A.length&&LA("Mismatched type converter count");for(var e=0;e<A.length;++e)zA(A[e],C[e])}A.forEach((function(A){KA[A]=g}));var e=new Array(g.length),i=[],t=0;g.forEach((function(A,g){YA.hasOwnProperty(A)?e[g]=YA[A]:(i.push(A),XA.hasOwnProperty(A)||(XA[A]=[]),XA[A].push((function(){e[g]=YA[A],++t===i.length&&C(e)})))})),0===i.length&&C(e)}function UA(A){switch(A){case 1:return 0;case 2:return 1;case 4:return 2;case 8:return 3;default:throw new TypeError("Unknown type size: "+A)}}var kA=void 0;function DA(A){for(var g="",I=A;p[I];)g+=kA[p[I++]];return g}var xA=void 0;function TA(A){throw new xA(A)}function zA(A,g,I){if(I=I||{},!("argPackAdvance"in g))throw new TypeError("registerType registeredInstance requires argPackAdvance");var C=g.name;if(A||TA('type "'+C+'" must have a positive integer typeid pointer'),YA.hasOwnProperty(A)){if(I.ignoreDuplicateRegistrations)return;TA("Cannot register type '"+C+"' twice")}if(YA[A]=g,delete KA[A],XA.hasOwnProperty(A)){var e=XA[A];delete XA[A],e.forEach((function(A){A()}))}}function PA(A){if(!(this instanceof Cg))return!1;if(!(A instanceof Cg))return!1;for(var g=this.$$.ptrType.registeredClass,I=this.$$.ptr,C=A.$$.ptrType.registeredClass,e=A.$$.ptr;g.baseClass;)I=g.upcast(I),g=g.baseClass;for(;C.baseClass;)e=C.upcast(e),C=C.baseClass;return g===C&&I===e}function OA(A){TA(A.$$.ptrType.registeredClass.name+" instance already deleted")}function jA(){if(this.$$.ptr||OA(this),this.$$.preservePointerOnDelete)return this.$$.count.value+=1,this;var A,g=Object.create(Object.getPrototypeOf(this),{$$:{value:(A=this.$$,{count:A.count,deleteScheduled:A.deleteScheduled,preservePointerOnDelete:A.preservePointerOnDelete,ptr:A.ptr,ptrType:A.ptrType,smartPtr:A.smartPtr,smartPtrType:A.smartPtrType})}});return g.$$.count.value+=1,g.$$.deleteScheduled=!1,g}function qA(){var A;this.$$.ptr||OA(this),this.$$.deleteScheduled&&!this.$$.preservePointerOnDelete&&TA("Object already scheduled for deletion"),this.$$.count.value-=1,0===this.$$.count.value&&((A=this.$$).smartPtr?A.smartPtrType.rawDestructor(A.smartPtr):A.ptrType.registeredClass.rawDestructor(A.ptr)),this.$$.preservePointerOnDelete||(this.$$.smartPtr=void 0,this.$$.ptr=void 0)}function _A(){return!this.$$.ptr}var $A=void 0,Ag=[];function gg(){for(;Ag.length;){var A=Ag.pop();A.$$.deleteScheduled=!1,A.delete()}}function Ig(){return this.$$.ptr||OA(this),this.$$.deleteScheduled&&!this.$$.preservePointerOnDelete&&TA("Object already scheduled for deletion"),Ag.push(this),1===Ag.length&&$A&&$A(gg),this.$$.deleteScheduled=!0,this}function Cg(){}var eg={};function ig(A,g,I){if(void 0===A[g].overloadTable){var C=A[g];A[g]=function(){return A[g].overloadTable.hasOwnProperty(arguments.length)||TA("Function '"+I+"' called with an invalid number of arguments ("+arguments.length+") - expects one of ("+A[g].overloadTable+")!"),A[g].overloadTable[arguments.length].apply(this,arguments)},A[g].overloadTable=[],A[g].overloadTable[C.argCount]=C}}function tg(A,g,I,C,e,i,t,o){this.name=A,this.constructor=g,this.instancePrototype=I,this.rawDestructor=C,this.baseClass=e,this.getActualType=i,this.upcast=t,this.downcast=o,this.pureVirtualFunctions=[]}function og(A,g,I){for(;g!==I;)g.upcast||TA("Expected null or instance of "+I.name+", got an instance of "+g.name),A=g.upcast(A),g=g.baseClass;return A}function ag(A,g){if(null===g)return this.isReference&&TA("null is not a valid "+this.name),0;g.$$||TA('Cannot pass "'+Yg(g)+'" as a '+this.name),g.$$.ptr||TA("Cannot pass deleted object as a pointer of type "+this.name);var I=g.$$.ptrType.registeredClass;return og(g.$$.ptr,I,this.registeredClass)}function Bg(A,g){var I;if(null===g)return this.isReference&&TA("null is not a valid "+this.name),this.isSmartPointer?(I=this.rawConstructor(),null!==A&&A.push(this.rawDestructor,I),I):0;g.$$||TA('Cannot pass "'+Yg(g)+'" as a '+this.name),g.$$.ptr||TA("Cannot pass deleted object as a pointer of type "+this.name),!this.isConst&&g.$$.ptrType.isConst&&TA("Cannot convert argument of type "+(g.$$.smartPtrType?g.$$.smartPtrType.name:g.$$.ptrType.name)+" to parameter type "+this.name);var C=g.$$.ptrType.registeredClass;if(I=og(g.$$.ptr,C,this.registeredClass),this.isSmartPointer)switch(void 0===g.$$.smartPtr&&TA("Passing raw pointer to smart pointer is illegal"),this.sharingPolicy){case 0:g.$$.smartPtrType===this?I=g.$$.smartPtr:TA("Cannot convert argument of type "+(g.$$.smartPtrType?g.$$.smartPtrType.name:g.$$.ptrType.name)+" to parameter type "+this.name);break;case 1:I=g.$$.smartPtr;break;case 2:if(g.$$.smartPtrType===this)I=g.$$.smartPtr;else{var e=g.clone();I=this.rawShare(I,Xg((function(){e.delete()}))),null!==A&&A.push(this.rawDestructor,I)}break;default:TA("Unsupporting sharing policy")}return I}function sg(A,g){if(null===g)return this.isReference&&TA("null is not a valid "+this.name),0;g.$$||TA('Cannot pass "'+Yg(g)+'" as a '+this.name),g.$$.ptr||TA("Cannot pass deleted object as a pointer of type "+this.name),g.$$.ptrType.isConst&&TA("Cannot convert argument of type "+g.$$.ptrType.name+" to parameter type "+this.name);var I=g.$$.ptrType.registeredClass;return og(g.$$.ptr,I,this.registeredClass)}function lg(A){return this.rawGetPointee&&(A=this.rawGetPointee(A)),A}function ng(A){this.rawDestructor&&this.rawDestructor(A)}function dg(A){null!==A&&A.delete()}function rg(A,g,I){if(g===I)return A;if(void 0===I.baseClass)return null;var C=rg(A,g,I.baseClass);return null===C?null:I.downcast(C)}function cg(){return Object.keys(mg).length}function hg(){var A=[];for(var g in mg)mg.hasOwnProperty(g)&&A.push(mg[g]);return A}function Gg(A){$A=A,Ag.length&&$A&&$A(gg)}var mg={};function bg(A,g){return g=function(A,g){for(void 0===g&&TA("ptr should not be undefined");A.baseClass;)g=A.upcast(g),A=A.baseClass;return g}(A,g),mg[g]}function Zg(A,g){return g.ptrType&&g.ptr||LA("makeClassHandle requires ptr and ptrType"),!!g.smartPtrType!==!!g.smartPtr&&LA("Both smartPtrType and smartPtr must be specified"),g.count={value:1},Object.create(A,{$$:{value:g}})}function ug(A){var g=this.getPointee(A);if(!g)return this.destructor(A),null;var I=bg(this.registeredClass,g);if(void 0!==I){if(0===I.$$.count.value)return I.$$.ptr=g,I.$$.smartPtr=A,I.clone();var C=I.clone();return this.destructor(A),C}function e(){return this.isSmartPointer?Zg(this.registeredClass.instancePrototype,{ptrType:this.pointeeType,ptr:g,smartPtrType:this,smartPtr:A}):Zg(this.registeredClass.instancePrototype,{ptrType:this,ptr:A})}var i,t=this.registeredClass.getActualType(g),o=eg[t];if(!o)return e.call(this);i=this.isConst?o.constPointerType:o.pointerType;var a=rg(g,this.registeredClass,i.registeredClass);return null===a?e.call(this):this.isSmartPointer?Zg(i.registeredClass.instancePrototype,{ptrType:i,ptr:a,smartPtrType:this,smartPtr:A}):Zg(i.registeredClass.instancePrototype,{ptrType:i,ptr:a})}function yg(A,g,I,C,e,i,t,o,a,B,s){this.name=A,this.registeredClass=g,this.isReference=I,this.isConst=C,this.isSmartPointer=e,this.pointeeType=i,this.sharingPolicy=t,this.rawGetPointee=o,this.rawConstructor=a,this.rawShare=B,this.rawDestructor=s,e||void 0!==g.baseClass?this.toWireType=Bg:C?(this.toWireType=ag,this.destructorFunction=null):(this.toWireType=sg,this.destructorFunction=null)}function Eg(A,g){var I;if(A=DA(A),void 0!==Module["FUNCTION_TABLE_"+A])I=Module["FUNCTION_TABLE_"+A][g];else if("undefined"!=typeof FUNCTION_TABLE)I=FUNCTION_TABLE[g];else{var C=Module.asm["dynCall_"+A];void 0===C&&void 0===(C=Module.asm["dynCall_"+A.replace(/f/g,"d")])&&TA("No dynCall invoker for signature: "+A),I=function(I){for(var C=[],e=1;e<A.length;++e)C.push("a"+e);var i="return function dynCall_"+A+"_"+g+"("+C.join(", ")+") {\n";return i+="    return dynCall(rawFunction"+(C.length?", ":"")+C.join(", ")+");\n",i+="};\n",new Function("dynCall","rawFunction",i)(I,g)}(C)}return"function"!=typeof I&&TA("unknown function pointer with signature "+A+": "+g),I}var Qg=void 0;function Vg(A){var g=_g(A),I=DA(g);return AI(g),I}function Rg(A,g){var I=[],C={};throw g.forEach((function A(g){C[g]||YA[g]||(KA[g]?KA[g].forEach(A):(I.push(g),C[g]=!0))})),new Qg(A+": "+I.map(Vg).join([", "]))}function Wg(A,g){for(var I=[],C=0;C<A;C++)I.push(w[(g>>2)+C]);return I}function pg(A,g,I,C,e){var i=g.length;i<2&&TA("argTypes array size mismatch! Must at least get return value and 'this' types!");for(var t=null!==g[1]&&null!==I,o=!1,a=1;a<g.length;++a)if(null!==g[a]&&void 0===g[a].destructorFunction){o=!0;break}var B="void"!==g[0].name,s="",l="";for(a=0;a<i-2;++a)s+=(0!==a?", ":"")+"arg"+a,l+=(0!==a?", ":"")+"arg"+a+"Wired";var n="return function "+NA(A)+"("+s+") {\nif (arguments.length !== "+(i-2)+") {\nthrowBindingError('function "+A+" called with ' + arguments.length + ' arguments, expected "+(i-2)+" args!');\n}\n";o&&(n+="var destructors = [];\n");var d=o?"destructors":"null",r=["throwBindingError","invoker","fn","runDestructors","retType","classParam"],c=[TA,C,e,wA,g[0],g[1]];t&&(n+="var thisWired = classParam.toWireType("+d+", this);\n");for(a=0;a<i-2;++a)n+="var arg"+a+"Wired = argType"+a+".toWireType("+d+", arg"+a+"); // "+g[a+2].name+"\n",r.push("argType"+a),c.push(g[a+2]);if(t&&(l="thisWired"+(l.length>0?", ":"")+l),n+=(B?"var rv = ":"")+"invoker(fn"+(l.length>0?", ":"")+l+");\n",o)n+="runDestructors(destructors);\n";else for(a=t?1:2;a<g.length;++a){var h=1===a?"thisWired":"arg"+(a-2)+"Wired";null!==g[a].destructorFunction&&(n+=h+"_dtor("+h+"); // "+g[a].name+"\n",r.push(h+"_dtor"),c.push(g[a].destructorFunction))}return B&&(n+="var ret = retType.fromWireType(rv);\nreturn ret;\n"),n+="}\n",r.push(n),function(A,g){if(!(A instanceof Function))throw new TypeError("new_ called with constructor type "+typeof A+" which is not a function");var I=vA(A.name||"unknownFunctionName",(function(){}));I.prototype=A.prototype;var C=new I,e=A.apply(C,g);return e instanceof Object?e:C}(Function,r).apply(null,c)}var Sg=[],Fg=[{},{value:void 0},{value:null},{value:!0},{value:!1}];function wg(){for(var A=0,g=5;g<Fg.length;++g)void 0!==Fg[g]&&++A;return A}function Hg(){for(var A=5;A<Fg.length;++A)if(void 0!==Fg[A])return Fg[A];return null}function Xg(A){switch(A){case void 0:return 1;case null:return 2;case!0:return 3;case!1:return 4;default:var g=Sg.length?Sg.pop():Fg.length;return Fg[g]={refcount:1,value:A},g}}function Yg(A){if(null===A)return"null";var g=typeof A;return"object"===g||"array"===g||"function"===g?A.toString():""+A}function Kg(A,g){switch(g){case 2:return function(A){return this.fromWireType(X[A>>2])};case 3:return function(A){return this.fromWireType(Y[A>>3])};default:throw new TypeError("Unknown float type: "+A)}}function Ng(A,g,I){switch(g){case 0:return I?function(A){return W[A]}:function(A){return p[A]};case 1:return I?function(A){return S[A>>1]}:function(A){return F[A>>1]};case 2:return I?function(A){return w[A>>2]}:function(A){return H[A>>2]};default:throw new TypeError("Unknown integer type: "+A)}}var vg=K+=16;var Jg={};var fg=K+=16;K+=48,G(Tg("GMT"),"i8",2);var Lg=K,Mg=K+=16,Ug=K+=16;function kg(){if(!kg.called){kg.called=!0,w[Ug>>2]=60*(new Date).getTimezoneOffset();var A=new Date(2e3,0,1),g=new Date(2e3,6,1);w[Mg>>2]=Number(A.getTimezoneOffset()!=g.getTimezoneOffset());var I=t(A),C=t(g),e=G(Tg(I),"i8",0),i=G(Tg(C),"i8",0);g.getTimezoneOffset()<A.getTimezoneOffset()?(w[Lg>>2]=e,w[Lg+4>>2]=i):(w[Lg>>2]=i,w[Lg+4>>2]=e)}function t(A){var g=A.toTimeString().match(/\(([A-Za-z ]+)\)$/);return g?g[1]:"GMT"}}K+=16;var Dg={};var xg=1;function Tg(A,g,I){var C=I>0?I:Q(A)+1,e=new Array(C),i=y(A,e,0,e.length);return g&&(e.length=i),e}WA.staticInit(),_.unshift((function(){Module.noFSInit||WA.init.initialized||WA.init()})),$.push((function(){WA.ignorePermissions=!1})),AA.push((function(){WA.quit()})),_.unshift((function(){})),AA.push((function(){})),fA=Module.InternalError=JA(Error,"InternalError"),function(){for(var A=new Array(256),g=0;g<256;++g)A[g]=String.fromCharCode(g);kA=A}(),xA=Module.BindingError=JA(Error,"BindingError"),Cg.prototype.isAliasOf=PA,Cg.prototype.clone=jA,Cg.prototype.delete=qA,Cg.prototype.isDeleted=_A,Cg.prototype.deleteLater=Ig,yg.prototype.getPointee=lg,yg.prototype.destructor=ng,yg.prototype.argPackAdvance=8,yg.prototype.readValueFromPointer=HA,yg.prototype.deleteObject=dg,yg.prototype.fromWireType=ug,Module.getInheritedInstanceCount=cg,Module.getLiveInheritedInstances=hg,Module.flushPendingDeletes=gg,Module.setDelayFunction=Gg,Qg=Module.UnboundTypeError=JA(Error,"UnboundTypeError"),Module.count_emval_handles=wg,Module.get_first_emval=Hg,function A(g){var I,C;A.called?(C=w[vg>>2],I=w[C>>2]):(A.called=!0,Jg.USER=Jg.LOGNAME="web_user",Jg.PATH="/",Jg.PWD="/",Jg.HOME="/home/web_user",Jg.LANG="C.UTF-8",Jg._=Module.thisProgram,I=s(1024),C=s(256),w[C>>2]=I,w[vg>>2]=C);var e=[],i=0;for(var t in g)if("string"==typeof g[t]){var o=t+"="+g[t];e.push(o),i+=o.length}if(i>1024)throw new Error("Environment size exceeded TOTAL_ENV_SIZE!");for(var a=0;a<e.length;a++){CA(o=e[a],I),w[C+4*a>>2]=I,I+=o.length+1}w[C+4*e.length>>2]=0}(Jg),L=s(4),v=J=n(K),f=n(v+P),w[L>>2]=f,N=!0,Module.wasmTableSize=316,Module.wasmMaxTableSize=316,Module.asmGlobalArg={},Module.asmLibraryArg={abort:tI,enlargeMemory:z,getTotalMemory:function(){return O},abortOnCannotGrowMemory:function(){tI("Cannot enlarge memory arrays. Either (1) compile with  -s TOTAL_MEMORY=X  with X higher than the current value "+O+", (2) compile with  -s ALLOW_MEMORY_GROWTH=1  which allows increasing the size at runtime, or (3) if you want malloc to return NULL (0) instead of this abort, compile with  -s ABORTING_MALLOC=0 ")},invoke_i:function(A){try{return Module.dynCall_i(A)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_ii:function(A,g){try{return Module.dynCall_ii(A,g)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_iii:function(A,g,I){try{return Module.dynCall_iii(A,g,I)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_iiii:function(A,g,I,C){try{return Module.dynCall_iiii(A,g,I,C)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_iiiii:function(A,g,I,C,e){try{return Module.dynCall_iiiii(A,g,I,C,e)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_iiiiiii:function(A,g,I,C,e,i,t){try{return Module.dynCall_iiiiiii(A,g,I,C,e,i,t)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_iiiiiiiiii:function(A,g,I,C,e,i,t,o,a,B){try{return Module.dynCall_iiiiiiiiii(A,g,I,C,e,i,t,o,a,B)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_iiiiiijii:function(A,g,I,C,e,i,t,o,a,B){try{return Module.dynCall_iiiiiijii(A,g,I,C,e,i,t,o,a,B)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_ijj:function(A,g,I,C,e){try{return Module.dynCall_ijj(A,g,I,C,e)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_ji:function(A,g){try{return Module.dynCall_ji(A,g)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_v:function(A){try{Module.dynCall_v(A)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_vi:function(A,g){try{Module.dynCall_vi(A,g)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_vii:function(A,g,I){try{Module.dynCall_vii(A,g,I)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_viii:function(A,g,I,C){try{Module.dynCall_viii(A,g,I,C)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_viiii:function(A,g,I,C,e){try{Module.dynCall_viiii(A,g,I,C,e)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_viiiii:function(A,g,I,C,e,i){try{Module.dynCall_viiiii(A,g,I,C,e,i)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_viiiiii:function(A,g,I,C,e,i,t){try{Module.dynCall_viiiiii(A,g,I,C,e,i,t)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_viiiiiiiii:function(A,g,I,C,e,i,t,o,a,B){try{Module.dynCall_viiiiiiiii(A,g,I,C,e,i,t,o,a,B)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_viiiiiiiiii:function(A,g,I,C,e,i,t,o,a,B,s){try{Module.dynCall_viiiiiiiiii(A,g,I,C,e,i,t,o,a,B,s)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_vij:function(A,g,I,C){try{Module.dynCall_vij(A,g,I,C)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},invoke_viji:function(A,g,I,C,e){try{Module.dynCall_viji(A,g,I,C,e)}catch(A){if("number"!=typeof A&&"longjmp"!==A)throw A;Module.setThrew(1,0)}},___cxa_allocate_exception:function(A){return gI(A)},___cxa_begin_catch:function(A){var g=cA.infos[A];return g&&!g.caught&&(g.caught=!0,rA.uncaught_exception--),g&&(g.rethrown=!1),cA.caught.push(A),cA.addRef(cA.deAdjust(A)),A},___cxa_end_catch:function(){Module.setThrew(0);var A=cA.caught.pop();A&&(cA.decRef(cA.deAdjust(A)),cA.last=0)},___cxa_find_matching_catch_2:function(){return GA.apply(null,arguments)},___cxa_find_matching_catch_3:function(){return GA.apply(null,arguments)},___cxa_find_matching_catch_4:function(){return GA.apply(null,arguments)},___cxa_free_exception:hA,___cxa_throw:function(A,g,I){throw cA.infos[A]={ptr:A,adjusted:A,type:g,destructor:I,refcount:0,caught:!1,rethrown:!1},cA.last=A,"uncaught_exception"in rA?rA.uncaught_exception++:rA.uncaught_exception=1,A},___lock:function(){},___map_file:function(A,g){return bA(mA.EPERM),-1},___resumeException:function(A){throw cA.last||(cA.last=A),A},___setErrNo:bA,___syscall140:function(A,g){pA.varargs=g;try{var I=pA.getStreamFromFD(),C=(pA.get(),pA.get()),e=pA.get(),i=pA.get(),t=C;return WA.llseek(I,t,i),w[e>>2]=I.position,I.getdents&&0===t&&0===i&&(I.getdents=null),0}catch(A){return void 0!==WA&&A instanceof WA.ErrnoError||tI(A),-A.errno}},___syscall145:function(A,g){pA.varargs=g;try{var I=pA.getStreamFromFD(),C=pA.get(),e=pA.get();return pA.doReadv(I,C,e)}catch(A){return void 0!==WA&&A instanceof WA.ErrnoError||tI(A),-A.errno}},___syscall146:function(A,g){pA.varargs=g;try{var I=pA.getStreamFromFD(),C=pA.get(),e=pA.get();return pA.doWritev(I,C,e)}catch(A){return void 0!==WA&&A instanceof WA.ErrnoError||tI(A),-A.errno}},___syscall183:function(A,g){pA.varargs=g;try{var I=pA.get(),C=pA.get();if(0===C)return-mA.EINVAL;var e=WA.cwd();return C<Q(e)+1?-mA.ERANGE:(E(e,I,C),I)}catch(A){return void 0!==WA&&A instanceof WA.ErrnoError||tI(A),-A.errno}},___syscall198:function(A,g){pA.varargs=g;try{var I=pA.getStr(),C=pA.get(),e=pA.get();return WA.chown(I,C,e),0}catch(A){return void 0!==WA&&A instanceof WA.ErrnoError||tI(A),-A.errno}},___syscall20:function(A,g){pA.varargs=g;try{return SA}catch(A){return void 0!==WA&&A instanceof WA.ErrnoError||tI(A),-A.errno}},___syscall6:function(A,g){pA.varargs=g;try{var I=pA.getStreamFromFD();return WA.close(I),0}catch(A){return void 0!==WA&&A instanceof WA.ErrnoError||tI(A),-A.errno}},___syscall60:function(A,g){pA.varargs=g;try{var I=pA.get(),C=pA.umask;return pA.umask=I,C}catch(A){return void 0!==WA&&A instanceof WA.ErrnoError||tI(A),-A.errno}},___syscall83:function(A,g){pA.varargs=g;try{var I=pA.getStr(),C=pA.getStr();return WA.symlink(I,C),0}catch(A){return void 0!==WA&&A instanceof WA.ErrnoError||tI(A),-A.errno}},___syscall91:function(A,g){pA.varargs=g;try{var I=pA.get(),C=pA.get(),e=pA.mappings[I];if(!e)return 0;if(C===e.len){var i=WA.getStream(e.fd);pA.doMsync(I,i,C,e.flags),WA.munmap(i),pA.mappings[I]=null,e.allocated&&AI(e.malloc)}return 0}catch(A){return void 0!==WA&&A instanceof WA.ErrnoError||tI(A),-A.errno}},___unlock:function(){},__embind_finalize_value_object:function(A){var g=FA[A];delete FA[A];var I=g.rawConstructor,C=g.rawDestructor,e=g.fields;MA([A],e.map((function(A){return A.getterReturnType})).concat(e.map((function(A){return A.setterArgumentType}))),(function(A){var i={};return e.forEach((function(g,I){var C=g.fieldName,t=A[I],o=g.getter,a=g.getterContext,B=A[I+e.length],s=g.setter,l=g.setterContext;i[C]={read:function(A){return t.fromWireType(o(a,A))},write:function(A,g){var I=[];s(l,A,B.toWireType(I,g)),wA(I)}}})),[{name:g.name,fromWireType:function(A){var g={};for(var I in i)g[I]=i[I].read(A);return C(A),g},toWireType:function(A,g){for(var e in i)if(!(e in g))throw new TypeError("Missing field");var t=I();for(e in i)i[e].write(t,g[e]);return null!==A&&A.push(C,t),t},argPackAdvance:8,readValueFromPointer:HA,destructorFunction:C}]}))},__embind_register_bool:function(A,g,I,C,e){var i=UA(I);zA(A,{name:g=DA(g),fromWireType:function(A){return!!A},toWireType:function(A,g){return g?C:e},argPackAdvance:8,readValueFromPointer:function(A){var C;if(1===I)C=W;else if(2===I)C=S;else{if(4!==I)throw new TypeError("Unknown boolean type size: "+g);C=w}return this.fromWireType(C[A>>i])},destructorFunction:null})},__embind_register_class:function(A,g,I,C,e,i,t,o,a,B,s,l,n){s=DA(s),i=Eg(e,i),o&&(o=Eg(t,o)),B&&(B=Eg(a,B)),n=Eg(l,n);var d=NA(s);!function(A,g,I){Module.hasOwnProperty(A)?((void 0===I||void 0!==Module[A].overloadTable&&void 0!==Module[A].overloadTable[I])&&TA("Cannot register public name '"+A+"' twice"),ig(Module,A,A),Module.hasOwnProperty(I)&&TA("Cannot register multiple overloads of a function with the same number of arguments ("+I+")!"),Module[A].overloadTable[I]=g):(Module[A]=g,void 0!==I&&(Module[A].numArguments=I))}(d,(function(){Rg("Cannot construct "+s+" due to unbound types",[C])})),MA([A,g,I],C?[C]:[],(function(g){var I,e;g=g[0],e=C?(I=g.registeredClass).instancePrototype:Cg.prototype;var t=vA(d,(function(){if(Object.getPrototypeOf(this)!==a)throw new xA("Use 'new' to construct "+s);if(void 0===l.constructor_body)throw new xA(s+" has no accessible constructor");var A=l.constructor_body[arguments.length];if(void 0===A)throw new xA("Tried to invoke ctor of "+s+" with invalid number of parameters ("+arguments.length+") - expected ("+Object.keys(l.constructor_body).toString()+") parameters instead!");return A.apply(this,arguments)})),a=Object.create(e,{constructor:{value:t}});t.prototype=a;var l=new tg(s,t,a,n,I,i,o,B),r=new yg(s,l,!0,!1,!1),c=new yg(s+"*",l,!1,!1,!1),h=new yg(s+" const*",l,!1,!0,!1);return eg[A]={pointerType:c,constPointerType:h},function(A,g,I){Module.hasOwnProperty(A)||LA("Replacing nonexistant public symbol"),void 0!==Module[A].overloadTable&&void 0!==I?Module[A].overloadTable[I]=g:(Module[A]=g,Module[A].argCount=I)}(d,t),[r,c,h]}))},__embind_register_class_constructor:function(A,g,I,C,e,i){var t=Wg(g,I);e=Eg(C,e),MA([],[A],(function(A){var I="constructor "+(A=A[0]).name;if(void 0===A.registeredClass.constructor_body&&(A.registeredClass.constructor_body=[]),void 0!==A.registeredClass.constructor_body[g-1])throw new xA("Cannot register multiple constructors with identical number of parameters ("+(g-1)+") for class '"+A.name+"'! Overload resolution is currently only performed using the parameter count, not actual type info!");return A.registeredClass.constructor_body[g-1]=function(){Rg("Cannot construct "+A.name+" due to unbound types",t)},MA([],t,(function(C){return A.registeredClass.constructor_body[g-1]=function(){arguments.length!==g-1&&TA(I+" called with "+arguments.length+" arguments, expected "+(g-1));var A=[],t=new Array(g);t[0]=i;for(var o=1;o<g;++o)t[o]=C[o].toWireType(A,arguments[o-1]);var a=e.apply(null,t);return wA(A),C[0].fromWireType(a)},[]})),[]}))},__embind_register_class_function:function(A,g,I,C,e,i,t,o){var a=Wg(I,C);g=DA(g),i=Eg(e,i),MA([],[A],(function(A){var C=(A=A[0]).name+"."+g;function e(){Rg("Cannot call "+C+" due to unbound types",a)}o&&A.registeredClass.pureVirtualFunctions.push(g);var B=A.registeredClass.instancePrototype,s=B[g];return void 0===s||void 0===s.overloadTable&&s.className!==A.name&&s.argCount===I-2?(e.argCount=I-2,e.className=A.name,B[g]=e):(ig(B,g,C),B[g].overloadTable[I-2]=e),MA([],a,(function(e){var o=pg(C,e,A,i,t);return void 0===B[g].overloadTable?(o.argCount=I-2,B[g]=o):B[g].overloadTable[I-2]=o,[]})),[]}))},__embind_register_emval:function(A,g){zA(A,{name:g=DA(g),fromWireType:function(A){var g=Fg[A].value;return function(A){A>4&&0==--Fg[A].refcount&&(Fg[A]=void 0,Sg.push(A))}(A),g},toWireType:function(A,g){return Xg(g)},argPackAdvance:8,readValueFromPointer:HA,destructorFunction:null})},__embind_register_float:function(A,g,I){var C=UA(I);zA(A,{name:g=DA(g),fromWireType:function(A){return A},toWireType:function(A,g){if("number"!=typeof g&&"boolean"!=typeof g)throw new TypeError('Cannot convert "'+Yg(g)+'" to '+this.name);return g},argPackAdvance:8,readValueFromPointer:Kg(g,C),destructorFunction:null})},__embind_register_integer:function(A,g,I,C,e){g=DA(g),-1===e&&(e=4294967295);var i=UA(I),t=function(A){return A};if(0===C){var o=32-8*I;t=function(A){return A<<o>>>o}}var a=-1!=g.indexOf("unsigned");zA(A,{name:g,fromWireType:t,toWireType:function(A,I){if("number"!=typeof I&&"boolean"!=typeof I)throw new TypeError('Cannot convert "'+Yg(I)+'" to '+this.name);if(I<C||I>e)throw new TypeError('Passing a number "'+Yg(I)+'" from JS side to C/C++ side to an argument of type "'+g+'", which is outside the valid range ['+C+", "+e+"]!");return a?I>>>0:0|I},argPackAdvance:8,readValueFromPointer:Ng(g,i,0!==C),destructorFunction:null})},__embind_register_memory_view:function(A,g,I){var C=[Int8Array,Uint8Array,Int16Array,Uint16Array,Int32Array,Uint32Array,Float32Array,Float64Array][g];function e(A){var g=H,I=g[A>>=2],e=g[A+1];return new C(g.buffer,e,I)}zA(A,{name:I=DA(I),fromWireType:e,argPackAdvance:8,readValueFromPointer:e},{ignoreDuplicateRegistrations:!0})},__embind_register_std_string:function(A,g){zA(A,{name:g=DA(g),fromWireType:function(A){for(var g=H[A>>2],I=new Array(g),C=0;C<g;++C)I[C]=String.fromCharCode(p[A+4+C]);return AI(A),I.join("")},toWireType:function(A,g){function I(A,g){return A[g]}var C;g instanceof ArrayBuffer&&(g=new Uint8Array(g)),g instanceof Uint8Array||g instanceof Uint8ClampedArray||g instanceof Int8Array?C=I:"string"==typeof g?C=function(A,g){return A.charCodeAt(g)}:TA("Cannot pass non-string to std::string");var e=g.length,i=gI(4+e);H[i>>2]=e;for(var t=0;t<e;++t){var o=C(g,t);o>255&&(AI(i),TA("String has UTF-16 code units that do not fit in 8 bits")),p[i+4+t]=o}return null!==A&&A.push(AI,i),i},argPackAdvance:8,readValueFromPointer:HA,destructorFunction:function(A){AI(A)}})},__embind_register_std_wstring:function(A,g,I){var C,e;I=DA(I),2===g?(C=function(){return F},e=1):4===g&&(C=function(){return H},e=2),zA(A,{name:I,fromWireType:function(A){for(var g=C(),I=H[A>>2],i=new Array(I),t=A+4>>e,o=0;o<I;++o)i[o]=String.fromCharCode(g[t+o]);return AI(A),i.join("")},toWireType:function(A,I){var i=C(),t=I.length,o=gI(4+t*g);H[o>>2]=t;for(var a=o+4>>e,B=0;B<t;++B)i[a+B]=I.charCodeAt(B);return null!==A&&A.push(AI,o),o},argPackAdvance:8,readValueFromPointer:HA,destructorFunction:function(A){AI(A)}})},__embind_register_value_object:function(A,g,I,C,e,i){FA[A]={name:DA(g),rawConstructor:Eg(I,C),rawDestructor:Eg(e,i),fields:[]}},__embind_register_value_object_field:function(A,g,I,C,e,i,t,o,a,B){FA[A].fields.push({fieldName:DA(g),getterReturnType:I,getter:Eg(C,e),getterContext:i,setterArgumentType:t,setter:Eg(o,a),setterContext:B})},__embind_register_void:function(A,g){zA(A,{isVoid:!0,name:g=DA(g),argPackAdvance:0,fromWireType:function(){},toWireType:function(A,g){}})},_abort:function(){Module.abort()},_emscripten_memcpy_big:function(A,g,I){return p.set(p.subarray(g,g+I),A),A},_getenv:function A(g){return 0===g?0:(g=m(g),Jg.hasOwnProperty(g)?(A.ret&&AI(A.ret),A.ret=(I=Jg[g],C=Q(I)+1,(e=gI(C))&&y(I,W,e,C),e)):0);var I,C,e},_getgrnam:function(){Module.printErr("missing function: getgrnam"),tI(-1)},_getpwnam:function(){throw"getpwnam: TODO"},_jsClose:function(){return i.close.apply(null,arguments)},_jsCreate:function(A){return i.create.call(null,V(A))},_jsOpen:function(A){return i.open.call(null,V(A))},_jsRead:function(){return i.read.apply(null,arguments)},_jsSeek:function(A,g,I){return i.seek.call(null,A,g,u(I))},_jsTell:function(){return i.tell.apply(null,arguments)},_jsWrite:function(){return i.write.apply(null,arguments)},_llvm_eh_typeid_for:function(A){return A},_localtime:function(A){return function(A,g){kg();var I=new Date(1e3*w[A>>2]);w[g>>2]=I.getSeconds(),w[g+4>>2]=I.getMinutes(),w[g+8>>2]=I.getHours(),w[g+12>>2]=I.getDate(),w[g+16>>2]=I.getMonth(),w[g+20>>2]=I.getFullYear()-1900,w[g+24>>2]=I.getDay();var C=new Date(I.getFullYear(),0,1),e=(I.getTime()-C.getTime())/864e5|0;w[g+28>>2]=e,w[g+36>>2]=-60*I.getTimezoneOffset();var i=new Date(2e3,6,1).getTimezoneOffset(),t=C.getTimezoneOffset(),o=0|(i!=t&&I.getTimezoneOffset()==Math.min(t,i));w[g+32>>2]=o;var a=w[Lg+(o?4:0)>>2];return w[g+40>>2]=a,g}(A,fg)},_mktime:function(A){kg();var g=new Date(w[A+20>>2]+1900,w[A+16>>2],w[A+12>>2],w[A+8>>2],w[A+4>>2],w[A>>2],0),I=w[A+32>>2],C=g.getTimezoneOffset(),e=new Date(g.getFullYear(),0,1),i=new Date(2e3,6,1).getTimezoneOffset(),t=e.getTimezoneOffset(),o=Math.min(t,i);if(I<0)w[A+32>>2]=Number(i!=t&&o==C);else if(I>0!=(o==C)){var a=Math.max(t,i),B=I>0?o:a;g.setTime(g.getTime()+6e4*(B-C))}w[A+24>>2]=g.getDay();var s=(g.getTime()-e.getTime())/864e5|0;return w[A+28>>2]=s,g.getTime()/1e3|0},_pthread_getspecific:function(A){return Dg[A]||0},_pthread_key_create:function(A,g){return 0==A?mA.EINVAL:(w[A>>2]=xg,Dg[xg]=0,xg++,0)},_pthread_once:function A(g,I){A.seen||(A.seen={}),g in A.seen||(Module.dynCall_v(I),A.seen[g]=1)},_pthread_setspecific:function(A,g){return A in Dg?(Dg[A]=g,0):mA.EINVAL},_time:function(A){var g=Date.now()/1e3|0;return A&&(w[A>>2]=g),g},DYNAMICTOP_PTR:L,STACKTOP:J};var zg=Module.asm(Module.asmGlobalArg,Module.asmLibraryArg,A);Module.asm=zg;var Pg=Module.__GLOBAL__sub_I_bind_cpp=function(){return Module.asm.__GLOBAL__sub_I_bind_cpp.apply(null,arguments)},Og=Module.__GLOBAL__sub_I_bridge_cpp=function(){return Module.asm.__GLOBAL__sub_I_bridge_cpp.apply(null,arguments)},jg=Module.__GLOBAL__sub_I_crc_cpp=function(){return Module.asm.__GLOBAL__sub_I_crc_cpp.apply(null,arguments)},qg=Module.__GLOBAL__sub_I_global_cpp=function(){return Module.asm.__GLOBAL__sub_I_global_cpp.apply(null,arguments)};Module.___cxa_can_catch=function(){return Module.asm.___cxa_can_catch.apply(null,arguments)},Module.___cxa_is_pointer_type=function(){return Module.asm.___cxa_is_pointer_type.apply(null,arguments)},Module.___errno_location=function(){return Module.asm.___errno_location.apply(null,arguments)};var _g=Module.___getTypeName=function(){return Module.asm.___getTypeName.apply(null,arguments)},$g=Module._emscripten_replace_memory=function(){return Module.asm._emscripten_replace_memory.apply(null,arguments)},AI=Module._free=function(){return Module.asm._free.apply(null,arguments)},gI=Module._malloc=function(){return Module.asm._malloc.apply(null,arguments)},II=Module.setTempRet0=function(){return Module.asm.setTempRet0.apply(null,arguments)};Module.setThrew=function(){return Module.asm.setThrew.apply(null,arguments)};var CI=Module.stackAlloc=function(){return Module.asm.stackAlloc.apply(null,arguments)};function eI(A){this.name="ExitStatus",this.message="Program terminated with exit("+A+")",this.status=A}function iI(A){function g(){Module.calledRun||(Module.calledRun=!0,r||(IA||(IA=!0,j(_)),j($),Module.onRuntimeInitialized&&Module.onRuntimeInitialized(),function(){if(Module.postRun)for("function"==typeof Module.postRun&&(Module.postRun=[Module.postRun]);Module.postRun.length;)A=Module.postRun.shift(),gA.unshift(A);var A;j(gA)}()))}A=A||Module.arguments,aA>0||(!function(){if(Module.preRun)for("function"==typeof Module.preRun&&(Module.preRun=[Module.preRun]);Module.preRun.length;)A=Module.preRun.shift(),q.unshift(A);var A;j(q)}(),aA>0||Module.calledRun||(Module.setStatus?(Module.setStatus("Running..."),setTimeout((function(){setTimeout((function(){Module.setStatus("")}),1),g()}),1)):g()))}function tI(A){throw Module.onAbort&&Module.onAbort(A),void 0!==A?(Module.print(A),Module.printErr(A),A=JSON.stringify(A)):A="",r=!0,"abort("+A+"). Build with -s ASSERTIONS=1 for more info."}if(Module.dynCall_dii=function(){return Module.asm.dynCall_dii.apply(null,arguments)},Module.dynCall_i=function(){return Module.asm.dynCall_i.apply(null,arguments)},Module.dynCall_ii=function(){return Module.asm.dynCall_ii.apply(null,arguments)},Module.dynCall_iii=function(){return Module.asm.dynCall_iii.apply(null,arguments)},Module.dynCall_iiii=function(){return Module.asm.dynCall_iiii.apply(null,arguments)},Module.dynCall_iiiii=function(){return Module.asm.dynCall_iiiii.apply(null,arguments)},Module.dynCall_iiiiii=function(){return Module.asm.dynCall_iiiiii.apply(null,arguments)},Module.dynCall_iiiiiii=function(){return Module.asm.dynCall_iiiiiii.apply(null,arguments)},Module.dynCall_iiiiiiiiii=function(){return Module.asm.dynCall_iiiiiiiiii.apply(null,arguments)},Module.dynCall_iiiiiijii=function(){return Module.asm.dynCall_iiiiiijii.apply(null,arguments)},Module.dynCall_ijj=function(){return Module.asm.dynCall_ijj.apply(null,arguments)},Module.dynCall_ji=function(){return Module.asm.dynCall_ji.apply(null,arguments)},Module.dynCall_v=function(){return Module.asm.dynCall_v.apply(null,arguments)},Module.dynCall_vi=function(){return Module.asm.dynCall_vi.apply(null,arguments)},Module.dynCall_vii=function(){return Module.asm.dynCall_vii.apply(null,arguments)},Module.dynCall_viid=function(){return Module.asm.dynCall_viid.apply(null,arguments)},Module.dynCall_viii=function(){return Module.asm.dynCall_viii.apply(null,arguments)},Module.dynCall_viiii=function(){return Module.asm.dynCall_viiii.apply(null,arguments)},Module.dynCall_viiiii=function(){return Module.asm.dynCall_viiiii.apply(null,arguments)},Module.dynCall_viiiiii=function(){return Module.asm.dynCall_viiiiii.apply(null,arguments)},Module.dynCall_viiiiiiiii=function(){return Module.asm.dynCall_viiiiiiiii.apply(null,arguments)},Module.dynCall_viiiiiiiiii=function(){return Module.asm.dynCall_viiiiiiiiii.apply(null,arguments)},Module.dynCall_vij=function(){return Module.asm.dynCall_vij.apply(null,arguments)},Module.dynCall_viji=function(){return Module.asm.dynCall_viji.apply(null,arguments)},Module.asm=zg,eI.prototype=new Error,eI.prototype.constructor=eI,BA=function A(){Module.calledRun||iI(),Module.calledRun||(BA=A)},Module.run=iI,Module.exit=function(A,g){g&&Module.noExitRuntime&&0===A||(Module.noExitRuntime||(r=!0,J=undefined,j(AA),Module.onExit&&Module.onExit(A)),B&&process.exit(A),Module.quit(A,new eI(A)))},Module.abort=tI,Module.preInit)for("function"==typeof Module.preInit&&(Module.preInit=[Module.preInit]);Module.preInit.length>0;)Module.preInit.pop()();return Module.noExitRuntime=!0,iI(),g};function getExtractor(A){return fetch(new Request(A)).then((A=>{if(A.ok)return A.arrayBuffer();throw new Error("404 Error: File not found.")})).then((A=>unpackBridge.createExtractorFromData(A)))}function extract({resourceId:A,url:g}){return new Promise((function(A,I){if(!unpackBridge)throw new Error("unpackBridge not detected");if(!unpack)throw new Error("unpack not detected");getExtractor(g).then((g=>{A(g.extractAll())}),(A=>{I(A)}))}))}function handleMessage$1(A,g){if("init"==A.type)unpack=initunpack(A.buffer),unpack.onRuntimeInitialized=()=>{g({type:"WASM_LOADED"})};else if("fetch"==A.type)extract(A).then((I=>{returnData(A,I,g)}),(I=>{const C={taskId:A.taskId,type:"ERROR",resourceId:A.resourceId,url:A.url};g(C)}));else if("unpack"==A.type){const{buffer:I}=A;if(!unpackBridge)throw new Error("unpackBridge not detected");if(!unpack)throw new Error("unpack not detected");const C=unpackBridge.createExtractorFromData(I).extractAll();returnData(A,C,g)}}function returnData(A,g,I){const[C,e]=g;if("FAIL"==C.state){return void I({taskId:A.taskId,type:"ERROR",reason:C.reason,msg:C.msg,resourceId:A.resourceId,url:A.url})}const i={taskId:A.taskId,type:"FINISHED",resourceId:A.resourceId,entries:{}},t=[];if(e&&e.files)for(const A of e.files)i.entries[A.fileHeader.name]=A.extract[1],t.push(A.extract[1].buffer);I(i,t)}globalThis.onmessage=function(A){handleMessage$1(A.data,self.postMessage)};let taskCounter=0;class WorkerPool extends EventEmitter{poolSize=Math.max(1,SystemDesc.hardwareConcurrency-1);workers=[];workerTaskCount=[];taskPromiseResolves={};taskQueue=[];availableWorkers=[];terminationTimeouts=[];terminateWorkersWhenFree=!0;terminationLatency=2e3;constructor(A){super(),this.terminateWorkersWhenFree=A}addTask(A,g){return this.addTaskCallback((()=>({taskData:A,transferables:g})))}addTaskCallback(A){taskCounter++;const g=taskCounter;return new Promise((async I=>{this.taskPromiseResolves[g]=I,this.taskQueue.push({taskId:g,dataFactory:A}),this.availableWorkers.length>0?this.consumeTask():this.workers.length<this.poolSize&&(await this.addWorker(),this.consumeTask())}))}async consumeTask(){const A=this.availableWorkers.pop();if(this.workerTaskCount[A]>0)return;if(0==this.taskQueue.length)return void(this.terminateWorkersWhenFree&&this.scheduleWorkerTermination(A));if(-1!=this.terminationTimeouts[A]?(clearTimeout(this.terminationTimeouts[A]),this.terminationTimeouts[A]=-1):this.workers[A]||await this.allocWorker(A),0==this.taskQueue.length)return void(this.terminateWorkersWhenFree&&this.scheduleWorkerTermination(A));const g=this.taskQueue.pop(),{taskData:I,transferables:C}=g.dataFactory(A);I.taskId=g.taskId,this.workerTaskCount[A]++,this.workers[A].postMessage(I,C)}addWorker(){const A=this.workers.length;return this.workers.push(null),this.allocWorker(A)}allocWorker(A){return new Promise((g=>{this.constructWorker().then((I=>{I.onmessage=g=>{if(g.data.taskId in this.taskPromiseResolves){const I=g.data.taskId;if(delete g.data.taskId,this.taskPromiseResolves[I](g.data),delete this.taskPromiseResolves[I],this.workerTaskCount[A]--,this.workerTaskCount[A]>0)return;-1==this.availableWorkers.indexOf(A)&&this.availableWorkers.push(A),this.taskQueue.length>0?this.consumeTask():this.terminateWorkersWhenFree&&this.scheduleWorkerTermination(A)}else g.data.eventName&&(g.data.workerId=A,this.emit(g.data.eventName,g.data))},this.workers[A]=I,this.terminationTimeouts[A]=-1,this.workerTaskCount[A]=0,this.availableWorkers.push(A),g()}))}))}scheduleWorkerTermination(A){this.terminationTimeouts[A]=setTimeout((()=>{this.terminateWorker(A),this.terminationTimeouts[A]=-1}),this.terminationLatency)}terminateWorker(A){this.workers[A].terminate(),this.workers[A]=null}messageWorker(A,g){taskCounter++;const I=taskCounter;return new Promise((C=>{this.taskPromiseResolves[I]=C,g.taskId=I,this.workers[A].postMessage(g)}))}}function decodeBase64(A,g){var I=atob(A);if(g){for(var C=new Uint8Array(I.length),e=0,i=I.length;e<i;++e)C[e]=I.charCodeAt(e);return String.fromCharCode.apply(null,new Uint16Array(C.buffer))}return I}function createURL(A,g,I){var C=void 0===g?null:g,e=decodeBase64(A,void 0!==I&&I),i=e.indexOf("\n",10)+1,t=e.substring(i)+(C?"//# sourceMappingURL="+C:""),o=new Blob([t],{type:"application/javascript"});return URL.createObjectURL(o)}function createBase64WorkerFactory(A,g,I){var C;return function(e){return C=C||createURL(A,g,I),new Worker(C,e)}}var WorkerFactory$2=createBase64WorkerFactory("/* rollup-plugin-web-worker-loader */
var worker_code = (function (exports) {
  'use strict';

  // <!-- prettier-ignore-start -->
  /* eslint-disable require-jsdoc */

  const Module = {};

  const WorkerScope = {};

  //--------------------unpackBridge.js----------------------------
  !(function (t, e) {
    // The following code has been _carefully_ modified by hand.
    // There were various cases for in what context the code might
    // be run, and I removed all but the webworker case.
    // There was code to handle loading in a nodeJS context, that tried to import("fs")
    // WebPack kepts tripping up on that code in its static analysis of the code, so
    // I carefully removed it.
    t.unpackBridge = e(t.fs);
  })(WorkerScope, function (t) {
    return (function (t) {
      var e = {};
      function r(n) {
        if (e[n]) return e[n].exports
        var i = (e[n] = { i: n, l: !1, exports: {} });
        return t[n].call(i.exports, i, i.exports, r), (i.l = !0), i.exports
      }
      return (
        (r.m = t),
        (r.c = e),
        (r.d = function (t, e, n) {
          r.o(t, e) || Object.defineProperty(t, e, { enumerable: !0, get: n });
        }),
        (r.r = function (t) {
          'undefined' != typeof Symbol &&
            Symbol.toStringTag &&
            Object.defineProperty(t, Symbol.toStringTag, { value: 'Module' }),
            Object.defineProperty(t, '__esModule', { value: !0 });
        }),
        (r.t = function (t, e) {
          if ((1 & e && (t = r(t)), 8 & e)) return t
          if (4 & e && 'object' == typeof t && t && t.__esModule) return t
          var n = Object.create(null);
          if ((r.r(n), Object.defineProperty(n, 'default', { enumerable: !0, value: t }), 2 & e && 'string' != typeof t))
            for (var i in t)
              r.d(
                n,
                i,
                function (e) {
                  return t[e]
                }.bind(null, i)
              );
          return n
        }),
        (r.n = function (t) {
          var e =
            t && t.__esModule
              ? function () {
                  return t.default
                }
              : function () {
                  return t
                };
          return r.d(e, 'a', e), e
        }),
        (r.o = function (t, e) {
          return Object.prototype.hasOwnProperty.call(t, e)
        }),
        (r.p = ''),
        r((r.s = 2))
      )
    })([
      function (t, e, r) {
        Object.defineProperty(e, '__esModule', { value: !0 });
        const n = r(1),
          i = {
            0: 'ERAR_SUCCESS',
            10: 'ERAR_END_ARCHIVE',
            11: 'ERAR_NO_MEMORY',
            12: 'ERAR_BAD_DATA',
            13: 'ERAR_BAD_ARCHIVE',
            14: 'ERAR_UNKNOWN_FORMAT',
            15: 'ERAR_EOPEN',
            16: 'ERAR_ECREATE',
            17: 'ERAR_ECLOSE',
            18: 'ERAR_EREAD',
            19: 'ERAR_EWRITE',
            20: 'ERAR_SMALL_BUF',
            21: 'ERAR_UNKNOWN',
            22: 'ERAR_MISSING_PASSWORD',
            23: 'ERAR_EREFERENCE',
            24: 'ERAR_BAD_PASSWORD',
          },
          o = {
            0: 'Success',
            11: 'Not enough memory',
            12: 'Archive header or data are damaged',
            13: 'File is not RAR archive',
            14: 'Unknown archive format',
            15: 'File open error',
            16: 'File create error',
            17: 'File close error',
            18: 'File read error',
            19: 'File write error',
            20: 'Buffer for archive comment is too small, comment truncated',
            21: 'Unknown error',
            22: 'Password for encrypted file or header is not specified',
            23: 'Cannot open file source for reference record',
            24: 'Wrong password is specified',
          };
        class s {
          constructor(t = '') {
  (this._password = t), (this._archive = null);
          }
          getFileList() {
            let t,
              [e, r] = this.openArc(!0);
            if ('SUCCESS' !== e.state) t = [e, null];
            else {
              let e,
                n,
                i = [];
              for (; ([e, n] = this.processNextFile(() => !0)), 'SUCCESS' === e.state; ) i.push(n.fileHeader);
              t = 'ERAR_END_ARCHIVE' !== e.reason ? [e, null] : [{ state: 'SUCCESS' }, { arcHeader: r, fileHeaders: i }];
            }
            return this.closeArc(), t
          }
          extractAll() {
            let t,
              [e, r] = this.openArc(!1);
            if ('SUCCESS' !== e.state) t = [e, null];
            else {
              let e,
                n,
                i = [];
              for (; ([e, n] = this.processNextFile(() => !1)), 'SUCCESS' === e.state; ) i.push(n);
              t = 'ERAR_END_ARCHIVE' !== e.reason ? [e, null] : [{ state: 'SUCCESS' }, { arcHeader: r, files: i }];
            }
            return this.closeArc(), t
          }
          extractFiles(t, e) {
            let r,
              [n, i] = this.openArc(!1, e),
              o = {};
            for (let e = 0; e < t.length; ++e) o[t[e]] = e;
            if ('SUCCESS' !== n.state) r = [n, null];
            else {
              let e,
                n,
                s = Array(t.length).fill(null),
                u = 0;
              for (;;) {
                let r = !1,
                  i = null;
                if (
                  (([e, n] = this.processNextFile((t) => (t in o ? ((i = o[t]), !1) : ((r = !0), !0)))),
                  'SUCCESS' !== e.state)
                )
                  break
                if (!r && ((s[i] = n), ++u === t.length)) {
                  e.reason = 'ERAR_END_ARCHIVE';
                  break
                }
              }
              r = 'ERAR_END_ARCHIVE' !== e.reason ? [e, null] : [{ state: 'SUCCESS' }, { arcHeader: i, files: s }];
            }
            return this.closeArc(), r
          }
          fileCreated(t) {}
          close(t) {
            this._lastFileContent = this.closeFile(t);
          }
          openArc(t, e) {
  (n.Ext.current = this), (this._archive = new unpack.RarArchive());
            let r,
              i = this._archive.open(this._filePath, e || this._password, t);
            return (
              (r =
                0 !== i.state.errCode
                  ? [this.getFailInfo(i.state.errCode, i.state.errType), null]
                  : [
                      { state: 'SUCCESS' },
                      {
                        comment: i.comment,
                        flags: {
                          volume: 0 != (1 & i.flags),
                          lock: 0 != (4 & i.flags),
                          solid: 0 != (8 & i.flags),
                          authInfo: 0 != (32 & i.flags),
                          recoveryRecord: 0 != (64 & i.flags),
                          headerEncrypted: 0 != (128 & i.flags),
                        },
                      },
                    ]),
              (n.Ext.current = null),
              r
            )
          }
          processNextFile(t) {
            let e;
            n.Ext.current = this;
            let r = this._archive.getFileHeader(),
              i = [{ state: 'SUCCESS' }, null];
            if (0 === r.state.errCode) {
              let e = t(r.name);
              this._lastFileContent = null;
              let n = this._archive.readFile(e);
              0 === n.errCode ||
                e ||
                ((i[0] = this.getFailInfo(n.errCode, n.errType)),
                22 === n.errCode ? (n = this._archive.readFile(!0)) : (n.errCode = 0)),
                0 === n.errCode
                  ? (i[1] = this._lastFileContent)
                  : ((r.state.errCode = n.errCode), (r.state.errType = n.errType)),
                (this._lastFileContent = null);
            }
            return (
              (e =
                0 !== r.state.errCode
                  ? [this.getFailInfo(r.state.errCode, r.state.errType), null]
                  : [
                      { state: 'SUCCESS' },
                      {
                        fileHeader: {
                          name: r.name,
                          flags: {
                            encrypted: 0 != (4 & r.flags),
                            solid: 0 != (16 & r.flags),
                            directory: 0 != (32 & r.flags),
                          },
                          packSize: r.packSize,
                          unpSize: r.unpSize,
                          crc: r.crc,
                          time: (function (t) {
                            const e = [5, 6, 5, 5, 4, 7];
                            let r = [];
                            for (let n of e) r.push(t & ((1 << n) - 1)), (t >>= n);
                            let n = (t) => (t < 10 ? '0' + t : '' + t);
                            return (
                              `${1980 + (r = r.reverse())[0]}-${n(r[1])}-${n(r[2])}` +
                              `T${n(r[3])}:${n(r[4])}:${n(2 * r[5])}.000`
                            )
                          })(r.time),
                          unpVer: `${Math.floor(r.unpVer / 10)}.${r.unpVer % 10}`,
                          method: (function (t) {
                            return (
                              { 48: 'Storing', 49: 'Fastest', 50: 'Fast', 51: 'Normal', 52: 'Good', 53: 'Best' }[t] ||
                              'Unknown'
                            )
                          })(r.method),
                        },
                        extract: i,
                      },
                    ]),
              (n.Ext.current = null),
              e
            )
          }
          closeArc() {
  (n.Ext.current = this), this._archive.delete(), (n.Ext.current = null), (this._archive = null);
          }
          getFailInfo(t, e) {
            return { state: 'FAIL', reason: i[t], msg: o[t] }
          }
        }
  (s._current = null), (e.Extractor = s);
      },
      function (t, e, r) {
        Object.defineProperty(e, '__esModule', { value: !0 }), (e.Ext = { current: null });
      },
      function (t, e, r) {
        Object.defineProperty(e, '__esModule', { value: !0 }),
          (function (t) {
            for (var r in t) e.hasOwnProperty(r) || (e[r] = t[r]);
          })(r(3));
        var n = r(1);
        e.Ext = n.Ext;
      },
      function (t, e, r) {
        Object.defineProperty(e, '__esModule', { value: !0 });
        const n = r(4),
          i = r(6)
        ;(e.createExtractorFromData = function (t, e = '') {
          return new n.DataExtractor(t, e)
        }),
          (e.createExtractorFromFile = function (t, e = '', r = '') {
            return new i.FileExtractor(t, e, r)
          });
      },
      function (t, e, r) {
        Object.defineProperty(e, '__esModule', { value: !0 });
        const n = r(5),
          i = r(0);
        e.DataExtractor = class extends i.Extractor {
          constructor(t, e) {
            super(e), (this.dataFiles = {}), (this.dataFileMap = {}), (this.currentFd = 1);
            let r = { file: new n.DataFile(new Uint8Array(t)), fd: this.currentFd++ }
            ;(this._filePath = '_defaultUnrarJS_.rar'),
              (this.dataFiles[this._filePath] = r),
              (this.dataFileMap[r.fd] = this._filePath);
          }
          open(t) {
            let e = this.dataFiles[t];
            return e ? e.fd : 0
          }
          create(t) {
            let e = this.currentFd++;
            return (this.dataFiles[t] = { file: new n.DataFile(), fd: this.currentFd++ }), (this.dataFileMap[e] = t), e
          }
          closeFile(t) {
            let e = this.dataFiles[this.dataFileMap[t]];
            if (!e) return null
            let r = e.file.readAll();
            return (
              1 !== t ? (delete this.dataFiles[this.dataFileMap[t]], delete this.dataFileMap[t]) : e.file.seek(0, 'SET'),
              r
            )
          }
          read(t, e, r) {
            let n = this.dataFiles[this.dataFileMap[t]];
            if (!n) return -1
            let i = n.file.read(r);
            return null === i ? -1 : (unpack.HEAPU8.set(i, e), i.byteLength)
          }
          write(t, e, r) {
            let n = this.dataFiles[this.dataFileMap[t]];
            return !!n && (n.file.write(unpack.HEAPU8.slice(e, e + r)), !0)
          }
          tell(t) {
            let e = this.dataFiles[this.dataFileMap[t]];
            return e ? e.file.tell() : -1
          }
          seek(t, e, r) {
            let n = this.dataFiles[this.dataFileMap[t]];
            return !!n && n.file.seek(e, r)
          }
        };
      },
      function (t, e, r) {
        Object.defineProperty(e, '__esModule', { value: !0 });
        e.DataFile = class {
          constructor(t) {
  (this.buffers = []),
              (this.pos = 0),
              (this.size = 0),
              t && (this.buffers.push(t), (this.size = t.byteLength), (this.pos = 0));
          }
          read(t) {
            if ((this.flatten(), t + this.pos > this.size)) return null
            let e = this.pos;
            return (this.pos += t), this.buffers[0].slice(e, this.pos)
          }
          readAll() {
            return this.flatten(), this.buffers[0]
          }
          write(t) {
            return this.buffers.push(t), (this.size += t.byteLength), (this.pos += t.byteLength), !0
          }
          tell() {
            return this.pos
          }
          seek(t, e) {
            let r = this.pos;
            return (
              'SET' === e ? (r = t) : 'CUR' === e ? (r += t) : (r = this.size - t),
              !(r < 0 || r > this.size || ((this.pos = r), 0))
            )
          }
          flatten() {
            if (this.buffers.length <= 1) return
            let t = new Uint8Array(this.size),
              e = 0;
            for (let r of this.buffers) t.set(r, e), (e += r.byteLength);
            this.buffers = [t];
          }
        };
      },
      function (t, e, r) {
  (function (t) {
          Object.defineProperty(e, '__esModule', { value: !0 });
          const n = r(12),
            i = r(13),
            o = r(0);
          e.FileExtractor = class extends o.Extractor {
            constructor(t, e, r) {
              super(r), (this._filePath = t), (this.fileMap = {}), (this._target = e);
            }
            open(t) {
              let e = n.openSync(t, 'r');
              return (this.fileMap[e] = { size: n.fstatSync(e).size, pos: 0, name: t }), e
            }
            create(t) {
              let e = i.join(this._target, t);
              i.parse(e)
                .dir.split('/')
                .reduce((t, e) => ((t += e + '/'), n.existsSync(t) || n.mkdirSync(t), t), '');
              let r = n.openSync(e, 'w');
              return (this.fileMap[r] = { size: 0, pos: 0, name: t }), r
            }
            closeFile(t) {
              return delete this.fileMap[t], n.closeSync(t), null
            }
            read(e, r, i) {
              let o = this.fileMap[e],
                s = new t(i),
                u = n.readSync(e, s, 0, i, o.pos);
              return unpack.HEAPU8.set(s, r), (o.pos += u), u
            }
            write(e, r, i) {
              let o = this.fileMap[e],
                s = n.writeSync(e, new t(unpack.HEAPU8.subarray(r, r + i)), 0, i);
              return (o.pos += s), (o.size += s), s === i
            }
            tell(t) {
              return this.fileMap[t].pos
            }
            seek(t, e, r) {
              let n = this.fileMap[t],
                i = n.pos;
              return (
                'SET' === r ? (i = 0) : 'END' === r && (i = n.size), !((i += e) < 0 || i > n.size || ((n.pos = i), 0))
              )
            }
          };
        }.call(this, r(7).Buffer));
      },
      function (t, e, r) {
  (function (t) {
          /*!
           * The buffer module from node.js, for the browser.
           *
           * @author   Feross Aboukhadijeh <feross@feross.org> <http://feross.org>
           * @license  MIT
           */
          var n = r(9),
            i = r(10),
            o = r(11);
          function s() {
            return a.TYPED_ARRAY_SUPPORT ? 2147483647 : 1073741823
          }
          function u(t, e) {
            if (s() < e) throw new RangeError('Invalid typed array length')
            return (
              a.TYPED_ARRAY_SUPPORT
                ? ((t = new Uint8Array(e)).__proto__ = a.prototype)
                : (null === t && (t = new a(e)), (t.length = e)),
              t
            )
          }
          function a(t, e, r) {
            if (!(a.TYPED_ARRAY_SUPPORT || this instanceof a)) return new a(t, e, r)
            if ('number' == typeof t) {
              if ('string' == typeof e)
                throw new Error('If encoding is specified then the first argument must be a string')
              return l(this, t)
            }
            return f(this, t, e, r)
          }
          function f(t, e, r, n) {
            if ('number' == typeof e) throw new TypeError('"value" argument must not be a number')
            return 'undefined' != typeof ArrayBuffer && e instanceof ArrayBuffer
              ? (function (t, e, r, n) {
                  if ((e.byteLength, r < 0 || e.byteLength < r)) throw new RangeError("'offset' is out of bounds")
                  if (e.byteLength < r + (n || 0)) throw new RangeError("'length' is out of bounds")
                  e =
                    void 0 === r && void 0 === n
                      ? new Uint8Array(e)
                      : void 0 === n
                      ? new Uint8Array(e, r)
                      : new Uint8Array(e, r, n);
                  a.TYPED_ARRAY_SUPPORT ? ((t = e).__proto__ = a.prototype) : (t = c(t, e));
                  return t
                })(t, e, r, n)
              : 'string' == typeof e
              ? (function (t, e, r) {
  ('string' == typeof r && '' !== r) || (r = 'utf8');
                  if (!a.isEncoding(r)) throw new TypeError('"encoding" must be a valid string encoding')
                  var n = 0 | g(e, r),
                    i = (t = u(t, n)).write(e, r);
                  i !== n && (t = t.slice(0, i));
                  return t
                })(t, e, r)
              : (function (t, e) {
                  if (a.isBuffer(e)) {
                    var r = 0 | p(e.length);
                    return 0 === (t = u(t, r)).length ? t : (e.copy(t, 0, 0, r), t)
                  }
                  if (e) {
                    if (('undefined' != typeof ArrayBuffer && e.buffer instanceof ArrayBuffer) || 'length' in e)
                      return 'number' != typeof e.length ||
                        (function (t) {
                          return t != t
                        })(e.length)
                        ? u(t, 0)
                        : c(t, e)
                    if ('Buffer' === e.type && o(e.data)) return c(t, e.data)
                  }
                  throw new TypeError(
                    'First argument must be a string, Buffer, ArrayBuffer, Array, or array-like object.'
                  )
                })(t, e)
          }
          function h(t) {
            if ('number' != typeof t) throw new TypeError('"size" argument must be a number')
            if (t < 0) throw new RangeError('"size" argument must not be negative')
          }
          function l(t, e) {
            if ((h(e), (t = u(t, e < 0 ? 0 : 0 | p(e))), !a.TYPED_ARRAY_SUPPORT)) for (var r = 0; r < e; ++r) t[r] = 0;
            return t
          }
          function c(t, e) {
            var r = e.length < 0 ? 0 : 0 | p(e.length);
            t = u(t, r);
            for (var n = 0; n < r; n += 1) t[n] = 255 & e[n];
            return t
          }
          function p(t) {
            if (t >= s())
              throw new RangeError(
                'Attempt to allocate Buffer larger than maximum size: 0x' + s().toString(16) + ' bytes'
              )
            return 0 | t
          }
          function g(t, e) {
            if (a.isBuffer(t)) return t.length
            if (
              'undefined' != typeof ArrayBuffer &&
              'function' == typeof ArrayBuffer.isView &&
              (ArrayBuffer.isView(t) || t instanceof ArrayBuffer)
            )
              return t.byteLength
            'string' != typeof t && (t = '' + t);
            var r = t.length;
            if (0 === r) return 0
            for (var n = !1; ; )
              switch (e) {
                case 'ascii':
                case 'latin1':
                case 'binary':
                  return r
                case 'utf8':
                case 'utf-8':
                case void 0:
                  return k(t).length
                case 'ucs2':
                case 'ucs-2':
                case 'utf16le':
                case 'utf-16le':
                  return 2 * r
                case 'hex':
                  return r >>> 1
                case 'base64':
                  return j(t).length
                default:
                  if (n) return k(t).length
                  ;(e = ('' + e).toLowerCase()), (n = !0);
              }
          }
          function d(t, e, r) {
            var n = t[e]
            ;(t[e] = t[r]), (t[r] = n);
          }
          function y(t, e, r, n, i) {
            if (0 === t.length) return -1
            if (
              ('string' == typeof r
                ? ((n = r), (r = 0))
                : r > 2147483647
                ? (r = 2147483647)
                : r < -2147483648 && (r = -2147483648),
              (r = +r),
              isNaN(r) && (r = i ? 0 : t.length - 1),
              r < 0 && (r = t.length + r),
              r >= t.length)
            ) {
              if (i) return -1
              r = t.length - 1;
            } else if (r < 0) {
              if (!i) return -1
              r = 0;
            }
            if (('string' == typeof e && (e = a.from(e, n)), a.isBuffer(e))) return 0 === e.length ? -1 : w(t, e, r, n, i)
            if ('number' == typeof e)
              return (
                (e &= 255),
                a.TYPED_ARRAY_SUPPORT && 'function' == typeof Uint8Array.prototype.indexOf
                  ? i
                    ? Uint8Array.prototype.indexOf.call(t, e, r)
                    : Uint8Array.prototype.lastIndexOf.call(t, e, r)
                  : w(t, [e], r, n, i)
              )
            throw new TypeError('val must be string, number or Buffer')
          }
          function w(t, e, r, n, i) {
            var o,
              s = 1,
              u = t.length,
              a = e.length;
            if (
              void 0 !== n &&
              ('ucs2' === (n = String(n).toLowerCase()) || 'ucs-2' === n || 'utf16le' === n || 'utf-16le' === n)
            ) {
              if (t.length < 2 || e.length < 2) return -1
              ;(s = 2), (u /= 2), (a /= 2), (r /= 2);
            }
            function f(t, e) {
              return 1 === s ? t[e] : t.readUInt16BE(e * s)
            }
            if (i) {
              var h = -1;
              for (o = r; o < u; o++)
                if (f(t, o) === f(e, -1 === h ? 0 : o - h)) {
                  if ((-1 === h && (h = o), o - h + 1 === a)) return h * s
                } else -1 !== h && (o -= o - h), (h = -1);
            } else
              for (r + a > u && (r = u - a), o = r; o >= 0; o--) {
                for (var l = !0, c = 0; c < a; c++)
                  if (f(t, o + c) !== f(e, c)) {
                    l = !1;
                    break
                  }
                if (l) return o
              }
            return -1
          }
          function E(t, e, r, n) {
            r = Number(r) || 0;
            var i = t.length - r;
            n ? (n = Number(n)) > i && (n = i) : (n = i);
            var o = e.length;
            if (o % 2 != 0) throw new TypeError('Invalid hex string')
            n > o / 2 && (n = o / 2);
            for (var s = 0; s < n; ++s) {
              var u = parseInt(e.substr(2 * s, 2), 16);
              if (isNaN(u)) return s
              t[r + s] = u;
            }
            return s
          }
          function v(t, e, r, n) {
            return z(k(e, t.length - r), t, r, n)
          }
          function A(t, e, r, n) {
            return z(
              (function (t) {
                for (var e = [], r = 0; r < t.length; ++r) e.push(255 & t.charCodeAt(r));
                return e
              })(e),
              t,
              r,
              n
            )
          }
          function _(t, e, r, n) {
            return A(t, e, r, n)
          }
          function b(t, e, r, n) {
            return z(j(e), t, r, n)
          }
          function m(t, e, r, n) {
            return z(
              (function (t, e) {
                for (var r, n, i, o = [], s = 0; s < t.length && !((e -= 2) < 0); ++s)
                  (r = t.charCodeAt(s)), (n = r >> 8), (i = r % 256), o.push(i), o.push(n);
                return o
              })(e, t.length - r),
              t,
              r,
              n
            )
          }
          function R(t, e, r) {
            return 0 === e && r === t.length ? n.fromByteArray(t) : n.fromByteArray(t.slice(e, r))
          }
          function S(t, e, r) {
            r = Math.min(t.length, r);
            for (var n = [], i = e; i < r; ) {
              var o,
                s,
                u,
                a,
                f = t[i],
                h = null,
                l = f > 239 ? 4 : f > 223 ? 3 : f > 191 ? 2 : 1;
              if (i + l <= r)
                switch (l) {
                  case 1:
                    f < 128 && (h = f);
                    break
                  case 2:
                    128 == (192 & (o = t[i + 1])) && (a = ((31 & f) << 6) | (63 & o)) > 127 && (h = a);
                    break
                  case 3:
  (o = t[i + 1]),
                      (s = t[i + 2]),
                      128 == (192 & o) &&
                        128 == (192 & s) &&
                        (a = ((15 & f) << 12) | ((63 & o) << 6) | (63 & s)) > 2047 &&
                        (a < 55296 || a > 57343) &&
                        (h = a);
                    break
                  case 4:
  (o = t[i + 1]),
                      (s = t[i + 2]),
                      (u = t[i + 3]),
                      128 == (192 & o) &&
                        128 == (192 & s) &&
                        128 == (192 & u) &&
                        (a = ((15 & f) << 18) | ((63 & o) << 12) | ((63 & s) << 6) | (63 & u)) > 65535 &&
                        a < 1114112 &&
                        (h = a);
                }
              null === h
                ? ((h = 65533), (l = 1))
                : h > 65535 && ((h -= 65536), n.push(((h >>> 10) & 1023) | 55296), (h = 56320 | (1023 & h))),
                n.push(h),
                (i += l);
            }
            return (function (t) {
              var e = t.length;
              if (e <= T) return String.fromCharCode.apply(String, t)
              var r = '',
                n = 0;
              for (; n < e; ) r += String.fromCharCode.apply(String, t.slice(n, (n += T)));
              return r
            })(n)
          }
  (e.Buffer = a),
            (e.SlowBuffer = function (t) {
  +t != t && (t = 0);
              return a.alloc(+t)
            }),
            (e.INSPECT_MAX_BYTES = 50),
            (a.TYPED_ARRAY_SUPPORT =
              void 0 !== t.TYPED_ARRAY_SUPPORT
                ? t.TYPED_ARRAY_SUPPORT
                : (function () {
                    try {
                      var t = new Uint8Array(1);
                      return (
                        (t.__proto__ = {
                          __proto__: Uint8Array.prototype,
                          foo: function () {
                            return 42
                          },
                        }),
                        42 === t.foo() && 'function' == typeof t.subarray && 0 === t.subarray(1, 1).byteLength
                      )
                    } catch (t) {
                      return !1
                    }
                  })()),
            (e.kMaxLength = s()),
            (a.poolSize = 8192),
            (a._augment = function (t) {
              return (t.__proto__ = a.prototype), t
            }),
            (a.from = function (t, e, r) {
              return f(null, t, e, r)
            }),
            a.TYPED_ARRAY_SUPPORT &&
              ((a.prototype.__proto__ = Uint8Array.prototype),
              (a.__proto__ = Uint8Array),
              'undefined' != typeof Symbol &&
                Symbol.species &&
                a[Symbol.species] === a &&
                Object.defineProperty(a, Symbol.species, { value: null, configurable: !0 })),
            (a.alloc = function (t, e, r) {
              return (function (t, e, r, n) {
                return (
                  h(e),
                  e <= 0
                    ? u(t, e)
                    : void 0 !== r
                    ? 'string' == typeof n
                      ? u(t, e).fill(r, n)
                      : u(t, e).fill(r)
                    : u(t, e)
                )
              })(null, t, e, r)
            }),
            (a.allocUnsafe = function (t) {
              return l(null, t)
            }),
            (a.allocUnsafeSlow = function (t) {
              return l(null, t)
            }),
            (a.isBuffer = function (t) {
              return !(null == t || !t._isBuffer)
            }),
            (a.compare = function (t, e) {
              if (!a.isBuffer(t) || !a.isBuffer(e)) throw new TypeError('Arguments must be Buffers')
              if (t === e) return 0
              for (var r = t.length, n = e.length, i = 0, o = Math.min(r, n); i < o; ++i)
                if (t[i] !== e[i]) {
  (r = t[i]), (n = e[i]);
                  break
                }
              return r < n ? -1 : n < r ? 1 : 0
            }),
            (a.isEncoding = function (t) {
              switch (String(t).toLowerCase()) {
                case 'hex':
                case 'utf8':
                case 'utf-8':
                case 'ascii':
                case 'latin1':
                case 'binary':
                case 'base64':
                case 'ucs2':
                case 'ucs-2':
                case 'utf16le':
                case 'utf-16le':
                  return !0
                default:
                  return !1
              }
            }),
            (a.concat = function (t, e) {
              if (!o(t)) throw new TypeError('"list" argument must be an Array of Buffers')
              if (0 === t.length) return a.alloc(0)
              var r;
              if (void 0 === e) for (e = 0, r = 0; r < t.length; ++r) e += t[r].length;
              var n = a.allocUnsafe(e),
                i = 0;
              for (r = 0; r < t.length; ++r) {
                var s = t[r];
                if (!a.isBuffer(s)) throw new TypeError('"list" argument must be an Array of Buffers')
                s.copy(n, i), (i += s.length);
              }
              return n
            }),
            (a.byteLength = g),
            (a.prototype._isBuffer = !0),
            (a.prototype.swap16 = function () {
              var t = this.length;
              if (t % 2 != 0) throw new RangeError('Buffer size must be a multiple of 16-bits')
              for (var e = 0; e < t; e += 2) d(this, e, e + 1);
              return this
            }),
            (a.prototype.swap32 = function () {
              var t = this.length;
              if (t % 4 != 0) throw new RangeError('Buffer size must be a multiple of 32-bits')
              for (var e = 0; e < t; e += 4) d(this, e, e + 3), d(this, e + 1, e + 2);
              return this
            }),
            (a.prototype.swap64 = function () {
              var t = this.length;
              if (t % 8 != 0) throw new RangeError('Buffer size must be a multiple of 64-bits')
              for (var e = 0; e < t; e += 8)
                d(this, e, e + 7), d(this, e + 1, e + 6), d(this, e + 2, e + 5), d(this, e + 3, e + 4);
              return this
            }),
            (a.prototype.toString = function () {
              var t = 0 | this.length;
              return 0 === t
                ? ''
                : 0 === arguments.length
                ? S(this, 0, t)
                : function (t, e, r) {
                    var n = !1;
                    if (((void 0 === e || e < 0) && (e = 0), e > this.length)) return ''
                    if (((void 0 === r || r > this.length) && (r = this.length), r <= 0)) return ''
                    if ((r >>>= 0) <= (e >>>= 0)) return ''
                    for (t || (t = 'utf8'); ; )
                      switch (t) {
                        case 'hex':
                          return U(this, e, r)
                        case 'utf8':
                        case 'utf-8':
                          return S(this, e, r)
                        case 'ascii':
                          return P(this, e, r)
                        case 'latin1':
                        case 'binary':
                          return C(this, e, r)
                        case 'base64':
                          return R(this, e, r)
                        case 'ucs2':
                        case 'ucs-2':
                        case 'utf16le':
                        case 'utf-16le':
                          return B(this, e, r)
                        default:
                          if (n) throw new TypeError('Unknown encoding: ' + t)
                          ;(t = (t + '').toLowerCase()), (n = !0);
                      }
                  }.apply(this, arguments)
            }),
            (a.prototype.equals = function (t) {
              if (!a.isBuffer(t)) throw new TypeError('Argument must be a Buffer')
              return this === t || 0 === a.compare(this, t)
            }),
            (a.prototype.inspect = function () {
              var t = '',
                r = e.INSPECT_MAX_BYTES;
              return (
                this.length > 0 &&
                  ((t = this.toString('hex', 0, r).match(/.{2}/g).join(' ')), this.length > r && (t += ' ... ')),
                '<Buffer ' + t + '>'
              )
            }),
            (a.prototype.compare = function (t, e, r, n, i) {
              if (!a.isBuffer(t)) throw new TypeError('Argument must be a Buffer')
              if (
                (void 0 === e && (e = 0),
                void 0 === r && (r = t ? t.length : 0),
                void 0 === n && (n = 0),
                void 0 === i && (i = this.length),
                e < 0 || r > t.length || n < 0 || i > this.length)
              )
                throw new RangeError('out of range index')
              if (n >= i && e >= r) return 0
              if (n >= i) return -1
              if (e >= r) return 1
              if (((e >>>= 0), (r >>>= 0), (n >>>= 0), (i >>>= 0), this === t)) return 0
              for (
                var o = i - n, s = r - e, u = Math.min(o, s), f = this.slice(n, i), h = t.slice(e, r), l = 0;
                l < u;
                ++l
              )
                if (f[l] !== h[l]) {
  (o = f[l]), (s = h[l]);
                  break
                }
              return o < s ? -1 : s < o ? 1 : 0
            }),
            (a.prototype.includes = function (t, e, r) {
              return -1 !== this.indexOf(t, e, r)
            }),
            (a.prototype.indexOf = function (t, e, r) {
              return y(this, t, e, r, !0)
            }),
            (a.prototype.lastIndexOf = function (t, e, r) {
              return y(this, t, e, r, !1)
            }),
            (a.prototype.write = function (t, e, r, n) {
              if (void 0 === e) (n = 'utf8'), (r = this.length), (e = 0);
              else if (void 0 === r && 'string' == typeof e) (n = e), (r = this.length), (e = 0);
              else {
                if (!isFinite(e))
                  throw new Error('Buffer.write(string, encoding, offset[, length]) is no longer supported')
                ;(e |= 0), isFinite(r) ? ((r |= 0), void 0 === n && (n = 'utf8')) : ((n = r), (r = void 0));
              }
              var i = this.length - e;
              if (((void 0 === r || r > i) && (r = i), (t.length > 0 && (r < 0 || e < 0)) || e > this.length))
                throw new RangeError('Attempt to write outside buffer bounds')
              n || (n = 'utf8');
              for (var o = !1; ; )
                switch (n) {
                  case 'hex':
                    return E(this, t, e, r)
                  case 'utf8':
                  case 'utf-8':
                    return v(this, t, e, r)
                  case 'ascii':
                    return A(this, t, e, r)
                  case 'latin1':
                  case 'binary':
                    return _(this, t, e, r)
                  case 'base64':
                    return b(this, t, e, r)
                  case 'ucs2':
                  case 'ucs-2':
                  case 'utf16le':
                  case 'utf-16le':
                    return m(this, t, e, r)
                  default:
                    if (o) throw new TypeError('Unknown encoding: ' + n)
                    ;(n = ('' + n).toLowerCase()), (o = !0);
                }
            }),
            (a.prototype.toJSON = function () {
              return { type: 'Buffer', data: Array.prototype.slice.call(this._arr || this, 0) }
            });
          var T = 4096;
          function P(t, e, r) {
            var n = '';
            r = Math.min(t.length, r);
            for (var i = e; i < r; ++i) n += String.fromCharCode(127 & t[i]);
            return n
          }
          function C(t, e, r) {
            var n = '';
            r = Math.min(t.length, r);
            for (var i = e; i < r; ++i) n += String.fromCharCode(t[i]);
            return n
          }
          function U(t, e, r) {
            var n = t.length
            ;(!e || e < 0) && (e = 0), (!r || r < 0 || r > n) && (r = n);
            for (var i = '', o = e; o < r; ++o) i += N(t[o]);
            return i
          }
          function B(t, e, r) {
            for (var n = t.slice(e, r), i = '', o = 0; o < n.length; o += 2)
              i += String.fromCharCode(n[o] + 256 * n[o + 1]);
            return i
          }
          function F(t, e, r) {
            if (t % 1 != 0 || t < 0) throw new RangeError('offset is not uint')
            if (t + e > r) throw new RangeError('Trying to access beyond buffer length')
          }
          function M(t, e, r, n, i, o) {
            if (!a.isBuffer(t)) throw new TypeError('"buffer" argument must be a Buffer instance')
            if (e > i || e < o) throw new RangeError('"value" argument is out of bounds')
            if (r + n > t.length) throw new RangeError('Index out of range')
          }
          function x(t, e, r, n) {
            e < 0 && (e = 65535 + e + 1);
            for (var i = 0, o = Math.min(t.length - r, 2); i < o; ++i)
              t[r + i] = (e & (255 << (8 * (n ? i : 1 - i)))) >>> (8 * (n ? i : 1 - i));
          }
          function I(t, e, r, n) {
            e < 0 && (e = 4294967295 + e + 1);
            for (var i = 0, o = Math.min(t.length - r, 4); i < o; ++i) t[r + i] = (e >>> (8 * (n ? i : 3 - i))) & 255;
          }
          function O(t, e, r, n, i, o) {
            if (r + n > t.length) throw new RangeError('Index out of range')
            if (r < 0) throw new RangeError('Index out of range')
          }
          function Y(t, e, r, n, o) {
            return o || O(t, 0, r, 4), i.write(t, e, r, n, 23, 4), r + 4
          }
          function L(t, e, r, n, o) {
            return o || O(t, 0, r, 8), i.write(t, e, r, n, 52, 8), r + 8
          }
  (a.prototype.slice = function (t, e) {
            var r,
              n = this.length;
            if (
              ((t = ~~t),
              (e = void 0 === e ? n : ~~e),
              t < 0 ? (t += n) < 0 && (t = 0) : t > n && (t = n),
              e < 0 ? (e += n) < 0 && (e = 0) : e > n && (e = n),
              e < t && (e = t),
              a.TYPED_ARRAY_SUPPORT)
            )
              (r = this.subarray(t, e)).__proto__ = a.prototype;
            else {
              var i = e - t;
              r = new a(i, void 0);
              for (var o = 0; o < i; ++o) r[o] = this[o + t];
            }
            return r
          }),
            (a.prototype.readUIntLE = function (t, e, r) {
  (t |= 0), (e |= 0), r || F(t, e, this.length);
              for (var n = this[t], i = 1, o = 0; ++o < e && (i *= 256); ) n += this[t + o] * i;
              return n
            }),
            (a.prototype.readUIntBE = function (t, e, r) {
  (t |= 0), (e |= 0), r || F(t, e, this.length);
              for (var n = this[t + --e], i = 1; e > 0 && (i *= 256); ) n += this[t + --e] * i;
              return n
            }),
            (a.prototype.readUInt8 = function (t, e) {
              return e || F(t, 1, this.length), this[t]
            }),
            (a.prototype.readUInt16LE = function (t, e) {
              return e || F(t, 2, this.length), this[t] | (this[t + 1] << 8)
            }),
            (a.prototype.readUInt16BE = function (t, e) {
              return e || F(t, 2, this.length), (this[t] << 8) | this[t + 1]
            }),
            (a.prototype.readUInt32LE = function (t, e) {
              return (
                e || F(t, 4, this.length), (this[t] | (this[t + 1] << 8) | (this[t + 2] << 16)) + 16777216 * this[t + 3]
              )
            }),
            (a.prototype.readUInt32BE = function (t, e) {
              return (
                e || F(t, 4, this.length), 16777216 * this[t] + ((this[t + 1] << 16) | (this[t + 2] << 8) | this[t + 3])
              )
            }),
            (a.prototype.readIntLE = function (t, e, r) {
  (t |= 0), (e |= 0), r || F(t, e, this.length);
              for (var n = this[t], i = 1, o = 0; ++o < e && (i *= 256); ) n += this[t + o] * i;
              return n >= (i *= 128) && (n -= Math.pow(2, 8 * e)), n
            }),
            (a.prototype.readIntBE = function (t, e, r) {
  (t |= 0), (e |= 0), r || F(t, e, this.length);
              for (var n = e, i = 1, o = this[t + --n]; n > 0 && (i *= 256); ) o += this[t + --n] * i;
              return o >= (i *= 128) && (o -= Math.pow(2, 8 * e)), o
            }),
            (a.prototype.readInt8 = function (t, e) {
              return e || F(t, 1, this.length), 128 & this[t] ? -1 * (255 - this[t] + 1) : this[t]
            }),
            (a.prototype.readInt16LE = function (t, e) {
              e || F(t, 2, this.length);
              var r = this[t] | (this[t + 1] << 8);
              return 32768 & r ? 4294901760 | r : r
            }),
            (a.prototype.readInt16BE = function (t, e) {
              e || F(t, 2, this.length);
              var r = this[t + 1] | (this[t] << 8);
              return 32768 & r ? 4294901760 | r : r
            }),
            (a.prototype.readInt32LE = function (t, e) {
              return e || F(t, 4, this.length), this[t] | (this[t + 1] << 8) | (this[t + 2] << 16) | (this[t + 3] << 24)
            }),
            (a.prototype.readInt32BE = function (t, e) {
              return e || F(t, 4, this.length), (this[t] << 24) | (this[t + 1] << 16) | (this[t + 2] << 8) | this[t + 3]
            }),
            (a.prototype.readFloatLE = function (t, e) {
              return e || F(t, 4, this.length), i.read(this, t, !0, 23, 4)
            }),
            (a.prototype.readFloatBE = function (t, e) {
              return e || F(t, 4, this.length), i.read(this, t, !1, 23, 4)
            }),
            (a.prototype.readDoubleLE = function (t, e) {
              return e || F(t, 8, this.length), i.read(this, t, !0, 52, 8)
            }),
            (a.prototype.readDoubleBE = function (t, e) {
              return e || F(t, 8, this.length), i.read(this, t, !1, 52, 8)
            }),
            (a.prototype.writeUIntLE = function (t, e, r, n) {
  ((t = +t), (e |= 0), (r |= 0), n) || M(this, t, e, r, Math.pow(2, 8 * r) - 1, 0);
              var i = 1,
                o = 0;
              for (this[e] = 255 & t; ++o < r && (i *= 256); ) this[e + o] = (t / i) & 255;
              return e + r
            }),
            (a.prototype.writeUIntBE = function (t, e, r, n) {
  ((t = +t), (e |= 0), (r |= 0), n) || M(this, t, e, r, Math.pow(2, 8 * r) - 1, 0);
              var i = r - 1,
                o = 1;
              for (this[e + i] = 255 & t; --i >= 0 && (o *= 256); ) this[e + i] = (t / o) & 255;
              return e + r
            }),
            (a.prototype.writeUInt8 = function (t, e, r) {
              return (
                (t = +t),
                (e |= 0),
                r || M(this, t, e, 1, 255, 0),
                a.TYPED_ARRAY_SUPPORT || (t = Math.floor(t)),
                (this[e] = 255 & t),
                e + 1
              )
            }),
            (a.prototype.writeUInt16LE = function (t, e, r) {
              return (
                (t = +t),
                (e |= 0),
                r || M(this, t, e, 2, 65535, 0),
                a.TYPED_ARRAY_SUPPORT ? ((this[e] = 255 & t), (this[e + 1] = t >>> 8)) : x(this, t, e, !0),
                e + 2
              )
            }),
            (a.prototype.writeUInt16BE = function (t, e, r) {
              return (
                (t = +t),
                (e |= 0),
                r || M(this, t, e, 2, 65535, 0),
                a.TYPED_ARRAY_SUPPORT ? ((this[e] = t >>> 8), (this[e + 1] = 255 & t)) : x(this, t, e, !1),
                e + 2
              )
            }),
            (a.prototype.writeUInt32LE = function (t, e, r) {
              return (
                (t = +t),
                (e |= 0),
                r || M(this, t, e, 4, 4294967295, 0),
                a.TYPED_ARRAY_SUPPORT
                  ? ((this[e + 3] = t >>> 24), (this[e + 2] = t >>> 16), (this[e + 1] = t >>> 8), (this[e] = 255 & t))
                  : I(this, t, e, !0),
                e + 4
              )
            }),
            (a.prototype.writeUInt32BE = function (t, e, r) {
              return (
                (t = +t),
                (e |= 0),
                r || M(this, t, e, 4, 4294967295, 0),
                a.TYPED_ARRAY_SUPPORT
                  ? ((this[e] = t >>> 24), (this[e + 1] = t >>> 16), (this[e + 2] = t >>> 8), (this[e + 3] = 255 & t))
                  : I(this, t, e, !1),
                e + 4
              )
            }),
            (a.prototype.writeIntLE = function (t, e, r, n) {
              if (((t = +t), (e |= 0), !n)) {
                var i = Math.pow(2, 8 * r - 1);
                M(this, t, e, r, i - 1, -i);
              }
              var o = 0,
                s = 1,
                u = 0;
              for (this[e] = 255 & t; ++o < r && (s *= 256); )
                t < 0 && 0 === u && 0 !== this[e + o - 1] && (u = 1), (this[e + o] = (((t / s) >> 0) - u) & 255);
              return e + r
            }),
            (a.prototype.writeIntBE = function (t, e, r, n) {
              if (((t = +t), (e |= 0), !n)) {
                var i = Math.pow(2, 8 * r - 1);
                M(this, t, e, r, i - 1, -i);
              }
              var o = r - 1,
                s = 1,
                u = 0;
              for (this[e + o] = 255 & t; --o >= 0 && (s *= 256); )
                t < 0 && 0 === u && 0 !== this[e + o + 1] && (u = 1), (this[e + o] = (((t / s) >> 0) - u) & 255);
              return e + r
            }),
            (a.prototype.writeInt8 = function (t, e, r) {
              return (
                (t = +t),
                (e |= 0),
                r || M(this, t, e, 1, 127, -128),
                a.TYPED_ARRAY_SUPPORT || (t = Math.floor(t)),
                t < 0 && (t = 255 + t + 1),
                (this[e] = 255 & t),
                e + 1
              )
            }),
            (a.prototype.writeInt16LE = function (t, e, r) {
              return (
                (t = +t),
                (e |= 0),
                r || M(this, t, e, 2, 32767, -32768),
                a.TYPED_ARRAY_SUPPORT ? ((this[e] = 255 & t), (this[e + 1] = t >>> 8)) : x(this, t, e, !0),
                e + 2
              )
            }),
            (a.prototype.writeInt16BE = function (t, e, r) {
              return (
                (t = +t),
                (e |= 0),
                r || M(this, t, e, 2, 32767, -32768),
                a.TYPED_ARRAY_SUPPORT ? ((this[e] = t >>> 8), (this[e + 1] = 255 & t)) : x(this, t, e, !1),
                e + 2
              )
            }),
            (a.prototype.writeInt32LE = function (t, e, r) {
              return (
                (t = +t),
                (e |= 0),
                r || M(this, t, e, 4, 2147483647, -2147483648),
                a.TYPED_ARRAY_SUPPORT
                  ? ((this[e] = 255 & t), (this[e + 1] = t >>> 8), (this[e + 2] = t >>> 16), (this[e + 3] = t >>> 24))
                  : I(this, t, e, !0),
                e + 4
              )
            }),
            (a.prototype.writeInt32BE = function (t, e, r) {
              return (
                (t = +t),
                (e |= 0),
                r || M(this, t, e, 4, 2147483647, -2147483648),
                t < 0 && (t = 4294967295 + t + 1),
                a.TYPED_ARRAY_SUPPORT
                  ? ((this[e] = t >>> 24), (this[e + 1] = t >>> 16), (this[e + 2] = t >>> 8), (this[e + 3] = 255 & t))
                  : I(this, t, e, !1),
                e + 4
              )
            }),
            (a.prototype.writeFloatLE = function (t, e, r) {
              return Y(this, t, e, !0, r)
            }),
            (a.prototype.writeFloatBE = function (t, e, r) {
              return Y(this, t, e, !1, r)
            }),
            (a.prototype.writeDoubleLE = function (t, e, r) {
              return L(this, t, e, !0, r)
            }),
            (a.prototype.writeDoubleBE = function (t, e, r) {
              return L(this, t, e, !1, r)
            }),
            (a.prototype.copy = function (t, e, r, n) {
              if (
                (r || (r = 0),
                n || 0 === n || (n = this.length),
                e >= t.length && (e = t.length),
                e || (e = 0),
                n > 0 && n < r && (n = r),
                n === r)
              )
                return 0
              if (0 === t.length || 0 === this.length) return 0
              if (e < 0) throw new RangeError('targetStart out of bounds')
              if (r < 0 || r >= this.length) throw new RangeError('sourceStart out of bounds')
              if (n < 0) throw new RangeError('sourceEnd out of bounds')
              n > this.length && (n = this.length), t.length - e < n - r && (n = t.length - e + r);
              var i,
                o = n - r;
              if (this === t && r < e && e < n) for (i = o - 1; i >= 0; --i) t[i + e] = this[i + r];
              else if (o < 1e3 || !a.TYPED_ARRAY_SUPPORT) for (i = 0; i < o; ++i) t[i + e] = this[i + r];
              else Uint8Array.prototype.set.call(t, this.subarray(r, r + o), e);
              return o
            }),
            (a.prototype.fill = function (t, e, r, n) {
              if ('string' == typeof t) {
                if (
                  ('string' == typeof e
                    ? ((n = e), (e = 0), (r = this.length))
                    : 'string' == typeof r && ((n = r), (r = this.length)),
                  1 === t.length)
                ) {
                  var i = t.charCodeAt(0);
                  i < 256 && (t = i);
                }
                if (void 0 !== n && 'string' != typeof n) throw new TypeError('encoding must be a string')
                if ('string' == typeof n && !a.isEncoding(n)) throw new TypeError('Unknown encoding: ' + n)
              } else 'number' == typeof t && (t &= 255);
              if (e < 0 || this.length < e || this.length < r) throw new RangeError('Out of range index')
              if (r <= e) return this
              var o;
              if (((e >>>= 0), (r = void 0 === r ? this.length : r >>> 0), t || (t = 0), 'number' == typeof t))
                for (o = e; o < r; ++o) this[o] = t;
              else {
                var s = a.isBuffer(t) ? t : k(new a(t, n).toString()),
                  u = s.length;
                for (o = 0; o < r - e; ++o) this[o + e] = s[o % u];
              }
              return this
            });
          var D = /[^+\/0-9A-Za-z-_]/g;
          function N(t) {
            return t < 16 ? '0' + t.toString(16) : t.toString(16)
          }
          function k(t, e) {
            var r;
            e = e || 1 / 0;
            for (var n = t.length, i = null, o = [], s = 0; s < n; ++s) {
              if ((r = t.charCodeAt(s)) > 55295 && r < 57344) {
                if (!i) {
                  if (r > 56319) {
  (e -= 3) > -1 && o.push(239, 191, 189);
                    continue
                  }
                  if (s + 1 === n) {
  (e -= 3) > -1 && o.push(239, 191, 189);
                    continue
                  }
                  i = r;
                  continue
                }
                if (r < 56320) {
  (e -= 3) > -1 && o.push(239, 191, 189), (i = r);
                  continue
                }
                r = 65536 + (((i - 55296) << 10) | (r - 56320));
              } else i && (e -= 3) > -1 && o.push(239, 191, 189);
              if (((i = null), r < 128)) {
                if ((e -= 1) < 0) break
                o.push(r);
              } else if (r < 2048) {
                if ((e -= 2) < 0) break
                o.push((r >> 6) | 192, (63 & r) | 128);
              } else if (r < 65536) {
                if ((e -= 3) < 0) break
                o.push((r >> 12) | 224, ((r >> 6) & 63) | 128, (63 & r) | 128);
              } else {
                if (!(r < 1114112)) throw new Error('Invalid code point')
                if ((e -= 4) < 0) break
                o.push((r >> 18) | 240, ((r >> 12) & 63) | 128, ((r >> 6) & 63) | 128, (63 & r) | 128);
              }
            }
            return o
          }
          function j(t) {
            return n.toByteArray(
              (function (t) {
                if (
                  (t = (function (t) {
                    return t.trim ? t.trim() : t.replace(/^\s+|\s+$/g, '')
                  })(t).replace(D, '')).length < 2
                )
                  return ''
                for (; t.length % 4 != 0; ) t += '=';
                return t
              })(t)
            )
          }
          function z(t, e, r, n) {
            for (var i = 0; i < n && !(i + r >= e.length || i >= t.length); ++i) e[i + r] = t[i];
            return i
          }
        }.call(this, r(8)));
      },
      function (t, e) {
        var r;
        r = (function () {
          return this
        })();
        try {
          r = r || Function('return this')() || (0, eval)('this');
        } catch (t) {
          'object' == typeof window && (r = window);
        }
        t.exports = r;
      },
      function (t, e, r) {
  (e.byteLength = function (t) {
          var e = f(t),
            r = e[0],
            n = e[1];
          return (3 * (r + n)) / 4 - n
        }),
          (e.toByteArray = function (t) {
            for (
              var e,
                r = f(t),
                n = r[0],
                s = r[1],
                u = new o(
                  (function (t, e, r) {
                    return (3 * (e + r)) / 4 - r
                  })(0, n, s)
                ),
                a = 0,
                h = s > 0 ? n - 4 : n,
                l = 0;
              l < h;
              l += 4
            )
              (e =
                (i[t.charCodeAt(l)] << 18) |
                (i[t.charCodeAt(l + 1)] << 12) |
                (i[t.charCodeAt(l + 2)] << 6) |
                i[t.charCodeAt(l + 3)]),
                (u[a++] = (e >> 16) & 255),
                (u[a++] = (e >> 8) & 255),
                (u[a++] = 255 & e);
            2 === s && ((e = (i[t.charCodeAt(l)] << 2) | (i[t.charCodeAt(l + 1)] >> 4)), (u[a++] = 255 & e));
            1 === s &&
              ((e = (i[t.charCodeAt(l)] << 10) | (i[t.charCodeAt(l + 1)] << 4) | (i[t.charCodeAt(l + 2)] >> 2)),
              (u[a++] = (e >> 8) & 255),
              (u[a++] = 255 & e));
            return u
          }),
          (e.fromByteArray = function (t) {
            for (var e, r = t.length, i = r % 3, o = [], s = 0, u = r - i; s < u; s += 16383)
              o.push(l(t, s, s + 16383 > u ? u : s + 16383));
            1 === i
              ? ((e = t[r - 1]), o.push(n[e >> 2] + n[(e << 4) & 63] + '=='))
              : 2 === i &&
                ((e = (t[r - 2] << 8) + t[r - 1]), o.push(n[e >> 10] + n[(e >> 4) & 63] + n[(e << 2) & 63] + '='));
            return o.join('')
          });
        for (
          var n = [],
            i = [],
            o = 'undefined' != typeof Uint8Array ? Uint8Array : Array,
            s = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/',
            u = 0,
            a = s.length;
          u < a;
          ++u
        )
          (n[u] = s[u]), (i[s.charCodeAt(u)] = u);
        function f(t) {
          var e = t.length;
          if (e % 4 > 0) throw new Error('Invalid string. Length must be a multiple of 4')
          var r = t.indexOf('=');
          return -1 === r && (r = e), [r, r === e ? 0 : 4 - (r % 4)]
        }
        function h(t) {
          return n[(t >> 18) & 63] + n[(t >> 12) & 63] + n[(t >> 6) & 63] + n[63 & t]
        }
        function l(t, e, r) {
          for (var n, i = [], o = e; o < r; o += 3)
            (n = ((t[o] << 16) & 16711680) + ((t[o + 1] << 8) & 65280) + (255 & t[o + 2])), i.push(h(n));
          return i.join('')
        }
  (i['-'.charCodeAt(0)] = 62), (i['_'.charCodeAt(0)] = 63);
      },
      function (t, e) {
  (e.read = function (t, e, r, n, i) {
          var o,
            s,
            u = 8 * i - n - 1,
            a = (1 << u) - 1,
            f = a >> 1,
            h = -7,
            l = r ? i - 1 : 0,
            c = r ? -1 : 1,
            p = t[e + l];
          for (l += c, o = p & ((1 << -h) - 1), p >>= -h, h += u; h > 0; o = 256 * o + t[e + l], l += c, h -= 8);
          for (s = o & ((1 << -h) - 1), o >>= -h, h += n; h > 0; s = 256 * s + t[e + l], l += c, h -= 8);
          if (0 === o) o = 1 - f;
          else {
            if (o === a) return s ? NaN : (1 / 0) * (p ? -1 : 1)
            ;(s += Math.pow(2, n)), (o -= f);
          }
          return (p ? -1 : 1) * s * Math.pow(2, o - n)
        }),
          (e.write = function (t, e, r, n, i, o) {
            var s,
              u,
              a,
              f = 8 * o - i - 1,
              h = (1 << f) - 1,
              l = h >> 1,
              c = 23 === i ? Math.pow(2, -24) - Math.pow(2, -77) : 0,
              p = n ? 0 : o - 1,
              g = n ? 1 : -1,
              d = e < 0 || (0 === e && 1 / e < 0) ? 1 : 0;
            for (
              e = Math.abs(e),
                isNaN(e) || e === 1 / 0
                  ? ((u = isNaN(e) ? 1 : 0), (s = h))
                  : ((s = Math.floor(Math.log(e) / Math.LN2)),
                    e * (a = Math.pow(2, -s)) < 1 && (s--, (a *= 2)),
                    (e += s + l >= 1 ? c / a : c * Math.pow(2, 1 - l)) * a >= 2 && (s++, (a /= 2)),
                    s + l >= h
                      ? ((u = 0), (s = h))
                      : s + l >= 1
                      ? ((u = (e * a - 1) * Math.pow(2, i)), (s += l))
                      : ((u = e * Math.pow(2, l - 1) * Math.pow(2, i)), (s = 0)));
              i >= 8;
              t[r + p] = 255 & u, p += g, u /= 256, i -= 8
            );
            for (s = (s << i) | u, f += i; f > 0; t[r + p] = 255 & s, p += g, s /= 256, f -= 8);
            t[r + p - g] |= 128 * d;
          });
      },
      function (t, e) {
        var r = {}.toString;
        t.exports =
          Array.isArray ||
          function (t) {
            return '[object Array]' == r.call(t)
          };
      },
      function (e, r) {
        e.exports = t;
      },
      function (t, e, r) {
  (function (t) {
          function r(t, e) {
            for (var r = 0, n = t.length - 1; n >= 0; n--) {
              var i = t[n];
              '.' === i ? t.splice(n, 1) : '..' === i ? (t.splice(n, 1), r++) : r && (t.splice(n, 1), r--);
            }
            if (e) for (; r--; r) t.unshift('..');
            return t
          }
          var n = /^(\/?|)([\s\S]*?)((?:\.{1,2}|[^\/]+?|)(\.[^.\/]*|))(?:[\/]*)$/,
            i = function (t) {
              return n.exec(t).slice(1)
            };
          function o(t, e) {
            if (t.filter) return t.filter(e)
            for (var r = [], n = 0; n < t.length; n++) e(t[n], n, t) && r.push(t[n]);
            return r
          }
  (e.resolve = function () {
            for (var e = '', n = !1, i = arguments.length - 1; i >= -1 && !n; i--) {
              var s = i >= 0 ? arguments[i] : t.cwd();
              if ('string' != typeof s) throw new TypeError('Arguments to path.resolve must be strings')
              s && ((e = s + '/' + e), (n = '/' === s.charAt(0)));
            }
            return (
              (e = r(
                o(e.split('/'), function (t) {
                  return !!t
                }),
                !n
              ).join('/')),
              (n ? '/' : '') + e || '.'
            )
          }),
            (e.normalize = function (t) {
              var n = e.isAbsolute(t),
                i = '/' === s(t, -1);
              return (
                (t = r(
                  o(t.split('/'), function (t) {
                    return !!t
                  }),
                  !n
                ).join('/')) ||
                  n ||
                  (t = '.'),
                t && i && (t += '/'),
                (n ? '/' : '') + t
              )
            }),
            (e.isAbsolute = function (t) {
              return '/' === t.charAt(0)
            }),
            (e.join = function () {
              var t = Array.prototype.slice.call(arguments, 0);
              return e.normalize(
                o(t, function (t, e) {
                  if ('string' != typeof t) throw new TypeError('Arguments to path.join must be strings')
                  return t
                }).join('/')
              )
            }),
            (e.relative = function (t, r) {
              function n(t) {
                for (var e = 0; e < t.length && '' === t[e]; e++);
                for (var r = t.length - 1; r >= 0 && '' === t[r]; r--);
                return e > r ? [] : t.slice(e, r - e + 1)
              }
  (t = e.resolve(t).substr(1)), (r = e.resolve(r).substr(1));
              for (
                var i = n(t.split('/')), o = n(r.split('/')), s = Math.min(i.length, o.length), u = s, a = 0;
                a < s;
                a++
              )
                if (i[a] !== o[a]) {
                  u = a;
                  break
                }
              var f = [];
              for (a = u; a < i.length; a++) f.push('..');
              return (f = f.concat(o.slice(u))).join('/')
            }),
            (e.sep = '/'),
            (e.delimiter = ':'),
            (e.dirname = function (t) {
              var e = i(t),
                r = e[0],
                n = e[1];
              return r || n ? (n && (n = n.substr(0, n.length - 1)), r + n) : '.'
            }),
            (e.basename = function (t, e) {
              var r = i(t)[2];
              return e && r.substr(-1 * e.length) === e && (r = r.substr(0, r.length - e.length)), r
            }),
            (e.extname = function (t) {
              return i(t)[3]
            });
          var s =
            'b' === 'ab'.substr(-1)
              ? function (t, e, r) {
                  return t.substr(e, r)
                }
              : function (t, e, r) {
                  return e < 0 && (e = t.length + e), t.substr(e, r)
                };
        }.call(this, r(14)));
      },
      function (t, e) {
        var r,
          n,
          i = (t.exports = {});
        function o() {
          throw new Error('setTimeout has not been defined')
        }
        function s() {
          throw new Error('clearTimeout has not been defined')
        }
        function u(t) {
          if (r === setTimeout) return setTimeout(t, 0)
          if ((r === o || !r) && setTimeout) return (r = setTimeout), setTimeout(t, 0)
          try {
            return r(t, 0)
          } catch (e) {
            try {
              return r.call(null, t, 0)
            } catch (e) {
              return r.call(this, t, 0)
            }
          }
        }
        !(function () {
          try {
            r = 'function' == typeof setTimeout ? setTimeout : o;
          } catch (t) {
            r = o;
          }
          try {
            n = 'function' == typeof clearTimeout ? clearTimeout : s;
          } catch (t) {
            n = s;
          }
        })();
        var a,
          f = [],
          h = !1,
          l = -1;
        function c() {
          h && a && ((h = !1), a.length ? (f = a.concat(f)) : (l = -1), f.length && p());
        }
        function p() {
          if (!h) {
            var t = u(c);
            h = !0;
            for (var e = f.length; e; ) {
              for (a = f, f = []; ++l < e; ) a && a[l].run()
              ;(l = -1), (e = f.length);
            }
  (a = null),
              (h = !1),
              (function (t) {
                if (n === clearTimeout) return clearTimeout(t)
                if ((n === s || !n) && clearTimeout) return (n = clearTimeout), clearTimeout(t)
                try {
                  n(t);
                } catch (e) {
                  try {
                    return n.call(null, t)
                  } catch (e) {
                    return n.call(this, t)
                  }
                }
              })(t);
          }
        }
        function g(t, e) {
  (this.fun = t), (this.array = e);
        }
        function d() {}
  (i.nextTick = function (t) {
          var e = new Array(arguments.length - 1);
          if (arguments.length > 1) for (var r = 1; r < arguments.length; r++) e[r - 1] = arguments[r];
          f.push(new g(t, e)), 1 !== f.length || h || u(p);
        }),
          (g.prototype.run = function () {
            this.fun.apply(null, this.array);
          }),
          (i.title = 'browser'),
          (i.browser = !0),
          (i.env = {}),
          (i.argv = []),
          (i.version = ''),
          (i.versions = {}),
          (i.on = d),
          (i.addListener = d),
          (i.once = d),
          (i.off = d),
          (i.removeListener = d),
          (i.removeAllListeners = d),
          (i.emit = d),
          (i.prependListener = d),
          (i.prependOnceListener = d),
          (i.listeners = function (t) {
            return []
          }),
          (i.binding = function (t) {
            throw new Error('process.binding is not supported')
          }),
          (i.cwd = function () {
            return '/'
          }),
          (i.chdir = function (t) {
            throw new Error('process.chdir is not supported')
          }),
          (i.umask = function () {
            return 0
          });
      },
    ])
  });
  //-------------------------------------------------------------

  const unpackBridge = WorkerScope.unpackBridge;

  let unpack;

  //-------------------------unpack--------------------------
  var initunpack = function (buffer) {
    // The following code has been _carefully_ modified by hand.
    // Due to WebPack embedding this script into the Zea engine
    // build, certain features broke.
    // There was code to handle loading in a nodeJS context, that tried to import("fs")
    // WebPack kepts tripping up on that code in its static analysis of the code, so
    // I carefully removed it.
    // The global scope of the script seems to be different, so unpackBridge was not available.
    // The unpackBridge code assigns unpackBridge to the passed in scope, which is 'this', but that
    // scope isn't available inside this 'unpack' function.
    const unpack = Module;

    // Note: the following is the URL of the unpack.wasm file in our ZeaEngine project on our
    // server. Ideally we could use a relative path from the ZeaEngine file, but
    // that isn't possible yet. (TODO: Ask Mauro about this)
    const credentials = 'omit';

    var Ext = unpackBridge.Ext;
    var jsAPI = {
      open: function () {
        return Ext.current.open.apply(Ext.current, arguments)
      },
      close: function () {
        return Ext.current.close.apply(Ext.current, arguments)
      },
      read: function () {
        return Ext.current.read.apply(Ext.current, arguments)
      },
      write: function () {
        return Ext.current.write.apply(Ext.current, arguments)
      },
      tell: function () {
        return Ext.current.tell.apply(Ext.current, arguments)
      },
      seek: function () {
        return Ext.current.seek.apply(Ext.current, arguments)
      },
      create: function () {
        return Ext.current.create.apply(Ext.current, arguments)
      },
    };
    var moduleOverrides = {};
    var key;
    for (key in Module) {
      if (Module.hasOwnProperty(key)) {
        moduleOverrides[key] = Module[key];
      }
    }
    Module['wasmBinary'] = buffer;
    Module['arguments'] = [];
    Module['thisProgram'] = './this.program';
    Module['quit'] = function (status, toThrow) {
      throw toThrow
    };
    Module['preRun'] = [];
    Module['postRun'] = [];
    var ENVIRONMENT_IS_WEB = false;
    var ENVIRONMENT_IS_WORKER = false;
    var ENVIRONMENT_IS_NODE = false;
    if (Module['ENVIRONMENT']) {
      if (Module['ENVIRONMENT'] === 'WEB') {
        ENVIRONMENT_IS_WEB = true;
      } else if (Module['ENVIRONMENT'] === 'WORKER') {
        ENVIRONMENT_IS_WORKER = true;
      } else if (Module['ENVIRONMENT'] === 'NODE') {
        ENVIRONMENT_IS_NODE = true;
      } else if (Module['ENVIRONMENT'] === 'SHELL') ; else {
        throw new Error("Module['ENVIRONMENT'] value is not valid. must be one of: WEB|WORKER|NODE|SHELL.")
      }
    } else {
      ENVIRONMENT_IS_WEB = typeof window === 'object';
      ENVIRONMENT_IS_WORKER = typeof importScripts === 'function';
      ENVIRONMENT_IS_NODE =
        typeof process === 'object' && typeof require === 'function' && !ENVIRONMENT_IS_WEB && !ENVIRONMENT_IS_WORKER;
    }
    if (ENVIRONMENT_IS_WEB || ENVIRONMENT_IS_WORKER) {
      Module['read'] = function shell_read(url) {
        var xhr = new XMLHttpRequest();
        xhr.open('GET', url, false);
        xhr.send(null);
        return xhr.responseText
      };
      if (ENVIRONMENT_IS_WORKER) {
        Module['readBinary'] = function readBinary(url) {
          var xhr = new XMLHttpRequest();
          xhr.open('GET', url, false);
          xhr.responseType = 'arraybuffer';
          xhr.send(null);
          return new Uint8Array(xhr.response)
        };
      }
      Module['readAsync'] = function readAsync(url, onload, onerror) {
        var xhr = new XMLHttpRequest();
        xhr.open('GET', url, true);
        xhr.responseType = 'arraybuffer';
        xhr.onload = function xhr_onload() {
          if (xhr.status == 200 || (xhr.status == 0 && xhr.response)) {
            onload(xhr.response);
            return
          }
          onerror();
        };
        xhr.onerror = onerror;
        xhr.send(null);
      };
      Module['setWindowTitle'] = function (title) {
        document.title = title;
      };
    }
    Module['print'] =
      typeof console !== 'undefined' ? console.log.bind(console) : typeof print !== 'undefined' ? print : null;
    Module['printErr'] =
      typeof printErr !== 'undefined'
        ? printErr
        : (typeof console !== 'undefined' && console.warn.bind(console)) || Module['print'];
    Module.print = Module['print'];
    Module.printErr = Module['printErr'];
    for (key in moduleOverrides) {
      if (moduleOverrides.hasOwnProperty(key)) {
        Module[key] = moduleOverrides[key];
      }
    }
    moduleOverrides = undefined;
    var STACK_ALIGN = 16;
    function staticAlloc(size) {
      assert(!staticSealed);
      var ret = STATICTOP;
      STATICTOP = (STATICTOP + size + 15) & -16;
      return ret
    }
    function dynamicAlloc(size) {
      assert(DYNAMICTOP_PTR);
      var ret = HEAP32[DYNAMICTOP_PTR >> 2];
      var end = (ret + size + 15) & -16;
      HEAP32[DYNAMICTOP_PTR >> 2] = end;
      if (end >= TOTAL_MEMORY) {
        var success = enlargeMemory();
        if (!success) {
          HEAP32[DYNAMICTOP_PTR >> 2] = ret;
          return 0
        }
      }
      return ret
    }
    function alignMemory(size, factor) {
      if (!factor) factor = STACK_ALIGN;
      var ret = (size = Math.ceil(size / factor) * factor);
      return ret
    }
    function getNativeTypeSize(type) {
      switch (type) {
        case 'i1':
        case 'i8':
          return 1
        case 'i16':
          return 2
        case 'i32':
          return 4
        case 'i64':
          return 8
        case 'float':
          return 4
        case 'double':
          return 8
        default: {
          if (type[type.length - 1] === '*') {
            return 4
          } else if (type[0] === 'i') {
            var bits = parseInt(type.substr(1));
            assert(bits % 8 === 0);
            return bits / 8
          } else {
            return 0
          }
        }
      }
    }
    new Array(0);
    var GLOBAL_BASE = 1024;
    var ABORT = 0;
    function assert(condition, text) {
      if (!condition) {
        abort('Assertion failed: ' + text);
      }
    }
    function setValue(ptr, value, type, noSafe) {
      type = type || 'i8';
      if (type.charAt(type.length - 1) === '*') type = 'i32';
      switch (type) {
        case 'i1':
          HEAP8[ptr >> 0] = value;
          break
        case 'i8':
          HEAP8[ptr >> 0] = value;
          break
        case 'i16':
          HEAP16[ptr >> 1] = value;
          break
        case 'i32':
          HEAP32[ptr >> 2] = value;
          break
        case 'i64':
  (tempI64 = [
            value >>> 0,
            ((tempDouble = value),
            +Math_abs(tempDouble) >= 1
              ? tempDouble > 0
                ? (Math_min(+Math_floor(tempDouble / 4294967296), 4294967295) | 0) >>> 0
                : ~~+Math_ceil((tempDouble - +(~~tempDouble >>> 0)) / 4294967296) >>> 0
              : 0),
          ]),
            (HEAP32[ptr >> 2] = tempI64[0]),
            (HEAP32[(ptr + 4) >> 2] = tempI64[1]);
          break
        case 'float':
          HEAPF32[ptr >> 2] = value;
          break
        case 'double':
          HEAPF64[ptr >> 3] = value;
          break
        default:
          abort('invalid type for setValue: ' + type);
      }
    }
    var ALLOC_NORMAL = 0;
    var ALLOC_STATIC = 2;
    var ALLOC_NONE = 4;
    function allocate(slab, types, allocator, ptr) {
      var zeroinit, size;
      if (typeof slab === 'number') {
        zeroinit = true;
        size = slab;
      } else {
        zeroinit = false;
        size = slab.length;
      }
      var singleType = typeof types === 'string' ? types : null;
      var ret;
      if (allocator == ALLOC_NONE) {
        ret = ptr;
      } else {
        ret = [typeof _malloc === 'function' ? _malloc : staticAlloc, stackAlloc, staticAlloc, dynamicAlloc][
          allocator === undefined ? ALLOC_STATIC : allocator
        ](Math.max(size, singleType ? 1 : types.length));
      }
      if (zeroinit) {
        var stop;
        ptr = ret;
        assert((ret & 3) == 0);
        stop = ret + (size & ~3);
        for (; ptr < stop; ptr += 4) {
          HEAP32[ptr >> 2] = 0;
        }
        stop = ret + size;
        while (ptr < stop) {
          HEAP8[ptr++ >> 0] = 0;
        }
        return ret
      }
      if (singleType === 'i8') {
        if (slab.subarray || slab.slice) {
          HEAPU8.set(slab, ret);
        } else {
          HEAPU8.set(new Uint8Array(slab), ret);
        }
        return ret
      }
      var i = 0,
        type,
        typeSize,
        previousType;
      while (i < size) {
        var curr = slab[i];
        type = singleType || types[i];
        if (type === 0) {
          i++;
          continue
        }
        if (type == 'i64') type = 'i32';
        setValue(ret + i, curr, type);
        if (previousType !== type) {
          typeSize = getNativeTypeSize(type);
          previousType = type;
        }
        i += typeSize;
      }
      return ret
    }
    function Pointer_stringify(ptr, length) {
      if (length === 0 || !ptr) return ''
      var hasUtf = 0;
      var t;
      var i = 0;
      while (1) {
        t = HEAPU8[(ptr + i) >> 0];
        hasUtf |= t;
        if (t == 0 && !length) break
        i++;
        if (length && i == length) break
      }
      if (!length) length = i;
      var ret = '';
      if (hasUtf < 128) {
        var MAX_CHUNK = 1024;
        var curr;
        while (length > 0) {
          curr = String.fromCharCode.apply(String, HEAPU8.subarray(ptr, ptr + Math.min(length, MAX_CHUNK)));
          ret = ret ? ret + curr : curr;
          ptr += MAX_CHUNK;
          length -= MAX_CHUNK;
        }
        return ret
      }
      return UTF8ToString(ptr)
    }
    var UTF8Decoder = typeof TextDecoder !== 'undefined' ? new TextDecoder('utf8') : undefined;
    function UTF8ArrayToString(u8Array, idx) {
      var endPtr = idx;
      while (u8Array[endPtr]) ++endPtr;
      if (endPtr - idx > 16 && u8Array.subarray && UTF8Decoder) {
        return UTF8Decoder.decode(u8Array.subarray(idx, endPtr))
      } else {
        var u0, u1, u2, u3, u4, u5;
        var str = '';
        while (1) {
          u0 = u8Array[idx++];
          if (!u0) return str
          if (!(u0 & 128)) {
            str += String.fromCharCode(u0);
            continue
          }
          u1 = u8Array[idx++] & 63;
          if ((u0 & 224) == 192) {
            str += String.fromCharCode(((u0 & 31) << 6) | u1);
            continue
          }
          u2 = u8Array[idx++] & 63;
          if ((u0 & 240) == 224) {
            u0 = ((u0 & 15) << 12) | (u1 << 6) | u2;
          } else {
            u3 = u8Array[idx++] & 63;
            if ((u0 & 248) == 240) {
              u0 = ((u0 & 7) << 18) | (u1 << 12) | (u2 << 6) | u3;
            } else {
              u4 = u8Array[idx++] & 63;
              if ((u0 & 252) == 248) {
                u0 = ((u0 & 3) << 24) | (u1 << 18) | (u2 << 12) | (u3 << 6) | u4;
              } else {
                u5 = u8Array[idx++] & 63;
                u0 = ((u0 & 1) << 30) | (u1 << 24) | (u2 << 18) | (u3 << 12) | (u4 << 6) | u5;
              }
            }
          }
          if (u0 < 65536) {
            str += String.fromCharCode(u0);
          } else {
            var ch = u0 - 65536;
            str += String.fromCharCode(55296 | (ch >> 10), 56320 | (ch & 1023));
          }
        }
      }
    }
    function UTF8ToString(ptr) {
      return UTF8ArrayToString(HEAPU8, ptr)
    }
    function stringToUTF8Array(str, outU8Array, outIdx, maxBytesToWrite) {
      if (!(maxBytesToWrite > 0)) return 0
      var startIdx = outIdx;
      var endIdx = outIdx + maxBytesToWrite - 1;
      for (var i = 0; i < str.length; ++i) {
        var u = str.charCodeAt(i);
        if (u >= 55296 && u <= 57343) u = (65536 + ((u & 1023) << 10)) | (str.charCodeAt(++i) & 1023);
        if (u <= 127) {
          if (outIdx >= endIdx) break
          outU8Array[outIdx++] = u;
        } else if (u <= 2047) {
          if (outIdx + 1 >= endIdx) break
          outU8Array[outIdx++] = 192 | (u >> 6);
          outU8Array[outIdx++] = 128 | (u & 63);
        } else if (u <= 65535) {
          if (outIdx + 2 >= endIdx) break
          outU8Array[outIdx++] = 224 | (u >> 12);
          outU8Array[outIdx++] = 128 | ((u >> 6) & 63);
          outU8Array[outIdx++] = 128 | (u & 63);
        } else if (u <= 2097151) {
          if (outIdx + 3 >= endIdx) break
          outU8Array[outIdx++] = 240 | (u >> 18);
          outU8Array[outIdx++] = 128 | ((u >> 12) & 63);
          outU8Array[outIdx++] = 128 | ((u >> 6) & 63);
          outU8Array[outIdx++] = 128 | (u & 63);
        } else if (u <= 67108863) {
          if (outIdx + 4 >= endIdx) break
          outU8Array[outIdx++] = 248 | (u >> 24);
          outU8Array[outIdx++] = 128 | ((u >> 18) & 63);
          outU8Array[outIdx++] = 128 | ((u >> 12) & 63);
          outU8Array[outIdx++] = 128 | ((u >> 6) & 63);
          outU8Array[outIdx++] = 128 | (u & 63);
        } else {
          if (outIdx + 5 >= endIdx) break
          outU8Array[outIdx++] = 252 | (u >> 30);
          outU8Array[outIdx++] = 128 | ((u >> 24) & 63);
          outU8Array[outIdx++] = 128 | ((u >> 18) & 63);
          outU8Array[outIdx++] = 128 | ((u >> 12) & 63);
          outU8Array[outIdx++] = 128 | ((u >> 6) & 63);
          outU8Array[outIdx++] = 128 | (u & 63);
        }
      }
      outU8Array[outIdx] = 0;
      return outIdx - startIdx
    }
    function stringToUTF8(str, outPtr, maxBytesToWrite) {
      return stringToUTF8Array(str, HEAPU8, outPtr, maxBytesToWrite)
    }
    function lengthBytesUTF8(str) {
      var len = 0;
      for (var i = 0; i < str.length; ++i) {
        var u = str.charCodeAt(i);
        if (u >= 55296 && u <= 57343) u = (65536 + ((u & 1023) << 10)) | (str.charCodeAt(++i) & 1023);
        if (u <= 127) {
          ++len;
        } else if (u <= 2047) {
          len += 2;
        } else if (u <= 65535) {
          len += 3;
        } else if (u <= 2097151) {
          len += 4;
        } else if (u <= 67108863) {
          len += 5;
        } else {
          len += 6;
        }
      }
      return len
    }
    typeof TextDecoder !== 'undefined' ? new TextDecoder('utf-16le') : undefined;
    function UTF32ToString(ptr) {
      var i = 0;
      var str = '';
      while (1) {
        var utf32 = HEAP32[(ptr + i * 4) >> 2];
        if (utf32 == 0) return str
        ++i;
        if (utf32 >= 65536) {
          var ch = utf32 - 65536;
          str += String.fromCharCode(55296 | (ch >> 10), 56320 | (ch & 1023));
        } else {
          str += String.fromCharCode(utf32);
        }
      }
    }
    function allocateUTF8(str) {
      var size = lengthBytesUTF8(str) + 1;
      var ret = _malloc(size);
      if (ret) stringToUTF8Array(str, HEAP8, ret, size);
      return ret
    }
    function demangle(func) {
      return func
    }
    function demangleAll(text) {
      var regex = /__Z[\w\d_]+/g;
      return text.replace(regex, function (x) {
        var y = demangle(x);
        return x === y ? x : x + ' [' + y + ']'
      })
    }
    function jsStackTrace() {
      var err = new Error();
      if (!err.stack) {
        try {
          throw new Error(0)
        } catch (e) {
          err = e;
        }
        if (!err.stack) {
          return '(no stack trace available)'
        }
      }
      return err.stack.toString()
    }
    function stackTrace() {
      var js = jsStackTrace();
      if (Module['extraStackTrace']) js += '\n' + Module['extraStackTrace']();
      return demangleAll(js)
    }
    var WASM_PAGE_SIZE = 65536;
    var ASMJS_PAGE_SIZE = 16777216;
    var MIN_TOTAL_MEMORY = 16777216;
    function alignUp(x, multiple) {
      if (x % multiple > 0) {
        x += multiple - (x % multiple);
      }
      return x
    }
    var buffer, HEAP8, HEAPU8, HEAP16, HEAPU16, HEAP32, HEAPU32, HEAPF32, HEAPF64;
    function updateGlobalBuffer(buf) {
      Module['buffer'] = buffer = buf;
    }
    function updateGlobalBufferViews() {
      Module['HEAP8'] = HEAP8 = new Int8Array(buffer);
      Module['HEAP16'] = HEAP16 = new Int16Array(buffer);
      Module['HEAP32'] = HEAP32 = new Int32Array(buffer);
      Module['HEAPU8'] = HEAPU8 = new Uint8Array(buffer);
      Module['HEAPU16'] = HEAPU16 = new Uint16Array(buffer);
      Module['HEAPU32'] = HEAPU32 = new Uint32Array(buffer);
      Module['HEAPF32'] = HEAPF32 = new Float32Array(buffer);
      Module['HEAPF64'] = HEAPF64 = new Float64Array(buffer);
    }
    var STATIC_BASE, STATICTOP, staticSealed;
    var STACK_BASE, STACKTOP, STACK_MAX;
    var DYNAMIC_BASE, DYNAMICTOP_PTR;
    STATIC_BASE = STATICTOP = STACK_BASE = STACKTOP = STACK_MAX = DYNAMIC_BASE = DYNAMICTOP_PTR = 0;
    staticSealed = false;
    function abortOnCannotGrowMemory() {
      abort(
        'Cannot enlarge memory arrays. Either (1) compile with  -s TOTAL_MEMORY=X  with X higher than the current value ' +
          TOTAL_MEMORY +
          ', (2) compile with  -s ALLOW_MEMORY_GROWTH=1  which allows increasing the size at runtime, or (3) if you want malloc to return NULL (0) instead of this abort, compile with  -s ABORTING_MALLOC=0 '
      );
    }
    if (!Module['reallocBuffer'])
      Module['reallocBuffer'] = function (size) {
        var ret;
        try {
          if (ArrayBuffer.transfer) {
            ret = ArrayBuffer.transfer(buffer, size);
          } else {
            var oldHEAP8 = HEAP8;
            ret = new ArrayBuffer(size);
            var temp = new Int8Array(ret);
            temp.set(oldHEAP8);
          }
        } catch (e) {
          return false
        }
        var success = _emscripten_replace_memory(ret);
        if (!success) return false
        return ret
      };
    function enlargeMemory() {
      var PAGE_MULTIPLE = Module['usingWasm'] ? WASM_PAGE_SIZE : ASMJS_PAGE_SIZE;
      var LIMIT = 2147483648 - PAGE_MULTIPLE;
      if (HEAP32[DYNAMICTOP_PTR >> 2] > LIMIT) {
        return false
      }
      var OLD_TOTAL_MEMORY = TOTAL_MEMORY;
      TOTAL_MEMORY = Math.max(TOTAL_MEMORY, MIN_TOTAL_MEMORY);
      while (TOTAL_MEMORY < HEAP32[DYNAMICTOP_PTR >> 2]) {
        if (TOTAL_MEMORY <= 536870912) {
          TOTAL_MEMORY = alignUp(2 * TOTAL_MEMORY, PAGE_MULTIPLE);
        } else {
          TOTAL_MEMORY = Math.min(alignUp((3 * TOTAL_MEMORY + 2147483648) / 4, PAGE_MULTIPLE), LIMIT);
        }
      }
      var replacement = Module['reallocBuffer'](TOTAL_MEMORY);
      if (!replacement || replacement.byteLength != TOTAL_MEMORY) {
        TOTAL_MEMORY = OLD_TOTAL_MEMORY;
        return false
      }
      updateGlobalBuffer(replacement);
      updateGlobalBufferViews();
      return true
    }
    var byteLength;
    try {
      byteLength = Function.prototype.call.bind(Object.getOwnPropertyDescriptor(ArrayBuffer.prototype, 'byteLength').get);
      byteLength(new ArrayBuffer(4));
    } catch (e) {
      byteLength = function (buffer) {
        return buffer.byteLength
      };
    }
    var TOTAL_STACK = Module['TOTAL_STACK'] || 5242880;
    var TOTAL_MEMORY = Module['TOTAL_MEMORY'] || 16777216;
    if (TOTAL_MEMORY < TOTAL_STACK)
      Module.printErr(
        'TOTAL_MEMORY should be larger than TOTAL_STACK, was ' + TOTAL_MEMORY + '! (TOTAL_STACK=' + TOTAL_STACK + ')'
      );
    if (Module['buffer']) {
      buffer = Module['buffer'];
    } else {
      if (typeof WebAssembly === 'object' && typeof WebAssembly.Memory === 'function') {
        Module['wasmMemory'] = new WebAssembly.Memory({ initial: TOTAL_MEMORY / WASM_PAGE_SIZE });
        buffer = Module['wasmMemory'].buffer;
      } else {
        buffer = new ArrayBuffer(TOTAL_MEMORY);
      }
      Module['buffer'] = buffer;
    }
    updateGlobalBufferViews();
    function getTotalMemory() {
      return TOTAL_MEMORY
    }
    HEAP32[0] = 1668509029;
    HEAP16[1] = 25459;
    if (HEAPU8[2] !== 115 || HEAPU8[3] !== 99) throw 'Runtime error: expected the system to be little-endian!'
    function callRuntimeCallbacks(callbacks) {
      while (callbacks.length > 0) {
        var callback = callbacks.shift();
        if (typeof callback == 'function') {
          callback();
          continue
        }
        var func = callback.func;
        if (typeof func === 'number') {
          if (callback.arg === undefined) {
            Module['dynCall_v'](func);
          } else {
            Module['dynCall_vi'](func, callback.arg);
          }
        } else {
          func(callback.arg === undefined ? null : callback.arg);
        }
      }
    }
    var __ATPRERUN__ = [];
    var __ATINIT__ = [];
    var __ATMAIN__ = [];
    var __ATEXIT__ = [];
    var __ATPOSTRUN__ = [];
    var runtimeInitialized = false;
    function preRun() {
      if (Module['preRun']) {
        if (typeof Module['preRun'] == 'function') Module['preRun'] = [Module['preRun']];
        while (Module['preRun'].length) {
          addOnPreRun(Module['preRun'].shift());
        }
      }
      callRuntimeCallbacks(__ATPRERUN__);
    }
    function ensureInitRuntime() {
      if (runtimeInitialized) return
      runtimeInitialized = true;
      callRuntimeCallbacks(__ATINIT__);
    }
    function preMain() {
      callRuntimeCallbacks(__ATMAIN__);
    }
    function exitRuntime() {
      callRuntimeCallbacks(__ATEXIT__);
    }
    function postRun() {
      if (Module['postRun']) {
        if (typeof Module['postRun'] == 'function') Module['postRun'] = [Module['postRun']];
        while (Module['postRun'].length) {
          addOnPostRun(Module['postRun'].shift());
        }
      }
      callRuntimeCallbacks(__ATPOSTRUN__);
    }
    function addOnPreRun(cb) {
      __ATPRERUN__.unshift(cb);
    }
    function addOnPostRun(cb) {
      __ATPOSTRUN__.unshift(cb);
    }
    function writeAsciiToMemory(str, buffer, dontAddNull) {
      for (var i = 0; i < str.length; ++i) {
        HEAP8[buffer++ >> 0] = str.charCodeAt(i);
      }
      if (!dontAddNull) HEAP8[buffer >> 0] = 0;
    }
    var Math_abs = Math.abs;
    var Math_ceil = Math.ceil;
    var Math_floor = Math.floor;
    var Math_min = Math.min;
    var runDependencies = 0;
    var dependenciesFulfilled = null;
    function addRunDependency(id) {
      runDependencies++;
      if (Module['monitorRunDependencies']) {
        Module['monitorRunDependencies'](runDependencies);
      }
    }
    function removeRunDependency(id) {
      runDependencies--;
      if (Module['monitorRunDependencies']) {
        Module['monitorRunDependencies'](runDependencies);
      }
      if (runDependencies == 0) {
        if (dependenciesFulfilled) {
          var callback = dependenciesFulfilled;
          dependenciesFulfilled = null;
          callback();
        }
      }
    }
    Module['preloadedImages'] = {};
    Module['preloadedAudios'] = {};
    var dataURIPrefix = 'data:application/octet-stream;base64,';
    function isDataURI(filename) {
      return String.prototype.startsWith ? filename.startsWith(dataURIPrefix) : filename.indexOf(dataURIPrefix) === 0
    }
    function integrateWasmJS() {
      var wasmTextFile = 'unpack.wast';
      var asmjsCodeFile = 'unpack.temp.asm.js';
      if (typeof Module['locateFile'] === 'function') {
        if (!isDataURI(wasmTextFile)) {
          wasmTextFile = Module['locateFile'](wasmTextFile);
        }
        if (!isDataURI(wasmBinaryFile)) {
          wasmBinaryFile = Module['locateFile'](wasmBinaryFile);
        }
        if (!isDataURI(asmjsCodeFile)) {
          asmjsCodeFile = Module['locateFile'](asmjsCodeFile);
        }
      }
      const wasmPageSize = 64 * 1024;
      const info = {
        asm2wasm: {
          'f64-rem': function (x, y) {
            return x % y
          },
          debugger: function () {
            debugger
          },
        },
        parent: Module,
      };
      var exports = null;
      function mergeMemory(newBuffer) {
        var oldBuffer = Module['buffer'];
        if (newBuffer.byteLength < oldBuffer.byteLength) {
          Module['printErr'](
            'the new buffer in mergeMemory is smaller than the previous one. in native wasm, we should grow memory here'
          );
        }
        var oldView = new Int8Array(oldBuffer);
        var newView = new Int8Array(newBuffer);
        newView.set(oldView);
        updateGlobalBuffer(newBuffer);
        updateGlobalBufferViews();
      }
      function fixImports(imports) {
        return imports
      }
      function getBinary() {
        try {
          if (Module['wasmBinary']) {
            return new Uint8Array(Module['wasmBinary'])
          }
          if (Module['readBinary']) {
            return Module['readBinary'](wasmBinaryFile)
          } else {
            throw "on the web, we need the wasm binary to be preloaded and set on Module['wasmBinary']. emcc.py will do that for you when generating HTML (but not JS)"
          }
        } catch (err) {
          abort(err);
        }
      }
      function getBinaryPromise() {
        if (!Module['wasmBinary'] && (ENVIRONMENT_IS_WEB || ENVIRONMENT_IS_WORKER) && typeof fetch === 'function') {
          return fetch(wasmBinaryFile, { credentials })
            .then(function (response) {
              if (!response['ok']) {
                throw "failed to load wasm binary file at '" + wasmBinaryFile + "'"
              }
              return response['arrayBuffer']()
            })
            .catch(function () {
              return getBinary()
            })
        }
        return new Promise(function (resolve, reject) {
          resolve(getBinary());
        })
      }
      function doNativeWasm(global, env, providedBuffer) {
        if (typeof WebAssembly !== 'object') {
          Module['printErr']('no native wasm support detected');
          return false
        }
        if (!(Module['wasmMemory'] instanceof WebAssembly.Memory)) {
          Module['printErr']('no native wasm Memory in use');
          return false
        }
        env['memory'] = Module['wasmMemory'];
        info['global'] = { NaN: NaN, Infinity: Infinity };
        info['global.Math'] = Math;
        info['env'] = env;
        function receiveInstance(instance, module) {
          exports = instance.exports;
          if (exports.memory) mergeMemory(exports.memory);
          Module['asm'] = exports;
          Module['usingWasm'] = true;
          removeRunDependency();
        }
        addRunDependency();
        if (Module['instantiateWasm']) {
          try {
            return Module['instantiateWasm'](info, receiveInstance)
          } catch (e) {
            Module['printErr']('Module.instantiateWasm callback failed with error: ' + e);
            return false
          }
        }
        function receiveInstantiatedSource(output) {
          receiveInstance(output['instance'], output['module']);
        }
        function instantiateArrayBuffer(receiver) {
          getBinaryPromise()
            .then(function (binary) {
              return WebAssembly.instantiate(binary, info)
            })
            .then(receiver)
            .catch(function (reason) {
              Module['printErr']('failed to asynchronously prepare wasm: ' + reason);
              abort(reason);
            });
        }
        if (
          !Module['wasmBinary'] &&
          typeof WebAssembly.instantiateStreaming === 'function' &&
          !isDataURI(wasmBinaryFile) &&
          typeof fetch === 'function'
        ) {
          WebAssembly.instantiateStreaming(fetch(wasmBinaryFile, { credentials }), info)
            .then(receiveInstantiatedSource)
            .catch(function (reason) {
              Module['printErr']('wasm streaming compile failed: ' + reason);
              Module['printErr']('falling back to ArrayBuffer instantiation');
              instantiateArrayBuffer(receiveInstantiatedSource);
            });
        } else {
          instantiateArrayBuffer(receiveInstantiatedSource);
        }
        return {}
      }
      Module['asmPreload'] = Module['asm'];
      var asmjsReallocBuffer = Module['reallocBuffer'];
      var wasmReallocBuffer = function (size) {
        var PAGE_MULTIPLE = Module['usingWasm'] ? WASM_PAGE_SIZE : ASMJS_PAGE_SIZE;
        size = alignUp(size, PAGE_MULTIPLE);
        var old = Module['buffer'];
        var oldSize = old.byteLength;
        if (Module['usingWasm']) {
          try {
            var result = Module['wasmMemory'].grow((size - oldSize) / wasmPageSize);
            if (result !== (-1 | 0)) {
              return (Module['buffer'] = Module['wasmMemory'].buffer)
            } else {
              return null
            }
          } catch (e) {
            return null
          }
        }
      };
      Module['reallocBuffer'] = function (size) {
        if (finalMethod === 'asmjs') {
          return asmjsReallocBuffer(size)
        } else {
          return wasmReallocBuffer(size)
        }
      };
      var finalMethod = '';
      Module['asm'] = function (global, env, providedBuffer) {
        env = fixImports(env);
        if (!env['table']) {
          var TABLE_SIZE = Module['wasmTableSize'];
          if (TABLE_SIZE === undefined) TABLE_SIZE = 1024;
          var MAX_TABLE_SIZE = Module['wasmMaxTableSize'];
          if (typeof WebAssembly === 'object' && typeof WebAssembly.Table === 'function') {
            if (MAX_TABLE_SIZE !== undefined) {
              env['table'] = new WebAssembly.Table({ initial: TABLE_SIZE, maximum: MAX_TABLE_SIZE, element: 'anyfunc' });
            } else {
              env['table'] = new WebAssembly.Table({ initial: TABLE_SIZE, element: 'anyfunc' });
            }
          } else {
            env['table'] = new Array(TABLE_SIZE);
          }
          Module['wasmTable'] = env['table'];
        }
        if (!env['memoryBase']) {
          env['memoryBase'] = Module['STATIC_BASE'];
        }
        if (!env['tableBase']) {
          env['tableBase'] = 0;
        }
        var exports;
        exports = doNativeWasm(global, env);
        if (!exports)
          abort(
            'no binaryen method succeeded. consider enabling more options, like interpreting, if you want that: https://github.com/kripken/emscripten/wiki/WebAssembly#binaryen-methods'
          );
        return exports
      };
    }
    integrateWasmJS();
    STATIC_BASE = GLOBAL_BASE;
    STATICTOP = STATIC_BASE + 66960;
    __ATINIT__.push(
      {
        func: function () {
          __GLOBAL__sub_I_global_cpp();
        },
      },
      {
        func: function () {
          __GLOBAL__sub_I_crc_cpp();
        },
      },
      {
        func: function () {
          __GLOBAL__sub_I_bridge_cpp();
        },
      },
      {
        func: function () {
          __GLOBAL__sub_I_bind_cpp();
        },
      }
    );
    var STATIC_BUMP = 66960;
    Module['STATIC_BASE'] = STATIC_BASE;
    Module['STATIC_BUMP'] = STATIC_BUMP;
    STATICTOP += 16;
    function ___cxa_allocate_exception(size) {
      return _malloc(size)
    }
    function __ZSt18uncaught_exceptionv() {
      return !!__ZSt18uncaught_exceptionv.uncaught_exception
    }
    var EXCEPTIONS = {
      last: 0,
      caught: [],
      infos: {},
      deAdjust: function (adjusted) {
        if (!adjusted || EXCEPTIONS.infos[adjusted]) return adjusted
        for (var key in EXCEPTIONS.infos) {
          var ptr = +key;
          var info = EXCEPTIONS.infos[ptr];
          if (info.adjusted === adjusted) {
            return ptr
          }
        }
        return adjusted
      },
      addRef: function (ptr) {
        if (!ptr) return
        var info = EXCEPTIONS.infos[ptr];
        info.refcount++;
      },
      decRef: function (ptr) {
        if (!ptr) return
        var info = EXCEPTIONS.infos[ptr];
        assert(info.refcount > 0);
        info.refcount--;
        if (info.refcount === 0 && !info.rethrown) {
          if (info.destructor) {
            Module['dynCall_vi'](info.destructor, ptr);
          }
          delete EXCEPTIONS.infos[ptr];
          ___cxa_free_exception(ptr);
        }
      },
      clearRef: function (ptr) {
        if (!ptr) return
        var info = EXCEPTIONS.infos[ptr];
        info.refcount = 0;
      },
    };
    function ___cxa_begin_catch(ptr) {
      var info = EXCEPTIONS.infos[ptr];
      if (info && !info.caught) {
        info.caught = true;
        __ZSt18uncaught_exceptionv.uncaught_exception--;
      }
      if (info) info.rethrown = false;
      EXCEPTIONS.caught.push(ptr);
      EXCEPTIONS.addRef(EXCEPTIONS.deAdjust(ptr));
      return ptr
    }
    function ___cxa_free_exception(ptr) {
      try {
        return _free(ptr)
      } catch (e) {}
    }
    function ___cxa_end_catch() {
      Module['setThrew'](0);
      var ptr = EXCEPTIONS.caught.pop();
      if (ptr) {
        EXCEPTIONS.decRef(EXCEPTIONS.deAdjust(ptr));
        EXCEPTIONS.last = 0;
      }
    }
    function ___cxa_find_matching_catch_2() {
      return ___cxa_find_matching_catch.apply(null, arguments)
    }
    function ___cxa_find_matching_catch_3() {
      return ___cxa_find_matching_catch.apply(null, arguments)
    }
    function ___cxa_find_matching_catch_4() {
      return ___cxa_find_matching_catch.apply(null, arguments)
    }
    function ___resumeException(ptr) {
      if (!EXCEPTIONS.last) {
        EXCEPTIONS.last = ptr;
      }
      throw ptr
    }
    function ___cxa_find_matching_catch() {
      var thrown = EXCEPTIONS.last;
      if (!thrown) {
        return (setTempRet0(0), 0) | 0
      }
      var info = EXCEPTIONS.infos[thrown];
      var throwntype = info.type;
      if (!throwntype) {
        return (setTempRet0(0), thrown) | 0
      }
      var typeArray = Array.prototype.slice.call(arguments);
      Module['___cxa_is_pointer_type'](throwntype);
      if (!___cxa_find_matching_catch.buffer) ___cxa_find_matching_catch.buffer = _malloc(4);
      HEAP32[___cxa_find_matching_catch.buffer >> 2] = thrown;
      thrown = ___cxa_find_matching_catch.buffer;
      for (var i = 0; i < typeArray.length; i++) {
        if (typeArray[i] && Module['___cxa_can_catch'](typeArray[i], throwntype, thrown)) {
          thrown = HEAP32[thrown >> 2];
          info.adjusted = thrown;
          return (setTempRet0(typeArray[i]), thrown) | 0
        }
      }
      thrown = HEAP32[thrown >> 2];
      return (setTempRet0(throwntype), thrown) | 0
    }
    function ___cxa_throw(ptr, type, destructor) {
      EXCEPTIONS.infos[ptr] = {
        ptr: ptr,
        adjusted: ptr,
        type: type,
        destructor: destructor,
        refcount: 0,
        caught: false,
        rethrown: false,
      };
      EXCEPTIONS.last = ptr;
      if (!('uncaught_exception' in __ZSt18uncaught_exceptionv)) {
        __ZSt18uncaught_exceptionv.uncaught_exception = 1;
      } else {
        __ZSt18uncaught_exceptionv.uncaught_exception++;
      }
      throw ptr
    }
    function ___lock() {}
    var ERRNO_CODES = {
      EPERM: 1,
      ENOENT: 2,
      ESRCH: 3,
      EINTR: 4,
      EIO: 5,
      ENXIO: 6,
      E2BIG: 7,
      ENOEXEC: 8,
      EBADF: 9,
      ECHILD: 10,
      EAGAIN: 11,
      EWOULDBLOCK: 11,
      ENOMEM: 12,
      EACCES: 13,
      EFAULT: 14,
      ENOTBLK: 15,
      EBUSY: 16,
      EEXIST: 17,
      EXDEV: 18,
      ENODEV: 19,
      ENOTDIR: 20,
      EISDIR: 21,
      EINVAL: 22,
      ENFILE: 23,
      EMFILE: 24,
      ENOTTY: 25,
      ETXTBSY: 26,
      EFBIG: 27,
      ENOSPC: 28,
      ESPIPE: 29,
      EROFS: 30,
      EMLINK: 31,
      EPIPE: 32,
      EDOM: 33,
      ERANGE: 34,
      ENOMSG: 42,
      EIDRM: 43,
      ECHRNG: 44,
      EL2NSYNC: 45,
      EL3HLT: 46,
      EL3RST: 47,
      ELNRNG: 48,
      EUNATCH: 49,
      ENOCSI: 50,
      EL2HLT: 51,
      EDEADLK: 35,
      ENOLCK: 37,
      EBADE: 52,
      EBADR: 53,
      EXFULL: 54,
      ENOANO: 55,
      EBADRQC: 56,
      EBADSLT: 57,
      EDEADLOCK: 35,
      EBFONT: 59,
      ENOSTR: 60,
      ENODATA: 61,
      ETIME: 62,
      ENOSR: 63,
      ENONET: 64,
      ENOPKG: 65,
      EREMOTE: 66,
      ENOLINK: 67,
      EADV: 68,
      ESRMNT: 69,
      ECOMM: 70,
      EPROTO: 71,
      EMULTIHOP: 72,
      EDOTDOT: 73,
      EBADMSG: 74,
      ENOTUNIQ: 76,
      EBADFD: 77,
      EREMCHG: 78,
      ELIBACC: 79,
      ELIBBAD: 80,
      ELIBSCN: 81,
      ELIBMAX: 82,
      ELIBEXEC: 83,
      ENOSYS: 38,
      ENOTEMPTY: 39,
      ENAMETOOLONG: 36,
      ELOOP: 40,
      EOPNOTSUPP: 95,
      EPFNOSUPPORT: 96,
      ECONNRESET: 104,
      ENOBUFS: 105,
      EAFNOSUPPORT: 97,
      EPROTOTYPE: 91,
      ENOTSOCK: 88,
      ENOPROTOOPT: 92,
      ESHUTDOWN: 108,
      ECONNREFUSED: 111,
      EADDRINUSE: 98,
      ECONNABORTED: 103,
      ENETUNREACH: 101,
      ENETDOWN: 100,
      ETIMEDOUT: 110,
      EHOSTDOWN: 112,
      EHOSTUNREACH: 113,
      EINPROGRESS: 115,
      EALREADY: 114,
      EDESTADDRREQ: 89,
      EMSGSIZE: 90,
      EPROTONOSUPPORT: 93,
      ESOCKTNOSUPPORT: 94,
      EADDRNOTAVAIL: 99,
      ENETRESET: 102,
      EISCONN: 106,
      ENOTCONN: 107,
      ETOOMANYREFS: 109,
      EUSERS: 87,
      EDQUOT: 122,
      ESTALE: 116,
      ENOTSUP: 95,
      ENOMEDIUM: 123,
      EILSEQ: 84,
      EOVERFLOW: 75,
      ECANCELED: 125,
      ENOTRECOVERABLE: 131,
      EOWNERDEAD: 130,
      ESTRPIPE: 86,
    };
    function ___setErrNo(value) {
      if (Module['___errno_location']) HEAP32[Module['___errno_location']() >> 2] = value;
      return value
    }
    function ___map_file(pathname, size) {
      ___setErrNo(ERRNO_CODES.EPERM);
      return -1
    }
    var ERRNO_MESSAGES = {
      0: 'Success',
      1: 'Not super-user',
      2: 'No such file or directory',
      3: 'No such process',
      4: 'Interrupted system call',
      5: 'I/O error',
      6: 'No such device or address',
      7: 'Arg list too long',
      8: 'Exec format error',
      9: 'Bad file number',
      10: 'No children',
      11: 'No more processes',
      12: 'Not enough core',
      13: 'Permission denied',
      14: 'Bad address',
      15: 'Block device required',
      16: 'Mount device busy',
      17: 'File exists',
      18: 'Cross-device link',
      19: 'No such device',
      20: 'Not a directory',
      21: 'Is a directory',
      22: 'Invalid argument',
      23: 'Too many open files in system',
      24: 'Too many open files',
      25: 'Not a typewriter',
      26: 'Text file busy',
      27: 'File too large',
      28: 'No space left on device',
      29: 'Illegal seek',
      30: 'Read only file system',
      31: 'Too many links',
      32: 'Broken pipe',
      33: 'Math arg out of domain of func',
      34: 'Math result not representable',
      35: 'File locking deadlock error',
      36: 'File or path name too long',
      37: 'No record locks available',
      38: 'Function not implemented',
      39: 'Directory not empty',
      40: 'Too many symbolic links',
      42: 'No message of desired type',
      43: 'Identifier removed',
      44: 'Channel number out of range',
      45: 'Level 2 not synchronized',
      46: 'Level 3 halted',
      47: 'Level 3 reset',
      48: 'Link number out of range',
      49: 'Protocol driver not attached',
      50: 'No CSI structure available',
      51: 'Level 2 halted',
      52: 'Invalid exchange',
      53: 'Invalid request descriptor',
      54: 'Exchange full',
      55: 'No anode',
      56: 'Invalid request code',
      57: 'Invalid slot',
      59: 'Bad font file fmt',
      60: 'Device not a stream',
      61: 'No data (for no delay io)',
      62: 'Timer expired',
      63: 'Out of streams resources',
      64: 'Machine is not on the network',
      65: 'Package not installed',
      66: 'The object is remote',
      67: 'The link has been severed',
      68: 'Advertise error',
      69: 'Srmount error',
      70: 'Communication error on send',
      71: 'Protocol error',
      72: 'Multihop attempted',
      73: 'Cross mount point (not really error)',
      74: 'Trying to read unreadable message',
      75: 'Value too large for defined data type',
      76: 'Given log. name not unique',
      77: 'f.d. invalid for this operation',
      78: 'Remote address changed',
      79: 'Can   access a needed shared lib',
      80: 'Accessing a corrupted shared lib',
      81: '.lib section in a.out corrupted',
      82: 'Attempting to link in too many libs',
      83: 'Attempting to exec a shared library',
      84: 'Illegal byte sequence',
      86: 'Streams pipe error',
      87: 'Too many users',
      88: 'Socket operation on non-socket',
      89: 'Destination address required',
      90: 'Message too long',
      91: 'Protocol wrong type for socket',
      92: 'Protocol not available',
      93: 'Unknown protocol',
      94: 'Socket type not supported',
      95: 'Not supported',
      96: 'Protocol family not supported',
      97: 'Address family not supported by protocol family',
      98: 'Address already in use',
      99: 'Address not available',
      100: 'Network interface is not configured',
      101: 'Network is unreachable',
      102: 'Connection reset by network',
      103: 'Connection aborted',
      104: 'Connection reset by peer',
      105: 'No buffer space available',
      106: 'Socket is already connected',
      107: 'Socket is not connected',
      108: "Can't send after socket shutdown",
      109: 'Too many references',
      110: 'Connection timed out',
      111: 'Connection refused',
      112: 'Host is down',
      113: 'Host is unreachable',
      114: 'Socket already connected',
      115: 'Connection already in progress',
      116: 'Stale file handle',
      122: 'Quota exceeded',
      123: 'No medium (in tape drive)',
      125: 'Operation canceled',
      130: 'Previous owner died',
      131: 'State not recoverable',
    };
    var PATH = {
      splitPath: function (filename) {
        var splitPathRe = /^(\/?|)([\s\S]*?)((?:\.{1,2}|[^\/]+?|)(\.[^.\/]*|))(?:[\/]*)$/;
        return splitPathRe.exec(filename).slice(1)
      },
      normalizeArray: function (parts, allowAboveRoot) {
        var up = 0;
        for (var i = parts.length - 1; i >= 0; i--) {
          var last = parts[i];
          if (last === '.') {
            parts.splice(i, 1);
          } else if (last === '..') {
            parts.splice(i, 1);
            up++;
          } else if (up) {
            parts.splice(i, 1);
            up--;
          }
        }
        if (allowAboveRoot) {
          for (; up; up--) {
            parts.unshift('..');
          }
        }
        return parts
      },
      normalize: function (path) {
        var isAbsolute = path.charAt(0) === '/',
          trailingSlash = path.substr(-1) === '/';
        path = PATH.normalizeArray(
          path.split('/').filter(function (p) {
            return !!p
          }),
          !isAbsolute
        ).join('/');
        if (!path && !isAbsolute) {
          path = '.';
        }
        if (path && trailingSlash) {
          path += '/';
        }
        return (isAbsolute ? '/' : '') + path
      },
      dirname: function (path) {
        var result = PATH.splitPath(path),
          root = result[0],
          dir = result[1];
        if (!root && !dir) {
          return '.'
        }
        if (dir) {
          dir = dir.substr(0, dir.length - 1);
        }
        return root + dir
      },
      basename: function (path) {
        if (path === '/') return '/'
        var lastSlash = path.lastIndexOf('/');
        if (lastSlash === -1) return path
        return path.substr(lastSlash + 1)
      },
      extname: function (path) {
        return PATH.splitPath(path)[3]
      },
      join: function () {
        var paths = Array.prototype.slice.call(arguments, 0);
        return PATH.normalize(paths.join('/'))
      },
      join2: function (l, r) {
        return PATH.normalize(l + '/' + r)
      },
      resolve: function () {
        var resolvedPath = '',
          resolvedAbsolute = false;
        for (var i = arguments.length - 1; i >= -1 && !resolvedAbsolute; i--) {
          var path = i >= 0 ? arguments[i] : FS.cwd();
          if (typeof path !== 'string') {
            throw new TypeError('Arguments to path.resolve must be strings')
          } else if (!path) {
            return ''
          }
          resolvedPath = path + '/' + resolvedPath;
          resolvedAbsolute = path.charAt(0) === '/';
        }
        resolvedPath = PATH.normalizeArray(
          resolvedPath.split('/').filter(function (p) {
            return !!p
          }),
          !resolvedAbsolute
        ).join('/');
        return (resolvedAbsolute ? '/' : '') + resolvedPath || '.'
      },
      relative: function (from, to) {
        from = PATH.resolve(from).substr(1);
        to = PATH.resolve(to).substr(1);
        function trim(arr) {
          var start = 0;
          for (; start < arr.length; start++) {
            if (arr[start] !== '') break
          }
          var end = arr.length - 1;
          for (; end >= 0; end--) {
            if (arr[end] !== '') break
          }
          if (start > end) return []
          return arr.slice(start, end - start + 1)
        }
        var fromParts = trim(from.split('/'));
        var toParts = trim(to.split('/'));
        var length = Math.min(fromParts.length, toParts.length);
        var samePartsLength = length;
        for (var i = 0; i < length; i++) {
          if (fromParts[i] !== toParts[i]) {
            samePartsLength = i;
            break
          }
        }
        var outputParts = [];
        for (var i = samePartsLength; i < fromParts.length; i++) {
          outputParts.push('..');
        }
        outputParts = outputParts.concat(toParts.slice(samePartsLength));
        return outputParts.join('/')
      },
    };
    var TTY = {
      ttys: [],
      init: function () {},
      shutdown: function () {},
      register: function (dev, ops) {
        TTY.ttys[dev] = { input: [], output: [], ops: ops };
        FS.registerDevice(dev, TTY.stream_ops);
      },
      stream_ops: {
        open: function (stream) {
          var tty = TTY.ttys[stream.node.rdev];
          if (!tty) {
            throw new FS.ErrnoError(ERRNO_CODES.ENODEV)
          }
          stream.tty = tty;
          stream.seekable = false;
        },
        close: function (stream) {
          stream.tty.ops.flush(stream.tty);
        },
        flush: function (stream) {
          stream.tty.ops.flush(stream.tty);
        },
        read: function (stream, buffer, offset, length, pos) {
          if (!stream.tty || !stream.tty.ops.get_char) {
            throw new FS.ErrnoError(ERRNO_CODES.ENXIO)
          }
          var bytesRead = 0;
          for (var i = 0; i < length; i++) {
            var result;
            try {
              result = stream.tty.ops.get_char(stream.tty);
            } catch (e) {
              throw new FS.ErrnoError(ERRNO_CODES.EIO)
            }
            if (result === undefined && bytesRead === 0) {
              throw new FS.ErrnoError(ERRNO_CODES.EAGAIN)
            }
            if (result === null || result === undefined) break
            bytesRead++;
            buffer[offset + i] = result;
          }
          if (bytesRead) {
            stream.node.timestamp = Date.now();
          }
          return bytesRead
        },
        write: function (stream, buffer, offset, length, pos) {
          if (!stream.tty || !stream.tty.ops.put_char) {
            throw new FS.ErrnoError(ERRNO_CODES.ENXIO)
          }
          for (var i = 0; i < length; i++) {
            try {
              stream.tty.ops.put_char(stream.tty, buffer[offset + i]);
            } catch (e) {
              throw new FS.ErrnoError(ERRNO_CODES.EIO)
            }
          }
          if (length) {
            stream.node.timestamp = Date.now();
          }
          return i
        },
      },
      default_tty_ops: {
        get_char: function (tty) {
          if (!tty.input.length) {
            var result = null;
            if (ENVIRONMENT_IS_NODE) {
              var BUFSIZE = 256;
              var buf = new Buffer(BUFSIZE);
              var bytesRead = 0;
              var isPosixPlatform = process.platform != 'win32';
              var fd = process.stdin.fd;
              if (isPosixPlatform) {
                var usingDevice = false;
                try {
                  fd = fs.openSync('/dev/stdin', 'r');
                  usingDevice = true;
                } catch (e) {}
              }
              try {
                bytesRead = fs.readSync(fd, buf, 0, BUFSIZE, null);
              } catch (e) {
                if (e.toString().indexOf('EOF') != -1) bytesRead = 0;
                else throw e
              }
              if (usingDevice) {
                fs.closeSync(fd);
              }
              if (bytesRead > 0) {
                result = buf.slice(0, bytesRead).toString('utf-8');
              } else {
                result = null;
              }
            } else if (typeof window != 'undefined' && typeof window.prompt == 'function') {
              result = window.prompt('Input: ');
              if (result !== null) {
                result += '\n';
              }
            } else if (typeof readline == 'function') {
              result = readline();
              if (result !== null) {
                result += '\n';
              }
            }
            if (!result) {
              return null
            }
            tty.input = intArrayFromString(result, true);
          }
          return tty.input.shift()
        },
        put_char: function (tty, val) {
          if (val === null || val === 10) {
            Module['print'](UTF8ArrayToString(tty.output, 0));
            tty.output = [];
          } else {
            if (val != 0) tty.output.push(val);
          }
        },
        flush: function (tty) {
          if (tty.output && tty.output.length > 0) {
            Module['print'](UTF8ArrayToString(tty.output, 0));
            tty.output = [];
          }
        },
      },
      default_tty1_ops: {
        put_char: function (tty, val) {
          if (val === null || val === 10) {
            Module['printErr'](UTF8ArrayToString(tty.output, 0));
            tty.output = [];
          } else {
            if (val != 0) tty.output.push(val);
          }
        },
        flush: function (tty) {
          if (tty.output && tty.output.length > 0) {
            Module['printErr'](UTF8ArrayToString(tty.output, 0));
            tty.output = [];
          }
        },
      },
    };
    var MEMFS = {
      ops_table: null,
      mount: function (mount) {
        return MEMFS.createNode(null, '/', 16384 | 511, 0)
      },
      createNode: function (parent, name, mode, dev) {
        if (FS.isBlkdev(mode) || FS.isFIFO(mode)) {
          throw new FS.ErrnoError(ERRNO_CODES.EPERM)
        }
        if (!MEMFS.ops_table) {
          MEMFS.ops_table = {
            dir: {
              node: {
                getattr: MEMFS.node_ops.getattr,
                setattr: MEMFS.node_ops.setattr,
                lookup: MEMFS.node_ops.lookup,
                mknod: MEMFS.node_ops.mknod,
                rename: MEMFS.node_ops.rename,
                unlink: MEMFS.node_ops.unlink,
                rmdir: MEMFS.node_ops.rmdir,
                readdir: MEMFS.node_ops.readdir,
                symlink: MEMFS.node_ops.symlink,
              },
              stream: { llseek: MEMFS.stream_ops.llseek },
            },
            file: {
              node: { getattr: MEMFS.node_ops.getattr, setattr: MEMFS.node_ops.setattr },
              stream: {
                llseek: MEMFS.stream_ops.llseek,
                read: MEMFS.stream_ops.read,
                write: MEMFS.stream_ops.write,
                allocate: MEMFS.stream_ops.allocate,
                mmap: MEMFS.stream_ops.mmap,
                msync: MEMFS.stream_ops.msync,
              },
            },
            link: {
              node: {
                getattr: MEMFS.node_ops.getattr,
                setattr: MEMFS.node_ops.setattr,
                readlink: MEMFS.node_ops.readlink,
              },
              stream: {},
            },
            chrdev: {
              node: { getattr: MEMFS.node_ops.getattr, setattr: MEMFS.node_ops.setattr },
              stream: FS.chrdev_stream_ops,
            },
          };
        }
        var node = FS.createNode(parent, name, mode, dev);
        if (FS.isDir(node.mode)) {
          node.node_ops = MEMFS.ops_table.dir.node;
          node.stream_ops = MEMFS.ops_table.dir.stream;
          node.contents = {};
        } else if (FS.isFile(node.mode)) {
          node.node_ops = MEMFS.ops_table.file.node;
          node.stream_ops = MEMFS.ops_table.file.stream;
          node.usedBytes = 0;
          node.contents = null;
        } else if (FS.isLink(node.mode)) {
          node.node_ops = MEMFS.ops_table.link.node;
          node.stream_ops = MEMFS.ops_table.link.stream;
        } else if (FS.isChrdev(node.mode)) {
          node.node_ops = MEMFS.ops_table.chrdev.node;
          node.stream_ops = MEMFS.ops_table.chrdev.stream;
        }
        node.timestamp = Date.now();
        if (parent) {
          parent.contents[name] = node;
        }
        return node
      },
      getFileDataAsRegularArray: function (node) {
        if (node.contents && node.contents.subarray) {
          var arr = [];
          for (var i = 0; i < node.usedBytes; ++i) arr.push(node.contents[i]);
          return arr
        }
        return node.contents
      },
      getFileDataAsTypedArray: function (node) {
        if (!node.contents) return new Uint8Array()
        if (node.contents.subarray) return node.contents.subarray(0, node.usedBytes)
        return new Uint8Array(node.contents)
      },
      expandFileStorage: function (node, newCapacity) {
        if (node.contents && node.contents.subarray && newCapacity > node.contents.length) {
          node.contents = MEMFS.getFileDataAsRegularArray(node);
          node.usedBytes = node.contents.length;
        }
        if (!node.contents || node.contents.subarray) {
          var prevCapacity = node.contents ? node.contents.length : 0;
          if (prevCapacity >= newCapacity) return
          var CAPACITY_DOUBLING_MAX = 1024 * 1024;
          newCapacity = Math.max(newCapacity, (prevCapacity * (prevCapacity < CAPACITY_DOUBLING_MAX ? 2 : 1.125)) | 0);
          if (prevCapacity != 0) newCapacity = Math.max(newCapacity, 256);
          var oldContents = node.contents;
          node.contents = new Uint8Array(newCapacity);
          if (node.usedBytes > 0) node.contents.set(oldContents.subarray(0, node.usedBytes), 0);
          return
        }
        if (!node.contents && newCapacity > 0) node.contents = [];
        while (node.contents.length < newCapacity) node.contents.push(0);
      },
      resizeFileStorage: function (node, newSize) {
        if (node.usedBytes == newSize) return
        if (newSize == 0) {
          node.contents = null;
          node.usedBytes = 0;
          return
        }
        if (!node.contents || node.contents.subarray) {
          var oldContents = node.contents;
          node.contents = new Uint8Array(new ArrayBuffer(newSize));
          if (oldContents) {
            node.contents.set(oldContents.subarray(0, Math.min(newSize, node.usedBytes)));
          }
          node.usedBytes = newSize;
          return
        }
        if (!node.contents) node.contents = [];
        if (node.contents.length > newSize) node.contents.length = newSize;
        else while (node.contents.length < newSize) node.contents.push(0);
        node.usedBytes = newSize;
      },
      node_ops: {
        getattr: function (node) {
          var attr = {};
          attr.dev = FS.isChrdev(node.mode) ? node.id : 1;
          attr.ino = node.id;
          attr.mode = node.mode;
          attr.nlink = 1;
          attr.uid = 0;
          attr.gid = 0;
          attr.rdev = node.rdev;
          if (FS.isDir(node.mode)) {
            attr.size = 4096;
          } else if (FS.isFile(node.mode)) {
            attr.size = node.usedBytes;
          } else if (FS.isLink(node.mode)) {
            attr.size = node.link.length;
          } else {
            attr.size = 0;
          }
          attr.atime = new Date(node.timestamp);
          attr.mtime = new Date(node.timestamp);
          attr.ctime = new Date(node.timestamp);
          attr.blksize = 4096;
          attr.blocks = Math.ceil(attr.size / attr.blksize);
          return attr
        },
        setattr: function (node, attr) {
          if (attr.mode !== undefined) {
            node.mode = attr.mode;
          }
          if (attr.timestamp !== undefined) {
            node.timestamp = attr.timestamp;
          }
          if (attr.size !== undefined) {
            MEMFS.resizeFileStorage(node, attr.size);
          }
        },
        lookup: function (parent, name) {
          throw FS.genericErrors[ERRNO_CODES.ENOENT]
        },
        mknod: function (parent, name, mode, dev) {
          return MEMFS.createNode(parent, name, mode, dev)
        },
        rename: function (old_node, new_dir, new_name) {
          if (FS.isDir(old_node.mode)) {
            var new_node;
            try {
              new_node = FS.lookupNode(new_dir, new_name);
            } catch (e) {}
            if (new_node) {
              for (var i in new_node.contents) {
                throw new FS.ErrnoError(ERRNO_CODES.ENOTEMPTY)
              }
            }
          }
          delete old_node.parent.contents[old_node.name];
          old_node.name = new_name;
          new_dir.contents[new_name] = old_node;
          old_node.parent = new_dir;
        },
        unlink: function (parent, name) {
          delete parent.contents[name];
        },
        rmdir: function (parent, name) {
          var node = FS.lookupNode(parent, name);
          for (var i in node.contents) {
            throw new FS.ErrnoError(ERRNO_CODES.ENOTEMPTY)
          }
          delete parent.contents[name];
        },
        readdir: function (node) {
          var entries = ['.', '..'];
          for (var key in node.contents) {
            if (!node.contents.hasOwnProperty(key)) {
              continue
            }
            entries.push(key);
          }
          return entries
        },
        symlink: function (parent, newname, oldpath) {
          var node = MEMFS.createNode(parent, newname, 511 | 40960, 0);
          node.link = oldpath;
          return node
        },
        readlink: function (node) {
          if (!FS.isLink(node.mode)) {
            throw new FS.ErrnoError(ERRNO_CODES.EINVAL)
          }
          return node.link
        },
      },
      stream_ops: {
        read: function (stream, buffer, offset, length, position) {
          var contents = stream.node.contents;
          if (position >= stream.node.usedBytes) return 0
          var size = Math.min(stream.node.usedBytes - position, length);
          assert(size >= 0);
          if (size > 8 && contents.subarray) {
            buffer.set(contents.subarray(position, position + size), offset);
          } else {
            for (var i = 0; i < size; i++) buffer[offset + i] = contents[position + i];
          }
          return size
        },
        write: function (stream, buffer, offset, length, position, canOwn) {
          if (!length) return 0
          var node = stream.node;
          node.timestamp = Date.now();
          if (buffer.subarray && (!node.contents || node.contents.subarray)) {
            if (canOwn) {
              node.contents = buffer.subarray(offset, offset + length);
              node.usedBytes = length;
              return length
            } else if (node.usedBytes === 0 && position === 0) {
              node.contents = new Uint8Array(buffer.subarray(offset, offset + length));
              node.usedBytes = length;
              return length
            } else if (position + length <= node.usedBytes) {
              node.contents.set(buffer.subarray(offset, offset + length), position);
              return length
            }
          }
          MEMFS.expandFileStorage(node, position + length);
          if (node.contents.subarray && buffer.subarray)
            node.contents.set(buffer.subarray(offset, offset + length), position);
          else {
            for (var i = 0; i < length; i++) {
              node.contents[position + i] = buffer[offset + i];
            }
          }
          node.usedBytes = Math.max(node.usedBytes, position + length);
          return length
        },
        llseek: function (stream, offset, whence) {
          var position = offset;
          if (whence === 1) {
            position += stream.position;
          } else if (whence === 2) {
            if (FS.isFile(stream.node.mode)) {
              position += stream.node.usedBytes;
            }
          }
          if (position < 0) {
            throw new FS.ErrnoError(ERRNO_CODES.EINVAL)
          }
          return position
        },
        allocate: function (stream, offset, length) {
          MEMFS.expandFileStorage(stream.node, offset + length);
          stream.node.usedBytes = Math.max(stream.node.usedBytes, offset + length);
        },
        mmap: function (stream, buffer, offset, length, position, prot, flags) {
          if (!FS.isFile(stream.node.mode)) {
            throw new FS.ErrnoError(ERRNO_CODES.ENODEV)
          }
          var ptr;
          var allocated;
          var contents = stream.node.contents;
          if (!(flags & 2) && (contents.buffer === buffer || contents.buffer === buffer.buffer)) {
            allocated = false;
            ptr = contents.byteOffset;
          } else {
            if (position > 0 || position + length < stream.node.usedBytes) {
              if (contents.subarray) {
                contents = contents.subarray(position, position + length);
              } else {
                contents = Array.prototype.slice.call(contents, position, position + length);
              }
            }
            allocated = true;
            ptr = _malloc(length);
            if (!ptr) {
              throw new FS.ErrnoError(ERRNO_CODES.ENOMEM)
            }
            buffer.set(contents, ptr);
          }
          return { ptr: ptr, allocated: allocated }
        },
        msync: function (stream, buffer, offset, length, mmapFlags) {
          if (!FS.isFile(stream.node.mode)) {
            throw new FS.ErrnoError(ERRNO_CODES.ENODEV)
          }
          if (mmapFlags & 2) {
            return 0
          }
          MEMFS.stream_ops.write(stream, buffer, 0, length, offset, false);
          return 0
        },
      },
    };
    var IDBFS = {
      dbs: {},
      indexedDB: function () {
        if (typeof indexedDB !== 'undefined') return indexedDB
        var ret = null;
        if (typeof window === 'object')
          ret = window.indexedDB || window.mozIndexedDB || window.webkitIndexedDB || window.msIndexedDB;
        assert(ret, 'IDBFS used, but indexedDB not supported');
        return ret
      },
      DB_VERSION: 21,
      DB_STORE_NAME: 'FILE_DATA',
      mount: function (mount) {
        return MEMFS.mount.apply(null, arguments)
      },
      syncfs: function (mount, populate, callback) {
        IDBFS.getLocalSet(mount, function (err, local) {
          if (err) return callback(err)
          IDBFS.getRemoteSet(mount, function (err, remote) {
            if (err) return callback(err)
            var src = populate ? remote : local;
            var dst = populate ? local : remote;
            IDBFS.reconcile(src, dst, callback);
          });
        });
      },
      getDB: function (name, callback) {
        var db = IDBFS.dbs[name];
        if (db) {
          return callback(null, db)
        }
        var req;
        try {
          req = IDBFS.indexedDB().open(name, IDBFS.DB_VERSION);
        } catch (e) {
          return callback(e)
        }
        if (!req) {
          return callback('Unable to connect to IndexedDB')
        }
        req.onupgradeneeded = function (e) {
          var db = e.target.result;
          var transaction = e.target.transaction;
          var fileStore;
          if (db.objectStoreNames.contains(IDBFS.DB_STORE_NAME)) {
            fileStore = transaction.objectStore(IDBFS.DB_STORE_NAME);
          } else {
            fileStore = db.createObjectStore(IDBFS.DB_STORE_NAME);
          }
          if (!fileStore.indexNames.contains('timestamp')) {
            fileStore.createIndex('timestamp', 'timestamp', { unique: false });
          }
        };
        req.onsuccess = function () {
          db = req.result;
          IDBFS.dbs[name] = db;
          callback(null, db);
        };
        req.onerror = function (e) {
          callback(this.error);
          e.preventDefault();
        };
      },
      getLocalSet: function (mount, callback) {
        var entries = {};
        function isRealDir(p) {
          return p !== '.' && p !== '..'
        }
        function toAbsolute(root) {
          return function (p) {
            return PATH.join2(root, p)
          }
        }
        var check = FS.readdir(mount.mountpoint).filter(isRealDir).map(toAbsolute(mount.mountpoint));
        while (check.length) {
          var path = check.pop();
          var stat;
          try {
            stat = FS.stat(path);
          } catch (e) {
            return callback(e)
          }
          if (FS.isDir(stat.mode)) {
            check.push.apply(check, FS.readdir(path).filter(isRealDir).map(toAbsolute(path)));
          }
          entries[path] = { timestamp: stat.mtime };
        }
        return callback(null, { type: 'local', entries: entries })
      },
      getRemoteSet: function (mount, callback) {
        var entries = {};
        IDBFS.getDB(mount.mountpoint, function (err, db) {
          if (err) return callback(err)
          try {
            var transaction = db.transaction([IDBFS.DB_STORE_NAME], 'readonly');
            transaction.onerror = function (e) {
              callback(this.error);
              e.preventDefault();
            };
            var store = transaction.objectStore(IDBFS.DB_STORE_NAME);
            var index = store.index('timestamp');
            index.openKeyCursor().onsuccess = function (event) {
              var cursor = event.target.result;
              if (!cursor) {
                return callback(null, { type: 'remote', db: db, entries: entries })
              }
              entries[cursor.primaryKey] = { timestamp: cursor.key };
              cursor.continue();
            };
          } catch (e) {
            return callback(e)
          }
        });
      },
      loadLocalEntry: function (path, callback) {
        var stat, node;
        try {
          var lookup = FS.lookupPath(path);
          node = lookup.node;
          stat = FS.stat(path);
        } catch (e) {
          return callback(e)
        }
        if (FS.isDir(stat.mode)) {
          return callback(null, { timestamp: stat.mtime, mode: stat.mode })
        } else if (FS.isFile(stat.mode)) {
          node.contents = MEMFS.getFileDataAsTypedArray(node);
          return callback(null, { timestamp: stat.mtime, mode: stat.mode, contents: node.contents })
        } else {
          return callback(new Error('node type not supported'))
        }
      },
      storeLocalEntry: function (path, entry, callback) {
        try {
          if (FS.isDir(entry.mode)) {
            FS.mkdir(path, entry.mode);
          } else if (FS.isFile(entry.mode)) {
            FS.writeFile(path, entry.contents, { canOwn: true });
          } else {
            return callback(new Error('node type not supported'))
          }
          FS.chmod(path, entry.mode);
          FS.utime(path, entry.timestamp, entry.timestamp);
        } catch (e) {
          return callback(e)
        }
        callback(null);
      },
      removeLocalEntry: function (path, callback) {
        try {
          var lookup = FS.lookupPath(path);
          var stat = FS.stat(path);
          if (FS.isDir(stat.mode)) {
            FS.rmdir(path);
          } else if (FS.isFile(stat.mode)) {
            FS.unlink(path);
          }
        } catch (e) {
          return callback(e)
        }
        callback(null);
      },
      loadRemoteEntry: function (store, path, callback) {
        var req = store.get(path);
        req.onsuccess = function (event) {
          callback(null, event.target.result);
        };
        req.onerror = function (e) {
          callback(this.error);
          e.preventDefault();
        };
      },
      storeRemoteEntry: function (store, path, entry, callback) {
        var req = store.put(entry, path);
        req.onsuccess = function () {
          callback(null);
        };
        req.onerror = function (e) {
          callback(this.error);
          e.preventDefault();
        };
      },
      removeRemoteEntry: function (store, path, callback) {
        var req = store.delete(path);
        req.onsuccess = function () {
          callback(null);
        };
        req.onerror = function (e) {
          callback(this.error);
          e.preventDefault();
        };
      },
      reconcile: function (src, dst, callback) {
        var total = 0;
        var create = [];
        Object.keys(src.entries).forEach(function (key) {
          var e = src.entries[key];
          var e2 = dst.entries[key];
          if (!e2 || e.timestamp > e2.timestamp) {
            create.push(key);
            total++;
          }
        });
        var remove = [];
        Object.keys(dst.entries).forEach(function (key) {
          dst.entries[key];
          var e2 = src.entries[key];
          if (!e2) {
            remove.push(key);
            total++;
          }
        });
        if (!total) {
          return callback(null)
        }
        var completed = 0;
        var db = src.type === 'remote' ? src.db : dst.db;
        var transaction = db.transaction([IDBFS.DB_STORE_NAME], 'readwrite');
        var store = transaction.objectStore(IDBFS.DB_STORE_NAME);
        function done(err) {
          if (err) {
            if (!done.errored) {
              done.errored = true;
              return callback(err)
            }
            return
          }
          if (++completed >= total) {
            return callback(null)
          }
        }
        transaction.onerror = function (e) {
          done(this.error);
          e.preventDefault();
        };
        create.sort().forEach(function (path) {
          if (dst.type === 'local') {
            IDBFS.loadRemoteEntry(store, path, function (err, entry) {
              if (err) return done(err)
              IDBFS.storeLocalEntry(path, entry, done);
            });
          } else {
            IDBFS.loadLocalEntry(path, function (err, entry) {
              if (err) return done(err)
              IDBFS.storeRemoteEntry(store, path, entry, done);
            });
          }
        });
        remove
          .sort()
          .reverse()
          .forEach(function (path) {
            if (dst.type === 'local') {
              IDBFS.removeLocalEntry(path, done);
            } else {
              IDBFS.removeRemoteEntry(store, path, done);
            }
          });
      },
    };
    var NODEFS = {
      isWindows: false,
      staticInit: function () {
        NODEFS.isWindows = !!process.platform.match(/^win/);
        var flags = process['binding']('constants');
        if (flags['fs']) {
          flags = flags['fs'];
        }
        NODEFS.flagsForNodeMap = {
          1024: flags['O_APPEND'],
          64: flags['O_CREAT'],
          128: flags['O_EXCL'],
          0: flags['O_RDONLY'],
          2: flags['O_RDWR'],
          4096: flags['O_SYNC'],
          512: flags['O_TRUNC'],
          1: flags['O_WRONLY'],
        };
      },
      bufferFrom: function (arrayBuffer) {
        return Buffer.alloc ? Buffer.from(arrayBuffer) : new Buffer(arrayBuffer)
      },
      mount: function (mount) {
        assert(ENVIRONMENT_IS_NODE);
        return NODEFS.createNode(null, '/', NODEFS.getMode(mount.opts.root), 0)
      },
      createNode: function (parent, name, mode, dev) {
        if (!FS.isDir(mode) && !FS.isFile(mode) && !FS.isLink(mode)) {
          throw new FS.ErrnoError(ERRNO_CODES.EINVAL)
        }
        var node = FS.createNode(parent, name, mode);
        node.node_ops = NODEFS.node_ops;
        node.stream_ops = NODEFS.stream_ops;
        return node
      },
      getMode: function (path) {
        var stat;
        try {
          stat = fs.lstatSync(path);
          if (NODEFS.isWindows) {
            stat.mode = stat.mode | ((stat.mode & 292) >> 2);
          }
        } catch (e) {
          if (!e.code) throw e
          throw new FS.ErrnoError(ERRNO_CODES[e.code])
        }
        return stat.mode
      },
      realPath: function (node) {
        var parts = [];
        while (node.parent !== node) {
          parts.push(node.name);
          node = node.parent;
        }
        parts.push(node.mount.opts.root);
        parts.reverse();
        return PATH.join.apply(null, parts)
      },
      flagsForNode: function (flags) {
        flags &= ~2097152;
        flags &= ~2048;
        flags &= ~32768;
        flags &= ~524288;
        var newFlags = 0;
        for (var k in NODEFS.flagsForNodeMap) {
          if (flags & k) {
            newFlags |= NODEFS.flagsForNodeMap[k];
            flags ^= k;
          }
        }
        if (!flags) {
          return newFlags
        } else {
          throw new FS.ErrnoError(ERRNO_CODES.EINVAL)
        }
      },
      node_ops: {
        getattr: function (node) {
          var path = NODEFS.realPath(node);
          var stat;
          try {
            stat = fs.lstatSync(path);
          } catch (e) {
            if (!e.code) throw e
            throw new FS.ErrnoError(ERRNO_CODES[e.code])
          }
          if (NODEFS.isWindows && !stat.blksize) {
            stat.blksize = 4096;
          }
          if (NODEFS.isWindows && !stat.blocks) {
            stat.blocks = ((stat.size + stat.blksize - 1) / stat.blksize) | 0;
          }
          return {
            dev: stat.dev,
            ino: stat.ino,
            mode: stat.mode,
            nlink: stat.nlink,
            uid: stat.uid,
            gid: stat.gid,
            rdev: stat.rdev,
            size: stat.size,
            atime: stat.atime,
            mtime: stat.mtime,
            ctime: stat.ctime,
            blksize: stat.blksize,
            blocks: stat.blocks,
          }
        },
        setattr: function (node, attr) {
          var path = NODEFS.realPath(node);
          try {
            if (attr.mode !== undefined) {
              fs.chmodSync(path, attr.mode);
              node.mode = attr.mode;
            }
            if (attr.timestamp !== undefined) {
              var date = new Date(attr.timestamp);
              fs.utimesSync(path, date, date);
            }
            if (attr.size !== undefined) {
              fs.truncateSync(path, attr.size);
            }
          } catch (e) {
            if (!e.code) throw e
            throw new FS.ErrnoError(ERRNO_CODES[e.code])
          }
        },
        lookup: function (parent, name) {
          var path = PATH.join2(NODEFS.realPath(parent), name);
          var mode = NODEFS.getMode(path);
          return NODEFS.createNode(parent, name, mode)
        },
        mknod: function (parent, name, mode, dev) {
          var node = NODEFS.createNode(parent, name, mode, dev);
          var path = NODEFS.realPath(node);
          try {
            if (FS.isDir(node.mode)) {
              fs.mkdirSync(path, node.mode);
            } else {
              fs.writeFileSync(path, '', { mode: node.mode });
            }
          } catch (e) {
            if (!e.code) throw e
            throw new FS.ErrnoError(ERRNO_CODES[e.code])
          }
          return node
        },
        rename: function (oldNode, newDir, newName) {
          var oldPath = NODEFS.realPath(oldNode);
          var newPath = PATH.join2(NODEFS.realPath(newDir), newName);
          try {
            fs.renameSync(oldPath, newPath);
          } catch (e) {
            if (!e.code) throw e
            throw new FS.ErrnoError(ERRNO_CODES[e.code])
          }
        },
        unlink: function (parent, name) {
          var path = PATH.join2(NODEFS.realPath(parent), name);
          try {
            fs.unlinkSync(path);
          } catch (e) {
            if (!e.code) throw e
            throw new FS.ErrnoError(ERRNO_CODES[e.code])
          }
        },
        rmdir: function (parent, name) {
          var path = PATH.join2(NODEFS.realPath(parent), name);
          try {
            fs.rmdirSync(path);
          } catch (e) {
            if (!e.code) throw e
            throw new FS.ErrnoError(ERRNO_CODES[e.code])
          }
        },
        readdir: function (node) {
          var path = NODEFS.realPath(node);
          try {
            return fs.readdirSync(path)
          } catch (e) {
            if (!e.code) throw e
            throw new FS.ErrnoError(ERRNO_CODES[e.code])
          }
        },
        symlink: function (parent, newName, oldPath) {
          var newPath = PATH.join2(NODEFS.realPath(parent), newName);
          try {
            fs.symlinkSync(oldPath, newPath);
          } catch (e) {
            if (!e.code) throw e
            throw new FS.ErrnoError(ERRNO_CODES[e.code])
          }
        },
        readlink: function (node) {
          var path = NODEFS.realPath(node);
          try {
            path = fs.readlinkSync(path);
            path = NODEJS_PATH.relative(NODEJS_PATH.resolve(node.mount.opts.root), path);
            return path
          } catch (e) {
            if (!e.code) throw e
            throw new FS.ErrnoError(ERRNO_CODES[e.code])
          }
        },
      },
      stream_ops: {
        open: function (stream) {
          var path = NODEFS.realPath(stream.node);
          try {
            if (FS.isFile(stream.node.mode)) {
              stream.nfd = fs.openSync(path, NODEFS.flagsForNode(stream.flags));
            }
          } catch (e) {
            if (!e.code) throw e
            throw new FS.ErrnoError(ERRNO_CODES[e.code])
          }
        },
        close: function (stream) {
          try {
            if (FS.isFile(stream.node.mode) && stream.nfd) {
              fs.closeSync(stream.nfd);
            }
          } catch (e) {
            if (!e.code) throw e
            throw new FS.ErrnoError(ERRNO_CODES[e.code])
          }
        },
        read: function (stream, buffer, offset, length, position) {
          if (length === 0) return 0
          try {
            return fs.readSync(stream.nfd, NODEFS.bufferFrom(buffer.buffer), offset, length, position)
          } catch (e) {
            throw new FS.ErrnoError(ERRNO_CODES[e.code])
          }
        },
        write: function (stream, buffer, offset, length, position) {
          try {
            return fs.writeSync(stream.nfd, NODEFS.bufferFrom(buffer.buffer), offset, length, position)
          } catch (e) {
            throw new FS.ErrnoError(ERRNO_CODES[e.code])
          }
        },
        llseek: function (stream, offset, whence) {
          var position = offset;
          if (whence === 1) {
            position += stream.position;
          } else if (whence === 2) {
            if (FS.isFile(stream.node.mode)) {
              try {
                var stat = fs.fstatSync(stream.nfd);
                position += stat.size;
              } catch (e) {
                throw new FS.ErrnoError(ERRNO_CODES[e.code])
              }
            }
          }
          if (position < 0) {
            throw new FS.ErrnoError(ERRNO_CODES.EINVAL)
          }
          return position
        },
      },
    };
    var WORKERFS = {
      DIR_MODE: 16895,
      FILE_MODE: 33279,
      reader: null,
      mount: function (mount) {
        assert(ENVIRONMENT_IS_WORKER);
        if (!WORKERFS.reader) WORKERFS.reader = new FileReaderSync();
        var root = WORKERFS.createNode(null, '/', WORKERFS.DIR_MODE, 0);
        var createdParents = {};
        function ensureParent(path) {
          var parts = path.split('/');
          var parent = root;
          for (var i = 0; i < parts.length - 1; i++) {
            var curr = parts.slice(0, i + 1).join('/');
            if (!createdParents[curr]) {
              createdParents[curr] = WORKERFS.createNode(parent, parts[i], WORKERFS.DIR_MODE, 0);
            }
            parent = createdParents[curr];
          }
          return parent
        }
        function base(path) {
          var parts = path.split('/');
          return parts[parts.length - 1]
        }
        Array.prototype.forEach.call(mount.opts['files'] || [], function (file) {
          WORKERFS.createNode(
            ensureParent(file.name),
            base(file.name),
            WORKERFS.FILE_MODE,
            0,
            file,
            file.lastModifiedDate
          );
        })
        ;(mount.opts['blobs'] || []).forEach(function (obj) {
          WORKERFS.createNode(ensureParent(obj['name']), base(obj['name']), WORKERFS.FILE_MODE, 0, obj['data']);
        })
        ;(mount.opts['packages'] || []).forEach(function (pack) {
          pack['metadata'].files.forEach(function (file) {
            var name = file.filename.substr(1);
            WORKERFS.createNode(
              ensureParent(name),
              base(name),
              WORKERFS.FILE_MODE,
              0,
              pack['blob'].slice(file.start, file.end)
            );
          });
        });
        return root
      },
      createNode: function (parent, name, mode, dev, contents, mtime) {
        var node = FS.createNode(parent, name, mode);
        node.mode = mode;
        node.node_ops = WORKERFS.node_ops;
        node.stream_ops = WORKERFS.stream_ops;
        node.timestamp = (mtime || new Date()).getTime();
        assert(WORKERFS.FILE_MODE !== WORKERFS.DIR_MODE);
        if (mode === WORKERFS.FILE_MODE) {
          node.size = contents.size;
          node.contents = contents;
        } else {
          node.size = 4096;
          node.contents = {};
        }
        if (parent) {
          parent.contents[name] = node;
        }
        return node
      },
      node_ops: {
        getattr: function (node) {
          return {
            dev: 1,
            ino: undefined,
            mode: node.mode,
            nlink: 1,
            uid: 0,
            gid: 0,
            rdev: undefined,
            size: node.size,
            atime: new Date(node.timestamp),
            mtime: new Date(node.timestamp),
            ctime: new Date(node.timestamp),
            blksize: 4096,
            blocks: Math.ceil(node.size / 4096),
          }
        },
        setattr: function (node, attr) {
          if (attr.mode !== undefined) {
            node.mode = attr.mode;
          }
          if (attr.timestamp !== undefined) {
            node.timestamp = attr.timestamp;
          }
        },
        lookup: function (parent, name) {
          throw new FS.ErrnoError(ERRNO_CODES.ENOENT)
        },
        mknod: function (parent, name, mode, dev) {
          throw new FS.ErrnoError(ERRNO_CODES.EPERM)
        },
        rename: function (oldNode, newDir, newName) {
          throw new FS.ErrnoError(ERRNO_CODES.EPERM)
        },
        unlink: function (parent, name) {
          throw new FS.ErrnoError(ERRNO_CODES.EPERM)
        },
        rmdir: function (parent, name) {
          throw new FS.ErrnoError(ERRNO_CODES.EPERM)
        },
        readdir: function (node) {
          var entries = ['.', '..'];
          for (var key in node.contents) {
            if (!node.contents.hasOwnProperty(key)) {
              continue
            }
            entries.push(key);
          }
          return entries
        },
        symlink: function (parent, newName, oldPath) {
          throw new FS.ErrnoError(ERRNO_CODES.EPERM)
        },
        readlink: function (node) {
          throw new FS.ErrnoError(ERRNO_CODES.EPERM)
        },
      },
      stream_ops: {
        read: function (stream, buffer, offset, length, position) {
          if (position >= stream.node.size) return 0
          var chunk = stream.node.contents.slice(position, position + length);
          var ab = WORKERFS.reader.readAsArrayBuffer(chunk);
          buffer.set(new Uint8Array(ab), offset);
          return chunk.size
        },
        write: function (stream, buffer, offset, length, position) {
          throw new FS.ErrnoError(ERRNO_CODES.EIO)
        },
        llseek: function (stream, offset, whence) {
          var position = offset;
          if (whence === 1) {
            position += stream.position;
          } else if (whence === 2) {
            if (FS.isFile(stream.node.mode)) {
              position += stream.node.size;
            }
          }
          if (position < 0) {
            throw new FS.ErrnoError(ERRNO_CODES.EINVAL)
          }
          return position
        },
      },
    };
    STATICTOP += 16;
    STATICTOP += 16;
    STATICTOP += 16;
    var FS = {
      root: null,
      mounts: [],
      devices: {},
      streams: [],
      nextInode: 1,
      nameTable: null,
      currentPath: '/',
      initialized: false,
      ignorePermissions: true,
      trackingDelegate: {},
      tracking: { openFlags: { READ: 1, WRITE: 2 } },
      ErrnoError: null,
      genericErrors: {},
      filesystems: null,
      syncFSRequests: 0,
      handleFSError: function (e) {
        if (!(e instanceof FS.ErrnoError)) throw e + ' : ' + stackTrace()
        return ___setErrNo(e.errno)
      },
      lookupPath: function (path, opts) {
        path = PATH.resolve(FS.cwd(), path);
        opts = opts || {};
        if (!path) return { path: '', node: null }
        var defaults = { follow_mount: true, recurse_count: 0 };
        for (var key in defaults) {
          if (opts[key] === undefined) {
            opts[key] = defaults[key];
          }
        }
        if (opts.recurse_count > 8) {
          throw new FS.ErrnoError(ERRNO_CODES.ELOOP)
        }
        var parts = PATH.normalizeArray(
          path.split('/').filter(function (p) {
            return !!p
          }),
          false
        );
        var current = FS.root;
        var current_path = '/';
        for (var i = 0; i < parts.length; i++) {
          var islast = i === parts.length - 1;
          if (islast && opts.parent) {
            break
          }
          current = FS.lookupNode(current, parts[i]);
          current_path = PATH.join2(current_path, parts[i]);
          if (FS.isMountpoint(current)) {
            if (!islast || (islast && opts.follow_mount)) {
              current = current.mounted.root;
            }
          }
          if (!islast || opts.follow) {
            var count = 0;
            while (FS.isLink(current.mode)) {
              var link = FS.readlink(current_path);
              current_path = PATH.resolve(PATH.dirname(current_path), link);
              var lookup = FS.lookupPath(current_path, { recurse_count: opts.recurse_count });
              current = lookup.node;
              if (count++ > 40) {
                throw new FS.ErrnoError(ERRNO_CODES.ELOOP)
              }
            }
          }
        }
        return { path: current_path, node: current }
      },
      getPath: function (node) {
        var path;
        while (true) {
          if (FS.isRoot(node)) {
            var mount = node.mount.mountpoint;
            if (!path) return mount
            return mount[mount.length - 1] !== '/' ? mount + '/' + path : mount + path
          }
          path = path ? node.name + '/' + path : node.name;
          node = node.parent;
        }
      },
      hashName: function (parentid, name) {
        var hash = 0;
        for (var i = 0; i < name.length; i++) {
          hash = ((hash << 5) - hash + name.charCodeAt(i)) | 0;
        }
        return ((parentid + hash) >>> 0) % FS.nameTable.length
      },
      hashAddNode: function (node) {
        var hash = FS.hashName(node.parent.id, node.name);
        node.name_next = FS.nameTable[hash];
        FS.nameTable[hash] = node;
      },
      hashRemoveNode: function (node) {
        var hash = FS.hashName(node.parent.id, node.name);
        if (FS.nameTable[hash] === node) {
          FS.nameTable[hash] = node.name_next;
        } else {
          var current = FS.nameTable[hash];
          while (current) {
            if (current.name_next === node) {
              current.name_next = node.name_next;
              break
            }
            current = current.name_next;
          }
        }
      },
      lookupNode: function (parent, name) {
        var err = FS.mayLookup(parent);
        if (err) {
          throw new FS.ErrnoError(err, parent)
        }
        var hash = FS.hashName(parent.id, name);
        for (var node = FS.nameTable[hash]; node; node = node.name_next) {
          var nodeName = node.name;
          if (node.parent.id === parent.id && nodeName === name) {
            return node
          }
        }
        return FS.lookup(parent, name)
      },
      createNode: function (parent, name, mode, rdev) {
        if (!FS.FSNode) {
          FS.FSNode = function (parent, name, mode, rdev) {
            if (!parent) {
              parent = this;
            }
            this.parent = parent;
            this.mount = parent.mount;
            this.mounted = null;
            this.id = FS.nextInode++;
            this.name = name;
            this.mode = mode;
            this.node_ops = {};
            this.stream_ops = {};
            this.rdev = rdev;
          };
          FS.FSNode.prototype = {};
          var readMode = 292 | 73;
          var writeMode = 146;
          Object.defineProperties(FS.FSNode.prototype, {
            read: {
              get: function () {
                return (this.mode & readMode) === readMode
              },
              set: function (val) {
                val ? (this.mode |= readMode) : (this.mode &= ~readMode);
              },
            },
            write: {
              get: function () {
                return (this.mode & writeMode) === writeMode
              },
              set: function (val) {
                val ? (this.mode |= writeMode) : (this.mode &= ~writeMode);
              },
            },
            isFolder: {
              get: function () {
                return FS.isDir(this.mode)
              },
            },
            isDevice: {
              get: function () {
                return FS.isChrdev(this.mode)
              },
            },
          });
        }
        var node = new FS.FSNode(parent, name, mode, rdev);
        FS.hashAddNode(node);
        return node
      },
      destroyNode: function (node) {
        FS.hashRemoveNode(node);
      },
      isRoot: function (node) {
        return node === node.parent
      },
      isMountpoint: function (node) {
        return !!node.mounted
      },
      isFile: function (mode) {
        return (mode & 61440) === 32768
      },
      isDir: function (mode) {
        return (mode & 61440) === 16384
      },
      isLink: function (mode) {
        return (mode & 61440) === 40960
      },
      isChrdev: function (mode) {
        return (mode & 61440) === 8192
      },
      isBlkdev: function (mode) {
        return (mode & 61440) === 24576
      },
      isFIFO: function (mode) {
        return (mode & 61440) === 4096
      },
      isSocket: function (mode) {
        return (mode & 49152) === 49152
      },
      flagModes: {
        r: 0,
        rs: 1052672,
        'r+': 2,
        w: 577,
        wx: 705,
        xw: 705,
        'w+': 578,
        'wx+': 706,
        'xw+': 706,
        a: 1089,
        ax: 1217,
        xa: 1217,
        'a+': 1090,
        'ax+': 1218,
        'xa+': 1218,
      },
      modeStringToFlags: function (str) {
        var flags = FS.flagModes[str];
        if (typeof flags === 'undefined') {
          throw new Error('Unknown file open mode: ' + str)
        }
        return flags
      },
      flagsToPermissionString: function (flag) {
        var perms = ['r', 'w', 'rw'][flag & 3];
        if (flag & 512) {
          perms += 'w';
        }
        return perms
      },
      nodePermissions: function (node, perms) {
        if (FS.ignorePermissions) {
          return 0
        }
        if (perms.indexOf('r') !== -1 && !(node.mode & 292)) {
          return ERRNO_CODES.EACCES
        } else if (perms.indexOf('w') !== -1 && !(node.mode & 146)) {
          return ERRNO_CODES.EACCES
        } else if (perms.indexOf('x') !== -1 && !(node.mode & 73)) {
          return ERRNO_CODES.EACCES
        }
        return 0
      },
      mayLookup: function (dir) {
        var err = FS.nodePermissions(dir, 'x');
        if (err) return err
        if (!dir.node_ops.lookup) return ERRNO_CODES.EACCES
        return 0
      },
      mayCreate: function (dir, name) {
        try {
          var node = FS.lookupNode(dir, name);
          return ERRNO_CODES.EEXIST
        } catch (e) {}
        return FS.nodePermissions(dir, 'wx')
      },
      mayDelete: function (dir, name, isdir) {
        var node;
        try {
          node = FS.lookupNode(dir, name);
        } catch (e) {
          return e.errno
        }
        var err = FS.nodePermissions(dir, 'wx');
        if (err) {
          return err
        }
        if (isdir) {
          if (!FS.isDir(node.mode)) {
            return ERRNO_CODES.ENOTDIR
          }
          if (FS.isRoot(node) || FS.getPath(node) === FS.cwd()) {
            return ERRNO_CODES.EBUSY
          }
        } else {
          if (FS.isDir(node.mode)) {
            return ERRNO_CODES.EISDIR
          }
        }
        return 0
      },
      mayOpen: function (node, flags) {
        if (!node) {
          return ERRNO_CODES.ENOENT
        }
        if (FS.isLink(node.mode)) {
          return ERRNO_CODES.ELOOP
        } else if (FS.isDir(node.mode)) {
          if (FS.flagsToPermissionString(flags) !== 'r' || flags & 512) {
            return ERRNO_CODES.EISDIR
          }
        }
        return FS.nodePermissions(node, FS.flagsToPermissionString(flags))
      },
      MAX_OPEN_FDS: 4096,
      nextfd: function (fd_start, fd_end) {
        fd_start = fd_start || 0;
        fd_end = fd_end || FS.MAX_OPEN_FDS;
        for (var fd = fd_start; fd <= fd_end; fd++) {
          if (!FS.streams[fd]) {
            return fd
          }
        }
        throw new FS.ErrnoError(ERRNO_CODES.EMFILE)
      },
      getStream: function (fd) {
        return FS.streams[fd]
      },
      createStream: function (stream, fd_start, fd_end) {
        if (!FS.FSStream) {
          FS.FSStream = function () {};
          FS.FSStream.prototype = {};
          Object.defineProperties(FS.FSStream.prototype, {
            object: {
              get: function () {
                return this.node
              },
              set: function (val) {
                this.node = val;
              },
            },
            isRead: {
              get: function () {
                return (this.flags & 2097155) !== 1
              },
            },
            isWrite: {
              get: function () {
                return (this.flags & 2097155) !== 0
              },
            },
            isAppend: {
              get: function () {
                return this.flags & 1024
              },
            },
          });
        }
        var newStream = new FS.FSStream();
        for (var p in stream) {
          newStream[p] = stream[p];
        }
        stream = newStream;
        var fd = FS.nextfd(fd_start, fd_end);
        stream.fd = fd;
        FS.streams[fd] = stream;
        return stream
      },
      closeStream: function (fd) {
        FS.streams[fd] = null;
      },
      chrdev_stream_ops: {
        open: function (stream) {
          var device = FS.getDevice(stream.node.rdev);
          stream.stream_ops = device.stream_ops;
          if (stream.stream_ops.open) {
            stream.stream_ops.open(stream);
          }
        },
        llseek: function () {
          throw new FS.ErrnoError(ERRNO_CODES.ESPIPE)
        },
      },
      major: function (dev) {
        return dev >> 8
      },
      minor: function (dev) {
        return dev & 255
      },
      makedev: function (ma, mi) {
        return (ma << 8) | mi
      },
      registerDevice: function (dev, ops) {
        FS.devices[dev] = { stream_ops: ops };
      },
      getDevice: function (dev) {
        return FS.devices[dev]
      },
      getMounts: function (mount) {
        var mounts = [];
        var check = [mount];
        while (check.length) {
          var m = check.pop();
          mounts.push(m);
          check.push.apply(check, m.mounts);
        }
        return mounts
      },
      syncfs: function (populate, callback) {
        if (typeof populate === 'function') {
          callback = populate;
          populate = false;
        }
        FS.syncFSRequests++;
        if (FS.syncFSRequests > 1) {
          console.log(
            'warning: ' + FS.syncFSRequests + ' FS.syncfs operations in flight at once, probably just doing extra work'
          );
        }
        var mounts = FS.getMounts(FS.root.mount);
        var completed = 0;
        function doCallback(err) {
          assert(FS.syncFSRequests > 0);
          FS.syncFSRequests--;
          return callback(err)
        }
        function done(err) {
          if (err) {
            if (!done.errored) {
              done.errored = true;
              return doCallback(err)
            }
            return
          }
          if (++completed >= mounts.length) {
            doCallback(null);
          }
        }
        mounts.forEach(function (mount) {
          if (!mount.type.syncfs) {
            return done(null)
          }
          mount.type.syncfs(mount, populate, done);
        });
      },
      mount: function (type, opts, mountpoint) {
        var root = mountpoint === '/';
        var pseudo = !mountpoint;
        var node;
        if (root && FS.root) {
          throw new FS.ErrnoError(ERRNO_CODES.EBUSY)
        } else if (!root && !pseudo) {
          var lookup = FS.lookupPath(mountpoint, { follow_mount: false });
          mountpoint = lookup.path;
          node = lookup.node;
          if (FS.isMountpoint(node)) {
            throw new FS.ErrnoError(ERRNO_CODES.EBUSY)
          }
          if (!FS.isDir(node.mode)) {
            throw new FS.ErrnoError(ERRNO_CODES.ENOTDIR)
          }
        }
        var mount = { type: type, opts: opts, mountpoint: mountpoint, mounts: [] };
        var mountRoot = type.mount(mount);
        mountRoot.mount = mount;
        mount.root = mountRoot;
        if (root) {
          FS.root = mountRoot;
        } else if (node) {
          node.mounted = mount;
          if (node.mount) {
            node.mount.mounts.push(mount);
          }
        }
        return mountRoot
      },
      unmount: function (mountpoint) {
        var lookup = FS.lookupPath(mountpoint, { follow_mount: false });
        if (!FS.isMountpoint(lookup.node)) {
          throw new FS.ErrnoError(ERRNO_CODES.EINVAL)
        }
        var node = lookup.node;
        var mount = node.mounted;
        var mounts = FS.getMounts(mount);
        Object.keys(FS.nameTable).forEach(function (hash) {
          var current = FS.nameTable[hash];
          while (current) {
            var next = current.name_next;
            if (mounts.indexOf(current.mount) !== -1) {
              FS.destroyNode(current);
            }
            current = next;
          }
        });
        node.mounted = null;
        var idx = node.mount.mounts.indexOf(mount);
        assert(idx !== -1);
        node.mount.mounts.splice(idx, 1);
      },
      lookup: function (parent, name) {
        return parent.node_ops.lookup(parent, name)
      },
      mknod: function (path, mode, dev) {
        var lookup = FS.lookupPath(path, { parent: true });
        var parent = lookup.node;
        var name = PATH.basename(path);
        if (!name || name === '.' || name === '..') {
          throw new FS.ErrnoError(ERRNO_CODES.EINVAL)
        }
        var err = FS.mayCreate(parent, name);
        if (err) {
          throw new FS.ErrnoError(err)
        }
        if (!parent.node_ops.mknod) {
          throw new FS.ErrnoError(ERRNO_CODES.EPERM)
        }
        return parent.node_ops.mknod(parent, name, mode, dev)
      },
      create: function (path, mode) {
        mode = mode !== undefined ? mode : 438;
        mode &= 4095;
        mode |= 32768;
        return FS.mknod(path, mode, 0)
      },
      mkdir: function (path, mode) {
        mode = mode !== undefined ? mode : 511;
        mode &= 511 | 512;
        mode |= 16384;
        return FS.mknod(path, mode, 0)
      },
      mkdirTree: function (path, mode) {
        var dirs = path.split('/');
        var d = '';
        for (var i = 0; i < dirs.length; ++i) {
          if (!dirs[i]) continue
          d += '/' + dirs[i];
          try {
            FS.mkdir(d, mode);
          } catch (e) {
            if (e.errno != ERRNO_CODES.EEXIST) throw e
          }
        }
      },
      mkdev: function (path, mode, dev) {
        if (typeof dev === 'undefined') {
          dev = mode;
          mode = 438;
        }
        mode |= 8192;
        return FS.mknod(path, mode, dev)
      },
      symlink: function (oldpath, newpath) {
        if (!PATH.resolve(oldpath)) {
          throw new FS.ErrnoError(ERRNO_CODES.ENOENT)
        }
        var lookup = FS.lookupPath(newpath, { parent: true });
        var parent = lookup.node;
        if (!parent) {
          throw new FS.ErrnoError(ERRNO_CODES.ENOENT)
        }
        var newname = PATH.basename(newpath);
        var err = FS.mayCreate(parent, newname);
        if (err) {
          throw new FS.ErrnoError(err)
        }
        if (!parent.node_ops.symlink) {
          throw new FS.ErrnoError(ERRNO_CODES.EPERM)
        }
        return parent.node_ops.symlink(parent, newname, oldpath)
      },
      rename: function (old_path, new_path) {
        var old_dirname = PATH.dirname(old_path);
        var new_dirname = PATH.dirname(new_path);
        var old_name = PATH.basename(old_path);
        var new_name = PATH.basename(new_path);
        var lookup, old_dir, new_dir;
        try {
          lookup = FS.lookupPath(old_path, { parent: true });
          old_dir = lookup.node;
          lookup = FS.lookupPath(new_path, { parent: true });
          new_dir = lookup.node;
        } catch (e) {
          throw new FS.ErrnoError(ERRNO_CODES.EBUSY)
        }
        if (!old_dir || !new_dir) throw new FS.ErrnoError(ERRNO_CODES.ENOENT)
        if (old_dir.mount !== new_dir.mount) {
          throw new FS.ErrnoError(ERRNO_CODES.EXDEV)
        }
        var old_node = FS.lookupNode(old_dir, old_name);
        var relative = PATH.relative(old_path, new_dirname);
        if (relative.charAt(0) !== '.') {
          throw new FS.ErrnoError(ERRNO_CODES.EINVAL)
        }
        relative = PATH.relative(new_path, old_dirname);
        if (relative.charAt(0) !== '.') {
          throw new FS.ErrnoError(ERRNO_CODES.ENOTEMPTY)
        }
        var new_node;
        try {
          new_node = FS.lookupNode(new_dir, new_name);
        } catch (e) {}
        if (old_node === new_node) {
          return
        }
        var isdir = FS.isDir(old_node.mode);
        var err = FS.mayDelete(old_dir, old_name, isdir);
        if (err) {
          throw new FS.ErrnoError(err)
        }
        err = new_node ? FS.mayDelete(new_dir, new_name, isdir) : FS.mayCreate(new_dir, new_name);
        if (err) {
          throw new FS.ErrnoError(err)
        }
        if (!old_dir.node_ops.rename) {
          throw new FS.ErrnoError(ERRNO_CODES.EPERM)
        }
        if (FS.isMountpoint(old_node) || (new_node && FS.isMountpoint(new_node))) {
          throw new FS.ErrnoError(ERRNO_CODES.EBUSY)
        }
        if (new_dir !== old_dir) {
          err = FS.nodePermissions(old_dir, 'w');
          if (err) {
            throw new FS.ErrnoError(err)
          }
        }
        try {
          if (FS.trackingDelegate['willMovePath']) {
            FS.trackingDelegate['willMovePath'](old_path, new_path);
          }
        } catch (e) {
          console.log(
            "FS.trackingDelegate['willMovePath']('" + old_path + "', '" + new_path + "') threw an exception: " + e.message
          );
        }
        FS.hashRemoveNode(old_node);
        try {
          old_dir.node_ops.rename(old_node, new_dir, new_name);
        } catch (e) {
          throw e
        } finally {
          FS.hashAddNode(old_node);
        }
        try {
          if (FS.trackingDelegate['onMovePath']) FS.trackingDelegate['onMovePath'](old_path, new_path);
        } catch (e) {
          console.log(
            "FS.trackingDelegate['onMovePath']('" + old_path + "', '" + new_path + "') threw an exception: " + e.message
          );
        }
      },
      rmdir: function (path) {
        var lookup = FS.lookupPath(path, { parent: true });
        var parent = lookup.node;
        var name = PATH.basename(path);
        var node = FS.lookupNode(parent, name);
        var err = FS.mayDelete(parent, name, true);
        if (err) {
          throw new FS.ErrnoError(err)
        }
        if (!parent.node_ops.rmdir) {
          throw new FS.ErrnoError(ERRNO_CODES.EPERM)
        }
        if (FS.isMountpoint(node)) {
          throw new FS.ErrnoError(ERRNO_CODES.EBUSY)
        }
        try {
          if (FS.trackingDelegate['willDeletePath']) {
            FS.trackingDelegate['willDeletePath'](path);
          }
        } catch (e) {
          console.log("FS.trackingDelegate['willDeletePath']('" + path + "') threw an exception: " + e.message);
        }
        parent.node_ops.rmdir(parent, name);
        FS.destroyNode(node);
        try {
          if (FS.trackingDelegate['onDeletePath']) FS.trackingDelegate['onDeletePath'](path);
        } catch (e) {
          console.log("FS.trackingDelegate['onDeletePath']('" + path + "') threw an exception: " + e.message);
        }
      },
      readdir: function (path) {
        var lookup = FS.lookupPath(path, { follow: true });
        var node = lookup.node;
        if (!node.node_ops.readdir) {
          throw new FS.ErrnoError(ERRNO_CODES.ENOTDIR)
        }
        return node.node_ops.readdir(node)
      },
      unlink: function (path) {
        var lookup = FS.lookupPath(path, { parent: true });
        var parent = lookup.node;
        var name = PATH.basename(path);
        var node = FS.lookupNode(parent, name);
        var err = FS.mayDelete(parent, name, false);
        if (err) {
          throw new FS.ErrnoError(err)
        }
        if (!parent.node_ops.unlink) {
          throw new FS.ErrnoError(ERRNO_CODES.EPERM)
        }
        if (FS.isMountpoint(node)) {
          throw new FS.ErrnoError(ERRNO_CODES.EBUSY)
        }
        try {
          if (FS.trackingDelegate['willDeletePath']) {
            FS.trackingDelegate['willDeletePath'](path);
          }
        } catch (e) {
          console.log("FS.trackingDelegate['willDeletePath']('" + path + "') threw an exception: " + e.message);
        }
        parent.node_ops.unlink(parent, name);
        FS.destroyNode(node);
        try {
          if (FS.trackingDelegate['onDeletePath']) FS.trackingDelegate['onDeletePath'](path);
        } catch (e) {
          console.log("FS.trackingDelegate['onDeletePath']('" + path + "') threw an exception: " + e.message);
        }
      },
      readlink: function (path) {
        var lookup = FS.lookupPath(path);
        var link = lookup.node;
        if (!link) {
          throw new FS.ErrnoError(ERRNO_CODES.ENOENT)
        }
        if (!link.node_ops.readlink) {
          throw new FS.ErrnoError(ERRNO_CODES.EINVAL)
        }
        return PATH.resolve(FS.getPath(link.parent), link.node_ops.readlink(link))
      },
      stat: function (path, dontFollow) {
        var lookup = FS.lookupPath(path, { follow: !dontFollow });
        var node = lookup.node;
        if (!node) {
          throw new FS.ErrnoError(ERRNO_CODES.ENOENT)
        }
        if (!node.node_ops.getattr) {
          throw new FS.ErrnoError(ERRNO_CODES.EPERM)
        }
        return node.node_ops.getattr(node)
      },
      lstat: function (path) {
        return FS.stat(path, true)
      },
      chmod: function (path, mode, dontFollow) {
        var node;
        if (typeof path === 'string') {
          var lookup = FS.lookupPath(path, { follow: !dontFollow });
          node = lookup.node;
        } else {
          node = path;
        }
        if (!node.node_ops.setattr) {
          throw new FS.ErrnoError(ERRNO_CODES.EPERM)
        }
        node.node_ops.setattr(node, { mode: (mode & 4095) | (node.mode & ~4095), timestamp: Date.now() });
      },
      lchmod: function (path, mode) {
        FS.chmod(path, mode, true);
      },
      fchmod: function (fd, mode) {
        var stream = FS.getStream(fd);
        if (!stream) {
          throw new FS.ErrnoError(ERRNO_CODES.EBADF)
        }
        FS.chmod(stream.node, mode);
      },
      chown: function (path, uid, gid, dontFollow) {
        var node;
        if (typeof path === 'string') {
          var lookup = FS.lookupPath(path, { follow: !dontFollow });
          node = lookup.node;
        } else {
          node = path;
        }
        if (!node.node_ops.setattr) {
          throw new FS.ErrnoError(ERRNO_CODES.EPERM)
        }
        node.node_ops.setattr(node, { timestamp: Date.now() });
      },
      lchown: function (path, uid, gid) {
        FS.chown(path, uid, gid, true);
      },
      fchown: function (fd, uid, gid) {
        var stream = FS.getStream(fd);
        if (!stream) {
          throw new FS.ErrnoError(ERRNO_CODES.EBADF)
        }
        FS.chown(stream.node, uid, gid);
      },
      truncate: function (path, len) {
        if (len < 0) {
          throw new FS.ErrnoError(ERRNO_CODES.EINVAL)
        }
        var node;
        if (typeof path === 'string') {
          var lookup = FS.lookupPath(path, { follow: true });
          node = lookup.node;
        } else {
          node = path;
        }
        if (!node.node_ops.setattr) {
          throw new FS.ErrnoError(ERRNO_CODES.EPERM)
        }
        if (FS.isDir(node.mode)) {
          throw new FS.ErrnoError(ERRNO_CODES.EISDIR)
        }
        if (!FS.isFile(node.mode)) {
          throw new FS.ErrnoError(ERRNO_CODES.EINVAL)
        }
        var err = FS.nodePermissions(node, 'w');
        if (err) {
          throw new FS.ErrnoError(err)
        }
        node.node_ops.setattr(node, { size: len, timestamp: Date.now() });
      },
      ftruncate: function (fd, len) {
        var stream = FS.getStream(fd);
        if (!stream) {
          throw new FS.ErrnoError(ERRNO_CODES.EBADF)
        }
        if ((stream.flags & 2097155) === 0) {
          throw new FS.ErrnoError(ERRNO_CODES.EINVAL)
        }
        FS.truncate(stream.node, len);
      },
      utime: function (path, atime, mtime) {
        var lookup = FS.lookupPath(path, { follow: true });
        var node = lookup.node;
        node.node_ops.setattr(node, { timestamp: Math.max(atime, mtime) });
      },
      open: function (path, flags, mode, fd_start, fd_end) {
        if (path === '') {
          throw new FS.ErrnoError(ERRNO_CODES.ENOENT)
        }
        flags = typeof flags === 'string' ? FS.modeStringToFlags(flags) : flags;
        mode = typeof mode === 'undefined' ? 438 : mode;
        if (flags & 64) {
          mode = (mode & 4095) | 32768;
        } else {
          mode = 0;
        }
        var node;
        if (typeof path === 'object') {
          node = path;
        } else {
          path = PATH.normalize(path);
          try {
            var lookup = FS.lookupPath(path, { follow: !(flags & 131072) });
            node = lookup.node;
          } catch (e) {}
        }
        var created = false;
        if (flags & 64) {
          if (node) {
            if (flags & 128) {
              throw new FS.ErrnoError(ERRNO_CODES.EEXIST)
            }
          } else {
            node = FS.mknod(path, mode, 0);
            created = true;
          }
        }
        if (!node) {
          throw new FS.ErrnoError(ERRNO_CODES.ENOENT)
        }
        if (FS.isChrdev(node.mode)) {
          flags &= ~512;
        }
        if (flags & 65536 && !FS.isDir(node.mode)) {
          throw new FS.ErrnoError(ERRNO_CODES.ENOTDIR)
        }
        if (!created) {
          var err = FS.mayOpen(node, flags);
          if (err) {
            throw new FS.ErrnoError(err)
          }
        }
        if (flags & 512) {
          FS.truncate(node, 0);
        }
        flags &= ~(128 | 512);
        var stream = FS.createStream(
          {
            node: node,
            path: FS.getPath(node),
            flags: flags,
            seekable: true,
            position: 0,
            stream_ops: node.stream_ops,
            ungotten: [],
            error: false,
          },
          fd_start,
          fd_end
        );
        if (stream.stream_ops.open) {
          stream.stream_ops.open(stream);
        }
        if (Module['logReadFiles'] && !(flags & 1)) {
          if (!FS.readFiles) FS.readFiles = {};
          if (!(path in FS.readFiles)) {
            FS.readFiles[path] = 1;
            Module['printErr']('read file: ' + path);
          }
        }
        try {
          if (FS.trackingDelegate['onOpenFile']) {
            var trackingFlags = 0;
            if ((flags & 2097155) !== 1) {
              trackingFlags |= FS.tracking.openFlags.READ;
            }
            if ((flags & 2097155) !== 0) {
              trackingFlags |= FS.tracking.openFlags.WRITE;
            }
            FS.trackingDelegate['onOpenFile'](path, trackingFlags);
          }
        } catch (e) {
          console.log("FS.trackingDelegate['onOpenFile']('" + path + "', flags) threw an exception: " + e.message);
        }
        return stream
      },
      close: function (stream) {
        if (FS.isClosed(stream)) {
          throw new FS.ErrnoError(ERRNO_CODES.EBADF)
        }
        if (stream.getdents) stream.getdents = null;
        try {
          if (stream.stream_ops.close) {
            stream.stream_ops.close(stream);
          }
        } catch (e) {
          throw e
        } finally {
          FS.closeStream(stream.fd);
        }
        stream.fd = null;
      },
      isClosed: function (stream) {
        return stream.fd === null
      },
      llseek: function (stream, offset, whence) {
        if (FS.isClosed(stream)) {
          throw new FS.ErrnoError(ERRNO_CODES.EBADF)
        }
        if (!stream.seekable || !stream.stream_ops.llseek) {
          throw new FS.ErrnoError(ERRNO_CODES.ESPIPE)
        }
        stream.position = stream.stream_ops.llseek(stream, offset, whence);
        stream.ungotten = [];
        return stream.position
      },
      read: function (stream, buffer, offset, length, position) {
        if (length < 0 || position < 0) {
          throw new FS.ErrnoError(ERRNO_CODES.EINVAL)
        }
        if (FS.isClosed(stream)) {
          throw new FS.ErrnoError(ERRNO_CODES.EBADF)
        }
        if ((stream.flags & 2097155) === 1) {
          throw new FS.ErrnoError(ERRNO_CODES.EBADF)
        }
        if (FS.isDir(stream.node.mode)) {
          throw new FS.ErrnoError(ERRNO_CODES.EISDIR)
        }
        if (!stream.stream_ops.read) {
          throw new FS.ErrnoError(ERRNO_CODES.EINVAL)
        }
        var seeking = typeof position !== 'undefined';
        if (!seeking) {
          position = stream.position;
        } else if (!stream.seekable) {
          throw new FS.ErrnoError(ERRNO_CODES.ESPIPE)
        }
        var bytesRead = stream.stream_ops.read(stream, buffer, offset, length, position);
        if (!seeking) stream.position += bytesRead;
        return bytesRead
      },
      write: function (stream, buffer, offset, length, position, canOwn) {
        if (length < 0 || position < 0) {
          throw new FS.ErrnoError(ERRNO_CODES.EINVAL)
        }
        if (FS.isClosed(stream)) {
          throw new FS.ErrnoError(ERRNO_CODES.EBADF)
        }
        if ((stream.flags & 2097155) === 0) {
          throw new FS.ErrnoError(ERRNO_CODES.EBADF)
        }
        if (FS.isDir(stream.node.mode)) {
          throw new FS.ErrnoError(ERRNO_CODES.EISDIR)
        }
        if (!stream.stream_ops.write) {
          throw new FS.ErrnoError(ERRNO_CODES.EINVAL)
        }
        if (stream.flags & 1024) {
          FS.llseek(stream, 0, 2);
        }
        var seeking = typeof position !== 'undefined';
        if (!seeking) {
          position = stream.position;
        } else if (!stream.seekable) {
          throw new FS.ErrnoError(ERRNO_CODES.ESPIPE)
        }
        var bytesWritten = stream.stream_ops.write(stream, buffer, offset, length, position, canOwn);
        if (!seeking) stream.position += bytesWritten;
        try {
          if (stream.path && FS.trackingDelegate['onWriteToFile']) FS.trackingDelegate['onWriteToFile'](stream.path);
        } catch (e) {
          console.log("FS.trackingDelegate['onWriteToFile']('" + path + "') threw an exception: " + e.message);
        }
        return bytesWritten
      },
      allocate: function (stream, offset, length) {
        if (FS.isClosed(stream)) {
          throw new FS.ErrnoError(ERRNO_CODES.EBADF)
        }
        if (offset < 0 || length <= 0) {
          throw new FS.ErrnoError(ERRNO_CODES.EINVAL)
        }
        if ((stream.flags & 2097155) === 0) {
          throw new FS.ErrnoError(ERRNO_CODES.EBADF)
        }
        if (!FS.isFile(stream.node.mode) && !FS.isDir(stream.node.mode)) {
          throw new FS.ErrnoError(ERRNO_CODES.ENODEV)
        }
        if (!stream.stream_ops.allocate) {
          throw new FS.ErrnoError(ERRNO_CODES.EOPNOTSUPP)
        }
        stream.stream_ops.allocate(stream, offset, length);
      },
      mmap: function (stream, buffer, offset, length, position, prot, flags) {
        if ((stream.flags & 2097155) === 1) {
          throw new FS.ErrnoError(ERRNO_CODES.EACCES)
        }
        if (!stream.stream_ops.mmap) {
          throw new FS.ErrnoError(ERRNO_CODES.ENODEV)
        }
        return stream.stream_ops.mmap(stream, buffer, offset, length, position, prot, flags)
      },
      msync: function (stream, buffer, offset, length, mmapFlags) {
        if (!stream || !stream.stream_ops.msync) {
          return 0
        }
        return stream.stream_ops.msync(stream, buffer, offset, length, mmapFlags)
      },
      munmap: function (stream) {
        return 0
      },
      ioctl: function (stream, cmd, arg) {
        if (!stream.stream_ops.ioctl) {
          throw new FS.ErrnoError(ERRNO_CODES.ENOTTY)
        }
        return stream.stream_ops.ioctl(stream, cmd, arg)
      },
      readFile: function (path, opts) {
        opts = opts || {};
        opts.flags = opts.flags || 'r';
        opts.encoding = opts.encoding || 'binary';
        if (opts.encoding !== 'utf8' && opts.encoding !== 'binary') {
          throw new Error('Invalid encoding type "' + opts.encoding + '"')
        }
        var ret;
        var stream = FS.open(path, opts.flags);
        var stat = FS.stat(path);
        var length = stat.size;
        var buf = new Uint8Array(length);
        FS.read(stream, buf, 0, length, 0);
        if (opts.encoding === 'utf8') {
          ret = UTF8ArrayToString(buf, 0);
        } else if (opts.encoding === 'binary') {
          ret = buf;
        }
        FS.close(stream);
        return ret
      },
      writeFile: function (path, data, opts) {
        opts = opts || {};
        opts.flags = opts.flags || 'w';
        var stream = FS.open(path, opts.flags, opts.mode);
        if (typeof data === 'string') {
          var buf = new Uint8Array(lengthBytesUTF8(data) + 1);
          var actualNumBytes = stringToUTF8Array(data, buf, 0, buf.length);
          FS.write(stream, buf, 0, actualNumBytes, undefined, opts.canOwn);
        } else if (ArrayBuffer.isView(data)) {
          FS.write(stream, data, 0, data.byteLength, undefined, opts.canOwn);
        } else {
          throw new Error('Unsupported data type')
        }
        FS.close(stream);
      },
      cwd: function () {
        return FS.currentPath
      },
      chdir: function (path) {
        var lookup = FS.lookupPath(path, { follow: true });
        if (lookup.node === null) {
          throw new FS.ErrnoError(ERRNO_CODES.ENOENT)
        }
        if (!FS.isDir(lookup.node.mode)) {
          throw new FS.ErrnoError(ERRNO_CODES.ENOTDIR)
        }
        var err = FS.nodePermissions(lookup.node, 'x');
        if (err) {
          throw new FS.ErrnoError(err)
        }
        FS.currentPath = lookup.path;
      },
      createDefaultDirectories: function () {
        FS.mkdir('/tmp');
        FS.mkdir('/home');
        FS.mkdir('/home/web_user');
      },
      createDefaultDevices: function () {
        FS.mkdir('/dev');
        FS.registerDevice(FS.makedev(1, 3), {
          read: function () {
            return 0
          },
          write: function (stream, buffer, offset, length, pos) {
            return length
          },
        });
        FS.mkdev('/dev/null', FS.makedev(1, 3));
        TTY.register(FS.makedev(5, 0), TTY.default_tty_ops);
        TTY.register(FS.makedev(6, 0), TTY.default_tty1_ops);
        FS.mkdev('/dev/tty', FS.makedev(5, 0));
        FS.mkdev('/dev/tty1', FS.makedev(6, 0));
        var random_device;
        if (typeof crypto !== 'undefined') {
          var randomBuffer = new Uint8Array(1);
          random_device = function () {
            crypto.getRandomValues(randomBuffer);
            return randomBuffer[0]
          };
        } else if (ENVIRONMENT_IS_NODE) {
          random_device = function () {
            return require('crypto')['randomBytes'](1)[0]
          };
        } else {
          random_device = function () {
            return (Math.random() * 256) | 0
          };
        }
        FS.createDevice('/dev', 'random', random_device);
        FS.createDevice('/dev', 'urandom', random_device);
        FS.mkdir('/dev/shm');
        FS.mkdir('/dev/shm/tmp');
      },
      createSpecialDirectories: function () {
        FS.mkdir('/proc');
        FS.mkdir('/proc/self');
        FS.mkdir('/proc/self/fd');
        FS.mount(
          {
            mount: function () {
              var node = FS.createNode('/proc/self', 'fd', 16384 | 511, 73);
              node.node_ops = {
                lookup: function (parent, name) {
                  var fd = +name;
                  var stream = FS.getStream(fd);
                  if (!stream) throw new FS.ErrnoError(ERRNO_CODES.EBADF)
                  var ret = {
                    parent: null,
                    mount: { mountpoint: 'fake' },
                    node_ops: {
                      readlink: function () {
                        return stream.path
                      },
                    },
                  };
                  ret.parent = ret;
                  return ret
                },
              };
              return node
            },
          },
          {},
          '/proc/self/fd'
        );
      },
      createStandardStreams: function () {
        if (Module['stdin']) {
          FS.createDevice('/dev', 'stdin', Module['stdin']);
        } else {
          FS.symlink('/dev/tty', '/dev/stdin');
        }
        if (Module['stdout']) {
          FS.createDevice('/dev', 'stdout', null, Module['stdout']);
        } else {
          FS.symlink('/dev/tty', '/dev/stdout');
        }
        if (Module['stderr']) {
          FS.createDevice('/dev', 'stderr', null, Module['stderr']);
        } else {
          FS.symlink('/dev/tty1', '/dev/stderr');
        }
        var stdin = FS.open('/dev/stdin', 'r');
        assert(stdin.fd === 0, 'invalid handle for stdin (' + stdin.fd + ')');
        var stdout = FS.open('/dev/stdout', 'w');
        assert(stdout.fd === 1, 'invalid handle for stdout (' + stdout.fd + ')');
        var stderr = FS.open('/dev/stderr', 'w');
        assert(stderr.fd === 2, 'invalid handle for stderr (' + stderr.fd + ')');
      },
      ensureErrnoError: function () {
        if (FS.ErrnoError) return
        FS.ErrnoError = function ErrnoError(errno, node) {
          this.node = node;
          this.setErrno = function (errno) {
            this.errno = errno;
            for (var key in ERRNO_CODES) {
              if (ERRNO_CODES[key] === errno) {
                this.code = key;
                break
              }
            }
          };
          this.setErrno(errno);
          this.message = ERRNO_MESSAGES[errno];
          if (this.stack) Object.defineProperty(this, 'stack', { value: new Error().stack, writable: true });
        };
        FS.ErrnoError.prototype = new Error();
        FS.ErrnoError.prototype.constructor = FS.ErrnoError
        ;[ERRNO_CODES.ENOENT].forEach(function (code) {
          FS.genericErrors[code] = new FS.ErrnoError(code);
          FS.genericErrors[code].stack = '<generic error, no stack>';
        });
      },
      staticInit: function () {
        FS.ensureErrnoError();
        FS.nameTable = new Array(4096);
        FS.mount(MEMFS, {}, '/');
        FS.createDefaultDirectories();
        FS.createDefaultDevices();
        FS.createSpecialDirectories();
        FS.filesystems = { MEMFS: MEMFS, IDBFS: IDBFS, NODEFS: NODEFS, WORKERFS: WORKERFS };
      },
      init: function (input, output, error) {
        assert(
          !FS.init.initialized,
          'FS.init was previously called. If you want to initialize later with custom parameters, remove any earlier calls (note that one is automatically added to the generated code)'
        );
        FS.init.initialized = true;
        FS.ensureErrnoError();
        Module['stdin'] = input || Module['stdin'];
        Module['stdout'] = output || Module['stdout'];
        Module['stderr'] = error || Module['stderr'];
        FS.createStandardStreams();
      },
      quit: function () {
        FS.init.initialized = false;
        var fflush = Module['_fflush'];
        if (fflush) fflush(0);
        for (var i = 0; i < FS.streams.length; i++) {
          var stream = FS.streams[i];
          if (!stream) {
            continue
          }
          FS.close(stream);
        }
      },
      getMode: function (canRead, canWrite) {
        var mode = 0;
        if (canRead) mode |= 292 | 73;
        if (canWrite) mode |= 146;
        return mode
      },
      joinPath: function (parts, forceRelative) {
        var path = PATH.join.apply(null, parts);
        if (forceRelative && path[0] == '/') path = path.substr(1);
        return path
      },
      absolutePath: function (relative, base) {
        return PATH.resolve(base, relative)
      },
      standardizePath: function (path) {
        return PATH.normalize(path)
      },
      findObject: function (path, dontResolveLastLink) {
        var ret = FS.analyzePath(path, dontResolveLastLink);
        if (ret.exists) {
          return ret.object
        } else {
          ___setErrNo(ret.error);
          return null
        }
      },
      analyzePath: function (path, dontResolveLastLink) {
        try {
          var lookup = FS.lookupPath(path, { follow: !dontResolveLastLink });
          path = lookup.path;
        } catch (e) {}
        var ret = {
          isRoot: false,
          exists: false,
          error: 0,
          name: null,
          path: null,
          object: null,
          parentExists: false,
          parentPath: null,
          parentObject: null,
        };
        try {
          var lookup = FS.lookupPath(path, { parent: true });
          ret.parentExists = true;
          ret.parentPath = lookup.path;
          ret.parentObject = lookup.node;
          ret.name = PATH.basename(path);
          lookup = FS.lookupPath(path, { follow: !dontResolveLastLink });
          ret.exists = true;
          ret.path = lookup.path;
          ret.object = lookup.node;
          ret.name = lookup.node.name;
          ret.isRoot = lookup.path === '/';
        } catch (e) {
          ret.error = e.errno;
        }
        return ret
      },
      createFolder: function (parent, name, canRead, canWrite) {
        var path = PATH.join2(typeof parent === 'string' ? parent : FS.getPath(parent), name);
        var mode = FS.getMode(canRead, canWrite);
        return FS.mkdir(path, mode)
      },
      createPath: function (parent, path, canRead, canWrite) {
        parent = typeof parent === 'string' ? parent : FS.getPath(parent);
        var parts = path.split('/').reverse();
        while (parts.length) {
          var part = parts.pop();
          if (!part) continue
          var current = PATH.join2(parent, part);
          try {
            FS.mkdir(current);
          } catch (e) {}
          parent = current;
        }
        return current
      },
      createFile: function (parent, name, properties, canRead, canWrite) {
        var path = PATH.join2(typeof parent === 'string' ? parent : FS.getPath(parent), name);
        var mode = FS.getMode(canRead, canWrite);
        return FS.create(path, mode)
      },
      createDataFile: function (parent, name, data, canRead, canWrite, canOwn) {
        var path = name ? PATH.join2(typeof parent === 'string' ? parent : FS.getPath(parent), name) : parent;
        var mode = FS.getMode(canRead, canWrite);
        var node = FS.create(path, mode);
        if (data) {
          if (typeof data === 'string') {
            var arr = new Array(data.length);
            for (var i = 0, len = data.length; i < len; ++i) arr[i] = data.charCodeAt(i);
            data = arr;
          }
          FS.chmod(node, mode | 146);
          var stream = FS.open(node, 'w');
          FS.write(stream, data, 0, data.length, 0, canOwn);
          FS.close(stream);
          FS.chmod(node, mode);
        }
        return node
      },
      createDevice: function (parent, name, input, output) {
        var path = PATH.join2(typeof parent === 'string' ? parent : FS.getPath(parent), name);
        var mode = FS.getMode(!!input, !!output);
        if (!FS.createDevice.major) FS.createDevice.major = 64;
        var dev = FS.makedev(FS.createDevice.major++, 0);
        FS.registerDevice(dev, {
          open: function (stream) {
            stream.seekable = false;
          },
          close: function (stream) {
            if (output && output.buffer && output.buffer.length) {
              output(10);
            }
          },
          read: function (stream, buffer, offset, length, pos) {
            var bytesRead = 0;
            for (var i = 0; i < length; i++) {
              var result;
              try {
                result = input();
              } catch (e) {
                throw new FS.ErrnoError(ERRNO_CODES.EIO)
              }
              if (result === undefined && bytesRead === 0) {
                throw new FS.ErrnoError(ERRNO_CODES.EAGAIN)
              }
              if (result === null || result === undefined) break
              bytesRead++;
              buffer[offset + i] = result;
            }
            if (bytesRead) {
              stream.node.timestamp = Date.now();
            }
            return bytesRead
          },
          write: function (stream, buffer, offset, length, pos) {
            for (var i = 0; i < length; i++) {
              try {
                output(buffer[offset + i]);
              } catch (e) {
                throw new FS.ErrnoError(ERRNO_CODES.EIO)
              }
            }
            if (length) {
              stream.node.timestamp = Date.now();
            }
            return i
          },
        });
        return FS.mkdev(path, mode, dev)
      },
      createLink: function (parent, name, target, canRead, canWrite) {
        var path = PATH.join2(typeof parent === 'string' ? parent : FS.getPath(parent), name);
        return FS.symlink(target, path)
      },
      forceLoadFile: function (obj) {
        if (obj.isDevice || obj.isFolder || obj.link || obj.contents) return true
        var success = true;
        if (typeof XMLHttpRequest !== 'undefined') {
          throw new Error(
            'Lazy loading should have been performed (contents set) in createLazyFile, but it was not. Lazy loading only works in web workers. Use --embed-file or --preload-file in emcc on the main thread.'
          )
        } else if (Module['read']) {
          try {
            obj.contents = intArrayFromString(Module['read'](obj.url), true);
            obj.usedBytes = obj.contents.length;
          } catch (e) {
            success = false;
          }
        } else {
          throw new Error('Cannot load without read() or XMLHttpRequest.')
        }
        if (!success) ___setErrNo(ERRNO_CODES.EIO);
        return success
      },
      createLazyFile: function (parent, name, url, canRead, canWrite) {
        function LazyUint8Array() {
          this.lengthKnown = false;
          this.chunks = [];
        }
        LazyUint8Array.prototype.get = function LazyUint8Array_get(idx) {
          if (idx > this.length - 1 || idx < 0) {
            return undefined
          }
          var chunkOffset = idx % this.chunkSize;
          var chunkNum = (idx / this.chunkSize) | 0;
          return this.getter(chunkNum)[chunkOffset]
        };
        LazyUint8Array.prototype.setDataGetter = function LazyUint8Array_setDataGetter(getter) {
          this.getter = getter;
        };
        LazyUint8Array.prototype.cacheLength = function LazyUint8Array_cacheLength() {
          var xhr = new XMLHttpRequest();
          xhr.open('HEAD', url, false);
          xhr.send(null);
          if (!((xhr.status >= 200 && xhr.status < 300) || xhr.status === 304))
            throw new Error("Couldn't load " + url + '. Status: ' + xhr.status)
          var datalength = Number(xhr.getResponseHeader('Content-length'));
          var header;
          var hasByteServing = (header = xhr.getResponseHeader('Accept-Ranges')) && header === 'bytes';
          var usesGzip = (header = xhr.getResponseHeader('Content-Encoding')) && header === 'gzip';
          var chunkSize = 1024 * 1024;
          if (!hasByteServing) chunkSize = datalength;
          var doXHR = function (from, to) {
            if (from > to) throw new Error('invalid range (' + from + ', ' + to + ') or no bytes requested!')
            if (to > datalength - 1) throw new Error('only ' + datalength + ' bytes available! programmer error!')
            var xhr = new XMLHttpRequest();
            xhr.open('GET', url, false);
            if (datalength !== chunkSize) xhr.setRequestHeader('Range', 'bytes=' + from + '-' + to);
            if (typeof Uint8Array != 'undefined') xhr.responseType = 'arraybuffer';
            if (xhr.overrideMimeType) {
              xhr.overrideMimeType('text/plain; charset=x-user-defined');
            }
            xhr.send(null);
            if (!((xhr.status >= 200 && xhr.status < 300) || xhr.status === 304))
              throw new Error("Couldn't load " + url + '. Status: ' + xhr.status)
            if (xhr.response !== undefined) {
              return new Uint8Array(xhr.response || [])
            } else {
              return intArrayFromString(xhr.responseText || '', true)
            }
          };
          var lazyArray = this;
          lazyArray.setDataGetter(function (chunkNum) {
            var start = chunkNum * chunkSize;
            var end = (chunkNum + 1) * chunkSize - 1;
            end = Math.min(end, datalength - 1);
            if (typeof lazyArray.chunks[chunkNum] === 'undefined') {
              lazyArray.chunks[chunkNum] = doXHR(start, end);
            }
            if (typeof lazyArray.chunks[chunkNum] === 'undefined') throw new Error('doXHR failed!')
            return lazyArray.chunks[chunkNum]
          });
          if (usesGzip || !datalength) {
            chunkSize = datalength = 1;
            datalength = this.getter(0).length;
            chunkSize = datalength;
            console.log('LazyFiles on gzip forces download of the whole file when length is accessed');
          }
          this._length = datalength;
          this._chunkSize = chunkSize;
          this.lengthKnown = true;
        };
        if (typeof XMLHttpRequest !== 'undefined') {
          if (!ENVIRONMENT_IS_WORKER)
            throw 'Cannot do synchronous binary XHRs outside webworkers in modern browsers. Use --embed-file or --preload-file in emcc'
          var lazyArray = new LazyUint8Array();
          Object.defineProperties(lazyArray, {
            length: {
              get: function () {
                if (!this.lengthKnown) {
                  this.cacheLength();
                }
                return this._length
              },
            },
            chunkSize: {
              get: function () {
                if (!this.lengthKnown) {
                  this.cacheLength();
                }
                return this._chunkSize
              },
            },
          });
          var properties = { isDevice: false, contents: lazyArray };
        } else {
          var properties = { isDevice: false, url: url };
        }
        var node = FS.createFile(parent, name, properties, canRead, canWrite);
        if (properties.contents) {
          node.contents = properties.contents;
        } else if (properties.url) {
          node.contents = null;
          node.url = properties.url;
        }
        Object.defineProperties(node, {
          usedBytes: {
            get: function () {
              return this.contents.length
            },
          },
        });
        var stream_ops = {};
        var keys = Object.keys(node.stream_ops);
        keys.forEach(function (key) {
          var fn = node.stream_ops[key];
          stream_ops[key] = function forceLoadLazyFile() {
            if (!FS.forceLoadFile(node)) {
              throw new FS.ErrnoError(ERRNO_CODES.EIO)
            }
            return fn.apply(null, arguments)
          };
        });
        stream_ops.read = function stream_ops_read(stream, buffer, offset, length, position) {
          if (!FS.forceLoadFile(node)) {
            throw new FS.ErrnoError(ERRNO_CODES.EIO)
          }
          var contents = stream.node.contents;
          if (position >= contents.length) return 0
          var size = Math.min(contents.length - position, length);
          assert(size >= 0);
          if (contents.slice) {
            for (var i = 0; i < size; i++) {
              buffer[offset + i] = contents[position + i];
            }
          } else {
            for (var i = 0; i < size; i++) {
              buffer[offset + i] = contents.get(position + i);
            }
          }
          return size
        };
        node.stream_ops = stream_ops;
        return node
      },
      createPreloadedFile: function (
        parent,
        name,
        url,
        canRead,
        canWrite,
        onload,
        onerror,
        dontCreateFile,
        canOwn,
        preFinish
      ) {
        Browser.init();
        var fullname = name ? PATH.resolve(PATH.join2(parent, name)) : parent;
        function processData(byteArray) {
          function finish(byteArray) {
            if (preFinish) preFinish();
            if (!dontCreateFile) {
              FS.createDataFile(parent, name, byteArray, canRead, canWrite, canOwn);
            }
            if (onload) onload();
            removeRunDependency();
          }
          var handled = false;
          Module['preloadPlugins'].forEach(function (plugin) {
            if (handled) return
            if (plugin['canHandle'](fullname)) {
              plugin['handle'](byteArray, fullname, finish, function () {
                if (onerror) onerror();
                removeRunDependency();
              });
              handled = true;
            }
          });
          if (!handled) finish(byteArray);
        }
        addRunDependency();
        if (typeof url == 'string') {
          Browser.asyncLoad(
            url,
            function (byteArray) {
              processData(byteArray);
            },
            onerror
          );
        } else {
          processData(url);
        }
      },
      indexedDB: function () {
        return window.indexedDB || window.mozIndexedDB || window.webkitIndexedDB || window.msIndexedDB
      },
      DB_NAME: function () {
        return 'EM_FS_' + window.location.pathname
      },
      DB_VERSION: 20,
      DB_STORE_NAME: 'FILE_DATA',
      saveFilesToDB: function (paths, onload, onerror) {
        onload = onload || function () {};
        onerror = onerror || function () {};
        var indexedDB = FS.indexedDB();
        try {
          var openRequest = indexedDB.open(FS.DB_NAME(), FS.DB_VERSION);
        } catch (e) {
          return onerror(e)
        }
        openRequest.onupgradeneeded = function openRequest_onupgradeneeded() {
          console.log('creating db');
          var db = openRequest.result;
          db.createObjectStore(FS.DB_STORE_NAME);
        };
        openRequest.onsuccess = function openRequest_onsuccess() {
          var db = openRequest.result;
          var transaction = db.transaction([FS.DB_STORE_NAME], 'readwrite');
          var files = transaction.objectStore(FS.DB_STORE_NAME);
          var ok = 0,
            fail = 0,
            total = paths.length;
          function finish() {
            if (fail == 0) onload();
            else onerror();
          }
          paths.forEach(function (path) {
            var putRequest = files.put(FS.analyzePath(path).object.contents, path);
            putRequest.onsuccess = function putRequest_onsuccess() {
              ok++;
              if (ok + fail == total) finish();
            };
            putRequest.onerror = function putRequest_onerror() {
              fail++;
              if (ok + fail == total) finish();
            };
          });
          transaction.onerror = onerror;
        };
        openRequest.onerror = onerror;
      },
      loadFilesFromDB: function (paths, onload, onerror) {
        onload = onload || function () {};
        onerror = onerror || function () {};
        var indexedDB = FS.indexedDB();
        try {
          var openRequest = indexedDB.open(FS.DB_NAME(), FS.DB_VERSION);
        } catch (e) {
          return onerror(e)
        }
        openRequest.onupgradeneeded = onerror;
        openRequest.onsuccess = function openRequest_onsuccess() {
          var db = openRequest.result;
          try {
            var transaction = db.transaction([FS.DB_STORE_NAME], 'readonly');
          } catch (e) {
            onerror(e);
            return
          }
          var files = transaction.objectStore(FS.DB_STORE_NAME);
          var ok = 0,
            fail = 0,
            total = paths.length;
          function finish() {
            if (fail == 0) onload();
            else onerror();
          }
          paths.forEach(function (path) {
            var getRequest = files.get(path);
            getRequest.onsuccess = function getRequest_onsuccess() {
              if (FS.analyzePath(path).exists) {
                FS.unlink(path);
              }
              FS.createDataFile(PATH.dirname(path), PATH.basename(path), getRequest.result, true, true, true);
              ok++;
              if (ok + fail == total) finish();
            };
            getRequest.onerror = function getRequest_onerror() {
              fail++;
              if (ok + fail == total) finish();
            };
          });
          transaction.onerror = onerror;
        };
        openRequest.onerror = onerror;
      },
    };
    var SYSCALLS = {
      DEFAULT_POLLMASK: 5,
      mappings: {},
      umask: 511,
      calculateAt: function (dirfd, path) {
        if (path[0] !== '/') {
          var dir;
          if (dirfd === -100) {
            dir = FS.cwd();
          } else {
            var dirstream = FS.getStream(dirfd);
            if (!dirstream) throw new FS.ErrnoError(ERRNO_CODES.EBADF)
            dir = dirstream.path;
          }
          path = PATH.join2(dir, path);
        }
        return path
      },
      doStat: function (func, path, buf) {
        try {
          var stat = func(path);
        } catch (e) {
          if (e && e.node && PATH.normalize(path) !== PATH.normalize(FS.getPath(e.node))) {
            return -ERRNO_CODES.ENOTDIR
          }
          throw e
        }
        HEAP32[buf >> 2] = stat.dev;
        HEAP32[(buf + 4) >> 2] = 0;
        HEAP32[(buf + 8) >> 2] = stat.ino;
        HEAP32[(buf + 12) >> 2] = stat.mode;
        HEAP32[(buf + 16) >> 2] = stat.nlink;
        HEAP32[(buf + 20) >> 2] = stat.uid;
        HEAP32[(buf + 24) >> 2] = stat.gid;
        HEAP32[(buf + 28) >> 2] = stat.rdev;
        HEAP32[(buf + 32) >> 2] = 0;
        HEAP32[(buf + 36) >> 2] = stat.size;
        HEAP32[(buf + 40) >> 2] = 4096;
        HEAP32[(buf + 44) >> 2] = stat.blocks;
        HEAP32[(buf + 48) >> 2] = (stat.atime.getTime() / 1e3) | 0;
        HEAP32[(buf + 52) >> 2] = 0;
        HEAP32[(buf + 56) >> 2] = (stat.mtime.getTime() / 1e3) | 0;
        HEAP32[(buf + 60) >> 2] = 0;
        HEAP32[(buf + 64) >> 2] = (stat.ctime.getTime() / 1e3) | 0;
        HEAP32[(buf + 68) >> 2] = 0;
        HEAP32[(buf + 72) >> 2] = stat.ino;
        return 0
      },
      doMsync: function (addr, stream, len, flags) {
        var buffer = new Uint8Array(HEAPU8.subarray(addr, addr + len));
        FS.msync(stream, buffer, 0, len, flags);
      },
      doMkdir: function (path, mode) {
        path = PATH.normalize(path);
        if (path[path.length - 1] === '/') path = path.substr(0, path.length - 1);
        FS.mkdir(path, mode, 0);
        return 0
      },
      doMknod: function (path, mode, dev) {
        switch (mode & 61440) {
          case 32768:
          case 8192:
          case 24576:
          case 4096:
          case 49152:
            break
          default:
            return -ERRNO_CODES.EINVAL
        }
        FS.mknod(path, mode, dev);
        return 0
      },
      doReadlink: function (path, buf, bufsize) {
        if (bufsize <= 0) return -ERRNO_CODES.EINVAL
        var ret = FS.readlink(path);
        var len = Math.min(bufsize, lengthBytesUTF8(ret));
        var endChar = HEAP8[buf + len];
        stringToUTF8(ret, buf, bufsize + 1);
        HEAP8[buf + len] = endChar;
        return len
      },
      doAccess: function (path, amode) {
        if (amode & ~7) {
          return -ERRNO_CODES.EINVAL
        }
        var node;
        var lookup = FS.lookupPath(path, { follow: true });
        node = lookup.node;
        var perms = '';
        if (amode & 4) perms += 'r';
        if (amode & 2) perms += 'w';
        if (amode & 1) perms += 'x';
        if (perms && FS.nodePermissions(node, perms)) {
          return -ERRNO_CODES.EACCES
        }
        return 0
      },
      doDup: function (path, flags, suggestFD) {
        var suggest = FS.getStream(suggestFD);
        if (suggest) FS.close(suggest);
        return FS.open(path, flags, 0, suggestFD, suggestFD).fd
      },
      doReadv: function (stream, iov, iovcnt, offset) {
        var ret = 0;
        for (var i = 0; i < iovcnt; i++) {
          var ptr = HEAP32[(iov + i * 8) >> 2];
          var len = HEAP32[(iov + (i * 8 + 4)) >> 2];
          var curr = FS.read(stream, HEAP8, ptr, len, offset);
          if (curr < 0) return -1
          ret += curr;
          if (curr < len) break
        }
        return ret
      },
      doWritev: function (stream, iov, iovcnt, offset) {
        var ret = 0;
        for (var i = 0; i < iovcnt; i++) {
          var ptr = HEAP32[(iov + i * 8) >> 2];
          var len = HEAP32[(iov + (i * 8 + 4)) >> 2];
          var curr = FS.write(stream, HEAP8, ptr, len, offset);
          if (curr < 0) return -1
          ret += curr;
        }
        return ret
      },
      varargs: 0,
      get: function (varargs) {
        SYSCALLS.varargs += 4;
        var ret = HEAP32[(SYSCALLS.varargs - 4) >> 2];
        return ret
      },
      getStr: function () {
        var ret = Pointer_stringify(SYSCALLS.get());
        return ret
      },
      getStreamFromFD: function () {
        var stream = FS.getStream(SYSCALLS.get());
        if (!stream) throw new FS.ErrnoError(ERRNO_CODES.EBADF)
        return stream
      },
      getSocketFromFD: function () {
        var socket = SOCKFS.getSocket(SYSCALLS.get());
        if (!socket) throw new FS.ErrnoError(ERRNO_CODES.EBADF)
        return socket
      },
      getSocketAddress: function (allowNull) {
        var addrp = SYSCALLS.get(),
          addrlen = SYSCALLS.get();
        if (allowNull && addrp === 0) return null
        var info = __read_sockaddr(addrp, addrlen);
        if (info.errno) throw new FS.ErrnoError(info.errno)
        info.addr = DNS.lookup_addr(info.addr) || info.addr;
        return info
      },
      get64: function () {
        var low = SYSCALLS.get(),
          high = SYSCALLS.get();
        if (low >= 0) assert(high === 0);
        else assert(high === -1);
        return low
      },
      getZero: function () {
        assert(SYSCALLS.get() === 0);
      },
    };
    function ___syscall140(which, varargs) {
      SYSCALLS.varargs = varargs;
      try {
        var stream = SYSCALLS.getStreamFromFD(),
          offset_high = SYSCALLS.get(),
          offset_low = SYSCALLS.get(),
          result = SYSCALLS.get(),
          whence = SYSCALLS.get();
        var offset = offset_low;
        FS.llseek(stream, offset, whence);
        HEAP32[result >> 2] = stream.position;
        if (stream.getdents && offset === 0 && whence === 0) stream.getdents = null;
        return 0
      } catch (e) {
        if (typeof FS === 'undefined' || !(e instanceof FS.ErrnoError)) abort(e);
        return -e.errno
      }
    }
    function ___syscall145(which, varargs) {
      SYSCALLS.varargs = varargs;
      try {
        var stream = SYSCALLS.getStreamFromFD(),
          iov = SYSCALLS.get(),
          iovcnt = SYSCALLS.get();
        return SYSCALLS.doReadv(stream, iov, iovcnt)
      } catch (e) {
        if (typeof FS === 'undefined' || !(e instanceof FS.ErrnoError)) abort(e);
        return -e.errno
      }
    }
    function ___syscall146(which, varargs) {
      SYSCALLS.varargs = varargs;
      try {
        var stream = SYSCALLS.getStreamFromFD(),
          iov = SYSCALLS.get(),
          iovcnt = SYSCALLS.get();
        return SYSCALLS.doWritev(stream, iov, iovcnt)
      } catch (e) {
        if (typeof FS === 'undefined' || !(e instanceof FS.ErrnoError)) abort(e);
        return -e.errno
      }
    }
    function ___syscall183(which, varargs) {
      SYSCALLS.varargs = varargs;
      try {
        var buf = SYSCALLS.get(),
          size = SYSCALLS.get();
        if (size === 0) return -ERRNO_CODES.EINVAL
        var cwd = FS.cwd();
        var cwdLengthInBytes = lengthBytesUTF8(cwd);
        if (size < cwdLengthInBytes + 1) return -ERRNO_CODES.ERANGE
        stringToUTF8(cwd, buf, size);
        return buf
      } catch (e) {
        if (typeof FS === 'undefined' || !(e instanceof FS.ErrnoError)) abort(e);
        return -e.errno
      }
    }
    function ___syscall198(which, varargs) {
      SYSCALLS.varargs = varargs;
      try {
        var path = SYSCALLS.getStr(),
          owner = SYSCALLS.get(),
          group = SYSCALLS.get();
        FS.chown(path, owner, group);
        return 0
      } catch (e) {
        if (typeof FS === 'undefined' || !(e instanceof FS.ErrnoError)) abort(e);
        return -e.errno
      }
    }
    var PROCINFO = { ppid: 1, pid: 42, sid: 42, pgid: 42 };
    function ___syscall20(which, varargs) {
      SYSCALLS.varargs = varargs;
      try {
        return PROCINFO.pid
      } catch (e) {
        if (typeof FS === 'undefined' || !(e instanceof FS.ErrnoError)) abort(e);
        return -e.errno
      }
    }
    function ___syscall6(which, varargs) {
      SYSCALLS.varargs = varargs;
      try {
        var stream = SYSCALLS.getStreamFromFD();
        FS.close(stream);
        return 0
      } catch (e) {
        if (typeof FS === 'undefined' || !(e instanceof FS.ErrnoError)) abort(e);
        return -e.errno
      }
    }
    function ___syscall60(which, varargs) {
      SYSCALLS.varargs = varargs;
      try {
        var mask = SYSCALLS.get();
        var old = SYSCALLS.umask;
        SYSCALLS.umask = mask;
        return old
      } catch (e) {
        if (typeof FS === 'undefined' || !(e instanceof FS.ErrnoError)) abort(e);
        return -e.errno
      }
    }
    function ___syscall83(which, varargs) {
      SYSCALLS.varargs = varargs;
      try {
        var target = SYSCALLS.getStr(),
          linkpath = SYSCALLS.getStr();
        FS.symlink(target, linkpath);
        return 0
      } catch (e) {
        if (typeof FS === 'undefined' || !(e instanceof FS.ErrnoError)) abort(e);
        return -e.errno
      }
    }
    function ___syscall91(which, varargs) {
      SYSCALLS.varargs = varargs;
      try {
        var addr = SYSCALLS.get(),
          len = SYSCALLS.get();
        var info = SYSCALLS.mappings[addr];
        if (!info) return 0
        if (len === info.len) {
          var stream = FS.getStream(info.fd);
          SYSCALLS.doMsync(addr, stream, len, info.flags);
          FS.munmap(stream);
          SYSCALLS.mappings[addr] = null;
          if (info.allocated) {
            _free(info.malloc);
          }
        }
        return 0
      } catch (e) {
        if (typeof FS === 'undefined' || !(e instanceof FS.ErrnoError)) abort(e);
        return -e.errno
      }
    }
    function ___unlock() {}
    var structRegistrations = {};
    function runDestructors(destructors) {
      while (destructors.length) {
        var ptr = destructors.pop();
        var del = destructors.pop();
        del(ptr);
      }
    }
    function simpleReadValueFromPointer(pointer) {
      return this['fromWireType'](HEAPU32[pointer >> 2])
    }
    var awaitingDependencies = {};
    var registeredTypes = {};
    var typeDependencies = {};
    var char_0 = 48;
    var char_9 = 57;
    function makeLegalFunctionName(name) {
      if (undefined === name) {
        return '_unknown'
      }
      name = name.replace(/[^a-zA-Z0-9_]/g, '$');
      var f = name.charCodeAt(0);
      if (f >= char_0 && f <= char_9) {
        return '_' + name
      } else {
        return name
      }
    }
    function createNamedFunction(name, body) {
      name = makeLegalFunctionName(name);
      return new Function(
        'body',
        'return function ' + name + '() {\n' + '    "use strict";' + '    return body.apply(this, arguments);\n' + '};\n'
      )(body)
    }
    function extendError(baseErrorType, errorName) {
      var errorClass = createNamedFunction(errorName, function (message) {
        this.name = errorName;
        this.message = message;
        var stack = new Error(message).stack;
        if (stack !== undefined) {
          this.stack = this.toString() + '\n' + stack.replace(/^Error(:[^\n]*)?\n/, '');
        }
      });
      errorClass.prototype = Object.create(baseErrorType.prototype);
      errorClass.prototype.constructor = errorClass;
      errorClass.prototype.toString = function () {
        if (this.message === undefined) {
          return this.name
        } else {
          return this.name + ': ' + this.message
        }
      };
      return errorClass
    }
    var InternalError = undefined;
    function throwInternalError(message) {
      throw new InternalError(message)
    }
    function whenDependentTypesAreResolved(myTypes, dependentTypes, getTypeConverters) {
      myTypes.forEach(function (type) {
        typeDependencies[type] = dependentTypes;
      });
      function onComplete(typeConverters) {
        var myTypeConverters = getTypeConverters(typeConverters);
        if (myTypeConverters.length !== myTypes.length) {
          throwInternalError('Mismatched type converter count');
        }
        for (var i = 0; i < myTypes.length; ++i) {
          registerType(myTypes[i], myTypeConverters[i]);
        }
      }
      var typeConverters = new Array(dependentTypes.length);
      var unregisteredTypes = [];
      var registered = 0;
      dependentTypes.forEach(function (dt, i) {
        if (registeredTypes.hasOwnProperty(dt)) {
          typeConverters[i] = registeredTypes[dt];
        } else {
          unregisteredTypes.push(dt);
          if (!awaitingDependencies.hasOwnProperty(dt)) {
            awaitingDependencies[dt] = [];
          }
          awaitingDependencies[dt].push(function () {
            typeConverters[i] = registeredTypes[dt];
            ++registered;
            if (registered === unregisteredTypes.length) {
              onComplete(typeConverters);
            }
          });
        }
      });
      if (0 === unregisteredTypes.length) {
        onComplete(typeConverters);
      }
    }
    function __embind_finalize_value_object(structType) {
      var reg = structRegistrations[structType];
      delete structRegistrations[structType];
      var rawConstructor = reg.rawConstructor;
      var rawDestructor = reg.rawDestructor;
      var fieldRecords = reg.fields;
      var fieldTypes = fieldRecords
        .map(function (field) {
          return field.getterReturnType
        })
        .concat(
          fieldRecords.map(function (field) {
            return field.setterArgumentType
          })
        );
      whenDependentTypesAreResolved([structType], fieldTypes, function (fieldTypes) {
        var fields = {};
        fieldRecords.forEach(function (field, i) {
          var fieldName = field.fieldName;
          var getterReturnType = fieldTypes[i];
          var getter = field.getter;
          var getterContext = field.getterContext;
          var setterArgumentType = fieldTypes[i + fieldRecords.length];
          var setter = field.setter;
          var setterContext = field.setterContext;
          fields[fieldName] = {
            read: function (ptr) {
              return getterReturnType['fromWireType'](getter(getterContext, ptr))
            },
            write: function (ptr, o) {
              var destructors = [];
              setter(setterContext, ptr, setterArgumentType['toWireType'](destructors, o));
              runDestructors(destructors);
            },
          };
        });
        return [
          {
            name: reg.name,
            fromWireType: function (ptr) {
              var rv = {};
              for (var i in fields) {
                rv[i] = fields[i].read(ptr);
              }
              rawDestructor(ptr);
              return rv
            },
            toWireType: function (destructors, o) {
              for (var fieldName in fields) {
                if (!(fieldName in o)) {
                  throw new TypeError('Missing field')
                }
              }
              var ptr = rawConstructor();
              for (fieldName in fields) {
                fields[fieldName].write(ptr, o[fieldName]);
              }
              if (destructors !== null) {
                destructors.push(rawDestructor, ptr);
              }
              return ptr
            },
            argPackAdvance: 8,
            readValueFromPointer: simpleReadValueFromPointer,
            destructorFunction: rawDestructor,
          },
        ]
      });
    }
    function getShiftFromSize(size) {
      switch (size) {
        case 1:
          return 0
        case 2:
          return 1
        case 4:
          return 2
        case 8:
          return 3
        default:
          throw new TypeError('Unknown type size: ' + size)
      }
    }
    function embind_init_charCodes() {
      var codes = new Array(256);
      for (var i = 0; i < 256; ++i) {
        codes[i] = String.fromCharCode(i);
      }
      embind_charCodes = codes;
    }
    var embind_charCodes = undefined;
    function readLatin1String(ptr) {
      var ret = '';
      var c = ptr;
      while (HEAPU8[c]) {
        ret += embind_charCodes[HEAPU8[c++]];
      }
      return ret
    }
    var BindingError = undefined;
    function throwBindingError(message) {
      throw new BindingError(message)
    }
    function registerType(rawType, registeredInstance, options) {
      options = options || {};
      if (!('argPackAdvance' in registeredInstance)) {
        throw new TypeError('registerType registeredInstance requires argPackAdvance')
      }
      var name = registeredInstance.name;
      if (!rawType) {
        throwBindingError('type "' + name + '" must have a positive integer typeid pointer');
      }
      if (registeredTypes.hasOwnProperty(rawType)) {
        if (options.ignoreDuplicateRegistrations) {
          return
        } else {
          throwBindingError("Cannot register type '" + name + "' twice");
        }
      }
      registeredTypes[rawType] = registeredInstance;
      delete typeDependencies[rawType];
      if (awaitingDependencies.hasOwnProperty(rawType)) {
        var callbacks = awaitingDependencies[rawType];
        delete awaitingDependencies[rawType];
        callbacks.forEach(function (cb) {
          cb();
        });
      }
    }
    function __embind_register_bool(rawType, name, size, trueValue, falseValue) {
      var shift = getShiftFromSize(size);
      name = readLatin1String(name);
      registerType(rawType, {
        name: name,
        fromWireType: function (wt) {
          return !!wt
        },
        toWireType: function (destructors, o) {
          return o ? trueValue : falseValue
        },
        argPackAdvance: 8,
        readValueFromPointer: function (pointer) {
          var heap;
          if (size === 1) {
            heap = HEAP8;
          } else if (size === 2) {
            heap = HEAP16;
          } else if (size === 4) {
            heap = HEAP32;
          } else {
            throw new TypeError('Unknown boolean type size: ' + name)
          }
          return this['fromWireType'](heap[pointer >> shift])
        },
        destructorFunction: null,
      });
    }
    function ClassHandle_isAliasOf(other) {
      if (!(this instanceof ClassHandle)) {
        return false
      }
      if (!(other instanceof ClassHandle)) {
        return false
      }
      var leftClass = this.$$.ptrType.registeredClass;
      var left = this.$$.ptr;
      var rightClass = other.$$.ptrType.registeredClass;
      var right = other.$$.ptr;
      while (leftClass.baseClass) {
        left = leftClass.upcast(left);
        leftClass = leftClass.baseClass;
      }
      while (rightClass.baseClass) {
        right = rightClass.upcast(right);
        rightClass = rightClass.baseClass;
      }
      return leftClass === rightClass && left === right
    }
    function shallowCopyInternalPointer(o) {
      return {
        count: o.count,
        deleteScheduled: o.deleteScheduled,
        preservePointerOnDelete: o.preservePointerOnDelete,
        ptr: o.ptr,
        ptrType: o.ptrType,
        smartPtr: o.smartPtr,
        smartPtrType: o.smartPtrType,
      }
    }
    function throwInstanceAlreadyDeleted(obj) {
      function getInstanceTypeName(handle) {
        return handle.$$.ptrType.registeredClass.name
      }
      throwBindingError(getInstanceTypeName(obj) + ' instance already deleted');
    }
    function ClassHandle_clone() {
      if (!this.$$.ptr) {
        throwInstanceAlreadyDeleted(this);
      }
      if (this.$$.preservePointerOnDelete) {
        this.$$.count.value += 1;
        return this
      } else {
        var clone = Object.create(Object.getPrototypeOf(this), { $$: { value: shallowCopyInternalPointer(this.$$) } });
        clone.$$.count.value += 1;
        clone.$$.deleteScheduled = false;
        return clone
      }
    }
    function runDestructor(handle) {
      var $$ = handle.$$;
      if ($$.smartPtr) {
        $$.smartPtrType.rawDestructor($$.smartPtr);
      } else {
        $$.ptrType.registeredClass.rawDestructor($$.ptr);
      }
    }
    function ClassHandle_delete() {
      if (!this.$$.ptr) {
        throwInstanceAlreadyDeleted(this);
      }
      if (this.$$.deleteScheduled && !this.$$.preservePointerOnDelete) {
        throwBindingError('Object already scheduled for deletion');
      }
      this.$$.count.value -= 1;
      var toDelete = 0 === this.$$.count.value;
      if (toDelete) {
        runDestructor(this);
      }
      if (!this.$$.preservePointerOnDelete) {
        this.$$.smartPtr = undefined;
        this.$$.ptr = undefined;
      }
    }
    function ClassHandle_isDeleted() {
      return !this.$$.ptr
    }
    var delayFunction = undefined;
    var deletionQueue = [];
    function flushPendingDeletes() {
      while (deletionQueue.length) {
        var obj = deletionQueue.pop();
        obj.$$.deleteScheduled = false;
        obj['delete']();
      }
    }
    function ClassHandle_deleteLater() {
      if (!this.$$.ptr) {
        throwInstanceAlreadyDeleted(this);
      }
      if (this.$$.deleteScheduled && !this.$$.preservePointerOnDelete) {
        throwBindingError('Object already scheduled for deletion');
      }
      deletionQueue.push(this);
      if (deletionQueue.length === 1 && delayFunction) {
        delayFunction(flushPendingDeletes);
      }
      this.$$.deleteScheduled = true;
      return this
    }
    function init_ClassHandle() {
      ClassHandle.prototype['isAliasOf'] = ClassHandle_isAliasOf;
      ClassHandle.prototype['clone'] = ClassHandle_clone;
      ClassHandle.prototype['delete'] = ClassHandle_delete;
      ClassHandle.prototype['isDeleted'] = ClassHandle_isDeleted;
      ClassHandle.prototype['deleteLater'] = ClassHandle_deleteLater;
    }
    function ClassHandle() {}
    var registeredPointers = {};
    function ensureOverloadTable(proto, methodName, humanName) {
      if (undefined === proto[methodName].overloadTable) {
        var prevFunc = proto[methodName];
        proto[methodName] = function () {
          if (!proto[methodName].overloadTable.hasOwnProperty(arguments.length)) {
            throwBindingError(
              "Function '" +
                humanName +
                "' called with an invalid number of arguments (" +
                arguments.length +
                ') - expects one of (' +
                proto[methodName].overloadTable +
                ')!'
            );
          }
          return proto[methodName].overloadTable[arguments.length].apply(this, arguments)
        };
        proto[methodName].overloadTable = [];
        proto[methodName].overloadTable[prevFunc.argCount] = prevFunc;
      }
    }
    function exposePublicSymbol(name, value, numArguments) {
      if (Module.hasOwnProperty(name)) {
        if (
          undefined === numArguments ||
          (undefined !== Module[name].overloadTable && undefined !== Module[name].overloadTable[numArguments])
        ) {
          throwBindingError("Cannot register public name '" + name + "' twice");
        }
        ensureOverloadTable(Module, name, name);
        if (Module.hasOwnProperty(numArguments)) {
          throwBindingError(
            'Cannot register multiple overloads of a function with the same number of arguments (' + numArguments + ')!'
          );
        }
        Module[name].overloadTable[numArguments] = value;
      } else {
        Module[name] = value;
        if (undefined !== numArguments) {
          Module[name].numArguments = numArguments;
        }
      }
    }
    function RegisteredClass(
      name,
      constructor,
      instancePrototype,
      rawDestructor,
      baseClass,
      getActualType,
      upcast,
      downcast
    ) {
      this.name = name;
      this.constructor = constructor;
      this.instancePrototype = instancePrototype;
      this.rawDestructor = rawDestructor;
      this.baseClass = baseClass;
      this.getActualType = getActualType;
      this.upcast = upcast;
      this.downcast = downcast;
      this.pureVirtualFunctions = [];
    }
    function upcastPointer(ptr, ptrClass, desiredClass) {
      while (ptrClass !== desiredClass) {
        if (!ptrClass.upcast) {
          throwBindingError('Expected null or instance of ' + desiredClass.name + ', got an instance of ' + ptrClass.name);
        }
        ptr = ptrClass.upcast(ptr);
        ptrClass = ptrClass.baseClass;
      }
      return ptr
    }
    function constNoSmartPtrRawPointerToWireType(destructors, handle) {
      if (handle === null) {
        if (this.isReference) {
          throwBindingError('null is not a valid ' + this.name);
        }
        return 0
      }
      if (!handle.$$) {
        throwBindingError('Cannot pass "' + _embind_repr(handle) + '" as a ' + this.name);
      }
      if (!handle.$$.ptr) {
        throwBindingError('Cannot pass deleted object as a pointer of type ' + this.name);
      }
      var handleClass = handle.$$.ptrType.registeredClass;
      var ptr = upcastPointer(handle.$$.ptr, handleClass, this.registeredClass);
      return ptr
    }
    function genericPointerToWireType(destructors, handle) {
      var ptr;
      if (handle === null) {
        if (this.isReference) {
          throwBindingError('null is not a valid ' + this.name);
        }
        if (this.isSmartPointer) {
          ptr = this.rawConstructor();
          if (destructors !== null) {
            destructors.push(this.rawDestructor, ptr);
          }
          return ptr
        } else {
          return 0
        }
      }
      if (!handle.$$) {
        throwBindingError('Cannot pass "' + _embind_repr(handle) + '" as a ' + this.name);
      }
      if (!handle.$$.ptr) {
        throwBindingError('Cannot pass deleted object as a pointer of type ' + this.name);
      }
      if (!this.isConst && handle.$$.ptrType.isConst) {
        throwBindingError(
          'Cannot convert argument of type ' +
            (handle.$$.smartPtrType ? handle.$$.smartPtrType.name : handle.$$.ptrType.name) +
            ' to parameter type ' +
            this.name
        );
      }
      var handleClass = handle.$$.ptrType.registeredClass;
      ptr = upcastPointer(handle.$$.ptr, handleClass, this.registeredClass);
      if (this.isSmartPointer) {
        if (undefined === handle.$$.smartPtr) {
          throwBindingError('Passing raw pointer to smart pointer is illegal');
        }
        switch (this.sharingPolicy) {
          case 0:
            if (handle.$$.smartPtrType === this) {
              ptr = handle.$$.smartPtr;
            } else {
              throwBindingError(
                'Cannot convert argument of type ' +
                  (handle.$$.smartPtrType ? handle.$$.smartPtrType.name : handle.$$.ptrType.name) +
                  ' to parameter type ' +
                  this.name
              );
            }
            break
          case 1:
            ptr = handle.$$.smartPtr;
            break
          case 2:
            if (handle.$$.smartPtrType === this) {
              ptr = handle.$$.smartPtr;
            } else {
              var clonedHandle = handle['clone']();
              ptr = this.rawShare(
                ptr,
                __emval_register(function () {
                  clonedHandle['delete']();
                })
              );
              if (destructors !== null) {
                destructors.push(this.rawDestructor, ptr);
              }
            }
            break
          default:
            throwBindingError('Unsupporting sharing policy');
        }
      }
      return ptr
    }
    function nonConstNoSmartPtrRawPointerToWireType(destructors, handle) {
      if (handle === null) {
        if (this.isReference) {
          throwBindingError('null is not a valid ' + this.name);
        }
        return 0
      }
      if (!handle.$$) {
        throwBindingError('Cannot pass "' + _embind_repr(handle) + '" as a ' + this.name);
      }
      if (!handle.$$.ptr) {
        throwBindingError('Cannot pass deleted object as a pointer of type ' + this.name);
      }
      if (handle.$$.ptrType.isConst) {
        throwBindingError('Cannot convert argument of type ' + handle.$$.ptrType.name + ' to parameter type ' + this.name);
      }
      var handleClass = handle.$$.ptrType.registeredClass;
      var ptr = upcastPointer(handle.$$.ptr, handleClass, this.registeredClass);
      return ptr
    }
    function RegisteredPointer_getPointee(ptr) {
      if (this.rawGetPointee) {
        ptr = this.rawGetPointee(ptr);
      }
      return ptr
    }
    function RegisteredPointer_destructor(ptr) {
      if (this.rawDestructor) {
        this.rawDestructor(ptr);
      }
    }
    function RegisteredPointer_deleteObject(handle) {
      if (handle !== null) {
        handle['delete']();
      }
    }
    function downcastPointer(ptr, ptrClass, desiredClass) {
      if (ptrClass === desiredClass) {
        return ptr
      }
      if (undefined === desiredClass.baseClass) {
        return null
      }
      var rv = downcastPointer(ptr, ptrClass, desiredClass.baseClass);
      if (rv === null) {
        return null
      }
      return desiredClass.downcast(rv)
    }
    function getInheritedInstanceCount() {
      return Object.keys(registeredInstances).length
    }
    function getLiveInheritedInstances() {
      var rv = [];
      for (var k in registeredInstances) {
        if (registeredInstances.hasOwnProperty(k)) {
          rv.push(registeredInstances[k]);
        }
      }
      return rv
    }
    function setDelayFunction(fn) {
      delayFunction = fn;
      if (deletionQueue.length && delayFunction) {
        delayFunction(flushPendingDeletes);
      }
    }
    function init_embind() {
      Module['getInheritedInstanceCount'] = getInheritedInstanceCount;
      Module['getLiveInheritedInstances'] = getLiveInheritedInstances;
      Module['flushPendingDeletes'] = flushPendingDeletes;
      Module['setDelayFunction'] = setDelayFunction;
    }
    var registeredInstances = {};
    function getBasestPointer(class_, ptr) {
      if (ptr === undefined) {
        throwBindingError('ptr should not be undefined');
      }
      while (class_.baseClass) {
        ptr = class_.upcast(ptr);
        class_ = class_.baseClass;
      }
      return ptr
    }
    function getInheritedInstance(class_, ptr) {
      ptr = getBasestPointer(class_, ptr);
      return registeredInstances[ptr]
    }
    function makeClassHandle(prototype, record) {
      if (!record.ptrType || !record.ptr) {
        throwInternalError('makeClassHandle requires ptr and ptrType');
      }
      var hasSmartPtrType = !!record.smartPtrType;
      var hasSmartPtr = !!record.smartPtr;
      if (hasSmartPtrType !== hasSmartPtr) {
        throwInternalError('Both smartPtrType and smartPtr must be specified');
      }
      record.count = { value: 1 };
      return Object.create(prototype, { $$: { value: record } })
    }
    function RegisteredPointer_fromWireType(ptr) {
      var rawPointer = this.getPointee(ptr);
      if (!rawPointer) {
        this.destructor(ptr);
        return null
      }
      var registeredInstance = getInheritedInstance(this.registeredClass, rawPointer);
      if (undefined !== registeredInstance) {
        if (0 === registeredInstance.$$.count.value) {
          registeredInstance.$$.ptr = rawPointer;
          registeredInstance.$$.smartPtr = ptr;
          return registeredInstance['clone']()
        } else {
          var rv = registeredInstance['clone']();
          this.destructor(ptr);
          return rv
        }
      }
      function makeDefaultHandle() {
        if (this.isSmartPointer) {
          return makeClassHandle(this.registeredClass.instancePrototype, {
            ptrType: this.pointeeType,
            ptr: rawPointer,
            smartPtrType: this,
            smartPtr: ptr,
          })
        } else {
          return makeClassHandle(this.registeredClass.instancePrototype, { ptrType: this, ptr: ptr })
        }
      }
      var actualType = this.registeredClass.getActualType(rawPointer);
      var registeredPointerRecord = registeredPointers[actualType];
      if (!registeredPointerRecord) {
        return makeDefaultHandle.call(this)
      }
      var toType;
      if (this.isConst) {
        toType = registeredPointerRecord.constPointerType;
      } else {
        toType = registeredPointerRecord.pointerType;
      }
      var dp = downcastPointer(rawPointer, this.registeredClass, toType.registeredClass);
      if (dp === null) {
        return makeDefaultHandle.call(this)
      }
      if (this.isSmartPointer) {
        return makeClassHandle(toType.registeredClass.instancePrototype, {
          ptrType: toType,
          ptr: dp,
          smartPtrType: this,
          smartPtr: ptr,
        })
      } else {
        return makeClassHandle(toType.registeredClass.instancePrototype, { ptrType: toType, ptr: dp })
      }
    }
    function init_RegisteredPointer() {
      RegisteredPointer.prototype.getPointee = RegisteredPointer_getPointee;
      RegisteredPointer.prototype.destructor = RegisteredPointer_destructor;
      RegisteredPointer.prototype['argPackAdvance'] = 8;
      RegisteredPointer.prototype['readValueFromPointer'] = simpleReadValueFromPointer;
      RegisteredPointer.prototype['deleteObject'] = RegisteredPointer_deleteObject;
      RegisteredPointer.prototype['fromWireType'] = RegisteredPointer_fromWireType;
    }
    function RegisteredPointer(
      name,
      registeredClass,
      isReference,
      isConst,
      isSmartPointer,
      pointeeType,
      sharingPolicy,
      rawGetPointee,
      rawConstructor,
      rawShare,
      rawDestructor
    ) {
      this.name = name;
      this.registeredClass = registeredClass;
      this.isReference = isReference;
      this.isConst = isConst;
      this.isSmartPointer = isSmartPointer;
      this.pointeeType = pointeeType;
      this.sharingPolicy = sharingPolicy;
      this.rawGetPointee = rawGetPointee;
      this.rawConstructor = rawConstructor;
      this.rawShare = rawShare;
      this.rawDestructor = rawDestructor;
      if (!isSmartPointer && registeredClass.baseClass === undefined) {
        if (isConst) {
          this['toWireType'] = constNoSmartPtrRawPointerToWireType;
          this.destructorFunction = null;
        } else {
          this['toWireType'] = nonConstNoSmartPtrRawPointerToWireType;
          this.destructorFunction = null;
        }
      } else {
        this['toWireType'] = genericPointerToWireType;
      }
    }
    function replacePublicSymbol(name, value, numArguments) {
      if (!Module.hasOwnProperty(name)) {
        throwInternalError('Replacing nonexistant public symbol');
      }
      if (undefined !== Module[name].overloadTable && undefined !== numArguments) {
        Module[name].overloadTable[numArguments] = value;
      } else {
        Module[name] = value;
        Module[name].argCount = numArguments;
      }
    }
    function embind__requireFunction(signature, rawFunction) {
      signature = readLatin1String(signature);
      function makeDynCaller(dynCall) {
        var args = [];
        for (var i = 1; i < signature.length; ++i) {
          args.push('a' + i);
        }
        var name = 'dynCall_' + signature + '_' + rawFunction;
        var body = 'return function ' + name + '(' + args.join(', ') + ') {\n';
        body += '    return dynCall(rawFunction' + (args.length ? ', ' : '') + args.join(', ') + ');\n';
        body += '};\n';
        return new Function('dynCall', 'rawFunction', body)(dynCall, rawFunction)
      }
      var fp;
      if (Module['FUNCTION_TABLE_' + signature] !== undefined) {
        fp = Module['FUNCTION_TABLE_' + signature][rawFunction];
      } else if (typeof FUNCTION_TABLE !== 'undefined') {
        fp = FUNCTION_TABLE[rawFunction];
      } else {
        var dc = Module['asm']['dynCall_' + signature];
        if (dc === undefined) {
          dc = Module['asm']['dynCall_' + signature.replace(/f/g, 'd')];
          if (dc === undefined) {
            throwBindingError('No dynCall invoker for signature: ' + signature);
          }
        }
        fp = makeDynCaller(dc);
      }
      if (typeof fp !== 'function') {
        throwBindingError('unknown function pointer with signature ' + signature + ': ' + rawFunction);
      }
      return fp
    }
    var UnboundTypeError = undefined;
    function getTypeName(type) {
      var ptr = ___getTypeName(type);
      var rv = readLatin1String(ptr);
      _free(ptr);
      return rv
    }
    function throwUnboundTypeError(message, types) {
      var unboundTypes = [];
      var seen = {};
      function visit(type) {
        if (seen[type]) {
          return
        }
        if (registeredTypes[type]) {
          return
        }
        if (typeDependencies[type]) {
          typeDependencies[type].forEach(visit);
          return
        }
        unboundTypes.push(type);
        seen[type] = true;
      }
      types.forEach(visit);
      throw new UnboundTypeError(message + ': ' + unboundTypes.map(getTypeName).join([', ']))
    }
    function __embind_register_class(
      rawType,
      rawPointerType,
      rawConstPointerType,
      baseClassRawType,
      getActualTypeSignature,
      getActualType,
      upcastSignature,
      upcast,
      downcastSignature,
      downcast,
      name,
      destructorSignature,
      rawDestructor
    ) {
      name = readLatin1String(name);
      getActualType = embind__requireFunction(getActualTypeSignature, getActualType);
      if (upcast) {
        upcast = embind__requireFunction(upcastSignature, upcast);
      }
      if (downcast) {
        downcast = embind__requireFunction(downcastSignature, downcast);
      }
      rawDestructor = embind__requireFunction(destructorSignature, rawDestructor);
      var legalFunctionName = makeLegalFunctionName(name);
      exposePublicSymbol(legalFunctionName, function () {
        throwUnboundTypeError('Cannot construct ' + name + ' due to unbound types', [baseClassRawType]);
      });
      whenDependentTypesAreResolved(
        [rawType, rawPointerType, rawConstPointerType],
        baseClassRawType ? [baseClassRawType] : [],
        function (base) {
          base = base[0];
          var baseClass;
          var basePrototype;
          if (baseClassRawType) {
            baseClass = base.registeredClass;
            basePrototype = baseClass.instancePrototype;
          } else {
            basePrototype = ClassHandle.prototype;
          }
          var constructor = createNamedFunction(legalFunctionName, function () {
            if (Object.getPrototypeOf(this) !== instancePrototype) {
              throw new BindingError("Use 'new' to construct " + name)
            }
            if (undefined === registeredClass.constructor_body) {
              throw new BindingError(name + ' has no accessible constructor')
            }
            var body = registeredClass.constructor_body[arguments.length];
            if (undefined === body) {
              throw new BindingError(
                'Tried to invoke ctor of ' +
                  name +
                  ' with invalid number of parameters (' +
                  arguments.length +
                  ') - expected (' +
                  Object.keys(registeredClass.constructor_body).toString() +
                  ') parameters instead!'
              )
            }
            return body.apply(this, arguments)
          });
          var instancePrototype = Object.create(basePrototype, { constructor: { value: constructor } });
          constructor.prototype = instancePrototype;
          var registeredClass = new RegisteredClass(
            name,
            constructor,
            instancePrototype,
            rawDestructor,
            baseClass,
            getActualType,
            upcast,
            downcast
          );
          var referenceConverter = new RegisteredPointer(name, registeredClass, true, false, false);
          var pointerConverter = new RegisteredPointer(name + '*', registeredClass, false, false, false);
          var constPointerConverter = new RegisteredPointer(name + ' const*', registeredClass, false, true, false);
          registeredPointers[rawType] = { pointerType: pointerConverter, constPointerType: constPointerConverter };
          replacePublicSymbol(legalFunctionName, constructor);
          return [referenceConverter, pointerConverter, constPointerConverter]
        }
      );
    }
    function heap32VectorToArray(count, firstElement) {
      var array = [];
      for (var i = 0; i < count; i++) {
        array.push(HEAP32[(firstElement >> 2) + i]);
      }
      return array
    }
    function __embind_register_class_constructor(
      rawClassType,
      argCount,
      rawArgTypesAddr,
      invokerSignature,
      invoker,
      rawConstructor
    ) {
      var rawArgTypes = heap32VectorToArray(argCount, rawArgTypesAddr);
      invoker = embind__requireFunction(invokerSignature, invoker);
      whenDependentTypesAreResolved([], [rawClassType], function (classType) {
        classType = classType[0];
        var humanName = 'constructor ' + classType.name;
        if (undefined === classType.registeredClass.constructor_body) {
          classType.registeredClass.constructor_body = [];
        }
        if (undefined !== classType.registeredClass.constructor_body[argCount - 1]) {
          throw new BindingError(
            'Cannot register multiple constructors with identical number of parameters (' +
              (argCount - 1) +
              ") for class '" +
              classType.name +
              "'! Overload resolution is currently only performed using the parameter count, not actual type info!"
          )
        }
        classType.registeredClass.constructor_body[argCount - 1] = function unboundTypeHandler() {
          throwUnboundTypeError('Cannot construct ' + classType.name + ' due to unbound types', rawArgTypes);
        };
        whenDependentTypesAreResolved([], rawArgTypes, function (argTypes) {
          classType.registeredClass.constructor_body[argCount - 1] = function constructor_body() {
            if (arguments.length !== argCount - 1) {
              throwBindingError(humanName + ' called with ' + arguments.length + ' arguments, expected ' + (argCount - 1));
            }
            var destructors = [];
            var args = new Array(argCount);
            args[0] = rawConstructor;
            for (var i = 1; i < argCount; ++i) {
              args[i] = argTypes[i]['toWireType'](destructors, arguments[i - 1]);
            }
            var ptr = invoker.apply(null, args);
            runDestructors(destructors);
            return argTypes[0]['fromWireType'](ptr)
          };
          return []
        });
        return []
      });
    }
    function new_(constructor, argumentList) {
      if (!(constructor instanceof Function)) {
        throw new TypeError('new_ called with constructor type ' + typeof constructor + ' which is not a function')
      }
      var dummy = createNamedFunction(constructor.name || 'unknownFunctionName', function () {});
      dummy.prototype = constructor.prototype;
      var obj = new dummy();
      var r = constructor.apply(obj, argumentList);
      return r instanceof Object ? r : obj
    }
    function craftInvokerFunction(humanName, argTypes, classType, cppInvokerFunc, cppTargetFunc) {
      var argCount = argTypes.length;
      if (argCount < 2) {
        throwBindingError("argTypes array size mismatch! Must at least get return value and 'this' types!");
      }
      var isClassMethodFunc = argTypes[1] !== null && classType !== null;
      var needsDestructorStack = false;
      for (var i = 1; i < argTypes.length; ++i) {
        if (argTypes[i] !== null && argTypes[i].destructorFunction === undefined) {
          needsDestructorStack = true;
          break
        }
      }
      var returns = argTypes[0].name !== 'void';
      var argsList = '';
      var argsListWired = '';
      for (var i = 0; i < argCount - 2; ++i) {
        argsList += (i !== 0 ? ', ' : '') + 'arg' + i;
        argsListWired += (i !== 0 ? ', ' : '') + 'arg' + i + 'Wired';
      }
      var invokerFnBody =
        'return function ' +
        makeLegalFunctionName(humanName) +
        '(' +
        argsList +
        ') {\n' +
        'if (arguments.length !== ' +
        (argCount - 2) +
        ') {\n' +
        "throwBindingError('function " +
        humanName +
        " called with ' + arguments.length + ' arguments, expected " +
        (argCount - 2) +
        " args!');\n" +
        '}\n';
      if (needsDestructorStack) {
        invokerFnBody += 'var destructors = [];\n';
      }
      var dtorStack = needsDestructorStack ? 'destructors' : 'null';
      var args1 = ['throwBindingError', 'invoker', 'fn', 'runDestructors', 'retType', 'classParam'];
      var args2 = [throwBindingError, cppInvokerFunc, cppTargetFunc, runDestructors, argTypes[0], argTypes[1]];
      if (isClassMethodFunc) {
        invokerFnBody += 'var thisWired = classParam.toWireType(' + dtorStack + ', this);\n';
      }
      for (var i = 0; i < argCount - 2; ++i) {
        invokerFnBody +=
          'var arg' +
          i +
          'Wired = argType' +
          i +
          '.toWireType(' +
          dtorStack +
          ', arg' +
          i +
          '); // ' +
          argTypes[i + 2].name +
          '\n';
        args1.push('argType' + i);
        args2.push(argTypes[i + 2]);
      }
      if (isClassMethodFunc) {
        argsListWired = 'thisWired' + (argsListWired.length > 0 ? ', ' : '') + argsListWired;
      }
      invokerFnBody +=
        (returns ? 'var rv = ' : '') + 'invoker(fn' + (argsListWired.length > 0 ? ', ' : '') + argsListWired + ');\n';
      if (needsDestructorStack) {
        invokerFnBody += 'runDestructors(destructors);\n';
      } else {
        for (var i = isClassMethodFunc ? 1 : 2; i < argTypes.length; ++i) {
          var paramName = i === 1 ? 'thisWired' : 'arg' + (i - 2) + 'Wired';
          if (argTypes[i].destructorFunction !== null) {
            invokerFnBody += paramName + '_dtor(' + paramName + '); // ' + argTypes[i].name + '\n';
            args1.push(paramName + '_dtor');
            args2.push(argTypes[i].destructorFunction);
          }
        }
      }
      if (returns) {
        invokerFnBody += 'var ret = retType.fromWireType(rv);\n' + 'return ret;\n';
      }
      invokerFnBody += '}\n';
      args1.push(invokerFnBody);
      var invokerFunction = new_(Function, args1).apply(null, args2);
      return invokerFunction
    }
    function __embind_register_class_function(
      rawClassType,
      methodName,
      argCount,
      rawArgTypesAddr,
      invokerSignature,
      rawInvoker,
      context,
      isPureVirtual
    ) {
      var rawArgTypes = heap32VectorToArray(argCount, rawArgTypesAddr);
      methodName = readLatin1String(methodName);
      rawInvoker = embind__requireFunction(invokerSignature, rawInvoker);
      whenDependentTypesAreResolved([], [rawClassType], function (classType) {
        classType = classType[0];
        var humanName = classType.name + '.' + methodName;
        if (isPureVirtual) {
          classType.registeredClass.pureVirtualFunctions.push(methodName);
        }
        function unboundTypesHandler() {
          throwUnboundTypeError('Cannot call ' + humanName + ' due to unbound types', rawArgTypes);
        }
        var proto = classType.registeredClass.instancePrototype;
        var method = proto[methodName];
        if (
          undefined === method ||
          (undefined === method.overloadTable && method.className !== classType.name && method.argCount === argCount - 2)
        ) {
          unboundTypesHandler.argCount = argCount - 2;
          unboundTypesHandler.className = classType.name;
          proto[methodName] = unboundTypesHandler;
        } else {
          ensureOverloadTable(proto, methodName, humanName);
          proto[methodName].overloadTable[argCount - 2] = unboundTypesHandler;
        }
        whenDependentTypesAreResolved([], rawArgTypes, function (argTypes) {
          var memberFunction = craftInvokerFunction(humanName, argTypes, classType, rawInvoker, context);
          if (undefined === proto[methodName].overloadTable) {
            memberFunction.argCount = argCount - 2;
            proto[methodName] = memberFunction;
          } else {
            proto[methodName].overloadTable[argCount - 2] = memberFunction;
          }
          return []
        });
        return []
      });
    }
    var emval_free_list = [];
    var emval_handle_array = [{}, { value: undefined }, { value: null }, { value: true }, { value: false }];
    function __emval_decref(handle) {
      if (handle > 4 && 0 === --emval_handle_array[handle].refcount) {
        emval_handle_array[handle] = undefined;
        emval_free_list.push(handle);
      }
    }
    function count_emval_handles() {
      var count = 0;
      for (var i = 5; i < emval_handle_array.length; ++i) {
        if (emval_handle_array[i] !== undefined) {
          ++count;
        }
      }
      return count
    }
    function get_first_emval() {
      for (var i = 5; i < emval_handle_array.length; ++i) {
        if (emval_handle_array[i] !== undefined) {
          return emval_handle_array[i]
        }
      }
      return null
    }
    function init_emval() {
      Module['count_emval_handles'] = count_emval_handles;
      Module['get_first_emval'] = get_first_emval;
    }
    function __emval_register(value) {
      switch (value) {
        case undefined: {
          return 1
        }
        case null: {
          return 2
        }
        case true: {
          return 3
        }
        case false: {
          return 4
        }
        default: {
          var handle = emval_free_list.length ? emval_free_list.pop() : emval_handle_array.length;
          emval_handle_array[handle] = { refcount: 1, value: value };
          return handle
        }
      }
    }
    function __embind_register_emval(rawType, name) {
      name = readLatin1String(name);
      registerType(rawType, {
        name: name,
        fromWireType: function (handle) {
          var rv = emval_handle_array[handle].value;
          __emval_decref(handle);
          return rv
        },
        toWireType: function (destructors, value) {
          return __emval_register(value)
        },
        argPackAdvance: 8,
        readValueFromPointer: simpleReadValueFromPointer,
        destructorFunction: null,
      });
    }
    function _embind_repr(v) {
      if (v === null) {
        return 'null'
      }
      var t = typeof v;
      if (t === 'object' || t === 'array' || t === 'function') {
        return v.toString()
      } else {
        return '' + v
      }
    }
    function floatReadValueFromPointer(name, shift) {
      switch (shift) {
        case 2:
          return function (pointer) {
            return this['fromWireType'](HEAPF32[pointer >> 2])
          }
        case 3:
          return function (pointer) {
            return this['fromWireType'](HEAPF64[pointer >> 3])
          }
        default:
          throw new TypeError('Unknown float type: ' + name)
      }
    }
    function __embind_register_float(rawType, name, size) {
      var shift = getShiftFromSize(size);
      name = readLatin1String(name);
      registerType(rawType, {
        name: name,
        fromWireType: function (value) {
          return value
        },
        toWireType: function (destructors, value) {
          if (typeof value !== 'number' && typeof value !== 'boolean') {
            throw new TypeError('Cannot convert "' + _embind_repr(value) + '" to ' + this.name)
          }
          return value
        },
        argPackAdvance: 8,
        readValueFromPointer: floatReadValueFromPointer(name, shift),
        destructorFunction: null,
      });
    }
    function integerReadValueFromPointer(name, shift, signed) {
      switch (shift) {
        case 0:
          return signed
            ? function readS8FromPointer(pointer) {
                return HEAP8[pointer]
              }
            : function readU8FromPointer(pointer) {
                return HEAPU8[pointer]
              }
        case 1:
          return signed
            ? function readS16FromPointer(pointer) {
                return HEAP16[pointer >> 1]
              }
            : function readU16FromPointer(pointer) {
                return HEAPU16[pointer >> 1]
              }
        case 2:
          return signed
            ? function readS32FromPointer(pointer) {
                return HEAP32[pointer >> 2]
              }
            : function readU32FromPointer(pointer) {
                return HEAPU32[pointer >> 2]
              }
        default:
          throw new TypeError('Unknown integer type: ' + name)
      }
    }
    function __embind_register_integer(primitiveType, name, size, minRange, maxRange) {
      name = readLatin1String(name);
      if (maxRange === -1) {
        maxRange = 4294967295;
      }
      var shift = getShiftFromSize(size);
      var fromWireType = function (value) {
        return value
      };
      if (minRange === 0) {
        var bitshift = 32 - 8 * size;
        fromWireType = function (value) {
          return (value << bitshift) >>> bitshift
        };
      }
      var isUnsignedType = name.indexOf('unsigned') != -1;
      registerType(primitiveType, {
        name: name,
        fromWireType: fromWireType,
        toWireType: function (destructors, value) {
          if (typeof value !== 'number' && typeof value !== 'boolean') {
            throw new TypeError('Cannot convert "' + _embind_repr(value) + '" to ' + this.name)
          }
          if (value < minRange || value > maxRange) {
            throw new TypeError(
              'Passing a number "' +
                _embind_repr(value) +
                '" from JS side to C/C++ side to an argument of type "' +
                name +
                '", which is outside the valid range [' +
                minRange +
                ', ' +
                maxRange +
                ']!'
            )
          }
          return isUnsignedType ? value >>> 0 : value | 0
        },
        argPackAdvance: 8,
        readValueFromPointer: integerReadValueFromPointer(name, shift, minRange !== 0),
        destructorFunction: null,
      });
    }
    function __embind_register_memory_view(rawType, dataTypeIndex, name) {
      var typeMapping = [
        Int8Array,
        Uint8Array,
        Int16Array,
        Uint16Array,
        Int32Array,
        Uint32Array,
        Float32Array,
        Float64Array,
      ];
      var TA = typeMapping[dataTypeIndex];
      function decodeMemoryView(handle) {
        handle = handle >> 2;
        var heap = HEAPU32;
        var size = heap[handle];
        var data = heap[handle + 1];
        return new TA(heap['buffer'], data, size)
      }
      name = readLatin1String(name);
      registerType(
        rawType,
        { name: name, fromWireType: decodeMemoryView, argPackAdvance: 8, readValueFromPointer: decodeMemoryView },
        { ignoreDuplicateRegistrations: true }
      );
    }
    function __embind_register_std_string(rawType, name) {
      name = readLatin1String(name);
      registerType(rawType, {
        name: name,
        fromWireType: function (value) {
          var length = HEAPU32[value >> 2];
          var a = new Array(length);
          for (var i = 0; i < length; ++i) {
            a[i] = String.fromCharCode(HEAPU8[value + 4 + i]);
          }
          _free(value);
          return a.join('')
        },
        toWireType: function (destructors, value) {
          if (value instanceof ArrayBuffer) {
            value = new Uint8Array(value);
          }
          function getTAElement(ta, index) {
            return ta[index]
          }
          function getStringElement(string, index) {
            return string.charCodeAt(index)
          }
          var getElement;
          if (value instanceof Uint8Array) {
            getElement = getTAElement;
          } else if (value instanceof Uint8ClampedArray) {
            getElement = getTAElement;
          } else if (value instanceof Int8Array) {
            getElement = getTAElement;
          } else if (typeof value === 'string') {
            getElement = getStringElement;
          } else {
            throwBindingError('Cannot pass non-string to std::string');
          }
          var length = value.length;
          var ptr = _malloc(4 + length);
          HEAPU32[ptr >> 2] = length;
          for (var i = 0; i < length; ++i) {
            var charCode = getElement(value, i);
            if (charCode > 255) {
              _free(ptr);
              throwBindingError('String has UTF-16 code units that do not fit in 8 bits');
            }
            HEAPU8[ptr + 4 + i] = charCode;
          }
          if (destructors !== null) {
            destructors.push(_free, ptr);
          }
          return ptr
        },
        argPackAdvance: 8,
        readValueFromPointer: simpleReadValueFromPointer,
        destructorFunction: function (ptr) {
          _free(ptr);
        },
      });
    }
    function __embind_register_std_wstring(rawType, charSize, name) {
      name = readLatin1String(name);
      var getHeap, shift;
      if (charSize === 2) {
        getHeap = function () {
          return HEAPU16
        };
        shift = 1;
      } else if (charSize === 4) {
        getHeap = function () {
          return HEAPU32
        };
        shift = 2;
      }
      registerType(rawType, {
        name: name,
        fromWireType: function (value) {
          var HEAP = getHeap();
          var length = HEAPU32[value >> 2];
          var a = new Array(length);
          var start = (value + 4) >> shift;
          for (var i = 0; i < length; ++i) {
            a[i] = String.fromCharCode(HEAP[start + i]);
          }
          _free(value);
          return a.join('')
        },
        toWireType: function (destructors, value) {
          var HEAP = getHeap();
          var length = value.length;
          var ptr = _malloc(4 + length * charSize);
          HEAPU32[ptr >> 2] = length;
          var start = (ptr + 4) >> shift;
          for (var i = 0; i < length; ++i) {
            HEAP[start + i] = value.charCodeAt(i);
          }
          if (destructors !== null) {
            destructors.push(_free, ptr);
          }
          return ptr
        },
        argPackAdvance: 8,
        readValueFromPointer: simpleReadValueFromPointer,
        destructorFunction: function (ptr) {
          _free(ptr);
        },
      });
    }
    function __embind_register_value_object(
      rawType,
      name,
      constructorSignature,
      rawConstructor,
      destructorSignature,
      rawDestructor
    ) {
      structRegistrations[rawType] = {
        name: readLatin1String(name),
        rawConstructor: embind__requireFunction(constructorSignature, rawConstructor),
        rawDestructor: embind__requireFunction(destructorSignature, rawDestructor),
        fields: [],
      };
    }
    function __embind_register_value_object_field(
      structType,
      fieldName,
      getterReturnType,
      getterSignature,
      getter,
      getterContext,
      setterArgumentType,
      setterSignature,
      setter,
      setterContext
    ) {
      structRegistrations[structType].fields.push({
        fieldName: readLatin1String(fieldName),
        getterReturnType: getterReturnType,
        getter: embind__requireFunction(getterSignature, getter),
        getterContext: getterContext,
        setterArgumentType: setterArgumentType,
        setter: embind__requireFunction(setterSignature, setter),
        setterContext: setterContext,
      });
    }
    function __embind_register_void(rawType, name) {
      name = readLatin1String(name);
      registerType(rawType, {
        isVoid: true,
        name: name,
        argPackAdvance: 0,
        fromWireType: function () {
          return undefined
        },
        toWireType: function (destructors, o) {
          return undefined
        },
      });
    }
    function _abort() {
      Module['abort']();
    }
    var _environ = STATICTOP;
    STATICTOP += 16;
    function ___buildEnvironment(env) {
      var MAX_ENV_VALUES = 64;
      var TOTAL_ENV_SIZE = 1024;
      var poolPtr;
      var envPtr;
      if (!___buildEnvironment.called) {
        ___buildEnvironment.called = true;
        ENV['USER'] = ENV['LOGNAME'] = 'web_user';
        ENV['PATH'] = '/';
        ENV['PWD'] = '/';
        ENV['HOME'] = '/home/web_user';
        ENV['LANG'] = 'C.UTF-8';
        ENV['_'] = Module['thisProgram'];
        poolPtr = staticAlloc(TOTAL_ENV_SIZE);
        envPtr = staticAlloc(MAX_ENV_VALUES * 4);
        HEAP32[envPtr >> 2] = poolPtr;
        HEAP32[_environ >> 2] = envPtr;
      } else {
        envPtr = HEAP32[_environ >> 2];
        poolPtr = HEAP32[envPtr >> 2];
      }
      var strings = [];
      var totalSize = 0;
      for (var key in env) {
        if (typeof env[key] === 'string') {
          var line = key + '=' + env[key];
          strings.push(line);
          totalSize += line.length;
        }
      }
      if (totalSize > TOTAL_ENV_SIZE) {
        throw new Error('Environment size exceeded TOTAL_ENV_SIZE!')
      }
      var ptrSize = 4;
      for (var i = 0; i < strings.length; i++) {
        var line = strings[i];
        writeAsciiToMemory(line, poolPtr);
        HEAP32[(envPtr + i * ptrSize) >> 2] = poolPtr;
        poolPtr += line.length + 1;
      }
      HEAP32[(envPtr + strings.length * ptrSize) >> 2] = 0;
    }
    var ENV = {};
    function _getenv(name) {
      if (name === 0) return 0
      name = Pointer_stringify(name);
      if (!ENV.hasOwnProperty(name)) return 0
      if (_getenv.ret) _free(_getenv.ret);
      _getenv.ret = allocateUTF8(ENV[name]);
      return _getenv.ret
    }
    function _getgrnam() {
      Module['printErr']('missing function: getgrnam');
      abort(-1);
    }
    function _getpwnam() {
      throw 'getpwnam: TODO'
    }
    function _jsClose() {
      return jsAPI.close.apply(null, arguments)
    }
    function _jsCreate(filename) {
      return jsAPI.create.call(null, UTF32ToString(filename))
    }
    function _jsOpen(filename) {
      return jsAPI.open.call(null, UTF32ToString(filename))
    }
    function _jsRead() {
      return jsAPI.read.apply(null, arguments)
    }
    function _jsSeek(fd, offset, method) {
      return jsAPI.seek.call(null, fd, offset, UTF8ToString(method))
    }
    function _jsTell() {
      return jsAPI.tell.apply(null, arguments)
    }
    function _jsWrite() {
      return jsAPI.write.apply(null, arguments)
    }
    function _llvm_eh_typeid_for(type) {
      return type
    }
    var ___tm_current = STATICTOP;
    STATICTOP += 48;
    allocate(intArrayFromString('GMT'), 'i8', ALLOC_STATIC);
    var _tzname = STATICTOP;
    STATICTOP += 16;
    var _daylight = STATICTOP;
    STATICTOP += 16;
    var _timezone = STATICTOP;
    STATICTOP += 16;
    function _tzset() {
      if (_tzset.called) return
      _tzset.called = true;
      HEAP32[_timezone >> 2] = new Date().getTimezoneOffset() * 60;
      var winter = new Date(2e3, 0, 1);
      var summer = new Date(2e3, 6, 1);
      HEAP32[_daylight >> 2] = Number(winter.getTimezoneOffset() != summer.getTimezoneOffset());
      function extractZone(date) {
        var match = date.toTimeString().match(/\(([A-Za-z ]+)\)$/);
        return match ? match[1] : 'GMT'
      }
      var winterName = extractZone(winter);
      var summerName = extractZone(summer);
      var winterNamePtr = allocate(intArrayFromString(winterName), 'i8', ALLOC_NORMAL);
      var summerNamePtr = allocate(intArrayFromString(summerName), 'i8', ALLOC_NORMAL);
      if (summer.getTimezoneOffset() < winter.getTimezoneOffset()) {
        HEAP32[_tzname >> 2] = winterNamePtr;
        HEAP32[(_tzname + 4) >> 2] = summerNamePtr;
      } else {
        HEAP32[_tzname >> 2] = summerNamePtr;
        HEAP32[(_tzname + 4) >> 2] = winterNamePtr;
      }
    }
    function _localtime_r(time, tmPtr) {
      _tzset();
      var date = new Date(HEAP32[time >> 2] * 1e3);
      HEAP32[tmPtr >> 2] = date.getSeconds();
      HEAP32[(tmPtr + 4) >> 2] = date.getMinutes();
      HEAP32[(tmPtr + 8) >> 2] = date.getHours();
      HEAP32[(tmPtr + 12) >> 2] = date.getDate();
      HEAP32[(tmPtr + 16) >> 2] = date.getMonth();
      HEAP32[(tmPtr + 20) >> 2] = date.getFullYear() - 1900;
      HEAP32[(tmPtr + 24) >> 2] = date.getDay();
      var start = new Date(date.getFullYear(), 0, 1);
      var yday = ((date.getTime() - start.getTime()) / (1e3 * 60 * 60 * 24)) | 0;
      HEAP32[(tmPtr + 28) >> 2] = yday;
      HEAP32[(tmPtr + 36) >> 2] = -(date.getTimezoneOffset() * 60);
      var summerOffset = new Date(2e3, 6, 1).getTimezoneOffset();
      var winterOffset = start.getTimezoneOffset();
      var dst = (summerOffset != winterOffset && date.getTimezoneOffset() == Math.min(winterOffset, summerOffset)) | 0;
      HEAP32[(tmPtr + 32) >> 2] = dst;
      var zonePtr = HEAP32[(_tzname + (dst ? 4 : 0)) >> 2];
      HEAP32[(tmPtr + 40) >> 2] = zonePtr;
      return tmPtr
    }
    function _localtime(time) {
      return _localtime_r(time, ___tm_current)
    }
    function _emscripten_memcpy_big(dest, src, num) {
      HEAPU8.set(HEAPU8.subarray(src, src + num), dest);
      return dest
    }
    function _mktime(tmPtr) {
      _tzset();
      var date = new Date(
        HEAP32[(tmPtr + 20) >> 2] + 1900,
        HEAP32[(tmPtr + 16) >> 2],
        HEAP32[(tmPtr + 12) >> 2],
        HEAP32[(tmPtr + 8) >> 2],
        HEAP32[(tmPtr + 4) >> 2],
        HEAP32[tmPtr >> 2],
        0
      );
      var dst = HEAP32[(tmPtr + 32) >> 2];
      var guessedOffset = date.getTimezoneOffset();
      var start = new Date(date.getFullYear(), 0, 1);
      var summerOffset = new Date(2e3, 6, 1).getTimezoneOffset();
      var winterOffset = start.getTimezoneOffset();
      var dstOffset = Math.min(winterOffset, summerOffset);
      if (dst < 0) {
        HEAP32[(tmPtr + 32) >> 2] = Number(summerOffset != winterOffset && dstOffset == guessedOffset);
      } else if (dst > 0 != (dstOffset == guessedOffset)) {
        var nonDstOffset = Math.max(winterOffset, summerOffset);
        var trueOffset = dst > 0 ? dstOffset : nonDstOffset;
        date.setTime(date.getTime() + (trueOffset - guessedOffset) * 6e4);
      }
      HEAP32[(tmPtr + 24) >> 2] = date.getDay();
      var yday = ((date.getTime() - start.getTime()) / (1e3 * 60 * 60 * 24)) | 0;
      HEAP32[(tmPtr + 28) >> 2] = yday;
      return (date.getTime() / 1e3) | 0
    }
    var PTHREAD_SPECIFIC = {};
    function _pthread_getspecific(key) {
      return PTHREAD_SPECIFIC[key] || 0
    }
    var PTHREAD_SPECIFIC_NEXT_KEY = 1;
    function _pthread_key_create(key, destructor) {
      if (key == 0) {
        return ERRNO_CODES.EINVAL
      }
      HEAP32[key >> 2] = PTHREAD_SPECIFIC_NEXT_KEY;
      PTHREAD_SPECIFIC[PTHREAD_SPECIFIC_NEXT_KEY] = 0;
      PTHREAD_SPECIFIC_NEXT_KEY++;
      return 0
    }
    function _pthread_once(ptr, func) {
      if (!_pthread_once.seen) _pthread_once.seen = {};
      if (ptr in _pthread_once.seen) return
      Module['dynCall_v'](func);
      _pthread_once.seen[ptr] = 1;
    }
    function _pthread_setspecific(key, value) {
      if (!(key in PTHREAD_SPECIFIC)) {
        return ERRNO_CODES.EINVAL
      }
      PTHREAD_SPECIFIC[key] = value;
      return 0
    }
    function _time(ptr) {
      var ret = (Date.now() / 1e3) | 0;
      if (ptr) {
        HEAP32[ptr >> 2] = ret;
      }
      return ret
    }
    FS.staticInit();
    __ATINIT__.unshift(function () {
      if (!Module['noFSInit'] && !FS.init.initialized) FS.init();
    });
    __ATMAIN__.push(function () {
      FS.ignorePermissions = false;
    });
    __ATEXIT__.push(function () {
      FS.quit();
    });
    __ATINIT__.unshift(function () {
    });
    __ATEXIT__.push(function () {
    });
    InternalError = Module['InternalError'] = extendError(Error, 'InternalError');
    embind_init_charCodes();
    BindingError = Module['BindingError'] = extendError(Error, 'BindingError');
    init_ClassHandle();
    init_RegisteredPointer();
    init_embind();
    UnboundTypeError = Module['UnboundTypeError'] = extendError(Error, 'UnboundTypeError');
    init_emval();
    ___buildEnvironment(ENV);
    DYNAMICTOP_PTR = staticAlloc(4);
    STACK_BASE = STACKTOP = alignMemory(STATICTOP);
    STACK_MAX = STACK_BASE + TOTAL_STACK;
    DYNAMIC_BASE = alignMemory(STACK_MAX);
    HEAP32[DYNAMICTOP_PTR >> 2] = DYNAMIC_BASE;
    staticSealed = true;
    function intArrayFromString(stringy, dontAddNull, length) {
      var len = length > 0 ? length : lengthBytesUTF8(stringy) + 1;
      var u8array = new Array(len);
      var numBytesWritten = stringToUTF8Array(stringy, u8array, 0, u8array.length);
      if (dontAddNull) u8array.length = numBytesWritten;
      return u8array
    }
    Module['wasmTableSize'] = 316;
    Module['wasmMaxTableSize'] = 316;
    function invoke_i(index) {
      try {
        return Module['dynCall_i'](index)
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_ii(index, a1) {
      try {
        return Module['dynCall_ii'](index, a1)
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_iii(index, a1, a2) {
      try {
        return Module['dynCall_iii'](index, a1, a2)
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_iiii(index, a1, a2, a3) {
      try {
        return Module['dynCall_iiii'](index, a1, a2, a3)
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_iiiii(index, a1, a2, a3, a4) {
      try {
        return Module['dynCall_iiiii'](index, a1, a2, a3, a4)
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_iiiiiii(index, a1, a2, a3, a4, a5, a6) {
      try {
        return Module['dynCall_iiiiiii'](index, a1, a2, a3, a4, a5, a6)
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_iiiiiiiiii(index, a1, a2, a3, a4, a5, a6, a7, a8, a9) {
      try {
        return Module['dynCall_iiiiiiiiii'](index, a1, a2, a3, a4, a5, a6, a7, a8, a9)
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_iiiiiijii(index, a1, a2, a3, a4, a5, a6, a7, a8, a9) {
      try {
        return Module['dynCall_iiiiiijii'](index, a1, a2, a3, a4, a5, a6, a7, a8, a9)
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_ijj(index, a1, a2, a3, a4) {
      try {
        return Module['dynCall_ijj'](index, a1, a2, a3, a4)
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_ji(index, a1) {
      try {
        return Module['dynCall_ji'](index, a1)
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_v(index) {
      try {
        Module['dynCall_v'](index);
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_vi(index, a1) {
      try {
        Module['dynCall_vi'](index, a1);
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_vii(index, a1, a2) {
      try {
        Module['dynCall_vii'](index, a1, a2);
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_viii(index, a1, a2, a3) {
      try {
        Module['dynCall_viii'](index, a1, a2, a3);
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_viiii(index, a1, a2, a3, a4) {
      try {
        Module['dynCall_viiii'](index, a1, a2, a3, a4);
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_viiiii(index, a1, a2, a3, a4, a5) {
      try {
        Module['dynCall_viiiii'](index, a1, a2, a3, a4, a5);
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_viiiiii(index, a1, a2, a3, a4, a5, a6) {
      try {
        Module['dynCall_viiiiii'](index, a1, a2, a3, a4, a5, a6);
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_viiiiiiiii(index, a1, a2, a3, a4, a5, a6, a7, a8, a9) {
      try {
        Module['dynCall_viiiiiiiii'](index, a1, a2, a3, a4, a5, a6, a7, a8, a9);
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_viiiiiiiiii(index, a1, a2, a3, a4, a5, a6, a7, a8, a9, a10) {
      try {
        Module['dynCall_viiiiiiiiii'](index, a1, a2, a3, a4, a5, a6, a7, a8, a9, a10);
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_vij(index, a1, a2, a3) {
      try {
        Module['dynCall_vij'](index, a1, a2, a3);
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    function invoke_viji(index, a1, a2, a3, a4) {
      try {
        Module['dynCall_viji'](index, a1, a2, a3, a4);
      } catch (e) {
        if (typeof e !== 'number' && e !== 'longjmp') throw e
        Module['setThrew'](1, 0);
      }
    }
    Module.asmGlobalArg = {};
    Module.asmLibraryArg = {
      abort: abort,
      enlargeMemory: enlargeMemory,
      getTotalMemory: getTotalMemory,
      abortOnCannotGrowMemory: abortOnCannotGrowMemory,
      invoke_i: invoke_i,
      invoke_ii: invoke_ii,
      invoke_iii: invoke_iii,
      invoke_iiii: invoke_iiii,
      invoke_iiiii: invoke_iiiii,
      invoke_iiiiiii: invoke_iiiiiii,
      invoke_iiiiiiiiii: invoke_iiiiiiiiii,
      invoke_iiiiiijii: invoke_iiiiiijii,
      invoke_ijj: invoke_ijj,
      invoke_ji: invoke_ji,
      invoke_v: invoke_v,
      invoke_vi: invoke_vi,
      invoke_vii: invoke_vii,
      invoke_viii: invoke_viii,
      invoke_viiii: invoke_viiii,
      invoke_viiiii: invoke_viiiii,
      invoke_viiiiii: invoke_viiiiii,
      invoke_viiiiiiiii: invoke_viiiiiiiii,
      invoke_viiiiiiiiii: invoke_viiiiiiiiii,
      invoke_vij: invoke_vij,
      invoke_viji: invoke_viji,
      ___cxa_allocate_exception: ___cxa_allocate_exception,
      ___cxa_begin_catch: ___cxa_begin_catch,
      ___cxa_end_catch: ___cxa_end_catch,
      ___cxa_find_matching_catch_2: ___cxa_find_matching_catch_2,
      ___cxa_find_matching_catch_3: ___cxa_find_matching_catch_3,
      ___cxa_find_matching_catch_4: ___cxa_find_matching_catch_4,
      ___cxa_free_exception: ___cxa_free_exception,
      ___cxa_throw: ___cxa_throw,
      ___lock: ___lock,
      ___map_file: ___map_file,
      ___resumeException: ___resumeException,
      ___setErrNo: ___setErrNo,
      ___syscall140: ___syscall140,
      ___syscall145: ___syscall145,
      ___syscall146: ___syscall146,
      ___syscall183: ___syscall183,
      ___syscall198: ___syscall198,
      ___syscall20: ___syscall20,
      ___syscall6: ___syscall6,
      ___syscall60: ___syscall60,
      ___syscall83: ___syscall83,
      ___syscall91: ___syscall91,
      ___unlock: ___unlock,
      __embind_finalize_value_object: __embind_finalize_value_object,
      __embind_register_bool: __embind_register_bool,
      __embind_register_class: __embind_register_class,
      __embind_register_class_constructor: __embind_register_class_constructor,
      __embind_register_class_function: __embind_register_class_function,
      __embind_register_emval: __embind_register_emval,
      __embind_register_float: __embind_register_float,
      __embind_register_integer: __embind_register_integer,
      __embind_register_memory_view: __embind_register_memory_view,
      __embind_register_std_string: __embind_register_std_string,
      __embind_register_std_wstring: __embind_register_std_wstring,
      __embind_register_value_object: __embind_register_value_object,
      __embind_register_value_object_field: __embind_register_value_object_field,
      __embind_register_void: __embind_register_void,
      _abort: _abort,
      _emscripten_memcpy_big: _emscripten_memcpy_big,
      _getenv: _getenv,
      _getgrnam: _getgrnam,
      _getpwnam: _getpwnam,
      _jsClose: _jsClose,
      _jsCreate: _jsCreate,
      _jsOpen: _jsOpen,
      _jsRead: _jsRead,
      _jsSeek: _jsSeek,
      _jsTell: _jsTell,
      _jsWrite: _jsWrite,
      _llvm_eh_typeid_for: _llvm_eh_typeid_for,
      _localtime: _localtime,
      _mktime: _mktime,
      _pthread_getspecific: _pthread_getspecific,
      _pthread_key_create: _pthread_key_create,
      _pthread_once: _pthread_once,
      _pthread_setspecific: _pthread_setspecific,
      _time: _time,
      DYNAMICTOP_PTR: DYNAMICTOP_PTR,
      STACKTOP: STACKTOP,
    };
    var asm = Module['asm'](Module.asmGlobalArg, Module.asmLibraryArg, buffer);
    Module['asm'] = asm;
    var __GLOBAL__sub_I_bind_cpp = (Module['__GLOBAL__sub_I_bind_cpp'] = function () {
      return Module['asm']['__GLOBAL__sub_I_bind_cpp'].apply(null, arguments)
    });
    var __GLOBAL__sub_I_bridge_cpp = (Module['__GLOBAL__sub_I_bridge_cpp'] = function () {
      return Module['asm']['__GLOBAL__sub_I_bridge_cpp'].apply(null, arguments)
    });
    var __GLOBAL__sub_I_crc_cpp = (Module['__GLOBAL__sub_I_crc_cpp'] = function () {
      return Module['asm']['__GLOBAL__sub_I_crc_cpp'].apply(null, arguments)
    });
    var __GLOBAL__sub_I_global_cpp = (Module['__GLOBAL__sub_I_global_cpp'] = function () {
      return Module['asm']['__GLOBAL__sub_I_global_cpp'].apply(null, arguments)
    });
    (Module['___cxa_can_catch'] = function () {
      return Module['asm']['___cxa_can_catch'].apply(null, arguments)
    });
    (Module['___cxa_is_pointer_type'] = function () {
      return Module['asm']['___cxa_is_pointer_type'].apply(null, arguments)
    });
    (Module['___errno_location'] = function () {
      return Module['asm']['___errno_location'].apply(null, arguments)
    });
    var ___getTypeName = (Module['___getTypeName'] = function () {
      return Module['asm']['___getTypeName'].apply(null, arguments)
    });
    var _emscripten_replace_memory = (Module['_emscripten_replace_memory'] = function () {
      return Module['asm']['_emscripten_replace_memory'].apply(null, arguments)
    });
    var _free = (Module['_free'] = function () {
      return Module['asm']['_free'].apply(null, arguments)
    });
    var _malloc = (Module['_malloc'] = function () {
      return Module['asm']['_malloc'].apply(null, arguments)
    });
    var setTempRet0 = (Module['setTempRet0'] = function () {
      return Module['asm']['setTempRet0'].apply(null, arguments)
    });
    (Module['setThrew'] = function () {
      return Module['asm']['setThrew'].apply(null, arguments)
    });
    var stackAlloc = (Module['stackAlloc'] = function () {
      return Module['asm']['stackAlloc'].apply(null, arguments)
    });
    (Module['dynCall_dii'] = function () {
      return Module['asm']['dynCall_dii'].apply(null, arguments)
    });
    (Module['dynCall_i'] = function () {
      return Module['asm']['dynCall_i'].apply(null, arguments)
    });
    (Module['dynCall_ii'] = function () {
      return Module['asm']['dynCall_ii'].apply(null, arguments)
    });
    (Module['dynCall_iii'] = function () {
      return Module['asm']['dynCall_iii'].apply(null, arguments)
    });
    (Module['dynCall_iiii'] = function () {
      return Module['asm']['dynCall_iiii'].apply(null, arguments)
    });
    (Module['dynCall_iiiii'] = function () {
      return Module['asm']['dynCall_iiiii'].apply(null, arguments)
    });
    (Module['dynCall_iiiiii'] = function () {
      return Module['asm']['dynCall_iiiiii'].apply(null, arguments)
    });
    (Module['dynCall_iiiiiii'] = function () {
      return Module['asm']['dynCall_iiiiiii'].apply(null, arguments)
    });
    (Module['dynCall_iiiiiiiiii'] = function () {
      return Module['asm']['dynCall_iiiiiiiiii'].apply(null, arguments)
    });
    (Module['dynCall_iiiiiijii'] = function () {
      return Module['asm']['dynCall_iiiiiijii'].apply(null, arguments)
    });
    (Module['dynCall_ijj'] = function () {
      return Module['asm']['dynCall_ijj'].apply(null, arguments)
    });
    (Module['dynCall_ji'] = function () {
      return Module['asm']['dynCall_ji'].apply(null, arguments)
    });
    (Module['dynCall_v'] = function () {
      return Module['asm']['dynCall_v'].apply(null, arguments)
    });
    (Module['dynCall_vi'] = function () {
      return Module['asm']['dynCall_vi'].apply(null, arguments)
    });
    (Module['dynCall_vii'] = function () {
      return Module['asm']['dynCall_vii'].apply(null, arguments)
    });
    (Module['dynCall_viid'] = function () {
      return Module['asm']['dynCall_viid'].apply(null, arguments)
    });
    (Module['dynCall_viii'] = function () {
      return Module['asm']['dynCall_viii'].apply(null, arguments)
    });
    (Module['dynCall_viiii'] = function () {
      return Module['asm']['dynCall_viiii'].apply(null, arguments)
    });
    (Module['dynCall_viiiii'] = function () {
      return Module['asm']['dynCall_viiiii'].apply(null, arguments)
    });
    (Module['dynCall_viiiiii'] = function () {
      return Module['asm']['dynCall_viiiiii'].apply(null, arguments)
    });
    (Module['dynCall_viiiiiiiii'] = function () {
      return Module['asm']['dynCall_viiiiiiiii'].apply(null, arguments)
    });
    (Module['dynCall_viiiiiiiiii'] = function () {
      return Module['asm']['dynCall_viiiiiiiiii'].apply(null, arguments)
    });
    (Module['dynCall_vij'] = function () {
      return Module['asm']['dynCall_vij'].apply(null, arguments)
    });
    (Module['dynCall_viji'] = function () {
      return Module['asm']['dynCall_viji'].apply(null, arguments)
    });
    Module['asm'] = asm;
    function ExitStatus(status) {
      this.name = 'ExitStatus';
      this.message = 'Program terminated with exit(' + status + ')';
      this.status = status;
    }
    ExitStatus.prototype = new Error();
    ExitStatus.prototype.constructor = ExitStatus;
    var initialStackTop;
    dependenciesFulfilled = function runCaller() {
      if (!Module['calledRun']) run();
      if (!Module['calledRun']) dependenciesFulfilled = runCaller;
    };
    function run(args) {
      args = args || Module['arguments'];
      if (runDependencies > 0) {
        return
      }
      preRun();
      if (runDependencies > 0) return
      if (Module['calledRun']) return
      function doRun() {
        if (Module['calledRun']) return
        Module['calledRun'] = true;
        if (ABORT) return
        ensureInitRuntime();
        preMain();
        if (Module['onRuntimeInitialized']) Module['onRuntimeInitialized']();
        postRun();
      }
      if (Module['setStatus']) {
        Module['setStatus']('Running...');
        setTimeout(function () {
          setTimeout(function () {
            Module['setStatus']('');
          }, 1);
          doRun();
        }, 1);
      } else {
        doRun();
      }
    }
    Module['run'] = run;
    function exit(status, implicit) {
      if (implicit && Module['noExitRuntime'] && status === 0) {
        return
      }
      if (Module['noExitRuntime']) ; else {
        ABORT = true;
        STACKTOP = initialStackTop;
        exitRuntime();
        if (Module['onExit']) Module['onExit'](status);
      }
      if (ENVIRONMENT_IS_NODE) {
        process['exit'](status);
      }
      Module['quit'](status, new ExitStatus(status));
    }
    Module['exit'] = exit;
    function abort(what) {
      if (Module['onAbort']) {
        Module['onAbort'](what);
      }
      if (what !== undefined) {
        Module.print(what);
        Module.printErr(what);
        what = JSON.stringify(what);
      } else {
        what = '';
      }
      ABORT = true;
      throw 'abort(' + what + '). Build with -s ASSERTIONS=1 for more info.'
    }
    Module['abort'] = abort;
    if (Module['preInit']) {
      if (typeof Module['preInit'] == 'function') Module['preInit'] = [Module['preInit']];
      while (Module['preInit'].length > 0) {
        Module['preInit'].pop()();
      }
    }
    Module['noExitRuntime'] = true;
    run();

    return unpack
  };
  //-------------------------------------------------------------

  /**
   * Returns a Promise containing the rar extractor for the given filename.
   * @private
   */
  function getExtractor(url) {
    return fetch(new Request(url))
      .then((response) => {
        if (response.ok) return response.arrayBuffer()
        else {
          throw new Error('404 Error: File not found.')
        }
      })
      .then((buffer) => unpackBridge.createExtractorFromData(buffer))
  }

  /**
   *  Returns a string representing the formatted contents of the given file.
   * @private
   */
  function extract({ resourceId, url }) {
    return new Promise(function (resolve, reject) {
      if (!unpackBridge) {
        throw new Error('unpackBridge not detected')
      }
      if (!unpack) {
        throw new Error('unpack not detected')
      }

      getExtractor(url).then(
        (extractor) => {
          // return extractor.extractAll();
          resolve(extractor.extractAll());
        },
        (err) => {
          reject(err);
        }
      );
    })
  }

  /**
   * Listen for messages sent to the worker.
   * @private
   */
  function handleMessage(data, postMessage) {
    if (data.type == 'init') {
      unpack = initunpack(data.buffer);
      unpack.onRuntimeInitialized = () => {
        postMessage({ type: 'WASM_LOADED' });
      };
    } else if (data.type == 'fetch') {
      extract(data).then(
        (unpacked) => {
          returnData(data, unpacked, postMessage);
        },
        (err) => {
          const result = {
            taskId: data.taskId,
            type: 'ERROR',
            resourceId: data.resourceId,
            url: data.url,
          };
          postMessage(result);
        }
      );
    } else if (data.type == 'unpack') {
      const { buffer } = data;

      if (!unpackBridge) {
        throw new Error('unpackBridge not detected')
      }
      if (!unpack) {
        throw new Error('unpack not detected')
      }

      const extractor = unpackBridge.createExtractorFromData(buffer);
      const unpacked = extractor.extractAll();
      returnData(data, unpacked, postMessage);
    }
  }

  function returnData(data, unpacked, postMessage) {
    const [state, list] = unpacked;
    if (state.state == 'FAIL') {
      const result = {
        taskId: data.taskId,
        type: 'ERROR',
        reason: state.reason,
        msg: state.msg,
        resourceId: data.resourceId,
        url: data.url,
      };
      postMessage(result);
      return
    }
    const result = {
      taskId: data.taskId,
      type: 'FINISHED',
      resourceId: data.resourceId,
      entries: {},
    };

    const transferables = [];
    if (list && list.files) {
      for (const file of list.files) {
        result.entries[file.fileHeader.name] = file.extract[1];
        transferables.push(file.extract[1].buffer);
      }
    }
    postMessage(result, transferables);
  }

  globalThis.onmessage = function (event) {
    handleMessage(event.data, self.postMessage);
  };

  /**
   * When the WASM runtime has been initialized on the unpack.js module, send a message indicating
   * that the library is ready.
   */
  // <!-- prettier-ignore-end -->

  exports.handleMessage = handleMessage;

  Object.defineProperty(exports, '__esModule', { value: true });

  return exports;

})({});

",null,!1);class ArchiveUnpackerMainThread{constructor(){const A=Buffer.from(unpackBase64Str,"base64");WebAssembly.compile(A).then((A=>{console.log("WebAssembly.compile",A)})),handleMessage$1({type:"init",buffer:A},(()=>{}))}addTask(A,g){return new Promise((g=>{handleMessage$1(A,(A=>{g(A)}))}))}terminate(){}}class ArchiveUnpackerWorkerPool extends WorkerPool{uint8Array;constructor(){super(!0),this.uint8Array=Uint8Array.from(atob(unpackBase64Str),(A=>A.charCodeAt(0)))}constructWorker(){return new Promise((A=>{const g=new WorkerFactory$2;g.onmessage=I=>{"WASM_LOADED"==I.data.type&&A(g)},g.postMessage({type:"init",buffer:this.uint8Array.buffer})}))}}let archiveUnpackerWorkerPool;archiveUnpackerWorkerPool="Node"==SystemDesc.OS?new ArchiveUnpackerMainThread:new ArchiveUnpackerWorkerPool;class ArchiveUnpackerPlugin extends ResourceLoaderPlugin{getType(){return"archive"}async loadFile(A){try{const g=await fetch(A);if(!g?.ok)throw new Error(`ArchiveUnpackerPlugin.loadFile: ${g.status} - ${g.statusText} : ${A}`);const I=await g.arrayBuffer();if(!I)throw new Error("Buffer is null");return await this.extractFile(I)}catch(A){throw A}}async extractFile(A){return new Promise(((g,I)=>{const C=A.byteLength;archiveUnpackerWorkerPool.addTask({type:"unpack",resourceId:C,buffer:A},[A]).then((A=>{"FINISHED"===A.type?g(A.entries):"ERROR"===A.type&&I(new Error(`Unable to open archive. The file may be corrupt: ${C}`))}))}))}}class JsonLoaderPlugin extends ResourceLoaderPlugin{getType(){return"json"}async loadFile(A){try{const g=await fetch(A);if(!g?.ok)throw new Error(`JsonLoaderPlugin.loadFile: ${g.status} - ${g.statusText} : ${A}`);return await g.json()}catch(A){throw A}}}class TextLoaderPlugin extends ResourceLoaderPlugin{getType(){return"text"}async loadFile(A){try{const g=await fetch(A);if(!g?.ok)throw new Error(`TextLoaderPlugin.loadFile: ${g.status} - ${g.statusText} : ${A}`);return await g.text()}catch(A){throw A}}}class BinaryLoaderPlugin extends ResourceLoaderPlugin{getType(){return"binary"}async loadFile(A){try{const g=await fetch(A);if(!g?.ok)throw new Error(`BinaryLoaderPlugin.loadFile: ${g.status} - ${g.statusText} : ${A}`);return await g.arrayBuffer()}catch(A){throw A}}}const MAX_LOAD_COUNT=1e3;class ResourceLoader extends EventEmitter{totalWork=0;doneWork=0;baseUrl="";plugins={};systemUrls={};commonResources={};loadCount=0;queue=[];cache;constructor(){super();const A="https://storage.googleapis.com/zea-engine-resources";this.systemUrls["ZeaEngine/Vive.vla"]=A+"/Vive.vla",this.systemUrls["ZeaEngine/Oculus.vla"]=A+"/Oculus.vla"}registerPlugin(A){this.plugins[A.getType()]=A}async loadFile(A,g,I=!0){const C=this.plugins[A];if(!C)throw new Error(`There's no plugin registered for the type of file "${A}". Did you add the plugins script? See: https://docs.zea.live/zea-engine/#/adding-default-plugins`);if(I&&this.incrementWorkload(),this.cache){const A=await this.cache.get(g);if(A)return I&&this.incrementWorkDone(),A}if(!(this.loadCount<1e3)){return new Promise(((A,e)=>{this.queue.push((()=>{this.loadCount++;C.loadFile(g).then((C=>{for(this.loadCount--,I&&this.incrementWorkDone(),this.emit("loaded",{url:g});this.loadCount<1e3&&this.queue.length>0;){this.queue.pop()()}A(C)}),(A=>{I&&this.incrementWorkDone(),e(A)}))}))}))}this.loadCount++;try{const A=C.loadFile(g);return A.then((A=>{for(this.cache&&this.cache.set(g,A),this.loadCount--,I&&this.incrementWorkDone(),this.emit("loaded",{url:g});this.loadCount<1e3&&this.queue.length>0;){this.queue.pop()()}}),(()=>{I&&this.incrementWorkDone()})),A}catch(A){throw I&&this.incrementWorkDone(),A}}getCommonResource(A){return this.commonResources[A]}setCommonResource(A,g){this.commonResources[A]=g}resetProgress(){this.totalWork=0,this.doneWork=0}incrementWorkload(A=1){this.totalWork+=A;const g=this.doneWork/this.totalWork*100;this.emit("progressIncremented",new ProgressEvent(g))}incrementWorkDone(A=1){this.doneWork+=A;const g=this.doneWork/this.totalWork*100;this.emit("progressIncremented",new ProgressEvent(g)),this.doneWork>this.totalWork&&console.warn("Mismatch between work loaded and work done.")}}const resourceLoader=new ResourceLoader,archiveUnpackerPlugin=new ArchiveUnpackerPlugin;resourceLoader.registerPlugin(archiveUnpackerPlugin);const jsonLoaderPlugin=new JsonLoaderPlugin;resourceLoader.registerPlugin(jsonLoaderPlugin);const textLoaderPlugin=new TextLoaderPlugin;resourceLoader.registerPlugin(textLoaderPlugin);const binaryLoaderPlugin=new BinaryLoaderPlugin;resourceLoader.registerPlugin(binaryLoaderPlugin);class IndexDBCache{name;objStoreName;version;db;constructor(A,g,I){this.name=A,this.objStoreName=g,this.version=I}async init(){return new Promise((A=>{const g=globalThis.indexedDB.open(this.name,this.version);g.onerror=()=>{console.error("IndexedDB Failed")},g.onsuccess=()=>{this.db=g.result,A()},g.onupgradeneeded=A=>{this.db=g.result,A.oldVersion&&A.oldVersion!=A.newVersion&&this.db.deleteObjectStore(this.objStoreName),this.db.createObjectStore(this.objStoreName,{keyPath:"key"})}}))}get(A){return new Promise(((g,I)=>{const C=this.db.transaction(this.objStoreName,"readonly").objectStore(this.objStoreName).get(A);C.onerror=A=>{I(A)},C.onsuccess=()=>{C.result?g(C.result.value):g(null)}}))}set(A,g){return new Promise(((I,C)=>{const e=this.db.transaction(this.objStoreName,"readwrite").objectStore(this.objStoreName).put({key:A,value:g});e.onsuccess=()=>{I()},e.onerror=A=>{C(A)}}))}}class Version{major=0;minor=0;patch=0;branch="";constructor(A){if("string"==typeof A){const g=A.split("-"),I=g[0].split(".");this.major=parseInt(I[0]),this.minor=I.length>1?parseInt(I[1]):0,this.patch=I.length>2?parseInt(I[2]):0,2==g.length&&(this.branch=g[1])}else if(Array.isArray(A)){const g=A;this.major=g[0],this.minor=g.length>1?g[1]:0,this.patch=g.length>2?g[2]:0}}compare(A){const g=[this.major,this.minor,this.patch];for(let I=0;I<3;I++)if(g[I]!==A[I])return g[I]-A[I];return 0}asArray(){return[this.major,this.minor,this.patch]}toString(){return`v${this.major}.${this.minor}.${this.patch}`+(""!=this.branch?`-${this.branch}`:"")}}class BinReader{__data;__byteOffset;__dataView;__isMobileDevice;utf8decoder;constructor(A,g=0,I=!0){this.__data=A,this.__byteOffset=g,this.__dataView=new DataView(this.__data),this.__isMobileDevice=I,this.utf8decoder=new TextDecoder}get isMobileDevice(){return this.__isMobileDevice}get data(){return this.__data}get byteLength(){return this.__dataView.byteLength}get remainingByteLength(){return this.__dataView.byteLength-this.__byteOffset}pos(){return this.__byteOffset}seek(A){this.__byteOffset=A}advance(A){this.__byteOffset+=A}loadUInt8(){const A=this.__dataView.getUint8(this.__byteOffset);return this.__byteOffset+=1,A}loadUInt16(){const A=this.__dataView.getUint16(this.__byteOffset,!0);return this.__byteOffset+=2,A}loadUInt32(){const A=this.__dataView.getUint32(this.__byteOffset,!0);return this.__byteOffset+=4,A}loadSInt32(){const A=this.__dataView.getInt32(this.__byteOffset,!0);return this.__byteOffset+=4,A}loadFloat16(){const A=this.loadUInt16();return MathFunctions.decode16BitFloat(A)}loadUFloat16(){const A=this.loadFloat16();return A<0?2048-A:A}loadFloat16From2xUInt8(){throw Error("loadFloat16From2xUInt8 not implemented!")}loadUInt32From2xUFloat16(){return this.loadUFloat16()+4096*this.loadUFloat16()}loadSInt32From2xFloat16(){return this.loadFloat16()+2048*this.loadFloat16()}loadFloat32(){const A=this.__dataView.getFloat32(this.__byteOffset,!0);return this.__byteOffset+=4,A}loadInt8Array(A,g=!0){let I;return null==A&&(A=this.loadUInt32()),g?(I=new Int8Array(this.__data.slice(this.__byteOffset,this.__byteOffset+A)),I.length!=A&&console.log("broken")):I=new Int8Array(this.__data,this.__byteOffset,A),this.__byteOffset+=A,I}loadUInt8Array(A,g=!0){let I;return null==A&&(A=this.loadUInt32()),g?(I=new Uint8Array(this.__data.slice(this.__byteOffset,this.__byteOffset+A)),I.length!=A&&console.log("broken")):I=new Uint8Array(this.__data,this.__byteOffset,A),this.__byteOffset+=A,I}loadUInt16Array(A,g=!0){if(null==A&&(A=this.loadUInt32()),0==A)return new Uint16Array;let I;if(this.readPad(2),this.__isMobileDevice){I=new Uint16Array(A);for(let g=0;g<A;g++)I[g]=this.__dataView.getUint16(this.__byteOffset,!0),this.__byteOffset+=2}else g?(I=new Uint16Array(this.__data.slice(this.__byteOffset,this.__byteOffset+2*A)),I.length!=A&&console.log("broken")):I=new Uint16Array(this.__data,this.__byteOffset,A),this.__byteOffset+=2*A;return I}loadUInt32Array(A,g=!0){if(null==A&&(A=this.loadUInt32()),0==A)return new Uint32Array;let I;if(this.readPad(4),this.__isMobileDevice){I=new Uint32Array(A);for(let g=0;g<A;g++)I[g]=this.__dataView.getUint32(this.__byteOffset,!0),this.__byteOffset+=4}else g?(I=new Uint32Array(this.__data.slice(this.__byteOffset,this.__byteOffset+4*A)),I.length!=A&&console.log("broken")):I=new Uint32Array(this.__data,this.__byteOffset,A),this.__byteOffset+=4*A;return I}loadFloat32Array(A,g=!0){if(null==A&&(A=this.loadUInt32()),0==A)return new Float32Array;let I;if(this.readPad(4),this.__isMobileDevice){I=new Float32Array(A);for(let g=0;g<A;g++)I[g]=this.__dataView.getFloat32(this.__byteOffset,!0),this.__byteOffset+=4}else I=g?new Float32Array(this.__data.slice(this.__byteOffset,this.__byteOffset+4*A)):new Float32Array(this.__data,this.__byteOffset,A),this.__byteOffset+=4*A;return I}loadStr(){const A=this.loadUInt32(),g=new Uint8Array(this.__data,this.__byteOffset,A);return this.__byteOffset+=A,this.utf8decoder.decode(g)}loadStrArray(){const A=this.loadUInt32(),g=[];for(let I=0;I<A;I++)g[I]=this.loadStr();return g}loadSInt32Vec2(){const A=this.loadSInt32(),g=this.loadSInt32();return new Vec2(A,g)}loadUInt32Vec2(){const A=this.loadUInt32(),g=this.loadUInt32();return new Vec2(A,g)}loadFloat16Vec2(){const A=this.loadFloat16(),g=this.loadFloat16();return new Vec2(A,g)}loadFloat32Vec2(){const A=this.loadFloat32(),g=this.loadFloat32();return new Vec2(A,g)}loadFloat16Vec3(){const A=this.loadFloat16(),g=this.loadFloat16(),I=this.loadFloat16();return new Vec3(A,g,I)}loadFloat32Vec3(){const A=this.loadFloat32(),g=this.loadFloat32(),I=this.loadFloat32();return new Vec3(A,g,I)}loadFloat16Quat(){const A=this.loadFloat16(),g=this.loadFloat16(),I=this.loadFloat16(),C=this.loadFloat16();return new Quat(A,g,I,C)}loadFloat32Quat(){const A=this.loadFloat32(),g=this.loadFloat32(),I=this.loadFloat32(),C=this.loadFloat32();return new Quat(A,g,I,C)}loadRGBFloat32Color(){const A=this.loadFloat32(),g=this.loadFloat32(),I=this.loadFloat32();return new Color(A,g,I)}loadRGBAFloat32Color(){const A=this.loadFloat32(),g=this.loadFloat32(),I=this.loadFloat32(),C=this.loadFloat32();return new Color(A,g,I,C)}loadRGBUInt8Color(){const A=this.loadUInt8(),g=this.loadUInt8(),I=this.loadUInt8();return new Color(A/255,g/255,I/255)}loadRGBAUInt8Color(){const A=this.loadUInt8(),g=this.loadUInt8(),I=this.loadUInt8(),C=this.loadUInt8();return new Color(A/255,g/255,I/255,C/255)}loadBox2(){return new Box2(this.loadFloat32Vec2(),this.loadFloat32Vec2())}loadBox3(){return new Box3(this.loadFloat32Vec3(),this.loadFloat32Vec3())}readPad(A){const g=this.__byteOffset%A;0!=g&&(this.__byteOffset+=A-g)}}class BinWriter{__data;__byteOffset;__reserved;__dataView;constructor(A=0){this.__data=new ArrayBuffer(A),this.__byteOffset=0,this.__reserved=A,this.__dataView=new DataView(this.__data)}pos(){return this.__byteOffset}seek(A){this.__byteOffset=A}seekEnd(){this.__byteOffset=this.__reserved}getBuffer(){if(this.__data.byteLength==this.__byteOffset)return this.__data;return new Uint8Array(this.__data).slice(0,this.__byteOffset).buffer}__grow(){const A=2*(this.__reserved>0?this.__reserved:1),g=new ArrayBuffer(A),I=new Uint8Array(g),C=new Uint8Array(this.__data);I.set(C),this.__data=g,this.__dataView=new DataView(this.__data),this.__reserved=A}__reserve(A){this.__byteOffset+A>this.__reserved&&this.__grow()}__offset(A){this.__byteOffset+=A,this.__byteOffset>this.__reserved&&this.__grow()}writeUInt8(A){this.__reserve(1),this.__dataView.setUint8(this.__byteOffset,A),this.__offset(1)}writeUInt16(A){this.__reserve(2),this.__dataView.setUint16(this.__byteOffset,A,!0),this.__offset(2)}writeUInt32(A){this.__reserve(4),this.__dataView.setUint32(this.__byteOffset,A,!0),this.__offset(4)}writeSInt32(A){this.__reserve(4),this.__dataView.setInt32(this.__byteOffset,A,!0),this.__offset(4)}writeFloat16(A){const g=MathFunctions.encode16BitFloat(A);this.writeUInt16(g)}writeFloat32(A){this.__reserve(4),this.__dataView.setFloat32(this.__byteOffset,A,!0),this.__offset(4)}writeUInt8Array(A,g=!0){const I=(A.length,A.length);this.__reserve(I+(g?4:0)),g&&this.writeUInt32(I);for(let g=0;g<I;g++)this.writeUInt8(A[g])}writeUInt16Array(A,g=!0){const I=(A.length,A.length);this.__reserve(2*I+(g?4:0)),g&&this.writeUInt32(I);for(let g=0;g<I;g++)this.writeUInt16(A[g])}writeUInt32Array(A,g=!0){const I=(A.length,A.length);this.__reserve(4*I+(g?4:0)),g&&this.writeUInt32(I);for(let g=0;g<I;g++)this.writeUInt32(A[g])}writeFloat32Array(A,g=!0){const I=(A.length,A.length);this.__reserve(4*I+(g?4:0)),g&&this.writeUInt32(I);for(let g=0;g<I;g++)this.writeFloat32(A[g])}writeStr(A,g=!0){const I=A.length;this.__reserve(4*I+(g?4:0)),g&&this.writeUInt32(I);for(let g=0;g<I;g++)this.writeFloat32(A.charCodeAt(g))}writeSInt32Vec2(A){this.writeSInt32(A.x),this.writeSInt32(A.y)}writeUInt32Vec2(A){this.writeUInt32(A.x),this.writeUInt32(A.y)}writeFloat16Vec2(A){this.writeFloat16(A.x),this.writeFloat16(A.y)}writeFloat32Vec2(A){this.writeFloat32(A.x),this.writeFloat32(A.y)}writeFloat16Vec3(A){this.writeFloat16(A.x),this.writeFloat16(A.y),this.writeFloat16(A.z)}writeFloat32Vec3(A){this.writeFloat32(A.x),this.writeFloat32(A.y),this.writeFloat32(A.z)}writeFloat16Quat(A){this.writeFloat16(A.x),this.writeFloat16(A.y),this.writeFloat16(A.z),this.writeFloat16(A.w)}writeFloat32Quat(A){this.writeFloat32(A.x),this.writeFloat32(A.y),this.writeFloat32(A.z),this.writeFloat32(A.w)}writeRGBFloat32Color(A){this.writeFloat32(A.r),this.writeFloat32(A.g),this.writeFloat32(A.b)}writeRGBAFloat32Color(A){this.writeFloat32(A.r),this.writeFloat32(A.g),this.writeFloat32(A.b),this.writeFloat32(A.a)}writeRGBUInt8Color(A){this.writeUInt8(A.r),this.writeUInt8(A.g),this.writeUInt8(A.b)}writeRGBAUInt8Color(A){this.writeUInt8(A.r),this.writeUInt8(A.g),this.writeUInt8(A.b),this.writeUInt8(A.a)}writeBox2(A){this.writeFloat32Vec2(A.p0),this.writeFloat32Vec2(A.p1)}writeBox3(A){this.writeFloat32Vec3(A.p0),this.writeFloat32Vec3(A.p1)}writePadd(A){const g=A-this.__byteOffset;this.__reserve(g),this.__offset(g)}writeAlignment(A){const g=this.__byteOffset%A;0!=g&&(this.__reserve(A-g),this.__offset(A-g))}}class CloneContext{assetItem;constructor(){this.assetItem=null}}var OperatorOutputMode,ColorSpace;!function(A){A[A.OP_WRITE=0]="OP_WRITE",A[A.OP_READ_WRITE=1]="OP_READ_WRITE"}(OperatorOutputMode||(OperatorOutputMode={}));class Parameter extends BaseItem{#C;dirty=!1;boundInputs=[];boundOutputs=[];cleaning=!1;dirtyOpIndex=0;firstOP_WRITE=0;dataType;constructor(A="",g,I){super(A),this.#C=g,this.dataType=I}getDataType(){return this.dataType}bindOperatorInput(A){this.boundInputs.push(A)}unbindOperatorInput(A){const g=this.boundInputs.indexOf(A);this.boundInputs.splice(g,1)}bindOperatorOutput(A,g=-1){-1==g&&(g=this.boundOutputs.length),this.boundOutputs.splice(g,0,A);for(let A=g;A<this.boundOutputs.length;A++)this.boundOutputs[A].setParamBindIndex(A);return this.__findFirstOP_WRITE(),this.dirtyOpIndex=this.boundOutputs.length,this.setDirty(g),g}unbindOperatorOutput(A){const g=A.getParamBindIndex();this.boundOutputs.splice(g,1);for(let A=g;A<this.boundOutputs.length;A++)this.boundOutputs[A].setParamBindIndex(A);return this.__findFirstOP_WRITE(),this.dirtyOpIndex=this.boundOutputs.length,this.setDirty(Math.max(0,g-1)),g}__findFirstOP_WRITE(){if(this.firstOP_WRITE=this.boundOutputs.length,this.boundOutputs.length>0)for(this.firstOP_WRITE--;this.firstOP_WRITE>0&&this.boundOutputs[this.firstOP_WRITE].getMode()!=OperatorOutputMode.OP_WRITE;this.firstOP_WRITE--);}isDrivenByOperator(){return 0==this.firstOP_WRITE&&this.boundOutputs.length>0}setDirty(A){if(A<this.dirtyOpIndex){let g=this.firstOP_WRITE;if(g<=A){for(this.dirtyOpIndex=g,g++;g<this.boundOutputs.length;g++)g!=A&&this.boundOutputs[g].getOperator().setDirty();for(let A=0;A<this.boundInputs.length;A++)this.boundInputs[A].setDirty();return this.emit("valueChanged"),!0}}return!1}isDirty(){return this.dirtyOpIndex<this.boundOutputs.length}getDirtyBindingIndex(){return this.dirtyOpIndex}setCleanFromOp(A,g){if(g!=this.dirtyOpIndex)if(g<this.dirtyOpIndex)console.log("Parameter is cleaned when it was already clean to that point in the stack:",this.getPath());else if(this.boundOutputs[g].getMode()!=OperatorOutputMode.OP_WRITE){const A=this.getClassName(),I=this.boundOutputs[g].getOperator(),C=I.constructor.name;throw new Error(`Parameter: ${A} with name: ${this.getName()} is not cleaning all outputs during evaluation of op: ${C} with name: ${I.name}`)}this.#C=A,this.dirtyOpIndex=g+1}getValueFromOp(A){return this.dirtyOpIndex<A&&this._clean(A),this.#C}_clean(A){if(this.cleaning)throw new Error(`Cycle detected when cleaning: ${this.getPath()}. Operators need to be rebound to fix errors`);for(this.cleaning=!0;this.dirtyOpIndex<A;){const A=this.dirtyOpIndex;if(this.boundOutputs[this.dirtyOpIndex].getOperator().evaluate(),A==this.dirtyOpIndex){const A=this.boundOutputs[this.dirtyOpIndex].getOperator(),g=A.constructor.name;console.warn(`Operator: ${g} with name: ${A.name} is not cleaning its outputs during evaluation`),this.dirtyOpIndex++}}this.cleaning=!1}getValue(){return this.dirtyOpIndex<this.boundOutputs.length&&this._clean(this.boundOutputs.length),this.#C}setValue(A){if(void 0===A)throw"undefined was passed into the set value for param:"+this.getName();if(this.boundOutputs.length>0)for(let g=this.boundOutputs.length-1;g>=0;g--){const I=this.boundOutputs[g];if(A=I.backPropagateValue(A),0==I.getMode())return}if("object"==typeof A||this.#C!=A){this.#C=A;for(let A=0;A<this.boundInputs.length;A++)this.boundInputs[A].paramValueChanged();this.emit("valueChanged")}}get value(){return this.getValue()}set value(A){this.setValue(A)}loadValue(A){this.#C=A}copyFrom(A,g){this.loadValue(A.value)}readBinary(A,g){console.warn(`TODO: Parameter: ${this.constructor.name} with name: ${this.name} does not implement readBinary`)}getPath(){return this.ownerItem instanceof BaseItem?[...this.ownerItem.getPath(),this.name]:[this.name]}resolvePath(A,g=0){if(0==g&&("."!=A[0]&&A[0]!=this.name||g++),".."==A[g]){if(this.ownerItem)return this.ownerItem.resolvePath(A,g+1);throw Error("this.ownerItem is undefined")}if(g==A.length)return this;if("value"==A[g]&&this.#C instanceof BaseItem)return this.#C.resolvePath(A,g+1);throw new Error("Invalid path:"+A+"["+g+"]. Path does not resolve to a BaseItem")}destroy(){console.warn("nothing destroyed. This method was not overwritten in subclass")}}class NumberParameter extends Parameter{range;step;constructor(A="",g=0,I,C){super(A,g,"Number"),this.range=I,this.step=C}getRange(){return this.range}setRange(A){this.range=A}getStep(){return this.step}setStep(A){this.step=A}setValue(A){if("number"!=typeof A)throw new Error("value provided is not a number. Check the source of this value");super.setValue(A)}toJSON(A){return{type:this.getClassName(),name:this.name,value:this.value}}fromJSON(A,g){this.value=A.value}readBinary(A,g){this.value=A.loadFloat32()}clone(){return new NumberParameter(this.name,this.value,this.range,this.step)}}class Float32Parameter extends NumberParameter{readBinary(A,g){this.value=A.loadFloat32()}}class SInt32Parameter extends NumberParameter{readBinary(A,g){this.value=A.loadSInt32()}}class UInt32Parameter extends NumberParameter{readBinary(A,g){this.value=A.loadUInt32()}}class AngleParameter extends Float32Parameter{}Registry.register("NumberParameter",NumberParameter),Registry.register("Property_SInt32",SInt32Parameter),Registry.register("Property_UInt32",UInt32Parameter),Registry.register("Property_Float32",Float32Parameter),Registry.register("AngleParameter",AngleParameter);class MultiChoiceParameter extends NumberParameter{choices;constructor(A,g,I=[]){super(A,g,[0,I.length],1),this.choices=I}getChoices(){return this.choices}setValue(A){if("string"==typeof A){const g=this.choices.indexOf(A);-1===g?console.error(`Invalid value for MultiChoiceParameter: ${A}. choices are: ${this.choices}.`):super.setValue(g)}else super.setValue(A)}clone(){return new MultiChoiceParameter(this.name,this.value,this.choices)}}Registry.register("MultiChoiceParameter",MultiChoiceParameter);class BooleanParameter extends Parameter{constructor(A="",g){super(A,null!=g&&g,"Boolean")}setValue(A){if("boolean"!=typeof A)throw new Error("value provided is not a boolean. Check the source of this value");super.setValue(A)}readBinary(A,g){this.value=0!=A.loadUInt8()}toJSON(A){return{type:this.getClassName(),name:this.name,value:this.value}}fromJSON(A,g){this.value=A.value}clone(){return new BooleanParameter(this.name,this.value)}}Registry.register("BooleanParameter",BooleanParameter),Registry.register("Property_Boolean",BooleanParameter);class Vec2Parameter extends Parameter{range;constructor(A="",g,I){super(A,g||new Vec2,"Vec2"),this.range=I}getRange(){return this.range}setRange(A){this.range=A,this.emit("rangeChanged",{range:A})}setValue(A){if(!(A instanceof Vec2))throw new Error("value provided is not an instance of a 'Vec2' class. Check the source of this value");super.setValue(A)}readBinary(A,g){this.value?.readBinary(A)}toJSON(A){return{type:this.getClassName(),name:this.name,value:this.value?.toJSON()}}fromJSON(A,g){const I=new Vec2;I.fromJSON(A.value),this.value=I,A.name&&(this.name=A.name)}clone(){const A=new Vec2Parameter(this.name,this.value?.clone());return this.range&&A.setRange(this.range),A}}Registry.register("Vec2Parameter",Vec2Parameter),Registry.register("Property_Vec2_32f",Vec2Parameter);class Vec3Parameter extends Parameter{range;constructor(A="",g,I){super(A,g||new Vec3,"Vec3"),this.range=I}setValue(A){if(!(A instanceof Vec3))throw new Error("value provided is not an instance of a 'Vec2' class. Check the source of this value");super.setValue(A)}readBinary(A,g){this.value?.readBinary(A)}toJSON(A){return{type:this.getClassName(),name:this.name,value:this.value?.toJSON()}}fromJSON(A,g){const I=new Vec3;I.fromJSON(A.value),this.value=I,A.name&&(this.name=A.name)}clone(){return new Vec3Parameter(this.name,this.value?.clone())}}Registry.register("Vec3Parameter",Vec3Parameter),Registry.register("Property_Vec3_32f",Vec3Parameter);class Vec4Parameter extends Parameter{constructor(A="",g){super(A,g||new Vec4,"Vec4")}setValue(A){if(!(A instanceof Vec4))throw new Error("value provided is not an instance of a 'Vec4' class. Check the source of this value");super.setValue(A)}readBinary(A,g){this.value?.readBinary(A)}toJSON(A){return{type:this.getClassName(),name:this.name,value:this.value?.toJSON()}}fromJSON(A,g){const I=new Vec4;I.fromJSON(A.value),this.value=I,A.name&&(this.name=A.name)}clone(){return new Vec4Parameter(this.name,this.value?.clone())}}Registry.register("Vec4Parameter",Vec4Parameter),Registry.register("Property_Vec4_32f",Vec4Parameter);class Box2Parameter extends Parameter{constructor(A="",g){super(A,g||new Box2,"Box2")}setValue(A){if(!(A instanceof Box2))throw new Error("value provided is not an instance of a 'Box2' class. Check the source of this value");super.setValue(A)}readBinary(A,g){this.value.p0.readBinary(A),this.value.p1.readBinary(A)}toJSON(A){return{type:this.getClassName(),name:this.name,value:{p0:this.value.p0.toJSON(),p1:this.value.p1.toJSON()}}}fromJSON(A,g){this.value.p0.fromJSON(A.p0),this.value.p1.fromJSON(A.p1)}clone(){return new Box2Parameter(this.name,this.value?.clone())}}Registry.register("Box2Parameter",Box2Parameter),Registry.register("Property_Box2_32f",Box2Parameter);class Box3Parameter extends Parameter{constructor(A="",g){super(A,g||new Box3,"Box3")}setValue(A){if(!(A instanceof Box3))throw new Error("value provided is not an instance of a 'Box3' class. Check the source of this value");super.setValue(A)}readBinary(A,g){this.value.p0.readBinary(A),this.value.p1.readBinary(A)}toJSON(A){return{type:this.getClassName(),name:this.name,value:this.value.toJSON()}}fromJSON(A,g){A.value&&this.value.fromJSON(A.value)}clone(){return new Box3Parameter(this.name,this.value?.clone())}}Registry.register("Box3Parameter",Box3Parameter),Registry.register("Property_Box3_32f",Box3Parameter);class ColorParameter extends Parameter{constructor(A="",g){super(A,g||new Color,"Color")}setValue(A){if(!(A instanceof Color))throw new Error("value provided is not an instance of a 'Color' class. Check the source of this value");super.setValue(A)}readBinary(A,g){const I=A.loadRGBAFloat32Color();I.applyGamma(2.2),this.value=I}toJSON(A){return{type:this.getClassName(),name:this.name,value:this.value?.toJSON()}}fromJSON(A,g){this.value?.fromJSON(A.value)}clone(){return new ColorParameter(this.name,this.value?.clone())}}Registry.register("ColorParameter",ColorParameter),Registry.register("Property_Color_32f",ColorParameter);class QuatParameter extends Parameter{constructor(A="",g){super(A,g||new Quat,"Quat")}setValue(A){if(!(A instanceof Quat))throw new Error("value provided is not an instance of a 'Quat' class. Check the source of this value");super.setValue(A)}readBinary(A,g){this.value?.readBinary(A)}toJSON(A){return{type:this.getClassName(),name:this.name,value:this.value?.toJSON()}}fromJSON(A,g){const I=new Quat;I.fromJSON(A.value),this.value=I}clone(){return new QuatParameter(this.name,this.value?.clone())}}Registry.register("QuatParameter",QuatParameter),Registry.register("Property_Quat_32f",QuatParameter);class Mat3Parameter extends Parameter{constructor(A="",g){super(A,g||new Mat3,"Mat3")}setValue(A){if(!(A instanceof Mat3))throw new Error("value provided is not an instance of a 'Mat3' class. Check the source of this value");super.setValue(A)}readBinary(A,g){this.value?.readBinary(A)}toJSON(A){return{type:this.getClassName(),name:this.name,value:this.value?.toJSON()}}fromJSON(A,g){const I=new Mat3;I.fromJSON(A.value),this.value=I}clone(){return new Mat3Parameter(this.name,this.value?.clone())}}Registry.register("Mat3Parameter",Mat3Parameter),Registry.register("Property_Mat3_32f",Mat3Parameter);class Mat4Parameter extends Parameter{constructor(A="",g){super(A,g||new Mat4,"Mat4")}setValue(A){if(!(A instanceof Mat4))throw new Error("value provided is not an instance of a 'Mat4' class. Check the source of this value");super.setValue(A)}readBinary(A,g){this.value?.readBinary(A)}toJSON(A){return{type:this.getClassName(),value:this.value?.toJSON()}}fromJSON(A,g){const I=new Mat4;I.fromJSON(A.value),this.value=I}clone(){return new Mat4Parameter(this.name,this.value?.clone())}}Registry.register("Mat4Parameter",Mat4Parameter),Registry.register("Property_Mat4_32f",Mat4Parameter);class XfoParameter extends Parameter{constructor(A="",g){super(A,g||new Xfo,"Xfo")}setValue(A){if(!(A instanceof Xfo))throw new Error("value provided is not an instance of a 'Xfo' class. Check the source of this value");super.setValue(A)}readBinary(A,g){this.value.readBinary(A)}toJSON(A){return{type:this.getClassName(),name:this.name,value:this.value.toJSON()}}fromJSON(A,g){const I=new Xfo;I.fromJSON(A.value),this.value=I,A.name&&(this.name=A.name)}clone(){return new XfoParameter(this.name,this.value.clone())}}Registry.register("XfoParameter",XfoParameter),Registry.register("Property_Xfo_32f",XfoParameter);class BaseImage extends ParameterOwner{width=0;height=0;format="RGB";type="UNSIGNED_BYTE";loaded=!1;mipMapped=!0;wrapS="REPEAT";wrapT="REPEAT";minFilter="LINEAR";magFilter="LINEAR";constructor(A="Image"){super(A),this.on("parameterValueChanged",(()=>{this.emit("updated")}))}isLoaded(){return this.loaded}getParams(){return{type:this.type,format:this.format,width:this.width,height:this.height,wrapS:this.wrapS,wrapT:this.wrapT,minFilter:this.minFilter,magFilter:this.magFilter,mipMapped:this.mipMapped}}}class ImageParameter extends Parameter{constructor(A="",g){super(A,g,"BaseImage")}setValue(A){if(!(A instanceof BaseImage))throw new Error("value provided is not an instance of 'BaseImage' class. Check the source of this value");super.setValue(A)}toJSON(A){const g={type:this.getClassName(),name:this.name};return this.value&&(g.imageType=this.value.getClassName(),g.value=this.value.toJSON()),g}fromJSON(A,g){A.imageType&&(this.value=Registry.constructClass(A.imageType),A.value&&this.value?.fromJSON(A.value,g))}clone(){return new ImageParameter(this.name,this.value)}}Registry.register("ImageParameter",ImageParameter);class StringParameter extends Parameter{multiLine;constructor(A="",g=""){super(A,g,"String"),this.multiLine=!1}setMultiLine(A){this.multiLine=A}getMultiLine(){return this.multiLine}setValue(A){if("string"!=typeof A)throw new Error("value provided is not a string. Check the source of this value");super.setValue(A)}readBinary(A,g){this.value=A.loadStr()}toJSON(A){return{type:this.getClassName(),name:this.name,value:this.value}}fromJSON(A,g){const I=A.value??"";this.value!=I&&(this.value=I)}clone(){return new StringParameter(this.name,this.value)}}Registry.register("StringParameter",StringParameter),Registry.register("Property_String",StringParameter);class StringListParameter extends Parameter{constructor(A="",g=[]){super(A,g,"String[]")}setValue(A){if(!Array.isArray(A))throw new Error("value provided is not an array. Check the source of this value");super.setValue(A)}readBinary(A,g){this.value=A.loadStrArray()}toJSON(A){return{type:this.getClassName(),name:this.name,value:this.value}}fromJSON(A,g){this.value=A.value}clone(){return new StringListParameter(this.name,this.value)}}Registry.register("StringListParameter",StringListParameter),Registry.register("Property_StringList",StringListParameter);class Float32ArrayParameter extends Parameter{constructor(A="",g=new Float32Array){super(A,g,"Float[]")}readBinary(A,g){this.value=A.loadFloat32Array()}toJSON(A){return{type:this.getClassName(),name:this.name,value:this.value}}fromJSON(A,g){this.value=A.value}clone(){return new Float32ArrayParameter(this.name,this.value)}}Registry.register("Float32ArrayParameter",Float32ArrayParameter),Registry.register("Property_Float32Array",Float32ArrayParameter);class ListParameter extends Parameter{constructor(A="",g){super(A,[],g)}filter(A){return!0}getCount(){return this.value?.length||0}getElement(A){if(this.value)return this.value[A]}setElement(A,g){this.value||(this.value=[]),this.value[A]=g,this.emit("valueChanged")}addElement(A){if((A||0==A)&&this.filter(A))return this.value||(this.value=[]),this.value.push(A),this.emit("elementAdded",{elem:A,index:this.value.length-1}),this.emit("valueChanged"),A}removeElement(A){this.value||(this.value=[]);const g=this.value[A];this.value.splice(A,1),this.emit("elementRemoved",{elem:g,index:A}),this.emit("valueChanged")}insertElement(A,g){this.value&&this.filter(g)&&(this.value.splice(A,0,g),this.emit("elementAdded",{elem:g,index:A}),this.emit("valueChanged"))}setValue(A){if(!Array.isArray(A))throw new Error("value provided is not an array. Check the source of this value");super.setValue(A)}toJSON(A){const g=[];if(this.value)for(const I of this.value)"string"==typeof this.dataType?g.push(I):g.push(I.toJSON(A));return{type:this.getClassName(),name:this.name,value:g}}fromJSON(A,g){if(null==A.items)return void console.warn("Invalid Parameter JSON");const I=[];for(let C=0;C<A.items.length;C++){let e;if("string"==typeof this.dataType)e=A.items[C];else{if(!this.dataType)throw"No DataType";e=Registry.constructClass(this.dataType),e.fromJSON(A.items[C],g)}I.push(e),this.emit("elementAdded",{elem:e,index:this.value.length-1})}this.value=I}clone(){const A=this.value?this.value.slice(0):[];if(!this.dataType)throw"This parameter does not have a DataType";const g=new ListParameter(this.name,this.dataType);return g.setValue(A),g}destroy(){if(this.value)for(let A=0;A<this.value.length;A++)this.value[A]instanceof Parameter&&this.value[A].destroy(),this.removeElement(A)}}Registry.register("ListParameter",ListParameter);class StructParameter extends Parameter{members;constructor(A){super(A,{},"Struct"),this.members=[]}addMember(A){return this.value&&(this.value[A.getName()]=A.value),A.on("valueChanged",(()=>{this.value&&(this.value[A.getName()]=A.value)})),this.members.push(A),this.emit("valueChanged"),A}getParameter(A){for(const g of this.members)if(g.getName()==A)return g}getMember(A){return this.getParameter(A)}getMemberNames(){const A=[];for(let g=0;g<this.members.length;g++){const I=this.members[g];null!=I&&(A[g]=I.getName())}return A}toJSON(A){const g=[];for(const I of this.members)g.push(I.toJSON(A));return{type:this.getClassName(),name:this.name,members:g}}fromJSON(A,g){if(null!=A.members){for(let I=0;I<A.members.length;I++)A.members[I]&&this.members[I].fromJSON(A.members[I],g);this.name=A.name}else console.warn("Invalid Parameter JSON")}clone(){return new StructParameter(this.name)}destroy(){for(const A of this.members)A.destroy()}}Registry.register("StructParameter",StructParameter);class OperatorOutput{name;mode;op=null;param;paramBindIndex;detached;constructor(A,g=OperatorOutputMode.OP_WRITE){this.name=A,this.mode=g,this.param=void 0,this.paramBindIndex=-1,this.detached=!1}setOperator(A){this.op=A}getOperator(){return this.op}getMode(){return this.mode}isConnected(){return null!=this.param}getParam(){return this.param}setParam(A,g=-1){this.param&&this.param.unbindOperatorOutput(this),this.param=A,this.param&&(this.paramBindIndex=this.param.bindOperatorOutput(this,g))}getParamBindIndex(){return this.paramBindIndex}setParamBindIndex(A){this.paramBindIndex=A}setDirty(){this.param&&this.param.setDirty(this.paramBindIndex)}getValue(){if(this.param)return this.param.getValueFromOp(this.paramBindIndex);throw new Error("Cannot call getValue on OperatorOutput that is not connected:",this.name)}backPropagateValue(A){return this.op&&(A=this.op.backPropagateValue(A)),A}setClean(A){this.param&&this.param.setCleanFromOp(A,this.paramBindIndex)}toJSON(A){const g=this.param?this.param.getPath():"";return{name:this.name,paramPath:A&&A.makeRelative?A.makeRelative(g):g,paramBindIndex:this.paramBindIndex}}fromJSON(A,g){A.paramPath&&g?.resolvePath(A.paramPath,(g=>{this.setParam(g,A.paramBindIndex)}),(g=>{console.warn("OperatorOutput: '"+this.name+"'. Unable to connect to:"+A.paramPath)}))}detach(){this.detached=!0,this.paramBindIndex=this.param?this.param.unbindOperatorOutput(this):-1}reattach(){this.detached=!1,this.param&&(this.paramBindIndex=this.param.bindOperatorOutput(this,this.paramBindIndex))}rebind(){this.param&&(this.param.unbindOperatorOutput(this),this.paramBindIndex=this.param.bindOperatorOutput(this))}}class BooleanOperatorOutput extends OperatorOutput{}class NumberOperatorOutput extends OperatorOutput{}class Vec2OperatorOutput extends OperatorOutput{}class Vec3OperatorOutput extends OperatorOutput{}class Vec4OperatorOutput extends OperatorOutput{}class ColorOperatorOutput extends OperatorOutput{}class QuatOperatorOutput extends OperatorOutput{}class XfoOperatorOutput extends OperatorOutput{}class Mat3OperatorOutput extends OperatorOutput{}class Mat4OperatorOutput extends OperatorOutput{}class Operator{name;inputs;outputs;constructor(A=""){this.name=A,this.inputs=new Map,this.outputs=new Map}setDirty(){this.outputs.forEach((A=>A.setDirty()))}addInput(A){return A.setOperator(this),this.inputs.set(A.name,A),this.setDirty(),A}removeInput(A){"string"==typeof A&&(A=this.getInput(A)),A.getParam()&&A.setParam(void 0),this.inputs.delete(A.name)}getNumInputs(){return this.inputs.size}getInputByIndex(A){return Array.from(this.inputs.values())[A]}getInput(A){const g=this.inputs.get(A);if(!g)throw`Couldn't find an Input with the name of '${A}'`;return g}addOutput(A){return A.setOperator(this),this.outputs.set(A.name,A),this.setDirty(),A}removeOutput(A){if("string"==typeof A&&(A=this.getOutput(A)),!(A instanceof OperatorOutput))throw new Error("removeOutput only accepts string or OperatorInput");A.getParam()&&A.setParam(),this.outputs.delete(A.name)}getNumOutputs(){return this.outputs.size}getOutputByIndex(A){return Array.from(this.outputs.values())[A]}getOutput(A){const g=this.outputs.get(A);if(!g)throw new Error(`Couldn't find an Output with the name of '${A}'`);return g}evaluate(){throw new Error("Not yet implemented")}backPropagateValue(A){return A}detach(){this.inputs.forEach((A=>A.detach())),this.outputs.forEach((A=>A.detach()))}reattach(){this.inputs.forEach((A=>A.reattach())),this.outputs.forEach((A=>A.reattach()))}rebind(){this.outputs.forEach((A=>A.rebind()))}}class OperatorInput{name;op;param;detached=!1;constructor(A){this.name=A}setOperator(A){this.op=A}getOperator(){return this.op}isConnected(){return null!=this.param}getParam(){return this.param}paramValueChanged(){this.op&&this.op.setDirty()}setParam(A){this.param&&this.param.unbindOperatorInput(this),this.param=A,this.param&&this.param.bindOperatorInput(this),this.op&&this.op.setDirty()}getValue(){if(this.param)return this.param.value;throw new Error("Unable to getValue")}setValue(A){this.param&&this.param.setValue(A)}setDirty(){this.op&&this.op.setDirty()}toJSON(A){const g=this.param?this.param.getPath():[],I=A&&A.makeRelative?A.makeRelative(g):g;return{name:this.name,paramPath:I}}fromJSON(A,g){A.paramPath&&g?.resolvePath(A.paramPath,(A=>{this.setParam(A)}),(()=>{console.warn("OperatorInput: '"+this.name+"'. Unable to connect to:"+A.paramPath)}))}detach(){this.param&&this.param.unbindOperatorInput(this)}reattach(){this.detached=!1,this.param&&this.param.bindOperatorInput(this)}}class BooleanOperatorInput extends OperatorInput{}class NumberOperatorInput extends OperatorInput{}class Vec2OperatorInput extends OperatorInput{}class Vec3OperatorInput extends OperatorInput{}class Vec4OperatorInput extends OperatorInput{}class ColorOperatorInput extends OperatorInput{}class QuatOperatorInput extends OperatorInput{}class XfoOperatorInput extends OperatorInput{}class Mat3OperatorInput extends OperatorInput{}class Mat4OperatorInput extends OperatorInput{}class CalcGlobalXfoOperator extends Operator{parentGlobal=new XfoOperatorInput("ParentGlobal");localXfo=new XfoOperatorInput("LocalXfo");globalXfo=new XfoOperatorOutput("GlobalXfo");constructor(A,g){super("CalcGlobalXfoOperator"),this.localXfo.setParam(g),this.globalXfo.setParam(A),this.addInput(this.parentGlobal),this.addInput(this.localXfo),this.addOutput(this.globalXfo)}backPropagateValue(A){if(this.parentGlobal.isConnected()){const g=this.parentGlobal.getValue();this.localXfo.setValue(g.inverse().multiply(A))}else this.localXfo.setValue(A)}evaluate(){const A=this.localXfo.getValue();if(this.parentGlobal.isConnected()){const g=this.parentGlobal.getValue();this.globalXfo.setClean(g.multiply(A))}else this.globalXfo.setClean(A)}}class BoundingBoxParameter extends Box3Parameter{treeItem;dirty=!0;constructor(A="",g){super(A),this.treeItem=g}setDirty(A){return this.dirty||(this.dirty=!0,this.emit("valueChanged")),!0}getValue(){return this.dirty&&(this.value=this.treeItem.cleanBoundingBox(),this.dirty=!1),super.getValue()}clone(){const A=new BoundingBoxParameter(this.name,this.treeItem);return A.value=this.value?.clone(),A}isDrivenByOperator(){return!0}loadValue(A){this.value=A.clone()}}Registry.register("BoundingBoxParameter",BoundingBoxParameter);class TreeItem extends ParameterOwner{disableBoundingBox=!1;__childItems=[];__childItemsEventHandlers=[];childItemsMapping={};childItemsMappingCorrupt=!1;globalXfoParam=new XfoParameter("GlobalXfo",new Xfo);localXfoParam=new XfoParameter("LocalXfo",new Xfo);boundingBoxParam=new BoundingBoxParameter("BoundingBox",this);visibleParam=new BooleanParameter("Visible",!0);pickableParam=new BooleanParameter("Pickable",!0);opacityParam=new NumberParameter("Opacity",1,[0,1]);highlightMapping={};highlights=[];#e=!0;#i=!0;visibleCounter=1;pickableCounter=1;opacity=1;inheritedOpacityValues=new Map;globalXfoOp;constructor(A){super(A),this.addParameter(this.visibleParam),this.addParameter(this.pickableParam),this.addParameter(this.opacityParam),this.addParameter(this.localXfoParam),this.addParameter(this.globalXfoParam),this.addParameter(this.boundingBoxParam),this.globalXfoOp=new CalcGlobalXfoOperator(this.globalXfoParam,this.localXfoParam),this.globalXfoParam.on("valueChanged",(A=>{this.setBoundingBoxDirty(),this.emit("globalXfoChanged",A)})),this.visibleParam.on("valueChanged",(()=>{this.visibleCounter+=this.visibleParam.value?1:-1,this.updateVisibility()})),this.pickableParam.on("valueChanged",(()=>{this.pickableCounter+=this.pickableParam.value?1:-1,this.updatePickable()})),this.opacityParam.on("valueChanged",(()=>{this.updateOpacity()}))}setOwner(A){if(A&&!(A instanceof TreeItem))throw new Error("cannot setOwner");if(this.ownerItem&&this.ownerItem instanceof TreeItem){this.ownerItem.isVisible()||this.visibleCounter++,this.ownerItem.isPickable()||this.pickableCounter++;const A=this.ownerItem.getChildIndex(this);A>=0&&this.ownerItem.unbindChild(A,this)}super.setOwner(A),this.ownerItem?this.ownerItem instanceof TreeItem&&(this.ownerItem.isVisible()||this.visibleCounter--,this.ownerItem.isPickable()||this.pickableCounter--,this.globalXfoOp.getInput("ParentGlobal").setParam(this.ownerItem.globalXfoParam)):this.globalXfoOp.getInput("ParentGlobal").setParam(void 0),this.updateVisibility()}getParentItem(){return this.getOwner()}setParentItem(A){this.setOwner(A)}get parent(){return this.getOwner()}set parent(A){this.setOwner(A)}isVisible(){return this.visibleCounter>0}setVisible(A){this.visibleParam.value=A}propagateVisibility(A){this.visibleCounter+=A,this.updateVisibility()}updateVisibility(){const A=this.visibleCounter>0;if(A!=this.#e){this.#e=A;for(const A of this.__childItems)A.propagateVisibility(this.#e?1:-1);return this.emit("visibilityChanged",new StateChangedEvent(A)),this.ownerItem instanceof TreeItem&&this.ownerItem.setBoundingBoxDirty(),!0}return!1}isPickable(){return this.pickableCounter>0}propagatePickable(A){this.pickableCounter+=A,this.updatePickable()}updatePickable(){const A=this.pickableCounter>0;if(A!=this.#i){this.#i=A;for(const A of this.__childItems)A.propagatePickable(this.#i?1:-1);return this.emit("pickabilityChanged",new StateChangedEvent(A)),!0}return!1}isOpaque(){return this.opacity>.999}setInheritedOpacity(A,g){this.inheritedOpacityValues.set(A,g),this.updateOpacity()}updateOpacity(){let A=1;this.inheritedOpacityValues.forEach((g=>{g<A&&(A=g)}));const g=this.opacity>.999;this.opacity=this.opacityParam.value*A;for(const A of this.__childItems)A.setInheritedOpacity(this,this.opacity);const I=this.opacity>.999;this.emit("opacityChanged",new OpacityStateChangedEvent(I,g!=I))}addHighlight(A,g,I=!0){if(A in this.highlightMapping)if(this.highlights[this.highlights.length-1]!=A){const I=this.highlights.indexOf(A);this.highlights.splice(I,1),this.highlights.push(A),this.emit("highlightChanged",{name:A,color:g})}else this.highlightMapping[A].isEqual(g)||(this.highlightMapping[A]=g,this.emit("highlightChanged",{name:A,color:g}));else this.highlights.push(A),this.highlightMapping[A]=g,this.emit("highlightChanged",{name:A,color:g});I&&this.__childItems.forEach((C=>{C.addHighlight(A,g,I)}))}removeHighlight(A,g=!0){if(A in this.highlightMapping){if(this.highlights[this.highlights.length-1]==A)if(this.highlights.pop(),delete this.highlightMapping[A],this.highlights.length>0){const A=this.highlights[this.highlights.length-1],g=this.highlightMapping[A];this.emit("highlightChanged",{name:A,color:g})}else this.emit("highlightChanged");else{const g=this.highlights.indexOf(A);this.highlights.splice(g,1),delete this.highlightMapping[A]}g&&this.__childItems.forEach((I=>{I.removeHighlight(A,g)}))}}getHighlight(){return 0==this.highlights.length?null:this.highlightMapping[this.highlights[this.highlights.length-1]]}getHighlightName(){return 0==this.highlights.length?null:this.highlights[this.highlights.length-1]}isHighlighted(){return this.highlights.length>0}cleanBoundingBox(){const A=new Box3;return this.__childItems.forEach((g=>{if(g.isVisible()&&g.isPickable()){const I=g.boundingBoxParam.value;I&&A.addBox3(I)}})),A}setBoundingBoxDirty(){this.boundingBoxParam&&this.boundingBoxParam.setDirty(-1),this.ownerItem instanceof TreeItem&&this.ownerItem.setBoundingBoxDirty()}getChildren(){return this.__childItems}getNumChildren(){return this.__childItems.length}get numChildren(){return this.__childItems.length}generateUniqueName(A){for(;null!=this.childItemsMapping[A.toLocaleLowerCase()];){const g=/(\d+)$/i,I=A.match(g);let C=1;I&&(C=parseInt(I[1])+1,A=A.replace(g,"")),A+=`${C}`.padStart(2,"0")}return A}updateChildNameMapping(A){for(let g=A;g<this.__childItems.length;g++){const A=this.__childItems[g].name.toLocaleLowerCase();this.childItemsMapping[A]=g}}childNameChanged(A){if(this.childItemsMappingCorrupt)this.updateChildNameMapping(0),this.childItemsMappingCorrupt=!1;else{const g=A.oldName.toLocaleLowerCase(),I=A.newName.toLocaleLowerCase(),C=this.childItemsMapping[g];null!=this.childItemsMapping[I]&&(this.childItemsMappingCorrupt=!0),delete this.childItemsMapping[g],this.childItemsMapping[I]=C}}insertChild(A,g,I=!1,C=!0){let e=A.name,i=e.toLocaleLowerCase();if(i in this.childItemsMapping){if(!C)throw new Error("Item '"+e+"' is already a child of :"+this.getPath());e=this.generateUniqueName(e),i=e.toLocaleLowerCase(),A.setName(e)}if(!(A instanceof TreeItem))throw new Error("Object is is not a tree item :"+A);const t={};let o;return t.nameChanged=A.on("nameChanged",(A=>{this.childNameChanged(A)})),I&&(o=A.globalXfoParam.value),this.setBoundingBoxDirty(),this.highlights.forEach((g=>{A.addHighlight(g,this.highlightMapping[g],!0)})),A.setInheritedOpacity(this,this.opacity),this.__childItems.splice(g,0,A),this.__childItemsEventHandlers.splice(g,0,t),this.childItemsMapping[i]&&(this.childItemsMappingCorrupt=!0),this.childItemsMapping[i]=g,this.updateChildNameMapping(g+1),A.setOwner(this),I&&(A.globalXfoParam.value=o),this.emit("childAdded",new ChildAddedEvent(g,A)),A}addChild(A,g=!0,I=!0){const C=this.__childItems.length;return this.insertChild(A,C,g,I),A}getChild(A){return this.__childItems[A]}getChildByName(A){const g=this.childItemsMapping[A.toLocaleLowerCase()];return null!=g?this.__childItems[g]:null}getChildNames(){const A=[];for(let g=0;g<this.__childItems.length;g++){const I=this.__childItems[g];null!=I&&(A[g]=I.getName())}return A}unbindChild(A,g){const I=this.__childItemsEventHandlers[A];g.off("nameChanged",I.nameChanged),this.__childItems.splice(A,1),this.__childItemsEventHandlers.splice(A,1),delete this.childItemsMapping[g.name.toLocaleLowerCase()],this.updateChildNameMapping(A),this.setBoundingBoxDirty(),this.emit("childRemoved",{childItem:g,index:A})}removeChild(A){const g=this.__childItems[A];g&&(this.unbindChild(A,g),g.setOwner(void 0))}removeChildByName(A){const g=this.childItemsMapping[A.toLocaleLowerCase()];if(null!=g)return this.removeChild(g)}removeChildByHandle(A){const g=this.__childItems.indexOf(A);if(-1==g)throw new Error("Error in removeChildByHandle. Child not found:"+A.getName());this.removeChild(g)}removeAllChildren(){let A=this.__childItems.length;for(;A--;)this.removeChild(A);this.setBoundingBoxDirty()}getChildIndex(A){return this.__childItems.indexOf(A)}resolvePath(A,g=0){if(0==g&&("."!=A[0]&&A[0]!=this.name||g++),".."==A[g]){if(this.ownerItem)return this.ownerItem.resolvePath(A,g+1);throw Error("this.ownerItem is undefined")}if(g==A.length)return this;const I=A[g],C=this.getChildByName(I);return C?C.resolvePath(A,g+1):super.resolvePath(A,g)}traverse(A,g=!0){const I=(A,g)=>{const I=A.getChildren();for(const A of I)A&&C(A,g)},C=(g,C)=>{0!=A(g,C)&&I(g,C+1)};g?C(this,1):I(this,0)}onPointerDown(A){this.emit("pointerDown",A),A.propagating&&this.ownerItem instanceof TreeItem&&this.ownerItem.onPointerDown(A)}onPointerUp(A){this.emit("pointerUp",A),A.propagating&&this.ownerItem instanceof TreeItem&&this.ownerItem.onPointerUp(A)}onPointerMove(A){this.emit("pointerMove",A),A.propagating&&this.ownerItem instanceof TreeItem&&this.ownerItem.onPointerMove(A)}onPointerEnter(A){this.emit("pointerEnter",A),A.propagating&&this.ownerItem instanceof TreeItem&&this.ownerItem.onPointerEnter(A)}onPointerLeave(A){this.emit("pointerLeave",A),A.propagating&&this.ownerItem instanceof TreeItem&&this.ownerItem.onPointerLeave(A)}onPointerClick(A){this.emit("pointerClick",A),A.propagating&&this.ownerItem instanceof TreeItem&&this.ownerItem.onPointerClick(A)}onPointerDoubleClick(A){this.emit("pointerDoubleClick",A),A.propagating&&this.ownerItem instanceof TreeItem&&this.ownerItem.onPointerDoubleClick(A)}onPointerLongPress(A){this.emit("pointerLongPress",A),A.propagating&&this.ownerItem instanceof TreeItem&&this.ownerItem.onPointerLongPress(A)}onWheel(A){this.emit("mouseWheel",A),A.propagating&&this.ownerItem instanceof TreeItem&&this.ownerItem.onWheel(A)}onTouchCancel(A){this.emit("touchCancel",A),A.propagating&&this.ownerItem instanceof TreeItem&&this.ownerItem.onTouchCancel(A)}toJSON(A){let g=super.toJSON(A);const I={};for(const g of this.__childItems)if(g){const C=g.toJSON(A);C&&(I[g.getName()]=C)}return Object.keys(I).length>0&&(g?g.children=I:g={name:this.name,children:I}),g}fromJSON(A,g,I){if(super.fromJSON(A,g),null!=A.children){const I=A.children;if(Array.isArray(I))for(const A of I){let I=this.getChildByName(A.name);I?I.fromJSON(A,g):A.type&&(I=Registry.constructClass(A.type),I&&(I.fromJSON(A,g),this.addChild(I,!1,!1)))}else for(const A in I){const C=I[A];let e=this.getChildByName(A);e?e.fromJSON(C,g):C.type&&(e=Registry.constructClass(C.type),e&&(e.fromJSON(C,g),this.addChild(e,!1,!1)))}}}readBinary(A,g){super.readBinary(A,g);const I=A.loadUInt8();2&I||this.setVisible(!1);if(4&I){const g=new Xfo;if(g.tr=A.loadFloat32Vec3(),g.ori=A.loadFloat32Quat(),32&I)g.sc=A.loadFloat32Vec3();else{const I=A.loadFloat32();g.sc.set(I,I,I)}this.localXfoParam.value=g}8&I&&this.boundingBoxParam.loadValue(new Box3(A.loadFloat32Vec3(),A.loadFloat32Vec3()));const C=A.loadUInt32();if(C>0){const I=A.loadUInt32Array(C);for(let e=0;e<C;e++)try{A.seek(I[e]);let C=A.loadStr();const i=Registry.constructClass(C);if(!i){const g=A.loadStr();console.warn("Unable to construct child:"+g+" of type:"+C);continue}A.seek(I[e]),i.readBinary(A,g),this.addChild(i,!1,!0)}catch(A){console.warn("Error loading tree item: ",A)}}}clone(A){const g=new TreeItem("");return g.copyFrom(this,A),g}copyFrom(A,g){if(!(A instanceof TreeItem))throw new Error("cannot copy from src");super.copyFrom(A,g),this.visibleCounter=this.visibleParam.value?1:0,this.updateVisibility(),this.updateOpacity(),this.removeAllChildren(),A.getChildren().forEach((A=>{A&&this.addChild(A.clone(g),!1,!1)}))}}Registry.register("TreeItem",TreeItem);class TreeItemParameter extends Parameter{filterFn;listenerIDs={};constructor(A="",g){super(A,null,"TreeItem"),this.filterFn=g}emitTreeItemGlobalXfoChanged(A){this.emit("treeItemGlobalXfoChanged",A)}setFilterFn(A){this.filterFn=A}getFilterFn(){return this.filterFn}setValue(A){if(!(A instanceof TreeItem))throw new Error("value provided is not an instance of a 'TreeItem' class. Check the source of this value");this.filterFn&&!this.filterFn(A)||this.value!==A&&(this.value&&this.value.off("globalXfoChanged",this.listenerIDs.globalXfoChanged),super.setValue(A),this.value&&(this.listenerIDs.globalXfoChanged=this.value.on("globalXfoChanged",(A=>{this.emitTreeItemGlobalXfoChanged(A)}))))}toJSON(A){return{type:this.getClassName(),name:this.name,value:A.makeRelative(this.value?.getPath())}}fromJSON(A,g){null!=A.value?g.resolvePath(A.value,(A=>{this.setValue(A)}),(()=>{console.warn("Unable to resolve tree item parameter value:"+A.paramPath)})):console.warn("Invalid Parameter JSON")}clone(A){const g=new TreeItemParameter(this.name,this.filterFn);return this.value&&g.setValue(this.value.clone(A)),g}}Registry.register("TreeItemParameter",TreeItemParameter);class ItemEvent extends BaseEvent{item;index;constructor(A,g){super(),this.index=g,this.item=A}}class ItemSetParameter extends Parameter{filterFn;constructor(A="",g){super(A,new Set,"TreeItem"),this.filterFn=g}setFilterFn(A){this.filterFn=A}getFilterFn(){return this.filterFn}getItem(A){return Array.from(this.value)[A]}addItem(A,g=!0){if(!this.filterFn||this.filterFn(A)){if(this.value.has(A))return-1;{this.value.add(A);const I=Array.from(this.value).indexOf(A);return this.emit("itemAdded",new ItemEvent(A,I)),g&&this.emit("valueChanged"),I}}console.warn("ItemSet __filterFn rejecting item:",A.getPath())}addItems(A,g=!0){A.forEach((A=>this.addItem(A,!1))),g&&this.emit("valueChanged")}removeItem(A,g=!0){const I=Array.from(this.value)[A];return this.value.delete(I),this.emit("itemRemoved",new ItemEvent(I,A)),g&&this.emit("valueChanged"),I}setItems(A,g=!0){const I=Array.from(this.value);for(let g=I.length-1;g>=0;g--){const C=I[g];A.has(C)||this.removeItem(g,!1)}for(const g of A)this.value.has(g)||this.addItem(g,!1);g&&this.emit("valueChanged")}clearItems(A=!0){this.value.clear(),A&&this.emit("valueChanged")}getNumItems(){return this.value.size}toJSON(A){this.value||(this.value=new Set);const g=[];for(const I of this.value){const C=I.getPath();g.push(A&&A.makeRelative?A.makeRelative(C):C)}return{type:this.getClassName(),name:this.name,value:g}}fromJSON(A,g){if(!g||!g.resolvePath)throw new Error("Unable to load JSON on a ItemSetParameter without a load context");A.value.forEach((A=>{g.resolvePath(A,(A=>{this.addItem(A,!1)}),(()=>{console.warn("BaseGroup: '"+this.getName()+"'. Unable to load item:"+A)}))}))}clone(){return new ItemSetParameter(this.name,this.filterFn)}}Registry.register("ItemSetParameter",ItemSetParameter);class MaterialFloatParam extends NumberParameter{image;constructor(A="",g,I){super(A,g,I)}getImage(){return this.image}setImage(A){const g=()=>{this.emit("textureDisconnected")};A?(null!=this.image&&this.image!==A&&g(),this.image=A,this.emit("textureConnected"),this.emit("valueChanged")):null!=this.image&&(g(),this.image=void 0,this.emit("textureDisconnected"))}setValue(A){if("number"!=typeof A&&!(A instanceof BaseImage))throw new Error("value provided is not of type number of an instance of 'BaseImage' class. Check the source of this value");A instanceof BaseImage?this.setImage(A):super.setValue(A)}readBinary(A,g){super.readBinary(A,g);const I=A.loadStr();""!=I&&(console.log("Load Texture"),this.setImage(g.assetItem.materialLibrary.getImage(I)))}clone(){return new MaterialFloatParam(this.name,this.value,this.range)}}Registry.register("MaterialFloatParam",MaterialFloatParam),function(A){A.Gamma="Gamma",A.Linear="Linear"}(ColorSpace||(ColorSpace={}));class MaterialColorParam extends ColorParameter{listenerIDs={};image;colorSpace=ColorSpace.Gamma;constructor(A,g){super(A,g)}getImage(){return this.image}imageUpdated(){this.emit("valueChanged")}setImage(A){const g=()=>{this.image&&(this.image.off("updated",this.listenerIDs.updated),this.image=void 0),this.emit("textureDisconnected")};A?(null!=this.image&&this.image!==A&&g(),this.image=A,this.listenerIDs.updated=this.image.on("updated",(()=>{this.imageUpdated()})),this.emit("textureConnected"),this.emit("valueChanged")):null!=this.image&&(g(),this.image=void 0,this.emit("textureDisconnected"))}setValue(A){if(!(A instanceof BaseImage||A instanceof Color))throw new Error("value provided is not an instance of a 'Color' or 'BaseImage' class. Check the source of this value");A instanceof BaseImage?this.setImage(A):super.setValue(A)}readBinary(A,g){super.readBinary(A,g);const I=A.loadStr();""!=I&&this.setImage(g.assetItem.materialLibrary.getImage(I)),this.colorSpace=ColorSpace.Linear}toJSON(A){const g=super.toJSON(A);return g.colorSpace=this.colorSpace,g}fromJSON(A,g){super.fromJSON(A,g),this.colorSpace=ColorSpace[A.colorSpace]??ColorSpace.Gamma}clone(){const A=new MaterialColorParam(this.name,this.value?.clone());return A.colorSpace=this.colorSpace,A}}Registry.register("MaterialColorParam",MaterialColorParam);class Material extends ParameterOwner{__isOpaque=!0;__isTextured=!1;__shaderName="";libraryIndex=-1;constructor(A,g){super(A),g&&this.setShaderName(g)}getShaderName(){return this.__shaderName}setShaderName(A){if(this.__shaderName==A)return;this.__shaderName=A;const g=Registry.getClassDefinition(A).getMaterialTemplate();if(!g)throw new Error("Error setting Shader. Material template not registered found:"+A);const I={};let C=g.getNumParameters();for(;C--;){const A=g.getParameterByIndex(C);this.getParameter(A.getName())||this.addParameter(A.clone()),I[A.getName()]=!0}for(const A of this.params)I[A.getName()]||this.removeParameter(A.getName());this.__shaderName=A,this.__checkOpacity({});const e=new ShaderNameChangedEvent(A);this.emit("shaderNameChanged",e)}removeAllTextures(){for(const A of this.params)A instanceof MaterialColorParam?A.getImage()&&A.setImage(null):A instanceof MaterialFloatParam&&A.getImage()&&A.setImage(null)}getPath(){return null==this.ownerItem?[this.name]:this.ownerItem instanceof Parameter?[...this.ownerItem.getPath(),"value"]:[...this.ownerItem.getPath(),this.name]}getParamTextures(){const A={};for(const g of this.params)g instanceof MaterialColorParam?g.getImage()&&(A[g.getName()]=g.getImage()):g instanceof MaterialFloatParam&&g.getImage()&&(A[g.getName()]=g.getImage());return A}isOpaque(){return this.__isOpaque}__checkOpacity(A){let g=!0;try{this.getShaderClass().isOpaque()||(g=!1)}catch(A){}if(g){const A=this.getParameter("Opacity");if(A&&(A.value<.99||A instanceof MaterialFloatParam&&A.getImage()))g=!1;else{const A=this.getParameter("BaseColor");if(A){if(A instanceof MaterialColorParam){const I=A.getImage();I&&"RGBA"==I.format&&(g=!1)}if(g&&A.value){A.value.a<1&&(g=!1)}}}}if(g!=this.__isOpaque){this.__isOpaque=g;const A=new OpacityStateChangedEvent(g,!0);this.emit("opacityChanged",A),this.emit("transparencyChanged",A)}}isTextured(){return this.__isTextured}__checkTextures(A){const g=A||{};let I=!1;for(const A of this.params)if(A instanceof MaterialColorParam){if(A.getImage()){I=!0;break}}else if(A instanceof MaterialFloatParam&&A.getImage()){I=!0;break}if(I!=this.__isTextured){this.__isTextured=I;let A=new TexturedChangedEvent(I,g);this.emit("texturedChanged",A)}}parameterValueChanged(A){this.__checkOpacity(A),this.__checkTextures(A),super.parameterValueChanged(A)}getShaderClass(){return Registry.getClassDefinition(this.getShaderName())}toJSON(A){const g=super.toJSON(A);return g.shader=this.__shaderName,g}fromJSON(A,g={}){A.shader?(this.setShaderName(A.shader),super.fromJSON(A,g),this.__checkOpacity(),this.__checkTextures()):console.warn("Invalid Material JSON")}readBinary(A,g){let I=A.loadStr();"StandardMaterial"==I&&(I="StandardSurfaceShader"),"TransparentMaterial"==I&&(I="StandardSurfaceShader"),this.setShaderName(I),super.readBinary(A,g),this.__checkOpacity(),this.__checkTextures()}clone(A){const g=new Material("clone","");return g.copyFrom(this,A),g}copyFrom(A,g){if(!(A instanceof Material))throw new Error("cannot copy from src");this.setShaderName(A.getShaderName()),super.copyFrom(A,g)}}Registry.register("Material",Material);class MaterialParameter extends Parameter{listenerIDs={};constructor(A="",g=null){super(A,null,"Material"),g&&this.setValue(g)}valueParameterValueChanged(A){this.emit("valueParameterValueChanged",A)}setValue(A){if(null!=A&&!(A instanceof Material))throw new Error("value provided is not an instance of a 'Material' class. Check the source of this value");this.value!==A&&(this.value&&(this.value.setOwner(null),this.value.off("parameterValueChanged",this.listenerIDs.parameterValueChanged)),super.setValue(A),this.value&&(this.value.setOwner(this),this.listenerIDs.parameterValueChanged=this.value.on("parameterValueChanged",(A=>{this.valueParameterValueChanged(A)}))))}loadValue(A){this.value&&(this.value.setOwner(null),this.value.off("parameterValueChanged",this.listenerIDs.parameterValueChanged)),super.loadValue(A),this.value&&(this.value.setOwner(this),this.listenerIDs.parameterValueChanged=this.value.on("parameterValueChanged",(A=>{this.valueParameterValueChanged(A)})))}toJSON(A){const g={type:this.getClassName(),name:this.name};return this.value&&(this.value.libraryIndex>=0?g.libraryIndex=this.value.libraryIndex:g.value=this.value.toJSON(A)),g}fromJSON(A,g){if(null!=A.libraryIndex){if(g&&g.assetItem){const I=g.assetItem.getMaterialLibrary().getMaterial(A.libraryIndex);I&&this.setValue(I)}}else if(A.value)if(this.value&&this.value.getClassName()==A.value.type)this.value.fromJSON(A.value);else{const I=Registry.constructClass(A.value.type);A.value&&I.fromJSON(A.value,g),this.setValue(I)}}clone(){return new MaterialParameter(this.name,this.value)}}function approxEqual(A,g){for(let I=0;I<A.length;I++)if(Math.abs(g[I]-A[I])>.001)return!1;return!0}Registry.register("MaterialParameter",MaterialParameter);const resizeArray=(A,g)=>{if(A instanceof Uint8Array){const I=new Uint8Array(g);return I.set(A),I}if(A instanceof Int8Array){const I=new Int8Array(g);return I.set(A),I}if(A instanceof Uint16Array){const I=new Uint16Array(g);return I.set(A),I}{const I=new Float32Array(g);return I.set(A),I}};class Attribute extends BaseClass{dataTypeName;stride;initValue=Number.NaN;normalized;data;mesh;splitValues=[];splits={};constructor(A,g,I=Number.NaN){super(),this.dataTypeName=A,this.stride=g,this.initValue=I,this.init()}init(){this.data=new Float32Array(0),this.initRange(0)}setMesh(A){this.mesh=A}asArray(){return this.data}getDataTypeName(){return this.dataTypeName}getCount(){return this.data.length/this.stride}get count(){return this.data.length/this.stride}setCount(A){const g=this.data.length,I=A*this.stride;I>g?(this.data=resizeArray(this.data,I),this.initRange(g)):I<g&&(this.data=this.data.slice(0,I)),this.splits={},this.splitValues=[]}initRange(A){for(let g=A;g<this.data.length;g++)this.data[g]=this.initValue}isInitialized(A){for(let g=0;g<A.length;g++)if(A[g]!=this.initValue)return!0;return!1}get numElements(){return this.stride}getFloat32Value(A){return this.data[A]}setFloat32Value(A,g){this.data[A]=g}getValues(A){if(A>=this.data.length/this.stride)throw new Error("Invalid vertex index:"+A+". Num Vertices:"+this.data.length/this.stride);const g=A*this.stride;return this.data.subarray(g,g+this.stride)}setValues(A,g){if(A>=this.data.length/this.stride)throw new Error("Invalid vertex index:"+A+". Num Vertices:"+this.data.length/this.stride);const I=A*this.stride;this.data.set(g,I)}merge(A,g=new Xfo){const I=this.data.length,C=A.data.length,e=new Float32Array(I+C);e.set(this.data,0);for(let g=0;g<C;g++)e[I+g]=A.data[g];this.data=e,this.splitValues=[...this.splitValues,...A.splitValues]}getSplits(){return this.splits}getFaceVertexValue_array(A,g){const I=this.mesh.getFaceVertexIndex(A,g);return I in this.splits&&A in this.splits[I]?this.splitValues[this.splits[I][A]]:this.data.subarray(I*this.stride,(I+1)*this.stride)}setFaceVertexValue_array(A,g,I){const C=this.mesh.getFaceVertexIndex(A,g);this.setFaceVertexValue_ByVertexIndex(A,C,I)}setFaceVertexValue_ByVertexIndex(A,g,I){const C=this.data.subarray(g*this.stride,(g+1)*this.stride);if(this.isInitialized(C))if(approxEqual(C,I));else{if(g in this.splits){const C=this.splits[g];for(const g in C){const e=C[g];if(approxEqual(this.splitValues[e],I))return void(C[A]=e)}if(A in this.splits[g])return void(this.splitValues[this.splits[g][A]]=I)}else this.splits[g]={};this.splits[g][A]=this.splitValues.length,this.splitValues.push(I)}else C.set(I)}setSplitVertexValue_array(A,g,I){if(A in this.splits||(this.splits[A]={}),g in this.splits[A]){if(approxEqual(this.splitValues[this.splits[A][g]],I))return;console.warn("Face Vertex Already Split with different value")}this.splits[A][g]=this.splitValues.length,this.splitValues.push(I)}setSplitVertexValues(A,g,I){A in this.splits||(this.splits[A]={});const C=this.splitValues.length;this.splitValues.push(I);for(const I of g)this.splits[A][I]=C}generateSplitValues(A,g){if(0==g)return this.data;const I=this.getCount(),C=resizeArray(this.data,(I+g)*this.stride);for(const g in A){const e=A[g];for(const A in e){const i=I+e[A];if(g in this.splits&&A in this.splits[g]){const I=this.splits[g][A],e=this.splitValues[I];for(let A=0;A<e.length;A++)C[i*this.stride+A]=e[A]}else{const A=parseInt(g);for(let g=0;g<this.stride;g++)C[i*this.stride+g]=this.data[A*this.stride+g]}}}return C}toJSON(A){return{data:Array.from(this.data),dataType:this.dataTypeName,length:this.data.length/this.stride}}fromJSON(A){const g=A.data;for(let A=0;A<g.length;A++)this.data[A]=g[A]}loadSplitValues(A){const g=A.loadUInt32Array();if(0==g.length)return;let I=0,C=0;for(;;){const A=g[I++],e=g[I++],i={};for(let A=0;A<e;A++){const A=g[I++],e=g[I++];i[A]=e,e>=C&&(C=e+1)}if(this.splits[A]=i,I>=g.length)break}const e=this.stride,i=A.loadFloat32Array(C*e);this.splitValues=[];for(let A=0;A<C;A++){const g=i.slice(A*e,A*e+e);this.splitValues.push(g)}}toString(){return JSON.stringify(this.toJSON(),null,2)}genBuffer(){return{values:this.data,count:this.getCount(),dimension:this.stride,dataType:this.dataTypeName,normalized:this.normalized}}}class Vec2Attribute extends Attribute{constructor(A="Vec2"){super(A,2),this.normalized=!1}getValue(A){if(A>=this.data.length/this.stride)throw new Error("Invalid vertex index:"+A+". Num Vertices:"+this.data.length/3);const g=A*this.stride;return new Vec2(this.data[g],this.data[g+1])}setValue(A,g){if(A>=this.data.length/this.stride)throw new Error("Invalid vertex index:"+A+". Num Vertices:"+this.data.length/3);const I=A*this.stride;this.data.set(g.asArray(),I)}getFaceVertexValue(A,g){const I=this.getFaceVertexValue_array(A,g);return new Vec2(I[0],I[1])}setFaceVertexValue(A,g,I){this.setFaceVertexValue_array(A,g,Float32Array.from(I.asArray()))}setSplitVertexValue(A,g,I){this.setSplitVertexValue_array(A,g,Float32Array.from(I.asArray()))}merge(A,g=new Xfo){const I=this.getCount(),C=A.getCount();this.setCount(I+C);for(let g=0;g<C;g++)this.setValue(I+g,A.getValue(g));this.splitValues=[...this.splitValues,...A.splitValues]}}Registry.register("Vec2Attribute",Vec2Attribute);const mapIn$2=A=>MathFunctions.encode16BitFloat(A),mapOut$2=A=>MathFunctions.decode16BitFloat(A);class Vec2f16Attribute extends Vec2Attribute{constructor(){super("Vec2f16")}init(){this.data=new Uint16Array(0),this.initRange(0)}initRange(A){for(let g=A;g<this.data.length;g++)this.data[g]=mapIn$2(Number.NaN)}isInitialized(A){for(let g=0;g<A.length;g++)if(Number.isFinite(mapOut$2(A[g])))return!0;return!1}getValue(A){if(A>=this.data.length/this.stride)throw new Error("Invalid vertex index:"+A+". Num Vertices:"+this.data.length/3);const g=A*this.stride,I=this.data.subarray(g,g+this.stride);return new Vec2(mapOut$2(I[0]),mapOut$2(I[1]))}setValue(A,g){if(A>=this.data.length/this.stride)throw new Error("Invalid vertex index:"+A+". Num Vertices:"+this.data.length/3);const I=A*this.stride,C=this.data.subarray(I,I+this.stride);C[0]=mapIn$2(g.x),C[1]=mapIn$2(g.y)}getFaceVertexValue(A,g){const I=this.getFaceVertexValue_array(A,g);return new Vec2(mapOut$2(I[0]),mapOut$2(I[1]))}setFaceVertexValue(A,g,I){const C=new Uint16Array(2);C[0]=mapIn$2(I.x),C[1]=mapIn$2(I.y),this.setFaceVertexValue_array(A,g,C)}}class Vec3Ref{data;constructor(A){this.data=A}get x(){return this.data[0]}set x(A){this.data[0]=A}get y(){return this.data[1]}set y(A){this.data[1]=A}get z(){return this.data[2]}set z(A){this.data[2]=A}set(A,g,I){this.data[0]=A,this.data[1]=g,this.data[2]=I}}class Vec3Attribute extends Attribute{constructor(A="Vec3"){super(A,3),this.normalized=!1}getValue(A){if(A>=this.data.length/this.stride)throw new Error("Invalid vertex index:"+A+". Num Vertices:"+this.data.length/3);const g=A*this.stride;return new Vec3(this.data[g+0],this.data[g+1],this.data[g+2])}getValueRef(A){if(A>=this.data.length/this.stride)throw new Error("Invalid vertex index:"+A+". Num Vertices:"+this.data.length/3);const g=A*this.stride;return new Vec3Ref(this.data.subarray(g,g+3))}setValue(A,g){this.setValues(A,g.asArray())}getFaceVertexValue(A,g){const I=this.getFaceVertexValue_array(A,g);return new Vec3(I[0],I[1],I[2])}setFaceVertexValue(A,g,I){this.setFaceVertexValue_array(A,g,Float32Array.from(I.asArray()))}setSplitVertexValue(A,g,I){this.setSplitVertexValue_array(A,g,Float32Array.from(I.asArray()))}merge(A,g=new Xfo){const I=this.getCount(),C=A.getCount();this.setCount(I+C);for(let e=0;e<C;e++)this.setValue(I+e,g.transformVec3(A.getValue(e)))}}Registry.register("Vec3Attribute",Vec3Attribute);class Vec3f8Ref{data;constructor(A){this.data=A}get x(){return MathFunctions.decode16BitFloat(this.data[0])}set x(A){this.data[0]=MathFunctions.encode16BitFloat(A)}get y(){return MathFunctions.decode16BitFloat(this.data[1])}set y(A){this.data[1]=MathFunctions.encode16BitFloat(A)}get z(){return MathFunctions.decode16BitFloat(this.data[2])}set z(A){this.data[2]=MathFunctions.encode16BitFloat(A)}set(A,g,I){this.data[0]=MathFunctions.encode16BitFloat(A),this.data[1]=MathFunctions.encode16BitFloat(g),this.data[2]=MathFunctions.encode16BitFloat(I)}}const mapIn$1=(A,g)=>MathFunctions.remap(A,g[0],g[1],-127,127),mapOut$1=(A,g)=>MathFunctions.remap(A,-127,127,g[0],g[1]);class Vec3f8Attribute extends Vec3Attribute{valueRange;constructor(A=[-1,1]){super("Vec3f8"),this.valueRange=A}init(){this.data=new Int8Array(0),this.initRange(0)}initRange(A){for(let g=A;g<this.data.length;g++)this.data[g]=mapIn$1(Number.NaN,this.valueRange)}isInitialized(A){for(let g=0;g<A.length;g++)if(Number.isFinite(mapOut$1(A[g],this.valueRange)))return!0;return!1}getValue(A){if(A>=this.data.length/this.stride)throw new Error("Invalid vertex index:"+A+". Num Vertices:"+this.data.length/3);const g=A*this.stride,I=this.data.subarray(g,g+this.stride);return new Vec3(mapOut$1(I[0],this.valueRange),mapOut$1(I[1],this.valueRange),mapOut$1(I[2],this.valueRange))}getValueRef(A){if(A>=this.data.length/this.stride)throw new Error("Invalid vertex index:"+A+". Num Vertices:"+this.data.length/3);const g=A*this.stride;return new Vec3f8Ref(this.data.subarray(g,g+3))}setValue(A,g){if(A>=this.data.length/this.stride)throw new Error("Invalid vertex index:"+A+". Num Vertices:"+this.data.length/3);const I=A*this.stride,C=this.data.subarray(I,I+this.stride);C[0]=mapIn$1(g.x,this.valueRange),C[1]=mapIn$1(g.y,this.valueRange),C[2]=mapIn$1(g.z,this.valueRange)}getFaceVertexValue(A,g){const I=this.getFaceVertexValue_array(A,g);return new Vec3(mapOut$1(I[0],this.valueRange),mapOut$1(I[1],this.valueRange),mapOut$1(I[2],this.valueRange))}setFaceVertexValue(A,g,I){const C=new Int8Array(3);C[0]=mapIn$1(I.x,this.valueRange),C[1]=mapIn$1(I.y,this.valueRange),C[2]=mapIn$1(I.z,this.valueRange),this.setFaceVertexValue_array(A,g,C)}setSplitVertexValues(A,g,I){super.setSplitVertexValues(A,g,I.map((A=>mapIn$1(A,this.valueRange)),this.valueRange))}}const mapIn=A=>MathFunctions.encode16BitFloat(A),mapOut=A=>MathFunctions.decode16BitFloat(A);class Vec3f16Ref{data;constructor(A){this.data=A}get x(){return mapOut(this.data[0])}set x(A){this.data[0]=mapIn(A)}get y(){return mapOut(this.data[1])}set y(A){this.data[1]=mapIn(A)}get z(){return mapOut(this.data[2])}set z(A){this.data[2]=mapIn(A)}set(A,g,I){this.data[0]=mapIn(A),this.data[1]=mapIn(g),this.data[2]=mapIn(I)}}class Vec3f16Attribute extends Vec3Attribute{constructor(){super("Vec3f16")}init(){this.data=new Uint16Array(0),this.initRange(0)}initRange(A){for(let g=A;g<this.data.length;g++)this.data[g]=mapIn(Number.NaN)}isInitialized(A){for(let g=0;g<A.length;g++)if(Number.isFinite(mapOut(A[g])))return!0;return!1}getValue(A){if(A>=this.data.length/this.stride)throw new Error("Invalid vertex index:"+A+". Num Vertices:"+this.data.length/3);const g=A*this.stride,I=this.data.subarray(g,g+this.stride);return new Vec3(mapOut(I[0]),mapOut(I[1]),mapOut(I[2]))}getValueRef(A){if(A>=this.data.length/this.stride)throw new Error("Invalid vertex index:"+A+". Num Vertices:"+this.data.length/3);const g=A*this.stride;return new Vec3f16Ref(this.data.subarray(g,g+3))}setValue(A,g){if(A>=this.data.length/this.stride)throw new Error("Invalid vertex index:"+A+". Num Vertices:"+this.data.length/3);const I=A*this.stride,C=this.data.subarray(I,I+this.stride);C[0]=mapIn(g.x),C[1]=mapIn(g.y),C[2]=mapIn(g.z)}getFaceVertexValue(A,g){const I=this.getFaceVertexValue_array(A,g);return new Vec3(mapOut(I[0]),mapOut(I[1]),mapOut(I[2]))}setFaceVertexValue(A,g,I){const C=new Uint16Array(3);C[0]=mapIn(I.x),C[1]=mapIn(I.y),C[2]=mapIn(I.z),this.setFaceVertexValue_array(A,g,C)}setSplitVertexValues(A,g,I){super.setSplitVertexValues(A,g,I.map((A=>mapIn(A))))}}class ColorAttribute extends Attribute{constructor(){super("Color",4),this.normalized=!1}getValue(A){if(A>=this.data.length/this.stride)throw new Error("Invalid vertex index:"+A+". Num Vertices:"+this.data.length/3);const g=A*this.stride;return new Color(this.data[g+0],this.data[g+1],this.data[g+2],this.data[g+3])}setValue(A,g){this.setValues(A,g.asArray())}getFaceVertexValue(A,g){const I=this.getFaceVertexValue_array(A,g);return new Color(I[0],I[1],I[2],I[3])}setFaceVertexValue(A,g,I){this.setFaceVertexValue_array(A,g,Float32Array.from(I.asArray()))}setSplitVertexValue(A,g,I){this.setSplitVertexValue_array(A,g,Float32Array.from(I.asArray()))}merge(A,g=new Xfo){const I=this.getCount(),C=A.getCount();this.setCount(I+C);for(let g=0;g<C;g++)this.setValue(I+g,A.getValue(g));this.splitValues=[...this.splitValues,...A.splitValues]}}Registry.register("ColorAttribute",ColorAttribute);const parse8BitPositionsArray=(A,g,I,C,e)=>{for(let i=A[0];i<A[1];i++){const A=new Vec3(C[3*i+0]/255,C[3*i+1]/255,C[3*i+2]/255);A.multiplyInPlace(I),A.addInPlace(g),e.setValue(i,A)}},parse16BitPositionsArray=(A,g,I,C,e)=>{for(let i=A[0];i<A[1];i++){const A=new Vec3(C[3*i+0]/65535,C[3*i+1]/65535,C[3*i+2]/65535);A.multiplyInPlace(I),A.addInPlace(g),e.setValue(i,A)}},parse8BitNormalsArray=(A,g,I,C,e)=>{I.isNull()&&I.set(1,1,1);for(let i=A[0];i<A[1];i++){const A=new Vec3(C[3*i+0]/255,C[3*i+1]/255,C[3*i+2]/255);A.multiplyInPlace(I),A.addInPlace(g),A.normalizeInPlace(),e.setValue(i,A)}},parse8BitTextureCoordsArray=(A,g,I,C,e)=>{for(let i=A[0];i<A[1];i++){const A=new Vec2(C[2*i+0]/255,C[2*i+1]/255);A.multiplyInPlace(I),A.addInPlace(g),e.setValue(i,A)}},parse16BitTextureCoordsArray=(A,g,I,C,e)=>{for(let i=A[0];i<A[1];i++){const A=new Vec2(C[2*i+0]/65535,C[2*i+1]/65535);A.multiplyInPlace(I),A.addInPlace(g),e.setValue(i,A)}};class BaseGeom extends ParameterOwner{boundingBox=new Box3;boundingBoxDirty=!0;__metaData=new Map;#t=0;__vertexAttributes=new Map;debugColor=new Color(1,0,0,1);constructor(){super(),this.addVertexAttribute("positions",new Vec3f16Attribute)}getPath(){return null==this.ownerItem?[this.name]:this.ownerItem instanceof Parameter?[...this.ownerItem.getPath(),"value"]:[...this.ownerItem.getPath(),this.name]}clear(){this.setNumVertices(0)}addVertexAttribute(A,g){g.setCount(this.#t),this.__vertexAttributes.set(A,g)}hasVertexAttribute(A){return this.__vertexAttributes.has(A)}getVertexAttribute(A){return this.__vertexAttributes.get(A)}getVertexAttributes(){const A={};for(const[g,I]of this.__vertexAttributes.entries())A[g]=I;return A}get positions(){return this.__vertexAttributes.get("positions")}numVertices(){return this.#t}getNumVertices(){return this.#t}setNumVertices(A){this.#t=A,this.__vertexAttributes.forEach((A=>A.setCount(this.#t))),this.setBoundingBoxDirty()}getBoundingBox(){return this.boundingBoxDirty&&this.updateBoundingBox(),this.boundingBox}setBoundingBoxDirty(){this.boundingBoxDirty=!0,this.emit("boundingBoxChanged")}updateBoundingBox(){const A=this.positions,g=new Box3;if(A){const I=A.getCount();for(let C=0;C<I;C++)g.addPoint(A.getValue(C))}this.boundingBox=g,this.boundingBoxDirty=!1}merge(A,g=new Xfo){const I=this.getNumVertices(),C=A.getNumVertices();for(const[I,C]of this.__vertexAttributes){const e=A.getVertexAttribute(I);e&&("positions"==I?C.merge(e,g):"normals"==I&&C.merge(e,new Xfo(new Vec3,g.ori)))}this.setNumVertices(I+C),this.updateBoundingBox()}genBuffers(A){const g={};for(const[A,I]of this.__vertexAttributes)g[A]=I.genBuffer();return{numVertices:this.numVertices(),attrBuffers:g}}freeBuffers(){}loadBaseGeomBinary(A,g){this.name=A.loadStr();const I=A.loadUInt8();this.debugColor=A.loadRGBFloat32Color();const C=A.loadUInt32();let e,i;this.boundingBox.set(A.loadFloat32Vec3(),A.loadFloat32Vec3()),2&I&&(e=this.getVertexAttribute("normals"),e||(e=new Vec3f8Attribute,this.addVertexAttribute("normals",e))),4&I&&(i=this.getVertexAttribute("texCoords"),i||(i=new Vec2f16Attribute,this.addVertexAttribute("texCoords",i)));const t=A.loadUInt32();if(0==t){this.positions.data=A.loadUInt16Array(3*C),e&&(e.data=A.loadInt8Array(3*C)),i&&(i.data=A.loadUInt16Array(2*C)),this.setNumVertices(C)}else if(1==t){this.addVertexAttribute("positions",new Vec3Attribute),this.setNumVertices(C);const I=this.positions;{const e=this.boundingBox;if(g.versions["zea-engine"].compare([3,9,1])>=0){const g=A.loadUInt16Array(3*C,!1);parse16BitPositionsArray([0,C],e.p0,e.diagonal(),g,I)}else{const g=A.loadUInt8Array(3*C,!1);parse8BitPositionsArray([0,C],e.p0,e.diagonal(),g,I)}}if(e){const g=new Box3(A.loadFloat32Vec3(),A.loadFloat32Vec3()),I=A.loadUInt8Array(3*C,!1);parse8BitNormalsArray([0,C],g.p0,g.diagonal(),I,e),e.loadSplitValues(A)}if(i){const I=new Box2(A.loadFloat32Vec2(),A.loadFloat32Vec2());if(g.versions["zea-engine"].compare([3,9,1])>=0){const g=A.loadUInt16Array(2*C,!1);parse16BitTextureCoordsArray([0,C],I.p0,I.diagonal(),g,i)}else{const g=A.loadUInt8Array(2*C,!1);parse8BitTextureCoordsArray([0,C],I.p0,I.diagonal(),g,i)}i.loadSplitValues(A)}}else{this.addVertexAttribute("positions",new Vec3Attribute),this.setNumVertices(C);const I=this.positions,o=[];let a,B=0;for(let g=0;g<t;g++){const g=A.loadUInt32(),I={range:[B,B+g],bbox:new Box3(A.loadFloat32Vec3(),A.loadFloat32Vec3()),normalsRange:new Box3,texCoordsRange:new Box2};e&&I.normalsRange.set(A.loadFloat32Vec3(),A.loadFloat32Vec3()),i&&I.texCoordsRange.set(A.loadFloat32Vec2(),A.loadFloat32Vec2()),o.push(I),B+=g}a=g.versions["zea-engine"].compare([3,9,1])>=0?A.loadUInt16Array(3*C,!1):A.loadUInt8Array(3*C,!1);let s=null,l=null;e&&(s=A.loadUInt8Array(3*C,!1)),i&&(l=A.loadUInt8Array(2*C,!1));for(let A=0;A<t;A++){{const C=o[A].bbox;g.versions["zea-engine"].compare([3,9,1])>=0?parse16BitPositionsArray(o[A].range,C.p0,C.diagonal(),a,I):parse8BitPositionsArray(o[A].range,C.p0,C.diagonal(),a,I)}if(s){const g=o[A].normalsRange;parse8BitNormalsArray(o[A].range,g.p0,g.diagonal(),s,e)}if(l){const I=o[A].texCoordsRange;g.versions["zea-engine"].compare([3,9,1])>=0?parse16BitTextureCoordsArray([0,C],I.p0,I.diagonal(),l,i):parse8BitTextureCoordsArray(o[A].range,I.p0,I.diagonal(),l,i)}}e&&e.loadSplitValues(A),i&&i.loadSplitValues(A)}this.boundingBoxDirty=!1}toJSON(A){const g=super.toJSON(A);A&&A.skipTopology||(g.numVertices=this.#t||0);const I={};for(const[g,C]of this.__vertexAttributes.entries())A&&"skipAttributes"in A&&A.skipAttributes.includes(g)||(I[g]=C.toJSON(A));return g.vertexAttributes=I,g}fromJSON(A,g){this.clear(),super.fromJSON(A,g),this.setNumVertices(A.numVertices);for(const g in A.vertexAttributes){let I=this.__vertexAttributes.get(g);const C=A.vertexAttributes[g];if(!I||I.dataTypeName!=C.dataType){switch(C.dataType){case"Vec2":I=new Vec2Attribute;break;case"Vec2f16":I=new Vec2f16Attribute;break;case"Vec3":I=new Vec3Attribute;break;case"Vec3f16":I=new Vec3f16Attribute;break;case"Vec3f8":I=new Vec3f8Attribute;break;case"Color":I=new ColorAttribute;break;default:throw"Invalid Vectex Type: "+I.dataTypeName}I.setCount(this.#t),this.addVertexAttribute(g,I)}I?I.fromJSON(C):console.warn("attr undefined, cannot execute fromJSON()")}this.emit("geomDataTopologyChanged")}toString(){return JSON.stringify(this.toJSON(),null,2)}}class Vec4Attribute extends Attribute{constructor(){super("Vec4",4),this.normalized=!1}getValue(A){if(A>=this.data.length/this.stride)throw new Error("Invalid vertex index:"+A+". Num Vertices:"+this.data.length/3);const g=A*this.stride;return new Vec4(this.data[g+0],this.data[g+1],this.data[g+2],this.data[g+3])}setValue(A,g){this.setValues(A,g.asArray())}getFaceVertexValue(A,g){const I=this.getFaceVertexValue_array(A,g);return new Vec4(I[0],I[1],I[2],I[3])}setFaceVertexValue(A,g,I){this.setFaceVertexValue_array(A,g,Float32Array.from(I.asArray()))}setSplitVertexValue(A,g,I){this.setSplitVertexValue_array(A,g,Float32Array.from(I.asArray()))}merge(A,g=new Xfo){const I=this.getCount(),C=A.getCount();this.setCount(I+C);for(let g=0;g<C;g++)this.setValue(I+g,A.getValue(g));this.splitValues=[...this.splitValues,...A.splitValues]}}Registry.register("Vec4Attribute",Vec4Attribute);class Points extends BaseGeom{constructor(){super()}clear(){this.setNumVertices(0),this.emit("geomDataTopologyChanged")}readBinary(A,g){super.loadBaseGeomBinary(A,g),this.emit("geomDataChanged")}}Registry.register("Points",Points);class ImageArrayParameter extends Parameter{constructor(A="",g=[]){super(A,g,"BaseImage[]")}toJSON(A){const g={type:this.getClassName(),name:this.name,images:this.value.map((A=>({imageType:A?.getClassName(),value:A?.toJSON()})))};return g}fromJSON(A,g){if(A.images){const I=A.images;this.value=I.map((A=>{const I=Registry.constructClass(A.imageType);return I&&A.value&&I.fromJSON(A.value,g),I}))}}clone(){return new ImageArrayParameter(this.name,this.value)}}Registry.register("ImageArrayParameter",ImageArrayParameter);class FatPoints extends Points{sprites=new ImageArrayParameter("Sprites");constructor(){super(),this.addVertexAttribute("sizes",new Attribute("Float32",1,1)),this.addVertexAttribute("colors",new ColorAttribute),this.addVertexAttribute("spriteIndices",new Attribute("Float32",1,-1)),this.addParameter(this.sprites)}}Registry.register("FatPoints",FatPoints);class Lines extends BaseGeom{__indices;constructor(){super(),this.__indices=new Uint32Array}clear(){this.setNumSegments(0),this.setNumVertices(0),this.emit("geomDataTopologyChanged")}getIndices(){return this.__indices}getNumSegments(){return this.__indices.length/2}getNumLineSegments(){return this.__indices.length/2}setNumSegments(A){if(A>this.getNumSegments()){const g=new Uint32Array(2*A);g.set(this.__indices),this.__indices=g}else this.__indices=this.__indices.slice(0,2*A)}setSegmentVertexIndices(A,g,I){if(A>=this.__indices.length/2)throw new Error("Invalid line index:"+A+". Num Segments:"+this.__indices.length/2);this.__indices[2*A+0]=g,this.__indices[2*A+1]=I}getSegmentVertexIndex(A,g){return A<this.getNumSegments()?this.__indices[2*A+g]:-1}merge(A,g=new Xfo){const I=this.getNumVertices();super.merge(A,g);const C=A.__indices,e=new Uint32Array(this.__indices.length+C.length);e.set(this.__indices,0),e.set(C.map((A=>A+I)),this.__indices.length),this.__indices=e}genBuffers(A){const g=super.genBuffers();let I;return I=g.numVertices<Math.pow(2,8)?new Uint8Array(this.__indices):g.numVertices<Math.pow(2,16)?new Uint16Array(this.__indices):this.__indices,g.indices=I,g}readBinary(A,g){super.loadBaseGeomBinary(A,g),this.setNumSegments(A.loadUInt32());const I=A.loadUInt8();1==I?this.__indices=A.loadUInt8Array():2==I?this.__indices=A.loadUInt16Array():4==I&&(this.__indices=A.loadUInt32Array()),this.emit("geomDataChanged")}toJSON(A){const g=super.toJSON(A);return A&&A.skipTopology||(g.indices=Array.from(this.__indices)),g}fromJSON(A,g){super.fromJSON(A,g),A.indices&&(this.__indices=Uint32Array.from(A.indices))}}Registry.register("Lines",Lines);class Mesh extends BaseGeom{faceCounts;faceVertexIndices;__logTopologyWarnings;__edgeAttributes;__faceAttributes;numEdges;edgeVerts;edgeAngles;edgeVecs;edgeFaces;faceEdges;vertexEdges;constructor(){super(),this.edgeFaces=[],this.faceEdges=[[]],this.faceCounts=[],this.faceVertexIndices=new Uint32Array,this.__logTopologyWarnings=!1,this.__edgeAttributes=new Map,this.__faceAttributes=new Map,this.numEdges=0,this.edgeVerts=[],this.vertexEdges=[],this.edgeAngles=new Float32Array,this.edgeVecs=[]}clear(){super.clear(),this.edgeVerts=[],this.vertexEdges=[],this.numEdges=0,this.edgeAngles=new Float32Array,this.emit("geomDataTopologyChanged")}addVertexAttribute(A,g){super.addVertexAttribute(A,g),g.setMesh(this)}getFaceCounts(){return this.faceCounts}getNumFaces(){return 0==this.faceCounts.length?0:this.faceCounts.reduce(((A,g)=>A+g))}getNumTriangles(){let A=0,g=1;for(const I of this.faceCounts)A+=I*g,g++;return A}setFaceCounts(A){let g=0,I=3;for(const C of A)g+=C*I,I++;if(0==this.getNumFaces())this.faceVertexIndices=new Uint32Array(g);else{const C=new Uint32Array(g);let e=0,i=0;g=0,I=3,A.forEach(((A,g)=>{const t=e+Math.min(A,this.faceCounts[g])*I;C.set(this.faceVertexIndices.slice(e,t),i),e+=this.faceCounts[g]*I,i+=A*I,I++})),this.faceVertexIndices=C}this.faceCounts=A}getNumFaceVertices(){let A=0;return this.faceCounts.forEach(((g,I)=>{A+=g*(I+3)})),A}getFaceVertexCount(A){let g=0,I=0;return this.faceCounts.some(((C,e)=>(g+=C,g>A&&(I=e+3,!0)))),I}getFaceVertexOffset(A){let g=0,I=0;return this.faceCounts.some(((C,e)=>g+C>A?(I+=(A-g)*(e+3),!0):(g+=C,I+=C*(e+3),!1))),I}setFaceVertexIndices(A,g){const I=this.getFaceVertexCount(A);if(g.length!=I)throw new Error(`Invalid indices for face:${A} vertexIndices:${g}. Expected ${I} indices`);const C=this.getFaceVertexOffset(A);this.faceVertexIndices.set(g,C)}addFace(A){const g=[...this.faceCounts];if(g.length<=A.length-3){for(let I=g.length;I<A.length-3;I++)g[I]=0;g[A.length-3]=1}else g[A.length-3]++;this.setFaceCounts(g);let I=0,C=0;return this.faceCounts.some(((g,e)=>e+3==A.length?(I+=g-1,C+=(g-1)*(e+3),!0):(I+=g,C+=g*(e+3),!1))),this.faceVertexIndices.set(A,C),I}getFaceVertexIndices(A){const g=[],I=this.getFaceVertexOffset(A),C=this.getFaceVertexCount(A);for(let A=0;A<C;A++)g.push(this.faceVertexIndices[I+A]);return g}getFaceVertexIndex(A,g){const I=this.getFaceVertexOffset(A);return this.faceVertexIndices[I+g]}addFaceAttribute(A,g){return g.setCount(this.getNumFaces()),this.__faceAttributes.set(A,g),g}hasFaceAttribute(A){return this.__faceAttributes.has(A)}getFaceAttribute(A){return this.__faceAttributes.get(A)}addEdgeAttribute(A,g){g.setCount(this.numEdges),this.__edgeAttributes.set(A,g)}hasEdgeAttribute(A){return this.__edgeAttributes.has(A)}getEdgeAttribute(A){return this.__edgeAttributes.get(A)}genTopologyInfo(){let A={};this.vertexEdges=[],this.edgeFaces=[],this.edgeVerts=[],this.faceEdges=[],this.numEdges=0;const g=this.positions,I=(I,C)=>{let e=I,i=C;if(i<e){const A=e;e=i,i=A}const t=e+">"+i;if(t in A)return A[t];const o=g.getValue(e),a=g.getValue(i).subtract(o),B={edgeIndex:this.edgeFaces.length/2,edgeVec:a};return A[t]=B,this.edgeFaces.push(-1),this.edgeFaces.push(-1),this.edgeVerts.push(e),this.edgeVerts.push(i),this.numEdges++,B},C=(A,g,C)=>{const e=I(A,g).edgeIndex;if(g<A){const A=2*e+0;this.__logTopologyWarnings&&-1!=this.edgeFaces[A]&&console.warn("Edge poly 0 already set. Mesh is non-manifold."),this.edgeFaces[A]=C}else{const A=2*e+1;this.__logTopologyWarnings&&-1!=this.edgeFaces[A]&&console.warn("Edge poly 1 already set. Mesh is non-manifold."),this.edgeFaces[A]=C}C in this.faceEdges||(this.faceEdges[C]=[]),this.faceEdges[C].push(e),null==this.vertexEdges[A]&&(this.vertexEdges[A]=new Set),null==this.vertexEdges[g]&&(this.vertexEdges[g]=new Set),this.vertexEdges[A].add(e),this.vertexEdges[g].add(e)},e=this.getNumFaces();for(let A=0;A<e;A++){const g=this.getFaceVertexIndices(A);for(let I=0;I<g.length;I++){C(g[I],g[(I+1)%g.length],A)}}}computeFaceNormals(){const A=this.positions,g=new Vec3f8Attribute;this.addFaceAttribute("normals",g);const I=this.getNumFaces();for(let C=0;C<I;C++){const I=this.getFaceVertexIndices(C),e=A.getValue(I[0]);let i=A.getValue(I[1]);const t=new Vec3;for(let g=2;g<I.length;g++){const C=A.getValue(I[g]),o=i.subtract(e),a=C.subtract(e);t.addInPlace(o.cross(a).normalize()),i=C}t.lengthSquared()<Number.EPSILON||g.setValue(C,t.normalize())}}calculateEdgeAngles(){0==this.vertexEdges.length&&this.genTopologyInfo(),this.computeFaceNormals();const A=this.positions,g=this.getFaceAttribute("normals");this.edgeVecs=[],this.edgeAngles=new Float32Array(this.numEdges);for(let I=0;I<this.edgeFaces.length;I+=2){const C=this.edgeVerts[I],e=this.edgeVerts[I+1],i=A.getValue(e).subtract(A.getValue(C));i.normalizeInPlace(),this.edgeVecs.push(i);const t=this.edgeFaces[I],o=this.edgeFaces[I+1];if(-1==t||-1==o){this.edgeAngles[I/2]=2*Math.PI;continue}const a=g.getValue(t),B=g.getValue(o);this.edgeAngles[I/2]=a.angleTo(B)}}computeVertexNormals(A=1){this.calculateEdgeAngles();const g=this.getFaceAttribute("normals"),I=new Vec3f8Attribute;this.addVertexAttribute("normals",I);const C=(A,g)=>{I.setValue(A,g)},e=(A,g)=>{let I,C;const e=this.faceEdges[A];for(const A of e)(this.edgeVerts[2*A]==g||this.edgeVerts[2*A+1]==g)&&(I?C=this.edgeVecs[A]:I=this.edgeVecs[A]);return[I,C]};for(let t=0;t<this.vertexEdges.length;t++){if(null==this.vertexEdges[t])continue;const o=this.vertexEdges[t],a=[],B=A=>{let g=!1;for(const I of a)if(g=I.includes(A),g)break;g||a.push([A])};for(const g of o){const I=this.edgeFaces[2*g],C=this.edgeFaces[2*g+1];if(-1!=I&&-1!=C&&this.edgeAngles[g]<A){let A=-1,g=-1;for(let e=0;e<a.length;e++)-1==A&&a[e].includes(I)&&(A=e),-1==g&&a[e].includes(C)&&(g=e);-1==A&&-1==g?a.push([I,C]):-1!=A&&-1!=g?A!=g&&(a[A]=a[A].concat(a[g]),a.splice(g,1)):(-1==A&&a[g].push(I),-1==g&&a[A].push(C))}else-1!=I&&B(I),-1!=C&&B(C)}a.sort(((A,g)=>A.length<g.length?1:A.length>g.length?-1:0));let s=!0;for(const A of a){const o=new Vec3;for(const I of A){const A=e(I,t);let C;A[0]&&A[1]?(C=A[0].angleTo(A[1]),o.addInPlace((i=I,g.getValue(i)).scale(C))):console.warn("variable weight is undefined because faceEdges[0] or faceEdges[1] is undefined")}o.normalizeInPlace(),s?(C(t,o),s=!1):I.setSplitVertexValues(t,A,o.asArray())}}var i;return I}computeHardEdgesIndices(A=1){0==this.edgeVerts.length&&this.calculateEdgeAngles();const g=[],I=A=>{g.push(this.edgeVerts[A]),g.push(this.edgeVerts[A+1])};for(let g=0;g<this.edgeAngles.length;g++)this.edgeAngles[g]>A&&I(2*g);return Uint32Array.from(g)}merge(A,g=new Xfo){const I=this.getNumVertices();super.merge(A,g);const C=A.faceVertexIndices,e=new Uint32Array(this.faceVertexIndices.length+C.length),i=A.getFaceCounts();let t=0,o=0,a=0;const B=Math.max(this.faceCounts.length,i.length);for(let A=0;A<B;A++){if(this.faceCounts.length>A){const g=this.faceCounts[A]*(A+3);e.set(this.faceVertexIndices.slice(t,t+g),a),t+=g,a+=g}if(i.length>A){const g=i[A]*(A+3);e.set(C.slice(o,o+g).map((A=>A+I)),a),o+=g,a+=g,this.faceCounts.length==A&&(this.faceCounts[A]=0),this.faceCounts[A]+=i[A]}}this.faceVertexIndices=e}genBuffers(A){const g={};let I=0;for(const[,A]of this.__vertexAttributes){const C=A.getSplits();for(const A in C){A in g||(g[A]={});const e=C[A];for(const C in e){const e=parseInt(C);e in g[A]||(g[A][e]=I,I++)}}}const C=this.positions.getCount(),e=C+I;let i;A&&0==A.includeIndices||(i=this.generateTriangulatedIndices(e,C,g));const t={};for(const[A,C]of this.__vertexAttributes){let e;e=0==I?C.asArray():C.generateSplitValues(g,I);const i=C.stride,o=e.length/i;t[A]={values:e,count:o,dimension:i,normalized:"normals"==A,dataType:C.getDataTypeName()}}return{numVertices:this.numVertices(),numRenderVerts:e,indices:i,attrBuffers:t}}computeNumTriangles(){let A=3,g=0;for(const I of this.faceCounts)g+=I*(A-2),A++;return g}generateTriangulatedIndices(A,g,I){const C=this.computeNumTriangles();let e;e=A<Math.pow(2,8)?new Uint8Array(3*C):A<Math.pow(2,16)?new Uint16Array(3*C):new Uint32Array(3*C);let i=0;const t=function(A,C){A in I&&C in I[A]&&(A=g+I[A][C]),e[i]=A,i++},o=this.getNumFaces();for(let A=0;A<o;A++){const g=this.getFaceVertexIndices(A);for(let I=0;I<g.length;I++)I>=3&&(t(g[0],A),t(g[I-1],A)),t(g[I],A)}return e}readBinary(A,g){super.loadBaseGeomBinary(A,g),this.setFaceCounts(Array.from(A.loadUInt32Array()));const I=this.getNumFaces(),C=A.loadUInt8Array(I,!1),e=A.loadSInt32Vec2(),i=A.loadUInt8();let t;if(1==i)t=A.loadUInt8Array(void 0,!1);else if(2==i)t=A.loadUInt16Array(void 0,!1);else{if(4!=i)throw Error("faceVertexIndexDeltas undefined");t=A.loadUInt32Array(void 0,!1)}let o=3,a=0;const B=this.faceCounts.map(((A,g)=>{const I=a;return a+=A*o,o++,I}));let s=0,l=0;const n=[];for(let A=0;A<I;A++){const g=C[A],I=B[g],i=g+3;n[A]=I;for(let g=0;g<i;g++){const C=I+g,i=t[s+g]+e.x;if(0==A)this.faceVertexIndices[C]=i;else{let I=n[A-1];I+=g<l?g:l-1,this.faceVertexIndices[C]=this.faceVertexIndices[I]+i}}s+=i,B[g]+=i,l=i}this.hasVertexAttribute("normals")||this.computeVertexNormals(),this.emit("geomDataChanged")}toJSON(A){const g=super.toJSON(A);return A&&A.skipTopology||(g.faceCounts=Array.from(this.faceCounts),g.faceVertexIndices=Array.from(this.faceVertexIndices)),g}fromJSON(A,g){super.fromJSON(A,g),A.faceCounts&&(this.faceCounts=A.faceCounts),A.faceVertexIndices&&(this.faceVertexIndices=Uint32Array.from(A.faceVertexIndices))}}Registry.register("Mesh",Mesh);class BaseProxy extends RefCounted{name;buffers;boundingBox;numVertices=0;libraryIndex=-1;constructor(A){super(),A&&(this.name=A.name,this.buffers=A.geomBuffers,this.boundingBox=new Box3,this.boundingBox.p0.fromJSON(A.bbox.p0),this.boundingBox.p1.fromJSON(A.bbox.p1),this.numVertices=this.buffers.numVertices)}get positions(){return this.buffers.attrBuffers.positions}getNumVertices(){return this.numVertices}getBoundingBox(){return this.boundingBox}genBuffers(){return this.buffers}freeBuffers(){for(const A in this.buffers.attrBuffers){this.buffers.attrBuffers[A].values=null}this.buffers.indices&&(this.buffers.indices=null)}toJSON(A){return{geomBuffers:this.buffers}}fromJSON(A,g){this.buffers=A.geomBuffers}copyFrom(A,g){this.buffers={...A.buffers},this.name=A.name,this.boundingBox=A.boundingBox,this.numVertices=this.buffers.numVertices}}class PointsProxy extends BaseProxy{constructor(A){super(A)}}class LinesProxy extends BaseProxy{numLineSegments=0;constructor(A){super(A),A&&(this.numLineSegments=this.buffers.indices.length/2)}getNumLineSegments(){return this.numLineSegments}clone(A){const g=new LinesProxy;return g.copyFrom(this,A),g}copyFrom(A,g){super.copyFrom(A,g),this.numLineSegments=this.buffers.indices.length/2}}class MeshProxy extends BaseProxy{numTriangles=0;constructor(A){super(A),A&&(this.numTriangles=this.buffers.indices.length/3)}getNumTriangles(){return this.numTriangles}clone(A){const g=new MeshProxy;return g.copyFrom(this,A),g}copyFrom(A,g){super.copyFrom(A,g),this.numTriangles=this.buffers.indices.length/3}}class SubGeom extends ParameterOwner{}Registry.register("SubGeom",SubGeom);class CompoundGeom extends BaseProxy{materials=[];subGeoms=[];counts;materialGroupsDirty=!1;constructor(A,g){if(super(A),A){this.counts=A.geomBuffers.counts;const I=g.getMaterials();A.geomBuffers.materialLibraryIndices.forEach(((A,g)=>{this.materials[g]=I[A]})),delete A.geomBuffers.materialLibraryIndices,this.buffers.materials=this.materials}}getNumSubGeoms(){return this.buffers.numSubGeoms}getNumFaces(){return this.buffers.subGeomCounts.TRIANGLES.length}getNumEdges(){return this.buffers.subGeomCounts.LINES.length}getNumTriangles(){return this.counts.TRIANGLES/3}getNumLineSegments(){return this.counts.LINES/2}getNumPoints(){return this.counts.POINTS}getSubGeomMaterial(A){const g=this.buffers.subGeomMaterialIndices[A]-1;if(-1!=g)return this.materials[g]}setSubGeomMaterial(A,g){if(g){let I=this.materials.indexOf(g);if(-1==I&&(I=this.materials.length,this.materials[I]=g),0==this.buffers.subGeomMaterialIndices.length&&(this.buffers.subGeomMaterialIndices=new Uint8Array(this.buffers.numSubGeoms)),this.materials.length>255&&this.buffers.subGeomMaterialIndices instanceof Uint8Array){const A=new Uint16Array(this.buffers.numSubGeoms);for(let g=0;g<this.buffers.subGeomMaterialIndices.length;g++)A[g]=this.buffers.subGeomMaterialIndices[g];this.buffers.subGeomMaterialIndices=A}this.buffers.subGeomMaterialIndices[A]=I+1}else this.buffers.subGeomMaterialIndices&&this.buffers.subGeomMaterialIndices[A]&&(this.buffers.subGeomMaterialIndices[A]=0);this.materialGroupsDirty=!0,this.emit("materialsChanged")}assignSubGeomMaterial(A,g){this.setSubGeomMaterial(A,g)}clearMaterials(){this.materials.splice(0,this.materials.length),this.buffers.subGeomMaterialIndices=this.buffers.subGeomMaterialIndices.map((()=>0)),this.emit("materialsChanged")}calcMaterialGroups(){const A={};if(0==this.buffers.subGeomMaterialIndices.length){this.buffers.materialSubGeoms={};let A=0;for(let g in this.buffers.counts){const I=this.buffers.counts[g];I>0&&(this.buffers.materialSubGeoms[g]=[{materialId:-1,offset:A,count:I}]),A+=I}return}let g=0,I=-99,C=null;for(let e=0;e<this.buffers.numSubGeoms;e++){let i,t=0;e<this.buffers.subGeomCounts.TRIANGLES.length?(A.TRIANGLES||(A.TRIANGLES=[]),i="TRIANGLES"):e<this.buffers.subGeomCounts.TRIANGLES.length+this.buffers.subGeomCounts.LINES.length?(t=this.buffers.subGeomCounts.TRIANGLES.length,i="LINES",A.LINES||(A.LINES=[])):(t=this.buffers.subGeomCounts.TRIANGLES.length+this.buffers.subGeomCounts.LINES.length,i="POINTS",A.POINTS||(A.POINTS=[]));const o=this.buffers.subGeomMaterialIndices[e];if(I!=o){for(I=o,C={materialId:o-1,offset:g,count:0};e<this.buffers.numSubGeoms&&I==this.buffers.subGeomMaterialIndices[e];e++){if(e-t==this.buffers.subGeomCounts[i].length){I=-99;break}C.count+=this.buffers.subGeomCounts[i][e-t]}g+=C.count,A[i].push(C),e--}}this.buffers.materialSubGeoms=A}genBuffers(){return this.materialGroupsDirty&&this.calcMaterialGroups(),this.buffers}toJSON(A){const g={materialPaths:this.materials.map((A=>A.path))};return g}fromJSON(A,g){if(this.counts=A.geomBuffers.counts,A.materialPaths&&g){const I=A.materialPaths;this.materials=[],I.forEach(((A,I)=>{g.resolvePath(A,(A=>{A instanceof Material&&(this.materials[I]=A)}),(()=>{}))}))}this.buffers.materials=this.materials,this.materialGroupsDirty=!0}loadMetadata(A,g){A.loadUInt32Array().forEach(((I,C)=>{const e=new SubGeom;A.seek(I),e.readBinary(A,g),this.subGeoms[C]=e}))}clone(A){const g=new CompoundGeom;return g.copyFrom(this,A),g}copyFrom(A,g){super.copyFrom(A,g),this.buffers={...A.buffers},this.buffers.subGeomMaterialIndices&&(this.buffers.subGeomMaterialIndices=new Uint8Array(this.buffers.numSubGeoms),this.buffers.subGeomMaterialIndices.set(A.buffers.subGeomMaterialIndices)),this.counts=A.counts,this.materials=A.materials,this.buffers.materials=this.materials,this.subGeoms=A.subGeoms,this.libraryIndex=A.libraryIndex}}Registry.register("CompoundGeom",CompoundGeom);class ProceduralPoints extends Points{dirtyTopology;dirtyVertices;topologyParams;constructor(){super(),this.dirtyTopology=!0,this.dirtyVertices=!0,this.topologyParams=[]}parameterValueChanged(A){this.setBoundingBoxDirty(),this.topologyParams.includes(A.param.getName())?(this.dirtyTopology=!0,this.emit("geomDataTopologyChanged")):(this.dirtyVertices=!0,this.setBoundingBoxDirty(),this.emit("geomDataChanged")),super.parameterValueChanged(A)}update(){this.dirtyTopology?(this.rebuild(),this.dirtyTopology=!1,this.dirtyVertices=!1,this.rebuild()):this.dirtyVertices&&(this.dirtyVertices=!1,this.resize(),this.dirtyVertices=!1)}getBoundingBox(){return this.update(),super.getBoundingBox()}getNumVertices(){return this.update(),super.getNumVertices()}genBuffers(A){return this.update(),super.genBuffers(A)}toJSON(A){A||(A={}),A.skipTopology=!0,A.skipAttributes=["positions","normals","texCoords"];const g=super.toJSON(A);return A.skipTopology=!1,A.skipAttributes=[],g}}class ProceduralLines extends Lines{dirtyTopology;dirtyVertices;topologyParams;constructor(){super(),this.dirtyTopology=!0,this.dirtyVertices=!0,this.topologyParams=[]}parameterValueChanged(A){this.setBoundingBoxDirty(),this.topologyParams.includes(A.param.getName())?(this.dirtyTopology=!0,this.emit("geomDataTopologyChanged")):(this.dirtyVertices=!0,this.setBoundingBoxDirty(),this.emit("geomDataChanged")),super.parameterValueChanged(A)}update(){this.dirtyTopology?(this.rebuild(),this.dirtyTopology=!1,this.dirtyVertices=!1,this.rebuild()):this.dirtyVertices&&(this.resize(),this.dirtyVertices=!1,this.resize())}getBoundingBox(){return this.update(),super.getBoundingBox()}getNumVertices(){return this.update(),super.getNumVertices()}genBuffers(A){return this.update(),super.genBuffers()}toJSON(A){A||(A={}),A.skipTopology=!0,A.skipAttributes=["positions","normals","texCoords"];const g=super.toJSON(A);return A.skipTopology=!1,A.skipAttributes=[],g}}class ProceduralMesh extends Mesh{dirtyTopology;dirtyVertices;topologyParams;constructor(){super(),this.dirtyTopology=!0,this.dirtyVertices=!0,this.topologyParams=[]}parameterValueChanged(A){this.setBoundingBoxDirty(),this.topologyParams.includes(A.param.getName())?(this.dirtyTopology=!0,this.emit("geomDataTopologyChanged")):(this.dirtyVertices=!0,this.setBoundingBoxDirty(),this.emit("geomDataChanged")),super.parameterValueChanged(A)}update(){this.dirtyTopology?(this.vertexEdges=[],this.dirtyTopology=!1,this.dirtyVertices=!1,this.rebuild()):this.dirtyVertices&&(this.dirtyVertices=!1,this.resize())}getBoundingBox(){return this.update(),super.getBoundingBox()}getNumVertices(){return this.update(),super.getNumVertices()}computeVertexNormals(A=1){return this.update(),super.computeVertexNormals(A)}computeHardEdgesIndices(A=1){return this.update(),super.computeHardEdgesIndices(A)}genBuffers(A){return this.update(),super.genBuffers(A)}toJSON(A){A||(A={}),A.skipTopology=!0,A.skipAttributes=["positions","normals","texCoords"];const g=super.toJSON(A);return A.skipTopology=!1,A.skipAttributes=[],g}}class PointGrid extends ProceduralPoints{sizeXParam=new NumberParameter("SizeX",1,[0,Number.MAX_VALUE]);sizeYParam=new NumberParameter("SizeY",1,[0,Number.MAX_VALUE]);divisionsXParam=new NumberParameter("XDivisions",1,[0,Number.MAX_VALUE],1);divisionsYParam=new NumberParameter("YDivisions",1,[0,Number.MAX_VALUE],1);constructor(A=1,g=1,I=1,C=1){if(super(),this.topologyParams=[],isNaN(A)||isNaN(g)||isNaN(I)||isNaN(C))throw new Error("Invalid geom args");this.addParameter(this.sizeXParam),this.addParameter(this.sizeYParam),this.addParameter(this.divisionsXParam),this.addParameter(this.divisionsYParam),this.sizeXParam.value=A,this.sizeYParam.value=g,this.divisionsXParam.value=I,this.divisionsYParam.value=C,this.topologyParams.push("XDivisions"),this.topologyParams.push("YDivisions")}rebuild(){const A=Math.round(this.divisionsXParam.value),g=Math.round(this.divisionsYParam.value);this.setNumVertices(A*g);const I=this.getVertexAttribute("texCoords");if(I)for(let C=0;C<g;C++){const e=C/(g-1);for(let g=0;g<A;g++){const i=g/(A-1);I.setValue(C*A+g,new Vec2(i,e))}}this.resize()}resize(){const A=Math.round(this.divisionsXParam.value),g=Math.round(this.divisionsYParam.value),I=this.sizeXParam.value,C=this.sizeYParam.value,e=this.positions;if(e)for(let i=0;i<g;i++){const t=(i/(g-1)-.5)*C;for(let g=0;g<A;g++){const C=(g/(A-1)-.5)*I;e.setValue(i*A+g,new Vec3(C,t,0))}}}}Registry.register("PointGrid",PointGrid);class Rect extends ProceduralLines{sizeXParam=new NumberParameter("SizeX",1,[0,Number.MAX_VALUE]);sizeYParam=new NumberParameter("SizeY",1,[0,Number.MAX_VALUE]);constructor(A=1,g=1){if(super(),isNaN(A)||isNaN(g))throw new Error("Invalid geom args");this.addParameter(this.sizeXParam),this.addParameter(this.sizeYParam),this.sizeXParam.value=A,this.sizeYParam.value=g,this.rebuild()}rebuild(){this.setNumVertices(4),this.setNumSegments(4),this.setSegmentVertexIndices(0,0,1),this.setSegmentVertexIndices(1,1,2),this.setSegmentVertexIndices(2,2,3),this.setSegmentVertexIndices(3,3,0),this.resize(!1)}resize(){const A=this.sizeXParam.value,g=this.sizeYParam.value,I=this.positions;I&&(I.setValue(0,new Vec3(-.5*A,-.5*g,0)),I.setValue(1,new Vec3(.5*A,-.5*g,0)),I.setValue(2,new Vec3(.5*A,.5*g,0)),I.setValue(3,new Vec3(-.5*A,.5*g,0)))}}Registry.register("Rect",Rect);class Circle extends ProceduralLines{angleParam=new AngleParameter("Angle",2*Math.PI,[0,2*Math.PI]);sidesParam=new NumberParameter("Sides",6,[0,Number.MAX_VALUE],1);radiusParam=new NumberParameter("Radius",1,[0,Number.MAX_VALUE]);constructor(A=1,g=32,I=2*Math.PI){if(super(),this.topologyParams=[],isNaN(A)||isNaN(g))throw new Error("Invalid geom args");this.radiusParam.value=A,this.sidesParam.value=g,this.angleParam.value=I,this.addParameter(this.radiusParam),this.addParameter(this.angleParam),this.addParameter(this.sidesParam),this.topologyParams.push("Angle"),this.topologyParams.push("Sides")}rebuild(){const A=this.angleParam.value,g=Math.round(this.sidesParam.value),I=A<2*Math.PI?g+1:g;this.setNumVertices(I),this.setNumSegments(g);for(let A=0;A<g;A++)this.setSegmentVertexIndices(A,A,(A+1)%I);this.resize()}resize(){const A=this.radiusParam.value,g=this.angleParam.value,I=g/(g<2*Math.PI?this.positions.count-1:this.positions.count);for(let g=0;g<this.positions.count;g++)this.positions.setValue(g,new Vec3(Math.cos(I*g)*A,Math.sin(I*g)*A,0))}}Registry.register("Circle",Circle);class Cross extends ProceduralLines{sizeParam=new NumberParameter("Size",1,[0,Number.MAX_VALUE]);constructor(A=1){if(super(),isNaN(A))throw new Error("Invalid geom args");this.addParameter(this.sizeParam),this.sizeParam.value=A}rebuild(){this.setNumVertices(6),this.setNumSegments(3),this.setSegmentVertexIndices(0,0,1),this.setSegmentVertexIndices(1,2,3),this.setSegmentVertexIndices(2,4,5),this.resize()}resize(){const A=this.sizeParam.value,g=this.positions;g&&(g.setValue(0,new Vec3(-.5*A,0,0)),g.setValue(1,new Vec3(.5*A,0,0)),g.setValue(2,new Vec3(0,.5*A,0)),g.setValue(3,new Vec3(0,-.5*A,0)),g.setValue(4,new Vec3(0,0,.5*A)),g.setValue(5,new Vec3(0,0,-.5*A)))}}Registry.register("Cross",Cross);class LinesCuboid extends ProceduralLines{baseZAtZeroParam=new BooleanParameter("BaseZAtZero",!1);sizeXParam=new NumberParameter("SizeX",1,[0,Number.MAX_VALUE]);sizeYParam=new NumberParameter("SizeY",1,[0,Number.MAX_VALUE]);sizeZParam=new NumberParameter("SizeZ",1,[0,Number.MAX_VALUE]);constructor(A=1,g=1,I=1,C=!1){super(),this.addParameter(this.sizeXParam),this.addParameter(this.sizeYParam),this.addParameter(this.sizeZParam),this.addParameter(this.baseZAtZeroParam),this.sizeXParam.value=A,this.sizeYParam.value=g,this.sizeZParam.value=I,this.baseZAtZeroParam.value=C}rebuild(){this.setNumVertices(8),this.setNumSegments(12),this.setSegmentVertexIndices(0,0,1),this.setSegmentVertexIndices(1,1,2),this.setSegmentVertexIndices(2,2,3),this.setSegmentVertexIndices(3,3,0),this.setSegmentVertexIndices(4,4,5),this.setSegmentVertexIndices(5,5,6),this.setSegmentVertexIndices(6,6,7),this.setSegmentVertexIndices(7,7,4),this.setSegmentVertexIndices(8,0,4),this.setSegmentVertexIndices(9,1,5),this.setSegmentVertexIndices(10,2,6),this.setSegmentVertexIndices(11,3,7),this.resize()}resize(){const A=this.sizeXParam.value,g=this.sizeYParam.value,I=this.sizeZParam.value,C=this.baseZAtZeroParam.value,e=this.positions;if(e){let i=.5;C&&(i=1),e.setValue(0,new Vec3(.5*A,-.5*g,i*I)),e.setValue(1,new Vec3(.5*A,.5*g,i*I)),e.setValue(2,new Vec3(-.5*A,.5*g,i*I)),e.setValue(3,new Vec3(-.5*A,-.5*g,i*I)),i=-.5,C&&(i=0),e.setValue(4,new Vec3(.5*A,-.5*g,i*I)),e.setValue(5,new Vec3(.5*A,.5*g,i*I)),e.setValue(6,new Vec3(-.5*A,.5*g,i*I)),e.setValue(7,new Vec3(-.5*A,-.5*g,i*I))}}}Registry.register("LinesCuboid",LinesCuboid);class LinesSphere extends ProceduralLines{radiusParam=new NumberParameter("Radius",6,[0,Number.MAX_VALUE]);sidesParam=new NumberParameter("Sides",12,[0,Number.MAX_VALUE],1);constructor(A=1,g=32){if(super(),this.topologyParams=[],isNaN(A)||isNaN(g))throw new Error("Invalid geom args");this.addParameter(this.radiusParam),this.addParameter(this.sidesParam),this.radiusParam.value=A,this.sidesParam.value=g,this.topologyParams.push("Sides")}rebuild(){const A=this.sidesParam.value;this.setNumVertices(3*A),this.setNumSegments(3*A);const g=g=>{for(let I=0;I<A;I++)this.setSegmentVertexIndices(I+g,I+g,(I+1)%A+g)};g(0),g(A),g(2*A),this.resize()}resize(){const A=this.radiusParam.value,g=this.sidesParam.value,I=2*Math.PI/g,C=this.positions;if(C)for(let e=0;e<g;e++)C.setValue(e,new Vec3(Math.cos(I*e)*A,Math.sin(I*e)*A,0)),C.setValue(e+g,new Vec3(Math.cos(I*e)*A,0,Math.sin(I*e)*A)),C.setValue(e+2*g,new Vec3(0,Math.cos(I*e)*A,Math.sin(I*e)*A))}}Registry.register("LinesSphere",LinesSphere);class LinesCylinder extends ProceduralLines{baseZAtZeroParam=new BooleanParameter("BaseZAtZero",!0);heightParam=new NumberParameter("Height",1,[0,Number.MAX_VALUE]);radiusParam=new NumberParameter("Radius",6,[0,Number.MAX_VALUE]);loopsParam=new NumberParameter("Loops",6,[0,Number.MAX_VALUE],1);sidesParam=new NumberParameter("Sides",12,[0,Number.MAX_VALUE],1);constructor(A=.5,g=1,I=32,C=2,e=!1){if(super(),this.topologyParams=[],isNaN(A)||isNaN(g)||isNaN(I)||isNaN(C))throw new Error("Invalid geom args");this.radiusParam.value=A,this.heightParam.value=g,this.sidesParam.value=I>=3?I:3,this.loopsParam.value=C>=2?C:2,this.baseZAtZeroParam.value=e,this.addParameter(this.radiusParam),this.addParameter(this.heightParam),this.addParameter(this.sidesParam),this.addParameter(this.loopsParam),this.addParameter(this.baseZAtZeroParam),this.topologyParams.push("Sides"),this.topologyParams.push("Loops")}rebuild(){const A=Math.round(this.sidesParam.value),g=Math.round(this.loopsParam.value);let I=A*g,C=A*g+g;this.setNumVertices(I),this.setNumSegments(C);let e=0;for(let I=0;I<g;I++){for(let g=0;g<A;g++){const C=A*I+g,i=A*I+(g+1)%A;this.setSegmentVertexIndices(e++,C,i)}if(I>0){{const g=(I-1)*A,C=I*A;this.setSegmentVertexIndices(e++,g,C)}{const g=(I-1)*A+Math.floor(.5*A),C=I*A+Math.floor(.5*A);this.setSegmentVertexIndices(e++,g,C)}}}this.resize()}resize(){const A=Math.round(this.sidesParam.value),g=Math.round(this.loopsParam.value),I=this.radiusParam.value,C=this.heightParam.value;let e=0,i=.5;this.baseZAtZeroParam.value&&(i=0);const t=this.positions;if(t)for(let o=0;o<g;o++){const a=o/(g-1)*C-C*i;for(let g=0;g<A;g++){const C=g/A*2*Math.PI;t.setValue(e,new Vec3(Math.sin(C)*I,Math.cos(C)*I,a)),e++}}this.dirtyTopology=!1,this.dirtyVertices=!1}}Registry.register("LinesCylinder",LinesCylinder);class Grid extends ProceduralLines{sizeXParam=new NumberParameter("SizeX",1,[0,Number.MAX_VALUE]);sizeYParam=new NumberParameter("SizeY",1,[0,Number.MAX_VALUE]);divisionsXParam=new NumberParameter("XDivisions",1,[0,Number.MAX_VALUE],1);divisionsYParam=new NumberParameter("YDivisions",1,[0,Number.MAX_VALUE],1);skipCenterLinesParam=new BooleanParameter("SkipCenterLines",!1);constructor(A=1,g=1,I=10,C=10,e=!1){if(super(),this.topologyParams=[],isNaN(A)||isNaN(g)||isNaN(I)||isNaN(C))throw new Error("Invalid geom args");this.addParameter(this.sizeXParam),this.addParameter(this.sizeYParam),this.addParameter(this.divisionsXParam),this.addParameter(this.divisionsYParam),this.addParameter(this.skipCenterLinesParam),this.sizeXParam.value=A,this.sizeYParam.value=g,this.divisionsXParam.value=I,this.divisionsYParam.value=C,this.skipCenterLinesParam.value=e,this.topologyParams.push("XDivisions"),this.topologyParams.push("YDivisions"),this.topologyParams.push("SkipCenterLines")}rebuild(){const A=this.divisionsXParam.value,g=this.divisionsYParam.value,I=this.skipCenterLinesParam.value&&A%2==0&&g%2==0;this.setNumVertices(2*(A+g+2-(I?1:0))),this.setNumSegments(A+g+2-(I?1:0));let C=0;for(let g=0;g<=A;g++){if(I&&g==A/2)continue;const e=2*C,i=2*C+1;this.setSegmentVertexIndices(C,e,i),C++}for(let e=0;e<=g;e++){if(I&&e==A/2)continue;const g=2*C,i=2*C+1;this.setSegmentVertexIndices(C,g,i),C++}this.resize()}resize(){const A=this.positions,g=this.divisionsXParam.value,I=this.divisionsYParam.value,C=this.sizeXParam.value,e=this.sizeYParam.value,i=this.skipCenterLinesParam.value&&g%2==0&&I%2==0;let t=0;for(let I=0;I<=g;I++){if(i&&I==g/2)continue;const o=2*t,a=2*t+1,B=(I/g-.5)*C;A&&(A.setValue(o,new Vec3(B,-.5*e,0)),A.setValue(a,new Vec3(B,.5*e,0))),t++}for(let o=0;o<=I;o++){if(i&&o==g/2)continue;const a=2*t,B=2*t+1,s=(o/I-.5)*e;A&&(A.setValue(a,new Vec3(-.5*C,s,0)),A.setValue(B,new Vec3(.5*C,s,0))),t++}}}Registry.register("Grid",Grid);class Cone extends ProceduralMesh{capParam=new BooleanParameter("Cap",!0);detailParam=new NumberParameter("Detail",6,[0,Number.MAX_VALUE],1);heightParam=new NumberParameter("Height",1,[0,Number.MAX_VALUE]);radiusParam=new NumberParameter("Radius",6,[0,Number.MAX_VALUE]);constructor(A=.5,g=1,I=32,C=!0,e=!0,i=!0){if(super(),this.topologyParams=[],isNaN(A)||isNaN(g)||isNaN(I))throw new Error("Invalid geom args");this.radiusParam.value=A,this.heightParam.value=g,this.detailParam.value=I,this.capParam.value=C,this.addParameter(this.radiusParam),this.addParameter(this.heightParam),this.addParameter(this.detailParam),this.addParameter(this.capParam),e&&this.addVertexAttribute("normals",new Vec3f8Attribute),i&&this.addVertexAttribute("texCoords",new Vec2f16Attribute),this.topologyParams.push("Detail"),this.topologyParams.push("Cap")}rebuild(){const A=Math.round(this.detailParam.value),g=this.radiusParam.value,I=this.heightParam.value,C=this.capParam.value;let e=A+1;C&&(e+=1),this.setNumVertices(e);const i=A,t=A+1,o=this.positions;if(o){o.setValue(i,new Vec3(0,0,I));for(let I=0;I<A;I++){const C=-I/A*2*Math.PI;o.setValue(I,new Vec3(g*Math.cos(C),g*Math.sin(C),0))}C&&o.setValue(t,new Vec3(0,0,0))}this.setFaceCounts([A+(C?A:0)]);for(let g=0;g<A;g++){const I=(g+1)%A;this.setFaceVertexIndices(g,[I,g,i])}if(C)for(let g=0;g<A;g++){const I=(g+1)%A;this.setFaceVertexIndices(A+g,[g,I,t])}const a=this.getVertexAttribute("texCoords");if(a){let g=0;for(let I=0;I<A;I++)"setFaceVertexValue"in a&&(a.setFaceVertexValue(g,0,new Vec2((I+1)/A,0)),a.setFaceVertexValue(g,1,new Vec2(I/A,0)),a.setFaceVertexValue(g,2,new Vec2((I+.5)/A,1)));if(C)for(let I=0;I<A;I++)a.setFaceVertexValue(g,0,new Vec2(I/A,0)),a.setFaceVertexValue(g,1,new Vec2((I+1)/A,0)),a.setFaceVertexValue(g,2,new Vec2((I+.5)/A,1)),g++}this.resize()}resize(){const A=Math.round(this.detailParam.value),g=this.radiusParam.value,I=this.heightParam.value,C=A,e=A+1,i=this.positions;if(i){i.setValue(C,new Vec3(0,0,I));for(let I=0;I<A;I++){const C=-I/A*2*Math.PI;i.setValue(I,new Vec3(g*Math.cos(C),g*Math.sin(C),0))}this.capParam.value&&i.setValue(e,new Vec3(0,0,0))}this.dirtyTopology=!1,this.dirtyVertices=!1;this.getVertexAttribute("normals")&&this.computeVertexNormals()}}Registry.register("Cone",Cone);class Cuboid extends ProceduralMesh{baseZAtZeroParam=new BooleanParameter("BaseZAtZero",!1);sizeXParam=new NumberParameter("SizeX",1,[0,Number.MAX_VALUE]);sizeYParam=new NumberParameter("SizeY",1,[0,Number.MAX_VALUE]);sizeZParam=new NumberParameter("SizeZ",1,[0,Number.MAX_VALUE]);constructor(A=1,g=1,I=1,C=!1){if(super(),isNaN(A)||isNaN(g)||isNaN(I))throw new Error("Invalid geom args");this.addParameter(this.sizeXParam),this.addParameter(this.sizeYParam),this.addParameter(this.sizeZParam),this.addParameter(this.baseZAtZeroParam),this.sizeXParam.value=A,this.sizeYParam.value=g,this.sizeZParam.value=I,this.baseZAtZeroParam.value=C,this.setFaceCounts([0,6]),this.setFaceVertexIndices(0,[0,1,2,3]),this.setFaceVertexIndices(1,[7,6,5,4]),this.setFaceVertexIndices(2,[1,0,4,5]),this.setFaceVertexIndices(3,[3,2,6,7]),this.setFaceVertexIndices(4,[0,3,7,4]),this.setFaceVertexIndices(5,[2,1,5,6]),this.setNumVertices(8),this.addVertexAttribute("normals",new Vec3f8Attribute)}rebuild(){const A=this.getVertexAttribute("normals");if(A)for(let g=0;g<6;g++){let I;switch(g){case 0:I=new Vec3(0,0,1);break;case 1:I=new Vec3(0,0,-1);break;case 2:I=new Vec3(1,0,0);break;case 3:I=new Vec3(-1,0,0);break;case 4:I=new Vec3(0,-1,0);break;default:I=new Vec3(0,1,0)}A.setFaceVertexValue(g,0,I),A.setFaceVertexValue(g,1,I),A.setFaceVertexValue(g,2,I),A.setFaceVertexValue(g,3,I)}this.resize()}resize(){const A=this.sizeXParam.value,g=this.sizeYParam.value,I=this.sizeZParam.value,C=this.baseZAtZeroParam.value;let e=.5;const i=this.positions;C&&(e=1),i&&(i.setValue(0,new Vec3(.5*A,-.5*g,e*I)),i.setValue(1,new Vec3(.5*A,.5*g,e*I)),i.setValue(2,new Vec3(-.5*A,.5*g,e*I)),i.setValue(3,new Vec3(-.5*A,-.5*g,e*I)),e=-.5,C&&(e=0),i.setValue(4,new Vec3(.5*A,-.5*g,e*I)),i.setValue(5,new Vec3(.5*A,.5*g,e*I)),i.setValue(6,new Vec3(-.5*A,.5*g,e*I)),i.setValue(7,new Vec3(-.5*A,-.5*g,e*I)))}}Registry.register("Cuboid",Cuboid);class Cylinder extends ProceduralMesh{baseZAtZeroParam=new BooleanParameter("BaseZAtZero",!0);capsParam=new BooleanParameter("Caps",!0);heightParam=new NumberParameter("Height",1,[0,Number.MAX_VALUE]);radiusParam=new NumberParameter("Radius",6,[0,Number.MAX_VALUE]);loopsParam=new NumberParameter("Loops",6,[0,Number.MAX_VALUE],1);sidesParam=new NumberParameter("Sides",12,[0,Number.MAX_VALUE],1);constructor(A=.5,g=1,I=32,C=2,e=!0,i=!1,t=!0,o=!0){if(super(),this.topologyParams=[],isNaN(A)||isNaN(g)||isNaN(I)||isNaN(C))throw new Error("Invalid geom args");this.radiusParam.value=A,this.heightParam.value=g,this.sidesParam.value=I>=3?I:3,this.loopsParam.value=C>=2?C:2,this.capsParam.value=e,this.baseZAtZeroParam.value=i,this.addParameter(this.radiusParam),this.addParameter(this.heightParam),this.addParameter(this.sidesParam),this.addParameter(this.loopsParam),this.addParameter(this.capsParam),this.addParameter(this.baseZAtZeroParam),t&&this.addVertexAttribute("normals",new Vec2f16Attribute),o&&this.addVertexAttribute("texCoords",new Vec3f8Attribute),this.topologyParams.push("Sides"),this.topologyParams.push("Loops"),this.topologyParams.push("Caps")}rebuild(){const A=Math.round(this.sidesParam.value),g=Math.round(this.loopsParam.value),I=this.capsParam.value;let C=A*g;I&&(C+=2),this.setNumVertices(C),I?this.setFaceCounts([2*A,A]):this.setFaceCounts([0,A]);let e=0;if(I){for(let g=0;g<A;g++){const I=C-1,i=g,t=(g+1)%A;this.setFaceVertexIndices(e++,[I,i,t])}for(let I=0;I<A;I++){const i=A*(g-1)+I,t=C-2,o=A*(g-1)+(I+1)%A;this.setFaceVertexIndices(e++,[i,t,o])}}for(let I=0;I<g-1;I++)for(let g=0;g<A;g++){const C=A*I+(g+1)%A,i=A*I+g,t=A*(I+1)+g,o=A*(I+1)+(g+1)%A;this.setFaceVertexIndices(e++,[C,i,t,o])}const i=this.getVertexAttribute("normals");if(i){if(e=0,I){const g=new Vec3(0,0,-1);for(let I=0;I<A;I++)i.setFaceVertexValue(e,0,g),i.setFaceVertexValue(e,1,g),i.setFaceVertexValue(e,2,g),e++;g.set(0,0,1);for(let I=0;I<A;I++)i.setFaceVertexValue(e,0,g),i.setFaceVertexValue(e,1,g),i.setFaceVertexValue(e,2,g),e++}for(let I=0;I<g-1;I++)for(let g=0;g<A;g++){let I=g/A*2*Math.PI;const C=new Vec3(Math.sin(I),Math.cos(I),0);i.setFaceVertexValue(e,0,C),i.setFaceVertexValue(e,1,C),I=(g+1)/A*2*Math.PI;const t=new Vec3(Math.sin(I),Math.cos(I),0);i.setFaceVertexValue(e,2,t),i.setFaceVertexValue(e,3,t),e++}}const t=this.getVertexAttribute("texCoords");if(t){if(e=0,I){for(let g=0;g<A;g++)t.setFaceVertexValue(e,0,new Vec2(g/A,0)),t.setFaceVertexValue(e,1,new Vec2((g+1)/A,0)),t.setFaceVertexValue(e,2,new Vec2((g+.5)/A,1)),e++;for(let g=0;g<A;g++)t.setFaceVertexValue(e,0,new Vec2(g/A,0)),t.setFaceVertexValue(e,1,new Vec2((g+1)/A,0)),t.setFaceVertexValue(e,2,new Vec2((g+.5)/A,1)),e++}for(let g=0;g<A;g++)t.setFaceVertexValue(e,0,new Vec2((g+1)/A,0)),t.setFaceVertexValue(e,2,new Vec2((g+1)/A,1)),t.setFaceVertexValue(e,1,new Vec2(g/A,0)),t.setFaceVertexValue(e,3,new Vec2(g/A,1)),e++}this.resize()}resize(){const A=Math.round(this.sidesParam.value),g=Math.round(this.loopsParam.value),I=this.radiusParam.value,C=this.heightParam.value,e=this.capsParam.value,i=this.baseZAtZeroParam.value;let t=A*g;e&&(t+=2);let o=0,a=.5;i&&(a=0);const B=this.positions;if(B){for(let e=0;e<g;e++){const i=e/(g-1)*C-C*a;for(let g=0;g<A;g++){const C=g/A*2*Math.PI;B.setValue(o,new Vec3(Math.sin(C)*I,Math.cos(C)*I,i)),o++}}e&&(B.setValue(t-1,new Vec3(0,0,C*(i?0:-.5))),B.setValue(t-2,new Vec3(0,0,C*(i?1:.5))))}this.dirtyTopology=!1,this.dirtyVertices=!1;this.getVertexAttribute("normals")&&this.computeVertexNormals()}}Registry.register("Cylinder",Cylinder);class Disc extends ProceduralMesh{radiusParam=new NumberParameter("Radius",6,[0,Number.MAX_VALUE]);sidesParam=new NumberParameter("Sides",12,[0,Number.MAX_VALUE],1);constructor(A=.5,g=32){if(super(),this.topologyParams=[],isNaN(A)||isNaN(g))throw new Error("Invalid geom args");this.addParameter(this.radiusParam),this.addParameter(this.sidesParam),this.radiusParam.value=A,this.sidesParam.value=g,this.addVertexAttribute("texCoords",new Vec2f16Attribute),this.addVertexAttribute("normals",new Vec3f8Attribute),this.topologyParams.push("Sides")}rebuild(){const A=Math.round(this.sidesParam.value);this.setNumVertices(A+1),this.setFaceCounts([A]);const g=this.positions;g&&g.setValue(0,new Vec3(0,0,0));for(let g=0;g<A;g++){const I=g%A+1,C=(g+1)%A+1;this.setFaceVertexIndices(g,[0,I,C])}const I=this.getVertexAttribute("normals");if(I){const g=new Vec3(0,0,1);I.setValue(0,g);for(let C=0;C<A;C++)I.setValue(C+1,g)}const C=this.getVertexAttribute("texCoords");if(C){C.setValue(0,new Vec2(.5,.5));for(let g=0;g<A;g++){const I=g/A*2*Math.PI;C.setValue(g+1,new Vec2(.5*Math.sin(I)+.5,.5*Math.cos(I)+.5))}}this.resize()}resize(){const A=Math.round(this.sidesParam.value),g=this.radiusParam.value,I=this.positions;if(I)for(let C=0;C<A;C++){const e=C/A*2*Math.PI;I.setValue(C+1,new Vec3(Math.sin(e)*g,Math.cos(e)*g,0))}}}Registry.register("Disc",Disc);class Plane extends ProceduralMesh{sizeXParam=new NumberParameter("SizeX",1,[0,Number.MAX_VALUE]);sizeYParam=new NumberParameter("SizeY",1,[0,Number.MAX_VALUE]);detailXParam=new NumberParameter("DetailX",1,[0,Number.MAX_VALUE],1);detailYParam=new NumberParameter("DetailY",1,[0,Number.MAX_VALUE],1);constructor(A=1,g=1,I=1,C=1,e=!0,i=!0){if(super(),this.topologyParams=[],isNaN(A)||isNaN(g)||isNaN(I)||isNaN(C))throw new Error("Invalid geom args");this.addParameter(this.sizeXParam),this.addParameter(this.sizeYParam),this.addParameter(this.detailXParam),this.addParameter(this.detailYParam),this.sizeXParam.value=A,this.sizeYParam.value=g,this.detailXParam.value=I,this.detailYParam.value=C,e&&this.addVertexAttribute("normals",new Vec3f8Attribute),i&&this.addVertexAttribute("texCoords",new Vec2f16Attribute),this.topologyParams.push("DetailX"),this.topologyParams.push("DetailY")}rebuild(){const A=this.detailXParam.value,g=this.detailYParam.value;this.setNumVertices((A+1)*(g+1)),this.setFaceCounts([0,A*g]);let I=0;for(let C=0;C<g;C++)for(let g=0;g<A;g++){const e=(A+1)*(C+1)+g,i=(A+1)*C+g,t=(A+1)*C+(g+1),o=(A+1)*(C+1)+(g+1);this.setFaceVertexIndices(I,[e,i,t,o]),I+=1}let C=0;const e=this.getVertexAttribute("normals");if(e)for(let I=0;I<=g;I++)for(let g=0;g<=A;g++)e.setValue(C,new Vec3(0,0,1)),C++;C=0;const i=this.getVertexAttribute("texCoords");if(i)for(let I=0;I<=g;I++){const e=I/g;for(let g=0;g<=A;g++){const I=g/A;i.setValue(C,new Vec2(I,e)),C++}}this.resize()}resize(){const A=this.sizeXParam.value,g=this.sizeYParam.value,I=this.detailXParam.value,C=this.detailYParam.value,e=this.positions;if(!e)return;let i=0;for(let t=0;t<=C;t++){const o=(t/C-.5)*g;for(let g=0;g<=I;g++){const C=(g/I-.5)*A;e.setValue(i,new Vec3(C,o,0)),i++}}}}Registry.register("Plane",Plane);class Sphere extends ProceduralMesh{radiusParam=new NumberParameter("Radius",1,[0,Number.MAX_VALUE]);sidesParam=new NumberParameter("Sides",12,[0,Number.MAX_VALUE],1);loopsParam=new NumberParameter("Loops",6,[0,Number.MAX_VALUE],1);constructor(A=1,g=12,I=12,C=!0,e=!0){if(super(),isNaN(A)||isNaN(g)||isNaN(I))throw new Error("Invalid geom args");this.addParameter(this.radiusParam),this.addParameter(this.sidesParam),this.addParameter(this.loopsParam),this.radiusParam.value=A,this.sidesParam.value=g,this.loopsParam.value=I,C&&this.addVertexAttribute("normals",new Vec3f8Attribute),e&&this.addVertexAttribute("texCoords",new Vec2f16Attribute),this.topologyParams.push("Sides"),this.topologyParams.push("Loops")}rebuild(){const A=this.radiusParam.value,g=Math.round(this.sidesParam.value),I=Math.round(this.loopsParam.value),C=2+g*I,e=2*g,i=g*I;this.setNumVertices(C),this.setFaceCounts([e,i]);const t=this.positions,o=this.getVertexAttribute("normals"),a=new Vec3(0,0,1);let B=0;if(!t)return;t.setValue(B,new Vec3(0,0,A)),o&&o.setValue(B,new Vec3(0,0,1)),B++;for(let C=0;C<I;C++){const e=(C+1)/(I+1)*Math.PI;for(let I=0;I<g;I++){const C=-I/g*2*Math.PI;a.set(Math.sin(e)*Math.cos(C),Math.sin(e)*Math.sin(C),Math.cos(e)),t.setValue(B,a.scale(A)),o&&o.setValue(B,a),B++}}t.setValue(B,new Vec3(0,0,-A)),o&&o.setValue(B,new Vec3(0,0,-1)),B++;const s=this.getVertexAttribute("texCoords");let l=0;for(let A=0;A<g;A++){const C=0,e=(A+1)%g+1,i=A+1;if(this.setFaceVertexIndices(l,[C,e,i]),s){const C=new Vec2(.5,0),e=new Vec2((A+1)/(g-1),1/(I+1)),i=new Vec2(A/(g-1),1/(I+1));s.setFaceVertexValue(l,0,C),s.setFaceVertexValue(l,1,e),s.setFaceVertexValue(l,2,i)}l++}for(let A=0;A<g;A++){const e=C-1,i=g*(I-1)+(A+1)%g+1,t=g*(I-1)+A+1;if(this.setFaceVertexIndices(l,[e,t,i]),s){const C=new Vec2(.5,1),e=new Vec2((A+1)/(g-1),1-1/(I+1)),i=new Vec2(A/(g-1),1-1/(I+1));s.setFaceVertexValue(l,0,C),s.setFaceVertexValue(l,1,e),s.setFaceVertexValue(l,2,i)}l++}for(let A=0;A<I-1;A++)for(let C=0;C<g;C++){const e=g*A+C+1,i=g*A+(C+1)%g+1,t=g*(A+1)+(C+1)%g+1,o=g*(A+1)+C+1;this.setFaceVertexIndices(l,[e,i,t,o]),s&&(s.setFaceVertexValue(l,0,new Vec2(C/g,(A+1)/I)),s.setFaceVertexValue(l,1,new Vec2((C+1)/g,(A+1)/I)),s.setFaceVertexValue(l,2,new Vec2((C+1)/g,(A+2)/I)),s.setFaceVertexValue(l,3,new Vec2(C/g,(A+2)/I))),l++}}resize(){const A=this.radiusParam.value,g=Math.round(this.sidesParam.value),I=Math.round(this.loopsParam.value);if(!g||!I)return void console.warn("resize() failed");const C=this.positions,e=this.getVertexAttribute("normals");let i=0;const t=new Vec3(0,0,1);C.setValue(i,new Vec3(0,0,A)),e&&e.setValue(i,new Vec3(0,0,1)),i++;for(let o=0;o<I;o++){const a=(o+1)/(I+1)*Math.PI;for(let I=0;I<g;I++){const o=-I/g*2*Math.PI;t.set(Math.sin(a)*Math.cos(o),Math.sin(a)*Math.sin(o),Math.cos(a)),C.setValue(i,t.scale(A)),e&&e.setValue(i,t),i++}}C.setValue(i,new Vec3(0,0,-A)),e&&e.setValue(i,new Vec3(0,0,-1)),i++}}Registry.register("Sphere",Sphere);class Torus extends ProceduralMesh{innerRadiusParam=new NumberParameter("InnerRadius",1,[0,Number.MAX_VALUE]);outerRadiusParam=new NumberParameter("OuterRadius",1,[0,Number.MAX_VALUE]);arcAngleParam=new AngleParameter("ArcAngle",2*Math.PI,[0,2*Math.PI]);detailParam=new NumberParameter("Detail",12,[0,Number.MAX_VALUE],1);constructor(A=.5,g=3,I=32,C=2*Math.PI){if(super(),isNaN(A)||isNaN(g)||isNaN(I))throw new Error("Invalid geom args");this.addParameter(this.innerRadiusParam),this.addParameter(this.outerRadiusParam),this.addParameter(this.arcAngleParam),this.addParameter(this.detailParam),this.innerRadiusParam.value=A,this.outerRadiusParam.value=g,this.detailParam.value=I>=3?I:33,this.arcAngleParam.value=C,this.addVertexAttribute("texCoords",new Vec2f16Attribute),this.addVertexAttribute("normals",new Vec3f8Attribute),this.topologyParams.push("Detail"),this.topologyParams.push("ArcAngle")}rebuild(){const A=this.arcAngleParam.value<2*Math.PI,g=Math.round(this.detailParam.value),I=g,C=2*g+(A?1:0),e=I*C;this.setNumVertices(e),this.setFaceCounts([0,I*C]);const i=this.getVertexAttribute("texCoords");if(i){let g=0;for(let e=0;e<(A?C-1:C);e++)for(let A=0;A<I;A++){const t=(e+1)%C,o=(A+1)%I,a=I*e+A,B=I*e+o,s=I*t+o,l=I*t+A;this.setFaceVertexIndices(g,[a,B,s,l]),i.setFaceVertexValue(g,0,new Vec2(e/C,A/C)),i.setFaceVertexValue(g,1,new Vec2(e/C,(A+1)/C)),i.setFaceVertexValue(g,2,new Vec2((e+1)/C,(A+1)/C)),i.setFaceVertexValue(g,3,new Vec2((e+1)/C,A/C)),g++}}this.resize()}resize(){const A=this.innerRadiusParam.value,g=this.outerRadiusParam.value,I=this.arcAngleParam.value,C=Math.round(this.detailParam.value),e=I<2*Math.PI,i=C,t=2*C+(e?1:0),o=this.positions,a=this.getVertexAttribute("normals");if(!o||!a)return;let B=0;for(let C=0;C<t;C++){const s=-C/(e?t-1:t)*I,l=Math.cos(s),n=Math.sin(s);for(let I=0;I<i;I++){const C=I/i*2*Math.PI,e=Math.sin(C),t=Math.cos(C),s=g+t*A;o.setValue(B,new Vec3(l*s,n*s,A*e)),a.setValue(B,new Vec3(l*t,n*t,e)),B++}}}}Registry.register("Torus",Torus);class GeometryParameter extends Parameter{listenerIDs={};constructor(A="",g=null){super(A,null,"Geometry"),g&&this.setValue(g)}emitBoundingBoxDirtied(A){this.emit("boundingBoxChanged",A)}setValue(A){if(!(null==A||A instanceof BaseGeom||A instanceof BaseProxy))throw new Error("value provided is not an instance of a 'BaseGeom' or 'BaseProxy' class. Check the source of this value");this.value!==A&&(this.value&&(this.value instanceof BaseGeom?this.value.setOwner(null):this.value instanceof BaseProxy&&this.value.removeRef(this),this.value.off("boundingBoxChanged",this.listenerIDs.boundingBoxChanged)),super.setValue(A),this.value instanceof BaseGeom?this.value.setOwner(this):this.value instanceof BaseProxy&&this.value.addRef(this),this.listenerIDs.boundingBoxChanged=this.value.on("boundingBoxChanged",(A=>{this.emitBoundingBoxDirtied(A)})))}loadValue(A){this.setValue(A)}toJSON(A){const g={type:this.getClassName(),name:this.name},I=this.value;return I instanceof BaseProxy?g.libraryIndex=I.libraryIndex:g.value=I?.toJSON(A),g}fromJSON(A,g){if(A.name&&(this.name=A.name),null!=A.libraryIndex){const I=g.assetItem.geomLibrary.getGeom(A.libraryIndex);this.setValue(I)}else if(null!=A.value)if(this.value&&this.value.getClassName()==A.value.type)this.value.fromJSON(A.value,g),this.emit("valueChanged");else{const I=Registry.constructClass(A.value.type);I.fromJSON(A.value,g),this.setValue(I)}}clone(){return new GeometryParameter(this.name,this.value)}}Registry.register("GeometryParameter",GeometryParameter);class DataImage extends BaseImage{__data=new Uint8Array(4);constructor(A){super(A),this.format="RGBA",this.type="UNSIGNED_BYTE",this.width=1,this.height=1}isLoaded(){return this.loaded}isStream(){return!1}setData(A,g,I){this.__data!=I&&(this.width=A,this.height=g,this.__data=I,this.loaded?this.emit("updated"):(this.loaded=!0,this.emit("loaded")))}getParams(){const A=super.getParams();return A.data=this.__data,A}}Registry.register("DataImage2D",DataImage),Registry.register("DataImage",DataImage);const imageDataLibrary$1={};class FileImage extends BaseImage{crossOrigin;url;__data=null;constructor(A,g="",I={}){super(A),this.type="UNSIGNED_BYTE",this.crossOrigin="anonymous",g&&""!=g&&this.load(g)}setCrossOrigin(A){this.crossOrigin=A}getDOMElement(){return this.__data}load(A,g="RGB"){return new Promise(((I,C)=>{if(!g){const I=A.lastIndexOf(".");if(-1!=I){".png"==A.substring(I).toLowerCase()&&(g="RGBA")}}this.format=g,this.loaded=!1;const e=()=>{this.url=A,this.width=this.__data.width,this.height=this.__data.height,this.loaded=!0,this.emit("loaded"),I()};A in imageDataLibrary$1?(this.__data=imageDataLibrary$1[A],this.__data.complete?e():(this.__data.addEventListener("load",e),this.__data.addEventListener("error",C))):(this.__data=new Image,this.__data.crossOrigin=this.crossOrigin,this.__data.src=A,this.__data.addEventListener("load",e),this.__data.addEventListener("error",C),imageDataLibrary$1[A]=this.__data)}))}setImageURL(A,g="RGB"){this.load(A,g)}getParams(){const A=super.getParams();return this.loaded&&(A.data=this.__data),A}readBinary(A,g){this.setName(A.loadStr());const I=A.loadStr();if("string"==typeof I&&""!=I){const A=g.url.substring(0,g.url.lastIndexOf("/"));this.load(A+"/"+I)}}}class FileImage2D extends FileImage{constructor(A,g={}){console.warn("FileImage2D is becoming deprecated in favor of simple FileImage"),super(A,g)}}Registry.register("FileImage2D",FileImage),Registry.register("FileImage",FileImage);class LDRImage extends FileImage{constructor(A,g,I){super(A,g,I)}}Registry.register("LDRImage",LDRImage);class HDRImage extends BaseImage{exposure=1;hdrTint=new Color(1,1,1,1);__data;constructor(A,g={}){let I;super(A),null!=A&&A.includes(".")&&(I=A,this.setName(A.substring(A.lastIndexOf("/")+1,A.lastIndexOf(".")))),this.type="HDR",I&&this.load(I)}__decodeData(A){return new Promise(((g,I)=>{const C=A.ldr,e=A.cdm,i=new Blob([C.buffer]),t=new Image;t.onload=()=>{this.width=t.width,this.height=t.height,this.__data={ldr:t,cdm:e},this.loaded?this.emit("updated"):(this.loaded=!0,this.emit("loaded")),g()},t.src=URL.createObjectURL(i)}))}load(A){return this.loaded=!1,new Promise(((g,I)=>{const C=A.lastIndexOf("/")>-1?A.substring(A.lastIndexOf("/")+1):"",e=C.substring(0,C.lastIndexOf("."));""==this.getName()&&this.setName(e),this.type="FLOAT",resourceLoader.loadFile("archive",A).then((A=>{if(!A.ldr||!A.cdm)for(const g in A)g.endsWith(".jpg")?(A.ldr=A[g],delete A[g]):g.endsWith(".bin")&&(A.cdm=A[g],delete A[g]);this.__decodeData(A).then((()=>{g()}))}),(A=>{this.emit("error",A),I(A)}))}))}isStream(){return!1}getParams(){const A=super.getParams();return this.loaded&&(A.data=this.__data,A.exposure=this.exposure),A}setHDRTint(A){this.hdrTint=A}getHDRTint(){return this.hdrTint}readBinary(A,g){this.setName(A.loadStr());let I=A.loadStr();"string"==typeof I&&""!=I&&this.load(I)}}Registry.register("HDRImage",HDRImage);class LDRVideo extends FileImage{videoElem=new HTMLVideoElement;muteParam=new BooleanParameter("Mute",!1);loopParam=new BooleanParameter("Loop",!0);spatializeAudioParam=new BooleanParameter("SpatializeAudio",!0);refDistanceParam=new NumberParameter("refDistance",2);maxDistanceParam=new NumberParameter("maxDistance",1e4);rolloffFactorParam=new NumberParameter("rolloffFactor",1);coneInnerAngleParam=new NumberParameter("coneInnerAngle",360);coneOuterAngleParam=new NumberParameter("coneOuterAngle",0);coneOuterGainParam=new NumberParameter("coneOuterGain",1);gainParam=new NumberParameter("Gain",2);constructor(A,g,I){super(A,g,I),this.format="RGB",this.type="UNSIGNED_BYTE",this.addParameter(this.muteParam),this.addParameter(this.loopParam),this.addParameter(this.spatializeAudioParam),this.addParameter(this.refDistanceParam),this.addParameter(this.maxDistanceParam),this.addParameter(this.rolloffFactorParam),this.addParameter(this.coneInnerAngleParam),this.addParameter(this.coneOuterAngleParam),this.addParameter(this.coneOuterGainParam),this.addParameter(this.gainParam).setRange([0,5])}getAudioSource(){return this.videoElem}load(A,g="RGB"){return new Promise(((g,I)=>{resourceLoader.incrementWorkload(1),this.videoElem.style.display="none",this.videoElem.preload="auto",this.videoElem.crossOrigin="anonymous",document.body.appendChild(this.videoElem),this.videoElem.addEventListener("loadedmetadata",(()=>{this.videoElem.muted=this.muteParam.value,this.muteParam.on("valueChanged",(()=>{this.videoElem.muted=this.muteParam.value})),this.videoElem.loop=this.loopParam.value,this.loopParam.on("valueChanged",(()=>{this.videoElem.loop=this.loopParam.value})),this.width=this.videoElem.videoHeight,this.height=this.videoElem.videoWidth,this.loaded=!0,resourceLoader.incrementWorkDone(1),this.emit("loaded"),g(C);let A=0;const I=()=>{if(this.videoElem.paused||this.videoElem.ended)return;const g=Math.floor(29.97*this.videoElem.currentTime);A!=g&&(this.emit("updated"),A=g),setTimeout(I,20)};I()}),!1),this.videoElem.src=A;const C=this.videoElem.play();void 0!==C&&C.then((A=>{console.log("Autoplay started!")})).catch((()=>{console.log("Autoplay was prevented.")}))}))}getParams(){const A=super.getParams();return this.loaded&&(A.data=this.videoElem),A}}function ByteStream(A){this.data=A,this.pos=0}function DataParser(A){this.stream=new ByteStream(A),this.output={}}function bitsToNum(A){return A.reduce((function(A,g){return 2*A+g}),0)}Registry.register("LDRVideo",LDRVideo),ByteStream.prototype.readByte=function(){return this.data[this.pos++]},ByteStream.prototype.peekByte=function(){return this.data[this.pos]},ByteStream.prototype.readBytes=function(A){for(var g=new Array(A),I=0;I<A;I++)g[I]=this.readByte();return g},ByteStream.prototype.peekBytes=function(A){for(var g=new Array(A),I=0;I<A;I++)g[I]=this.data[this.pos+I];return g},ByteStream.prototype.readString=function(A){for(var g="",I=0;I<A;I++)g+=String.fromCharCode(this.readByte());return g},ByteStream.prototype.readBitArray=function(){for(var A=[],g=this.readByte(),I=7;I>=0;I--)A.push(!!(g&1<<I));return A},ByteStream.prototype.readUnsigned=function(A){var g=this.readBytes(2);return A?(g[1]<<8)+g[0]:(g[0]<<8)+g[1]},DataParser.prototype.parse=function(A){return this.parseParts(this.output,A),this.output},DataParser.prototype.parseParts=function(A,g){for(var I=0;I<g.length;I++){var C=g[I];this.parsePart(A,C)}},DataParser.prototype.parsePart=function(A,g){var I,C=g.label;if(!g.requires||g.requires(this.stream,this.output,A))if(g.loop){for(var e=[];g.loop(this.stream);){var i={};this.parseParts(i,g.parts),e.push(i)}A[C]=e}else g.parts?(I={},this.parseParts(I,g.parts),A[C]=I):g.parser?(I=g.parser(this.stream,this.output,A),g.skip||(A[C]=I)):g.bits&&(A[C]=this.parseBits(g.bits))},DataParser.prototype.parseBits=function(A){var g={},I=this.stream.readBitArray();for(var C in A){var e=A[C];e.length?g[C]=bitsToNum(I.slice(e.index,e.index+e.length)):g[C]=I[e.index]}return g};var Parsers={readByte:function(){return function(A){return A.readByte()}},readBytes:function(A){return function(g){return g.readBytes(A)}},readString:function(A){return function(g){return g.readString(A)}},readUnsigned:function(A){return function(g){return g.readUnsigned(A)}},readArray:function(A,g){return function(I,C,e){for(var i=g(I,C,e),t=new Array(i),o=0;o<i;o++)t[o]=I.readBytes(A);return t}}},subBlocks={label:"blocks",parser:function(A){for(var g=[],I=A.readByte();0!==I;I=A.readByte())g=g.concat(A.readBytes(I));return g}},gce={label:"gce",requires:function(A){var g=A.peekBytes(2);return 33===g[0]&&249===g[1]},parts:[{label:"codes",parser:Parsers.readBytes(2),skip:!0},{label:"byteSize",parser:Parsers.readByte()},{label:"extras",bits:{future:{index:0,length:3},disposal:{index:3,length:3},userInput:{index:6},transparentColorGiven:{index:7}}},{label:"delay",parser:Parsers.readUnsigned(!0)},{label:"transparentColorIndex",parser:Parsers.readByte()},{label:"terminator",parser:Parsers.readByte(),skip:!0}]},image={label:"image",requires:function(A){return 44===A.peekByte()},parts:[{label:"code",parser:Parsers.readByte(),skip:!0},{label:"descriptor",parts:[{label:"left",parser:Parsers.readUnsigned(!0)},{label:"top",parser:Parsers.readUnsigned(!0)},{label:"width",parser:Parsers.readUnsigned(!0)},{label:"height",parser:Parsers.readUnsigned(!0)},{label:"lct",bits:{exists:{index:0},interlaced:{index:1},sort:{index:2},future:{index:3,length:2},size:{index:5,length:3}}}]},{label:"lct",requires:function(A,g,I){return I.descriptor.lct.exists},parser:Parsers.readArray(3,(function(A,g,I){return Math.pow(2,I.descriptor.lct.size+1)}))},{label:"data",parts:[{label:"minCodeSize",parser:Parsers.readByte()},subBlocks]}]},text={label:"text",requires:function(A){var g=A.peekBytes(2);return 33===g[0]&&1===g[1]},parts:[{label:"codes",parser:Parsers.readBytes(2),skip:!0},{label:"blockSize",parser:Parsers.readByte()},{label:"preData",parser:function(A,g,I){return A.readBytes(I.text.blockSize)}},subBlocks]},application={label:"application",requires:function(A,g,I){var C=A.peekBytes(2);return 33===C[0]&&255===C[1]},parts:[{label:"codes",parser:Parsers.readBytes(2),skip:!0},{label:"blockSize",parser:Parsers.readByte()},{label:"id",parser:function(A,g,I){return A.readString(I.blockSize)}},subBlocks]},comment={label:"comment",requires:function(A,g,I){var C=A.peekBytes(2);return 33===C[0]&&254===C[1]},parts:[{label:"codes",parser:Parsers.readBytes(2),skip:!0},subBlocks]},frames={label:"frames",parts:[gce,application,comment,image,text],loop:function(A){var g=A.peekByte();return 33===g||44===g}},schemaGIF=[{label:"header",parts:[{label:"signature",parser:Parsers.readString(3)},{label:"version",parser:Parsers.readString(3)}]},{label:"lsd",parts:[{label:"width",parser:Parsers.readUnsigned(!0)},{label:"height",parser:Parsers.readUnsigned(!0)},{label:"gct",bits:{exists:{index:0},resolution:{index:1,length:3},sort:{index:4},size:{index:5,length:3}}},{label:"backgroundColorIndex",parser:Parsers.readByte()},{label:"pixelAspectRatio",parser:Parsers.readByte()}]},{label:"gct",requires:function(A,g){return g.lsd.gct.exists},parser:Parsers.readArray(3,(function(A,g){return Math.pow(2,g.lsd.gct.size+1)}))},frames],gifSchema=schemaGIF;function GIF(A){var g=new DataParser(new Uint8Array(A));this.raw=g.parse(gifSchema),this.raw.hasImages=!1;for(var I=0;I<this.raw.frames.length;I++)if(this.raw.frames[I].image){this.raw.hasImages=!0;break}}GIF.prototype.decompressFrame=function(A,g){if(A>=this.raw.frames.length)return null;var I=this.raw.frames[A];if(I.image){var C=I.image.descriptor.width*I.image.descriptor.height,e=function(A,g,I){var C,e,i,t,o,a,B,s,l,n,d,r,c,h,G,m,b=4096,Z=-1,u=I,y=new Array(I),E=new Array(b),Q=new Array(b),V=new Array(b+1);for(o=(e=1<<(r=A))+1,C=e+2,B=Z,i=(1<<(t=r+1))-1,l=0;l<e;l++)E[l]=0,Q[l]=l;for(d=s=c=h=m=G=0,n=0;n<u;){if(0===h){if(s<t){d+=g[G]<<s,s+=8,G++;continue}if(l=d&i,d>>=t,s-=t,l>C||l==o)break;if(l==e){i=(1<<(t=r+1))-1,C=e+2,B=Z;continue}if(B==Z){V[h++]=Q[l],B=l,c=l;continue}for(a=l,l==C&&(V[h++]=c,l=B);l>e;)V[h++]=Q[l],l=E[l];c=255&Q[l],V[h++]=c,C<b&&(E[C]=B,Q[C]=c,!(++C&i)&&C<b&&(t++,i+=C)),B=a}h--,y[m++]=V[h],n++}for(n=m;n<u;n++)y[n]=0;return y}(I.image.data.minCodeSize,I.image.data.blocks,C);I.image.descriptor.lct.interlaced&&(e=function(A,g){for(var I=new Array(A.length),C=A.length/g,e=function(C,e){var i=A.slice(e*g,(e+1)*g);I.splice.apply(I,[C*g,g].concat(i))},i=[0,4,2,1],t=[8,8,4,2],o=0,a=0;a<4;a++)for(var B=i[a];B<C;B+=t[a])e(B,o),o++;return I}(e,I.image.descriptor.width));var i={pixels:e,dims:{top:I.image.descriptor.top,left:I.image.descriptor.left,width:I.image.descriptor.width,height:I.image.descriptor.height}};return I.image.descriptor.lct&&I.image.descriptor.lct.exists?i.colorTable=I.image.lct:i.colorTable=this.raw.gct,I.gce&&(i.delay=10*(I.gce.delay||10),i.disposalType=I.gce.extras.disposal,I.gce.extras.transparentColorGiven&&(i.transparentIndex=I.gce.transparentColorIndex)),g&&(i.patch=function(A){for(var g=A.pixels.length,I=new Uint8ClampedArray(4*g),C=0;C<g;C++){var e=4*C,i=A.pixels[C],t=A.colorTable[i];I[e]=t[0],I[e+1]=t[1],I[e+2]=t[2],I[e+3]=i!==A.transparentIndex?255:0}return I}(i)),i}return null},GIF.prototype.decompressFrames=function(A){for(var g=[],I=0;I<this.raw.frames.length;I++){this.raw.frames[I].image&&g.push(this.decompressFrame(I,A))}return g};const imageDataLibrary={};class GIFImage extends FileImage{__streamAtlas;play;stop;__resourcePromise;__unpackedData;streamAtlasDescParam=new Vec4Parameter("StreamAtlasDesc");streamAtlasIndexParam=new NumberParameter("StreamAtlasIndex",0);constructor(A,g="",I={}){super(A,g,I),this.format="RGBA",this.type="UNSIGNED_BYTE",this.__streamAtlas=!0,this.addParameter(this.streamAtlasDescParam),this.addParameter(this.streamAtlasIndexParam);const C=this.streamAtlasIndexParam;let e;C.setRange([0,1]);let i=0;const t=A=>{C.value=i,e&&setTimeout((()=>t(A)),this.getFrameDelay(i)),i=(i+1)%A};this.play=()=>{this.__resourcePromise.then((()=>{e=!0;let A=C.getRange();if(!A)return void console.warn("numFrames is null");const g=A[1];t(g)}))},this.stop=()=>{e=!1}}getFrameDelay(A){return 10*this.__unpackedData.frameDelays[A]}load(A,g="RGB"){return A in imageDataLibrary?(this.__resourcePromise=imageDataLibrary[A],this.__resourcePromise):(this.__resourcePromise=new Promise(((g,I)=>{resourceLoader.incrementWorkload(1),loadBinfile(A,(I=>{console.warn("Unpacking Gif client side:"+A);const C=performance.now(),e=new GIF(I).decompressFrames(!0),i=Math.sqrt(e.length),t=[i,i];MathFunctions.fract(i)>0&&(t[0]=Math.floor(t[0]+1),MathFunctions.fract(i)>.5?t[1]=Math.floor(t[1]+1):t[1]=Math.floor(t[1]));const o=e[0].dims.width,a=e[0].dims.height,B=document.createElement("canvas"),s=B.getContext("2d"),l=document.createElement("canvas"),n=l.getContext("2d");l.width=o,l.height=a;const d=document.createElement("canvas"),r=d.getContext("2d");let c;d.width=t[0]*o,d.height=t[1]*a;const h=[],G=(A,g)=>{const I=A.dims;h.push(A.delay/10),c&&I.width==c.width&&I.height==c.height||(B.width=I.width,B.height=I.height,c=s?.createImageData(I.width,I.height)),c.data.set(A.patch),s?.putImageData(c,0,0),2==A.disposalType&&n?.clearRect(0,0,l.width,l.height),n?.drawImage(B,I.left,I.top),r?.drawImage(l,g%t[0]*o,Math.floor(g/t[0])*a)};for(let A=0;A<e.length;A++)G(e[A],A);resourceLoader.incrementWorkDone(1);const m=r?.getImageData(0,0,d.width,d.height),b=performance.now()-C;console.log(`Decode GIF '${A}' time:`+b),g({width:d.width,height:d.height,atlasSize:t,frameRange:[0,e.length],frameDelays:h,imageData:m})}),(g=>{const C="Unable to Load URL:"+g+":"+A;console.warn(C),I(C)}))})),imageDataLibrary[A]=this.__resourcePromise,this.__resourcePromise.then((A=>{this.width=A.width,this.height=A.height,this.streamAtlasDescParam.value=new Vec4(A.atlasSize[0],A.atlasSize[1],0,0),this.streamAtlasIndexParam.setRange(A.frameRange),this.__unpackedData=A,this.__data=A.imageData,this.loaded=!0,this.emit("loaded")})),this.__resourcePromise)}}Registry.register("GIFImage",GIFImage);class EnvMap extends HDRImage{utf8decoder;shCoeffs;luminanceData;headlightModeParam=new BooleanParameter("HeadLightMode",!1);constructor(A,g={}){super(A,g),this.addParameter(this.headlightModeParam),this.utf8decoder=new TextDecoder,this.shCoeffs=[]}__decodeData(A){const g=A.samples;if(g&&(this.luminanceData=JSON.parse(this.utf8decoder.decode(g)),this.luminanceData.shCoeffs))for(let A=0;A<9;A++)this.shCoeffs[A]=new Color(this.luminanceData.shCoeffs[3*A+0],this.luminanceData.shCoeffs[3*A+1],this.luminanceData.shCoeffs[3*A+2]);return super.__decodeData(A)}dirToLuminance(A){const g=A.x,I=A.y,C=A.z,e=this.shCoeffs[0].scale(.886227);return e.addInPlace(this.shCoeffs[1].scale(1.023328*I)),e.addInPlace(this.shCoeffs[2].scale(1.023328*C)),e.addInPlace(this.shCoeffs[3].scale(1.023328*g)),e.addInPlace(this.shCoeffs[4].scale(.858086*g*I)),e.addInPlace(this.shCoeffs[5].scale(.858086*I*C)),e.addInPlace(this.shCoeffs[6].scale(.743125*C*C-.247708)),e.addInPlace(this.shCoeffs[7].scale(.858086*g*C)),e.addInPlace(this.shCoeffs[8].scale(.429043*(g*g-I*I))),e.luminance()}}function getLanguage(){if("Node"==SystemDesc.OS)return"en";const A=new URLSearchParams(globalThis.location.search);if(A.has("lang"))return A.get("lang");const g=globalThis.navigator;let I,C;const e=A=>A.startsWith("en")?"En":A.startsWith("es")?"Es":A.startsWith("fr")?"Fr":A.startsWith("gb")||A.startsWith("de")?"Gb":A;if(Array.isArray(g.languages))for(I=0;I<g.languages.length;I++)if(C=g.languages[I],C&&C.length)return e(C);return null}Registry.register("EnvMap",EnvMap);class LabelManager extends EventEmitter{__language;__foundLabelLibraries;__labelLibraries;constructor(){super(),this.__labelLibraries={},this.__language=getLanguage(),this.__foundLabelLibraries={}}loadLibrary(A,g){const I=A.substring(0,A.lastIndexOf("."));if(this.__foundLabelLibraries[I]=g,A.endsWith(".labels"))loadTextfile(g,(A=>{this.__labelLibraries[I]=JSON.parse(A),this.emit("labelLibraryLoaded",{library:I})}));else if(A.endsWith(".xlsx")){const A=globalThis.XLSX;loadBinfile(g,(g=>{const C=new Uint8Array(g),e=A.read(C,{type:"array"}),i={};e.SheetNames.forEach((function(g){A.utils.sheet_to_json(e.Sheets[g],{}).forEach((function(A){const g=A.Identifier;delete A.Identifier,i[g]=A}))})),this.__labelLibraries[I]=i,this.emit("labelLibraryLoaded",{library:I})}))}}isLibraryFound(A){return A in this.__foundLabelLibraries}isLibraryLoaded(A){return A in this.__labelLibraries}getLabelText(A,g){const I=this.__labelLibraries[A];if(!I)throw new Error("LabelLibrary: '"+A+"' not found in LabelManager. Found: ["+Object.keys(this.__labelLibraries)+"]");const C=I[g];if(!C)throw new Error("Label: '"+g+"' not found in LabelLibrary: '"+A+"'. Found: ["+Object.keys(I)+"]");const e=C[this.__language];if(!e){if(C.En)return C.En;throw new Error("labelText: '"+this.__language+"' not found in Label. Found: ["+Object.keys(C)+"]")}return e}setLabelText(A,g,I){let C=this.__labelLibraries[A];C||(C={},this.__labelLibraries[A]=C);let e=C[g];e||(e={},C[g]=e),e[this.__language]=I}setLanguage(A){this.__language=A}}const labelManager=new LabelManager;function roundRect(A,g,I,C,e,i,t=!1,o=!0,a){if(void 0===o&&(o=!0),void 0===i&&(i=5),"number"==typeof i)i={tl:i,tr:i,br:i,bl:i};else{const A={tl:0,tr:0,br:0,bl:0};for(const g in A)i[g]=i[g]||A[g]}A.beginPath(),A.moveTo(g+i.tl,I),A.lineTo(g+C-i.tr,I),A.quadraticCurveTo(g+C,I,g+C,I+i.tr),A.lineTo(g+C,I+e-i.br),A.quadraticCurveTo(g+C,I+e,g+C-i.br,I+e),A.lineTo(g+i.bl,I+e),A.quadraticCurveTo(g,I+e,g,I+e-i.bl),A.lineTo(g,I+i.tl),A.quadraticCurveTo(g,I,g+i.tl,I),A.closePath(),t&&A.fill(),o&&(A.lineWidth=a,A.stroke())}class Label extends DataImage{needsRender;canvasElem;requestedReRender=!1;marginParam;borderRadiusParam;libraryParam=new StringParameter("Library");textParam=new StringParameter("Text","");fontColorParam=new ColorParameter("FontColor",new Color(0,0,0));fontSizeParam=new NumberParameter("FontSize",22);fontParam=new StringParameter("Font","Helvetica");borderWidthParam=new NumberParameter("BorderWidth",2);outlineParam=new BooleanParameter("Outline",!1);outlineColorParam=new ColorParameter("OutlineColor",new Color(0,0,0));backgroundParam=new BooleanParameter("Background",!0);backgroundColorParam=new ColorParameter("BackgroundColor",new Color("#FBC02D"));fillBackgroundParam=new BooleanParameter("FillBackground",!0);strokeBackgroundOutlineParam=new BooleanParameter("StrokeBackgroundOutline",!0);constructor(A,g){super(A),this.canvasElem=document.createElement("canvas");this.marginParam=new NumberParameter("Margin",11),this.borderRadiusParam=new NumberParameter("BorderRadius",11),this.addParameter(this.marginParam),this.addParameter(this.borderRadiusParam),this.addParameter(this.libraryParam),this.addParameter(this.textParam),this.addParameter(this.fontColorParam),this.addParameter(this.fontSizeParam),this.addParameter(this.fontParam),this.addParameter(this.borderWidthParam),this.addParameter(this.outlineParam),this.addParameter(this.outlineColorParam),this.addParameter(this.backgroundParam),this.addParameter(this.backgroundColorParam),this.addParameter(this.fillBackgroundParam),this.addParameter(this.strokeBackgroundOutlineParam);this.on("nameChanged",(()=>{this.loadLabelData()})),g&&(this.libraryParam.value=g),this.requestedReRender=!1,this.needsRender=!1,this.loadLabelData()}__parameterValueChanged(A){super.parameterValueChanged(A),this.requestedReRender||(this.requestedReRender=!0,this.loadLabelData())}loadLabelData(){Promise.all([(()=>new Promise((A=>{const g=this.libraryParam.value;if(""==g)return void A();if(!labelManager.isLibraryFound(g))return console.warn("Label Library not found:",g),void A();const I=()=>{try{const A=this.getName(),I=labelManager.getLabelText(g,A);this.textParam.value=I}catch(A){console.warn(A)}A()};labelManager.isLibraryLoaded(g)?I():labelManager.on("labelLibraryLoaded",(A=>{A.library==g&&I()}))})))(),(()=>new Promise((A=>{if(null!=document.fonts){const g=this.fontParam.value,I=this.fontSizeParam.value;document.fonts.load(I+'px "'+g+'"').then((()=>{A()}))}else A()})))()]).then((()=>{this.requestedReRender=!1,this.needsRender=!0,this.loaded?this.emit("updated"):(this.loaded=!0,this.emit("loaded"))}))}renderLabelToImage(){const A=this.canvasElem.getContext("2d",{alpha:!0});let g=this.textParam.value;""==g&&(g=this.getName());const I=this.fontParam.value,C=this.fontColorParam.value,e=this.fontSizeParam.value,i=this.marginParam.value,t=this.borderWidthParam.value,o=this.borderRadiusParam.value,a=this.outlineParam.value,B=this.outlineColorParam.value,s=this.backgroundParam.value,l=this.backgroundColorParam.value,n=this.fillBackgroundParam.value,d=this.strokeBackgroundOutlineParam.value,r=i+t,c=g.split("\n");A.font=e+'px "'+I+'"';let h=0;c.forEach((g=>{h=Math.max(A.measureText(g).width,h)}));const G=e;this.width=Math.ceil(h+2*r),this.height=Math.ceil(G*c.length+2*r),A.canvas.width=this.width,A.canvas.height=this.height,this.canvasElem.width=this.width,this.canvasElem.height=this.height,A.fillStyle="rgba(0, 0, 0, 0.0)",A.fillRect(0,0,this.width,this.height),s&&(A.fillStyle=l.toHex(),A.strokeStyle=B.toHex(),roundRect(A,t,t,this.width-2*t,this.height-2*t,o,n,d,t)),A.font=e+'px "'+I+'"',A.textAlign="left",A.fillStyle=C.toHex(),A.textBaseline="hanging",c.forEach(((g,I)=>{A.fillText(g,r,r+I*G)})),a&&(A.strokeStyle=B.toHex(),A.lineWidth=1.5,A.strokeText(g,r,r)),this.__data=A.getImageData(0,0,this.width,this.height),this.needsRender=!1,this.emit("labelRendered",{width:this.width,height:this.height,data:this.__data})}getParams(){return this.needsRender&&this.renderLabelToImage(),super.getParams()}}Registry.register("Label",Label);class VideoStreamImage2D extends BaseImage{__data;__intervalId;constructor(){super("")}connectWebcam(A,g,I=!1){const C={width:A,height:g,frameRate:{ideal:60,max:60}};C.facingMode=I?{exact:"environment"}:{facingMode:"user"};const e=document.createElement("video");e.style.display="none",e.preload="auto",e.crossOrigin="anonymous",document.body.appendChild(e),navigator.mediaDevices.getUserMedia({audio:!1,video:C}).then((A=>{e.srcObject=A,e.onloadedmetadata=A=>{e.play(),this.width=e.videoWidth,this.height=e.videoHeight,console.log("Webcam:["+this.width+", "+this.height+"]"),this.__data=e,this.loaded=!0,this.emit("loaded");let g=0;const I=()=>{if(e.paused||e.ended)return;const A=Math.floor(60*e.currentTime);g!=A&&(this.emit("updated"),g=A),setTimeout(I,20)};I()}})).catch((function(A){}))}setVideoStream(A){this.loaded=!1,this.width=A.videoWidth,this.height=A.videoHeight,this.start(),this.__data=A,this.loaded=!0,this.emit("loaded")}stop(){clearInterval(this.__intervalId)}start(){this.__intervalId=setInterval((()=>{this.emit("updated")}),20)}isLoaded(){return this.loaded}getParams(){return{type:this.type,format:this.format,width:this.width,height:this.height,data:this.__data}}}Registry.register("VideoStreamImage2D",VideoStreamImage2D);class FatLinesMaterial extends Material{baseColorParam=new MaterialColorParam("BaseColor",new Color(1,1,.5));opacityParam=new NumberParameter("Opacity",1);lineThicknessParam=new NumberParameter("LineThickness",.01);overlayParam=new NumberParameter("Overlay",0);constructor(A){super(A),this.__shaderName="FatLinesShader",this.addParameter(this.baseColorParam),this.addParameter(this.opacityParam),this.addParameter(this.overlayParam),this.addParameter(this.lineThicknessParam)}isTextured(){return!0}}Registry.register("FatLinesMaterial",FatLinesMaterial);class FatPointsMaterial extends Material{baseColorParam=new MaterialColorParam("BaseColor",new Color(1,1,1));pointSizeParam=new NumberParameter("PointSize",1);borderWidthParam=new NumberParameter("BorderWidth",.2);overlayParam=new NumberParameter("Overlay",0);constructor(A){super(A),this.__shaderName="FatPointsShader",this.addParameter(this.baseColorParam),this.addParameter(this.pointSizeParam),this.addParameter(this.borderWidthParam),this.addParameter(this.overlayParam)}clone(A){const g=new FatPointsMaterial;return g.copyFrom(this,A),g}}Registry.register("FatPointsMaterial",FatPointsMaterial);class FlatSurfaceMaterial extends Material{baseColorParam=new MaterialColorParam("BaseColor",new Color(1,1,.5));overlayParam=new MaterialFloatParam("Overlay",0);constructor(A){super(A),this.__shaderName="FlatSurfaceShader",this.addParameter(this.baseColorParam),this.addParameter(this.overlayParam)}clone(A){const g=new FlatSurfaceMaterial;return g.copyFrom(this,A),g}}Registry.register("FlatSurfaceMaterial",FlatSurfaceMaterial);class LinesMaterial extends Material{baseColorParam=new MaterialColorParam("BaseColor",new Color(1,1,.5));opacityParam=new MaterialFloatParam("Opacity",.7,[0,1]);overlayParam=new MaterialFloatParam("Overlay",1e-6);constructor(A){super(A),this.__shaderName="LinesShader",this.addParameter(this.baseColorParam),this.addParameter(this.opacityParam),this.addParameter(this.overlayParam)}clone(A){const g=new LinesMaterial;return g.copyFrom(this,A),g}}Registry.register("LinesMaterial",LinesMaterial);class PointsMaterial extends Material{baseColorParam=new MaterialColorParam("BaseColor",new Color(1,1,.5));pointSizeParam=new MaterialFloatParam("PointSize",2);overlayParam=new MaterialFloatParam("Overlay",2e-5);constructor(A){super(A),this.__shaderName="PointsShader",this.addParameter(this.baseColorParam),this.addParameter(this.pointSizeParam),this.addParameter(this.overlayParam)}clone(A){const g=new PointsMaterial;return g.copyFrom(this,A),g}}Registry.register("PointsMaterial",PointsMaterial);class ScreenSpaceMaterial extends Material{baseColorParam=new MaterialColorParam("BaseColor",new Color(1,1,.5));constructor(A){super(A),this.__shaderName="ScreenSpaceShader",this.addParameter(this.baseColorParam)}clone(A){const g=new ScreenSpaceMaterial;return g.copyFrom(this,A),g}}Registry.register("ScreenSpaceMaterial",ScreenSpaceMaterial);class SimpleSurfaceMaterial extends Material{baseColorParam=new MaterialColorParam("BaseColor",new Color(1,1,.5));opacityParam=new MaterialFloatParam("Opacity",1,[0,1]);emissiveStrengthParam=new MaterialFloatParam("EmissiveStrength",0,[0,1]);constructor(A){super(A),this.__shaderName="SimpleSurfaceShader",this.addParameter(this.baseColorParam),this.addParameter(this.opacityParam),this.addParameter(this.emissiveStrengthParam)}clone(A){const g=new SimpleSurfaceMaterial;return g.copyFrom(this,A),g}}Registry.register("SimpleSurfaceMaterial",SimpleSurfaceMaterial);class StandardSurfaceMaterial extends Material{baseColorParam=new MaterialColorParam("BaseColor",new Color(1,1,.5));normalParam=new MaterialColorParam("Normal",new Color(1,1,.5));ambientOcclusion=new MaterialFloatParam("AmbientOcclusion",1,[0,1]);metallicParam=new MaterialFloatParam("Metallic",.05,[0,1]);roughnessParam=new MaterialFloatParam("Roughness",.5,[0,1]);reflectanceParam=new MaterialFloatParam("Reflectance",.5,[0,1]);emissiveStrengthParam=new MaterialFloatParam("EmissiveStrength",0,[0,1]);opacityParam=new MaterialFloatParam("Opacity",1,[0,1]);edgeWeightParam=new MaterialFloatParam("EdgeWeight",.7,[0,1]);overlayParam=new MaterialFloatParam("Overlay",0,[0,1]);edgeColorParam=new MaterialColorParam("EdgeColor",new Color(.1,.1,.1,.8));pointColorParam=new MaterialColorParam("PointColor",new Color(.1,.1,.1));pointSizeParam=new MaterialFloatParam("PointSize",4,[0,Number.MAX_VALUE]);constructor(A){super(A),this.__shaderName="StandardSurfaceShader",this.addParameter(this.baseColorParam),this.addParameter(this.normalParam),this.addParameter(this.ambientOcclusion),this.addParameter(this.metallicParam),this.addParameter(this.roughnessParam),this.addParameter(this.reflectanceParam),this.addParameter(this.emissiveStrengthParam),this.addParameter(this.opacityParam),this.addParameter(this.edgeWeightParam),this.addParameter(this.overlayParam),this.addParameter(this.edgeColorParam),this.addParameter(this.pointColorParam),this.addParameter(this.pointSizeParam)}clone(A){const g=new StandardSurfaceMaterial;return g.copyFrom(this,A),g}}Registry.register("StandardSurfaceMaterial",StandardSurfaceMaterial);class VertexColorMaterial extends Material{constructor(A){super(A),this.__shaderName="VertexColorShader"}clone(A){const g=new VertexColorMaterial;return g.copyFrom(this,A),g}static supportsInstancing(){return!1}}Registry.register("VertexColorMaterial",VertexColorMaterial);class MaskMaterial extends SimpleSurfaceMaterial{constructor(A){super(A),this.baseColorParam.value=new Color(1,0,0,.2)}clone(A){const g=new MaskMaterial;return g.copyFrom(this,A),g}}Registry.register("MaskMaterial",MaskMaterial);class EnvProjectionMaterial extends Material{projectionCenter=new Vec3Parameter("projectionCenter",new Vec3(0,0,1.7));constructor(A){super(A),this.__shaderName="EnvProjectionShader",this.addParameter(this.projectionCenter)}clone(A){const g=new EnvProjectionMaterial;return g.copyFrom(this,A),g}isTextured(){return!0}}Registry.register("EnvProjectionMaterial",EnvProjectionMaterial);class InstanceItem extends TreeItem{srcTreePath=[];srcTree=null;constructor(A){super(A)}setSrcTree(A){this.srcTree=A;const g=new CloneContext,I=this.srcTree.clone(g);I.localXfoParam.value=new Xfo,this.addChild(I,!1,!1)}getSrcTree(){return this.srcTree}readBinary(A,g){super.readBinary(A,g),this.srcTreePath=A.loadStrArray(),this.srcTreePath.length>0&&g.resolvePath(this.srcTreePath,(A=>{this.setSrcTree(A)}),(A=>{console.warn(`Error loading InstanceItem: ${this.getPath()}, unable to resolve: ${this.srcTreePath}. `+A.message)}))}clone(A){const g=new InstanceItem;return g.copyFrom(this,A),g}copyFrom(A,g){if(!(A instanceof InstanceItem))throw new Error("cannot copy from src");super.copyFrom(A,g),this.srcTreePath=A.srcTreePath,this.srcTreePath.length>0&&0==this.getNumChildren()&&A.once("childAdded",(A=>{const g=A.childItem;this.setSrcTree(g)}))}}Registry.register("InstanceItem",InstanceItem);class BaseGeomItem extends TreeItem{overlay;cutAway;cutAwayVector;cutAwayDist;layers;materialParam=new MaterialParameter("Material");constructor(A){super(A),this.addParameter(this.materialParam),this.overlay=!1,this.cutAway=!1,this.cutAwayVector=new Vec3(0,0,0),this.cutAwayDist=0,this.layers=[]}setOverlay(A){this.overlay=A}isOverlay(){return this.overlay}addLayer(A){this.layers.push(A)}getLayers(){return this.layers}isCutawayEnabled(){return this.cutAway}setCutawayEnabled(A){this.cutAway=A,this.emit("cutAwayChanged")}getCutVector(){return this.cutAwayVector}setCutVector(A){this.cutAwayVector=A,this.emit("cutAwayChanged")}getCutDist(){return this.cutAwayDist}setCutDist(A){this.cutAwayDist=A,this.emit("cutAwayChanged")}readBinary(A,g){if(super.readBinary(A,g),g.versions["zea-engine"].compare([0,0,4])>=0){const I=A.loadStr();let C=g.assetItem.getMaterialLibrary().getMaterial(I);if(C||(C=new Material(I,"SimpleSurfaceShader"),C.getParameter("BaseColor").loadValue(Color.random(.25)),g.assetItem.getMaterialLibrary().addMaterial(C)),this.materialParam.loadValue(C),this.layers=A.loadStrArray(),this.layers.length>0)for(const A of this.layers)g.addGeomToLayer(this,A)}}}let calculatePreciseBoundingBoxes=!1;class CalcGeomMatOperator extends Operator{globalXfo=new XfoOperatorInput("GlobalXfo");geomOffsetXfo=new XfoOperatorInput("GeomOffsetXfo");geomMat=new Mat4OperatorOutput("GeomMat");constructor(A,g,I){super("CalcGeomMatOperator"),this.globalXfo.setParam(A),this.geomOffsetXfo.setParam(g),this.geomMat.setParam(I),this.addInput(this.globalXfo),this.addInput(this.geomOffsetXfo),this.addOutput(this.geomMat)}evaluate(){const A=this.globalXfo.getValue().toMat4(),g=this.geomOffsetXfo.getValue().toMat4();this.geomMat.setClean(A.multiply(g))}}class GeomItem extends BaseGeomItem{listenerIDs={};geomBBox;geomIndex=-1;assetItem=null;calcGeomMatOperator;cullable=!0;geomOffsetXfoParam=new XfoParameter("GeomOffsetXfo");geomParam=new GeometryParameter("Geometry");geomMatParam=new Mat4Parameter("GeomMat");constructor(A,g,I,C){super(A),this.addParameter(this.geomParam),this.addParameter(this.geomOffsetXfoParam),this.addParameter(this.geomMatParam);const e=()=>{this.setBoundingBoxDirty()};this.geomParam.on("valueChanged",e),this.geomParam.on("boundingBoxChanged",e),this.calcGeomMatOperator=new CalcGeomMatOperator(this.globalXfoParam,this.geomOffsetXfoParam,this.geomMatParam),g&&this.geomParam.loadValue(g),I&&this.materialParam.loadValue(I),C&&(this.localXfoParam.value=C)}cleanBoundingBox(){if(this.disableBoundingBox)return new Box3;const A=super.cleanBoundingBox();if(this.geomBBox){const g=this.geomMatParam.value;A.addPoint(g.transformVec3(this.geomBBox.p0)),A.addPoint(g.transformVec3(this.geomBBox.p1))}else{const g=this.geomParam.value;if(g)if(calculatePreciseBoundingBoxes){const I=this.geomMatParam.value,C=g.genBuffers();if(g instanceof BaseProxy){const e=C.attrBuffers.positions.values,i=A=>{const g=3*A,I=new Vec3;return I.fromArray(e.subarray(g,g+3)),I};for(let C=0;C<g.getNumVertices();C++)A.addPoint(I.transformVec3(i(C)))}else{const C=g.getVertexAttribute("positions");for(let e=0;e<g.getNumVertices();e++)A.addPoint(I.transformVec3(C.getValue(e)))}}else A.addBox3(g.getBoundingBox(),this.geomMatParam.value)}return A}fromJSON(A,g){super.fromJSON(A,g)}readBinary(A,g){super.readBinary(A,g);const I=A.loadUInt8(),C=A.loadUInt32(),e=g.assetItem.getGeometryLibrary();this.geomIndex=C,this.assetItem=g.assetItem;const i=e.getGeom(C);if(i)this.geomParam.loadValue(i);else{const A=A=>{const{range:I}=A;if(C>=I[0]&&C<I[1]){const A=e.getGeom(C);A?this.geomParam.value=A:console.warn("Geom not loaded:",this.getName()),e.off("rangeLoaded",g)}},g=e.on("rangeLoaded",A)}if(4&I&&(this.geomOffsetXfoParam.value=new Xfo(A.loadFloat32Vec3(),A.loadFloat32Quat(),A.loadFloat32Vec3())),g.versions["zea-engine"].compare([0,0,4])<0){if(I&8){const I=g.assetItem.getMaterialLibrary(),C=A.loadStr();let e=I.getMaterial(C);e||(console.warn("Geom :'"+this.name+"' Material not found:"+C),e=I.getMaterial("Default")),this.materialParam.loadValue(e)}else this.materialParam.loadValue(g.assetItem.getMaterialLibrary().getMaterial("Default"))}g.versions["zea-engine"].compare([3,0,0])<0?A.loadFloat32Vec2():this.geomBBox=new Box3(A.loadFloat32Vec3(),A.loadFloat32Vec3())}toString(A){return JSON.stringify(this.toJSON(A),null,2)}clone(A){const g=new GeomItem;return g.copyFrom(this,A),g}copyFrom(A,g){if(!(A instanceof GeomItem))throw new Error("cannot copy from src");if(super.copyFrom(A,g),this.geomOffsetXfoParam.value=A.geomOffsetXfoParam.value,!A.geomParam.value&&-1!=A.geomIndex){const g=A.assetItem.getGeometryLibrary();this.assetItem=A.assetItem,this.geomIndex=A.geomIndex,this.geomBBox=A.geomBBox;const I=A=>{const{range:I}=A;if(this.geomIndex>=I[0]&&this.geomIndex<I[1]){const A=g.getGeom(this.geomIndex);A?this.geomParam.value=A:console.warn("Geom not loaded:",this.getName()),g.off("rangeLoaded",this.listenerIDs.rangeLoaded)}};this.listenerIDs.rangeLoaded=g.on("rangeLoaded",I)}this.geomMatParam.setDirty(0)}static setCalculatePreciseBoundingBoxes(A){calculatePreciseBoundingBoxes=A}}Registry.register("GeomItem",GeomItem);class CompoundGeomLoader extends BaseGeom{numSubGeoms=0;indices=new Uint8Array(0);offsets={};counts={};materialSubGeoms={};subGeomOffsets={};subGeomCounts={};materialLibraryIndices=new Uint32Array(0);subGeomMaterialIndices=new Uint8Array(0);constructor(){super()}genBuffers(){const A={};for(const[g,I]of this.__vertexAttributes)A[g]=I.genBuffer();const g=this.numVertices();return{numVertices:g,numRenderVerts:g,indices:this.indices,attrBuffers:A,offsets:this.offsets,counts:this.counts,numSubGeoms:this.numSubGeoms,subGeomOffsets:this.subGeomOffsets,subGeomCounts:this.subGeomCounts,materialLibraryIndices:this.materialLibraryIndices,subGeomMaterialIndices:this.subGeomMaterialIndices,materialSubGeoms:this.materialSubGeoms}}readBinary(A,g){super.loadBaseGeomBinary(A,g);const I=A.loadUInt32Array(3);this.offsets.TRIANGLES=0,this.counts.TRIANGLES=I[0],this.offsets.LINES=I[0],this.counts.LINES=I[1],this.offsets.POINTS=I[0]+I[1],this.counts.POINTS=I[2];const C=A.loadUInt8();1==C?this.indices=A.loadUInt8Array():2==C?this.indices=A.loadUInt16Array():4==C&&(this.indices=A.loadUInt32Array());const e=A.loadUInt8();let i;if(1==e)i=A.loadUInt8Array();else if(2==e)i=A.loadUInt16Array();else{if(4!=e)throw Error("subGeomOffsets undefined");i=A.loadUInt32Array()}const t=new Uint32Array(i.length);let o=0;for(let A=0;A<i.length;A++)t[A]=o,o+=i[A];this.subGeomOffsets.TRIANGLES=t,this.subGeomCounts.TRIANGLES=i;const a=A.loadUInt8();let B;if(1==a)B=A.loadUInt8Array();else if(2==a)B=A.loadUInt16Array();else{if(4!=a)throw Error("subGeomOffsets undefined");B=A.loadUInt32Array()}const s=new Uint32Array(B.length);for(let A=0;A<B.length;A++)s[A]=o,o+=B[A];this.subGeomOffsets.LINES=s,this.subGeomCounts.LINES=B;const l=A.loadUInt32(),n=new Uint32Array(l),d=new Uint8Array(l);for(let A=0;A<l;A++)n[A]=o,d[A]=1,o++;this.subGeomOffsets.POINTS=n,this.subGeomCounts.POINTS=d,g.versions["zea-engine"].compare([3,11,0])>0?this.numSubGeoms=i.length+B.length+l:this.numSubGeoms=i.length+B.length;const r=A.loadUInt32();if(r>0){this.materialLibraryIndices=A.loadUInt32Array(r),this.subGeomMaterialIndices=A.loadUInt8Array(this.numSubGeoms);let g=0,I=-99,C=null;for(let A=0;A<this.numSubGeoms;A++){let e,i=0;A<this.subGeomCounts.TRIANGLES.length?(this.materialSubGeoms.TRIANGLES||(this.materialSubGeoms.TRIANGLES=[]),e="TRIANGLES"):A<this.subGeomCounts.TRIANGLES.length+this.subGeomCounts.LINES.length?(i=this.subGeomCounts.TRIANGLES.length,e="LINES",this.materialSubGeoms.LINES||(this.materialSubGeoms.LINES=[])):(i=this.subGeomCounts.TRIANGLES.length+this.subGeomCounts.LINES.length,e="POINTS",this.materialSubGeoms.POINTS||(this.materialSubGeoms.POINTS=[]));const t=this.subGeomMaterialIndices[A];if(I!=t){for(I=t,C={materialId:t-1,offset:g,count:0};A<this.numSubGeoms&&I==this.subGeomMaterialIndices[A];A++){if(A-i==this.subGeomCounts[e].length){I=-99;break}C.count+=this.subGeomCounts[e][A-i]}g+=C.count,this.materialSubGeoms[e].push(C),A--}}this.subGeomCounts.POINTS.length>0&&!this.materialSubGeoms.POINTS&&(this.materialSubGeoms.POINTS=[{materialId:-1,offset:g,count:this.subGeomCounts.POINTS.length}])}else{this.materialSubGeoms={};let A=0;for(let g in this.counts){const I=this.counts[g];I>0&&(this.materialSubGeoms[g]=[{materialId:-1,offset:A,count:I}]),A+=I}}this.emit("geomDataChanged",{})}}const parseGeomsBinary=(A,g)=>{for(const g in A.context.versions){const I=A.context.versions[g],C=new Version("");C.major=I.major,C.minor=I.minor,C.patch=I.patch,C.branch=I.branch,A.context.versions[g]=C}const I=[],C=A.byteOffset,e=[];for(let g=A.geomsRange[0];g<A.geomsRange[1];g++){const i=new BinReader(A.bufferSlice,A.toc[g]-C,A.isMobileDevice),t=i.loadStr(),o=i.pos();let a;switch(t){case"Points":a=new Points;break;case"Lines":a=new Lines;break;case"Mesh":a=new Mesh;break;case"CompoundGeom":a=new CompoundGeomLoader;break;default:throw new Error("Unsupported Geom type:"+t)}try{i.seek(o),a.readBinary(i,A.context)}catch(A){console.warn("Error loading:"+a.name+"\n:"+A),I.push({});continue}const B=a.genBuffers(A.genBuffersOpts);B.indices&&e.push(B.indices.buffer);for(const A in B.attrBuffers){const g=B.attrBuffers[A];e.push(g.values.buffer)}if(B.materialLibraryIndices&&e.push(B.materialLibraryIndices.buffer),B.subGeomMaterialIndices&&e.push(B.subGeomMaterialIndices.buffer),B.subGeomOffsets)for(let A in B.subGeomOffsets)e.push(B.subGeomOffsets[A].buffer);I.push({name:a.name,type:t,geomBuffers:B,bbox:a.getBoundingBox()})}g({taskId:A.taskId,geomLibraryId:A.geomLibraryId,geomFileID:A.geomFileID,geomIndexOffset:A.geomIndexOffset,geomsRange:A.geomsRange,geomDatas:I},e)},handleMessage=function(A,g){parseGeomsBinary(A,((A,I)=>{g(A,I)}))};void 0===globalThis.document&&(globalThis.onmessage=function(A){A.data?A.data.context&&handleMessage(A.data,self.postMessage):console.warn("GeomLibrary worker.postMessage failed. data was lost on the way to the web worker.")});var WorkerFactory$1=createBase64WorkerFactory("/* rollup-plugin-web-worker-loader */
var worker_code = (function (exports) {
    'use strict';

    /**
     * String functions
     *
     */
    class StringFunctions {
        /**
         * Replaces all matches in a string.
         *
         * @static
         * @param str -
         * @param pattern -
         * @param replacement -
         * @return -
         */
        static replaceAll(str, pattern, replacement) {
            return str.replace(new RegExp(pattern, 'g'), replacement);
        }
        /**
         * Returns JSON object as a formatted string, but the numeric values are fixed to the specified precision.
         *
         * @static
         * @param val -
         * @param space -
         * @param precision -
         * @return -
         */
        static stringifyJSONWithFixedPrecision(val, space = 0, precision = 5) {
            return JSON.stringify(val, (_, val) => {
                return val ? (val.toFixed ? Number(val.toFixed(precision)) : val) : val;
            }, space);
        }
        /**
         * Transforms the given string into a numeric value.
         *
         * @static
         * @param str -
         * @return -
         */
        static hashStr(str) {
            let hash = 0;
            let i;
            let chr;
            let len;
            if (str.length === 0)
                return hash;
            for (i = 0, len = str.length; i < len; i++) {
                chr = str.charCodeAt(i);
                hash = (hash << 5) - hash + chr;
                hash |= 0; // Convert to 32bit integer
            }
            return Math.abs(hash);
        }
    }

    /**
     * Representing a Vec2(two-dimensional floating point vector). A Vec2 is for representing 2 dimensional values, such as screen coordinates or pixel coordinates within an image.
     *
     * Math types internally store values in {@link https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Float32Array|Float32Array} and
     * expose getters and setters for the component values.
     *
     */
    class Vec2 {
        x;
        y;
        /**
         * Creates a Vec2.
         *
         * ```javascript
         *  const myVec2 = new Vec2(1.2, 3.4)
         * ```
         *
         */
        constructor(x = 0, y = 0) {
            this.x = x;
            this.y = y;
        }
        /**
         * Setter from scalar components.
         * @param x - The x component.
         * @param y  - The y component.
         */
        set(x, y) {
            this.x = x;
            this.y = y;
        }
        /**
         * Replaces this Vec2 data with the Vec2 data passed as parameter.
         *
         * @param other - The other Vec2 to set from.
         */
        setFromOther(other) {
            this.x = other.x;
            this.y = other.y;
        }
        /**
         * Checks if this Vec2 contains the same values as the other Vec2.
         *
         * @param other - The other Vec2 to compare with.
         * @return - Returns `true` if are the same Vector, otherwise, `false`.
         */
        isEqual(other) {
            return this.x == other.x && this.y == other.y;
        }
        /**
         * Checks if this Vec2 is different from another Vec2.
         *
         * @param other - The other Vec2 to compare with.
         * @return - Returns `true` if the Vec2s are different, otherwise, `false`.
         */
        notEqual(other) {
            return this.x != other.x && this.y != other.y;
        }
        /**
         * Returns true if this Vec2 is approximately the same as other.
         *
         * @param other - The other Vec2 to compare with.
         * @param precision - The precision to which the values must match.
         * @return - Returns true or false.
         */
        approxEqual(other, precision = Number.EPSILON) {
            return Math.abs(this.x - other.x) < precision && Math.abs(this.y - other.y) < precision;
        }
        /**
         * Adds other to this Vec2 and returns the result as a new Vec2.
         *
         * @param other - The other Vec2 to add.
         * @return - Returns a new Vec2.
         */
        add(other) {
            return new Vec2(this.x + other.x, this.y + other.y);
        }
        /**
         * Adds a Vec2 to this Vec2.
         *
         * @param other - The other Vec2 to add.
         */
        addInPlace(other) {
            this.x += other.x;
            this.y += other.y;
        }
        /**
         * Subtracts a Vec2 from this Vec2 and returns the result as a new Vec2.
         *
         * @param other - The other Vec2 to subtract.
         * @return - Returns a new Vec2.
         */
        subtract(other) {
            return new Vec2(this.x - other.x, this.y - other.y);
        }
        /**
         * Subtracts a Vec2 from this Vec2.
         *
         * @param other - The other Vec2 to subtract.
         * @return - Returns a new Vec2.
         */
        subtractInPlace(other) {
            this.x -= other.x;
            this.y -= other.y;
            return this;
        }
        /**
         * Scales this Vec2 by scalar and returns the result as a new Vec2.
         *
         * @param scalar - The scalar value.
         * @return - Returns a new Vec2.
         */
        scale(scalar) {
            return new Vec2(this.x * scalar, this.y * scalar);
        }
        /**
         * Scales this Vec2 by scalar.
         *
         * @param scalar - The scalar value.
         */
        scaleInPlace(scalar) {
            this.x *= scalar;
            this.y *= scalar;
        }
        /**
         * Inverts this Vec2 and returns the result as a new Vec2.
         *
         * @return - Returns a new Vec2.
         */
        invert() {
            return new Vec2(1.0 / this.x, 1.0 / this.y);
        }
        /**
         * Inverts this Vec2.
         *
         * @return - The return value.
         */
        invertInPlace() {
            this.x = 1.0 / this.x;
            this.y = 1.0 / this.y;
            return this;
        }
        /**
         * Multiplies a Vec2 with this Vec2 and returns the result as a new Vec2.
         *
         * @param other - The other Vec2 to multiply with.
         * @return - Returns a new Vec2.
         */
        multiply(other) {
            return new Vec2(this.x * other.x, this.y * other.y);
        }
        /**
         * Multiplies a Vec2 with this Vec2.
         *
         * @param other - The other Vec2 to multiply with.
         */
        multiplyInPlace(other) {
            this.x *= other.x;
            this.y *= other.y;
        }
        /**
         * Calculates the squared length of this Vec2.
         *
         * @return - Returns the length squared.
         */
        lengthSquared() {
            const x = this.x;
            const y = this.y;
            return x * x + y * y;
        }
        /**
         * Calculates the length of this Vec2.
         *
         * @return - Returns the length.
         */
        length() {
            return Math.sqrt(this.lengthSquared());
        }
        /**
         * Calculates the distance to another vector.
         *
         * @param other - The other value.
         * @return - Returns the distance between vectors.
         */
        distanceTo(other) {
            const x = this.x - other.x;
            const y = this.y - other.y;
            return Math.sqrt(x * x + y * y);
        }
        /**
         * Normalizes the Vec2 and returns it as a new Vec2.
         * Multiplies coordinates value by the inverse of the vector length.
         *
         * @return - Returns the Vec2 normalized.
         */
        normalize() {
            const x = this.x;
            const y = this.y;
            let len = x * x + y * y;
            if (len < Number.EPSILON) {
                return new Vec2();
            }
            // TODO: evaluate use of glm_invsqrt here?
            len = 1 / Math.sqrt(len);
            return new Vec2(x * len, y * len);
        }
        /**
         * Normalizes this Vec2 multiplying coordinate values by the inverse of the vector length.
         */
        normalizeInPlace() {
            const x = this.x;
            const y = this.y;
            let len = x * x + y * y;
            if (len < Number.EPSILON) {
                return;
            }
            len = 1 / Math.sqrt(len);
            this.set(x * len, y * len);
        }
        /**
         * Calculates the dot product of this Vec2 against another Vec2.
         *
         * @param other - The other Vec2 to compare with.
         * @return - Returns the dot product.
         */
        dot(other) {
            return this.x * other.x + this.y * other.y;
        }
        /**
         * Calculates the cross product of this Vec2 against another Vec2.
         *
         * @param other - The other Vec2 to compare with.
         * @return - Returns the cross product.
         */
        cross(other) {
            // just calculate the z-component
            return this.x * other.y - this.y * other.x;
        }
        /**
         * Gets the angle between this Vec2 and other assuming both are normalized vectors.
         *
         * @param other - The other Vec2 to compare with.
         * @return - Returns the angle in radians.
         */
        angleTo(other) {
            const cosine = this.normalize().dot(other.normalize());
            if (cosine > 1.0)
                return 0.0;
            else if (cosine < -1.0)
                return Math.PI;
            else
                return Math.acos(cosine);
        }
        /**
         * Gets the angle between this Vec2 and other.
         *
         * @param other - The other Vec2 to compare with.
         * @return - Returns the angle in radians.
         */
        signedAngleTo(other) {
            const angle = this.angleTo(other);
            if (this.cross(other) < 0.0)
                return -angle;
            else
                return angle;
        }
        /**
         * Rotates a Vec2 in a clockwise direction and returns a new rotated Vec2.
         *
         * @param angle - The angle of rotation.
         * @return - Returns the rotated vector.
         */
        rotate(angle) {
            const cosA = Math.cos(angle);
            const sinA = Math.sin(angle);
            return new Vec2(this.x * cosA - this.y * sinA, this.x * sinA + this.y * cosA);
        }
        /**
         * Performs a linear interpolation between this Vec2 and other Vec2.
         *
         * @param other - The other Vec2 to interpolate between.
         * @param t - Interpolation amount between the two inputs.
         * @return - Returns a new Vec2.
         */
        lerp(other, t) {
            const ax = this.x;
            const ay = this.y;
            return new Vec2(ax + t * (other.x - ax), ay + t * (other.y - ay));
        }
        /**
         * Generates a random vector with the given scale.
         *
         * @param scale - Length of the resulting vector. If omitted, a unit vector will be returned.
         * @return - The return value.
         */
        setRandomDir(scale = 1.0) {
            const r = Math.random() * 2.0 * Math.PI;
            this.x = Math.cos(r) * scale;
            this.y = Math.sin(r) * scale;
            return this;
        }
        /**
         * Randomizes the scale of this Vec2 coordinates.
         *
         * @param scale - The scale value.
         * @return - The return value.
         */
        setRandom(scale = 1.0) {
            this.x = Math.random() * scale;
            this.y = Math.random() * scale;
            return this;
        }
        /**
         * Clones this Vec2 and returns a new Vec2.
         *
         * @return - Returns a new Vec2.
         */
        clone() {
            return new Vec2(this.x, this.y);
        }
        /**
         * Returns current Vec2 data as array. Often used to pass types to the GPU.
         *
         * @return - Returns as an array.
         */
        asArray() {
            return [this.x, this.y];
        }
        /**
         * Setter from an array.
         */
        fromArray(vals) {
            this.x = vals[0];
            this.y = vals[1];
        }
        // ///////////////////////////
        // Persistence
        /**
         * Converts this Vec3 to a string in JSON format.
         *
         * @return - The return value.
         */
        toString() {
            // eslint-disable-next-line new-cap
            return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON());
        }
        /**
         * Encodes Vec2 Class as a JSON object for persistence.
         *
         * @return - The json object.
         */
        toJSON() {
            return {
                x: this.x,
                y: this.y,
            };
        }
        /**
         * Decodes a JSON object to set the state of this class.
         *
         * @param j - The json object.
         */
        fromJSON(j) {
            this.x = j.x;
            this.y = j.y;
        }
        /**
         * Loads the state of the value from a binary reader.
         *
         * @param reader - The reader value.
         */
        readBinary(reader) {
            this.x = reader.loadFloat32();
            this.y = reader.loadFloat32();
        }
        /**
         * Calculate the intersection point of 2 2d lines, returning the parameters values for each line.
         *
         * @param p0 - The point of the first line
         * @param d0 - The direction of the first line
         * @param p1 - The point of the second line
         * @param d1 - The direction of the second line
         * @return - Returns an array containing 2 parameter values for the 2 lines.
         */
        static intersectionOfLines(p1, p2, p3, p4) {
            // https://dirask.com/posts/JavaScript-how-to-calculate-intersection-point-of-two-lines-for-given-4-points-VjvnAj
            // down part of intersection point formula
            const d1 = (p1.x - p2.x) * (p3.y - p4.y); // (x1 - x2) * (y3 - y4)
            const d2 = (p1.y - p2.y) * (p3.x - p4.x); // (y1 - y2) * (x3 - x4)
            const d = d1 - d2;
            if (d == 0) {
                return null;
            }
            // upper part of intersection point formula
            const u1 = p1.x * p2.y - p1.y * p2.x; // (x1 * y2 - y1 * x2)
            const u4 = p3.x * p4.y - p3.y * p4.x; // (x3 * y4 - y3 * x4)
            const u2x = p3.x - p4.x; // (x3 - x4)
            const u3x = p1.x - p2.x; // (x1 - x2)
            const u2y = p3.y - p4.y; // (y3 - y4)
            const u3y = p1.y - p2.y; // (y1 - y2)
            // intersection point formula
            const px = (u1 * u2x - u3x * u4) / d;
            const py = (u1 * u2y - u3y * u4) / d;
            return new Vec2(px, py);
        }
        isValid() {
            for (const v of this.asArray()) {
                if (v == Infinity || isNaN(v))
                    return false;
            }
            return true;
        }
    }

    /**
     * Represents a three dimensional coordinate, such as 3D scene values, or mesh vertex positions.
     */
    class Vec3 {
        x;
        y;
        z;
        /**
         * Creates a Vec3.
         */
        constructor(x = 0, y = 0, z = 0) {
            this.x = x;
            this.y = y;
            this.z = z;
        }
        /**
         * Getter for `xy` swizzel.
         *
         * @return - Returns the xy components as a Vec2.
         */
        get xy() {
            return new Vec2(this.x, this.y);
        }
        /**
         * Getter for `yz` swizzel.
         *
         * @return - Returns the yz components as a Vec2.
         */
        get yz() {
            return new Vec2(this.y, this.z);
        }
        /**
         * Setter from scalar components.
         *
         * @param x - The x component.
         * @param y - The y component.
         * @param z - The y component.
         */
        set(x, y, z) {
            this.x = x;
            this.y = y !== undefined ? y : x;
            this.z = z !== undefined ? z : x;
        }
        /**
         * Sets the state of a Vec3 Object from another Vec3.
         *
         * @param other - The other Vec3 to set from.
         */
        setFromOther(other) {
            this.x = other.x;
            this.y = other.y;
            this.z = other.z;
        }
        /**
         * Checks if the coordinates of this Vec3 are 0 0 0.
         *
         * @return - Returns `true` if the coordinates are(0, 0, 0), otherwise, `false`.
         */
        isNull() {
            return Math.abs(this.x) < Number.EPSILON && Math.abs(this.y) < Number.EPSILON && Math.abs(this.z) < Number.EPSILON;
        }
        /**
         * Checks if the coordinates of this Vec3 are 1 1 1.
         *
         * @return - Returns `true` if the coordinates are(1, 1, 1), otherwise, `false`.
         */
        is111() {
            return (Math.abs(1.0 - this.x) < Number.EPSILON &&
                Math.abs(1.0 - this.y) < Number.EPSILON &&
                Math.abs(1.0 - this.z) < Number.EPSILON);
        }
        /**
         * Checks if this Vec3 contains the same values as the other Vec3.
         *
         * @param other - The other Vec3 to compare with.
         * @return - Returns `true` if the values are the same, otherwise, `false`.
         */
        isEqual(other) {
            return this.x == other.x && this.y == other.y && this.z == other.z;
        }
        /**
         * Checks if this Vec2 is different from another Vec2.
         *
         * @param other - The other Vec3 to compare with.
         * @return - Returns `true` if the Vec3s are different, otherwise, `false`.
         */
        notEqual(other) {
            return this.x != other.x && this.y != other.y && this.z != other.z;
        }
        /**
         * Returns true if this Vec2 is approximately the same as other.
         *
         * @param other - The other Vec3 to compare with.
         * @param precision - The precision to which the values must match.
         * @return - Returns true or false.
         */
        approxEqual(other, precision = Number.EPSILON) {
            return (Math.abs(this.x - other.x) < precision &&
                Math.abs(this.y - other.y) < precision &&
                Math.abs(this.z - other.z) < precision);
        }
        /**
         * Adds other to this Vec3 and return the result as a new Vec3.
         *
         * @param other - The other Vec3 to add.
         * @return - Returns a new Vec3.
         */
        add(other) {
            return new Vec3(this.x + other.x, this.y + other.y, this.z + other.z);
        }
        /**
         * Adds other to this Vec3.
         *
         * @param other - The other Vec3 to add.
         */
        addInPlace(other) {
            this.x += other.x;
            this.y += other.y;
            this.z += other.z;
        }
        /**
         * Subtracts other from this Vec3 and returns the result as a new Vec3.
         *
         * @param other - The other Vec3 to subtract.
         * @return - Returns a new Vec3.
         */
        subtract(other) {
            return new Vec3(this.x - other.x, this.y - other.y, this.z - other.z);
        }
        /**
         * Subtracts other from this Vec3.
         *
         * @param other - The other Vec3 to subtract.
         */
        subtractInPlace(other) {
            this.x -= other.x;
            this.y -= other.y;
            this.z -= other.z;
        }
        /**
         * Multiplies two Vec3s and returns the result as a new Vec3.
         *
         * @param other - The other Vec3 to multiply with.
         * @return - Returns a new Vec3.
         */
        multiply(other) {
            return new Vec3(this.x * other.x, this.y * other.y, this.z * other.z);
        }
        /**
         * Multiplies two Vec3s.
         *
         * @param other - The other Vec3 to multiply with.
         */
        multiplyInPlace(other) {
            this.x *= other.x;
            this.y *= other.y;
            this.z *= other.z;
        }
        /**
         * Divides two Vec3s and returns the result as a new Vec3.
         *
         * @param vec3 - The other Vec3 to divide by.
         * @return - Returns a new Vec3.
         */
        divide(vec3) {
            return new Vec3(this.x / vec3.x, this.y / vec3.y, this.z / vec3.z);
        }
        /**
         * Divides two Vec3s.
         *
         * @param vec3 - The other Vec3 to divide by.
         */
        divideInPlace(vec3) {
            this.x /= vec3.x;
            this.y /= vec3.y;
            this.z /= vec3.z;
        }
        /**
         * Scales this Vec3 by scalar and returns the result as a new Vec3.
         *
         * @param scalar - The scalar value.
         * @return - Returns a new Vec3.
         */
        scale(scalar) {
            return new Vec3(this.x * scalar, this.y * scalar, this.z * scalar);
        }
        /**
         * Scales this Vec3 by scalar.
         *
         * @param scalar - The scalar value.
         */
        scaleInPlace(scalar) {
            this.x *= scalar;
            this.y *= scalar;
            this.z *= scalar;
        }
        /**
         * Negates this Vec3 (x = -x, y = -y and z = -z), but returns the result as a new Vec3.
         *
         * @return - Returns a new Vec3.
         */
        negate() {
            return new Vec3(-this.x, -this.y, -this.z);
        }
        /**
         * Returns the inverse of this Vec3, but returns. the result as a new Vec3
         *
         * @return - Returns a new Vec3.
         */
        inverse() {
            return new Vec3(1.0 / this.x, 1.0 / this.y, 1.0 / this.z);
        }
        /**
         * Calculates the squared length of this Vec3.
         *
         * @return - Returns the length.
         */
        lengthSquared() {
            const x = this.x;
            const y = this.y;
            const z = this.z;
            return x * x + y * y + z * z;
        }
        /**
         * Calculates the length of this Vec3.
         *
         * @return - Returns the length.
         */
        length() {
            return Math.sqrt(this.lengthSquared());
        }
        /**
         * Calculates the distance to another Vec3.
         *
         * @param other - The other Vec3 to calculate the distance to.
         * @return - Returns the distance between vectors.
         */
        distanceTo(other) {
            const x = this.x - other.x;
            const y = this.y - other.y;
            const z = this.z - other.z;
            return Math.sqrt(x * x + y * y + z * z);
        }
        /**
         * Normalizes the Vec3 and returns it as a new Vec3.
         * Multiplies coordinates value by the inverse of the vector length.
         *
         * @return - Returns the Vec3 normalized.
         */
        normalize() {
            let len = this.x * this.x + this.y * this.y + this.z * this.z;
            if (len < Number.EPSILON) {
                return new Vec3();
            }
            // TODO: evaluate use of glm_invsqrt here?
            len = 1.0 / Math.sqrt(len);
            return new Vec3(this.x * len, this.y * len, this.z * len);
        }
        /**
         * Normalizes this Vec3 multiplying coordinate values by the inverse of the vector length.
         *
         * @return - The return value.
         */
        normalizeInPlace() {
            let len = this.x * this.x + this.y * this.y + this.z * this.z;
            if (len < Number.EPSILON) {
                return;
            }
            len = Math.sqrt(len);
            const tmp = 1.0 / len;
            this.x *= tmp;
            this.y *= tmp;
            this.z *= tmp;
            return len;
        }
        /**
         * Creates and returns a new Vec3 with the new coordinates(calculated with this Vec3 coordinates and the specified length).
         *
         * @param length - The length value.
         * @return - The return value.
         */
        resize(length) {
            const currLen = this.x * this.x + this.y * this.y + this.z * this.z;
            if (currLen < Number.EPSILON) {
                return;
            }
            const scl = length / Math.sqrt(currLen);
            return new Vec3(this.x * scl, this.y * scl, this.z * scl);
        }
        /**
         * Modifies current coordinates using the specified length.
         *
         * @param length - The length value.
         */
        resizeInPlace(length) {
            const currLen = this.x * this.x + this.y * this.y + this.z * this.z;
            if (currLen < Number.EPSILON) {
                return;
            }
            const scl = length / Math.sqrt(currLen);
            this.x *= scl;
            this.y *= scl;
            this.z *= scl;
        }
        /**
         * Calculates the dot product of this Vec3 against another Vec3.
         *
         * @param other - The other Vec3 to compare with.
         * @return - Returns the dot product.
         */
        dot(other) {
            return this.x * other.x + this.y * other.y + this.z * other.z;
        }
        /**
         * Calculates the cross product of two Vec3s and returns the result as a new Vec3.
         *
         * @param other - The other Vec3 to calculate with.
         * @return - Returns the cross product as a new Vec3.
         */
        cross(other) {
            const ax = this.x;
            const ay = this.y;
            const az = this.z;
            const bx = other.x;
            const by = other.y;
            const bz = other.z;
            return new Vec3(ay * bz - az * by, az * bx - ax * bz, ax * by - ay * bx);
        }
        /**
         * Gets the angle between this Vec3 and b.
         *
         * @param other - The other Vec3 to compare with.
         * @return - Returns the angle in radians.
         */
        angleTo(other) {
            const cosine = this.dot(other);
            if (cosine > 1.0) {
                return 0;
            }
            else {
                return Math.acos(cosine);
            }
        }
        /**
         * Performs a linear interpolation between this Vec3 and other.
         *
         * @param other - The other Vec3 to interpolate towards.
         * @param t - Interpolation ratio.
         * @return - Returns a new Vec3.
         */
        lerp(other, t) {
            const ax = this.x;
            const ay = this.y;
            const az = this.z;
            return new Vec3(ax + t * (other.x - ax), ay + t * (other.y - ay), az + t * (other.z - az));
        }
        /**
         * Returns a new Vec3 whose component values are the abs of this Vec3s component values.
         *
         * @return - Returns a new Vec3.
         */
        abs() {
            return new Vec3(Math.abs(this.x), Math.abs(this.y), Math.abs(this.z));
        }
        /**
         * Sets the vector a random vector on the surface of a sphere with the radius of the given scale value.
         *
         * @param scale - The radius of the surface sphere.
         * @return - The random Vec3.
         */
        setRandomDir(scale = 1.0) {
            const r = Math.random() * 2.0 * Math.PI;
            const z = Math.random() * 2.0 - 1.0;
            const zScale = Math.sqrt(1.0 - z * z) * scale;
            this.x = Math.cos(r) * zScale;
            this.y = Math.sin(r) * zScale;
            this.z = z * scale;
            return this;
        }
        /**
         * Generates a random vector anywhere in the sphere defined by the provided scale value.
         *
         * @param scale - The radius of the bounding sphere.
         * @return - The random Vec3.
         */
        setRandom(scale = 1.0) {
            this.x = (Math.random() - 0.5) * scale;
            this.y = (Math.random() - 0.5) * scale;
            this.z = (Math.random() - 0.5) * scale;
            return this;
        }
        /**
         * Clones this Vec3 and returns a new Vec3.
         *
         * @return - Returns a new Vec3.
         */
        clone() {
            return new Vec3(this.x, this.y, this.z);
        }
        /**
         * Returns the type as an array. Often used to pass types to the GPU.
         *
         * @return - Returns as an array.
         */
        asArray() {
            return [this.x, this.y, this.z];
        }
        /**
         * Setter from an array.
         */
        fromArray(vals) {
            this.x = vals[0];
            this.y = vals[1];
            this.z = vals[1];
        }
        // ///////////////////////////
        // Persistence
        /**
         * Converts this Vec3 to a string in JSON format.
         *
         * @return - The return value.
         */
        toString() {
            // eslint-disable-next-line new-cap
            return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON());
        }
        /**
         * Encodes Vec3 Class as a JSON object for persistence.
         *
         * @return - The json object.
         */
        toJSON() {
            return {
                x: this.x,
                y: this.y,
                z: this.z,
            };
        }
        /**
         * Decodes a JSON object to set the state of this class.
         *
         * @param j - The json object.
         */
        fromJSON(j) {
            this.x = j.x;
            this.y = j.y;
            this.z = j.z;
        }
        /**
         * Loads the state of the value from a binary reader.
         *
         * @param reader - The reader value.
         */
        readBinary(reader) {
            this.x = reader.loadFloat32();
            this.y = reader.loadFloat32();
            this.z = reader.loadFloat32();
        }
        isValid() {
            for (const v of this.asArray()) {
                if (v == Infinity || isNaN(v))
                    return false;
            }
            return true;
        }
    }

    /* eslint-disable new-cap */
    /**
     * Represents a four-dimensional coordinate.
     * Math types internally store values in {@link https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Float32Array|Float32Array} and
     * expose getters and setters for the component values.
     *
     */
    class Vec4 {
        x;
        y;
        z;
        w;
        /**
         * Creates a Vec4.
         *
         */
        constructor(x = 0, y = 0, z = 0, w = 0) {
            this.x = x;
            this.y = y;
            this.z = z;
            this.w = w;
        }
        /**
         * Getter for `xyz` swizzel.
         *
         * @return - Returns the z value.
         */
        get xyz() {
            return new Vec3(this.x, this.y, this.z);
        }
        /**
         * Setter from scalar components.
         *
         * @param x - The x value.
         * @param y  - The y value.
         * @param z  - The y value.
         * @param w  - The w value.
         */
        set(x, y, z, w) {
            this.x = x;
            this.y = y;
            this.z = z;
            this.w = w;
        }
        /**
         * Sets the state of a Vec4 Object from another Vec4.
         *
         * @param other - The other Vec4 to set from.
         */
        setFromOther(other) {
            this.x = other.x;
            this.y = other.y;
            this.z = other.z;
            this.w = other.w;
        }
        /**
         * Checks if this Vec4 contains the same values as the other Vec4.
         *
         * @param other - The other Vec4 to compare with.
         * @return - Returns true or false.
         */
        isEqual(other) {
            return this.x == other.x && this.y == other.y && this.z == other.z && this.w == other.w;
        }
        /**
         * Checks if this Vec4 is different from another Vec4.
         *
         * @param other - The other Vec4 to compare with.
         * @return - Returns true or false.
         */
        notEqual(other) {
            return this.x != other.x && this.y != other.y && this.z != other.z && this.w != other.w;
        }
        /**
         * Returns true if this Vec4 is approximately the same as other.
         *
         * @param other - The other Vec4 to compare with.
         * @param precision - The precision to which the values must match.
         * @return - The return value.
         */
        approxEqual(other, precision = Number.EPSILON) {
            return (Math.abs(this.x - other.x) < precision &&
                Math.abs(this.y - other.y) < precision &&
                Math.abs(this.z - other.z) < precision &&
                Math.abs(this.w - other.w) < precision);
        }
        /**
         * Adds other to this Vec4 and returns the result as a new Vec4.
         *
         * @param other - The other Vec4 to add.
         * @return - Returns a new Vec4.
         */
        add(other) {
            return new Vec4(this.x + other.x, this.y + other.y, this.z + other.z, this.w + other.w);
        }
        /**
         * Adds other to this Vec4 mutating the values of this instance
         *
         * @param other - The other Vec4 to add.
         */
        addInPlace(other) {
            this.x += other.x;
            this.y += other.y;
            this.z += other.z;
            this.w += other.w;
        }
        /**
         * Subtracts other from this Vec4 and returns then result as a new Vec4.
         *
         * @param other - The other Vec4 to subtract.
         * @return - Returns a new Vec4.
         */
        subtract(other) {
            return new Vec4(this.x - other.x, this.y - other.y, this.z - other.z, this.w - other.w);
        }
        /**
         * Subtracts other from this Vec4 mutating the values of this instance
         *
         * @param other - The other Vec4 to subtract.
         */
        subtractInPlace(other) {
            this.x -= other.x;
            this.y -= other.y;
            this.z -= other.z;
            this.w -= other.w;
        }
        /**
         * Multiplies two Vec4s and returns the result as a new Vec4.
         *
         * @param other - The other Vec4 to multiply with.
         * @return - Returns a new Vec4.
         */
        multiply(other) {
            return new Vec4(this.x * other.x, this.y * other.y, this.z * other.z, this.w * other.w);
        }
        /**
         * Multiplies two Vec4s mutating the values of this instance
         *
         * @param other - The other Vec4 to multiply with.
         */
        multiplyInPlace(other) {
            this.x *= other.x;
            this.y *= other.y;
            this.z *= other.z;
            this.w *= other.w;
        }
        /**
         * Divides two Vec4s and returns the result as a new Vec4.
         *
         * @param other - The other Vec4 to divide by.
         * @return - Returns a new Vec4.
         */
        divide(other) {
            return new Vec4(this.x / other.x, this.y / other.y, this.z / other.z, this.w / other.w);
        }
        /**
         * Divides two Vec4s.
         *
         * @param other - The other Vec4 to divide by.
         */
        divideInPlace(other) {
            this.x /= other.x;
            this.y /= other.y;
            this.z /= other.z;
            this.w /= other.w;
        }
        /**
         * Scales this Vec4 by scalar and returns the result as a new Vec4.
         *
         * @param scalar - The scalar value.
         * @return - The return value.
         */
        scale(scalar) {
            return new Vec4(this.x * scalar, this.y * scalar, this.z * scalar, this.w * scalar);
        }
        /**
         * Scales this Vec4 by scalar.
         *
         * @param scalar - The scalar value.
         */
        scaleInPlace(scalar) {
            this.set(this.x * scalar, this.y * scalar, this.z * scalar, this.w * scalar);
        }
        /**
         * Calculates the length of this Vec4.
         *
         * @return - Returns the length.
         */
        length() {
            const x = this.x;
            const y = this.y;
            const z = this.z;
            const w = this.w;
            return Math.sqrt(x * x + y * y + z * z + w * w);
        }
        /**
         * Calculates the squared length of this Vec4.
         *
         * @return - Returns the length.
         */
        lengthSquared() {
            const x = this.x;
            const y = this.y;
            const z = this.z;
            const w = this.w;
            return x * x + y * y + z * z + w * w;
        }
        /**
         * Normalizes the Vec4 and returns it as a new Vec4.
         * Multiplies coordinates value by the inverse of the vector length.
         *
         * @return - Returns the Vec4 normalized.
         */
        normalize() {
            const x = this.x;
            const y = this.y;
            const z = this.z;
            const w = this.w;
            let len = x * x + y * y + z * z + w * w;
            if (len < Number.EPSILON) {
                return new Vec4();
            }
            // TODO: evaluate use of glm_invsqrt here?
            len = 1 / Math.sqrt(len);
            return new Vec4(x * len, y * len, z * len);
        }
        /**
         * Normalizes this Vec4 multiplying coordinate values by the inverse of the vector length.
         */
        normalizeInPlace() {
            const x = this.x;
            const y = this.y;
            const z = this.z;
            const w = this.w;
            let len = x * x + y * y + z * z + w * w;
            if (len < Number.EPSILON) {
                return;
            }
            len = 1 / Math.sqrt(len);
            this.set(x * len, y * len, z * len, w * len);
        }
        /**
         * Calculates the dot product of this Vec4 against another Vec4.
         *
         * @param other - The other Vec4 to compare with.
         * @return - Returns the dot product.
         */
        dot(other) {
            return this.x * other.x + this.y * other.y + this.z * other.z + this.w * other.w; // TODO: other.w used to be b.w?
        }
        /**
         * Calculates the cross product of two Vec4s and returns the result as a new Vec4.
         *
         * @param other - The other Vec4 to calculate with.
         * @return - Returns the cross product as a new Vec4.
         */
        cross(other) {
            const ax = this.x;
            const ay = this.y;
            const az = this.z;
            const at = this.w;
            const bx = other.x;
            const by = other.y;
            const bz = other.z;
            const bt = other.w;
            return new Vec4(ay * bz - az * by, az * bt - at * bz, at * bx - ax * bt, ax * by - ay * bx);
        }
        /**
         * Gets the angle between this Vec4 and b.
         *
         * @param other - The other Vec4 to compare with.
         * @return - Returns the angle in radians.
         */
        angleTo(other) {
            const tempA = this.normalize();
            const tempB = other.normalize();
            const cosine = tempA.dot(tempB);
            if (cosine > 1.0) {
                return 0;
            }
            else {
                return Math.acos(cosine);
            }
        }
        /**
         * Performs a linear interpolation between this Vec4 and other.
         *
         * @param other - The other Vec4 to interpolate between.
         * @param w - Interpolation amount between the two inputs.
         * @return - Returns a new Vec4.
         */
        lerp(other, t) {
            const ax = this.x;
            const ay = this.y;
            const az = this.z;
            const at = this.w;
            return new Vec4(ax + t * (other.x - ax), ay + t * (other.y - ay), az + t * (other.z - az), at + t * (other.w - at));
        }
        /**
         * Generates a random vector with the given scale.
         *
         * @param scale - Length of the resulting vector. If omitted, a unit vector will be returned.
         * @return - The return value.
         */
        // random(scale = 1.0) {
        //   const r = glMatrix.RANDOM() * 2.0 * Math.PI
        //   const z = glMatrix.RANDOM() * 2.0 - 1.0
        //   const zScale = Math.sqrt(1.0 - z * z) * scale
        //   out[0] = Math.cos(r) * zScale
        //   out[1] = Math.sin(r) * zScale
        //   out[2] = z * scale
        //   return out
        // }
        /**
         * Clones this Vec4 and returns a new Vec4.
         *
         * @return - Returns a new Vec4.
         */
        clone() {
            return new Vec4(this.x, this.y, this.z, this.w);
        }
        /**
         * Converts this Vec4 into a Vec3.
         *
         * @return - Returns the value as a new Vec3.
         */
        toVec3() {
            return new Vec3(this.x, this.y, this.z);
        }
        /**
         * Returns the type as an array. Often used to pass types to the GPU.
         *
         * @return - Returns as an array.
         */
        asArray() {
            return [this.x, this.y, this.z, this.w];
        }
        /**
         * Setter from an array.
         */
        fromArray(vals) {
            this.x = vals[0];
            this.y = vals[1];
            this.z = vals[1];
            this.w = vals[1];
        }
        // ///////////////////////////
        // Persistence
        /**
         * Converts this Vec3 to a string in JSON format.
         *
         * @return - The return value.
         */
        toString() {
            // eslint-disable-next-line new-cap
            return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON());
        }
        /**
         * The toJSON method encodes this type as a json object for persistence.
         * @return - The json object.
         */
        toJSON() {
            return {
                x: this.x,
                y: this.y,
                z: this.z,
                w: this.w,
            };
        }
        /**
         * Decodes a JSON object to set the state of this class.
         *
         * @param j - The json object.
         */
        fromJSON(j) {
            this.x = j.x;
            this.y = j.y;
            this.z = j.z;
            this.w = j.w;
        }
        /**
         * Loads the state of the value from a binary reader.
         *
         * @param reader - The reader value.
         */
        readBinary(reader) {
            this.x = reader.loadFloat32();
            this.y = reader.loadFloat32();
            this.z = reader.loadFloat32();
            this.w = reader.loadFloat32();
        }
        /**
         * Verifies if the values stored in this Math type are valid numeric values.
         * Returns `false` If at least one of the values is either {@link https://developer.mozilla.org/en-US/docs/Web/JavaScript/Referencia/Objetos_globales/Infinity|Infinity} or
         * {@link https://developer.mozilla.org/en-US/docs/Web/JavaScript/Referencia/Objetos_globales/NaN|NaN}.
         *
         * @return - Returns the result as a boolean.
         */
        isValid() {
            for (const v of this.asArray()) {
                if (v == Infinity || isNaN(v))
                    return false;
            }
            return true;
        }
    }

    let registeredClasses = {};
    let classNames = {};
    let classDefinitions = [];
    /**
     * Registry is a static factory that handles registration/reconstruction of
     * classes bases on BaseClass. Registered classes can then be constructed by the Registry by name.
     *
     * Note: className is required because on minification process
     * the name of classes change and we can't simply use '....constructor.name'.
     * So, we need a way of relating minified class names to the one stored for persistency.
     *
     * i.e.
     * ```javascript
     * // Import registry class
     * class Foo() extends BaseClass {}
     *
     * Registry.register('Foo', Foo)
     * // In case 'Foo' class gets its name changed to 'c' on minification,
     * // and the persisted data type is 'Foo', we would know how to relate them.
     * ```
     *
     * @static
     * @class Registry
     */
    class Registry {
        /**
         * Registers a new class to the factory.
         *
         * @param className - Name of the registered class
         * @param classDef - Class representation(Class function, type)
         */
        static register(className, classDef) {
            if (className in registeredClasses) {
                console.warn(`There's a class registered with '${className}' name. Second registration failed.`);
                return;
            }
            // Note: To provide backwards compatibility, same classDef can be stored under multiple names.
            // Thats the reason behind using indexes instead of the classDef.
            const index = classDefinitions.length;
            classDefinitions.push(classDef);
            classNames[index] = className;
            registeredClasses[className] = index;
        }
        /**
         * Returns class definition using the name it was registered with.
         *
         * @param className - Name of the registered class
         * @return - Class representation(Class function, type)
         */
        static getClassDefinition(className) {
            if (!(className in registeredClasses))
                throw new Error(`${className} class is not registered`);
            return classDefinitions[registeredClasses[className]];
        }
        /**
         * Returns class name registered for the instantiated object.
         * @param classDefinition - Class type definition.
         * @return - Name of the registered class
         */
        static getClassName(classDefinition) {
            const classId = classDefinitions.indexOf(classDefinition);
            if (classId >= 0 && classNames[classId])
                return classNames[classId];
            throw new Error(`class is not registered`);
        }
        /**
         * The factory function that construct the class registered under the given name.
         *
         * @param className - Name of the registered class
         * @return - Instantiated object of the specified class
         */
        static constructClass(className) {
            const classDefinition = classDefinitions[registeredClasses[className]];
            if (!classDefinition)
                throw new Error(`${className} class is not registered`);
            return new classDefinition();
        }
        /**
         * For testing purpose only, never call this outside of the test scope.
         *
         * @private
         */
        static flush() {
            registeredClasses = {};
            classNames = {};
            classDefinitions = [];
        }
    }

    /* eslint-disable @typescript-eslint/no-explicit-any */
    let counter = 0;
    /**
     * Class representing a BaseClass.
     * The BaseClass is the foundation class of the SceneTree, as almost all classes derive from it.
     */
    class BaseClass {
        __id;
        /**
         * Create an BaseClass.
         */
        constructor() {
            this.__id = ++counter;
        }
        /**
         * Every instance of each class based on BaseClass is assigned a unique number.
         * This number is not persistent in between different loads of a scene.
         * Returns the unique id of the object.
         * @return - The Id of the object.
         */
        getId() {
            return this.__id;
        }
        /**
         * Returns the unmangled name of the class.
         * @return - The name of the class definition.
         */
        getClassName() {
            return Registry.getClassName(Object.getPrototypeOf(this).constructor);
        }
    }

    /** Class representing a BaseEvent. */
    class BaseEvent {
        /**
         * Create an BaseEvent.
         */
        constructor() { }
    }

    /* eslint-disable @typescript-eslint/no-explicit-any */
    /**
     * Provides an interface for emitting events under given names, and registering listeners to those events.
     * This is a base class for most classes in the Scene Tree and Renderer, enabling observers to listen to changes throughout the system.
     * The interface exposed is similar to [EventEmitter](https://nodejs.org/api/events.html#events_class_eventemitter) in Node.
     *
     * Similar to how the DOM event system in the browser works, events are registered by name.
     * Example: Registering a listener for a custom event, and then emitting that event.
     * ```javascript
     *  const ee = new EventEmitter()
     *
     *  const eventID = ee.on('myEvent', (event) => {
     *    console.log('My Event was emitted:', event)
     *  })
     *
     *  ee.emit('myEvent', { data: 42 })
     *  // We no longer want to listen to this event, so let's remove the listener.
     *  ee.off('myEvent', eventID)
     * ```
     *
     *
     */
    class EventEmitter extends BaseClass {
        listeners = {};
        /**
         * Initializes an empty `listeners` map that will host all the events,
         * which implies that it doesn't allow multiple events with the same name.
         *
         */
        constructor() {
            super();
        }
        /**
         * Adds a listener function for a given event name.
         *
         * @param eventName - The name of the event.
         * @param listener - The listener function(callback).
         * @return - the id that can be used to remove the listener.
         */
        on(eventName, listener) {
            if (!listener) {
                throw new Error('Missing listener.');
            }
            if (!this.listeners[eventName]) {
                this.listeners[eventName] = [];
            }
            const listeners = this.listeners[eventName];
            if (listeners.includes(listener)) {
                throw new Error(`Listener "${listener.name}" already connected to event "${eventName}".`);
            }
            // TODO: Deprecate alongside #addListener.
            const id = listeners.length;
            listeners[id] = listener;
            return id;
        }
        /**
         * Similar to the `on` method with the difference that when the event is triggered,
         * it is automatically unregistered meaning that the event listener will be triggered at most one time.
         *
         * Useful for events that we expect to trigger one time, such as when assets load.
         * ```javascript
         * const asset = new Asset();
         * asset.once('loaded', () => {
         *   console.log("Yay! the asset is loaded")
         * })
         * ```
         *
         * @param eventName - The eventName value
         * @param listener - The listener value
         * @return - the id that can be used to remove the listener.
         */
        once(eventName, listener) {
            const cb = (event) => {
                this.off(eventName, cb);
                listener(event);
            };
            return this.on(eventName, cb);
        }
        /**
         * Removes a listener from the specified event, using either the function or the index id. Depends on what is passed in.
         *
         * @param eventName - The name of the event.
         * @param listenerOrId - The listener function or the id number returned by 'on'.
         */
        off(eventName, listenerOrId) {
            if (listenerOrId == undefined) {
                throw new Error('Missing callback function (listener).');
            }
            const listeners = this.listeners[eventName] || [];
            if (typeof listenerOrId == 'number') {
                const id = listenerOrId;
                // Note: do not splice the array as that would change the indexes of existing listeners.
                listeners[id] = null;
                return;
            }
            const listener = listenerOrId;
            listeners.forEach((e, i) => {
                if (e === listener) {
                    // Note: do not splice the array as that would change the indexes of existing listeners.
                    listeners[i] = null;
                }
            });
        }
        /**
         * remove listener by ID returned from #on
         *
         * @param eventName - The name of the event.
         * @param id - The id returned by addListener
         */
        removeListenerById(eventName, id) {
            this.off(eventName, id);
        }
        /**
         * Triggers all listener functions in an event.
         *
         * @param eventName - The name of the event.
         * @param event - The data you want to pass down to all listener functions as parameter.
         *
         */
        emit(eventName, event = new BaseEvent()) {
            const listeners = this.listeners[eventName] || [];
            listeners.forEach((fn) => {
                // Skip disconnected listeners.
                if (fn) {
                    try {
                        fn(event);
                    }
                    catch (e) {
                        console.warn(e);
                    }
                }
            });
        }
    }

    /* eslint-disable @typescript-eslint/no-explicit-any */
    /**
     * Math Functions
     */
    class MathFunctions {
        /**
         * Converts Radians to Degrees
         *
         * @static
         * @param rad - Radians value
         * @return - Degrees equivalent
         */
        static radToDeg(rad) {
            return rad / (Math.PI / 180);
        }
        /**
         * Converts Degrees to Radiants
         *
         * @static
         * @param deg - Degrees value
         * @return -  Radians equivalent
         */
        static degToRad(deg) {
            return deg * (Math.PI / 180);
        }
        /**
         * Verifies if the specified parameter is numeric.
         *
         * @static
         * @param number - Number to test
         * @return - `true` when is a valid number
         */
        static isNumeric(number) {
            return !isNaN(parseFloat(number)) && isFinite(number);
        }
        /**
         * Generates and returns a random integer within the specified range.
         *
         * @static
         * @param min - Lower value random int can be.
         * @param max - Highest value random int can be.
         * @return - Random number inside range.
         */
        static randomInt(min, max) {
            min = Math.ceil(min);
            max = Math.floor(max);
            return Math.floor(Math.random() * (max - min)) + min;
        }
        /**
         * Calculates a lineal interpolation between two inputs for the specified parameter(t).
         *
         * @static
         * @param v0 -
         * @param v1 -
         * @param t -
         * @return -
         */
        static lerp(v0, v1, t) {
            return v0 + t * (v1 - v0);
        }
        /**
         * Restricts the specified value between two numbers
         *
         * @static
         * @param value
         * @param min
         * @param max
         * @return
         */
        static clamp(value, min, max) {
            return Math.min(Math.max(value, min), max);
        }
        /**
         * Returns the nearest pow of two value of the specified number.
         *
         * @static
         * @param value -
         * @return -
         */
        static nearestPow2(value) {
            return Math.pow(2, Math.round(Math.log(value) / Math.log(2)));
        }
        /**
         * Returns the nearest pow of ten value of the specified number.
         *
         * @static
         * @param value -
         * @return -
         */
        static nearestPow10(value) {
            return Math.pow(10, Math.round(Math.log10(value) / Math.log10(10)));
        }
        /**
         * Returns the next pow of two value of the specified number.
         *
         * @static
         * @param value -
         * @return -
         */
        static nextPow2(value) {
            if (this.fract(Math.log2(value)) == 0) {
                return value;
            }
            let exp = 0;
            while (value > 0) {
                exp++;
                value = value >> 1;
            }
            return 1 << exp;
        }
        /**
         * Returns the fractional component of a number
         *
         * @static
         * @param value -
         * @return -
         */
        static fract(value) {
            if (value == 0)
                return 0;
            if (value < 0) {
                if (value > -1.0)
                    return -value;
                return -value % Math.floor(-value);
            }
            if (value < 1.0)
                return value;
            return value % Math.floor(value);
        }
        /**
         * Moves the specified value from one numeric domain(range) to another.
         *
         * @static
         * @param value -
         * @param start1 -
         * @param end1 -
         * @param start2 -
         * @param end2 -
         * @return -
         */
        static remap(value, start1, end1, start2, end2) {
            return start2 + (end2 - start2) * ((value - start1) / (end1 - start1));
        }
        /**
         * Perform Hermite interpolation between two values
         *
         * @static
         * @param edge0 -
         * @param edge1 -
         * @param x -
         * @return -
         */
        static smoothStep(edge0, edge1, x) {
            const t = this.clamp((x - edge0) / (edge1 - edge0), 0.0, 1.0);
            return t * t * (3.0 - 2.0 * t);
        }
        /**
         * Performs - interpolation between two values
         *
         * @static
         * @param edge0 -
         * @param edge1 -
         * @param x -
         * @return -
         */
        static linStep(edge0, edge1, x) {
            return this.clamp((x - edge0) / (edge1 - edge0), 0.0, 1.0);
        }
        /**
         * Decodes a Float16 from two unsigned Int8
         *
         * @static
         * @param c - Array with the two UInt8
         * @return - Decoded Float16
         */
        static decode16BitFloatFrom2xUInt8(c) {
            const ix = c[0]; // 1st byte: 1 bit signed num, 4 bits exponent, 3 bits mantissa (MSB)
            const iy = c[1]; // 2nd byte: 8 bit mantissa (LSB)
            const s = ix & 0x80 ? 1 : -1; // get bit 8
            const iexp = (ix & 0x78) >> 3; // mask bits 7-4
            const msb = ix & 0x7; // mask bits 3-1
            let norm = iexp == 0 ? 0 : 2048; // distinguish between normalized and sub-normalized numbers
            const mantissa = norm + (msb << 8) + iy; // implicit preceding 1 or 0 added here
            norm = iexp == 0 ? 1 : 0; // normalization toggle
            const exponent = Math.pow(2, iexp + norm - 16); // -5 for the the exponent bias from 2^-5 to 2^10 plus another -11 for the normalized 12 bit mantissa
            const v = s * mantissa * exponent;
            return v;
        }
        /**
         * Encodes an array of two unsigned Int8 to a Float16
         *
         * @static
         * @param v - Float16 number
         * @return - Encoded Unsigned Int8 array
         */
        static encode16BitFloatInto2xUInt8(v) {
            const c = new Uint8Array(2);
            // const c = [0, 0];
            const signum = v >= 0 ? 128 : 0;
            v = Math.abs(v);
            let exponent = 15;
            let limit = 1024; // considering the bias from 2^-5 to 2^10 (==1024)
            for (let exp = 15; exp > 0; exp--) {
                if (v < limit) {
                    limit /= 2;
                    exponent--;
                }
            }
            let rest;
            if (exponent == 0) {
                rest = v / limit / 2; // "sub-normalize" implicit preceding 0.
            }
            else {
                rest = (v - limit) / limit; // normalize accordingly to implicit preceding 1.
            }
            const mantissa = Math.round(rest * 2048); // 2048 = 2^11 for the (split) 11 bit mantissa
            const msb = mantissa / 256; // the most significant 3 bits go into the lower part of the first byte
            const lsb = mantissa - msb * 256; // there go the other 8 bit of the lower significance
            c[0] = signum + exponent * 8 + msb; // color normalization for texture2D
            c[1] = lsb;
            if (v >= 2048) {
                c[0] = 255;
            }
            return c;
        }
        /**
         * Transforms a 16 bit float to an encoded integer.
         *
         * @static
         * @param v - Float16 number to encode
         * @return - Encoded number
         */
        static encode16BitFloat(v) {
            const float32Array = new Float32Array(1);
            float32Array[0] = v;
            const int32View = new Int32Array(float32Array.buffer);
            const toUInt16 = (x) => {
                let bits = (x >> 16) & 0x8000; /* Get the sign */
                let m = (x >> 12) & 0x07ff; /* Keep one extra bit for rounding */
                const e = (x >> 23) & 0xff; /* Using int is faster here */
                /* If zero, or de-normal, or exponent underflows too much for a de-normal
                 * half, return signed zero. */
                if (e < 103) {
                    return bits;
                }
                /* If NaN, return NaN. If Inf or exponent overflow, return Inf. */
                if (e > 142) {
                    bits |= 0x7c00;
                    /* If exponent was 0xff and one mantissa bit was set, it means NaN,
                     * not Inf, so make sure we set one mantissa bit too. */
                    bits |= (e == 255 ? 0 : 1) && x & 0x007fffff;
                    return bits;
                }
                /* If exponent underflows but not too much, return a de-normal */
                if (e < 113) {
                    m |= 0x0800;
                    /* Extra rounding may overflow and set mantissa to 0 and exponent
                     * to 1, which is OK. */
                    bits |= (m >> (114 - e)) + ((m >> (113 - e)) & 1);
                    return bits;
                }
                bits |= ((e - 112) << 10) | (m >> 1);
                /* Extra rounding. An overflow will set mantissa to 0 and increment
                 * the exponent, which is OK. */
                bits += m & 1;
                return bits;
            };
            return toUInt16(int32View[0]);
        }
        /**
         * As opposite of the `encode16BitFloat` method, this takes an encoded integer value,
         * and returns the 16 bit float.
         *
         * @static
         * @param h - Encoded integer
         * @return - Decoded 16 bit float.
         */
        static decode16BitFloat(h) {
            const s = (h & 0x8000) >> 15;
            const e = (h & 0x7c00) >> 10;
            const f = h & 0x03ff;
            if (e == 0) {
                return (s ? -1 : 1) * Math.pow(2, -14) * (f / Math.pow(2, 10));
            }
            else if (e == 0x1f) {
                return f ? NaN : (s ? -1 : 1) * Infinity;
            }
            return (s ? -1 : 1) * Math.pow(2, e - 15) * (1 + f / Math.pow(2, 10));
        }
        /**
         * Transforms an array of Float 32 to an array of unsigned Int16.
         *
         * @static
         * @param float32Array -
         * @return - Unsigned Int16 array representative of the Float32Array
         */
        static convertFloat32ArrayToUInt16Array(float32Array) {
            const unit16s = new Uint16Array(float32Array.length);
            const int32View = new Int32Array(float32Array.buffer);
            const toUInt16 = (x) => {
                let bits = (x >> 16) & 0x8000; /* Get the sign */
                let m = (x >> 12) & 0x07ff; /* Keep one extra bit for rounding */
                const e = (x >> 23) & 0xff; /* Using int is faster here */
                /* If zero, or de-normal, or exponent underflows too much for a de-normal
                 * half, return signed zero. */
                if (e < 103) {
                    return bits;
                }
                /* If NaN, return NaN. If Inf or exponent overflow, return Inf. */
                if (e > 142) {
                    bits |= 0x7c00;
                    /* If exponent was 0xff and one mantissa bit was set, it means NaN,
                     * not Inf, so make sure we set one mantissa bit too. */
                    bits |= (e == 255 ? 0 : 1) && x & 0x007fffff;
                    return bits;
                }
                /* If exponent underflows but not too much, return a de-normal */
                if (e < 113) {
                    m |= 0x0800;
                    /* Extra rounding may overflow and set mantissa to 0 and exponent
                     * to 1, which is OK. */
                    bits |= (m >> (114 - e)) + ((m >> (113 - e)) & 1);
                    return bits;
                }
                bits |= ((e - 112) << 10) | (m >> 1);
                /* Extra rounding. An overflow will set mantissa to 0 and increment
                 * the exponent, which is OK. */
                bits += m & 1;
                return bits;
            };
            for (let i = 0; i < float32Array.length; i++) {
                unit16s[i] = toUInt16(int32View[i]);
            }
            return unit16s;
        }
    }

    class ChildAddedEvent extends BaseEvent {
        index;
        childItem;
        constructor(index, childItem) {
            super();
            this.index = index;
            this.childItem = childItem;
        }
    }

    class NameChangedEvent extends BaseEvent {
        oldName;
        newName;
        constructor(oldName, newName) {
            super();
            this.oldName = oldName;
            this.newName = newName;
        }
    }

    class OpacityStateChangedEvent extends BaseEvent {
        isOpaque;
        isOpaqueStateChanged;
        constructor(isOpaque, isOpaqueStateChanged) {
            super();
            this.isOpaque = isOpaque;
            this.isOpaqueStateChanged = isOpaqueStateChanged;
        }
    }

    class ParameterAddedEvent extends BaseEvent {
        name;
        constructor(name) {
            super();
            this.name = name;
        }
    }

    class ParameterRemovedEvent extends BaseEvent {
        name;
        constructor(name) {
            super();
            this.name = name;
        }
    }

    class SelectedEvent extends BaseEvent {
        selected;
        constructor(selected) {
            super();
            this.selected = selected;
        }
    }

    class StateChangedEvent extends BaseEvent {
        state;
        constructor(state) {
            super();
            this.state = state;
        }
    }

    /**
     * Class representing a color as 4 floating point values.
     */
    class Color {
        r = 0;
        g = 0;
        b = 0;
        a = 255;
        /**
         * Creates a `Color` object with an RGBA structure.
         *
         * @param r - The red channel of a color.
         * @param g - The green channel of a color.
         * @param b - The blue channel of a color.
         * @param a - The alpha (transparency) channel of a color.
         */
        constructor(r = 0, g = 0, b = 0, a = 1.0) {
            if (typeof r == 'string') {
                if (r.startsWith('#')) {
                    this.setFromHex(r);
                }
                else {
                    this.setFromCSSColorName(r);
                }
            }
            else {
                this.r = r;
                this.g = g;
                this.b = b;
                this.a = a;
            }
        }
        /**
         * Setter from scalar components.
         *
         * @param r - The red channel.
         * @param g  - The green channel.
         * @param b  - The blue channel.
         * @param a  - The alpha channel.
         */
        set(r, g, b, a = 1.0) {
            this.r = r;
            this.g = g;
            this.b = b;
            this.a = a;
        }
        /**
         * Sets current color state with another `Color` object.
         *
         * @param other - The other color to set from.
         */
        setFromOther(other) {
            this.r = other.r;
            this.g = other.g;
            this.b = other.b;
            this.a = other.a;
        }
        /**
         * Getter from an RGB array.
         *
         * @return - The return value.
         */
        getAsRGBArray() {
            return [this.r * 255, this.g * 255, this.b * 255];
        }
        /**
         * Getter from an RGB dict.
         *
         * @return - The return value.
         */
        getAsRGBDict() {
            return {
                r: this.r * 255,
                g: this.g * 255,
                b: this.b * 255,
            };
        }
        /**
         * Setter from a RGB value.
         *
         * @param r - The red channel.
         * @param g  - The green channel.
         * @param b  - The blue channel.
         * @param a  - The alpha channel.
         */
        setFromRGB(r, g, b, a) {
            this.r = r / 255;
            this.g = g / 255;
            this.b = b / 255;
            this.a = a ? a / 255 : 1.0;
        }
        /**
         * Setter from an RGB dict.
         *
         * @param vals - The vals param.
         */
        setFromRGBDict(vals) {
            this.r = vals.r / 255;
            this.g = vals.g / 255;
            this.b = vals.b / 255;
            this.a = vals.a == 4 ? vals.a / 255 : 1.0;
        }
        /**
         * Setter from a hexadecimal value.
         * E.g. #ff0000
         * @param hex - The hex value.
         */
        setFromHex(hex) {
            function hexToRgb(hex) {
                const result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex);
                return result
                    ? {
                        r: parseInt(result[1], 16),
                        g: parseInt(result[2], 16),
                        b: parseInt(result[3], 16),
                    }
                    : null;
            }
            const rgb = hexToRgb(hex);
            if (!rgb) {
                console.warn('Invalid hex code:' + hex);
                return;
            }
            this.setFromRGB(rgb.r, rgb.g, rgb.b);
        }
        /**
         * Sets the Color values from a CSS color name.
         * E.g. "red"
         * @param name - The CSS color name.
         */
        setFromCSSColorName(name) {
            const colourNameToHex = (colour) => {
                const colors = {
                    aliceblue: '#f0f8ff',
                    antiquewhite: '#faebd7',
                    aqua: '#00ffff',
                    aquamarine: '#7fffd4',
                    azure: '#f0ffff',
                    beige: '#f5f5dc',
                    bisque: '#ffe4c4',
                    black: '#000000',
                    blanchedalmond: '#ffebcd',
                    blue: '#0000ff',
                    blueviolet: '#8a2be2',
                    brown: '#a52a2a',
                    burlywood: '#deb887',
                    cadetblue: '#5f9ea0',
                    chartreuse: '#7fff00',
                    chocolate: '#d2691e',
                    coral: '#ff7f50',
                    cornflowerblue: '#6495ed',
                    cornsilk: '#fff8dc',
                    crimson: '#dc143c',
                    cyan: '#00ffff',
                    darkblue: '#00008b',
                    darkcyan: '#008b8b',
                    darkgoldenrod: '#b8860b',
                    darkgray: '#a9a9a9',
                    darkgreen: '#006400',
                    darkkhaki: '#bdb76b',
                    darkmagenta: '#8b008b',
                    darkolivegreen: '#556b2f',
                    darkorange: '#ff8c00',
                    darkorchid: '#9932cc',
                    darkred: '#8b0000',
                    darksalmon: '#e9967a',
                    darkseagreen: '#8fbc8f',
                    darkslateblue: '#483d8b',
                    darkslategray: '#2f4f4f',
                    darkturquoise: '#00ced1',
                    darkviolet: '#9400d3',
                    deeppink: '#ff1493',
                    deepskyblue: '#00bfff',
                    dimgray: '#696969',
                    dodgerblue: '#1e90ff',
                    firebrick: '#b22222',
                    floralwhite: '#fffaf0',
                    forestgreen: '#228b22',
                    fuchsia: '#ff00ff',
                    gainsboro: '#dcdcdc',
                    ghostwhite: '#f8f8ff',
                    gold: '#ffd700',
                    goldenrod: '#daa520',
                    gray: '#808080',
                    green: '#008000',
                    greenyellow: '#adff2f',
                    honeydew: '#f0fff0',
                    hotpink: '#ff69b4',
                    'indianred ': '#cd5c5c',
                    indigo: '#4b0082',
                    ivory: '#fffff0',
                    khaki: '#f0e68c',
                    lavender: '#e6e6fa',
                    lavenderblush: '#fff0f5',
                    lawngreen: '#7cfc00',
                    lemonchiffon: '#fffacd',
                    lightblue: '#add8e6',
                    lightcoral: '#f08080',
                    lightcyan: '#e0ffff',
                    lightgoldenrodyellow: '#fafad2',
                    lightgrey: '#d3d3d3',
                    lightgreen: '#90ee90',
                    lightpink: '#ffb6c1',
                    lightsalmon: '#ffa07a',
                    lightseagreen: '#20b2aa',
                    lightskyblue: '#87cefa',
                    lightslategray: '#778899',
                    lightsteelblue: '#b0c4de',
                    lightyellow: '#ffffe0',
                    lime: '#00ff00',
                    limegreen: '#32cd32',
                    linen: '#faf0e6',
                    magenta: '#ff00ff',
                    maroon: '#800000',
                    mediumaquamarine: '#66cdaa',
                    mediumblue: '#0000cd',
                    mediumorchid: '#ba55d3',
                    mediumpurple: '#9370d8',
                    mediumseagreen: '#3cb371',
                    mediumslateblue: '#7b68ee',
                    mediumspringgreen: '#00fa9a',
                    mediumturquoise: '#48d1cc',
                    mediumvioletred: '#c71585',
                    midnightblue: '#191970',
                    mintcream: '#f5fffa',
                    mistyrose: '#ffe4e1',
                    moccasin: '#ffe4b5',
                    navajowhite: '#ffdead',
                    navy: '#000080',
                    oldlace: '#fdf5e6',
                    olive: '#808000',
                    olivedrab: '#6b8e23',
                    orange: '#ffa500',
                    orangered: '#ff4500',
                    orchid: '#da70d6',
                    palegoldenrod: '#eee8aa',
                    palegreen: '#98fb98',
                    paleturquoise: '#afeeee',
                    palevioletred: '#d87093',
                    papayawhip: '#ffefd5',
                    peachpuff: '#ffdab9',
                    peru: '#cd853f',
                    pink: '#ffc0cb',
                    plum: '#dda0dd',
                    powderblue: '#b0e0e6',
                    purple: '#800080',
                    rebeccapurple: '#663399',
                    red: '#ff0000',
                    rosybrown: '#bc8f8f',
                    royalblue: '#4169e1',
                    saddlebrown: '#8b4513',
                    salmon: '#fa8072',
                    sandybrown: '#f4a460',
                    seagreen: '#2e8b57',
                    seashell: '#fff5ee',
                    sienna: '#a0522d',
                    silver: '#c0c0c0',
                    skyblue: '#87ceeb',
                    slateblue: '#6a5acd',
                    slategray: '#708090',
                    snow: '#fffafa',
                    springgreen: '#00ff7f',
                    steelblue: '#4682b4',
                    tan: '#d2b48c',
                    teal: '#008080',
                    thistle: '#d8bfd8',
                    tomato: '#ff6347',
                    turquoise: '#40e0d0',
                    violet: '#ee82ee',
                    wheat: '#f5deb3',
                    white: '#ffffff',
                    whitesmoke: '#f5f5f5',
                    yellow: '#ffff00',
                    yellowgreen: '#9acd32',
                };
                return colors[colour.toLowerCase()];
                //  if (typeof colors[colour.toLowerCase()] != 'undefined') return colors[colour.toLowerCase()]
                // return false
            };
            if (name.startsWith('#')) {
                this.setFromHex(name);
            }
            else {
                const hexColor = colourNameToHex(name);
                if (hexColor)
                    this.setFromHex(hexColor);
            }
        }
        /**
         * Returns the hexadecimal value of this color, including the leading "#" character.
         *
         * @return - Returns the hex value.
         */
        toHex() {
            function componentToHex(c) {
                const int = Math.round(c * 255);
                const hex = int.toString(16);
                return hex.length == 1 ? '0' + hex : hex;
            }
            return '#' + componentToHex(this.r) + componentToHex(this.g) + componentToHex(this.b);
        }
        /**
         * Checks if this Color  contains the same values as the other.
         *
         * @param other - The other Color to compare with.
         * @return - Returns `true` if the values are the same, otherwise, `false`.
         */
        isEqual(other) {
            return this.r == other.r && this.g == other.g && this.b == other.b && this.a == other.a;
        }
        /**
         * Returns true if this color is NOT exactly the same as other.
         *
         * @param other - The other color to compare with.
         * @return - Returns true or false.
         */
        notEquals(other) {
            return this.r != other.r && this.g != other.g && this.b != other.b && this.a != other.a;
        }
        /**
         * Returns true if this color is approximately the same as other.
         *
         * @param other - The other color to compare with.
         * @param precision - The precision to which the values must match.
         * @return - Returns true or false.
         */
        approxEqual(other, precision = Number.EPSILON) {
            return (Math.abs(this.r - other.r) < precision &&
                Math.abs(this.g - other.g) < precision &&
                Math.abs(this.b - other.b) < precision &&
                Math.abs(this.a - other.a) < precision);
        }
        /**
         * Returns a new Color which is this Color added to other.
         *
         * @param other - The other color to add.
         * @return - Returns a new color.
         */
        add(other) {
            return new Color(this.r + other.r, this.g + other.g, this.b + other.b, this.a + other.a);
        }
        /**
         * Updates this Color by adding the values from the other color.
         *
         * @param other - The other color to add.
         */
        addInPlace(other) {
            this.r += other.r;
            this.g += other.g;
            this.b += other.b;
            this.a += other.a;
        }
        /**
         * Returns a new color which is this color subtracted from other.
         *
         * @param other - The other color to subtract.
         * @return - Returns a new color.
         */
        subtract(other) {
            return new Color(this.r - other.r, this.g - other.g, this.b - other.b, this.a - other.a);
        }
        /**
         * Scales this color by scalar and return the result as a new Vec4.
         *
         * @param scalar - The scalar value.
         * @return - Returns a new color.
         */
        scale(scalar) {
            return new Color(this.r * scalar, this.g * scalar, this.b * scalar, this.a * scalar);
        }
        /**
         * Scales this color by scalar.
         *
         * @param scalar - The scalar value.
         */
        scaleInPlace(scalar) {
            this.r *= scalar;
            this.g *= scalar;
            this.b *= scalar;
            this.a *= scalar;
        }
        /**
         * Apply gamma correction to this color
         *
         * @param gamma - The gamma value.
         */
        applyGamma(gamma) {
            this.set(Math.pow(this.r, gamma), Math.pow(this.g, gamma), Math.pow(this.b, gamma), this.a);
        }
        /**
         * Converts to linear color space and returns a new color
         *
         * @param gamma - The gamma value.
         * @return - Returns a new color.
         */
        toLinear(gamma = 2.2) {
            return new Color(Math.pow(this.r, gamma), Math.pow(this.g, gamma), Math.pow(this.b, gamma), this.a);
        }
        /**
         * returns a new color value value is mapped into a gamma curve
         *
         * @param gamma - The gamma value.
         * @return - Returns a new color.
         */
        toGamma(gamma = 2.2) {
            return new Color(Math.pow(this.r, 1.0 / gamma), Math.pow(this.g, 1.0 / gamma), Math.pow(this.b, 1.0 / gamma), this.a);
        }
        /**
         * Calculates and returns the luminance of the linear RGB components.
         *
         * @return - The return value.
         */
        luminance() {
            return 0.2126 * this.r + 0.7152 * this.g + 0.0722 * this.b;
        }
        /**
         * Performs a linear interpolation between this color and other.
         *
         * @param other - The other color to interpolate between.
         * @param t - Interpolation amount between the two inputs.
         * @return - Returns a new color.
         */
        lerp(other, t) {
            const ar = this.r;
            const ag = this.g;
            const ab = this.b;
            const aa = this.a;
            return new Color(ar + t * (other.r - ar), ag + t * (other.g - ag), ab + t * (other.b - ab), aa + t * (other.a - aa));
        }
        /**
         * Creates a random color.
         *
         * @param gammaOffset - The gamma offset. Values between 0 and 1 increase the average brightness of the generated color. Values between 0 and -1 darken the generated color values.
         * @param randomAlpha - Determines whether the alpha channel is random. If not, the alpha values will be 1.0.
         * @return - The new random color.
         */
        static random(gammaOffset = 0.0, randomAlpha = false) {
            if (gammaOffset > 0.0) {
                return new Color(gammaOffset + Math.random() * (1.0 - gammaOffset), gammaOffset + Math.random() * (1.0 - gammaOffset), gammaOffset + Math.random() * (1.0 - gammaOffset), randomAlpha ? gammaOffset + Math.random() * (1.0 - gammaOffset) : 1.0);
            }
            if (gammaOffset < 0.0) {
                return new Color(Math.random() * (1.0 + gammaOffset), Math.random() * (1.0 + gammaOffset), Math.random() * (1.0 + gammaOffset), randomAlpha ? Math.random() * (1.0 + gammaOffset) : 1.0);
            }
            return new Color(Math.random(), Math.random(), Math.random(), randomAlpha ? Math.random() : 1.0);
        }
        /**
         * Clones this color and returns a new color.
         *
         * @return - Returns a new color.
         */
        clone() {
            return new Color(this.r, this.g, this.b, this.a);
        }
        /**
         * Returns the type as an array. Often used to pass types to the GPU.
         *
         * @return - Returns as an array.
         */
        asArray() {
            return [this.r, this.g, this.b, this.a];
        }
        /**
         * Setter from an RGB array.
         *
         * @param vals - The vals param.
         */
        fromArray(vals) {
            this.r = vals[0] / 255;
            this.g = vals[1] / 255;
            this.b = vals[2] / 255;
            this.a = vals.length == 4 ? vals[3] / 255 : 1.0;
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * The toJSON method encodes this type as a json object for persistence.
         *
         * @return - The json object.
         */
        toJSON() {
            return {
                r: this.r,
                g: this.g,
                b: this.b,
                a: this.a,
            };
        }
        /**
         * The fromJSON method decodes a json object for this type.
         *
         * @param j - The json object.
         */
        fromJSON(j) {
            this.r = j.r;
            this.g = j.g;
            this.b = j.b;
            this.a = j.a;
        }
        /**
         * Loads the state of the value from a binary reader.
         *
         * @param reader - The reader value.
         */
        readBinary(reader) {
            this.r = reader.loadFloat32();
            this.g = reader.loadFloat32();
            this.b = reader.loadFloat32();
            this.a = reader.loadFloat32();
        }
        /**
         * Returns the CSS rgba string.
         *
         * @return - The return value.
         */
        toCSSString() {
            return ('rgba(' +
                Math.round(this.r * 255) +
                ', ' +
                Math.round(this.g * 255) +
                ', ' +
                Math.round(this.b * 255) +
                ', ' +
                this.a +
                ')');
        }
        /**
         * Converts this Vec3 to a string in JSON format.
         *
         * @return - The return value.
         */
        toString() {
            // eslint-disable-next-line new-cap
            return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON());
        }
    }

    var EulerAnglesAxisOrder;
    (function (EulerAnglesAxisOrder) {
        EulerAnglesAxisOrder[EulerAnglesAxisOrder["XYZ"] = 0] = "XYZ";
        EulerAnglesAxisOrder[EulerAnglesAxisOrder["YZX"] = 1] = "YZX";
        EulerAnglesAxisOrder[EulerAnglesAxisOrder["ZXY"] = 2] = "ZXY";
        EulerAnglesAxisOrder[EulerAnglesAxisOrder["XZY"] = 3] = "XZY";
        EulerAnglesAxisOrder[EulerAnglesAxisOrder["ZYX"] = 4] = "ZYX";
        EulerAnglesAxisOrder[EulerAnglesAxisOrder["YXZ"] = 5] = "YXZ";
    })(EulerAnglesAxisOrder || (EulerAnglesAxisOrder = {}));
    /**
     * Class representing euler angles. Euler angles describe rotating an object
     * around its various axis in a specified axis order.
     *
     */
    class EulerAngles {
        x;
        y;
        z;
        order;
        /**
         * Create a euler angle. Receives the xyz values in radians and the order that the rotations are applied.
         *
         * Order parameter values: `XYZ: 0`, `YZX: 1`, `ZXY: 2`, `XZY: 3`, `ZYX: 4`, `YXZ: 5`
         *
         * It could be either the `string` or the `number` value.
         *
         * @param x - The angle of the x axis in radians. Default is 0.
         * @param y - The angle of the y axis in radians. Default is 0.
         * @param z - The angle of the z axis in radians. Default is 0.
         * @param order - The order in which the rotations are applied.
         */
        constructor(x = 0, y = 0, z = 0, order = 0) {
            this.x = x;
            this.y = y;
            this.z = z;
            if (typeof order === 'number' && !isNaN(order))
                this.order = order;
            else {
                switch (order) {
                    case 'XYZ':
                        this.order = 0;
                        break;
                    case 'YZX':
                        this.order = 1;
                        break;
                    case 'ZXY':
                        this.order = 2;
                        break;
                    case 'XZY':
                        this.order = 3;
                        break;
                    case 'ZYX':
                        this.order = 4;
                        break;
                    case 'YXZ':
                        this.order = 5;
                        break;
                    default:
                        throw new Error('Invalid Euler Angles Order:' + order);
                }
            }
        }
        /**
         * Sets the EulerAngles
         *
         * @param x - The x axis rotation in radians.
         * @param y - The y axis rotation in radians.
         * @param z - The z axis rotation in radians.
         */
        set(x, y, z) {
            this.x = x;
            this.y = y;
            this.z = z;
        }
        // ///////////////////////////
        // Persistence
        /**
         * Converts this Vec3 to a string in JSON format.
         *
         * @return - The return value.
         */
        toString() {
            // eslint-disable-next-line new-cap
            return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON());
        }
        toJSON() {
            return {
                x: this.x,
                y: this.y,
                z: this.z,
                order: this.order,
            };
        }
        fromJSON(json) {
            this.x = json.x;
            this.y = json.y;
            this.z = json.z;
            this.order = json.order;
        }
    }

    /* eslint-disable new-cap */
    /**
     * A class representing a 3x3 matrix.
     * This matrix class is based on GLM, and is column major.
     *
     */
    class Mat3 {
        m00 = 1;
        m01 = 0;
        m02 = 0;
        m10 = 0;
        m11 = 1;
        m12 = 0;
        m20 = 0;
        m21 = 0;
        m22 = 1;
        /**
         * Initializes the Mat3 class with given data.
         *
         * @param m00 - Row 0, column 0.
         * @param m01 - Row 0, column 1.
         * @param m02 - Row 0, column 2.
         * @param m10 - Row 1, column 0.
         * @param m11 - Row 1, column 1.
         * @param m12 - Row 1, column 2.
         * @param m20 - Row 2, column 0.
         * @param m21 - Row 2, column 1.
         * @param m22 - Row 2, column 2.
         */
        constructor(m00 = 1, m01 = 0, m02 = 0, m10 = 0, m11 = 1, m12 = 0, m20 = 0, m21 = 0, m22 = 1) {
            if (m00 instanceof Vec3 && m01 instanceof Vec3 && m02 instanceof Vec3) {
                this.set(m00.x, m00.y, m00.z, m01.x, m01.y, m01.z, m02.x, m02.y, m02.z);
            }
            else {
                this.set(m00, m01, m02, m10, m11, m12, m20, m21, m22);
            }
        }
        /**
         * Getter for the `x` axis.
         *
         * @return - Returns the `x` axis as a Vec3.
         */
        get xAxis() {
            return new Vec3(this.m00, this.m01, this.m02);
        }
        /**
         * Setter for the `x` axis.
         *
         * @param vec3 - The vec3 value.
         */
        set xAxis(vec3) {
            this.xAxis.set(vec3.x, vec3.y, vec3.z);
        }
        /**
         * Getter for the `y` axis.
         * * @return - Returns the `y` axis as a Vec3.
         */
        get yAxis() {
            return new Vec3(this.m10, this.m11, this.m12);
        }
        /**
         * Setter for the `y` axis.
         * @param vec3 - The vec3 value.
         */
        set yAxis(vec3) {
            this.yAxis.set(vec3.x, vec3.y, vec3.z);
        }
        /**
         * Getter for the `z` axis.
         * * @return - Returns the `z` axis as a Vec3.
         */
        get zAxis() {
            return new Vec3(this.m20, this.m21, this.m22);
        }
        /**
         * Setter for the `z` axis.
         * @param vec3 - The vec3 value.
         */
        set zAxis(vec3) {
            this.zAxis.set(vec3.x, vec3.y, vec3.z);
        }
        // /////////////////////////////////////////
        // Setters
        /**
         * Sets the state of the Mat3 class
         *
         * @param m00 - Row 0, column 0.
         * @param m01 - Row 0, column 1.
         * @param m02 - Row 0, column 2.
         * @param m10 - Row 1, column 0.
         * @param m11 - Row 1, column 1.
         * @param m12 - Row 1, column 2.
         * @param m20 - Row 2, column 0.
         * @param m21 - Row 2, column 1.
         * @param m22 - Row 2, column 2.
         */
        set(m00 = 1, m01 = 0, m02 = 0, m10 = 0, m11 = 1, m12 = 0, m20 = 0, m21 = 0, m22 = 1) {
            this.m00 = m00;
            this.m01 = m01;
            this.m02 = m02;
            this.m10 = m10;
            this.m11 = m11;
            this.m12 = m12;
            this.m20 = m20;
            this.m21 = m21;
            this.m22 = m22;
        }
        /**
         * Sets state of the Mat3 with the identity  Matrix
         */
        setIdentity() {
            this.set();
        }
        /**
         * Sets state of the Mat3 from another Mat3
         *
         * Note: works with either Mat3 or Mat4.
         *
         * @param mat - The mat value.
         */
        setFromMat(mat) {
            this.m00 = mat.m00;
            this.m01 = mat.m01;
            this.m02 = mat.m02;
            this.m10 = mat.m10;
            this.m11 = mat.m11;
            this.m12 = mat.m12;
            this.m20 = mat.m20;
            this.m21 = mat.m21;
            this.m22 = mat.m22;
        }
        /**
         * Scales and calculates the cross product of the `Vec3` and sets the result in the Mat3
         * Note: the resulting matrix +Z axis is aligned with the provided direction value.
         *
         * @param dir - The dir value.
         * @param up - The up value.
         */
        setFromDirectionAndUpvector(dir, up) {
            const zAxis = dir;
            const zLen = zAxis.length();
            if (zLen < Number.EPSILON) {
                this.setIdentity();
                return;
            }
            zAxis.scaleInPlace(1 / zLen);
            const xAxis = up.cross(zAxis);
            const xLen = xAxis.length();
            if (xLen > Number.EPSILON)
                xAxis.scaleInPlace(1 / xLen);
            const yAxis = zAxis.cross(xAxis);
            const yLen = yAxis.length();
            if (yLen > Number.EPSILON)
                yAxis.scaleInPlace(1 / yLen);
            this.set(xAxis.x, xAxis.y, xAxis.z, yAxis.x, yAxis.y, yAxis.z, zAxis.x, zAxis.y, zAxis.z);
        }
        /**
         * Inverts a Mat3 and returns the result as a new instance.
         *
         * @return - Returns a new Mat3.
         */
        inverse() {
            const a00 = this.m00;
            const a01 = this.m01;
            const a02 = this.m02;
            const a10 = this.m10;
            const a11 = this.m11;
            const a12 = this.m12;
            const a20 = this.m20;
            const a21 = this.m21;
            const a22 = this.m22;
            const b01 = a22 * a11 - a12 * a21;
            const b11 = -a22 * a10 + a12 * a20;
            const b21 = a21 * a10 - a11 * a20;
            // Calculate the determinant
            let det = a00 * b01 + a01 * b11 + a02 * b21;
            if (!det) {
                console.warn('Unable to invert Mat3');
                return new Mat3();
            }
            det = 1.0 / det;
            return new Mat3(b01 * det, (-a22 * a01 + a02 * a21) * det, (a12 * a01 - a02 * a11) * det, b11 * det, (a22 * a00 - a02 * a20) * det, (-a12 * a00 + a02 * a10) * det, b21 * det, (-a21 * a00 + a01 * a20) * det, (a11 * a00 - a01 * a10) * det);
        }
        /**
         * Inverts a Mat3 in place modifying its values.
         *
         * @return - The return value.
         */
        invertInPlace() {
            const a00 = this.m00;
            const a01 = this.m01;
            const a02 = this.m02;
            const a10 = this.m10;
            const a11 = this.m11;
            const a12 = this.m12;
            const a20 = this.m20;
            const a21 = this.m21;
            const a22 = this.m22;
            const b01 = a22 * a11 - a12 * a21;
            const b11 = -a22 * a10 + a12 * a20;
            const b21 = a21 * a10 - a11 * a20;
            // Calculate the determinant
            let det = a00 * b01 + a01 * b11 + a02 * b21;
            if (!det) {
                console.warn('Unable to invert Mat3');
                return false;
            }
            det = 1.0 / det;
            this.set(b01 * det, (-a22 * a01 + a02 * a21) * det, (a12 * a01 - a02 * a11) * det, b11 * det, (a22 * a00 - a02 * a20) * det, (-a12 * a00 + a02 * a10) * det, b21 * det, (-a21 * a00 + a01 * a20) * det, (a11 * a00 - a01 * a10) * det);
            return true;
        }
        /**
         * Transposes (exchanges columns with rows) this matrix
         * and returns the result as a new instance.
         *
         * @return - Return a new transposed Mat3.
         */
        transpose() {
            return new Mat3(this.m00, this.m10, this.m20, this.m01, this.m11, this.m21, this.m02, this.m12, this.m22);
        }
        /**
         * Transposes (exchanges columns with rows) this matrix modifying its values.
         */
        transposeInPlace() {
            // If we are transposing ourselves we can skip a few steps but have to cache some values
            const a01 = this.m01;
            const a02 = this.m02;
            const a12 = this.m12;
            this.m01 = this.m10;
            this.m02 = this.m20;
            this.m10 = a01;
            this.m12 = this.m21;
            this.m20 = a02;
            this.m21 = a12;
        }
        /**
         * Transforms the Vec3 with a Mat3.
         *
         * @param vec3 - The vec3 value.
         * @return - Return the result as a new Vec3.
         */
        transformVec3(vec3) {
            return new Vec3(this.m00 * vec3.x + this.m01 * vec3.y + this.m02 * vec3.z, this.m10 * vec3.x + this.m11 * vec3.y + this.m12 * vec3.z, this.m20 * vec3.x + this.m21 * vec3.y + this.m22 * vec3.z);
        }
        /**
         * Clones this Mat3 returning a new instance.
         *
         * @return - Returns a new Mat3.
         */
        clone() {
            return new Mat3(this.m00, this.m01, this.m02, this.m10, this.m11, this.m12, this.m20, this.m21, this.m22);
        }
        // ///////////////////////////
        // Persistence
        /**
         * Loads the state of the value from a binary reader.
         *
         * @param reader - The reader value.
         */
        readBinary(reader) {
            const data = reader.loadFloat32Array(9);
            this.fromArray(data);
        }
        /**
         * The toJSON method encodes this type as a json object for persistence.
         */
        toJSON() {
            return this.asArray();
        }
        /**
         * The fromJSON method decodes a json object for this type.
         *
         * @param json - The json param.
         */
        fromJSON(json) {
            this.fromArray(json);
        }
        // ///////////////////////////
        // Debugging
        /**
         * Converts this Vec3 to a string in JSON format.
         *
         * @return - The return value.
         */
        toString() {
            // eslint-disable-next-line new-cap
            return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON());
        }
        /**
         * Returns current Math type data as array. Often used to pass types to the GPU.
         *
         * @return - Returns the result as an array.
         */
        asArray() {
            return [this.m00, this.m01, this.m02, this.m10, this.m11, this.m12, this.m20, this.m21, this.m22];
        }
        fromArray(array) {
            this.m00 = array[0];
            this.m01 = array[1];
            this.m02 = array[2];
            this.m10 = array[3];
            this.m11 = array[4];
            this.m12 = array[5];
            this.m20 = array[6];
            this.m21 = array[7];
            this.m22 = array[8];
        }
    }

    /**
     * A class representing a 4x4 matrix.
     * This matrix class is based on GLM, and is column major.
     *
     */
    class Mat4 {
        m00;
        m01;
        m02;
        m03;
        m10;
        m11;
        m12;
        m13;
        m20;
        m21;
        m22;
        m23;
        m30;
        m31;
        m32;
        m33;
        constructor(m00 = 1, m01 = 0, m02 = 0, m03 = 0, m10 = 0, m11 = 1, m12 = 0, m13 = 0, m20 = 0, m21 = 0, m22 = 1, m23 = 0, m30 = 0, m31 = 0, m32 = 0, m33 = 1) {
            this.m00 = m00;
            this.m01 = m01;
            this.m02 = m02;
            this.m03 = m03;
            this.m10 = m10;
            this.m11 = m11;
            this.m12 = m12;
            this.m13 = m13;
            this.m20 = m20;
            this.m21 = m21;
            this.m22 = m22;
            this.m23 = m23;
            this.m30 = m30;
            this.m31 = m31;
            this.m32 = m32;
            this.m33 = m33;
        }
        /**
         * Getter for the `x` axis.
         *
         * @return - Returns the `x` axis as a Vec3.
         */
        get xAxis() {
            return new Vec3(this.m00, this.m01, this.m02);
        }
        /**
         * Setter for the `x` axis.
         *
         * @param vec3 - The vec3 value.
         */
        set xAxis(vec3) {
            this.xAxis.set(vec3.x, vec3.y, vec3.z);
        }
        /**
         * Getter for the `y` axis.
         *
         * @return - Returns the `y` axis as a Vec3.
         */
        get yAxis() {
            return new Vec3(this.m10, this.m11, this.m12);
        }
        /**
         * Setter for the `y` axis.
         *
         * @param vec3 - The vec3 value.
         */
        set yAxis(vec3) {
            this.yAxis.set(vec3.x, vec3.y, vec3.z);
        }
        /**
         * Getter for the `z` axis.
         *
         * @return - Returns the `z` axis as a Vec3.
         */
        get zAxis() {
            return new Vec3(this.m20, this.m21, this.m22);
        }
        /**
         * Setter for the `z` axis.
         *
         * @param vec3 - The vec3 value.
         */
        set zAxis(vec3) {
            this.zAxis.set(vec3.x, vec3.y, vec3.z);
        }
        /**
         * Getter for the translation of the matrix. Assumes the translation values are 12, 13, & 14.
         *
         * @return - Returns the translation.
         */
        get translation() {
            return new Vec3(this.m30, this.m31, this.m32);
        }
        /**
         * Setter for the translation of the matrix. Assumes the translation values are 12, 13, & 14.
         *
         * @param vec3 - The translation.
         */
        set translation(vec3) {
            this.m30 = vec3.x;
            this.m31 = vec3.y;
            this.m32 = vec3.z;
        }
        // /////////////////////////////////////////
        // Setters
        /**
         * Sets the state of the Mat4 class
         *
         * @param m00 - Row 0, column 0.
         * @param m01 - Row 0, column 1.
         * @param m02 - Row 0, column 2.
         * @param m03 - Row 0, column 3.
         * @param m10 - Row 1, column 0.
         * @param m11 - Row 1, column 1.
         * @param m12 - Row 1, column 2.
         * @param m13 - Row 1, column 3.
         * @param m20 - Row 2, column 0.
         * @param m21 - Row 2, column 1.
         * @param m22 - Row 2, column 2.
         * @param m23 - Row 2, column 3.
         * @param m30 - Row 3, column 0.
         * @param m31 - Row 3, column 1.
         * @param m32 - Row 3, column 2.
         * @param m33 - Row 3, column 3.
         */
        set(m00 = 1, m01 = 0, m02 = 0, m03 = 0, m10 = 0, m11 = 1, m12 = 0, m13 = 0, m20 = 0, m21 = 0, m22 = 1, m23 = 0, m30 = 0, m31 = 0, m32 = 0, m33 = 1) {
            this.m00 = m00;
            this.m01 = m01;
            this.m02 = m02;
            this.m03 = m03;
            this.m10 = m10;
            this.m11 = m11;
            this.m12 = m12;
            this.m13 = m13;
            this.m20 = m20;
            this.m21 = m21;
            this.m22 = m22;
            this.m23 = m23;
            this.m30 = m30;
            this.m31 = m31;
            this.m32 = m32;
            this.m33 = m33;
        }
        /**
         * Sets state of the Mat4 with the identity  Matrix
         */
        setIdentity() {
            this.set();
        }
        /**
         * Sets state of the Mat4 from another Mat4
         *
         * Note: works with either Mat3 or Mat4.
         *
         * @param mat4 - The mat4 value.
         */
        setFromMat4(mat4) {
            this.m00 = mat4.m00;
            this.m01 = mat4.m01;
            this.m02 = mat4.m02;
            this.m03 = mat4.m03;
            this.m10 = mat4.m10;
            this.m11 = mat4.m11;
            this.m12 = mat4.m12;
            this.m13 = mat4.m13;
            this.m20 = mat4.m20;
            this.m21 = mat4.m21;
            this.m22 = mat4.m22;
            this.m23 = mat4.m23;
            this.m30 = mat4.m30;
            this.m31 = mat4.m31;
            this.m32 = mat4.m32;
            this.m33 = mat4.m33;
        }
        /**
         * Converts a Mat4 to a Mat3.
         *
         * @return - Returns a new Mat3.
         */
        toMat3() {
            return new Mat3(this.m00, this.m01, this.m02, this.m10, this.m11, this.m12, this.m20, this.m21, this.m22);
        }
        /**
         * Transposes (exchanges columns with rows) this matrix.
         */
        transposeInPlace() {
            // If we are transposing ourselves we can skip a few steps but have to cache some values
            const a01 = this.m01;
            const a02 = this.m02;
            const a03 = this.m03;
            const a12 = this.m12;
            const a13 = this.m13;
            const a23 = this.m23;
            this.m01 = this.m10;
            this.m02 = this.m20;
            this.m03 = this.m30;
            this.m10 = a01;
            this.m12 = this.m21;
            this.m13 = this.m31;
            this.m20 = a02;
            this.m21 = a12;
            this.m23 = this.m32;
            this.m30 = a03;
            this.m31 = a13;
            this.m32 = a23;
        }
        /**
         * Transposes (exchanges columns with rows) this matrix
         * and returns the result as a new instance.
         *
         * @return - Return a new transposed Mat4.
         */
        transpose() {
            return new Mat4(this.m00, this.m10, this.m20, this.m30, this.m01, this.m11, this.m21, this.m31, this.m02, this.m12, this.m22, this.m32, this.m03, this.m13, this.m23, this.m33);
        }
        /**
         * Inverts a Mat4 and returns the result as a new instance.
         *
         * @return - Returns a new Mat4.
         */
        inverse() {
            const a00 = this.m00;
            const a01 = this.m01;
            const a02 = this.m02;
            const a03 = this.m03;
            const a10 = this.m10;
            const a11 = this.m11;
            const a12 = this.m12;
            const a13 = this.m13;
            const a20 = this.m20;
            const a21 = this.m21;
            const a22 = this.m22;
            const a23 = this.m23;
            const a30 = this.m30;
            const a31 = this.m31;
            const a32 = this.m32;
            const a33 = this.m33;
            const b00 = a00 * a11 - a01 * a10;
            const b01 = a00 * a12 - a02 * a10;
            const b02 = a00 * a13 - a03 * a10;
            const b03 = a01 * a12 - a02 * a11;
            const b04 = a01 * a13 - a03 * a11;
            const b05 = a02 * a13 - a03 * a12;
            const b06 = a20 * a31 - a21 * a30;
            const b07 = a20 * a32 - a22 * a30;
            const b08 = a20 * a33 - a23 * a30;
            const b09 = a21 * a32 - a22 * a31;
            const b10 = a21 * a33 - a23 * a31;
            const b11 = a22 * a33 - a23 * a32;
            // Calculate the determinant
            let det = b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;
            if (!det) {
                console.warn('Unable to invert Mat4');
                return this;
            }
            det = 1.0 / det;
            return new Mat4((a11 * b11 - a12 * b10 + a13 * b09) * det, (a02 * b10 - a01 * b11 - a03 * b09) * det, (a31 * b05 - a32 * b04 + a33 * b03) * det, (a22 * b04 - a21 * b05 - a23 * b03) * det, (a12 * b08 - a10 * b11 - a13 * b07) * det, (a00 * b11 - a02 * b08 + a03 * b07) * det, (a32 * b02 - a30 * b05 - a33 * b01) * det, (a20 * b05 - a22 * b02 + a23 * b01) * det, (a10 * b10 - a11 * b08 + a13 * b06) * det, (a01 * b08 - a00 * b10 - a03 * b06) * det, (a30 * b04 - a31 * b02 + a33 * b00) * det, (a21 * b02 - a20 * b04 - a23 * b00) * det, (a11 * b07 - a10 * b09 - a12 * b06) * det, (a00 * b09 - a01 * b07 + a02 * b06) * det, (a31 * b01 - a30 * b03 - a32 * b00) * det, (a20 * b03 - a21 * b01 + a22 * b00) * det);
        }
        /**
         * Inverts a Mat4.
         *
         * @return - The return value.
         */
        invertInPlace() {
            const a00 = this.m00;
            const a01 = this.m01;
            const a02 = this.m02;
            const a03 = this.m03;
            const a10 = this.m10;
            const a11 = this.m11;
            const a12 = this.m12;
            const a13 = this.m13;
            const a20 = this.m20;
            const a21 = this.m21;
            const a22 = this.m22;
            const a23 = this.m23;
            const a30 = this.m30;
            const a31 = this.m31;
            const a32 = this.m32;
            const a33 = this.m33;
            const b00 = a00 * a11 - a01 * a10;
            const b01 = a00 * a12 - a02 * a10;
            const b02 = a00 * a13 - a03 * a10;
            const b03 = a01 * a12 - a02 * a11;
            const b04 = a01 * a13 - a03 * a11;
            const b05 = a02 * a13 - a03 * a12;
            const b06 = a20 * a31 - a21 * a30;
            const b07 = a20 * a32 - a22 * a30;
            const b08 = a20 * a33 - a23 * a30;
            const b09 = a21 * a32 - a22 * a31;
            const b10 = a21 * a33 - a23 * a31;
            const b11 = a22 * a33 - a23 * a32;
            // Calculate the determinant
            let det = b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;
            if (!det) {
                console.warn('Unable to invert Mat4');
                return false;
            }
            det = 1.0 / det;
            this.set((a11 * b11 - a12 * b10 + a13 * b09) * det, (a02 * b10 - a01 * b11 - a03 * b09) * det, (a31 * b05 - a32 * b04 + a33 * b03) * det, (a22 * b04 - a21 * b05 - a23 * b03) * det, (a12 * b08 - a10 * b11 - a13 * b07) * det, (a00 * b11 - a02 * b08 + a03 * b07) * det, (a32 * b02 - a30 * b05 - a33 * b01) * det, (a20 * b05 - a22 * b02 + a23 * b01) * det, (a10 * b10 - a11 * b08 + a13 * b06) * det, (a01 * b08 - a00 * b10 - a03 * b06) * det, (a30 * b04 - a31 * b02 + a33 * b00) * det, (a21 * b02 - a20 * b04 - a23 * b00) * det, (a11 * b07 - a10 * b09 - a12 * b06) * det, (a00 * b09 - a01 * b07 + a02 * b06) * det, (a31 * b01 - a30 * b03 - a32 * b00) * det, (a20 * b03 - a21 * b01 + a22 * b00) * det);
            return true;
        }
        /**
         * Sets this matrix as the inverse of the given Mat4.
         *
         * @param mat4 - The mat4 value.
         * @return - In case the `determinant` can't be calculated, a `null` will be returned, otherwise, nothing is returned
         */
        setInverse(mat4) {
            const a00 = mat4.m00;
            const a01 = mat4.m01;
            const a02 = mat4.m02;
            const a03 = mat4.m03;
            const a10 = mat4.m10;
            const a11 = mat4.m11;
            const a12 = mat4.m12;
            const a13 = mat4.m13;
            const a20 = mat4.m20;
            const a21 = mat4.m21;
            const a22 = mat4.m22;
            const a23 = mat4.m23;
            const a30 = mat4.m30;
            const a31 = mat4.m31;
            const a32 = mat4.m32;
            const a33 = mat4.m33;
            const b00 = a00 * a11 - a01 * a10;
            const b01 = a00 * a12 - a02 * a10;
            const b02 = a00 * a13 - a03 * a10;
            const b03 = a01 * a12 - a02 * a11;
            const b04 = a01 * a13 - a03 * a11;
            const b05 = a02 * a13 - a03 * a12;
            const b06 = a20 * a31 - a21 * a30;
            const b07 = a20 * a32 - a22 * a30;
            const b08 = a20 * a33 - a23 * a30;
            const b09 = a21 * a32 - a22 * a31;
            const b10 = a21 * a33 - a23 * a31;
            const b11 = a22 * a33 - a23 * a32;
            // Calculate the determinant
            let det = b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;
            if (!det) {
                throw new Error('Unable to invert Mat4');
            }
            det = 1.0 / det;
            this.set((a11 * b11 - a12 * b10 + a13 * b09) * det, (a02 * b10 - a01 * b11 - a03 * b09) * det, (a31 * b05 - a32 * b04 + a33 * b03) * det, (a22 * b04 - a21 * b05 - a23 * b03) * det, (a12 * b08 - a10 * b11 - a13 * b07) * det, (a00 * b11 - a02 * b08 + a03 * b07) * det, (a32 * b02 - a30 * b05 - a33 * b01) * det, (a20 * b05 - a22 * b02 + a23 * b01) * det, (a10 * b10 - a11 * b08 + a13 * b06) * det, (a01 * b08 - a00 * b10 - a03 * b06) * det, (a30 * b04 - a31 * b02 + a33 * b00) * det, (a21 * b02 - a20 * b04 - a23 * b00) * det, (a11 * b07 - a10 * b09 - a12 * b06) * det, (a00 * b09 - a01 * b07 + a02 * b06) * det, (a31 * b01 - a30 * b03 - a32 * b00) * det, (a20 * b03 - a21 * b01 + a22 * b00) * det);
        }
        /**
         * Multiplies two Mat4s and returns the result as a new instance.
         *
         * @param other - The other Mat4 to multiply with.
         * @return - Returns a new Mat4.
         */
        multiply(other) {
            const a00 = this.m00;
            const a01 = this.m01;
            const a02 = this.m02;
            const a03 = this.m03;
            const a10 = this.m10;
            const a11 = this.m11;
            const a12 = this.m12;
            const a13 = this.m13;
            const a20 = this.m20;
            const a21 = this.m21;
            const a22 = this.m22;
            const a23 = this.m23;
            const a30 = this.m30;
            const a31 = this.m31;
            const a32 = this.m32;
            const a33 = this.m33;
            // Cache only the current line of the second matrix
            const b = other.asArray();
            let b0 = b[0];
            let b1 = b[1];
            let b2 = b[2];
            let b3 = b[3];
            const result = new Mat4();
            result.m00 = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;
            result.m01 = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;
            result.m02 = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;
            result.m03 = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;
            b0 = b[4];
            b1 = b[5];
            b2 = b[6];
            b3 = b[7];
            result.m10 = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;
            result.m11 = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;
            result.m12 = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;
            result.m13 = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;
            b0 = b[8];
            b1 = b[9];
            b2 = b[10];
            b3 = b[11];
            result.m20 = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;
            result.m21 = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;
            result.m22 = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;
            result.m23 = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;
            b0 = b[12];
            b1 = b[13];
            b2 = b[14];
            b3 = b[15];
            result.m30 = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;
            result.m31 = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;
            result.m32 = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;
            result.m33 = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;
            return result;
        }
        /**
         * Multiplies two Mat4s in place explicitly not using SIMD.
         *
         * @param other - The other Mat4 to multiply with.
         * @return - Returns a new Mat4.
         */
        multiplyInPlace(other) {
            const a = this.asArray();
            const a00 = a[0];
            const a01 = a[1];
            const a02 = a[2];
            const a03 = a[3];
            const a10 = a[4];
            const a11 = a[5];
            const a12 = a[6];
            const a13 = a[7];
            const a20 = a[8];
            const a21 = a[9];
            const a22 = a[10];
            const a23 = a[11];
            const a30 = a[12];
            const a31 = a[13];
            const a32 = a[14];
            const a33 = a[15];
            // Cache only the current line of the second matrix
            const b = other.asArray();
            let b0 = b[0];
            let b1 = b[1];
            let b2 = b[2];
            let b3 = b[3];
            this.m00 = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;
            this.m01 = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;
            this.m02 = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;
            this.m03 = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;
            b0 = b[4];
            b1 = b[5];
            b2 = b[6];
            b3 = b[7];
            this.m10 = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;
            this.m11 = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;
            this.m12 = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;
            this.m13 = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;
            b0 = b[8];
            b1 = b[9];
            b2 = b[10];
            b3 = b[11];
            this.m20 = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;
            this.m21 = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;
            this.m22 = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;
            this.m23 = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;
            b0 = b[12];
            b1 = b[13];
            b2 = b[14];
            b3 = b[15];
            this.m30 = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;
            this.m31 = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;
            this.m32 = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;
            this.m33 = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;
            return this;
        }
        /**
         * Post multiplies two Mat4s in place explicitly not using SIMD.
         *
         * @param other - The other Mat4 to multiply with.
         * @return - Returns the result as a new Mat4.
         */
        postMultiplyInPlace(other) {
            const a = other.asArray();
            const a00 = a[0];
            const a01 = a[1];
            const a02 = a[2];
            const a03 = a[3];
            const a10 = a[4];
            const a11 = a[5];
            const a12 = a[6];
            const a13 = a[7];
            const a20 = a[8];
            const a21 = a[9];
            const a22 = a[10];
            const a23 = a[11];
            const a30 = a[12];
            const a31 = a[13];
            const a32 = a[14];
            const a33 = a[15];
            // Cache only the current line of the second matrix
            const b = this.asArray();
            let b0 = b[0];
            let b1 = b[1];
            let b2 = b[2];
            let b3 = b[3];
            this.m00 = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;
            this.m01 = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;
            this.m02 = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;
            this.m03 = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;
            b0 = b[4];
            b1 = b[5];
            b2 = b[6];
            b3 = b[7];
            this.m10 = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;
            this.m11 = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;
            this.m12 = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;
            this.m13 = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;
            b0 = b[8];
            b1 = b[9];
            b2 = b[10];
            b3 = b[11];
            this.m20 = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;
            this.m21 = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;
            this.m22 = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;
            this.m23 = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;
            b0 = b[12];
            b1 = b[13];
            b2 = b[14];
            b3 = b[15];
            this.m30 = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;
            this.m31 = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;
            this.m32 = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;
            this.m33 = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;
            return this;
        }
        /**
         * Translate a Mat4 by the given vector not using SIMD.
         *
         * @param v3 - The given vector to translate along.
         * @return - The return value.
         */
        translateInPlace(v3) {
            const a = this.asArray();
            const x = v3.x;
            const y = v3.y;
            const z = v3.z;
            a[12] = a[0] * x + a[4] * y + a[8] * z + a[12];
            a[13] = a[1] * x + a[5] * y + a[9] * z + a[13];
            a[14] = a[2] * x + a[6] * y + a[10] * z + a[14];
            a[15] = a[3] * x + a[7] * y + a[11] * z + a[15];
            return this;
        }
        /**
         * Generates a look-at matrix with the given position, focal point, and up axis.
         *
         * @param pos - Position of the viewer.
         * @param target - Point the viewer is looking at.
         * @param up - Vec3 pointing up.
         */
        setLookAt(pos, target, up) {
            const zAxis = pos.subtract(target);
            const zLen = zAxis.length();
            if (zLen < Number.EPSILON) {
                this.setIdentity();
                return;
            }
            zAxis.scaleInPlace(1.0 / zLen);
            const xAxis = up.cross(zAxis);
            const xLen = xAxis.length();
            if (xLen > Number.EPSILON)
                xAxis.scaleInPlace(1.0 / xLen);
            const yAxis = zAxis.cross(xAxis);
            const yLen = yAxis.length();
            if (yLen > Number.EPSILON)
                yAxis.scaleInPlace(1.0 / yLen);
            /* eslint-disable prettier/prettier*/
            this.set(xAxis.x, xAxis.y, xAxis.z, 0, yAxis.x, yAxis.y, yAxis.z, 0, zAxis.x, zAxis.y, zAxis.z, 0, pos.x, pos.y, pos.z, 1);
            /* eslint-enable prettier/prettier*/
        }
        /**
         * Creates a matrix from a given angle around a given axis.
         * This is equivalent to (but much faster than):
         *
         *     mat4.identity(dest);
         *     mat4.rotate(dest, dest, rad, axis);
         *
         * @param axis - The axis to rotate around.
         * @param rad - The angle to rotate the matrix by.
         * @return - The return value.
         */
        setRotation(axis, rad) {
            const len = axis.length();
            if (Math.abs(len) < Number.EPSILON) {
                return null;
            }
            const x = axis.x / len;
            const y = axis.y / len;
            const z = axis.z / len;
            const s = Math.sin(rad);
            const c = Math.cos(rad);
            const t = 1 - c;
            // Perform rotation-specific matrix multiplication
            const a = this.asArray();
            a[0] = x * x * t + c;
            a[1] = y * x * t + z * s;
            a[2] = z * x * t - y * s;
            a[3] = 0;
            a[4] = x * y * t - z * s;
            a[5] = y * y * t + c;
            a[6] = z * y * t + x * s;
            a[7] = 0;
            a[8] = x * z * t + y * s;
            a[9] = y * z * t - x * s;
            a[10] = z * z * t + c;
            a[11] = 0;
            a[12] = 0;
            a[13] = 0;
            a[14] = 0;
            a[15] = 1;
            return this;
        }
        /**
         * Creates a matrix from the given angle around the X axis.
         * This is equivalent to (but much faster than):
         *
         *     mat4.identity(dest);
         *     mat4.rotateX(dest, dest, rad);
         *
         * @param rad - The angle to rotate the matrix by.
         * @return - The return value.
         */
        setXRotation(rad) {
            const s = Math.sin(rad);
            const c = Math.cos(rad);
            // Perform axis-specific matrix multiplication
            const a = this.asArray();
            /* eslint-disable prettier/prettier*/
            a[0] = 1;
            a[1] = 0;
            a[2] = 0;
            a[3] = 0;
            a[4] = 0;
            a[5] = c;
            a[6] = s;
            a[7] = 0;
            a[8] = 0;
            a[9] = -s;
            a[10] = c;
            a[11] = 0;
            a[12] = 0;
            a[13] = 0;
            a[14] = 0;
            a[15] = 1;
            /* eslint-enable prettier/prettier*/
            return this;
        }
        /**
         * Creates a matrix from the given angle around the Y axis.
         * This is equivalent to (but much faster than):
         *
         *     mat4.identity(dest);
         *     mat4.rotateY(dest, dest, rad);
         *
         * @param rad - The angle to rotate the matrix by.
         * @return - The return value.
         */
        setYRotation(rad) {
            const s = Math.sin(rad);
            const c = Math.cos(rad);
            // Perform axis-specific matrix multiplication
            const a = this.asArray();
            /* eslint-disable prettier/prettier*/
            a[0] = c;
            a[1] = 0;
            a[2] = -s;
            a[3] = 0;
            a[4] = 0;
            a[5] = 1;
            a[6] = 0;
            a[7] = 0;
            a[8] = s;
            a[9] = 0;
            a[10] = c;
            a[11] = 0;
            a[12] = 0;
            a[13] = 0;
            a[14] = 0;
            a[15] = 1;
            /* eslint-enable prettier/prettier*/
            return this;
        }
        /**
         * Creates a matrix from the given angle around the Z axis.
         * This is equivalent to (but much faster than):
         *
         *     mat4.identity(dest);
         *     mat4.rotateZ(dest, dest, rad);
         *
         * @param rad - The angle to rotate the matrix by.
         * @return - The return value.
         */
        setZRotation(rad) {
            const s = Math.sin(rad);
            const c = Math.cos(rad);
            // Perform axis-specific matrix multiplication
            const a = this.asArray();
            /* eslint-disable prettier/prettier*/
            a[0] = c;
            a[1] = s;
            a[2] = 0;
            a[3] = 0;
            a[4] = -s;
            a[5] = c;
            a[6] = 0;
            a[7] = 0;
            a[8] = 0;
            a[9] = 0;
            a[10] = 1;
            a[11] = 0;
            a[12] = 0;
            a[13] = 0;
            a[14] = 0;
            a[15] = 1;
            /* eslint-enable prettier/prettier*/
            return this;
        }
        /**
         * Transforms the Vec4 with a Mat4.
         *
         * @param vec - The vec value.
         * @return - Return the result as a new Vec4.
         */
        transformVec4(vec) {
            const a = this.asArray();
            const x = vec.x;
            const y = vec.y;
            const z = vec.z;
            const w = vec.w;
            return new Vec4(a[0] * x + a[4] * y + a[8] * z + a[12] * w, a[1] * x + a[5] * y + a[9] * z + a[13] * w, a[2] * x + a[6] * y + a[10] * z + a[14] * w, a[3] * x + a[7] * y + a[11] * z + a[15] * w);
        }
        /**
         * Transforms the Vec3 with a Mat4.
         *
         * @param vec - The vec value.
         * @return - Return the result as a new Vec3.
         */
        transformVec3(vec) {
            const a = this.asArray();
            const x = vec.x;
            const y = vec.y;
            const z = vec.z;
            return new Vec3(a[0] * x + a[4] * y + a[8] * z + a[12], a[1] * x + a[5] * y + a[9] * z + a[13], a[2] * x + a[6] * y + a[10] * z + a[14]);
        }
        /**
         * Rotates a given `Vec3` and the result is returned as a new `Vec3`, applying only the top left components of the matrix, so not applying any translation.
         * @param vec - The vec value.
         * @return - Return the result as a new Vec3.
         */
        rotateVec3(vec) {
            const a = this.asArray();
            const x = vec.x;
            const y = vec.y;
            const z = vec.z;
            return new Vec3(a[0] * x + a[4] * y + a[8] * z, a[1] * x + a[5] * y + a[9] * z, a[2] * x + a[6] * y + a[10] * z);
        }
        /**
         * Set the perspective from a Mat4.
         *
         * @param fovY - The fovY value.
         * @param aspect - The aspect value.
         * @param near - The near value.
         * @param far - The far value.
         */
        setPerspectiveMatrix(fovy, aspect, near, far) {
            const f = Math.tan(Math.PI * 0.5 - 0.5 * fovy);
            const rangeInv = 1.0 / (near - far);
            /* eslint-disable prettier/prettier*/
            this.set(f / aspect, 0, 0, 0, 0, f, 0, 0, 0, 0, (near + far) * rangeInv, -1, 0, 0, near * far * rangeInv * 2, 0);
            /* eslint-enable prettier/prettier*/
        }
        /**
         * Calculates the orthographic matrix and sets the state of the Mat4 class
         *
         * @param left - The left value.
         * @param right - The right value.
         * @param bottom - The bottom value.
         * @param top - The top value.
         * @param near - The near value.
         * @param far - The far value.
         */
        setOrthographicMatrix(left, right, bottom, top, near, far) {
            const lr = 1 / (left - right);
            const bt = 1 / (bottom - top);
            const nf = 1 / (near - far);
            /* eslint-disable prettier/prettier*/
            this.set(-2 * lr, 0, 0, 0, 0, -2 * bt, 0, 0, 0, 0, 2 * nf, 0, (left + right) * lr, (top + bottom) * bt, (far + near) * nf, 1);
            /* eslint-enable prettier/prettier*/
        }
        /**
         * Set the Matrix to be a scale matrix.
         *
         * @param x - The x value.
         * @param y - The y value.
         * @param z - The z value.
         */
        setScale(x, y, z) {
            /* eslint-disable prettier/prettier*/
            if (x instanceof Vec3) {
                this.set(x.x, 0, 0, 0, 0, x.y, 0, 0, 0, 0, x.z, 0, 0, 0, 0, 1);
            }
            else {
                this.set(x, 0, 0, 0, 0, y, 0, 0, 0, 0, z, 0, 0, 0, 0, 1);
            }
            /* eslint-enable prettier/prettier*/
        }
        /**
         * Transforms a 3x4 matrix into a 4x4 matrix and set the result to the Math4 state.
         *
         * @param m3x4 - The m3x4 value.
         */
        setFromMat3x4Array(m3x4) {
            /* eslint-disable prettier/prettier*/
            this.set(m3x4[0], m3x4[1], m3x4[2], 0, m3x4[3], m3x4[4], m3x4[5], 0, m3x4[6], m3x4[7], m3x4[8], 0, m3x4[9], m3x4[10], m3x4[11], 1);
            /* eslint-enable prettier/prettier*/
        }
        /**
         * Clones this Mat4 returning a new instance.
         *
         * @return - Returns a new Mat4.
         */
        clone() {
            return new Mat4(this.m00, this.m01, this.m02, this.m03, this.m10, this.m11, this.m12, this.m13, this.m20, this.m21, this.m22, this.m23, this.m30, this.m31, this.m32, this.m33);
        }
        // ///////////////////////////
        // Persistence
        /**
         * Converts this Vec3 to a string in JSON format.
         *
         * @return - The return value.
         */
        toString() {
            // eslint-disable-next-line new-cap
            return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON());
        }
        /**
         * The toJSON method encodes this type as a json object for persistence.
         *
         */
        toJSON() {
            return this.asArray();
        }
        /**
         * The fromJSON method decodes a json object for this type.
         *
         */
        fromJSON(json) {
            this.fromArray(json);
        }
        /**
         * Loads the state of the value from a binary reader.
         *
         * @param reader - The reader value.
         */
        readBinary(reader) {
            this.fromArray(reader.loadFloat32Array(16));
        }
        /**
         * Returns current Math type data as array. Often used to pass types to the GPU.
         *
         * @return - Returns the result as an array.
         */
        asArray() {
            return [
                this.m00,
                this.m01,
                this.m02,
                this.m03,
                this.m10,
                this.m11,
                this.m12,
                this.m13,
                this.m20,
                this.m21,
                this.m22,
                this.m23,
                this.m30,
                this.m31,
                this.m32,
                this.m33,
            ];
        }
        fromArray(array) {
            this.m00 = array[0];
            this.m01 = array[1];
            this.m02 = array[2];
            this.m03 = array[3];
            this.m10 = array[4];
            this.m11 = array[5];
            this.m12 = array[6];
            this.m13 = array[7];
            this.m20 = array[8];
            this.m21 = array[9];
            this.m22 = array[10];
            this.m23 = array[11];
            this.m30 = array[12];
            this.m31 = array[13];
            this.m32 = array[14];
            this.m33 = array[15];
        }
    }

    /* eslint-disable no-unused-vars */
    /**
     * Class representing a quaternion. Quaternions are used to represent 3 dimensional rotations.
     *
     * While Quaternions are difficult to understand they have important mathematical properties that make them very useful in 3d engines.
     * They can be directly multiplied together in the same was as matrices.
     * They can be interpolated from one value to another while maintaining constant angular velocity.
     * They can be converted to other more easily understood representations such as EulerAngles or Matrices.
     *

     */
    class Quat {
        x;
        y;
        z;
        w;
        /**
         * Creates a quaternion.
         */
        constructor(x = 0, y = 0, z = 0, w = 1) {
            this.x = x;
            this.y = y;
            this.z = z;
            this.w = w;
        }
        /**
         * Setter from scalar components.
         *
         * @param x - The x axis rotation.
         * @param y  - The y axis rotation.
         * @param z  - The z axis rotation.
         * @param w  - The w value.
         */
        set(x, y, z, w) {
            this.x = x;
            this.y = y;
            this.z = z;
            this.w = w;
        }
        /**
         * Setter from another vector.
         *
         * @param other - The other vector to set from.
         */
        setFromOther(other) {
            this.x = other.x;
            this.y = other.y;
            this.z = other.z;
            this.w = other.w;
        }
        /**
         * Set this Quat from a euler rotation.
         *
         * @param eulerAngles - The euler angles rotation.
         */
        setFromEulerAngles(eulerAngles) {
            const ordered = new Vec3();
            switch (eulerAngles.order) {
                case EulerAnglesAxisOrder.XYZ:
                    ordered.set(eulerAngles.x, -eulerAngles.y, eulerAngles.z);
                    break;
                case EulerAnglesAxisOrder.YZX:
                    ordered.set(eulerAngles.y, -eulerAngles.z, eulerAngles.x);
                    break;
                case EulerAnglesAxisOrder.ZXY:
                    ordered.set(eulerAngles.z, -eulerAngles.x, eulerAngles.y);
                    break;
                case EulerAnglesAxisOrder.XZY:
                    ordered.set(eulerAngles.x, eulerAngles.z, eulerAngles.y);
                    break;
                case EulerAnglesAxisOrder.ZYX:
                    ordered.set(eulerAngles.z, eulerAngles.y, eulerAngles.x);
                    break;
                case EulerAnglesAxisOrder.YXZ:
                    ordered.set(eulerAngles.y, eulerAngles.x, eulerAngles.z);
                    break;
                default:
                    throw new Error(`Invalid EulerAngles order: ${eulerAngles.order}`);
            }
            const ti = ordered.x * 0.5;
            const tj = ordered.y * 0.5;
            const tk = ordered.z * 0.5;
            const ci = Math.cos(ti);
            const cj = Math.cos(tj);
            const ck = Math.cos(tk);
            const si = Math.sin(ti);
            const sj = Math.sin(tj);
            const sk = Math.sin(tk);
            const cc = ci * ck;
            const cs = ci * sk;
            const sc = si * ck;
            const ss = si * sk;
            const ai = cj * sc - sj * cs;
            const aj = cj * ss + sj * cc;
            const ak = cj * cs - sj * sc;
            this.w = cj * cc + sj * ss;
            switch (eulerAngles.order) {
                case 0:
                    // ' XYZ'
                    this.x = ai;
                    this.y = -aj;
                    this.z = ak;
                    break;
                case 1:
                    // 'YZX'
                    this.x = ak;
                    this.y = ai;
                    this.z = -aj;
                    break;
                case 2:
                    // 'ZXY'
                    this.x = -aj;
                    this.y = ak;
                    this.z = ai;
                    break;
                case 3:
                    // 'XZY'
                    this.x = ai;
                    this.y = ak;
                    this.z = aj;
                    break;
                case 4:
                    // 'ZYX'
                    this.x = ak;
                    this.y = aj;
                    this.z = ai;
                    break;
                case 5:
                    // 'YXZ'
                    this.x = aj;
                    this.y = ai;
                    this.z = ak;
                    break;
                default:
                    throw new Error(`Invalid EulerAngles order: ${eulerAngles.order}`);
            }
        }
        /**
         * Converts Quat to an EulerAngles
         *
         * @param rotationOrder - The order in which the rotations are applied.
         * @return - The return value.
         */
        toEulerAngles(rotationOrder) {
            const ordered = new Vec3();
            switch (rotationOrder) {
                case EulerAnglesAxisOrder.XYZ:
                case 'XYZ':
                    ordered.set(this.z, this.x, this.y);
                    break;
                case EulerAnglesAxisOrder.YZX:
                case 'YZX':
                    ordered.set(this.x, this.y, this.z);
                    break;
                case EulerAnglesAxisOrder.ZXY:
                case 'ZXY':
                    ordered.set(this.y, this.z, this.x);
                    break;
                case EulerAnglesAxisOrder.XZY:
                case 'XZY':
                    ordered.set(this.y, -this.x, this.z);
                    break;
                case EulerAnglesAxisOrder.ZYX:
                case 'ZYX':
                    ordered.set(this.x, -this.z, this.y);
                    break;
                case EulerAnglesAxisOrder.YXZ:
                case 'YXZ':
                    ordered.set(this.z, -this.y, this.x);
                    break;
                default:
                    throw new Error('Invalid rotation order:' + rotationOrder);
            }
            const euler = new Vec3();
            const test = ordered.x * ordered.y + ordered.z * this.w;
            if (test > 0.49999) {
                // singularity at north pole
                euler.y = 2.0 * Math.atan2(ordered.x, this.w);
                euler.z = Math.PI * 0.5;
                euler.x = 0.0;
            }
            else if (test < -0.49999) {
                // singularity at south pole
                euler.y = -2.0 * Math.atan2(ordered.x, this.w);
                euler.z = Math.PI * -0.5;
                euler.x = 0.0;
            }
            else {
                const sqx = ordered.x * ordered.x;
                const sqy = ordered.y * ordered.y;
                const sqz = ordered.z * ordered.z;
                euler.y = Math.atan2(2.0 * ordered.y * this.w - 2.0 * ordered.x * ordered.z, 1.0 - 2.0 * sqy - 2.0 * sqz);
                euler.z = Math.asin(2.0 * test);
                euler.x = Math.atan2(2.0 * ordered.x * this.w - 2.0 * ordered.y * ordered.z, 1.0 - 2.0 * sqx - 2.0 * sqz);
            }
            switch (rotationOrder) {
                case EulerAnglesAxisOrder.XYZ:
                case 'XYZ':
                    return new EulerAngles(euler.y, euler.z, euler.x, rotationOrder);
                case EulerAnglesAxisOrder.YZX:
                case 'YZX':
                    return new EulerAngles(euler.x, euler.y, euler.z, rotationOrder);
                case EulerAnglesAxisOrder.ZXY:
                case 'ZXY':
                    return new EulerAngles(euler.z, euler.x, euler.y, rotationOrder);
                case EulerAnglesAxisOrder.XZY:
                case 'XZY':
                    return new EulerAngles(-euler.y, euler.x, euler.z, rotationOrder);
                case EulerAnglesAxisOrder.ZYX:
                case 'ZYX':
                    return new EulerAngles(euler.x, euler.z, -euler.y, rotationOrder);
                case EulerAnglesAxisOrder.YXZ:
                case 'YXZ':
                    return new EulerAngles(euler.z, -euler.y, euler.x, rotationOrder);
            }
        }
        /**
         * Set this Quat to a rotation defined by an axis and an angle (in radians).
         *
         * @param axis - The axis around which to rotate.
         * @param angle - The angle to rotate
         */
        setFromAxisAndAngle(axis, angle) {
            const halfAngle = angle / 2.0;
            const vec = axis.normalize().scale(Math.sin(halfAngle));
            this.set(vec.x, vec.y, vec.z, Math.cos(halfAngle));
        }
        /**
         * Sets the state of the Quat to look in a particular direction along the z axis.
         * > The camera looks down the negative z axis, so to set a rotation value
         * > for the camera, remember to negate the direction vector.
         *
         * @param dir - The direction value.
         * @param up - The up vector.
         */
        setFromDirectionAndUpvector(dir, up) {
            const mat3 = new Mat3();
            mat3.setFromDirectionAndUpvector(dir, up);
            this.setFromMat3(mat3);
        }
        /**
         * Sets the state of the `Quat` from two `Vec3`. The quaternion would then represent the rotation from v0 to v1 in 3d space.
         *
         * @param v0 - The v0 unit vector.
         * @param v1 - The v1 unit vector.
         */
        setFrom2Vectors(v0, v1) {
            const c = v0.cross(v1);
            const d = v0.dot(v1);
            const s = Math.sqrt((1 + d) * 2);
            // this.set( s/2, c.x / s, c.y / s, c.z / s );
            this.set(c.x / s, c.y / s, c.z / s, s / 2);
            this.normalizeInPlace();
        }
        /**
         * Set the Quat from a Mat3.
         *
         * @param mat3 - The mat3 value.
         */
        setFromMat3(mat3) {
            // Algorithm in Ken Shoemake's article in 1987 SIGGRAPH course notes
            // article "Quaternion Calculus and Fast Animation".
            const data = mat3.asArray();
            const fTrace = data[0] + data[4] + data[8];
            let fRoot;
            if (fTrace > 0.0) {
                // |w| > 1/2, may as well choose w > 1/2
                fRoot = Math.sqrt(fTrace + 1); // 2w
                this.w = 0.5 * fRoot;
                fRoot = 0.5 / fRoot; // 1/(4w)
                this.x = (data[5] - data[7]) * fRoot;
                this.y = (data[6] - data[2]) * fRoot;
                this.z = (data[1] - data[3]) * fRoot;
            }
            else {
                // |w| <= 1/2
                let i = 0;
                if (data[4] > data[0])
                    i = 1;
                if (data[8] > data[i * 3 + i])
                    i = 2;
                const j = (i + 1) % 3;
                const k = (i + 2) % 3;
                fRoot = Math.sqrt(data[i * 3 + i] - data[j * 3 + j] - data[k * 3 + k] + 1.0);
                const array = [0, 0, 0, 0];
                array[i] = 0.5 * fRoot;
                fRoot = 0.5 / fRoot;
                array[3] = (data[j * 3 + k] - data[k * 3 + j]) * fRoot;
                array[j] = (data[j * 3 + i] + data[i * 3 + j]) * fRoot;
                array[k] = (data[k * 3 + i] + data[i * 3 + k]) * fRoot;
                this.fromArray(array);
            }
            this.normalizeInPlace();
        }
        /**
         * Set the Quat from a Mat4.
         *
         * @param mat4 - The mat4 value.
         */
        setFromMat4(mat4) {
            // Algorithm in Ken Shoemake's article in 1987 SIGGRAPH course notes
            // article "Quaternion Calculus and Fast Animation".
            const data = mat4.asArray();
            const fTrace = data[0] + data[5] + data[10];
            let fRoot;
            if (fTrace > 0.0) {
                // |w| > 1/2, may as well choose w > 1/2
                fRoot = Math.sqrt(fTrace + 1); // 2w
                this.w = 0.5 * fRoot;
                fRoot = 0.5 / fRoot; // 1/(4w)
                this.x = (data[6] - data[9]) * fRoot;
                this.y = (data[8] - data[2]) * fRoot;
                this.z = (data[1] - data[4]) * fRoot;
            }
            else {
                // |w| <= 1/2
                let i = 0;
                if (data[5] > data[0])
                    i = 1;
                if (data[10] > data[i * 4 + i])
                    i = 2;
                const j = (i + 1) % 3;
                const k = (i + 2) % 3;
                fRoot = Math.sqrt(data[i * 4 + i] - data[j * 4 + j] - data[k * 4 + k] + 1.0);
                const array = [0, 0, 0, 0];
                array[i] = 0.5 * fRoot;
                fRoot = 0.5 / fRoot;
                array[3] = (data[j * 4 + k] - data[k * 4 + j]) * fRoot;
                array[j] = (data[j * 4 + i] + data[i * 4 + j]) * fRoot;
                array[k] = (data[k * 4 + i] + data[i * 4 + k]) * fRoot;
                this.fromArray(array);
            }
            this.normalizeInPlace();
        }
        /**
         * Checks if the angle of the Quat is less that ` Number.EPSILON`
         *
         * @return - Returns true or false.
         */
        isIdentity() {
            return this.getAngle() < Number.EPSILON;
        }
        /**
         * Return the angle of the Quat.
         *
         * @return - The return value.
         */
        getAngle() {
            return Math.acos(this.w) * 2.0;
        }
        /**
         * Checks if this Quat contains the same values as the other Quat.
         *
         * @param other - The other Quat to compare with.
         * @return - Returns `true` if are the same Vector, otherwise, `false`.
         */
        isEqual(other) {
            return this.x == other.x && this.y == other.y && this.z == other.z && this.w == other.w;
        }
        /**
         * Returns true if this Quat is NOT exactly the same other.
         *
         * @param other - The other Quat to compare with.
         * @return - Returns true or false.
         */
        notEquals(other) {
            return this.x != other.x && this.y != other.y && this.z != other.z && this.w != other.w;
        }
        /**
         * Returns true if this Quat is approximately the same as other
         *
         * @param other - The other Quat to compare with.
         * @param precision - The precision to which the values must match.
         * @return - Returns true or false.
         */
        approxEqual(other, precision = Number.EPSILON) {
            return (Math.abs(this.x - other.x) < precision &&
                Math.abs(this.y - other.y) < precision &&
                Math.abs(this.z - other.z) < precision &&
                Math.abs(this.w - other.w) < precision);
        }
        /**
         * Adds other to this Quat and return the result as a new Quat.
         *
         * @param other - The other Quat to add.
         * @return - Returns a new Quat.
         */
        add(other) {
            return new Quat(this.x + other.x, this.y + other.y, this.z + other.z, this.w + other.w);
        }
        /**
         * Adds other to this Quat.
         *
         * @param other - The other Quat to add.
         */
        addInPlace(other) {
            this.x += other.x;
            this.y += other.y;
            this.z += other.z;
            this.w += other.w;
        }
        /**
         * Subtracts other from this Quat and returns the result as a new Quat.
         *
         * @param other - The other Quat to subtract.
         * @return - Returns a new Quat.
         */
        subtract(other) {
            return new Quat(this.x - other.x, this.y - other.y, this.z - other.z, this.w - other.w);
        }
        /**
         * Scales this Quat by scalar and returns the result as a new Quat.
         *
         * @param scalar - The scalar value.
         * @return - Returns a new Vec3.
         */
        scale(scalar) {
            return new Quat(this.x * scalar, this.y * scalar, this.z * scalar, this.w * scalar);
        }
        /**
         * Scales this Quat by scalar.
         *
         * @param scalar - The scalar value.
         */
        scaleInPlace(scalar) {
            this.x *= scalar;
            this.y *= scalar;
            this.z *= scalar;
            this.w *= scalar;
        }
        /**
         * Calculates the length of this Quat.
         *
         * @return - Returns the length.
         */
        length() {
            const x = this.x;
            const y = this.y;
            const z = this.z;
            const w = this.w;
            return Math.sqrt(x * x + y * y + z * z + w * w);
        }
        /**
         * Calculates the squared length of this Quat.
         *
         * @return - Returns the length.
         */
        lengthSquared() {
            const x = this.x;
            const y = this.y;
            const z = this.z;
            const w = this.w;
            return x * x + y * y + z * z + w * w;
        }
        /**
         * Normalizes the Quat and returns it as a new Quat.
         *
         * @return - Returns the Quat normalized.
         */
        normalize() {
            const x = this.x;
            const y = this.y;
            const z = this.z;
            const w = this.w;
            let len = x * x + y * y + z * z + w * w;
            if (len < Number.EPSILON) {
                return new Quat();
            }
            // TODO: evaluate use of glm_invsqrt here?
            len = 1 / Math.sqrt(len);
            return new Quat(x * len, y * len, z * len, w * len);
        }
        /**
         * Normalizes the Quat, modifying its values in place.
         */
        normalizeInPlace() {
            const x = this.x;
            const y = this.y;
            const z = this.z;
            const w = this.w;
            let len = x * x + y * y + z * z + w * w;
            if (len < Number.EPSILON) {
                return;
            }
            len = 1 / Math.sqrt(len);
            this.set(x * len, y * len, z * len, w * len);
        }
        /**
         * Calculates the dot product of this quat against another.
         *
         * @param other - The other Quat to compare with.
         * @return - Returns the dot product.
         */
        dot(other) {
            return this.x * other.x + this.y * other.y + this.z * other.z + this.w * other.w;
        }
        /**
         * Calculates the cross product of two Quats and returns the result as a new Quat.
         *
         * @param other - The other Quat to calculate with.
         * @return - Returns the cross product as a new Quat.
         */
        cross(other) {
            const ax = this.x;
            const ay = this.y;
            const az = this.z;
            const at = this.w;
            const bx = other.x;
            const by = other.y;
            const bz = other.z;
            const bt = other.w;
            return new Quat(ay * bz - az * by, az * bt - at * bz, at * bx - ax * bt, ax * by - ay * bx);
        }
        /**
         * Returns the rotational conjugate of this Quat.
         * Conjugation represents the same rotation of the Quat but
         * in the opposite direction around the rotational axis.
         *
         * @return - the return value.
         */
        conjugate() {
            return new Quat(-this.x, -this.y, -this.z, this.w);
        }
        /**
         * Return the inverse of the `Quat`
         *
         * @return - Returns a new Quat.
         */
        inverse() {
            return this.conjugate();
        }
        /**
         * Aligns this quaternion with another one ensuring that the delta between
         * the Quat values is the shortest path over the hyper-sphere.
         *
         *  @param other - The other Quat to divide by.
         */
        alignWith(other) {
            if (this.dot(other) < 0.0) {
                this.set(-this.x, -this.y, -this.z, -this.w);
            }
        }
        /**
         * Multiplies two this quat by another returning the result as a new Quat.
         *
         * @param other - The other Quat to multiply.
         * @return - Returns a new Quat.
         */
        multiply(other) {
            const ax = this.x;
            const ay = this.y;
            const az = this.z;
            const aw = this.w;
            const bx = other.x;
            const by = other.y;
            const bz = other.z;
            const bw = other.w;
            return new Quat(ax * bw + aw * bx + ay * bz - az * by, ay * bw + aw * by + az * bx - ax * bz, az * bw + aw * bz + ax * by - ay * bx, aw * bw - ax * bx - ay * by - az * bz);
        }
        /**
         * Multiplies this quat by another, modifying its values in place.
         *
         * @param other - The other Quat to multiply.
         */
        multiplyInPlace(other) {
            const ax = this.x;
            const ay = this.y;
            const az = this.z;
            const aw = this.w;
            const bx = other.x;
            const by = other.y;
            const bz = other.z;
            const bw = other.w;
            this.set(ax * bw + aw * bx + ay * bz - az * by, ay * bw + aw * by + az * bx - ax * bz, az * bw + aw * bz + ax * by - ay * bx, aw * bw - ax * bx - ay * by - az * bz);
        }
        /**
         * Rotates a vector by this quaternion.
         * Don't forget to normalize the quaternion unless
         * you want axial translation as well as rotation.
         *
         * @param vec3 - The vec3 value.
         * @return - Returns a new Vec3.
         */
        rotateVec3(vec3) {
            const vq = new Quat(vec3.x, vec3.y, vec3.z, 0.0);
            const pq = this.multiply(vq).multiply(this.conjugate());
            return new Vec3(pq.x, pq.y, pq.z);
        }
        /**
         * Sets this quaternion to a rotation by the given angle about the X axis.
         *
         * @param rad - Angle (in radians) to rotate.
         */
        rotateX(rad) {
            rad *= 0.5;
            const ax = this.x;
            const ay = this.y;
            const az = this.z;
            const aw = this.w;
            const bx = Math.sin(rad);
            const bw = Math.cos(rad);
            this.x = ax * bw + aw * bx;
            this.y = ay * bw + az * bx;
            this.z = az * bw - ay * bx;
            this.w = aw * bw - ax * bx;
        }
        /**
         * Sets this quaternion to a rotation by the given angle about the Y axis.
         *
         * @param rad - Angle (in radians) to rotate.
         */
        rotateY(rad) {
            rad *= 0.5;
            const ax = this.x;
            const ay = this.y;
            const az = this.z;
            const aw = this.w;
            const by = Math.sin(rad);
            const bw = Math.cos(rad);
            this.x = ax * bw - az * by;
            this.y = ay * bw + aw * by;
            this.z = az * bw + ax * by;
            this.w = aw * bw - ay * by;
        }
        /**
         * Sets this quaternion to a rotation by the given angle about the Z axis.
         *
         * @param rad - Angle (in radians) to rotate.
         */
        rotateZ(rad) {
            rad *= 0.5;
            const ax = this.x;
            const ay = this.y;
            const az = this.z;
            const aw = this.w;
            const bz = Math.sin(rad);
            const bw = Math.cos(rad);
            this.x = ax * bw + ay * bz;
            this.y = ay * bw - ax * bz;
            this.z = az * bw + aw * bz;
            this.w = aw * bw - az * bz;
        }
        /**
         * Converts this Quat to a Mat3 (a 3x3 matrix).
         *
         * @return - TReturns a new Mat3.
         */
        toMat3() {
            const x = this.x;
            const y = this.y;
            const z = this.z;
            const w = this.w;
            const x2 = x + x;
            const y2 = y + y;
            const z2 = z + z;
            const xx = x * x2;
            const yx = y * x2;
            const yy = y * y2;
            const zx = z * x2;
            const zy = z * y2;
            const zz = z * z2;
            const wx = w * x2;
            const wy = w * y2;
            const wz = w * z2;
            const mat3 = new Mat3();
            mat3.m00 = 1 - yy - zz;
            mat3.m10 = yx - wz;
            mat3.m20 = zx + wy;
            mat3.m01 = yx + wz;
            mat3.m11 = 1 - xx - zz;
            mat3.m21 = zy - wx;
            mat3.m02 = zx - wy;
            mat3.m12 = zy + wx;
            mat3.m22 = 1 - xx - yy;
            return mat3;
        }
        /**
         * Calculates a Vec3 value aligned with the X axis of this quaternion.
         *
         * @return - The resulting Vec3 value
         */
        getXaxis() {
            const xy = this.x * this.y;
            const xz = this.x * this.z;
            const yy = this.y * this.y;
            const yw = this.y * this.w;
            const zz = this.z * this.z;
            const zw = this.z * this.w;
            return new Vec3(1.0 - 2.0 * (zz + yy), 2.0 * (xy + zw), 2.0 * (xz - yw));
        }
        /**
         * Calculates a Vec3 value aligned with the Y axis of this quaternion.
         *
         * @return - The resulting Vec3 value
         */
        getYaxis() {
            const xx = this.x * this.x;
            const xy = this.x * this.y;
            const xw = this.x * this.w;
            const yz = this.y * this.z;
            const zz = this.z * this.z;
            const zw = this.z * this.w;
            return new Vec3(2.0 * (xy - zw), 1.0 - 2.0 * (zz + xx), 2.0 * (yz + xw));
        }
        /**
         * Calculates a Vec3 value aligned with the Z axis of this quaternion.
         *
         * @return - The resulting Vec3 value
         */
        getZaxis() {
            const xx = this.x * this.x;
            const xz = this.x * this.z;
            const xw = this.x * this.w;
            const yy = this.y * this.y;
            const yz = this.y * this.z;
            const yw = this.y * this.w;
            // const temp = new Vec3()
            return new Vec3(2.0 * (yw + xz), 2.0 * (yz - xw), 1.0 - 2.0 * (yy + xx));
        }
        /**
         * Reflects this quaternion according to the axis provided.
         *
         * @param axisIndex - An integer with value of 0 for the X axis, 1 for the Y axis, and 2 for the Z axis.
         * @return - Returns a new Quat.
         */
        mirror(axisIndex) {
            switch (axisIndex) {
                case 0:
                    return new Quat(this.z, this.w, this.x, this.y);
                case 1:
                    return new Quat(-this.w, this.z, this.y, -this.x);
                case 2:
                    return new Quat(this.x, this.y, this.z, -this.w);
                case 0:
                default:
                    return new Quat(this.z, this.w, this.x, this.y);
            }
        }
        /**
         * Converts this Quat to a Mat4 (a 4x4 matrix).
         *
         * @return - Returns a new Mat4.
         */
        toMat4() {
            const x = this.x;
            const y = this.y;
            const z = this.z;
            const w = this.w;
            const x2 = x + x;
            const y2 = y + y;
            const z2 = z + z;
            const xx = x * x2;
            const yx = y * x2;
            const yy = y * y2;
            const zx = z * x2;
            const zy = z * y2;
            const zz = z * z2;
            const wx = w * x2;
            const wy = w * y2;
            const wz = w * z2;
            // Set the columns
            const mat4 = new Mat4();
            mat4.m00 = 1 - yy - zz;
            mat4.m10 = yx - wz;
            mat4.m20 = zx + wy;
            mat4.m01 = yx + wz;
            mat4.m11 = 1 - xx - zz;
            mat4.m21 = zy - wx;
            mat4.m02 = zx - wy;
            mat4.m12 = zy + wx;
            mat4.m22 = 1 - xx - yy;
            return mat4;
        }
        /**
         * Performs a linear interpolation of this Quat towards another Quat, returning the result as a new Quat.
         *
         * @param other  - The other Quat to interpolate towards.
         * @param t - Interpolation ratio.
         * @return - Returns a new Quat.
         */
        lerp(other, t) {
            const result = new Quat(this.x + t * (other.x - this.x), this.y + t * (other.y - this.y), this.z + t * (other.z - this.z), this.w + t * (other.w - this.w));
            result.normalizeInPlace();
            return result;
        }
        /**
         * Performs a spherical linear interpolation of this Quat towards another Quat, returning the result as a new Quat.
         *
         * @param other - The other Quat to interpolate towards.
         * @param t - Interpolation amount between the two inputs.
         * @return - Returns a new Quat.
         */
        slerp(other, lambda) {
            /// https://www.geometrictools.com/Documentation/FastAndAccurateSlerp.pdf
            const dotProduct = this.dot(other);
            if (dotProduct > 0.999)
                return this;
            // algorithm adapted from Shoemake's paper
            // lambda is in (0, π/2]
            const theta = Math.acos(dotProduct);
            const st = Math.sin(theta);
            const sut = Math.sin(lambda * theta);
            const sout = Math.sin((1 - lambda) * theta);
            const coeff1 = sout / st;
            const coeff2 = sut / st;
            const result = new Quat(coeff1 * this.x + coeff2 * other.x, coeff1 * this.y + coeff2 * other.y, coeff1 * this.z + coeff2 * other.z, coeff1 * this.w + coeff2 * other.w);
            result.normalizeInPlace();
            return result;
        }
        /**
         * Clones this Quat and returns a new Quat.
         *
         * @return - Returns a new Quat.
         */
        clone() {
            return new Quat(this.x, this.y, this.z, this.w);
        }
        /**
         * Returns the type as an array. Often used to pass types to the GPU.
         *
         * @return - Returns as an array.
         */
        asArray() {
            return [this.x, this.y, this.z, this.w];
        }
        fromArray(array) {
            this.x = array[0];
            this.y = array[1];
            this.z = array[2];
            this.w = array[3];
        }
        // ///////////////////////////
        // Persistence
        /**
         * Converts this Vec3 to a string in JSON format.
         *
         * @return - The return value.
         */
        toString() {
            // eslint-disable-next-line new-cap
            return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON());
        }
        /**
         * The toJSON method encodes this type as a json object for persistence.
         *
         * @return - The json object.
         */
        toJSON() {
            return {
                x: this.x,
                y: this.y,
                z: this.z,
                w: this.w,
            };
        }
        /**
         * The fromJSON method decodes a json object for this type.
         *
         * @param j - The json object.
         */
        fromJSON(j) {
            this.x = j.x;
            this.y = j.y;
            this.z = j.z;
            this.w = j.w;
            this.normalizeInPlace();
        }
        /**
         * Loads the state of the value from a binary reader.
         *
         * @param reader - The reader value.
         */
        readBinary(reader) {
            this.x = reader.loadFloat32();
            this.y = reader.loadFloat32();
            this.z = reader.loadFloat32();
            this.w = reader.loadFloat32();
        }
    }

    /* eslint-disable no-unused-vars */
    /**
     * Class representing an Xfo transform, which is a transformation decomposed into 3 component values. Translation, Orientation, and Scaling.
     */
    class Xfo {
        tr;
        ori;
        sc;
        /**
         * Initializes the Xfo object.
         *
         * @param tr - The translation value.
         * @param ori - The orientation value.
         * @param sc - The scaling value.
         */
        constructor(tr = new Vec3(), ori = new Quat(), sc = new Vec3(1, 1, 1)) {
            this.tr = tr;
            this.ori = ori;
            this.sc = sc;
        }
        /**
         * Sets the state of the Xfo object.
         *
         * @param tr - The translation value.
         * @param ori - The orientation value.
         * @param sc - The scaling value.
         */
        set(tr, ori, sc) {
            this.tr = tr;
            this.ori = ori;
            if (sc instanceof Vec3)
                this.sc = sc;
        }
        /**
         * Sets the state of the Xfo object using another Xfo object.
         *
         * @param other - The other Xfo to set from.
         */
        setFromOther(other) {
            this.tr = other.tr;
            this.ori = other.ori;
            this.sc = other.sc;
        }
        /**
         * Verifies that the Xfo object is an `identity`, checking that the translation, orientation and scaling attributes are in their initial state.
         *
         * @return - The return value.
         */
        isIdentity() {
            return this.tr.isNull() && this.ori.isIdentity() && this.sc.is111();
        }
        /**
         * Checks if this Vec3 contains the same values as the other Vec3.
         *
         * @param other - The other Vec3 to compare with.
         * @return - Returns `true` if are the same Vector, otherwise, `false`.
         */
        isEqual(other) {
            return this.tr.isEqual(other.tr) && this.ori.isEqual(other.ori) && this.sc.isEqual(other.sc);
        }
        /**
         * Returns true if this Vec2 is approximately the same as other.
         *
         * @param other - The other Vec3 to compare with.
         * @param precision - The precision to which the values must match.
         * @return - Returns true or false.
         */
        approxEqual(other, precision = Number.EPSILON) {
            return ((other.tr ? this.tr.approxEqual(other.tr, precision) : true) &&
                (other.ori ? this.ori.approxEqual(other.ori, precision) : true) &&
                (other.sc ? this.sc.approxEqual(other.sc, precision) : true));
        }
        /**
         * The setLookAt method.
         * @param pos - The position value.
         * @param target - The target value.
         * @param up - The up value.
         */
        setLookAt(pos, target, up) {
            // Note: We look along the -z axis. Negate the direction.
            const dir = pos.subtract(target);
            const dirLen = dir.length();
            if (dirLen < Number.EPSILON) {
                throw new Error('Invalid dir');
            }
            this.ori.setFromDirectionAndUpvector(dir, up);
            this.tr = pos;
        }
        /**
         * Multiplies two Xfo transforms.
         *
         * @param xfo - The xfo to multiply with.
         * @return - Returns an Xfo.
         */
        multiply(xfo) {
            let this_sc = this.sc;
            // check for non-uniform scale.
            if (Math.abs(this.sc.y - this.sc.x) > 0.001 ||
                Math.abs(this.sc.z - this.sc.x) > 0.001 ||
                Math.abs(this.sc.z - this.sc.y) > 0.001) {
                const this_mat4 = this.toMat4();
                const other_mat4 = xfo.ori.toMat4();
                const resM4 = this_mat4.multiply(other_mat4);
                this_sc = new Vec3(resM4.xAxis.length(), resM4.yAxis.length(), resM4.zAxis.length());
            }
            const result = new Xfo(this.tr.add(this.ori.rotateVec3(this.sc.multiply(xfo.tr))), this.ori.multiply(xfo.ori), this_sc.multiply(xfo.sc));
            return result;
        }
        /**
         * Returns the inverse of the Xfo object, but returns. the result as a new Xfo.
         *
         * @return - Returns a new Xfo.
         */
        inverse() {
            const result = new Xfo();
            result.ori = this.ori.inverse();
            // check for non-uniform scale.
            if (Math.abs(this.sc.y - this.sc.x) > 0.001 ||
                Math.abs(this.sc.z - this.sc.x) > 0.001 ||
                Math.abs(this.sc.z - this.sc.y) > 0.001) {
                const this_mat4 = this.toMat4().inverse();
                result.sc = new Vec3(this_mat4.xAxis.length(), this_mat4.yAxis.length(), this_mat4.zAxis.length());
            }
            else {
                result.sc = this.sc.inverse();
            }
            result.tr = result.ori.rotateVec3(this.tr.negate().multiply(result.sc));
            return result;
        }
        /**
         * Transforms Xfo object using a `Vec3` object. First scaling it, then rotating and finally adding the result to current translation object.
         *
         * @param vec3 - The vec3 value.
         * @return - The return value.
         */
        transformVec3(vec3) {
            return this.tr.add(this.ori.rotateVec3(this.sc.multiply(vec3)));
        }
        /**
         * Performs a linear interpolation between this Xfo and other.
         *
         * @param other - The other Xfo to interpolate towards.
         * @param t - Interpolation ratio.
         * @return - Returns a new Xfo.
         */
        lerp(other, t) {
            return new Xfo(this.tr.lerp(other.tr, t), this.ori.slerp(other.ori, t), this.sc.lerp(other.sc, t));
        }
        /**
         * Converts this Xfo to a Mat4 (a 4x4 matrix).
         *
         * @return - Returns a new Mat4.
         */
        toMat4() {
            const scl = new Mat4(this.sc.x, 0, 0, 0, 0, this.sc.y, 0, 0, 0, 0, this.sc.z, 0, 0, 0, 0, 1.0);
            const rot = this.ori.toMat4();
            const trn = new Mat4();
            trn.translation = this.tr;
            return trn.multiply(rot).multiply(scl);
        }
        /**
         * Sets the state of the Xfo object using Mat4.
         * @param mat4 - The mat4 value.
         */
        setFromMat4(mat4) {
            this.tr = mat4.translation;
            this.ori.setFromMat4(mat4);
            this.sc.set(mat4.xAxis.length(), mat4.yAxis.length(), mat4.zAxis.length());
        }
        /**
         * Clones this Xfo and returns a new Xfo.
         *
         * @return - Returns a new Xfo.
         */
        clone() {
            return new Xfo(this.tr.clone(), this.ori.clone(), this.sc.clone());
        }
        // ///////////////////////////
        // Persistence
        /**
         * The toJSON method encodes this type as a json object for persistence.
         *
         * @return - The json object.
         */
        toJSON() {
            const j = {
                tr: this.tr.toJSON(),
                ori: this.ori.toJSON(),
                sc: this.sc.toJSON(),
            };
            return j;
        }
        /**
         * The fromJSON method decodes a json object for this type.
         *
         * @param j - The json object.
         */
        fromJSON(j) {
            this.tr.fromJSON(j.tr);
            this.ori.fromJSON(j.ori);
            if (j.sc) {
                this.sc.fromJSON(j.sc);
            }
        }
        /**
         * Loads the state of the value from a binary reader.
         *
         * @param reader - The reader value.
         */
        readBinary(reader) {
            this.tr.readBinary(reader);
            this.ori.readBinary(reader);
            this.sc.readBinary(reader);
        }
        /**
         * The fromJSON method decodes a json object for this type.
         *
         * @return - The return value.
         */
        toString() {
            // eslint-disable-next-line new-cap
            return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON());
        }
    }

    /**
     * Represents a box in 2D space. Needing two Vec2 vectors describing the corners
     */
    class Box2 {
        p0;
        p1;
        /**
         * Creates a Box2 object using Vec2s.
         * In case the parameters are not passed by, their values are pre-defined:
         *
         * p0 is a Vec2 with {@link https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/POSITIVE_INFINITY|`Number.POSITIVE_INFINITY`}
         *
         * p1 is a Vec2 with {@link https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/NEGATIVE_INFINITY|`Number.NEGATIVE_INFINITY`}
         *
         * @param p0 - A point representing the corners of a 2D box.
         * @param p1 - A point representing the corners of a 2D box.
         */
        constructor(p0, p1) {
            if (p0 instanceof Vec2) {
                this.p0 = p0;
            }
            else {
                this.p0 = new Vec2(Number.POSITIVE_INFINITY, Number.POSITIVE_INFINITY);
            }
            if (p1 instanceof Vec2) {
                this.p1 = p1;
            }
            else {
                this.p1 = new Vec2(Number.NEGATIVE_INFINITY, Number.NEGATIVE_INFINITY);
            }
        }
        /**
         * Sets both Vec2 points
         *
         * @param p0 - A point representing the corners of a 2D box.
         * @param p1 - A point representing the corners of a 2D box.
         */
        set(p0, p1) {
            this.p0 = p0;
            this.p1 = p1;
        }
        /**
         * Resets the box2 back to an uninitialized state.
         *
         * @see {@link https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/POSITIVE_INFINITY|`Number.POSITIVE_INFINITY`}
         * and {@link https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/NEGATIVE_INFINITY|`Number.NEGATIVE_INFINITY`}
         */
        reset() {
            this.p0.x = Number.POSITIVE_INFINITY;
            this.p1.x = Number.NEGATIVE_INFINITY;
            this.p0.y = Number.POSITIVE_INFINITY;
            this.p1.y = Number.NEGATIVE_INFINITY;
        }
        /**
         * Returns `true` if the box has been expanded to contain a point.
         *
         * @return - The return value.
         */
        isValid() {
            return (this.p0.x != Number.POSITIVE_INFINITY &&
                this.p1.x != Number.NEGATIVE_INFINITY &&
                this.p0.y != Number.POSITIVE_INFINITY &&
                this.p1.y != Number.NEGATIVE_INFINITY);
        }
        /**
         * Expands the Box2 to contain the new point.
         *
         * @param point - A point represents the corners of a 2D box.
         */
        addPoint(point) {
            if (this.p0.x == Number.POSITIVE_INFINITY || point.x < this.p0.x)
                this.p0.x = point.x;
            if (this.p0.y == Number.POSITIVE_INFINITY || point.y < this.p0.y)
                this.p0.y = point.y;
            if (this.p1.y == Number.NEGATIVE_INFINITY || point.x > this.p1.x)
                this.p1.x = point.x;
            if (this.p1.y == Number.NEGATIVE_INFINITY || point.y > this.p1.y)
                this.p1.y = point.y;
        }
        /**
         * Returns the length of the diagonal of the box.
         *
         * @return - Returns the distance.
         */
        size() {
            return this.p1.distanceTo(this.p0);
        }
        /**
         * Returns the size of a Box2 - the same as size().
         *
         * @return - Returns a Vec2.
         */
        diagonal() {
            return this.p1.subtract(this.p0);
        }
        /**
         * Returns the center point of a Box2.
         *
         * @return - Returns a Vec2.
         */
        center() {
            const result = this.p1.subtract(this.p0);
            result.scaleInPlace(0.5);
            result.addInPlace(this.p0);
            return result;
        }
        /**
         * Clones this Vec2 and returns a new Vec2.
         *
         * @return - Returns a new Vec2.
         */
        clone() {
            return new Box2(this.p0.clone(), this.p1.clone());
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * Encodes `Box2` Class as a JSON object for persistence.
         *
         * @return - The json object.
         */
        toJSON() {
            return {
                p0: this.p0.toJSON(),
                p1: this.p1.toJSON(),
            };
        }
        /**
         * Decodes a JSON object to set the state of this class.
         *
         * @param j - The json object.
         */
        fromJSON(j) {
            // We need to verify that p0 and p1 axes are numeric, so in case they are not, we restore them to their default values.
            // This, because 'Infinity' and '-Infinity' are stringified as 'null'.
            const p0 = {
                x: MathFunctions.isNumeric(j.p0.x) ? j.p0.x : Number.POSITIVE_INFINITY,
                y: MathFunctions.isNumeric(j.p0.y) ? j.p0.y : Number.POSITIVE_INFINITY,
            };
            const p1 = {
                x: MathFunctions.isNumeric(j.p1.x) ? j.p1.x : Number.NEGATIVE_INFINITY,
                y: MathFunctions.isNumeric(j.p1.y) ? j.p1.y : Number.NEGATIVE_INFINITY,
            };
            this.p0.fromJSON(p0);
            this.p1.fromJSON(p1);
        }
        /**
         * Calls `toJSON` method and stringifies it.
         *
         * @return - The return value.
         */
        toString() {
            // eslint-disable-next-line new-cap
            return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON());
        }
    }

    /* eslint-disable new-cap */
    /**
     * Class representing a mathematical sphere, as opposed to the Sphere class derived from ProceduralMesh.
     *
     */
    class SphereType {
        pos;
        radius;
        /**
         * Create a sphere.
         * @param pos - The position of the sphere.
         * @param radius - The radius of the sphere.
         */
        constructor(pos = new Vec3(), radius = 0) {
            if (pos instanceof Vec3) {
                this.pos = pos;
            }
            else {
                this.pos = new Vec3();
            }
            this.radius = radius;
        }
        /**
         * Clones this sphere and returns a new sphere.
         *
         * @return - Returns a new sphere.
         */
        clone() {
            return new SphereType(this.pos.clone(), this.radius);
        }
        /**
         * Checks if this sphere intersects a box.
         *
         * @param box - The box value.
         * @return - The return value.
         */
        intersectsBox(box) {
            return box.intersectsSphere(this);
        }
        // ///////////////////////////
        // Persistence
        /**
         * The toJSON method encodes this type as a json object for persistence.
         *
         * @return - The json object.
         */
        toJSON() {
            return {
                pos: this.pos.toJSON(),
                radius: this.radius,
            };
        }
        /**
         * Calls `toJSON` method and stringifies it.
         *
         * @return - The return value.
         */
        toString() {
            return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON());
        }
    }

    /**
     * Class representing a box in 3D space.
     * Represents a box in 3D space defined by two Vec3 values which define opposing corners of the box.
     */
    class Box3 {
        p0;
        p1;
        /**
         * Creates a Box3 object using Vec3s.
         * In case the parameters are not passed by, their values are pre-defined:
         *
         * p0 is a Vec2 with {@link https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/POSITIVE_INFINITY|`Number.POSITIVE_INFINITY`}
         *
         * p1 is a Vec2 with {@link https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/NEGATIVE_INFINITY|`Number.NEGATIVE_INFINITY`}
         *
         * @param p0 - A point representing the corners of a 3D box.
         * @param p1 - A point representing the corners of a 3D box.
         */
        constructor(p0 = new Vec3(Number.POSITIVE_INFINITY, Number.POSITIVE_INFINITY, Number.POSITIVE_INFINITY), p1 = new Vec3(Number.NEGATIVE_INFINITY, Number.NEGATIVE_INFINITY, Number.NEGATIVE_INFINITY)) {
            this.p0 = p0;
            this.p1 = p1;
        }
        /**
         * Getter for the lower (x, y, z) boundary of the box.
         *
         * @return - Returns the minimum Vec3.
         */
        get min() {
            return this.p0;
        }
        /**
         * Getter for the upper (x, y, z) boundary of the box.
         *
         * @return - Returns the maximum Vec3.
         */
        get max() {
            return this.p1;
        }
        /**
         * Sets both Vec3 points
         *
         * @param p0 - A point representing the corners of a 3D box.
         * @param p1 - A point representing the corners of a 3D box.
         */
        set(p0, p1) {
            this.p0 = p0;
            this.p1 = p1;
        }
        /**
         * Resets the box3 back to an uninitialized state.
         */
        reset() {
            this.p0.x = Number.POSITIVE_INFINITY;
            this.p0.y = Number.POSITIVE_INFINITY;
            this.p0.z = Number.POSITIVE_INFINITY;
            this.p1.x = Number.NEGATIVE_INFINITY;
            this.p1.y = Number.NEGATIVE_INFINITY;
            this.p1.z = Number.NEGATIVE_INFINITY;
        }
        /**
         * Returns `true` if the box has been expanded to contain a point.
         *
         * @return - The return value.
         */
        isValid() {
            return (this.p0.x != Number.POSITIVE_INFINITY &&
                this.p1.x != Number.NEGATIVE_INFINITY &&
                this.p0.y != Number.POSITIVE_INFINITY &&
                this.p1.y != Number.NEGATIVE_INFINITY &&
                this.p0.z != Number.POSITIVE_INFINITY &&
                this.p1.z != Number.NEGATIVE_INFINITY);
        }
        /**
         * Expands the Box3 to contain the new point.
         *
         * @param point - A point represents the corners of a 3D box.
         */
        addPoint(point) {
            if (point.x != Number.POSITIVE_INFINITY && point.x != Number.NEGATIVE_INFINITY) {
                if (point.x < this.p0.x)
                    this.p0.x = point.x;
                if (point.x > this.p1.x)
                    this.p1.x = point.x;
            }
            if (point.y != Number.POSITIVE_INFINITY && point.y != Number.NEGATIVE_INFINITY) {
                if (point.y < this.p0.y)
                    this.p0.y = point.y;
                if (point.y > this.p1.y)
                    this.p1.y = point.y;
            }
            if (point.z != Number.POSITIVE_INFINITY && point.z != Number.NEGATIVE_INFINITY) {
                if (point.z < this.p0.z)
                    this.p0.z = point.z;
                if (point.z > this.p1.z)
                    this.p1.z = point.z;
            }
        }
        /**
         * Adds `Box3` to this `Box3`, of the Xfo instance is passed in the parameters
         * it proceeds to apply the transform for the Vec3.
         *
         * @param box3 - A 3D box.
         * @param xfo - A 3D transform.
         */
        addBox3(box3, transform) {
            if (transform) {
                // Transform each corner of the Box3 into the new coordinate system.
                this.addPoint(transform.transformVec3(box3.p0));
                this.addPoint(transform.transformVec3(new Vec3(box3.p0.x, box3.p0.y, box3.p1.z)));
                this.addPoint(transform.transformVec3(new Vec3(box3.p0.x, box3.p1.y, box3.p0.z)));
                this.addPoint(transform.transformVec3(new Vec3(box3.p1.x, box3.p0.y, box3.p0.z)));
                this.addPoint(transform.transformVec3(new Vec3(box3.p0.x, box3.p1.y, box3.p1.z)));
                this.addPoint(transform.transformVec3(new Vec3(box3.p1.x, box3.p0.y, box3.p1.z)));
                this.addPoint(transform.transformVec3(new Vec3(box3.p1.x, box3.p1.y, box3.p0.z)));
                this.addPoint(transform.transformVec3(box3.p1));
            }
            else {
                this.addPoint(box3.p0);
                this.addPoint(box3.p1);
            }
        }
        /**
         * Returns the length of the diagonal of the box.
         *
         * @return - Returns the distance.
         */
        size() {
            return this.p1.distanceTo(this.p0);
        }
        /**
         * Returns the diagonal vector of the B=box from p0 to p1.
         *
         * @return - Returns a Box3.
         */
        diagonal() {
            return this.p1.subtract(this.p0);
        }
        /**
         * Returns the center point of a Box3.
         *
         * @return - Returns a Vec3.
         */
        center() {
            const result = this.p1.subtract(this.p0);
            result.scaleInPlace(0.5);
            result.addInPlace(this.p0);
            return result;
        }
        /**
         * Converts this Box3 to a Mat4 (a 4x4 matrix). The returned mat4 would transform a unit cube into the shape of the Bounding box.
         *
         * @return - Returns a new Mat4.
         */
        toMat4() {
            const scx = this.p1.x - this.p0.x;
            const scy = this.p1.y - this.p0.y;
            const scz = this.p1.z - this.p0.z;
            return new Mat4(scx, 0, 0, 0, 0, scy, 0, 0, 0, 0, scz, 0, this.p0.x, this.p0.y, this.p0.z, 1.0);
        }
        /**
         * Calculates and returns the bounding Sphere of the Box3
         *
         * @return - The return value.
         */
        getBoundingSphere() {
            return new SphereType(this.center(), this.diagonal().length() * 0.5);
        }
        /**
         * Determines if this Box3 intersects a given box value.
         *
         * @param box - The box to check for intersection against.
         * @return - Returns true if the shapes intersect.
         */
        intersectsBox(box) {
            // Using 6 splitting planes to rule out intersections.
            return box.max.x < this.min.x ||
                box.min.x > this.max.x ||
                box.max.y < this.min.y ||
                box.min.y > this.max.y ||
                box.max.z < this.min.z ||
                box.min.z > this.max.z
                ? false
                : true;
        }
        /**
         * Determines if this Box3 intersects a sphere.
         *
         * @param sphere - The sphere to check for intersection against.
         * @return - Returns true if the shapes intersect.
         */
        intersectsSphere(sphere) {
            let closestPoint = new Vec3();
            // Find the point on the AABB closest to the sphere center.
            // this.clampPoint( sphere.center, closestPoint );
            // If that point is inside the sphere, the AABB and sphere intersect.
            return closestPoint.distanceTo(sphere.pos) <= sphere.radius * sphere.radius;
        }
        /**
         * Determines if this Box3 intersects a plane.
         *
         * @param plane - The plane to check for intersection against.
         * @return - The return value.
         */
        intersectsPlane(plane) {
            // We compute the minimum and maximum dot product values. If those values
            // are on the same side (back or front) of the plane, then there is no intersection.
            let min;
            let max;
            if (plane.normal.x > 0) {
                min = plane.normal.x * this.min.x;
                max = plane.normal.x * this.max.x;
            }
            else {
                min = plane.normal.x * this.max.x;
                max = plane.normal.x * this.min.x;
            }
            if (plane.normal.y > 0) {
                min += plane.normal.y * this.min.y;
                max += plane.normal.y * this.max.y;
            }
            else {
                min += plane.normal.y * this.max.y;
                max += plane.normal.y * this.min.y;
            }
            if (plane.normal.z > 0) {
                min += plane.normal.z * this.min.z;
                max += plane.normal.z * this.max.z;
            }
            else {
                min += plane.normal.z * this.max.z;
                max += plane.normal.z * this.min.z;
            }
            return min <= -plane.w && max >= -plane.w;
        }
        /**
         * Clones this Box3 and returns a new Box3.
         * @return - Returns a new Box3.
         */
        clone() {
            return new Box3(this.p0.clone(), this.p1.clone());
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * Encodes `Box3` Class as a JSON object for persistence.
         *
         * @return - The json object.
         */
        toJSON() {
            return {
                p0: this.p0.toJSON(),
                p1: this.p1.toJSON(),
            };
        }
        /**
         * Decodes a JSON object to set the state of this class.
         *
         * @param j - The json object.
         */
        fromJSON(j) {
            // We need to verify that p0 and p1 axes are numeric, so in case they are not, we restore them to their default values.
            // This, because 'Infinity' and '-Infinity' are stringified as 'null'.
            const p0 = {
                x: MathFunctions.isNumeric(j.p0.x) ? j.p0.x : Number.POSITIVE_INFINITY,
                y: MathFunctions.isNumeric(j.p0.y) ? j.p0.y : Number.POSITIVE_INFINITY,
                z: MathFunctions.isNumeric(j.p0.z) ? j.p0.z : Number.POSITIVE_INFINITY,
            };
            const p1 = {
                x: MathFunctions.isNumeric(j.p1.x) ? j.p1.x : Number.NEGATIVE_INFINITY,
                y: MathFunctions.isNumeric(j.p1.y) ? j.p1.y : Number.NEGATIVE_INFINITY,
                z: MathFunctions.isNumeric(j.p1.z) ? j.p1.z : Number.NEGATIVE_INFINITY,
            };
            this.p0.fromJSON(p0);
            this.p1.fromJSON(p1);
        }
        /**
         * Calls `toJSON` method and stringifies it.
         *
         * @return - The return value.
         */
        toString() {
            // eslint-disable-next-line new-cap
            return StringFunctions.stringifyJSONWithFixedPrecision(this.toJSON());
        }
    }

    let numBaseItems = 0;
    /**
     * Base class for Items in the scene. It can be parameterized and can emit events.
     *
     * **Events**
     * * **nameChanged:** Emitted every time the Item's name is change. mostly in `setName` method.
     * * **selectedChanged:** Emitted `selected` status changes, mostly in `setSelected` method.
     *
     * @extends {EventEmitter}
     */
    class BaseItem extends EventEmitter {
        #name;
        #ownerItem = undefined;
        selected = false;
        /**
         * Create a base item by defining its name.
         *
         * @param name - The name of the base item.
         */
        constructor(name = '') {
            super();
            this.#name = name;
            numBaseItems++;
        }
        // ////////////////////////////////////////
        // Static Methods
        /**
         * The getNumBaseItems method returns the total number of base items created.
         * This method is used in debugging memory consumption.
         *
         * @return - Returns the total number of base items created.
         */
        static getNumBaseItems() {
            return numBaseItems;
        }
        // ////////////////////////////////////////
        // Name and Path
        /**
         * Returns the name of the base item.
         *
         * @return - Returns the base item name.
         */
        get name() {
            return this.#name;
        }
        /**
         * Sets the name of the base item(Updates path).
         *
         * @emits `nameChanged` with `newName` and `oldName` data.
         * @param name - The base item name.
         */
        set name(value) {
            if (this.#name != value) {
                const oldName = this.#name;
                this.#name = value;
                const event = new NameChangedEvent(oldName, value);
                this.emit('nameChanged', event);
            }
        }
        /**
         * Returns the current path of the item in the tree as an array of names.
         *
         * @return - Returns an array.
         */
        get path() {
            return this.getPath();
        }
        /**
         * Returns the current path of the item in the tree as an array of names.
         *
         * @return - Returns an array.
         */
        get ownerItem() {
            return this.#ownerItem;
        }
        /**
         * Returns the name of the base item.
         *
         * @return - Returns the base item name.
         */
        getName() {
            return this.#name;
        }
        /**
         * Sets the name of the base item(Updates path).
         *
         * @emits `nameChanged` with `newName` and `oldName` data.
         * @param name - The base item name.
         */
        setName(name) {
            this.name = name;
        }
        /**
         * Returns the current path of the item in the tree as an array of names.
         *
         * @return - Returns an array.
         */
        getPath() {
            if (this.#ownerItem == undefined)
                return [this.#name];
            return [...this.#ownerItem.getPath(), this.#name];
        }
        // Path Traversal
        /**
         * The resolvePath method traverses the subtree from this item down
         * matching each name in the path with a child until it reaches the
         * end of the path.
         *
         * @param path - The path value.
         * @param index - The index value.
         * @return - The return value.
         */
        resolvePath(path, index = 0) {
            if (index == 0) {
                if (path[0] == '.' || path[0] == this.#name)
                    index++;
            }
            if (path[index] == '..') {
                if (this.#ownerItem) {
                    return this.#ownerItem.resolvePath(path, index + 1);
                }
                else {
                    throw Error('this.#ownerItem is undefined');
                }
            }
            if (index == path.length) {
                return this;
            }
        }
        // ////////////////////////////////////////
        // Owner Item
        /**
         * The getOwner method returns the current owner of the item.
         * The item is a child of the current owner.
         *
         * @return - Returns the current owner.
         */
        getOwner() {
            return this.#ownerItem;
        }
        /**
         * The setOwner method assigns a new owner to the item.
         *
         * @param ownerItem - The new owner item.
         */
        setOwner(ownerItem) {
            this.#ownerItem = ownerItem;
        }
        // ////////////////////////////////////////
        // Selection
        /**
         * The isSelected method.
         * @return - The return value.
         */
        isSelected() {
            return this.selected;
        }
        /**
         * Changes the current state of the selection of this item.
         *
         * @emits `selectedChanged` with selected state
         * @param sel - Boolean indicating the new selection state.
         */
        setSelected(sel) {
            this.selected = sel;
            let event = new SelectedEvent(this.selected);
            this.emit('selectedChanged', event);
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * Encodes the current object as a json object.
         *
         * @param context - The context value.
         * @return - Returns the json object.
         */
        toJSON(context) {
            const json = {
                type: this.getClassName(),
                name: this.#name,
            };
            return json;
        }
        /**
         * Decodes a json object for this type.
         *
         * @param json - The json object this item must decode.
         * @param context - The context value.
         */
        fromJSON(json, context) {
            if (json.name)
                this.#name = json.name;
        }
        /**
         * Sets state of current Item(Including parameters) using a binary reader object.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            // read the type, but don't use it. This line must not be removed.
            // as the binary pointer is incremented.
            /*const type = */ reader.loadStr();
            this.setName(reader.loadStr());
        }
        /**
         * Converts object's JSON value and converts it to a string.
         * @param context
         * @return - String of object's parameter list state.
         */
        toString(context) {
            return JSON.stringify(this.toJSON(), null, 2);
        }
        // ////////////////////////////////////////
        // Clone and Destroy
        /**
         * Clones this base item and returns a new base item.
         *
         * **Note:** Each class should implement clone to be clonable.
         * @param context - The context value.
         */
        clone(context) {
            throw new Error(this.constructor.name + ' does not implement its clone method');
        }
        /**
         * When a BaseItem is cloned, initially the constructor is
         * called to generate a new instance. This instance then copies
         * its values from the source using this method.
         * This method copies any relevant data from the source object to
         * ensure that it represents a valid clone.
         * Derived classes override this method to copy any relevant
         * data from the source object.
         *
         * @param src - The BaseItem to copy from.
         * @param context - The context value
         */
        copyFrom(src, context) {
            this.setName(src.getName());
        }
    }

    /**
     * Class that allows other classes to be parameterized by `Parameter` type of objects.
     * Not only hosting parameters, but their events.
     *
     * @extends {BaseItem}
     */
    class ParameterOwner extends BaseItem {
        paramEventListenerIDs = {};
        paramMapping = {};
        params = [];
        deprecatedParamMapping = {};
        /**
         * Creates an instance of ParameterOwner by initializing parameter hosting mappings and events.
         *
         * Every Object has a unique identifier which is based on a counter that is incremented.
         */
        constructor(name) {
            super(name);
        }
        /**
         * Returns the number of parameters current object has.
         */
        getNumParameters() {
            return this.params.length;
        }
        /**
         * Returns the number of parameters current object has.
         */
        get numParameters() {
            return this.params.length;
        }
        /**
         * Returns all the parameters of the object.
         *
         * @return - Parameter List
         */
        getParameters() {
            return this.params;
        }
        /**
         * Returns the index of a parameter in parameter list.
         *
         * @param paramName - Name of the parameter.
         * @return - Position in the array
         */
        getParameterIndex(paramName) {
            return this.paramMapping[paramName];
        }
        /**
         * Returns `Parameter` object in a given index
         *
         * @param index - Position of the parameter in the array
         * @return - Parameter object value
         */
        getParameterByIndex(index) {
            return this.params[index];
        }
        /**
         * Validates if the specified parameter exists in the object.
         *
         * @param paramName - The parameter name.
         * @return - The return value.
         */
        hasParameter(paramName) {
            return paramName in this.paramMapping;
        }
        /**
         * Add a mapping from one name to a new parameter.
         * This is used to handle migrating parameters to new names.
         *
         * @param key - The parameter name.
         * @param paramName - The parameter name.
         * @return - The return value.
         */
        addParameterDeprecationMapping(key, paramName) {
            this.deprecatedParamMapping[key] = paramName;
        }
        /**
         * Returns `Parameter` object using the given name
         *
         * @param paramName - The parameter name.
         * @return - Parameter object value
         */
        getParameter(paramName) {
            let index = this.paramMapping[paramName];
            if (index == undefined) {
                const newParamName = this.deprecatedParamMapping[paramName];
                if (!newParamName) {
                    // TODO: Should this method not return null?
                    return null;
                    // throw Error(`No Parameter with that name exists: ${paramName} `)
                }
                else {
                    console.warn(`Parameter name ${paramName} is now deprecated. Please use ${newParamName} instead.`);
                    index = this.paramMapping[newParamName];
                }
            }
            return this.params[index];
        }
        /**
         * This method can be overridden in derived classes
         * to perform general updates (see GLPass or BaseItem).
         * @param event - The event object emitted by the parameter.
         * @private
         */
        parameterValueChanged(event) {
            this.emit('parameterValueChanged', event);
        }
        /**
         * Adds `Parameter` object to the owner's parameter list.
         *
         * @emits `parameterAdded` with the name of the param.
         * @param param - The parameter to add.
         * @return - With `owner` and `valueChanged` event set.
         */
        addParameter(param) {
            return this.insertParameter(param, this.params.length);
        }
        /**
         * Adds `Parameter` object to the owner's parameter list using the index.
         * It replaces the event in the specified index.
         *
         *
         * @emits `parameterAdded` with the name of the param.
         * @param param - The parameter to insert.
         * @param index - The index value.
         * @return - With `owner` and `valueChanged` event set.
         */
        insertParameter(param, index) {
            const name = param.getName();
            if (this.paramMapping[name] != undefined) {
                console.warn('Replacing Parameter:' + name);
                this.removeParameter(name);
            }
            param.setOwner(this);
            this.paramEventListenerIDs[name] = param.on('valueChanged', (event) => {
                // Note: spread operators cause errors on iOS 11.
                const newEvent = { param };
                for (const key in event)
                    newEvent[key] = event[key];
                this.parameterValueChanged(newEvent);
            });
            this.params.splice(index, 0, param);
            for (let i = index; i < this.params.length; i++) {
                this.paramMapping[this.params[i].getName()] = i;
            }
            const event = new ParameterAddedEvent(name);
            this.emit('parameterAdded', event);
            return param;
        }
        /**
         * Removes `Parameter` from owner, by using parameter's name.
         * @emits `parameterRemoved` with the name of the param.
         * @param name - The parameter name.
         */
        removeParameter(name) {
            if (this.paramMapping[name] == undefined) {
                throw new Error('Unable to remove Parameter:' + name);
            }
            const index = this.paramMapping[name];
            const param = this.params[this.paramMapping[name]];
            param.off('valueChanged', this.paramEventListenerIDs[name]);
            this.params.splice(index, 1);
            delete this.paramMapping[name];
            for (let i = index; i < this.params.length; i++) {
                this.paramMapping[this.params[i].getName()] = i;
            }
            const event = new ParameterRemovedEvent(name);
            this.emit('parameterRemoved', event);
        }
        /**
         * Replaces old `Parameter` by passing a new one with the same name.
         *
         * @param param - The parameter to replace.
         * @return - `Parameter` with `valueChanged` event set.
         */
        replaceParameter(param) {
            const name = param.getName();
            if (this.paramMapping[name] == undefined) {
                throw new Error('Unable to replace Parameter:' + name);
            }
            const index = this.paramMapping[name];
            this.removeParameter(name);
            this.insertParameter(param, index);
            return param;
        }
        /**
         * The resolvePath method traverses the subtree from this item down
         * matching each name in the path with a child until it reaches the
         * end of the path.
         *
         * @param path - The path value.
         * @param index - The index value.
         * @return - The return value.
         */
        resolvePath(path, index = 0) {
            if (index == 0) {
                if (path[0] == '.' || path[0] == this.name)
                    index++;
            }
            if (path[index] == '..') {
                if (this.ownerItem) {
                    return this.ownerItem.resolvePath(path, index + 1);
                }
                else {
                    throw Error('this.ownerItem is undefined');
                }
            }
            if (index == path.length) {
                return this;
            }
            // Maybe the name is a parameter name.
            const param = this.getParameter(path[index]);
            if (param) {
                if (index < path.length) {
                    return param.resolvePath(path, index + 1);
                }
                return param;
            }
            throw new Error(`Unable to resolve path : [${path.toString()}] after: ${this.getName()} \nNo child or parameter called : "${path[index]}"`);
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * The toJSON method encodes this type as a json object for persistence.
         *
         * @param context - The context value.
         * @return - Returns the json object.
         */
        toJSON(context) {
            const json = super.toJSON(context);
            const paramsJSON = {};
            let savedParams = 0;
            for (const param of this.params) {
                if (param.isDrivenByOperator())
                    continue;
                const paramJSON = param.toJSON(context);
                if (paramJSON) {
                    paramsJSON[param.getName()] = paramJSON;
                    savedParams++;
                }
            }
            if (savedParams > 0)
                json.params = paramsJSON;
            return json;
        }
        /**
         * The fromJSON method decodes a json object for this type.
         *
         * @param json - The json object this item must decode.
         * @param context - The context value.
         */
        fromJSON(json, context) {
            super.fromJSON(json, context);
            if (json.params) {
                for (const key in json.params) {
                    const pj = json.params[key];
                    if (pj.paramPath) {
                        context?.resolvePath(pj.paramPath, (param) => {
                            this.replaceParameter(param);
                        }, () => {
                            console.warn('Unable to resolve shared parameter:' + pj.paramPath);
                        });
                    }
                    else {
                        let param = this.getParameter(key);
                        if (!param && pj.type && pj.name) {
                            param = Registry.constructClass(pj.type);
                            if (!param) {
                                console.error('Unable to construct prop:' + pj.name + ' of type:' + pj.type);
                                continue;
                            }
                            param.setName(pj.name);
                            this.addParameter(param);
                        }
                        if (param)
                            param.fromJSON(pj, context);
                    }
                }
            }
        }
        /**
         * Uses passed in BinReader object(containing an Int32 array with all the parameters) to reconstruct all parameters state.
         *
         * In each iteration of the array, propType and propName are extracted and
         * used to build the right `Parameter` class. Then all of them are added to the object.
         *
         * @emits `parameterAdded` with the name of the param.
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            super.readBinary(reader, context);
            this.readBinaryParams(reader, context);
        }
        readBinaryParams(reader, context) {
            if (context?.versions['zea-engine'].compare([0, 0, 3]) >= 0) {
                const numProps = reader.loadUInt32();
                for (let i = 0; i < numProps; i++) {
                    const propType = reader.loadStr();
                    const propName = reader.loadStr();
                    let param = this.getParameter(propName);
                    if (!param) {
                        param = Registry.constructClass(propType);
                        if (!param) {
                            console.error('Unable to construct prop:' + propName + ' of type:' + propType);
                            continue;
                        }
                        param.setName(propName);
                        this.addParameter(param);
                    }
                    param.readBinary(reader, context);
                }
            }
        }
        // ////////////////////////////////////////
        // Clone and Destroy
        /**
         * Clones this base item and returns a new base item.
         *
         * **Note:** Each class should implement clone to be clonable.
         * @param context - The context value.
         */
        clone(context) {
            throw new Error(this.constructor.name + ' does not implement its clone method');
        }
        /**
         * Copies Parameters from another `ParameterOwner` to current object.
         *
         * @param src - The ParameterOwner copy from.
         * @param context - The context value
         */
        copyFrom(src, context) {
            if (!(src instanceof ParameterOwner)) {
                throw new Error('cannot copy from src');
            }
            super.copyFrom(src, context);
            // Note: Loop over the parameters in reverse order,
            // this is because often, parameter dependencies
            // are bottom to top (bottom params dependent on higher params).
            // This means that as a parameter is set with a new value
            // it will dirty the params below it.
            let i = src.getNumParameters();
            while (i--) {
                const srcParam = src.getParameterByIndex(i);
                const param = this.getParameter(srcParam.getName());
                if (param) {
                    // Note: we are not cloning the values.
                    param.copyFrom(srcParam);
                }
                else {
                    this.addParameter(srcParam.clone());
                }
            }
        }
    }

    /* eslint-disable @typescript-eslint/no-explicit-any */
    function approxEqual(a, b) {
        for (let i = 0; i < a.length; i++) {
            if (Math.abs(b[i] - a[i]) > 0.001)
                return false;
        }
        return true;
    }
    const resizeArray = (inArray, newSize) => {
        if (inArray instanceof Uint8Array) {
            const newArray = new Uint8Array(newSize);
            newArray.set(inArray);
            return newArray;
        }
        else if (inArray instanceof Int8Array) {
            const newArray = new Int8Array(newSize);
            newArray.set(inArray);
            return newArray;
        }
        else if (inArray instanceof Uint16Array) {
            const newArray = new Uint16Array(newSize);
            newArray.set(inArray);
            return newArray;
        }
        else {
            const newArray = new Float32Array(newSize);
            newArray.set(inArray);
            return newArray;
        }
    };
    class Attribute extends BaseClass {
        dataTypeName;
        stride;
        initValue = Number.NaN;
        normalized;
        data;
        mesh;
        splitValues = [];
        splits = {};
        constructor(dataTypeName, stride, initValue = Number.NaN) {
            super();
            this.dataTypeName = dataTypeName;
            this.stride = stride;
            this.initValue = initValue;
            this.init();
        }
        init() {
            this.data = new Float32Array(0);
            this.initRange(0);
        }
        /**
         * Sets the Mesh reference to the VertexAttribute. This is needed for attributes
         * assigned to meshes, and is used to calculate face vertex indices.
         * > Note: the mesh automatically calls this method when a vertex attribute is assigned.
         *
         * @param mesh - The mesh object
         */
        setMesh(mesh) {
            this.mesh = mesh;
        }
        /**
         * Returns the backing array for this attribute
         *
         * @return - The return value.
         */
        asArray() {
            return this.data;
        }
        /**
         * Returns the name of the math type this attribute stores.
         *
         * @return - The return value.
         */
        getDataTypeName() {
            return this.dataTypeName;
        }
        /**
         * Returns the count of attribute values in the data.
         *
         * @return - The return value.
         */
        getCount() {
            return this.data.length / this.stride;
        }
        /**
         * Returns the count of attribute values in the data.
         *
         * @return - The return value.
         */
        get count() {
            return this.data.length / this.stride;
        }
        /**
         * Sets the count of attribute values in the data.
         *
         * @param size - The size value.
         */
        setCount(count) {
            const prevLength = this.data.length;
            const newLength = count * this.stride;
            if (newLength > prevLength) {
                this.data = resizeArray(this.data, newLength);
                this.initRange(prevLength);
            }
            else if (newLength < prevLength) {
                this.data = this.data.slice(0, newLength);
            }
            else ;
            this.splits = {};
            this.splitValues = [];
        }
        /**
         * Fills up data values with default ones starting from the specified index.
         *
         * @param start - The start value.
         */
        initRange(start) {
            // Initialize the values to invalid values.
            for (let i = start; i < this.data.length; i++) {
                this.data[i] = this.initValue;
            }
        }
        isInitialized(a) {
            for (let i = 0; i < a.length; i++) {
                if (a[i] != this.initValue)
                    return true;
            }
            return false;
        }
        /**
         * Returns the number of elements stored in each `T`.
         *
         * @return - The return value.
         */
        get numElements() {
            return this.stride;
        }
        /**
         * Returns data value of the specified index.
         *
         * @param index - The index value.
         * @return - The return value.
         */
        getFloat32Value(index) {
            return this.data[index];
        }
        /**
         * Sets data value in the specified index.
         *
         * @param index - The index value.
         * @param value - The value param.
         */
        setFloat32Value(index, value) {
            this.data[index] = value;
        }
        /**
         * Sets data values in the specified index.
         */
        getValues(index) {
            if (index >= this.data.length / this.stride)
                throw new Error('Invalid vertex index:' + index + '. Num Vertices:' + this.data.length / this.stride);
            const offset = index * this.stride;
            return this.data.subarray(offset, offset + this.stride);
        }
        /**
         * Sets data values in the specified index.
         */
        setValues(index, values) {
            if (index >= this.data.length / this.stride)
                throw new Error('Invalid vertex index:' + index + '. Num Vertices:' + this.data.length / this.stride);
            const offset = index * this.stride;
            this.data.set(values, offset);
        }
        merge(other, xfo = new Xfo()) {
            const prevNumValues = this.data.length;
            const addedValues = other.data.length;
            const newLength = prevNumValues + addedValues;
            const data = new Float32Array(newLength);
            data.set(this.data, 0);
            for (let i = 0; i < addedValues; i++) {
                data[prevNumValues + i] = other.data[i];
            }
            this.data = data;
            this.splitValues = [...this.splitValues, ...other.splitValues];
        }
        // //////////////////////////////////////////////////
        // Face Vertex Values
        /**
         * The getSplits method.
         * @return - The return value.
         */
        getSplits() {
            return this.splits;
        }
        /**
         * Gets the value of a corner vertex of a face.
         * @param face - The face index.
         * @param faceVertex - The index of vertex within the face. [0... num face vertices]
         * @return - The return value.
         */
        getFaceVertexValue_array(face, faceVertex) {
            const vertex = this.mesh.getFaceVertexIndex(face, faceVertex);
            if (vertex in this.splits && face in this.splits[vertex]) {
                return this.splitValues[this.splits[vertex][face]];
            }
            return this.data.subarray(vertex * this.stride, (vertex + 1) * this.stride);
        }
        /**
         * Sets the value of a corner vertex of a face.
         * @param face - The face index.
         * @param faceVertex - The index of vertex within the face. [0... num face vertices]
         * @param value - The value value.
         */
        setFaceVertexValue_array(face, faceVertex, value) {
            const vertex = this.mesh.getFaceVertexIndex(face, faceVertex);
            this.setFaceVertexValue_ByVertexIndex(face, vertex, value);
        }
        /**
         * The setFaceVertexValue_ByVertexIndex method.
         * @param face - The face index.
         * @param vertex - The vertex value.
         * @param value - The value value.
         */
        setFaceVertexValue_ByVertexIndex(face, vertex, value) {
            const currValue = this.data.subarray(vertex * this.stride, (vertex + 1) * this.stride);
            if (!this.isInitialized(currValue)) {
                // the value is uninitialized. Initialize it.
                currValue.set(value);
            }
            else if (approxEqual(currValue, value)) ;
            else {
                // The new value is different from the existing value
                if (vertex in this.splits) {
                    // Now check if any existing splits for this vertex match the value being set.
                    // i.e. for faces around a vertex, there will often be a seam along 2 edges
                    // where the values differ. On each side of the seam, all faces can use the same
                    // value. We should see then only one split value for the vertex.
                    const vertexSplitIds = this.splits[vertex];
                    for (const fid in vertexSplitIds) {
                        const splitId = vertexSplitIds[fid];
                        if (approxEqual(this.splitValues[splitId], value)) {
                            // re-use this split value
                            vertexSplitIds[face] = splitId;
                            return;
                        }
                    }
                    // If a split already exists for this face, re-use it.
                    if (face in this.splits[vertex]) {
                        this.splitValues[this.splits[vertex][face]] = value;
                        return;
                    }
                }
                else {
                    this.splits[vertex] = {};
                }
                this.splits[vertex][face] = this.splitValues.length;
                this.splitValues.push(value);
            }
        }
        /**
         * The setSplitVertexValue method.
         * @param vertex - The vertex value.
         * @param face - The face index.
         * @param value - The value value.
         */
        setSplitVertexValue_array(vertex, face, value) {
            if (!(vertex in this.splits))
                this.splits[vertex] = {};
            if (face in this.splits[vertex]) {
                const currValue = this.splitValues[this.splits[vertex][face]];
                if (approxEqual(currValue, value))
                    return;
                console.warn('Face Vertex Already Split with different value');
            }
            this.splits[vertex][face] = this.splitValues.length;
            this.splitValues.push(value);
        }
        /**
         * The setSplitVertexValues method.
         * @param vertex - The vertex value.
         * @param faceGroup - The faceGroup value.
         * @param value - The value value.
         */
        setSplitVertexValues(vertex, faceGroup, values) {
            if (!(vertex in this.splits))
                this.splits[vertex] = {};
            const splitIndex = this.splitValues.length;
            this.splitValues.push(values);
            for (const face of faceGroup) {
                // if (face in this.splits[vertex]) {
                //     let currValue = this.splitValues[this.splits[vertex][face]];
                //     if (currValue.approxEqual(value))
                //         return;
                //     console.warn("Face Vertex Already Split with different value");
                // }
                this.splits[vertex][face] = splitIndex;
            }
        }
        /**
         * The generateSplitValues method.
         * @param splitIndices - The splitIndices value.
         * @param splitCount - The splitCount value.
         * @return - The return value.
         */
        generateSplitValues(splitIndices, splitCount) {
            if (splitCount == 0)
                return this.data;
            const numUnSplitValues = this.getCount();
            const data = resizeArray(this.data, (numUnSplitValues + splitCount) * this.stride);
            // Now duplicate the split values to generate an attributes array
            // using the shared splits across all attributes.
            // eslint-disable-next-line guard-for-in
            for (const vertex in splitIndices) {
                const faces = splitIndices[vertex];
                // eslint-disable-next-line guard-for-in
                for (const face in faces) {
                    const tgt = numUnSplitValues + faces[face];
                    if (vertex in this.splits && face in this.splits[vertex]) {
                        // this attribute has a split value in its array.
                        // we must use that value...
                        const src = this.splits[vertex][face];
                        const srcArray = this.splitValues[src];
                        for (let i = 0; i < srcArray.length; i++) {
                            data[tgt * this.stride + i] = srcArray[i];
                        }
                    }
                    else {
                        // Copy each scalar value to the new place in the array.
                        const src = parseInt(vertex);
                        for (let e = 0; e < this.stride; e++) {
                            data[tgt * this.stride + e] = this.data[src * this.stride + e];
                        }
                    }
                }
            }
            return data;
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * The toJSON method encodes this type as a json object for persistence.
         *
         * @param context - The context value.
         * @return - Returns the json object.
         */
        toJSON(context) {
            return {
                data: Array.from(this.data),
                dataType: this.dataTypeName,
                length: this.data.length / this.stride,
            };
        }
        /**
         * The fromJSON method decodes a json object for this type.
         *
         * @param j - The json object this item must decode.
         */
        fromJSON(j) {
            const data = j.data;
            for (let i = 0; i < data.length; i++) {
                this.data[i] = data[i];
            }
        }
        /**
         * The loadSplitValues method.
         * @param reader - The reader value.
         */
        loadSplitValues(reader) {
            const splitIndices = reader.loadUInt32Array();
            if (splitIndices.length == 0)
                return;
            let offset = 0;
            let numSplitValues = 0;
            while (true) {
                const vertexId = splitIndices[offset++];
                const numSplits = splitIndices[offset++];
                const splits = {};
                for (let i = 0; i < numSplits; i++) {
                    const faceId = splitIndices[offset++];
                    const splitId = splitIndices[offset++];
                    splits[faceId] = splitId;
                    if (splitId >= numSplitValues)
                        numSplitValues = splitId + 1;
                }
                this.splits[vertexId] = splits;
                if (offset >= splitIndices.length)
                    break;
            }
            const dim = this.stride;
            const splitValues = reader.loadFloat32Array(numSplitValues * dim);
            this.splitValues = [];
            for (let i = 0; i < numSplitValues; i++) {
                const val = splitValues.slice(i * dim, i * dim + dim);
                this.splitValues.push(val);
            }
        }
        /**
         * Returns the string representation of the object's state.
         *
         * @return - The return value.
         */
        toString() {
            return JSON.stringify(this.toJSON(), null, 2);
        }
        // ////////////////////////////////////////
        // Memory
        /**
         * Returns vertex attributes buffers and its count.
         *
         * @return - The return value.
         */
        genBuffer() {
            return {
                values: this.data,
                count: this.getCount(),
                dimension: this.stride,
                dataType: this.dataTypeName,
                normalized: this.normalized,
            };
        }
    }

    /**
     * Class representing an attribute.
     */
    class Vec2Attribute extends Attribute {
        /**
         * Create a Vec2Attribute.
         */
        constructor(dataTypeName = 'Vec2') {
            super(dataTypeName, 2);
            this.normalized = false;
        }
        /**
         * Returns the Vec2 from the specified index.
         *
         * @param index - The index value.
         * @return Vec2 - The return value.
         */
        getValue(index) {
            if (index >= this.data.length / this.stride)
                throw new Error('Invalid vertex index:' + index + '. Num Vertices:' + this.data.length / 3);
            const offset = index * this.stride;
            return new Vec2(this.data[offset], this.data[offset + 1]);
        }
        /**
         * Sets Vec2 at the specified index.
         *
         * @param index - The index value.
         * @param value - The value param.
         */
        setValue(index, value) {
            if (index >= this.data.length / this.stride)
                throw new Error('Invalid vertex index:' + index + '. Num Vertices:' + this.data.length / 3);
            const offset = index * this.stride;
            this.data.set(value.asArray(), offset);
        }
        /**
         * Gets the value of a corner vertex of a face.
         * > Note: 'Ref' means that the value contains a reference to the data in the attribute.
         * > The components of the value can be changed causing the attributes data is changed.
         * > No need to call 'setFaceVertexValue'.
         * @param face - The face index.
         * @param faceVertex - The index of vertex within the face. [0... num face vertices]
         * @return - The return value.
         */
        getFaceVertexValue(face, faceVertex) {
            const array = this.getFaceVertexValue_array(face, faceVertex);
            return new Vec2(array[0], array[1]);
        }
        /**
         * Sets the value of a corner vertex of a face.
         * @param face - The face index.
         * @param faceVertex - The index of vertex within the face. [0... num face vertices]
         * @param value - The value value.
         */
        setFaceVertexValue(face, faceVertex, value) {
            this.setFaceVertexValue_array(face, faceVertex, Float32Array.from(value.asArray()));
        }
        /**
         * The setSplitVertexValue method.
         * @param vertex - The vertex value.
         * @param face - The face index.
         * @param value - The value value.
         */
        setSplitVertexValue(vertex, face, value) {
            this.setSplitVertexValue_array(vertex, face, Float32Array.from(value.asArray()));
        }
        merge(other, xfo = new Xfo()) {
            const prevNumValues = this.getCount();
            const addedValues = other.getCount();
            this.setCount(prevNumValues + addedValues);
            for (let i = 0; i < addedValues; i++) {
                this.setValue(prevNumValues + i, other.getValue(i));
            }
            this.splitValues = [...this.splitValues, ...other.splitValues];
        }
    }
    Registry.register('Vec2Attribute', Vec2Attribute);

    const mapIn$2 = (value) => {
        return MathFunctions.encode16BitFloat(value);
    };
    const mapOut$2 = (value) => {
        return MathFunctions.decode16BitFloat(value);
    };
    /**
     * Class representing an attribute.
     */
    class Vec2f16Attribute extends Vec2Attribute {
        /**
         * Create a Vec3f8Attribute.
         */
        constructor() {
            super('Vec2f16');
        }
        init() {
            this.data = new Uint16Array(0);
            this.initRange(0);
        }
        initRange(start) {
            // Initialize the values to invalid values.
            for (let i = start; i < this.data.length; i++) {
                this.data[i] = mapIn$2(Number.NaN);
            }
        }
        isInitialized(a) {
            for (let i = 0; i < a.length; i++) {
                if (Number.isFinite(mapOut$2(a[i])))
                    return true;
            }
            return false;
        }
        /**
         * Returns a copy of the Vec2 value at the specified index.
         *
         * @param index - The index value.
         * @return Vec2 - The value at the specified index.
         */
        getValue(index) {
            if (index >= this.data.length / this.stride)
                throw new Error('Invalid vertex index:' + index + '. Num Vertices:' + this.data.length / 3);
            const offset = index * this.stride;
            const valueData = this.data.subarray(offset, offset + this.stride);
            return new Vec2(mapOut$2(valueData[0]), mapOut$2(valueData[1]));
        }
        /**
         * Sets Vec2 at the specified index.
         *
         * @param index - The index value.
         * @param value - The value param.
         */
        setValue(index, value) {
            if (index >= this.data.length / this.stride)
                throw new Error('Invalid vertex index:' + index + '. Num Vertices:' + this.data.length / 3);
            const offset = index * this.stride;
            const valueData = this.data.subarray(offset, offset + this.stride);
            valueData[0] = mapIn$2(value.x);
            valueData[1] = mapIn$2(value.y);
        }
        /**
         * Gets the value of a corner vertex of a face.
         * > Note: 'Ref' means that the value contains a reference to the data in the attribute.
         * > The components of the value can be changed causing the attributes data is changed.
         * > No need to call 'setFaceVertexValue'.
         * @param face - The face index.
         * @param faceVertex - The index of vertex within the face. [0... num face vertices]
         * @return - The return value.
         */
        getFaceVertexValue(face, faceVertex) {
            const array = this.getFaceVertexValue_array(face, faceVertex);
            return new Vec2(mapOut$2(array[0]), mapOut$2(array[1]));
        }
        /**
         * Sets the value of a corner vertex of a face.
         * @param face - The face index.
         * @param faceVertex - The index of vertex within the face. [0... num face vertices]
         * @param value - The value value.
         */
        setFaceVertexValue(face, faceVertex, value) {
            const valueData = new Uint16Array(2);
            valueData[0] = mapIn$2(value.x);
            valueData[1] = mapIn$2(value.y);
            this.setFaceVertexValue_array(face, faceVertex, valueData);
        }
    }

    class Vec3Ref {
        data;
        constructor(data) {
            this.data = data;
        }
        get x() {
            return this.data[0];
        }
        set x(value) {
            this.data[0] = value;
        }
        get y() {
            return this.data[1];
        }
        set y(value) {
            this.data[1] = value;
        }
        get z() {
            return this.data[2];
        }
        set z(value) {
            this.data[2] = value;
        }
        set(x, y, z) {
            this.data[0] = x;
            this.data[1] = y;
            this.data[2] = z;
        }
    }
    /**
     * Class representing an attribute.
     */
    class Vec3Attribute extends Attribute {
        /**
         * Create a Vec3Attribute.
         */
        constructor(dataTypeName = 'Vec3') {
            super(dataTypeName, 3);
            this.normalized = false;
        }
        /**
         * Returns a copy of the Vec3 value at the specified index.
         *
         * @param index - The index value.
         * @return Vec3 - The value at the specified index.
         */
        getValue(index) {
            if (index >= this.data.length / this.stride)
                throw new Error('Invalid vertex index:' + index + '. Num Vertices:' + this.data.length / 3);
            const offset = index * this.stride;
            return new Vec3(this.data[offset + 0], this.data[offset + 1], this.data[offset + 2]);
        }
        /**
         * Returns a copy of the Vec3 value at the specified index.
         *
         * @deprecated - This method will soon be removed.
         * @return Vec3 - The value at the specified index.
         */
        getValueRef(index) {
            if (index >= this.data.length / this.stride)
                throw new Error('Invalid vertex index:' + index + '. Num Vertices:' + this.data.length / 3);
            const offset = index * this.stride;
            return new Vec3Ref(this.data.subarray(offset, offset + 3));
        }
        /**
         * Sets Vec3 at the specified index.
         *
         * @param index - The index value.
         * @param value - The value param.
         */
        setValue(index, value) {
            this.setValues(index, value.asArray());
        }
        /**
         * Gets the value of a corner vertex of a face.
         * > Note: 'Ref' means that the value contains a reference to the data in the attribute.
         * > The components of the value can be changed causing the attributes data is changed.
         * > No need to call 'setFaceVertexValue'.
         * @param face - The face index.
         * @param faceVertex - The index of vertex within the face. [0... num face vertices]
         * @return - The return value.
         */
        getFaceVertexValue(face, faceVertex) {
            const array = this.getFaceVertexValue_array(face, faceVertex);
            return new Vec3(array[0], array[1], array[2]);
        }
        /**
         * Sets the value of a corner vertex of a face.
         * @param face - The face index.
         * @param faceVertex - The index of vertex within the face. [0... num face vertices]
         * @param value - The value value.
         */
        setFaceVertexValue(face, faceVertex, value) {
            this.setFaceVertexValue_array(face, faceVertex, Float32Array.from(value.asArray()));
        }
        /**
         * The setSplitVertexValue method.
         * @param vertex - The vertex value.
         * @param face - The face index.
         * @param value - The value value.
         */
        setSplitVertexValue(vertex, face, value) {
            this.setSplitVertexValue_array(vertex, face, Float32Array.from(value.asArray()));
        }
        merge(other, xfo = new Xfo()) {
            const prevNumValues = this.getCount();
            const addedValues = other.getCount();
            // We cached these values before calling 'setCount'.
            // Maybe 'setCount' shoulnd't clear the splits. It seems heavy, but I don't want to change anything.
            // We should just re-write the system without splits anyway.
            // const splitValues = [...this.splitValues, ...other.splitValues]
            // const splits = [...this.splits, ...other.splits]
            this.setCount(prevNumValues + addedValues);
            for (let i = 0; i < addedValues; i++) {
                this.setValue(prevNumValues + i, xfo.transformVec3(other.getValue(i)));
            }
            // this.splitValues = [...this.splitValues, ...other.splitValues]
        }
    }
    Registry.register('Vec3Attribute', Vec3Attribute);

    class Vec3f8Ref {
        data;
        constructor(data) {
            this.data = data;
        }
        get x() {
            return MathFunctions.decode16BitFloat(this.data[0]);
        }
        set x(value) {
            this.data[0] = MathFunctions.encode16BitFloat(value);
        }
        get y() {
            return MathFunctions.decode16BitFloat(this.data[1]);
        }
        set y(value) {
            this.data[1] = MathFunctions.encode16BitFloat(value);
        }
        get z() {
            return MathFunctions.decode16BitFloat(this.data[2]);
        }
        set z(value) {
            this.data[2] = MathFunctions.encode16BitFloat(value);
        }
        set(x, y, z) {
            this.data[0] = MathFunctions.encode16BitFloat(x);
            this.data[1] = MathFunctions.encode16BitFloat(y);
            this.data[2] = MathFunctions.encode16BitFloat(z);
        }
    }
    const mapIn$1 = (value, valueRange) => {
        return MathFunctions.remap(value, valueRange[0], valueRange[1], -127, 127);
    };
    const mapOut$1 = (value, valueRange) => {
        return MathFunctions.remap(value, -127, 127, valueRange[0], valueRange[1]);
    };
    /**
     * Class representing an attribute.
     */
    class Vec3f8Attribute extends Vec3Attribute {
        valueRange;
        /**
         * Create a Vec3f8Attribute.
         */
        constructor(valueRange = [-1, 1]) {
            super('Vec3f8');
            this.valueRange = valueRange;
        }
        init() {
            this.data = new Int8Array(0);
            this.initRange(0);
        }
        initRange(start) {
            // Initialize the values to invalid values.
            for (let i = start; i < this.data.length; i++) {
                this.data[i] = mapIn$1(Number.NaN, this.valueRange);
            }
        }
        isInitialized(a) {
            for (let i = 0; i < a.length; i++) {
                if (Number.isFinite(mapOut$1(a[i], this.valueRange)))
                    return true;
            }
            return false;
        }
        /**
         * Returns a copy of the Vec3 value at the specified index.
         *
         * @param index - The index value.
         * @return Vec3 - The value at the specified index.
         */
        getValue(index) {
            if (index >= this.data.length / this.stride)
                throw new Error('Invalid vertex index:' + index + '. Num Vertices:' + this.data.length / 3);
            const offset = index * this.stride;
            const valueData = this.data.subarray(offset, offset + this.stride);
            return new Vec3(mapOut$1(valueData[0], this.valueRange), mapOut$1(valueData[1], this.valueRange), mapOut$1(valueData[2], this.valueRange));
        }
        /**
         * Returns a copy of the Vec3 value at the specified index.
         *
         * @deprecated - This method will soon be removed.
         * @return Vec3 - The value at the specified index.
         */
        getValueRef(index) {
            if (index >= this.data.length / this.stride)
                throw new Error('Invalid vertex index:' + index + '. Num Vertices:' + this.data.length / 3);
            const offset = index * this.stride;
            return new Vec3f8Ref(this.data.subarray(offset, offset + 3));
        }
        /**
         * Sets Vec3 at the specified index.
         *
         * @param index - The index value.
         * @param value - The value param.
         */
        setValue(index, value) {
            if (index >= this.data.length / this.stride)
                throw new Error('Invalid vertex index:' + index + '. Num Vertices:' + this.data.length / 3);
            const offset = index * this.stride;
            const valueData = this.data.subarray(offset, offset + this.stride);
            valueData[0] = mapIn$1(value.x, this.valueRange);
            valueData[1] = mapIn$1(value.y, this.valueRange);
            valueData[2] = mapIn$1(value.z, this.valueRange);
        }
        /**
         * Gets the value of a corner vertex of a face.
         * > Note: 'Ref' means that the value contains a reference to the data in the attribute.
         * > The components of the value can be changed causing the attributes data is changed.
         * > No need to call 'setFaceVertexValue'.
         * @param face - The face index.
         * @param faceVertex - The index of vertex within the face. [0... num face vertices]
         * @return - The return value.
         */
        getFaceVertexValue(face, faceVertex) {
            const array = this.getFaceVertexValue_array(face, faceVertex);
            return new Vec3(mapOut$1(array[0], this.valueRange), mapOut$1(array[1], this.valueRange), mapOut$1(array[2], this.valueRange));
        }
        /**
         * Sets the value of a corner vertex of a face.
         * @param face - The face index.
         * @param faceVertex - The index of vertex within the face. [0... num face vertices]
         * @param value - The value value.
         */
        setFaceVertexValue(face, faceVertex, value) {
            const valueData = new Int8Array(3);
            valueData[0] = mapIn$1(value.x, this.valueRange);
            valueData[1] = mapIn$1(value.y, this.valueRange);
            valueData[2] = mapIn$1(value.z, this.valueRange);
            this.setFaceVertexValue_array(face, faceVertex, valueData);
        }
        /**
         * The setSplitVertexValues method.
         * @param vertex - The vertex value.
         * @param faceGroup - The faceGroup value.
         * @param value - The value value.
         */
        setSplitVertexValues(vertex, faceGroup, values) {
            super.setSplitVertexValues(vertex, faceGroup, values.map((v) => mapIn$1(v, this.valueRange), this.valueRange));
        }
    }

    const mapIn = (value) => {
        return MathFunctions.encode16BitFloat(value);
    };
    const mapOut = (value) => {
        return MathFunctions.decode16BitFloat(value);
    };
    class Vec3f16Ref {
        data;
        constructor(data) {
            this.data = data;
        }
        get x() {
            return mapOut(this.data[0]);
        }
        set x(value) {
            this.data[0] = mapIn(value);
        }
        get y() {
            return mapOut(this.data[1]);
        }
        set y(value) {
            this.data[1] = mapIn(value);
        }
        get z() {
            return mapOut(this.data[2]);
        }
        set z(value) {
            this.data[2] = mapIn(value);
        }
        set(x, y, z) {
            this.data[0] = mapIn(x);
            this.data[1] = mapIn(y);
            this.data[2] = mapIn(z);
        }
    }
    /**
     * Class representing an attribute.
     */
    class Vec3f16Attribute extends Vec3Attribute {
        /**
         * Create a Vec3f8Attribute.
         */
        constructor() {
            super('Vec3f16');
        }
        init() {
            this.data = new Uint16Array(0);
            this.initRange(0);
        }
        initRange(start) {
            // Initialize the values to invalid values.
            for (let i = start; i < this.data.length; i++) {
                this.data[i] = mapIn(Number.NaN);
            }
        }
        isInitialized(a) {
            for (let i = 0; i < a.length; i++) {
                if (Number.isFinite(mapOut(a[i])))
                    return true;
            }
            return false;
        }
        /**
         * Returns a copy of the Vec3 value at the specified index.
         *
         * @param index - The index value.
         * @return Vec3 - The value at the specified index.
         */
        getValue(index) {
            if (index >= this.data.length / this.stride)
                throw new Error('Invalid vertex index:' + index + '. Num Vertices:' + this.data.length / 3);
            const offset = index * this.stride;
            const valueData = this.data.subarray(offset, offset + this.stride);
            return new Vec3(mapOut(valueData[0]), mapOut(valueData[1]), mapOut(valueData[2]));
        }
        /**
         * Returns a copy of the Vec3 value at the specified index.
         *
         * @deprecated - This method will soon be removed.
         * @return Vec3 - The value at the specified index.
         */
        getValueRef(index) {
            if (index >= this.data.length / this.stride)
                throw new Error('Invalid vertex index:' + index + '. Num Vertices:' + this.data.length / 3);
            const offset = index * this.stride;
            return new Vec3f16Ref(this.data.subarray(offset, offset + 3));
        }
        /**
         * Sets Vec3 at the specified index.
         *
         * @param index - The index value.
         * @param value - The value param.
         */
        setValue(index, value) {
            if (index >= this.data.length / this.stride)
                throw new Error('Invalid vertex index:' + index + '. Num Vertices:' + this.data.length / 3);
            const offset = index * this.stride;
            const valueData = this.data.subarray(offset, offset + this.stride);
            valueData[0] = mapIn(value.x);
            valueData[1] = mapIn(value.y);
            valueData[2] = mapIn(value.z);
        }
        /**
         * Gets the value of a corner vertex of a face.
         * > Note: 'Ref' means that the value contains a reference to the data in the attribute.
         * > The components of the value can be changed causing the attributes data is changed.
         * > No need to call 'setFaceVertexValue'.
         * @param face - The face index.
         * @param faceVertex - The index of vertex within the face. [0... num face vertices]
         * @return - The return value.
         */
        getFaceVertexValue(face, faceVertex) {
            const array = this.getFaceVertexValue_array(face, faceVertex);
            return new Vec3(mapOut(array[0]), mapOut(array[1]), mapOut(array[2]));
        }
        /**
         * Sets the value of a corner vertex of a face.
         * @param face - The face index.
         * @param faceVertex - The index of vertex within the face. [0... num face vertices]
         * @param value - The value value.
         */
        setFaceVertexValue(face, faceVertex, value) {
            const valueData = new Uint16Array(3);
            valueData[0] = mapIn(value.x);
            valueData[1] = mapIn(value.y);
            valueData[2] = mapIn(value.z);
            this.setFaceVertexValue_array(face, faceVertex, valueData);
        }
        /**
         * The setSplitVertexValues method.
         * @param vertex - The vertex value.
         * @param faceGroup - The faceGroup value.
         * @param value - The value value.
         */
        setSplitVertexValues(vertex, faceGroup, values) {
            super.setSplitVertexValues(vertex, faceGroup, values.map((v) => mapIn(v)));
        }
    }

    /**
     * Class representing an attribute.
     */
    class ColorAttribute extends Attribute {
        /**
         * Create a ColorAttribute.
         */
        constructor() {
            super('Color', 4);
            this.normalized = false;
        }
        /**
         * Returns a copy of the Color value at the specified index.
         *
         * @param index - The index value.
         * @return Color - The return value.
         */
        getValue(index) {
            if (index >= this.data.length / this.stride)
                throw new Error('Invalid vertex index:' + index + '. Num Vertices:' + this.data.length / 3);
            const offset = index * this.stride;
            return new Color(this.data[offset + 0], this.data[offset + 1], this.data[offset + 2], this.data[offset + 3]);
        }
        /**
         * Sets Color at the specified index.
         *
         * @param index - The index value.
         * @param value - The value param.
         */
        setValue(index, value) {
            this.setValues(index, value.asArray());
        }
        /**
         * Gets the value of a corner vertex of a face.
         * > Note: 'Ref' means that the value contains a reference to the data in the attribute.
         * > The components of the value can be changed causing the attributes data is changed.
         * > No need to call 'setFaceVertexValue'.
         * @param face - The face index.
         * @param faceVertex - The index of vertex within the face. [0... num face vertices]
         * @return - The return value.
         */
        getFaceVertexValue(face, faceVertex) {
            const array = this.getFaceVertexValue_array(face, faceVertex);
            return new Color(array[0], array[1], array[2], array[3]);
        }
        /**
         * Sets the value of a corner vertex of a face.
         * @param face - The face index.
         * @param faceVertex - The index of vertex within the face. [0... num face vertices]
         * @param value - The value value.
         */
        setFaceVertexValue(face, faceVertex, value) {
            this.setFaceVertexValue_array(face, faceVertex, Float32Array.from(value.asArray()));
        }
        /**
         * The setSplitVertexValue method.
         * @param vertex - The vertex value.
         * @param face - The face index.
         * @param value - The value value.
         */
        setSplitVertexValue(vertex, face, value) {
            this.setSplitVertexValue_array(vertex, face, Float32Array.from(value.asArray()));
        }
        merge(other, xfo = new Xfo()) {
            const prevNumValues = this.getCount();
            const addedValues = other.getCount();
            this.setCount(prevNumValues + addedValues);
            for (let i = 0; i < addedValues; i++) {
                this.setValue(prevNumValues + i, other.getValue(i));
            }
            this.splitValues = [...this.splitValues, ...other.splitValues];
        }
    }
    Registry.register('ColorAttribute', ColorAttribute);

    var OperatorOutputMode;
    (function (OperatorOutputMode) {
        OperatorOutputMode[OperatorOutputMode["OP_WRITE"] = 0] = "OP_WRITE";
        OperatorOutputMode[OperatorOutputMode["OP_READ_WRITE"] = 1] = "OP_READ_WRITE";
    })(OperatorOutputMode || (OperatorOutputMode = {}));

    /**
     * Represents a reactive type of attribute that can be owned by a `ParameterOwner` class.
     *
     * **Events**
     * * **nameChanged:** Triggered when the name of the parameter changes.
     * * **valueChanged:** Triggered when the value of the parameter changes.
     */
    class Parameter extends BaseItem {
        #value;
        dirty = false;
        boundInputs = [];
        boundOutputs = [];
        cleaning = false;
        dirtyOpIndex = 0;
        firstOP_WRITE = 0;
        dataType;
        /**
         * When initializing a new parameter, the passed in value could be anything.
         * If it is a new type of value, just ensure you register it in the `Registry`.
         *
         * How to use it:
         *
         * ```javascript
         *  // Creating a parameter object
         *  const param = new Parameter('Title', 'Awesome Parameter Value', 'String')
         *
         *   // Capturing events
         *  param.on('valueChanged', (...params) => console.log('Value changed!'))
         *
         *  // Changing parameter's value will cause `valueChanged` event to trigger.
         *  param.setValue('A New Awesome Parameter Value')
         *  // As result the console log code will execute: Value Changed!
         * ```
         *
         * @param name - The name of the parameter.
         * @param value - The value of the parameter.
         * @param dataType - The data type of the parameter.
         */
        constructor(name = '', value, dataType) {
            super(name);
            this.#value = value;
            this.dataType = dataType;
        }
        /**
         * Returns parameter's data type.
         *
         * @return - The return value.
         */
        getDataType() {
            return this.dataType;
        }
        // ////////////////////////////////////////////////
        // Operator bindings
        /**
         * When an Operator is reading from a parameter, it must be dirtied when the parameter value
         * changes. The Parameter maintains a list of bound inputs and will propagate dirty to
         * them explicitly.
         *
         * @param operatorInput - The output that we are unbinding from the Parameter
         * @param index - The index(optional) that the output is being bound at.
         * @return - The index of the bound output.
         */
        bindOperatorInput(operatorInput) {
            this.boundInputs.push(operatorInput);
        }
        /**
         * When an operator is being removed from reading from a Parameter, the Input is removed
         * This means the operator will no longer receive updates when the operator changes.
         *
         * @param operatorInput - The output that we are unbinding from the Parameter
         * @return - The return value.
         */
        unbindOperatorInput(operatorInput) {
            const index = this.boundInputs.indexOf(operatorInput);
            this.boundInputs.splice(index, 1);
        }
        /**
         * When an Operator writes to a parameter, it binds its outputs to the parameter at a given
         * index. Then when the operator is dirtied by one of its inputs, it explicitly dirties
         * the output parameters.
         *
         * @param operatorOutput - The output that we are unbinding from the Parameter
         * @param index - The index(optional) that the output is being bound at.
         * @return - The index of the bound output.
         */
        bindOperatorOutput(operatorOutput, index = -1) {
            if (index == -1)
                index = this.boundOutputs.length;
            this.boundOutputs.splice(index, 0, operatorOutput);
            // Update the remaining binding indices
            for (let i = index; i < this.boundOutputs.length; i++) {
                this.boundOutputs[i].setParamBindIndex(i);
            }
            // If we weren't already dirty, make sure to emit a 'valueChanged' anyway.
            this.__findFirstOP_WRITE();
            // This ensures that the operator stack is considered 'clean'
            // and then we call set dirty to force it to become dirty from the insertion point down.
            // Without this line, the operator is considered already 'dirty', and so won't propagate.
            this.dirtyOpIndex = this.boundOutputs.length;
            this.setDirty(index);
            return index;
        }
        /**
         * When an operator is unbinding from a parameter, it removes its self from the list maintained
         * by the parameter.
         *
         * @param operatorOutput - The output that we are unbinding from the Parameter
         * @return - The return value.
         */
        unbindOperatorOutput(operatorOutput) {
            const index = operatorOutput.getParamBindIndex();
            this.boundOutputs.splice(index, 1);
            // Update the remaining binding indices
            for (let i = index; i < this.boundOutputs.length; i++) {
                this.boundOutputs[i].setParamBindIndex(i);
            }
            this.__findFirstOP_WRITE();
            this.dirtyOpIndex = this.boundOutputs.length;
            this.setDirty(Math.max(0, index - 1));
            return index;
        }
        /**
         * Find the first operator in our stack which writes using an OP_WRITE connection.
         * All operators before this op can be ignored during dirty propagation.
         * @private
         */
        __findFirstOP_WRITE() {
            this.firstOP_WRITE = this.boundOutputs.length;
            if (this.boundOutputs.length > 0) {
                for (this.firstOP_WRITE--; this.firstOP_WRITE > 0; this.firstOP_WRITE--) {
                    // Find the first OP_WRITE binding. (Note: we could cache this)
                    if (this.boundOutputs[this.firstOP_WRITE].getMode() == OperatorOutputMode.OP_WRITE)
                        break;
                }
            }
        }
        isDrivenByOperator() {
            return this.firstOP_WRITE == 0 && this.boundOutputs.length > 0;
        }
        /**
         * Dirties this Parameter so subsequent calls to `getValue` will cause an evaluation of its bound operators.
         *
         * @param index - Index of the operator
         * @return - `true` if the Parameter was made dirty, else `false` if it was already dirty.
         */
        setDirty(index) {
            // Determine the first operator in the stack that must evaluate to clean the parameter.
            // Note: if a READ_WRITE op is becoming dirty, then we dirty back up to that op.
            if (index < this.dirtyOpIndex) {
                // If we must dirty all operators in the stack from the last OP_WRITE to the end.
                // Note: If a setDirty call comes from an op that precedes an OP_WRITE operator, we
                // can safely discard it, as its output will have no effect on the value of this parameter.
                let newDirtyIndex = this.firstOP_WRITE;
                if (newDirtyIndex <= index) {
                    this.dirtyOpIndex = newDirtyIndex;
                    for (newDirtyIndex++; newDirtyIndex < this.boundOutputs.length; newDirtyIndex++) {
                        // Dirty all the other bound ops from the OP_WRITE to the top of the stack.
                        if (newDirtyIndex != index) {
                            // This will cause the other outputs of the operator to become dirty.
                            this.boundOutputs[newDirtyIndex].getOperator().setDirty();
                        }
                    }
                    for (let i = 0; i < this.boundInputs.length; i++) {
                        this.boundInputs[i].setDirty();
                    }
                    this.emit('valueChanged');
                    return true;
                }
            }
            return false;
        }
        /**
         * Returns true if this parameter is currently dirty and will evaluate its bound
         * operators if its value is requested by a call to getValue.
         *
         * @return - Returns a boolean.
         */
        isDirty() {
            return this.dirtyOpIndex < this.boundOutputs.length;
        }
        /**
         * Returns the index of the first 'dirty' binding in the stack. This will be the index of the
         * first operator that will evaluate when the parameter needs to be cleaned.
         *
         * @return - The index of the dirty binding in the binding stack.
         */
        getDirtyBindingIndex() {
            return this.dirtyOpIndex;
        }
        /**
         * The setCleanFromOp method.
         * @param value - The computed value to be stored in the Parameter.
         * @param index - The index of the bound OperatorOutput.
         */
        setCleanFromOp(value, index) {
            if (index != this.dirtyOpIndex) {
                if (index < this.dirtyOpIndex) {
                    // This can happen when an operator in the following case.
                    // ParamA [OpC, OpB, OpA]
                    // ParamB [OpC, OpA]
                    // When OpB dirties ParamA, and is evaluated, ParamB is considered clean because OpA was never dirtied
                    // We see this message when parameters are evaluated as soon as a change is detected instead of
                    // in batches. Now that all rendering code is pulling data only during the render cycle, we ara
                    // not seeing it anymore. However, maybe with a UI open, it will start emitting this warning.
                    // Note: this would be caused, if a Parameter is already cleaned by an Operator, and yet the Operator
                    // is re-evaluating. I am not sure how this can occur.
                    // const op = operatorOutput.getOperator()
                    // console.log(
                    //   `Operator:: ${
                    //     op.constructor.name
                    //   } with name: ${op.name} is being cleaned immediately, instead of lazily.`
                    // )
                    console.log(`Parameter is cleaned when it was already clean to that point in the stack:`, this.getPath());
                }
                else if (this.boundOutputs[index].getMode() != OperatorOutputMode.OP_WRITE) {
                    // A parameter can become dirty (so __dirtyOpIndex == 0), and then another operator bound on top.
                    // if the next op is a WRITE op, then we can fast forward the dirty index.
                    const thisClassName = this.getClassName();
                    const op = this.boundOutputs[index].getOperator();
                    const opClassName = op.constructor.name;
                    throw new Error(`Parameter: ${thisClassName} with name: ${this.getName()} is not cleaning all outputs during evaluation of op: ${opClassName} with name: ${op.name}`);
                }
            }
            this.#value = value;
            // As each operator writes its value, the dirty value is incremented
            this.dirtyOpIndex = index + 1;
        }
        /**
         * During operator evaluation, operators can use this method to retrieve the existing
         * value of one of their outputs.
         * @param index - The index of the bound OperatorOutput to evaluate up to.
         * @return - The return value.
         */
        getValueFromOp(index) {
            // Note: during evaluation of an Operator that writes to multiple outputs,
            // it can write to an output with an IO setting, which means it retrieves
            // the previous value while calculating the next.
            if (this.dirtyOpIndex < index) {
                this._clean(index);
            }
            return this.#value;
        }
        /**
         * Cleans the parameter up tp the index of the specified index of the bound OperatorOutput
         *
         * @param index - The index of the bound OperatorOutput to evaluate up to.
         */
        _clean(index) {
            if (this.cleaning) {
                throw new Error(`Cycle detected when cleaning: ${this.getPath()}. Operators need to be rebound to fix errors`);
            }
            this.cleaning = true;
            while (this.dirtyOpIndex < index) {
                const tmp = this.dirtyOpIndex;
                const operatorOutput = this.boundOutputs[this.dirtyOpIndex];
                // The op can get the current value and modify it in place
                // and set the output to clean.
                operatorOutput.getOperator().evaluate();
                if (tmp == this.dirtyOpIndex) {
                    // During initial configuration of an operator, cleaning outputs might be disabled.
                    const op = this.boundOutputs[this.dirtyOpIndex].getOperator();
                    const opClassName = op.constructor.name;
                    console.warn(`Operator: ${opClassName} with name: ${op.name} is not cleaning its outputs during evaluation`);
                    this.dirtyOpIndex++;
                }
            }
            this.cleaning = false;
        }
        /**
         * Returns parameter's value.
         * @return - The return value.
         */
        getValue() {
            if (this.dirtyOpIndex < this.boundOutputs.length) {
                this._clean(this.boundOutputs.length);
            }
            return this.#value;
        }
        /**
         * Sets value of the parameter.
         *
         * @param value - The value param.
         */
        setValue(value) {
            if (value === undefined) {
                // eslint-disable-next-line no-throw-literal
                throw 'undefined was passed into the set value for param:' + this.getName();
            }
            if (this.boundOutputs.length > 0) {
                for (let i = this.boundOutputs.length - 1; i >= 0; i--) {
                    const operatorOutput = this.boundOutputs[i];
                    value = operatorOutput.backPropagateValue(value);
                    if (operatorOutput.getMode() == 0 /* OP_WRITE */)
                        return;
                }
            }
            if (typeof value !== 'object') {
                // Note: equality tests on anything but simple values is going to be super expensive.
                if (this.#value == value)
                    return;
            }
            this.#value = value;
            // Note: only users call 'setValue'. Operators call 'setCleanFromOp'
            for (let i = 0; i < this.boundInputs.length; i++) {
                this.boundInputs[i].paramValueChanged();
            }
            this.emit('valueChanged');
        }
        get value() {
            return this.getValue();
        }
        set value(value) {
            this.setValue(value);
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * The loadValue is used to change the value of a parameter, without triggering a
         * valueChanges.
         *
         * @param value - The context value.
         */
        loadValue(value) {
            this.#value = value;
        }
        copyFrom(src, context) {
            this.loadValue(src.value);
        }
        /**
         * The readBinary method.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            console.warn(`TODO: Parameter: ${this.constructor.name} with name: ${this.name} does not implement readBinary`);
        }
        /**
         * Returns the parameter's path as an array of strings.
         * Includes owner's path in case it is owned by a `ParameterOwner`.
         *
         * @return - The return value.
         */
        getPath() {
            if (this.ownerItem instanceof BaseItem) {
                return [...this.ownerItem.getPath(), this.name];
            }
            else {
                return [this.name];
            }
        }
        resolvePath(path, index = 0) {
            if (index == 0) {
                if (path[0] == '.' || path[0] == this.name)
                    index++;
            }
            if (path[index] == '..') {
                if (this.ownerItem) {
                    return this.ownerItem.resolvePath(path, index + 1);
                }
                else {
                    throw Error('this.ownerItem is undefined');
                }
            }
            if (index == path.length) {
                return this;
            }
            if (path[index] == 'value') {
                // The path to the parameter, assumes that the next item will be a sub-param
                // of the value of the parameter.
                if (this.#value instanceof BaseItem)
                    return this.#value.resolvePath(path, index + 1);
            }
            throw new Error('Invalid path:' + path + '[' + index + ']. Path does not resolve to a BaseItem');
        }
        /**
         * The readBinary method.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        destroy() {
            console.warn('nothing destroyed. This method was not overwritten in subclass');
        }
    }

    /**
     * Represents a specific type of parameter, that only stores numeric values.
     *
     * ```javascript
     * const numberParam = new NumberParameter('MyNumber', 15)
     * //'myParameterOwnerItem' is an instance of a 'ParameterOwner' class.
     * // Remember that only 'ParameterOwner' and classes that extend from it can host 'Parameter' objects.
     * myParameterOwnerItem.addParameter(numberParam)
     * ```
     *
     * @extends Parameter
     */
    class NumberParameter extends Parameter {
        range; // TODO: should create type with two fields for range. Must change how range is used.
        step;
        /**
         * Create a number parameter.
         * @param name - The name of the number parameter.
         * @param value - The value of the parameter.
         * @param range - An array with two numbers. If defined, the parameter value will be clamped.
         * @param step - The step value. If defined, the parameter value will be rounded to the nearest integer.
         */
        constructor(name = '', value = 0, range, step) {
            super(name, value, 'Number');
            this.range = range;
            this.step = step;
        }
        /**
         * Returns the range to which the parameter is restrained.
         *
         * @return - The return value.
         */
        getRange() {
            return this.range;
        }
        /**
         * Sets the range to which the parameter is restrained.
         *
         * @param range - The range value.
         */
        setRange(range) {
            this.range = range;
        }
        /**
         * Returns the step number, which is the one used for rounding.
         *
         * @return - The return value.
         */
        getStep() {
            return this.step;
        }
        /**
         * Returns step value.
         *
         * @param step - The step value.
         */
        setStep(step) {
            this.step = step;
        }
        setValue(value) {
            if (typeof value != 'number') {
                throw new Error(`value provided is not a number. Check the source of this value`);
            }
            super.setValue(value);
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * The toJSON method encodes this type as a json object for persistence.
         *
         * @param context - The context value.
         * @return - Returns the json object.
         */
        toJSON(context) {
            return { type: this.getClassName(), name: this.name, value: this.value };
        }
        /**
         * The fromJSON method decodes a json object for this type.
         *
         * @param j - The json object this item must decode.
         * @param context - The context value.
         */
        fromJSON(j, context) {
            this.value = j.value;
        }
        /**
         * Extracts a number value from a buffer, updating current parameter state.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            this.value = reader.loadFloat32();
        }
        // ////////////////////////////////////////
        // Clone
        /**
         * The clone method constructs a new number parameter, copies its values
         * from this parameter and returns it.
         *
         * @return - Returns a new number parameter.
         */
        clone() {
            return new NumberParameter(this.name, this.value, this.range, this.step);
        }
    }
    // eslint-disable-next-line require-jsdoc
    class Float32Parameter extends NumberParameter {
        // eslint-disable-next-line require-jsdoc
        readBinary(reader, context) {
            this.value = reader.loadFloat32();
        }
    }
    // eslint-disable-next-line require-jsdoc
    class SInt32Parameter extends NumberParameter {
        // eslint-disable-next-line require-jsdoc
        readBinary(reader, context) {
            this.value = reader.loadSInt32();
        }
    }
    // eslint-disable-next-line require-jsdoc
    class UInt32Parameter extends NumberParameter {
        // eslint-disable-next-line require-jsdoc
        readBinary(reader, context) {
            this.value = reader.loadUInt32();
        }
    }
    // eslint-disable-next-line require-jsdoc
    class AngleParameter extends Float32Parameter {
    }
    Registry.register('NumberParameter', NumberParameter);
    Registry.register('Property_SInt32', SInt32Parameter);
    Registry.register('Property_UInt32', UInt32Parameter);
    Registry.register('Property_Float32', Float32Parameter);
    Registry.register('AngleParameter', AngleParameter);

    /**
     * Represents a specific type of parameter, that stores multiple choice(array) values.
     *
     * i.e.:
     * ```javascript
     * const multiChoiceParameter =  new MultiChoiceParameter('InitialXfoMode', GROUP_INITIAL_XFO_MODES.average, [
     *                                  'manual',
     *                                  'first',
     *                                  'average',
     *                                  'global',
     *                                ])
     * //'myParameterOwnerItem' is an instance of a 'ParameterOwner' class.
     * // Remember that only 'ParameterOwner' and classes that extend from it can host 'Parameter' objects.
     * myParameterOwnerItem.addParameter(multiChoiceParameter)
     * ```
     * @extends NumberParameter
     */
    class MultiChoiceParameter extends NumberParameter {
        choices;
        /**
         * Create a multi choice parameter.
         * @param name - The name of the multi choice parameter.
         * @param index - The index value.
         * @param choices - The choices value.
         */
        constructor(name, index, choices = []) {
            super(name, index, [0, choices.length], 1);
            this.choices = choices;
        }
        /**
         * Returns choices array.
         *
         * @return - The return value.
         */
        getChoices() {
            return this.choices;
        }
        /**
         * Sets parameter index value.
         *
         * @param value - The value param.
         */
        setValue(value) {
            if (typeof value === 'string') {
                const index = this.choices.indexOf(value);
                if (index === -1) {
                    console.error(`Invalid value for MultiChoiceParameter: ${value}. choices are: ${this.choices}.`);
                }
                else {
                    super.setValue(index);
                }
            }
            else {
                super.setValue(value);
            }
        }
        clone() {
            return new MultiChoiceParameter(this.name, this.value, this.choices);
        }
    }
    Registry.register('MultiChoiceParameter', MultiChoiceParameter);

    /* eslint-disable @typescript-eslint/no-unused-vars */
    /**
     * Represents a specific type of parameter, that only stores `boolean` values.
     *
     * i.e.:
     * ```javascript
     * const booleanParam = new BooleanParameter('MyBoolean', true)
     * //'myParameterOwnerItem' is an instance of a 'ParameterOwner' class.
     * // Remember that only 'ParameterOwner' and classes that extend from it can host 'Parameter' objects.
     * myParameterOwnerItem.addParameter(booleanParam)
     * ```
     * @extends Parameter
     */
    class BooleanParameter extends Parameter {
        /**
         * Creates a new parameter with `Boolean` data type.
         *
         * @param name - The name of the boolean parameter.
         * @param value - The value of the parameter.
         */
        constructor(name = '', value) {
            super(name, value != undefined ? value : false, 'Boolean');
        }
        setValue(value) {
            if (typeof value != 'boolean') {
                throw new Error(`value provided is not a boolean. Check the source of this value`);
            }
            super.setValue(value);
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * Loads the boolean values from the binary buffer.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            this.value = reader.loadUInt8() != 0;
        }
        /**
         * The toJSON method serializes this instance as a JSON.
         * It can be used for persistence, data transfer, etc.
         *
         * @param context - The context value.
         * @return - Returns the json object.
         */
        toJSON(context) {
            return { type: this.getClassName(), name: this.name, value: this.value };
        }
        /**
         * The fromJSON method takes a JSON and deserializes into an instance of this type.
         *
         * @param j - The json object this item must decode.
         * @param context - The context value.
         */
        fromJSON(j, context) {
            this.value = j.value;
        }
        /**
         * The clone method constructs a new parameter, copies its values
         * from this parameter and returns it.
         *
         * @return - Returns a new cloned parameter.
         */
        clone() {
            return new BooleanParameter(this.name, this.value);
        }
    }
    Registry.register('BooleanParameter', BooleanParameter);
    Registry.register('Property_Boolean', BooleanParameter);

    /* eslint-disable @typescript-eslint/no-explicit-any */
    /**
     * Represents a specific type of parameter, that only stores Vec2(two-dimensional coordinate) values.
     *
     * i.e.:
     * ```javascript
     * const vec2Param = new Vec2Parameter('MyVec2', new Vec2(1.2, 3.4))
     * //'myParameterOwnerItem' is an instance of a 'ParameterOwner' class.
     * // Remember that only 'ParameterOwner' and classes that extend from it can host 'Parameter' objects.
     * myParameterOwnerItem.addParameter(vec2Param)
     * ```
     *
     * **Events**
     * * **rangeChanged:** Triggered when rage array changes.
     *
     * @extends Parameter
     */
    class Vec2Parameter extends Parameter {
        range;
        /**
         * Create a Vec2 parameter.
         *
         * @param name - The name of the Vec2 parameter.
         * @param value - The value of the parameter.
         * @param range - The range value is an array of two `Vec2` objects.
         */
        constructor(name = '', value, range) {
            super(name, value ? value : new Vec2(), 'Vec2');
            this.range = range;
        }
        /**
         * Returns the range of values in which current parameter can be.
         *
         * @return - The return value.
         */
        getRange() {
            // Range should be an array of 2 vec2s. [min(x,y), max(x,y)]
            return this.range;
        }
        /**
         * The __setRange method.
         * @param range - The range value.
         * @private
         */
        setRange(range) {
            // Should be an array [0, 20]
            this.range = range;
            this.emit('rangeChanged', { range });
        }
        setValue(value) {
            if (!(value instanceof Vec2)) {
                throw new Error(`value provided is not an instance of a 'Vec2' class. Check the source of this value`);
            }
            super.setValue(value);
        }
        /**
         * Extracts a number value from a buffer, updating current parameter state.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            this.value?.readBinary(reader);
        }
        toJSON(context) {
            return {
                type: this.getClassName(),
                name: this.name,
                value: this.value?.toJSON(),
            };
        }
        fromJSON(j, context) {
            const vec2 = new Vec2();
            vec2.fromJSON(j.value);
            this.value = vec2;
            if (j.name)
                this.name = j.name;
        }
        /**
         * The clone method constructs a new Vec2 parameter, copies its values
         * from this parameter and returns it.
         *
         * @return - Returns a new Vec2 parameter.
         */
        clone() {
            const clonedParam = new Vec2Parameter(this.name, this.value?.clone());
            if (this.range)
                clonedParam.setRange(this.range);
            return clonedParam;
        }
    }
    Registry.register('Vec2Parameter', Vec2Parameter);
    Registry.register('Property_Vec2_32f', Vec2Parameter);

    /* eslint-disable @typescript-eslint/no-explicit-any */
    /**
     * Represents a specific type of parameter, that only stores Vec3(three-dimensional coordinate) values.
     *
     * i.e.:
     * ```javascript
     * const vec3Param = new Vec3Parameter('MyVec3', new Vec3(1.2, 3.4, 1))
     * //'myParameterOwnerItem' is an instance of a 'ParameterOwner' class.
     * // Remember that only 'ParameterOwner' and classes that extend from it can host 'Parameter' objects.
     * myParameterOwnerItem.addParameter(vec3Param)
     * ```
     * @extends Parameter
     */
    class Vec3Parameter extends Parameter {
        range;
        /**
         * Create a Vec3 parameter.
         *
         * @param name - The name of the Vec3 parameter.
         * @param value - The value of the parameter.
         * @param range - The range value is an array of two `Vec3` objects.
         */
        constructor(name = '', value, range) {
            super(name, value ? value : new Vec3(), 'Vec3');
            this.range = range;
        }
        setValue(value) {
            if (!(value instanceof Vec3)) {
                throw new Error(`value provided is not an instance of a 'Vec2' class. Check the source of this value`);
            }
            super.setValue(value);
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * Extracts a number value from a buffer, updating current parameter state.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            this.value?.readBinary(reader);
        }
        toJSON(context) {
            return {
                type: this.getClassName(),
                name: this.name,
                value: this.value?.toJSON(),
            };
        }
        fromJSON(j, context) {
            const vec4 = new Vec3();
            vec4.fromJSON(j.value);
            this.value = vec4;
            if (j.name)
                this.name = j.name;
        }
        /**
         * The clone method constructs a new Vec3 parameter, copies its values
         * from this parameter and returns it.
         *
         * @return - Returns a new Vec3 parameter.
         */
        clone() {
            const clonedParam = new Vec3Parameter(this.name, this.value?.clone());
            return clonedParam;
        }
    }
    Registry.register('Vec3Parameter', Vec3Parameter);
    Registry.register('Property_Vec3_32f', Vec3Parameter);

    /* eslint-disable @typescript-eslint/no-explicit-any */
    /**
     * Represents a specific type of parameter, that only stores Vec4(four-dimensional coordinate) values.
     *
     * i.e.:
     * ```javascript
     * const vec4Param = new Vec4Parameter('MyVec4', new Vec4(1.2, 3.4, 1, 4.2))
     * //'myParameterOwnerItem' is an instance of a 'ParameterOwner' class.
     * // Remember that only 'ParameterOwner' and classes that extend from it can host 'Parameter' objects.
     * myParameterOwnerItem.addParameter(vec4Param)
     * ```
     *
     * @extends Parameter
     */
    class Vec4Parameter extends Parameter {
        /**
         * Create a Vec4 parameter.
         * @param name - The name of the Vec4 parameter.
         * @param value - The value of the parameter.
         */
        constructor(name = '', value) {
            super(name, value ? value : new Vec4(), 'Vec4');
        }
        setValue(value) {
            if (!(value instanceof Vec4)) {
                throw new Error(`value provided is not an instance of a 'Vec4' class. Check the source of this value`);
            }
            super.setValue(value);
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * Extracts a number value from a buffer, updating current parameter state.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            this.value?.readBinary(reader);
        }
        toJSON(context) {
            return { type: this.getClassName(), name: this.name, value: this.value?.toJSON() };
        }
        fromJSON(j, context) {
            const vec4 = new Vec4();
            vec4.fromJSON(j.value);
            this.value = vec4;
            if (j.name)
                this.name = j.name;
        }
        // ////////////////////////////////////////
        // Clone
        /**
         * The clone method constructs a new Vec4 parameter, copies its values
         * from this parameter and returns it.
         *
         * @return - Returns a new Vec4 parameter.
         */
        clone() {
            const clonedParam = new Vec4Parameter(this.name, this.value?.clone());
            return clonedParam;
        }
    }
    Registry.register('Vec4Parameter', Vec4Parameter);
    Registry.register('Property_Vec4_32f', Vec4Parameter);

    /* eslint-disable @typescript-eslint/no-unused-vars */
    /**
     * @extends Parameter
     */
    class Box2Parameter extends Parameter {
        /**
         * Create a Box2 parameter.
         * @param name - The name of the Box2 parameter.
         * @param value - The value of the parameter.
         */
        constructor(name = '', value) {
            super(name, value ? value : new Box2(), 'Box2');
        }
        setValue(value) {
            if (!(value instanceof Box2)) {
                throw new Error(`value provided is not an instance of a 'Box2' class. Check the source of this value`);
            }
            super.setValue(value);
        }
        /**
         * Extracts a number value from a buffer, updating current parameter state.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            this.value.p0.readBinary(reader);
            this.value.p1.readBinary(reader);
        }
        toJSON(context) {
            return {
                type: this.getClassName(),
                name: this.name,
                value: {
                    p0: this.value.p0.toJSON(),
                    p1: this.value.p1.toJSON(),
                },
            };
        }
        fromJSON(j, context) {
            this.value.p0.fromJSON(j.p0);
            this.value.p1.fromJSON(j.p1);
        }
        /**
         * The clone method constructs a new Box2 parameter,
         * copies its values from this parameter and returns it.
         *
         * @return - Returns a new cloned Box2 parameter.
         */
        clone() {
            const clonedParam = new Box2Parameter(this.name, this.value?.clone());
            return clonedParam;
        }
    }
    Registry.register('Box2Parameter', Box2Parameter);
    Registry.register('Property_Box2_32f', Box2Parameter);

    /* eslint-disable @typescript-eslint/no-unused-vars */
    /**
     * @extends Parameter
     */
    class Box3Parameter extends Parameter {
        /**
         * Create a Box3 parameter.
         * @param name - The name of the Box3 parameter.
         * @param value - The value of the parameter.
         */
        constructor(name = '', value) {
            super(name, value ? value : new Box3(), 'Box3');
        }
        setValue(value) {
            if (!(value instanceof Box3)) {
                throw new Error(`value provided is not an instance of a 'Box3' class. Check the source of this value`);
            }
            super.setValue(value);
        }
        /**
         * Extracts a number value from a buffer, updating current parameter state.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            this.value.p0.readBinary(reader);
            this.value.p1.readBinary(reader);
        }
        toJSON(context) {
            return {
                type: this.getClassName(),
                name: this.name,
                value: this.value.toJSON(),
            };
        }
        fromJSON(j, context) {
            if (j.value) {
                // @ts-ignore
                this.value.fromJSON(j.value);
            }
        }
        /**
         * The clone method constructs a new Box3 parameter,
         * copies its values from this parameter and returns it.
         *
         * @return - Returns a new cloned Box3 parameter.
         */
        clone() {
            const clonedParam = new Box3Parameter(this.name, this.value?.clone());
            return clonedParam;
        }
    }
    Registry.register('Box3Parameter', Box3Parameter);
    Registry.register('Property_Box3_32f', Box3Parameter);

    /* eslint-disable @typescript-eslint/no-explicit-any */
    /**
     * Represents a specific type of parameter, that only stores `Color` values.
     *
     * i.e.:
     * ```javascript
     * const colorParam = new ColorParameter('MyColor', new Color(0, 254, 2))
     * //'myParameterOwnerItem' is an instance of a 'ParameterOwner' class.
     * // Remember that only 'ParameterOwner' and classes that extend from it can host 'Parameter' objects.
     * myParameterOwnerItem.addParameter(colorParam)
     * ```
     *
     * @extends Parameter
     */
    class ColorParameter extends Parameter {
        /**
         * Create a color parameter.
         * @param name - The name of the color parameter.
         * @param value - The value of the parameter.
         */
        constructor(name = '', value) {
            super(name, value ? value : new Color(), 'Color');
        }
        setValue(value) {
            if (!(value instanceof Color)) {
                throw new Error(`value provided is not an instance of a 'Color' class. Check the source of this value`);
            }
            super.setValue(value);
        }
        /**
         * Extracts `Color` values from a buffer, updating current parameter state.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            const value = reader.loadRGBAFloat32Color();
            // If the value is in linear space, then we should convert it to gamma space.
            // Note: !! this should always be done in preprocessing...
            value.applyGamma(2.2);
            this.value = value;
        }
        toJSON(context) {
            return {
                type: this.getClassName(),
                name: this.name,
                value: this.value?.toJSON(),
            };
        }
        fromJSON(j, context) {
            // if (j.value.type) this.value = Registry.constructClass('Color') as Color // TODO: commented out Registry.constructClass
            this.value?.fromJSON(j.value);
        }
        /**
         * The clone method constructs a new color parameter,
         * copies its values from this parameter and returns it.
         *
         * @return - Returns a new cloned color parameter.
         */
        clone() {
            const clonedParam = new ColorParameter(this.name, this.value?.clone());
            return clonedParam;
        }
    }
    Registry.register('ColorParameter', ColorParameter);
    Registry.register('Property_Color_32f', ColorParameter);

    /* eslint-disable @typescript-eslint/no-explicit-any */
    /**
     * Represents a specific type of parameter, that only stores Vec3(four-dimensional coordinate) values.
     *
     * i.e.:
     * ```javascript
     * const quatParam = new QuatParameter('MyQuat', new Quat(1.2, 3.4, 1, 4.2))
     * //'myParameterOwnerItem' is an instance of a 'ParameterOwner' class.
     * // Remember that only 'ParameterOwner' and classes that extend from it can host 'Parameter' objects.
     * myParameterOwnerItem.addParameter(quatParam)
     * ```
     *
     * @extends Parameter
     */
    class QuatParameter extends Parameter {
        /**
         * Create a Quat parameter.
         * @param name - The name of the Quat parameter.
         * @param value - The value of the parameter.
         */
        constructor(name = '', value) {
            super(name, value ? value : new Quat(), 'Quat');
        }
        setValue(value) {
            if (!(value instanceof Quat)) {
                throw new Error(`value provided is not an instance of a 'Quat' class. Check the source of this value`);
            }
            super.setValue(value);
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * Extracts a number value from a buffer, updating current parameter state.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            this.value?.readBinary(reader);
        }
        toJSON(context) {
            return {
                type: this.getClassName(),
                name: this.name,
                value: this.value?.toJSON(),
            };
        }
        fromJSON(j, context) {
            const quat = new Quat();
            quat.fromJSON(j.value);
            this.value = quat;
        }
        // ////////////////////////////////////////
        // Clone
        /**
         * The clone method constructs a new Quat parameter, copies its values
         * from this parameter and returns it.
         *
         * @return - Returns a new Quat parameter.
         */
        clone() {
            const clonedParam = new QuatParameter(this.name, this.value?.clone());
            return clonedParam;
        }
    }
    Registry.register('QuatParameter', QuatParameter);
    Registry.register('Property_Quat_32f', QuatParameter);

    /* eslint-disable @typescript-eslint/no-unused-vars */
    /**
     * Represents a specific type of parameter, that only stores Mat3(3x3 matrix) values.
     *
     * i.e.:
     * ```javascript
     * const mat3Param = new Ma3Parameter('MyMat3', new Mat3(...args))
     * //'myParameterOwnerItem' is an instance of a 'ParameterOwner' class.
     * // Remember that only 'ParameterOwner' and classes that extend from it can host 'Parameter' objects.
     * myParameterOwnerItem.addParameter(mat3Param)
     * ```
     *
     * @extends Parameter
     */
    class Mat3Parameter extends Parameter {
        /**
         * Create a Mat3 parameter.
         * @param name - The name of the Mat3 parameter.
         * @param value - The value of the parameter.
         */
        constructor(name = '', value) {
            super(name, value ? value : new Mat3(), 'Mat3');
        }
        setValue(value) {
            if (!(value instanceof Mat3)) {
                throw new Error(`value provided is not an instance of a 'Mat3' class. Check the source of this value`);
            }
            super.setValue(value);
        }
        /**
         * Extracts a number value from a buffer, updating current parameter state.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            this.value?.readBinary(reader);
        }
        toJSON(context) {
            return {
                type: this.getClassName(),
                name: this.name,
                value: this.value?.toJSON(),
            };
        }
        fromJSON(j, context) {
            const mat3 = new Mat3();
            mat3.fromJSON(j.value);
            this.value = mat3;
        }
        /**
         * The clone method constructs a new Mat3 parameter,
         * copies its values from this parameter and returns it.
         *
         * @return - Returns a new cloned Mat3 parameter.
         */
        clone() {
            const clonedParam = new Mat3Parameter(this.name, this.value?.clone());
            return clonedParam;
        }
    }
    Registry.register('Mat3Parameter', Mat3Parameter);
    Registry.register('Property_Mat3_32f', Mat3Parameter);

    /* eslint-disable @typescript-eslint/no-explicit-any */
    /**
     * Represents a specific type of parameter, that only stores Mat4(4x4 matrix) values.
     *
     * i.e.:
     * ```javascript
     * const mat4Param = new Ma3Parameter('MyMat4', new Mat4(...args))
     * //'myParameterOwnerItem' is an instance of a 'ParameterOwner' class.
     * // Remember that only 'ParameterOwner' and classes that extend from it can host 'Parameter' objects.
     * myParameterOwnerItem.addParameter(mat4Param)
     * ```
     *
     * @extends Parameter
     */
    class Mat4Parameter extends Parameter {
        /**
         * Create a Mat4 parameter.
         *
         * @param name - The name of the Mat4 parameter.
         * @param value - The value of the parameter.
         */
        constructor(name = '', value) {
            super(name, value ? value : new Mat4(), 'Mat4');
        }
        setValue(value) {
            if (!(value instanceof Mat4)) {
                throw new Error(`value provided is not an instance of a 'Mat4' class. Check the source of this value`);
            }
            super.setValue(value);
        }
        /**
         * Extracts a number value from a buffer, updating current parameter state.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            this.value?.readBinary(reader);
        }
        toJSON(context) {
            return {
                type: this.getClassName(),
                value: this.value?.toJSON(),
            };
        }
        fromJSON(j, context) {
            const mat4 = new Mat4();
            mat4.fromJSON(j.value);
            this.value = mat4;
        }
        /**
         * The clone method constructs a new Mat4 parameter,
         * copies its values from this parameter and returns it.
         *
         * @return - Returns a new cloned Mat4 parameter.
         */
        clone() {
            const clonedParam = new Mat4Parameter(this.name, this.value?.clone());
            return clonedParam;
        }
    }
    Registry.register('Mat4Parameter', Mat4Parameter);
    Registry.register('Property_Mat4_32f', Mat4Parameter);

    /* eslint-disable @typescript-eslint/no-explicit-any */
    /**
     * Represents a specific type of parameter, that only stores `Xfo` transform values.
     *
     * ```javascript
     * const xfoParam = new XfoParameter('MyXfo', new Xfo(new Vec3(1.2, 3.4, 1)))
     * //'myParameterOwnerItem' is an instance of a 'ParameterOwner' class.
     * // Remember that only 'ParameterOwner' and classes that extend from it can host 'Parameter' objects.
     * myParameterOwnerItem.addParameter(xfoParam)
     * ```
     *
     * @extends Parameter
     */
    class XfoParameter extends Parameter {
        /**
         * Create a Xfo parameter.
         * @param name - The name of the Xfo parameter.
         * @param value - The value of the parameter.
         */
        constructor(name = '', value) {
            super(name, value ? value : new Xfo(), 'Xfo');
        }
        setValue(value) {
            if (!(value instanceof Xfo)) {
                throw new Error(`value provided is not an instance of a 'Xfo' class. Check the source of this value`);
            }
            super.setValue(value);
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * Extracts a number value from a buffer, updating current parameter state.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            this.value.readBinary(reader);
        }
        toJSON(context) {
            return { type: this.getClassName(), name: this.name, value: this.value.toJSON() };
        }
        fromJSON(j, context) {
            const xfo = new Xfo();
            xfo.fromJSON(j.value);
            this.value = xfo;
            if (j.name)
                this.name = j.name;
        }
        // ////////////////////////////////////////
        // Clone
        /**
         * The clone method constructs a new Xfo parameter, copies its values
         * from this parameter and returns it.
         *
         * @return - Returns a new Xfo parameter.
         */
        clone() {
            const clonedParam = new XfoParameter(this.name, this.value.clone());
            return clonedParam;
        }
    }
    Registry.register('XfoParameter', XfoParameter);
    Registry.register('Property_Xfo_32f', XfoParameter);

    /**
     * Represents a 2D image item, containing width and height.
     *
     * **Events**
     * * **updated:** Triggered when the value of any of the parameters listed above changes.
     *
     * @extends ParameterOwner
     */
    class BaseImage extends ParameterOwner {
        width = 0;
        height = 0;
        format = 'RGB';
        type = 'UNSIGNED_BYTE';
        loaded = false;
        mipMapped = true;
        wrapS = 'REPEAT';
        wrapT = 'REPEAT';
        minFilter = 'LINEAR';
        magFilter = 'LINEAR';
        /**
         * Creates an instance of BaseImage.
         * @param name - name of the item
         */
        constructor(name = 'Image') {
            super(name);
            this.on('parameterValueChanged', () => {
                this.emit('updated');
            });
        }
        /**
         * Returns true if loaded.
         * @private
         * @return - Returns a boolean.
         */
        isLoaded() {
            return this.loaded;
        }
        /**
         * Returns all parameters and class state values.
         *
         * @return - The return value.
         */
        getParams() {
            return {
                type: this.type,
                format: this.format,
                width: this.width,
                height: this.height,
                wrapS: this.wrapS,
                wrapT: this.wrapT,
                minFilter: this.minFilter,
                magFilter: this.magFilter,
                mipMapped: this.mipMapped,
            };
        }
    }

    /* eslint-disable @typescript-eslint/no-unused-vars */
    /**
     * Represents a specific type of parameter, that only stores `BaseImage` values.
     *
     * i.e.:
     * ```javascript
     * // Since `Label` is a `BaseImage` implementation, it helps us with the example.
     * const label = new Label('My awesome label', 'LabelPack')
     * const imageParam = new ImageParameter('MyImage', label)
     * //'myParameterOwnerItem' is an instance of a 'ParameterOwner' class.
     * // Remember that only 'ParameterOwner' and classes that extend from it can host 'Parameter' objects.
     * myParameterOwnerItem.addParameter(imageParam)
     * ```
     *
     * @extends Parameter
     */
    class ImageParameter extends Parameter {
        /**
         * Create an image parameter.
         *
         * @param name - The name of the image parameter.
         * @param value - The value of the parameter.
         */
        constructor(name = '', value) {
            super(name, value, 'BaseImage');
        }
        setValue(value) {
            if (!(value instanceof BaseImage)) {
                throw new Error(`value provided is not an instance of 'BaseImage' class. Check the source of this value`);
            }
            super.setValue(value);
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * The toJSON method encodes this type as a json object for persistence.
         *
         * @param context - The context value.
         * @return - Returns the json object.
         */
        toJSON(context) {
            const j = {
                type: this.getClassName(),
                name: this.name,
            };
            if (this.value) {
                j.imageType = this.value.getClassName();
                j.value = this.value.toJSON();
            }
            return j;
        }
        /**
         * The fromJSON method decodes a json object for this type.
         *
         * @param j - The json object this item must decode.
         * @param context - The context value.
         * @return - Returns the json object.
         */
        fromJSON(j, context) {
            if (j.imageType) {
                this.value = Registry.constructClass(j.imageType);
                if (j.value)
                    this.value?.fromJSON(j.value, context);
            }
        }
        // ////////////////////////////////////////
        // Clone
        /**
         * The clone method constructs a new image parameter,
         * copies its values from this parameter and returns it.
         *
         * @return - Returns a new cloned image parameter.
         */
        clone() {
            const clonedParam = new ImageParameter(this.name, this.value);
            return clonedParam;
        }
    }
    Registry.register('ImageParameter', ImageParameter);

    /* eslint-disable @typescript-eslint/no-unused-vars */
    /**
     * Represents a specific type of parameter, that only stores Mat4(4x4 matrix) values.
     *
     * i.e.:
     * ```javascript
     * const stringParam = new StringParameter('MyString', 'A String value goes here')
     * //'myParameterOwnerItem' is an instance of a 'ParameterOwner' class.
     * // Remember that only 'ParameterOwner' and classes that extend from it can host 'Parameter' objects.
     * myParameterOwnerItem.addParameter(stringParam)
     * ```
     *
     * @extends Parameter
     */
    class StringParameter extends Parameter {
        multiLine;
        /**
         * Create a string parameter.
         * @param name - The name of the material color parameter.
         * @param value - The value of the parameter.
         */
        constructor(name = '', value = '') {
            super(name, value, 'String');
            this.multiLine = false;
        }
        /**
         * Sets flag that indicates if the string contains new line feeds.
         *
         * @param multiLine - The multiLine value.
         */
        setMultiLine(multiLine) {
            this.multiLine = multiLine;
        }
        /**
         * Returns multi-line flag value.
         *
         * @return - The return value.
         */
        getMultiLine() {
            return this.multiLine;
        }
        setValue(value) {
            if (typeof value != 'string') {
                throw new Error(`value provided is not a string. Check the source of this value`);
            }
            super.setValue(value);
        }
        /**
         * Extracts the string value from a buffer, updating current parameter state.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            this.value = reader.loadStr();
        }
        /**
         * The toJSON method serializes this instance as a JSON.
         * It can be used for persistence, data transfer, etc.
         *
         * @param context - The context value.
         * @return - Returns the json object.
         */
        toJSON(context) {
            return { type: this.getClassName(), name: this.name, value: this.value };
        }
        /**
         * The fromJSON method takes a JSON and deserializes into an instance of this type.
         *
         * @param j - The json object this item must decode.
         * @param context - The context value.
         */
        fromJSON(j, context) {
            const newValue = j.value ?? '';
            if (this.value != newValue) {
                this.value = newValue;
            }
        }
        /**
         * The clone method constructs a new string parameter, copies its values
         * from this parameter and returns it.
         *
         * @return - Returns a new string parameter.
         */
        clone() {
            return new StringParameter(this.name, this.value);
        }
    }
    Registry.register('StringParameter', StringParameter);
    Registry.register('Property_String', StringParameter);

    /**
     * A parameter for storing an array of string values.
     *
     * @extends Parameter
     */
    class StringListParameter extends Parameter {
        /**
         * Create a string parameter.
         * @param {string} name - The name of the material color parameter.
         * @param {string} value - The value of the parameter.
         */
        constructor(name = '', value = []) {
            super(name, value, 'String[]');
        }
        setValue(value) {
            if (!Array.isArray(value)) {
                throw new Error(`value provided is not an array. Check the source of this value`);
            }
            super.setValue(value);
        }
        /**
         * Extracts the string value from a buffer, updating current parameter state.
         *
         * @param {BinReader} reader - The reader value.
         * @param {object} context - The context value.
         */
        readBinary(reader, context) {
            this.value = reader.loadStrArray();
        }
        /**
         * The toJSON method serializes this instance as a JSON.
         * It can be used for persistence, data transfer, etc.
         *
         * @param {Record<string, unknown>} context - The context value.
         * @return {Record<string, boolean | undefined>} - Returns the json object.
         */
        toJSON(context) {
            return { type: this.getClassName(), name: this.name, value: this.value };
        }
        /**
         * The fromJSON method takes a JSON and deserializes into an instance of this type.
         *
         * @param {Record<string, boolean | undefined>} j - The json object this item must decode.
         * @param {Record<string, unknown>} context - The context value.
         */
        fromJSON(j, context) {
            this.value = j.value;
        }
        /**
         * The clone method constructs a new string parameter, copies its values
         * from this parameter and returns it.
         *
         * @return {StringListParameter} - Returns a new string parameter.
         */
        clone() {
            const clonedParam = new StringListParameter(this.name, this.value);
            return clonedParam;
        }
    }
    Registry.register('StringListParameter', StringListParameter);
    Registry.register('Property_StringList', StringListParameter);

    /**
     * A parameter for storing an array of string values.
     *
     * @extends Parameter
     */
    class Float32ArrayParameter extends Parameter {
        /**
         * Create a string parameter.
         * @param {string} name - The name of the material color parameter.
         * @param {string} value - The value of the parameter.
         */
        constructor(name = '', value = new Float32Array()) {
            super(name, value, 'Float[]');
        }
        /**
         * Extracts the string value from a buffer, updating current parameter state.
         *
         * @param {BinReader} reader - The reader value.
         * @param {object} context - The context value.
         */
        readBinary(reader, context) {
            this.value = reader.loadFloat32Array();
        }
        /**
         * The toJSON method serializes this instance as a JSON.
         * It can be used for persistence, data transfer, etc.
         *
         * @param {Record<string, unknown>} context - The context value.
         * @return {Record<string, boolean | undefined>} - Returns the json object.
         */
        toJSON(context) {
            return { type: this.getClassName(), name: this.name, value: this.value };
        }
        /**
         * The fromJSON method takes a JSON and deserializes into an instance of this type.
         *
         * @param {Record<string, boolean | undefined>} j - The json object this item must decode.
         * @param {Record<string, unknown>} context - The context value.
         */
        fromJSON(j, context) {
            this.value = j.value;
        }
        /**
         * The clone method constructs a new string parameter, copies its values
         * from this parameter and returns it.
         *
         * @return {Float32ArrayParameter} - Returns a new string parameter.
         */
        clone() {
            const clonedParam = new Float32ArrayParameter(this.name, this.value);
            return clonedParam;
        }
    }
    Registry.register('Float32ArrayParameter', Float32ArrayParameter);
    Registry.register('Property_Float32Array', Float32ArrayParameter);

    /* eslint-disable @typescript-eslint/no-unused-vars */
    /**
     * A Parameter for storing list(array) values.
     *
     * i.e.:
     * ```javascript
     * const listParam = new ListParameter('MyList', GearParameter)
     * //'myParameterOwnerItem' is an instance of a 'ParameterOwner' class.
     * // Remember that only 'ParameterOwner' and classes that extend from it can host 'Parameter' objects.
     * myParameterOwnerItem.addParameter(listParam)
     * ```
     *
     * **Events**
     * * **valueChanged:** Triggered when setting a value changes in the array(insert, add, remove).
     * * **elementAdded:** Triggered when an element is added to the array(add, insert).
     * * **elementRemoved:** Triggered when an element is removed from the array
     *
     * @extends Parameter
     */
    class ListParameter extends Parameter {
        /**
         * Create a list parameter.
         * @param name - The name of the list parameter.
         * @param dataType - The dataType value.
         */
        constructor(name = '', dataType) {
            super(name, [], dataType);
        }
        /**
         * The filter method.
         * @param item - The item value.
         * @return - The return value.
         *
         * @private
         */
        filter(item) {
            return true;
        }
        /**
         * Returns the count of items in the array.
         *
         * @return - The return value.
         */
        getCount() {
            return this.value?.length || 0;
        }
        /**
         * Returns value from the array in the specified index.
         *
         * @param index - The index value.
         * @return - The return value.
         */
        getElement(index) {
            if (!this.value)
                return;
            return this.value[index];
        }
        /**
         * Sets a value in the specified array's index.
         *
         * @param index - The index value.
         * @param value - The value value.
         */
        setElement(index, value) {
            if (!this.value)
                this.value = [];
            this.value[index] = value;
            this.emit('valueChanged');
        }
        /**
         * Adds a new element at the end of the array pile.
         *
         * @param elem - The elem value.
         * @return - The return value.
         */
        addElement(elem) {
            if ((!elem && elem != 0) || !this.filter(elem))
                return;
            if (!this.value)
                this.value = [];
            this.value.push(elem);
            this.emit('elementAdded', { elem, index: this.value.length - 1 });
            this.emit('valueChanged');
            return elem;
        }
        /**
         * Removes an array element from the specified index
         *
         * @param index - The index value.
         */
        removeElement(index) {
            if (!this.value)
                this.value = [];
            const elem = this.value[index];
            this.value.splice(index, 1);
            this.emit('elementRemoved', { elem, index });
            this.emit('valueChanged');
        }
        /**
         * Inserts a new element in the specified index.
         *
         * @param index - The index value.
         * @param elem - The elem value.
         */
        insertElement(index, elem) {
            if (!this.value || !this.filter(elem))
                return;
            this.value.splice(index, 0, elem);
            this.emit('elementAdded', { elem, index });
            this.emit('valueChanged');
        }
        setValue(value) {
            if (!Array.isArray(value)) {
                throw new Error(`value provided is not an array. Check the source of this value`);
            }
            super.setValue(value);
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * The toJSON method encodes this type as a json object for persistence.
         *
         * @param context - The context value.
         * @return - Returns the json object.
         */
        toJSON(context) {
            const items = [];
            if (this.value) {
                for (const p of this.value) {
                    if (typeof this.dataType === 'string')
                        items.push(p);
                    else
                        items.push(p.toJSON(context));
                }
            }
            return {
                type: this.getClassName(),
                name: this.name,
                value: items,
            };
        }
        /**
         * The fromJSON method decodes a json object for this type.
         *
         * @param j - The json object this item must decode.
         * @param context - The context value.
         */
        fromJSON(j, context) {
            if (j.items == undefined) {
                console.warn('Invalid Parameter JSON');
                return;
            }
            const value = [];
            for (let i = 0; i < j.items.length; i++) {
                let elem;
                if (typeof this.dataType === 'string') {
                    elem = j.items[i];
                }
                else {
                    if (!this.dataType)
                        throw 'No DataType';
                    elem = Registry.constructClass(this.dataType);
                    elem.fromJSON(j.items[i], context);
                }
                value.push(elem);
                this.emit('elementAdded', { elem, index: this.value.length - 1 });
            }
            this.value = value;
        }
        // ////////////////////////////////////////
        // Clone and Destroy
        /**
         * The clone method constructs a new list parameter, copies its values
         * from this parameter and returns it.
         *
         * @return - Returns a new list parameter.
         */
        clone() {
            const clonedValue = this.value ? this.value.slice(0) : [];
            if (!this.dataType)
                throw 'This parameter does not have a DataType';
            const clonedParam = new ListParameter(this.name, this.dataType);
            clonedParam.setValue(clonedValue);
            return clonedParam;
        }
        /**
         * The destroy is called by the system to cause explicit resources cleanup.
         * Users should never need to call this method directly.
         */
        destroy() {
            if (!this.value)
                return;
            for (let i = 0; i < this.value.length; i++) {
                if (this.value[i] instanceof Parameter)
                    this.value[i].destroy();
                this.removeElement(i);
            }
        }
    }
    Registry.register('ListParameter', ListParameter);

    /* eslint-disable @typescript-eslint/explicit-module-boundary-types */
    /**
     * Represents a specific type of parameter, that stores multiple parameters in object format.
     *
     * i.e.:
     * ```javascript
     * const structParam = new StructParameter('MyStructParam')
     * //'myParameterOwnerItem' is an instance of a 'ParameterOwner' class.
     * // Remember that only 'ParameterOwner' and classes that extend from it can host 'Parameter' objects.
     * myParameterOwnerItem.addParameter(structParam)
     * ```
     *
     * **Events**
     * * **valueChanged:** Triggered whenever parameter's value changes.
     *
     * @extends Parameter
     */
    class StructParameter extends Parameter {
        members;
        /**
         * Create a struct parameter.
         * @param name - The name of the struct parameter.
         */
        constructor(name) {
            super(name, {}, 'Struct');
            this.members = [];
        }
        /**
         * The _addMember method.
         * @param parameter - The parameter value.
         * @return - The return value.
         * @private
         */
        addMember(parameter) {
            if (this.value)
                this.value[parameter.getName()] = parameter.value;
            parameter.on('valueChanged', () => {
                if (this.value)
                    this.value[parameter.getName()] = parameter.value;
            });
            this.members.push(parameter);
            this.emit('valueChanged');
            return parameter;
        }
        /**
         * The getParameter method.
         *
         * @private
         * @param name - The parameter name.
         * @return - The return value.
         */
        getParameter(name) {
            for (const p of this.members) {
                if (p.getName() == name)
                    return p;
            }
            return undefined;
        }
        /**
         * Looks for a member parameter with the specified name and returns it.
         *
         * @param name - The parameter name.
         * @return - The return value.
         */
        getMember(name) {
            return this.getParameter(name);
        }
        /**
         * Returns the name of all parameters in StructParameter.
         *
         * @return - The return value.
         */
        getMemberNames() {
            const names = [];
            for (let i = 0; i < this.members.length; i++) {
                const member = this.members[i];
                if (member != null)
                    names[i] = member.getName();
            }
            return names;
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * The toJSON method encodes this type as a json object for persistence.
         *
         * @param context - The context value.
         * @return - Returns the json object.
         */
        toJSON(context) {
            const members = [];
            for (const p of this.members)
                members.push(p.toJSON(context));
            return { type: this.getClassName(), name: this.name, members };
        }
        /**
         * The fromJSON method decodes a json object for this type.
         *
         * @param j - The json object this item must decode.
         * @param context - The context value.
         */
        fromJSON(j, context) {
            if (j.members == undefined) {
                console.warn('Invalid Parameter JSON');
                return;
            }
            for (let i = 0; i < j.members.length; i++) {
                if (j.members[i]) {
                    this.members[i].fromJSON(j.members[i], context);
                }
            }
            this.name = j.name;
        }
        clone() {
            const clonedParam = new StructParameter(this.name);
            return clonedParam;
        }
        // ////////////////////////////////////////
        // Destroy
        /**
         * The destroy is called by the system to cause explicit resources cleanup.
         * Users should never need to call this method directly.
         */
        destroy() {
            for (const p of this.members) {
                // TODO: not sure about this. I added a do-nothing destroy method in Parameter<T> to be overwritten
                // since only some subclasses use destroy.
                p.destroy();
            }
        }
    }
    Registry.register('StructParameter', StructParameter);

    /** Class representing an operator output.
     * @extends EventEmitter
     */
    class OperatorOutput {
        name;
        mode;
        op = null;
        param;
        paramBindIndex;
        detached;
        /**
         * Create an operator output.
         * @param name - The name value.
         * @param operatorOutputMode - The mode which the OperatorOutput uses to bind to its target parameter.
         */
        constructor(name, operatorOutputMode = OperatorOutputMode.OP_WRITE) {
            this.name = name;
            this.mode = operatorOutputMode;
            this.param = undefined;
            this.paramBindIndex = -1;
            this.detached = false;
        }
        /**
         * Sets operator that owns this output. Called by the operator when adding outputs
         * @param op - The operator object.
         */
        setOperator(op) {
            this.op = op;
        }
        /**
         * Returns operator that owns this output.
         * @return - The operator object.
         */
        getOperator() {
            return this.op;
        }
        /**
         * Returns mode that the output writes to be parameter. Must be a number from OperatorOutputMode
         * @return - The mode value.
         */
        getMode() {
            return this.mode;
        }
        /**
         * Returns true if this output is connected to a parameter.
         * @return - The return value.
         */
        isConnected() {
            return this.param != undefined;
        }
        /**
         * The getParam method.
         * @return - The return value.
         */
        getParam() {
            return this.param;
        }
        /**
         * Sets the Parameter for this output to write to.
         * @param param - The param value.
         * @param index - The index to bind at in the Parameter.
         */
        setParam(param, index = -1) {
            if (this.param) {
                this.param.unbindOperatorOutput(this);
            }
            this.param = param;
            if (this.param) {
                this.paramBindIndex = this.param.bindOperatorOutput(this, index);
            }
        }
        /**
         * Returns the index of the binding on the parameter of this OperatorOutput
         * up to date.
         * @return index - The index of the binding on the parameter.
         */
        getParamBindIndex() {
            return this.paramBindIndex;
        }
        /**
         * If bindings change on a Parameter, it will call this method to ensure the output index is
         * up to date.
         * @param index - The index of the binding on the parameter.
         */
        setParamBindIndex(index) {
            this.paramBindIndex = index;
        }
        /**
         * Propagates dirty to the connected parameter.
         */
        setDirty() {
            if (this.param) {
                this.param.setDirty(this.paramBindIndex);
            }
        }
        /**
         * The getValue method.
         * @return - The return value.
         */
        getValue() {
            if (this.param) {
                return this.param.getValueFromOp(this.paramBindIndex);
            }
            else {
                // @ts-expect-error ts-migrate(2554) FIXME: Expected 0-1 arguments, but got 2.
                throw new Error('Cannot call getValue on OperatorOutput that is not connected:', this.name);
            }
        }
        /**
         * When the value on a Parameter is modified by a user by calling 'setValue,
         * then if any operators are bound, the value of the Parameter cannot be modified
         * directly as it is the result of a computation. Instead, the Parameter calls
         * 'backPropagateValue' on the Operator to cause the Operator to handle propagating
         * the value to one or more of its inputs.
         * to its inputs.
         * @param value - The value param.
         * @return - The modified value.
         */
        backPropagateValue(value) {
            if (this.op) {
                value = this.op.backPropagateValue(value);
            }
            return value;
        }
        /**
         * The setClean method.
         * @param value - The value param.
         */
        setClean(value) {
            if (this.param) {
                this.param.setCleanFromOp(value, this.paramBindIndex);
            }
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * The toJSON method encodes this type as a json object for persistence.
         * @param context - The context value.
         * @return - Returns the json object.
         */
        toJSON(context) {
            const paramPath = this.param ? this.param.getPath() : '';
            return {
                name: this.name,
                paramPath: context && context.makeRelative ? context.makeRelative(paramPath) : paramPath,
                paramBindIndex: this.paramBindIndex,
            };
        }
        /**
         * The fromJSON method decodes a json object for this type.
         * @param j - The json object this item must decode.
         * @param context - The context value.
         */
        fromJSON(j, context) {
            if (j.paramPath) {
                // Note: the tree should have fully loaded by the time we are loading operators
                // even new items and groups should have been created. Operators and state machines
                // are loaded last.
                context?.resolvePath(j.paramPath, (param) => {
                    this.setParam(param, j.paramBindIndex);
                }, (reason) => {
                    console.warn("OperatorOutput: '" + this.name + "'. Unable to connect to:" + j.paramPath);
                });
            }
        }
        /**
         * The detach method is called when an operator is being removed from the scene tree.
         * It removes all connections to parameters in the scene.
         */
        detach() {
            // This function is called when we want to suspend an operator
            // from functioning because it is deleted and on the undo stack.
            // Once operators have persistent connections,
            // we will simply uninstall the output from the parameter.
            this.detached = true;
            this.paramBindIndex = this.param ? this.param.unbindOperatorOutput(this) : -1;
        }
        /**
         * The reattach method can be called when re-instating an operator in the scene.
         */
        reattach() {
            this.detached = false;
            if (this.param) {
                this.paramBindIndex = this.param.bindOperatorOutput(this, this.paramBindIndex);
            }
        }
        /**
         * The rebind rebinds the outputs to be at the top of the stack for its parameter.
         */
        rebind() {
            if (this.param) {
                this.param.unbindOperatorOutput(this);
                this.paramBindIndex = this.param.bindOperatorOutput(this);
            }
        }
    }
    class XfoOperatorOutput extends OperatorOutput {
    }

    /**
     * Class representing an operator.
     *
     */
    class Operator {
        name;
        inputs;
        outputs;
        /**
         * Create an operator.
         * @param name - The name value.
         */
        constructor(name = '') {
            this.name = name;
            this.inputs = new Map();
            this.outputs = new Map();
        }
        /**
         * This method sets the state of the operator to dirty which propagates
         * to the outputs of this operator, and which may then propagate to other
         * operators. When the scene is cleaned, which usually is caused by rendering
         * then the chain of operators are cleaned by triggering evaluation.
         * @private
         */
        setDirty() {
            this.outputs.forEach((output) => output.setDirty());
        }
        /**
         * The addInput method.
         * @param input - The name of the input, or the input object
         * @return - The return value.
         */
        addInput(input) {
            input.setOperator(this);
            this.inputs.set(input.name, input);
            this.setDirty();
            return input;
        }
        /**
         * The removeInput method.
         * @param input - The name of the input, or the input object
         */
        removeInput(input) {
            if (typeof input == 'string')
                input = this.getInput(input);
            if (input.getParam())
                input.setParam(undefined);
            this.inputs.delete(input.name);
        }
        /**
         * Getter for the number of inputs in this operator.
         * @return - Returns the number of inputs.
         */
        getNumInputs() {
            return this.inputs.size;
        }
        /**
         * The getInputByIndex method.
         * @param index - The index value.
         * @return - The return value.
         */
        getInputByIndex(index) {
            return Array.from(this.inputs.values())[index];
        }
        /**
         * The getInput method.
         * @param name - The name value.
         * @return - The return value.
         */
        getInput(name) {
            const input = this.inputs.get(name);
            if (!input)
                throw `Couldn't find an Input with the name of '${name}'`;
            return input;
        }
        /**
         * The addOutput method.
         * @param output - The name of the output, or the output object
         * @return - The return value.
         */
        addOutput(output) {
            output.setOperator(this);
            // if (this.getOutput(output.name)) throw new Error(`Operator output already exists ${output.name}`)
            this.outputs.set(output.name, output);
            this.setDirty();
            return output;
        }
        /**
         * The removeOutput method.
         * @param output - The name of the output, or the output object
         */
        removeOutput(output) {
            if (typeof output == 'string')
                output = this.getOutput(output);
            if (!(output instanceof OperatorOutput)) {
                throw new Error(`removeOutput only accepts string or OperatorInput`);
            }
            if (output.getParam())
                output.setParam();
            this.outputs.delete(output.name);
        }
        /**
         * Getter for the number of outputs in this operator.
         * @return - Returns the number of outputs.
         */
        getNumOutputs() {
            return this.outputs.size;
        }
        /**
         * The getOutputByIndex method.
         * @param index - The index value.
         * @return - The return value.
         */
        getOutputByIndex(index) {
            return Array.from(this.outputs.values())[index];
        }
        /**
         * The getOutput method.
         * @param name - The name value.
         * @return - The return value.
         */
        getOutput(name) {
            const output = this.outputs.get(name);
            if (!output)
                throw new Error(`Couldn't find an Output with the name of '${name}'`);
            return output;
        }
        /**
         * The evaluate method.
         * Computes the values of each of the outputs based on the values of the inputs
         * and the values of outputs with mode OP_READ_WRITE.
         * This method must be implemented by all Operators.
         */
        evaluate() {
            throw new Error('Not yet implemented');
        }
        /**
         * When the value on a Parameter is modified by a user by calling 'setValue,
         * then if any operators are bound, the value of the Parameter cannot be modified
         * directly as it is the result of a computation. Instead, the Parameter calls
         * 'backPropagateValue' on the Operator to cause the Operator to handle propagating
         * the value to one or more of its inputs.
         * to its inputs.
         * @param value - The value param.
         * @return - The modified value.
         */
        backPropagateValue(value) {
            // TODO: Implement me for custom manipulations.
            return value;
        }
        /**
         * The detach method.
         */
        detach() {
            this.inputs.forEach((input) => input.detach());
            this.outputs.forEach((output) => output.detach());
        }
        /**
         * The reattach method.
         */
        reattach() {
            this.inputs.forEach((input) => input.reattach());
            this.outputs.forEach((output) => output.reattach());
        }
        /**
         * The rebind method.
         */
        rebind() {
            this.outputs.forEach((output) => output.rebind());
        }
    }

    /** Class representing an operator input.
     * @extends EventEmitter
     */
    class OperatorInput {
        name;
        op;
        param;
        detached = false;
        /**
         * Create an operator input.
         * @param name - The name value.
         */
        constructor(name) {
            this.name = name;
        }
        /**
         * Sets operator that owns this input. Called by the operator when adding inputs
         * @param op - The operator object.
         */
        setOperator(op) {
            this.op = op;
        }
        /**
         * Returns operator that owns this input.
         * @return - The operator object.
         */
        getOperator() {
            return this.op;
        }
        /**
         * Returns true if this input is connected to a parameter.
         * @return - The return value.
         */
        isConnected() {
            return this.param != null;
        }
        /**
         * The getParam method.
         * @return - The return value.
         */
        getParam() {
            return this.param;
        }
        /**
         * @private
         * The handler function for when the input paramter changes.
         * @param event - The event object.
         */
        paramValueChanged() {
            if (this.op)
                this.op.setDirty();
        }
        /**
         * Assigns the Paramter to be used to provide the input value.
         * @param param - The param value.
         */
        setParam(param) {
            if (this.param) {
                this.param.unbindOperatorInput(this);
            }
            this.param = param;
            if (this.param) {
                this.param.bindOperatorInput(this);
            }
            // When an input param is assigned, the op should
            // propagate dirty to its outputs.
            if (this.op)
                this.op.setDirty();
        }
        /**
         * The getValue method.
         * @return - The return value.
         */
        getValue() {
            if (this.param)
                return this.param.value;
            throw new Error('Unable to getValue');
        }
        /**
         * The setValue method.
         * @param value - The value param.
         */
        setValue(value) {
            if (this.param) {
                this.param.setValue(value);
            }
        }
        /**
         * Propagates from the upstream parameter to the connected operator.
         */
        setDirty() {
            if (this.op) {
                this.op.setDirty();
            }
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * The toJSON method encodes this type as a json object for persistence.
         *
         * @param context - The context value.
         * @return - Returns the json object.
         */
        toJSON(context) {
            const absPath = this.param ? this.param.getPath() : [];
            const paramPath = (context && context.makeRelative ? context.makeRelative(absPath) : absPath);
            return {
                name: this.name,
                paramPath: paramPath,
            };
        }
        /**
         * The fromJSON method decodes a json object for this type.
         * @param j - The json object this item must decode.
         * @param context - The context value.
         */
        fromJSON(j, context) {
            if (j.paramPath) {
                // Note: the tree should have fully loaded by the time we are loading operators
                // even new items and groups should have been created. Operators and state machines
                // are loaded last.
                context?.resolvePath(j.paramPath, (param) => {
                    this.setParam(param);
                }, () => {
                    console.warn("OperatorInput: '" + this.name + "'. Unable to connect to:" + j.paramPath);
                });
            }
        }
        /**
         * The detach method is called when an operator is being removed from the scene tree.
         * It removes all connections to parameters in the scene.
         */
        detach() {
            // This function is called when we want to suspend an operator
            // from functioning because it is deleted and on the undo stack.
            // Once operators have persistent connections,
            // we will simply uninstall the output from the parameter.
            if (this.param) {
                this.param.unbindOperatorInput(this);
            }
        }
        /**
         * The reattach method can be called when re-instating an operator in the scene.
         */
        reattach() {
            this.detached = false;
            if (this.param) {
                this.param.bindOperatorInput(this);
            }
        }
    }
    class XfoOperatorInput extends OperatorInput {
    }

    /** The operator the calculates the global Xfo of a TreeItem based on its parents GlobalXfo and its own LocalXfo
     * @extends Operator
     * @private
     */
    class CalcGlobalXfoOperator extends Operator {
        parentGlobal = new XfoOperatorInput('ParentGlobal');
        localXfo = new XfoOperatorInput('LocalXfo');
        globalXfo = new XfoOperatorOutput('GlobalXfo');
        /**
         * Create a CalcGlobalXfoOperator operator.
         *
         * @param groupGlobalXfoParam - The GlobalXfo param found on the Group.
         * @param cuttingPlaneParam - The parameter on the Group which defines the displacement to apply to the members.
         */
        // TODO: adding new XfoP... to make inheritence work
        constructor(globalXfoParam, localXfoParam) {
            super('CalcGlobalXfoOperator');
            this.localXfo.setParam(localXfoParam);
            this.globalXfo.setParam(globalXfoParam);
            this.addInput(this.parentGlobal);
            this.addInput(this.localXfo);
            this.addOutput(this.globalXfo);
        }
        /**
         * The backPropagateValue method inverts the mathematics of the 'evaluate'
         * method so it can propagate the value backwards to its inputs.
         * @param value - the new value being set on the output GlobalXfo
         */
        backPropagateValue(value) {
            if (this.parentGlobal.isConnected()) {
                const parentGlobalXfo = this.parentGlobal.getValue();
                this.localXfo.setValue(parentGlobalXfo.inverse().multiply(value));
            }
            else {
                this.localXfo.setValue(value);
            }
        }
        /**
         * The evaluate method calculates a new global Xfo based on the parents Global Xfo,
         * and the local Xfo value.
         */
        evaluate() {
            const localXfo = this.localXfo.getValue();
            if (this.parentGlobal.isConnected()) {
                const parentGlobalXfo = this.parentGlobal.getValue();
                this.globalXfo.setClean(parentGlobalXfo.multiply(localXfo));
            }
            else {
                this.globalXfo.setClean(localXfo);
            }
        }
    }

    /**
     * Represents a specific type of parameter, that only stores `Box3` values.
     *
     * i.e.:
     * ```javascript
     * const boundingBox = new BoundingBoxParameter('MyBBox', new TreeItem())
     * //'myParameterOwnerItem' is an instance of a 'ParameterOwner' class.
     * // Remember that only 'ParameterOwner' and classes that extend from it can host 'Parameter' objects.
     * myParameterOwnerItem.addParameter(boundingBox)
     * ```
     * @extends Parameter
     */
    class BoundingBoxParameter extends Box3Parameter {
        // protected dirty: boolean, value, name
        treeItem;
        dirty = true;
        /**
         * Creates an instance of BoundingBoxParameter.
         * @param name - Name of the parameter
         * @param treeItem - `TreeItem` that contains `Box3` representing the Bounding Box
         */
        constructor(name = '', treeItem) {
            super(name);
            this.treeItem = treeItem;
        }
        /**
         * Makes parameter value be dirty, so when `getValue` is called,
         * an evaluation is then executed to re-calculate the BoundingBox
         *
         * @memberof BoundingBoxParameter
         */
        setDirty(index) {
            if (!this.dirty) {
                this.dirty = true;
                this.emit('valueChanged');
            }
            return true;
        }
        getValue() {
            if (this.dirty) {
                //@ts-ignore
                this.value = this.treeItem.cleanBoundingBox();
                this.dirty = false;
            }
            return super.getValue();
        }
        clone() {
            const bBox3Clone = new BoundingBoxParameter(this.name, this.treeItem);
            bBox3Clone.value = this.value?.clone();
            return bBox3Clone;
        }
        /**
         * We do not want this parameter serialized.
         */
        isDrivenByOperator() {
            return true;
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * The loadValue is used to change the value of a parameter, without triggering a
         * valueChanges, or setting the USER_EDITED state.
         *
         * @param value - The context value.
         */
        loadValue(value) {
            this.value = value.clone();
        }
    }
    Registry.register('BoundingBoxParameter', BoundingBoxParameter);

    /**
     * Class representing an Item in the scene tree with hierarchy capabilities (has children).
     * It has the capability to add and remove children.
     * **Parameters**
     * * **Visible(`BooleanParameter`):** Shows/Hides the item.
     * * **LocalXfo(`XfoParameter`):** Specifies the offset of this tree item from its parent.
     * * **GlobalXfo(`XfoParameter`):** Provides the computed world Xfo of this tree item.
     * * **BoundingBox(`BoundingBox`):** Provides the bounding box for the tree item and all of its children in the 3d scene.
     *
     * **Events**
     * * **globalXfoChanged:** Emitted when the value of GlobalXfo parameter changes.
     * * **visibilityChanged:** Emitted when the visibility on the tree item changes.
     * * **highlightChanged:** Emitted when the highlight on the tree item changes.
     * * **childAdded:** Emitted when a item is added as a child.
     * * **childRemoved:** Emitted when an item is removed from the child nodes.
     * * **pointerDown:** Emitted when a pointerDown event happens in an item.
     * * **pointerUp:** Emitted when a pointerUp event happens in an item.
     * * **pointerMove:** Emitted when a pointerMove event happens in an item.
     * * **pointerEnter:** Emitted when a pointerEnter event happens in an item.
     * * **pointerClick:** Emitted when a pointer is clicked on an item.
     * * **pointerDoubleClick:** Emitted when a pointer is double-clicked on an item.
     * * **pointerLongPress:** Emitted when a pointer is clicked and held on an item for a long time.
     *
     *
     * @extends {ParameterOwner}
     */
    class TreeItem extends ParameterOwner {
        // Controls if this TreeItem or its children contribute to the bounding boxes
        // in the scene. If set to false, Camera framing will ignore this item,
        disableBoundingBox = false;
        __childItems = [];
        __childItemsEventHandlers = [];
        childItemsMapping = {};
        childItemsMappingCorrupt = false;
        /**
         * @member globalXfoParam - Stores the global Xfo for this tree item.
         * global xfos are calculated from the localXfo and parentXfo.
         */
        globalXfoParam = new XfoParameter('GlobalXfo', new Xfo());
        /**
         * @member localXfoParam - Stores the local Xfo for this tree item.
         * local Xfos are the offset from the parent's coordinate frame.
         */
        localXfoParam = new XfoParameter('LocalXfo', new Xfo());
        /**
         * @member boundingBoxParam - Stores the bounding box for this tree item
         */
        boundingBoxParam = new BoundingBoxParameter('BoundingBox', this);
        /**
         * @member visibleParam - Whether this tree item is visible or not.
         * Any given tree item is also is affected by parent's visibility.
         */
        visibleParam = new BooleanParameter('Visible', true);
        /**
         * @member pickableParam - Whether this tree item ic pickable in the viewport or not.
         * Any given tree item is also is affected by parent's pick-ablility.
         */
        pickableParam = new BooleanParameter('Pickable', true);
        /**
         * @member opacityParam - Controls, in combination with Material transparency,
         * the opacity of this item and its children.
         */
        opacityParam = new NumberParameter('Opacity', 1, [0, 1]);
        highlightMapping = {};
        highlights = [];
        #visible = true;
        #pickable = true;
        visibleCounter = 1; // Visible by Default.
        pickableCounter = 1; // Pickable by Default.
        opacity = 1; // Opaque by Default.
        inheritedOpacityValues = new Map();
        globalXfoOp;
        /**
         * Creates a tree item with the specified name.
         *
         * @param name - The name of the tree item. It's the identifier of the tree item.
         * It's an identifier intended to be human readable.
         * It's included in the path that we use to access a particular item.
         * It's used to display it in the tree.
         */
        constructor(name) {
            super(name);
            // /////////////////////////////////////
            // Add parameters.
            this.addParameter(this.visibleParam);
            this.addParameter(this.pickableParam);
            this.addParameter(this.opacityParam);
            this.addParameter(this.localXfoParam);
            this.addParameter(this.globalXfoParam);
            this.addParameter(this.boundingBoxParam);
            this.globalXfoOp = new CalcGlobalXfoOperator(this.globalXfoParam, this.localXfoParam);
            this.globalXfoParam.on('valueChanged', (event) => {
                this.setBoundingBoxDirty();
                // Note: deprecate this event.
                this.emit('globalXfoChanged', event);
            });
            this.visibleParam.on('valueChanged', () => {
                this.visibleCounter += this.visibleParam.value ? 1 : -1;
                this.updateVisibility();
            });
            this.pickableParam.on('valueChanged', () => {
                this.pickableCounter += this.pickableParam.value ? 1 : -1;
                this.updatePickable();
            });
            this.opacityParam.on('valueChanged', () => {
                this.updateOpacity();
            });
        }
        setOwner(ownerItem) {
            if (ownerItem && !(ownerItem instanceof TreeItem)) {
                throw new Error('cannot setOwner');
            }
            if (this.ownerItem) {
                if (this.ownerItem instanceof TreeItem) {
                    // The effect of the invisible owner is removed.
                    if (!this.ownerItem.isVisible())
                        this.visibleCounter++;
                    if (!this.ownerItem.isPickable())
                        this.pickableCounter++;
                    const index = this.ownerItem.getChildIndex(this);
                    if (index >= 0)
                        this.ownerItem.unbindChild(index, this);
                }
            }
            super.setOwner(ownerItem);
            if (this.ownerItem) {
                if (this.ownerItem instanceof TreeItem) {
                    // The effect of the invisible owner is added.
                    if (!this.ownerItem.isVisible())
                        this.visibleCounter--;
                    if (!this.ownerItem.isPickable())
                        this.pickableCounter--;
                    this.globalXfoOp.getInput('ParentGlobal').setParam(this.ownerItem.globalXfoParam);
                }
            }
            else {
                this.globalXfoOp.getInput('ParentGlobal').setParam(undefined);
            }
            this.updateVisibility();
        }
        /**
         * Returns the parent of current TreeItem.
         *
         * @return - Returns the parent item.
         */
        getParentItem() {
            return this.getOwner();
        }
        /**
         * Sets the parent of current TreeItem.
         *
         * @param parentItem - The parent item.
         */
        setParentItem(parentItem) {
            this.setOwner(parentItem);
        }
        get parent() {
            return this.getOwner();
        }
        set parent(treeItem) {
            this.setOwner(treeItem);
        }
        // ////////////////////////////////////////
        // Visibility
        /**
         * Returns visible parameter value for current TreeItem.
         *
         * @return - The visible param value.
         */
        isVisible() {
            // Should never be more than 1, but can be less than 0.
            return this.visibleCounter > 0;
        }
        /**
         * Sets visible parameter value.
         *
         * @param val - The val param.
         */
        setVisible(visible) {
            this.visibleParam.value = visible;
        }
        /**
         * Updates current TreeItem visible state and propagates its value to children elements.
         *
         * @param val - The val param.
         */
        propagateVisibility(val) {
            this.visibleCounter += val;
            this.updateVisibility();
        }
        /**
         * The updateVisibility method.
         * @return - Returns a boolean.
         */
        updateVisibility() {
            const visible = this.visibleCounter > 0;
            if (visible != this.#visible) {
                this.#visible = visible;
                for (const childItem of this.__childItems) {
                    childItem.propagateVisibility(this.#visible ? 1 : -1);
                }
                this.emit('visibilityChanged', new StateChangedEvent(visible));
                // Note: we used to handle this by listening to a 'valueChanged' event on the
                // parameter.
                if (this.ownerItem instanceof TreeItem) {
                    this.ownerItem.setBoundingBoxDirty();
                }
                return true;
            }
            return false;
        }
        // ////////////////////////////////////////
        // Pickable
        /**
         * Returns visible parameter value for current TreeItem.
         *
         * @return - The visible param value.
         */
        isPickable() {
            // Should never be more than 1, but can be less than 0.
            return this.pickableCounter > 0;
        }
        /**
         * Updates current TreeItem visible state and propagates its value to children elements.
         *
         * @param val - The val param.
         */
        propagatePickable(val) {
            this.pickableCounter += val;
            this.updatePickable();
        }
        /**
         * The updatePickable method.
         * @return - Returns a boolean.
         */
        updatePickable() {
            const pickable = this.pickableCounter > 0;
            if (pickable != this.#pickable) {
                this.#pickable = pickable;
                for (const childItem of this.__childItems) {
                    childItem.propagatePickable(this.#pickable ? 1 : -1);
                }
                this.emit('pickabilityChanged', new StateChangedEvent(pickable));
                return true;
            }
            return false;
        }
        // ////////////////////////////////////////
        // Opacity
        /**
         * Returns the current status of the opacity value.
         *
         * @return - Returns true if the opacity value is less than 1.
         */
        isOpaque() {
            return this.opacity > 0.999;
        }
        /**
         * Set the value of opacity inherited by a given tree item.
         */
        setInheritedOpacity(parent, value) {
            this.inheritedOpacityValues.set(parent, value);
            this.updateOpacity();
        }
        /**
         * Calculates the new opacity value based on the opacityParam value
         * and the lowest of the inherited opacity values.
         */
        updateOpacity() {
            let inheritedOpacityValue = 1;
            this.inheritedOpacityValues.forEach((value) => {
                if (value < inheritedOpacityValue)
                    inheritedOpacityValue = value;
            });
            const wasOpaque = this.opacity > 0.999;
            this.opacity = this.opacityParam.value * inheritedOpacityValue;
            // else this.opacity = this.opacityParam.value
            for (const childItem of this.__childItems) {
                childItem.setInheritedOpacity(this, this.opacity);
            }
            const isOpaque = this.opacity > 0.999;
            this.emit('opacityChanged', new OpacityStateChangedEvent(isOpaque, wasOpaque != isOpaque));
        }
        // ////////////////////////////////////////
        // Highlights
        /**
         * Adds a highlight to the tree item.
         *
         * @param name - The name of the tree item.
         * @param color - The color of the highlight.
         * @param propagateToChildren - A boolean indicating whether to propagate to children.
         */
        addHighlight(name, color, propagateToChildren = true) {
            // If the highlight was already in the list,
            // remove it and put it at the top.
            if (name in this.highlightMapping) {
                if (this.highlights[this.highlights.length - 1] != name) {
                    // The highlight was already in the list, but not at the top. Move it to the top.
                    const id = this.highlights.indexOf(name);
                    this.highlights.splice(id, 1);
                    this.highlights.push(name);
                    this.emit('highlightChanged', { name, color });
                }
                else {
                    // This item is already highlighted with this highlight
                    if (!this.highlightMapping[name].isEqual(color)) {
                        this.highlightMapping[name] = color;
                        this.emit('highlightChanged', { name, color });
                    }
                }
            }
            else {
                this.highlights.push(name);
                this.highlightMapping[name] = color;
                this.emit('highlightChanged', { name, color });
            }
            if (propagateToChildren) {
                this.__childItems.forEach((childItem) => {
                    childItem.addHighlight(name, color, propagateToChildren);
                });
            }
        }
        /**
         * Removes a highlight to the tree item.
         *
         * @param name - The name of the tree item.
         * @param propagateToChildren - A boolean indicating whether to propagate to children.
         */
        removeHighlight(name, propagateToChildren = true) {
            if (name in this.highlightMapping) {
                if (this.highlights[this.highlights.length - 1] == name) {
                    this.highlights.pop();
                    delete this.highlightMapping[name];
                    if (this.highlights.length > 0) {
                        const nextName = this.highlights[this.highlights.length - 1];
                        const nextColor = this.highlightMapping[nextName];
                        this.emit('highlightChanged', { name: nextName, color: nextColor });
                    }
                    else {
                        // The last highlight was removed, so emit an event saying we are no longer highlighted.
                        this.emit('highlightChanged');
                    }
                }
                else {
                    // The removed highlight was not the current highlight, so no change needs to be shown.
                    const id = this.highlights.indexOf(name);
                    this.highlights.splice(id, 1);
                    delete this.highlightMapping[name];
                }
                if (propagateToChildren) {
                    this.__childItems.forEach((childItem) => {
                        childItem.removeHighlight(name, propagateToChildren);
                    });
                }
            }
        }
        /**
         * Returns the color of the current highlight.
         *
         * @return - The color value.
         */
        getHighlight() {
            if (this.highlights.length == 0) {
                return null;
            }
            return this.highlightMapping[this.highlights[this.highlights.length - 1]];
        }
        /**
         * Returns the name of the current highlight.
         *
         * @return - The color value.
         */
        getHighlightName() {
            if (this.highlights.length == 0) {
                return null;
            }
            return this.highlights[this.highlights.length - 1];
        }
        /**
         * Returns `true` if this items has a highlight color assigned.
         *
         * @return - `True` if this item is highlighted.
         */
        isHighlighted() {
            return this.highlights.length > 0;
        }
        // ////////////////////////////////////////
        // Bounding Box
        cleanBoundingBox() {
            const bbox = new Box3();
            this.__childItems.forEach((childItem) => {
                if (childItem.isVisible() && childItem.isPickable()) {
                    // console.log(" - ", childItem.constructor.name, childItem.getName(), childItem.globalXfoParam.value.sc.x, childItem.getBoundingBox().toString())
                    const box3 = childItem.boundingBoxParam.value;
                    if (box3)
                        bbox.addBox3(box3);
                }
            });
            // console.log(this.getName(), bbox.toString())
            return bbox;
        }
        setBoundingBoxDirty() {
            if (this.boundingBoxParam) {
                // Will cause boundingChanged to emit
                this.boundingBoxParam.setDirty(-1);
            }
            // Note: we used to handle this by listening to a 'valueChanged' event on the
            // parameter.
            if (this.ownerItem instanceof TreeItem) {
                this.ownerItem.setBoundingBoxDirty();
            }
        }
        // ////////////////////////////////////////
        // Children
        /**
         * Returns children list, but children are not required to have hierarchy structure(`TreeItem`).
         * Meaning that it could be another kind of item than `TreeItem`.
         *
         * i.e. **BaseImage**
         *
         * @return - List of `TreeItem` owned by current TreeItem.
         */
        getChildren() {
            return this.__childItems;
        }
        /**
         * Returns the number of child tree items.
         */
        getNumChildren() {
            return this.__childItems.length;
        }
        /**
         * Returns the number of child tree items.
         */
        get numChildren() {
            return this.__childItems.length;
        }
        /**
         * Apply an index to the name if name exists within this item's children.
         *
         * @param name - The name value.
         * @return - Returns a unique name.
         */
        generateUniqueName(name) {
            while (this.childItemsMapping[name.toLocaleLowerCase()] != undefined) {
                const indexRegexp = /(\d+)$/i;
                const indexMatches = name.match(indexRegexp);
                let newIndex = 1;
                if (indexMatches) {
                    newIndex = parseInt(indexMatches[1]) + 1;
                    name = name.replace(indexRegexp, ''); // remove old index
                }
                name += `${newIndex}`.padStart(2, '0');
            }
            return name;
        }
        /**
         * Updates the internal acceleration structure that speeds up looking up children by name.
         * @param start - The start value.
         * @private
         */
        updateChildNameMapping(start) {
            // If a child has been added or removed from the
            // tree item, we need to update the acceleration structure.
            for (let i = start; i < this.__childItems.length; i++) {
                const name = this.__childItems[i].name.toLocaleLowerCase();
                this.childItemsMapping[name] = i;
            }
        }
        /**
         * When a child's name changed, we update our acceleration structure.
         * @param event - The start value.
         * @private
         */
        childNameChanged(event) {
            // Update the acceleration structure.
            if (this.childItemsMappingCorrupt) {
                this.updateChildNameMapping(0);
                this.childItemsMappingCorrupt = false;
            }
            else {
                const oldName = event.oldName.toLocaleLowerCase();
                const newName = event.newName.toLocaleLowerCase();
                const index = this.childItemsMapping[oldName];
                if (this.childItemsMapping[newName] != undefined)
                    this.childItemsMappingCorrupt = true;
                delete this.childItemsMapping[oldName];
                this.childItemsMapping[newName] = index;
            }
        }
        /**
         * Inserts a child. It accepts all kind of `TreeItem`, not only `TreeItem`.
         *
         * @param childItem - The child TreeItem to insert.
         * @param index - The index to add the child item.
         * @param maintainXfo - Boolean that determines if the Xfo value is maintained.
         * @param fixCollisions - Modify the name of the item to avoid name collisions.
         * If false, an exception wll be thrown instead if a name collision occurs.
         * @return - The index of the child item in this items children array.
         */
        insertChild(childItem, index, maintainXfo = false, fixCollisions = true) {
            let name = childItem.name;
            let nameLc = name.toLocaleLowerCase();
            if (nameLc in this.childItemsMapping) {
                if (fixCollisions) {
                    name = this.generateUniqueName(name);
                    nameLc = name.toLocaleLowerCase();
                    childItem.setName(name);
                }
                else {
                    throw new Error("Item '" + name + "' is already a child of :" + this.getPath());
                }
            }
            if (!(childItem instanceof TreeItem)) {
                throw new Error('Object is is not a tree item :' + childItem); // TODO: need better output here+ childItem.constructor.name)
            }
            const listenerIDs = {};
            listenerIDs['nameChanged'] = childItem.on('nameChanged', (event) => {
                this.childNameChanged(event);
            });
            let prevGlobal;
            if (maintainXfo) {
                prevGlobal = childItem.globalXfoParam.value;
            }
            this.setBoundingBoxDirty();
            this.highlights.forEach((name) => {
                childItem.addHighlight(name, this.highlightMapping[name], true);
            });
            childItem.setInheritedOpacity(this, this.opacity);
            this.__childItems.splice(index, 0, childItem);
            this.__childItemsEventHandlers.splice(index, 0, listenerIDs);
            // If we have non-unique names, we need to regenerate this mapping.
            if (this.childItemsMapping[nameLc])
                this.childItemsMappingCorrupt = true;
            this.childItemsMapping[nameLc] = index;
            this.updateChildNameMapping(index + 1);
            childItem.setOwner(this);
            if (maintainXfo) {
                // Mainain the previous global Xfo.
                childItem.globalXfoParam.value = prevGlobal;
            }
            this.emit('childAdded', new ChildAddedEvent(index, childItem));
            return childItem;
        }
        /**
         * Adds a child.
         *
         * @param childItem - The child TreeItem to add.
         * @param maintainXfo - Boolean that determines if
         * the Global Xfo value is maintained. If true, when moving
         * items in the hierarchy from one parent to another, the local Xfo
         * of the item will be modified to maintain and the Global Xfo.
         * Note: this option defaults to false because we expect that is the
         * behavior users would expect when manipulating the tree in code.
         * To be safe and unambiguous, always try to specify this value.
         * @param fixCollisions - Modify the name of the item to avoid
         * name collisions with other children of the same parent.
         * If false, an exception wll be thrown instead if a name collision occurs.
         * @return childItem - The child TreeItem that was added.
         */
        addChild(childItem, maintainXfo = true, fixCollisions = true) {
            const index = this.__childItems.length;
            this.insertChild(childItem, index, maintainXfo, fixCollisions);
            return childItem;
        }
        /**
         * Returns child element in the specified index.
         *
         * @param index - The index to remove the child TreeItem.
         * @return - Return the child TreeItem.
         */
        getChild(index) {
            return this.__childItems[index];
        }
        /**
         * Returns child element with the specified name.
         *
         * @param name - The name value.
         * @return - Return the child TreeItem.
         */
        getChildByName(name) {
            const index = this.childItemsMapping[name.toLocaleLowerCase()];
            if (index != undefined) {
                return this.__childItems[index];
            }
            return null;
        }
        /**
         * Returns children names as an array of strings.
         *
         * @return - An array of names for each child.
         */
        getChildNames() {
            const names = [];
            for (let i = 0; i < this.__childItems.length; i++) {
                const childItem = this.__childItems[i];
                if (childItem != null)
                    names[i] = childItem.getName();
            }
            return names;
        }
        /**
         * UnBind an item from the group. This method is called
         * automatically when an item is removed from the group.
         * @param index - The index value.
         * @param childItem - item to unbind.
         * @private
         */
        unbindChild(index, childItem) {
            const listenerIDs = this.__childItemsEventHandlers[index];
            childItem.off('nameChanged', listenerIDs['nameChanged']);
            this.__childItems.splice(index, 1);
            this.__childItemsEventHandlers.splice(index, 1);
            delete this.childItemsMapping[childItem.name.toLocaleLowerCase()];
            this.updateChildNameMapping(index);
            this.setBoundingBoxDirty();
            this.emit('childRemoved', { childItem, index });
        }
        /**
         * Removes a child TreeItem by specifying its index.
         *
         * @param index - The index value.
         */
        removeChild(index) {
            const childItem = this.__childItems[index];
            if (!childItem) {
                return;
            }
            this.unbindChild(index, childItem);
            childItem.setOwner(undefined);
        }
        /**
         * Removes a child TreeItem by specifying its name.
         *
         * @param name - The name param.
         * @return - Return the child TreeItem.
         */
        removeChildByName(name) {
            const index = this.childItemsMapping[name.toLocaleLowerCase()];
            if (index != undefined) {
                return this.removeChild(index);
            }
        }
        /**
         * Removes the provided item from this TreeItem if it is one of its children.
         * An exception is thrown if the item is not a child of this tree item.
         *
         * @param childItem - The child TreeItem to remove.
         */
        removeChildByHandle(childItem) {
            const index = this.__childItems.indexOf(childItem);
            if (index == -1)
                throw new Error('Error in removeChildByHandle. Child not found:' + childItem.getName());
            this.removeChild(index);
        }
        /**
         * Removes all children Items.
         */
        removeAllChildren() {
            let index = this.__childItems.length;
            while (index--) {
                this.removeChild(index);
            }
            this.setBoundingBoxDirty();
        }
        /**
         * Returns index position of the specified item.
         *
         * @param childItem - The child TreeItem value.
         * @return - Child index in children array.
         */
        getChildIndex(childItem) {
            return this.__childItems.indexOf(childItem);
        }
        // ////////////////////////////////////////
        // Path Traversal
        // Note: Path resolution starts at the root of the
        // tree the path was generated from (so index=1, because we don't resolve root).
        // Note: When a path is made relative to an item in its tree, the path
        // starts with the child elements.
        /**
         * The resolvePath method traverses the subtree from this item down
         * matching each name in the path with a child until it reaches the
         * end of the path.
         *
         * @param path - The path value.
         * @param index - The index value.
         * @return - The return value.
         */
        resolvePath(path, index = 0) {
            if (index == 0) {
                if (path[0] == '.' || path[0] == this.name)
                    index++;
            }
            if (path[index] == '..') {
                if (this.ownerItem) {
                    return this.ownerItem.resolvePath(path, index + 1);
                }
                else {
                    throw Error('this.ownerItem is undefined');
                }
            }
            if (index == path.length) {
                return this;
            }
            const childName = path[index];
            const childItem = this.getChildByName(childName);
            if (childItem) {
                return childItem.resolvePath(path, index + 1);
            }
            return super.resolvePath(path, index);
        }
        /**
         * Traverse the tree structure from this point down
         * and fire the callback for each visited item.
         * Note: Depth only used by selection sets for now.
         *
         * @param callback - The callback value.
         * @param includeThis - Fire the callback for this item.
         */
        traverse(callback, includeThis = true) {
            const __c = (treeItem, depth) => {
                const children = treeItem.getChildren();
                for (const childItem of children) {
                    if (childItem)
                        __t(childItem, depth);
                }
            };
            const __t = (treeItem, depth) => {
                if (callback(treeItem, depth) == false)
                    return;
                __c(treeItem, depth + 1);
            };
            if (includeThis) {
                __t(this, 1);
            }
            else {
                __c(this, 0);
            }
        }
        // ///////////////////////
        // Events
        /**
         * Called by the Viewport when events are received by the canvas element.
         * The event is propagated to a TreeItem if it is under the pointer at the time.
         * The ZeaPointerEvent abstracts the Mouse, touch and our custom XR events.
         * This method emits the ZeaPointerEvent with the key 'pointerDown', and
         * propagates it up to the TreeItem's owner.
         *
         * @param event - The event value
         */
        onPointerDown(event) {
            this.emit('pointerDown', event);
            if (event.propagating && this.ownerItem instanceof TreeItem) {
                this.ownerItem.onPointerDown(event);
            }
        }
        /**
         * Called by the Viewport when events are received by the canvas element.
         * The event is propagated to a TreeItem if it is under the pointer at the time.
         * The ZeaPointerEvent abstracts the Mouse, touch and our custom XR events.
         * This method emits the ZeaPointerEvent with the key 'pointerDown', and
         * propagates it up to the TreeItem's owner.
         *
         * @param event - The pointer event that was generated from the user interaction
         */
        onPointerUp(event) {
            this.emit('pointerUp', event);
            if (event.propagating && this.ownerItem instanceof TreeItem) {
                this.ownerItem.onPointerUp(event);
            }
        }
        /**
         * Called by the Viewport when events are received by the canvas element.
         * The event is propagated to a TreeItem if it is under the pointer at the time.
         * The ZeaPointerEvent abstracts the Mouse, touch and our custom XR events.
         * This method emits the ZeaPointerEvent with the key 'pointerMove', and
         * propagates it up to the TreeItem's owner.
         *
         * @param event - The pointer event that was generated from the user interaction
         */
        onPointerMove(event) {
            this.emit('pointerMove', event);
            if (event.propagating && this.ownerItem instanceof TreeItem) {
                this.ownerItem.onPointerMove(event);
            }
        }
        /**
         * Called by the Viewport when the mouse or other pointer enters the canvas element.
         * The event is propagated to a TreeItem if it is under the pointer at the time.
         * The ZeaPointerEvent abstracts the Mouse, touch and our custom XR events.
         * This method emits the ZeaPointerEvent with the key 'pointerEnter', and
         * propagates it up to the TreeItem's owner.
         *
         * @param event - The pointer event that was generated from the user interaction
         */
        onPointerEnter(event) {
            this.emit('pointerEnter', event);
            if (event.propagating && this.ownerItem instanceof TreeItem) {
                this.ownerItem.onPointerEnter(event);
            }
        }
        /**
         * Called by the Viewport when the mouse or other pointer leaves the canvas element.
         * The event is propagated to a TreeItem if it is under the pointer at the time.
         * The ZeaPointerEvent abstracts the Mouse, touch and our custom XR events.
         * This method emits the ZeaPointerEvent with the key 'pointerLeave', and
         * propagates it up to the TreeItem's owner.
         *
         * @param event - The pointer event that was generated from the user interaction
         */
        onPointerLeave(event) {
            this.emit('pointerLeave', event);
            if (event.propagating && this.ownerItem instanceof TreeItem) {
                this.ownerItem.onPointerLeave(event);
            }
        }
        /**
         * Called by the Viewport when the mouse or other pointer is clicked on this item.
         *
         * @param event - The pointer event that was generated from the user interaction
         */
        onPointerClick(event) {
            this.emit('pointerClick', event);
            if (event.propagating && this.ownerItem instanceof TreeItem) {
                this.ownerItem.onPointerClick(event);
            }
        }
        /**
         * Called by the Viewport when the mouse or other pointer is double-clicked on this item.
         *
         * @param event - The pointer event that was generated from the user interaction
         */
        onPointerDoubleClick(event) {
            this.emit('pointerDoubleClick', event);
            if (event.propagating && this.ownerItem instanceof TreeItem) {
                this.ownerItem.onPointerDoubleClick(event);
            }
        }
        /**
         * Called by the Viewport when the mouse or other pointer is double-clicked on this item.
         *
         * @param event - The pointer event that was generated from the user interaction
         */
        onPointerLongPress(event) {
            this.emit('pointerLongPress', event);
            if (event.propagating && this.ownerItem instanceof TreeItem) {
                this.ownerItem.onPointerLongPress(event);
            }
        }
        /**
         * Called by the Viewport when the mouse wheel event is received by the canvas element.
         * Emits the ZeaWheelEvent with the key 'mouseWheel', and Propagates is up to the TreeItem's owner.
         *
         * @param event - The wheel event that occurs.
         */
        onWheel(event) {
            this.emit('mouseWheel', event);
            if (event.propagating && this.ownerItem instanceof TreeItem) {
                this.ownerItem.onWheel(event);
            }
        }
        /**
         * Called by the Viewport when the touch cancel event is received by the canvas element.
         * Emits the ZeaTouchEvent with the key 'touchCancel', and Propagates is up to the TreeItem's owner.
         *
         * @param event - The wheel event that occurs.
         */
        onTouchCancel(event) {
            this.emit('touchCancel', event);
            if (event.propagating && this.ownerItem instanceof TreeItem) {
                this.ownerItem.onTouchCancel(event);
            }
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * The toJSON method serializes this instance as a JSON.
         * It can be used for persistence, data transfer, etc.
         *
         * @param context - The context value.
         * @return - Returns the json object.
         */
        toJSON(context) {
            let j = super.toJSON(context);
            // Some Items, such as the SliderSceneWidget do not need their children
            // to be saved.
            const childItemsJSON = {};
            for (const childItem of this.__childItems) {
                if (childItem) {
                    const childJSON = childItem.toJSON(context);
                    if (childJSON)
                        childItemsJSON[childItem.getName()] = childJSON;
                }
            }
            if (Object.keys(childItemsJSON).length > 0) {
                if (j) {
                    j.children = childItemsJSON;
                }
                else {
                    j = {
                        name: this.name,
                        children: childItemsJSON,
                    };
                }
            }
            return j;
        }
        /**
         * The fromJSON method takes a JSON and deserializes into an instance of this type.
         *
         * @param j - The json object this item must decode.
         * @param context - The context value.
         */
        fromJSON(j, context, onDone) {
            super.fromJSON(j, context);
            // if ('bbox' in j){
            //     let box = new Box3();
            //     box.fromJSON(j.bbox);
            //     this.boundingBoxParam.value = box)
            // }
            if (j.children != null) {
                const childrenJson = j.children;
                if (Array.isArray(childrenJson)) {
                    for (const childJson of childrenJson) {
                        // Note: During loading of asset trees, we have an
                        // existing tree generated by loading a bin data file.
                        let childItem = this.getChildByName(childJson.name);
                        if (childItem) {
                            childItem.fromJSON(childJson, context);
                        }
                        else {
                            if (childJson.type) {
                                childItem = Registry.constructClass(childJson.type);
                                if (childItem) {
                                    // Note: we should load the json first, as it
                                    // may contain the unique name of the item.
                                    childItem.fromJSON(childJson, context);
                                    this.addChild(childItem, false, false);
                                }
                            }
                        }
                    }
                }
                else {
                    // eslint-disable-next-line guard-for-in
                    for (const childName in childrenJson) {
                        const childJson = childrenJson[childName];
                        // Note: During loading of asset trees, we have an
                        // existing tree generated by loading a bin data file.
                        let childItem = this.getChildByName(childName);
                        if (childItem) {
                            childItem.fromJSON(childJson, context);
                        }
                        else if (childJson.type) {
                            childItem = Registry.constructClass(childJson.type);
                            if (childItem) {
                                // Note: we add the child now before loading.
                                // This is because certain items. (e.g. Groups)
                                // Calculate their global Xfo, and use it to modify
                                // the transform of their members.
                                // Note: Groups bind to items in the scene which are
                                // already added as children, and so have global Xfos.
                                // We prefer to add a child after its loaded, because sometimes
                                // In the tree is asset items, who will only toggled as
                                // unloaded once they are loaded(else they are considered inline assets.)
                                childItem.fromJSON(childJson, context);
                                this.addChild(childItem, false, false);
                            }
                        }
                        else ;
                    }
                }
            }
            // if (j.components) {
            //   for (const cj of j.components) {
            //     const component = Registry.constructClass(cj.type ? cj.type : cj.name)
            //     if (component) {
            //       component.fromJSON(cj, context)
            //       this.addComponent(component)
            //     }
            //   }
            // }
        }
        /**
         * Sets state of current Item(Including parameters & children) using a binary reader object.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            super.readBinary(reader, context);
            const itemFlags = reader.loadUInt8();
            const visibilityFlag = 1 << 1;
            // Note: XRef loads a visiblity setting for itself when loading in the parent assembly tree,
            // then it loads the zcad file which contains a visibility setting for the CADAssset. This
            // visibility setting would override the XRef. Ideally the zcad file would not store a visiblity
            // setting for the CADAsset, as that should be set by the loading context.
            // Visible is true by default, so only set it to false. This is to work around the above conflict.
            if ((itemFlags & visibilityFlag) == 0)
                this.setVisible(false);
            // Note: to save space, some values are skipped if they are identity values
            const localXfoFlag = 1 << 2;
            const localXfoIndependentScFlag = 1 << 5;
            if (itemFlags & localXfoFlag) {
                const xfo = new Xfo();
                xfo.tr = reader.loadFloat32Vec3();
                xfo.ori = reader.loadFloat32Quat();
                if (itemFlags & localXfoIndependentScFlag) {
                    xfo.sc = reader.loadFloat32Vec3();
                }
                else {
                    const sc = reader.loadFloat32();
                    xfo.sc.set(sc, sc, sc);
                }
                this.localXfoParam.value = xfo;
            }
            const bboxFlag = 1 << 3;
            if (itemFlags & bboxFlag) {
                this.boundingBoxParam.loadValue(new Box3(reader.loadFloat32Vec3(), reader.loadFloat32Vec3()));
            }
            const numChildren = reader.loadUInt32();
            if (numChildren > 0) {
                const toc = reader.loadUInt32Array(numChildren);
                for (let i = 0; i < numChildren; i++) {
                    try {
                        reader.seek(toc[i]); // Reset the pointer to the start of the item data.
                        let childType = reader.loadStr();
                        const childItem = Registry.constructClass(childType);
                        if (!childItem) {
                            const childName = reader.loadStr();
                            console.warn('Unable to construct child:' + childName + ' of type:' + childType);
                            continue;
                        }
                        reader.seek(toc[i]); // Reset the pointer to the start of the item data.
                        childItem.readBinary(reader, context);
                        this.addChild(childItem, false, true);
                    }
                    catch (e) {
                        console.warn('Error loading tree item: ', e);
                    }
                }
            }
        }
        // ////////////////////////////////////////
        // Clone and Destroy
        clone(context) {
            const cloned = new TreeItem('');
            cloned.copyFrom(this, context);
            return cloned;
        }
        /**
         * Copies current TreeItem with all its children.
         *
         * @param src - The tree item to copy from.
         * @param context - The context value.
         */
        copyFrom(src, context) {
            if (!(src instanceof TreeItem)) {
                throw new Error('cannot copy from src');
            }
            super.copyFrom(src, context);
            // Note: configure visiblity and opacity before adding children. They will
            // inherit these settings as they are added.
            this.visibleCounter = this.visibleParam.value ? 1 : 0;
            this.updateVisibility();
            this.updateOpacity();
            // Note: If a tree item already had children, we must clear them before adding from the src.
            // An XRef stored in an assembly may contain a cache of its subtree.
            // If the XRef url does resolve to a file, we must clear this cache before loading the actual data.
            this.removeAllChildren();
            src.getChildren().forEach((srcChildItem) => {
                if (srcChildItem)
                    this.addChild(srcChildItem.clone(context), false, false);
            });
        }
    }
    Registry.register('TreeItem', TreeItem);

    /* eslint-disable @typescript-eslint/no-explicit-any */
    /**
     * Represents a specific type of parameter, that only stores `TreeItem` values.
     *
     * i.e.:
     * ```javascript
     * const treeItem = new TreeItem('tree1')
     * const treeItemParam = new TreeItemParameter('MyTreeItem', treeItem)
     * //'myParameterOwnerItem' is an instance of a 'ParameterOwner' class.
     * // Remember that only 'ParameterOwner' and classes that extend from it can host 'Parameter' objects.
     * myParameterOwnerItem.addParameter(treeItemParam)
     * ```
     *
     * **Events**
     * * **treeItemGlobalXfoChanged:** Triggered when computed world Xfo of parameter's `TreeItem` changes.
     * * **valueChanged:** Triggered when parameter's value changes.
     *
     * @extends Parameter
     */
    class TreeItemParameter extends Parameter {
        filterFn;
        listenerIDs = {};
        /**
         * Create a tree item parameter.
         * @param name - The name of the tree item parameter.
         * @param filterFn - The filterFn value.
         */
        constructor(name = '', filterFn) {
            super(name, null, 'TreeItem');
            this.filterFn = filterFn;
        }
        emitTreeItemGlobalXfoChanged(event) {
            this.emit('treeItemGlobalXfoChanged', event);
        }
        /**
         * The setFilterFn method.
         * @param filterFn - The filterFn value.
         */
        setFilterFn(filterFn) {
            this.filterFn = filterFn;
        }
        /**
         * The getFilterFn method.
         * @return - The return value.
         */
        getFilterFn() {
            return this.filterFn;
        }
        /**
         * Sets parameter's `TreeItem` value.
         *
         * @param value - The treeItem value
         * @return - The return value.
         */
        setValue(value) {
            if (!(value instanceof TreeItem)) {
                throw new Error(`value provided is not an instance of a 'TreeItem' class. Check the source of this value`);
            }
            // 0 == normal set. 1 = changed via cleaner fn, 2=change by loading/cloning code.
            if (this.filterFn && !this.filterFn(value))
                return;
            if (this.value !== value) {
                if (this.value) {
                    this.value.off('globalXfoChanged', this.listenerIDs['globalXfoChanged']);
                }
                super.setValue(value);
                if (this.value) {
                    this.listenerIDs['globalXfoChanged'] = this.value.on('globalXfoChanged', (event) => {
                        this.emitTreeItemGlobalXfoChanged(event);
                    });
                }
            }
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * The toJSON method encodes this type as a json object for persistence.
         *
         * @param context - The context value.
         * @return - Returns the json object.
         */
        toJSON(context) {
            return {
                type: this.getClassName(),
                name: this.name,
                value: context.makeRelative(this.value?.getPath()),
            };
        }
        /**
         * The fromJSON method decodes a json object for this type.
         *
         * @param j - The json object this item must decode.
         * @param context - The context value.
         */
        fromJSON(j, context) {
            if (j.value == null) {
                console.warn('Invalid Parameter JSON');
                return;
            }
            context.resolvePath(j.value, (treeItem) => {
                this.setValue(treeItem);
            }, () => {
                console.warn('Unable to resolve tree item parameter value:' + j.paramPath);
            });
        }
        // ////////////////////////////////////////
        // Clone and Destroy
        /**
         * The clone method constructs a new tree item parameter, copies its values
         * from this parameter and returns it.
         *
         * @return - Returns a new tree item parameter.
         */
        clone(context) {
            const clonedParam = new TreeItemParameter(this.name, this.filterFn);
            if (this.value)
                clonedParam.setValue(this.value.clone(context));
            return clonedParam;
        }
    }
    Registry.register('TreeItemParameter', TreeItemParameter);

    /* eslint-disable @typescript-eslint/no-explicit-any */
    class ItemEvent extends BaseEvent {
        item;
        index;
        constructor(item, index) {
            super();
            this.index = index;
            this.item = item;
        }
    }
    /** Class representing an item set parameter.
     * @extends Parameter
     * @private
     */
    class ItemSetParameter extends Parameter {
        filterFn;
        /**
         * Create an item set parameter.
         * @param name - The name of the item set parameter.
         * @param filterFn - The filterFn value.
         */
        constructor(name = '', filterFn) {
            super(name, new Set(), 'TreeItem');
            this.filterFn = filterFn; // Note: the filter Fn indicates that users will edit the set.
        }
        /**
         * The setFilterFn method.
         * @param filterFn - The filterFn value.
         */
        setFilterFn(filterFn) {
            this.filterFn = filterFn;
        }
        /**
         * The getFilterFn method.
         * @return - The return value.
         */
        getFilterFn() {
            return this.filterFn;
        }
        /**
         * The getItem method.
         * @param index - The index param.
         * @return - The return value.
         */
        getItem(index) {
            // if (!this.__items) return undefined
            return Array.from(this.value)[index];
        }
        /**
         * The addItem method.
         * @param item - The item value.
         * @param emitValueChanged - The emit value.
         * @return - The return value.
         */
        addItem(item, emitValueChanged = true) {
            if (this.filterFn && !this.filterFn(item)) {
                console.warn('ItemSet __filterFn rejecting item:', item.getPath());
                return;
            }
            if (!this.value.has(item)) {
                this.value.add(item);
                const index = Array.from(this.value).indexOf(item);
                this.emit('itemAdded', new ItemEvent(item, index));
                if (emitValueChanged)
                    this.emit('valueChanged');
                return index;
            }
            else {
                return -1;
            }
        }
        /**
         * Adds items to the parameter value
         *
         * @param items - list of items to add to the parameter
         * @param emitValueChanged
         * @memberof ItemSetParameter
         */
        addItems(items, emitValueChanged = true) {
            items.forEach((item) => this.addItem(item, false));
            if (emitValueChanged)
                this.emit('valueChanged');
        }
        /**
         * The removeItem method.
         * @param index - The index value.
         * @param emitValueChanged - The emit param.
         * @return - The return value.
         */
        removeItem(index, emitValueChanged = true) {
            const item = Array.from(this.value)[index];
            this.value.delete(item);
            this.emit('itemRemoved', new ItemEvent(item, index));
            if (emitValueChanged)
                this.emit('valueChanged');
            return item;
        }
        /**
         * The setItems method.
         * @param items - The item param.
         * @param emit - The emit param.
         */
        setItems(items, emit = true) {
            const values = Array.from(this.value);
            for (let i = values.length - 1; i >= 0; i--) {
                const item = values[i];
                if (!items.has(item)) {
                    this.removeItem(i, false);
                }
            }
            for (const item of items) {
                if (!this.value.has(item)) {
                    this.addItem(item, false);
                }
            }
            if (emit)
                this.emit('valueChanged');
        }
        /**
         * The clearItems method.
         * @param emit - The emit value.
         */
        clearItems(emitValueChanged = true) {
            this.value.clear();
            if (emitValueChanged)
                this.emit('valueChanged');
        }
        /**
         * The getNumItems method.
         * @return - The return value.
         */
        getNumItems() {
            return this.value.size; // might be faster
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * The toJSON method encodes this type as a json object for persistence.
         * @param context - The context value.
         * @return - The return value.
         */
        toJSON(context) {
            if (!this.value)
                this.value = new Set();
            const paths = [];
            for (const item of this.value) {
                const path = item.getPath();
                paths.push(context && context.makeRelative ? context.makeRelative(path) : path);
            }
            return {
                type: this.getClassName(),
                name: this.name,
                value: paths,
            };
        }
        /**
         * The fromJSON method decodes a json object for this type.
         * @param j - The json object this item must decode.
         * @param context - The context value.
         */
        fromJSON(j, context) {
            if (!context || !context.resolvePath) {
                throw new Error('Unable to load JSON on a ItemSetParameter without a load context');
            }
            const paths = j.value;
            paths.forEach((path) => {
                context.resolvePath(path, (treeItem) => {
                    this.addItem(treeItem, false);
                }, () => {
                    console.warn("BaseGroup: '" + this.getName() + "'. Unable to load item:" + path);
                });
            });
        }
        // ////////////////////////////////////////
        // Clone
        /**
         * The clone method constructs a item set new parameter, copies its values
         * from this parameter and returns it.
         *
         * @return - Returns a new item set parameter.
         */
        clone() {
            const clonedParam = new ItemSetParameter(this.name, this.filterFn);
            return clonedParam;
        }
    }
    Registry.register('ItemSetParameter', ItemSetParameter);

    /* eslint-disable @typescript-eslint/no-explicit-any */
    const parse8BitPositionsArray = (range, offset, sclVec, positions_quantized, positionsAttr) => {
        for (let i = range[0]; i < range[1]; i++) {
            const pos = new Vec3(positions_quantized[i * 3 + 0] / 255.0, positions_quantized[i * 3 + 1] / 255.0, positions_quantized[i * 3 + 2] / 255.0);
            pos.multiplyInPlace(sclVec);
            pos.addInPlace(offset);
            positionsAttr.setValue(i, pos);
        }
    };
    const parse16BitPositionsArray = (range, offset, sclVec, positions_quantized, positionsAttr) => {
        for (let i = range[0]; i < range[1]; i++) {
            const pos = new Vec3(positions_quantized[i * 3 + 0] / 65535.0, positions_quantized[i * 3 + 1] / 65535.0, positions_quantized[i * 3 + 2] / 65535.0);
            pos.multiplyInPlace(sclVec);
            pos.addInPlace(offset);
            positionsAttr.setValue(i, pos);
        }
    };
    const parse8BitNormalsArray = (range, offset, sclVec, normals_quantized, normalsAttr) => {
        if (sclVec.isNull())
            sclVec.set(1, 1, 1);
        for (let i = range[0]; i < range[1]; i++) {
            const normal = new Vec3(normals_quantized[i * 3 + 0] / 255.0, normals_quantized[i * 3 + 1] / 255.0, normals_quantized[i * 3 + 2] / 255.0);
            normal.multiplyInPlace(sclVec);
            normal.addInPlace(offset);
            normal.normalizeInPlace();
            normalsAttr.setValue(i, normal);
        }
    };
    const parse8BitTextureCoordsArray = (range, offset, sclVec, texCoords_quantized, texCoordsAttr) => {
        // if (sclVec.isNull())
        //     sclVec.set(1, 1, 1);
        for (let i = range[0]; i < range[1]; i++) {
            const textureCoord = new Vec2(texCoords_quantized[i * 2 + 0] / 255.0, texCoords_quantized[i * 2 + 1] / 255.0);
            textureCoord.multiplyInPlace(sclVec);
            textureCoord.addInPlace(offset);
            texCoordsAttr.setValue(i, textureCoord);
        }
    };
    const parse16BitTextureCoordsArray = (range, offset, sclVec, texCoords_quantized, texCoordsAttr) => {
        // if (sclVec.isNull())
        //     sclVec.set(1, 1, 1);
        for (let i = range[0]; i < range[1]; i++) {
            const textureCoord = new Vec2(texCoords_quantized[i * 2 + 0] / 65535.0, texCoords_quantized[i * 2 + 1] / 65535.0);
            textureCoord.multiplyInPlace(sclVec);
            textureCoord.addInPlace(offset);
            texCoordsAttr.setValue(i, textureCoord);
        }
    };
    /**
     * Represents a base class for 3D geometry items.
     *
     * **Events**
     * * **boundingBoxChanged:** Triggered when the bounding box changes.
     * * **geomDataChanged:** Emitted when the geometry attributes have changed. The topology did not change. The Renderer will upload the new attributes to the GPU.
     * * **geomDataTopologyChanged:** Emitted when the geometry attributes and topology have changed.  The Renderer will upload the new attributes and topology to the GPU.
     *
     * @extends ParameterOwner
     */
    class BaseGeom extends ParameterOwner {
        boundingBox = new Box3();
        boundingBoxDirty = true;
        __metaData = new Map();
        #numVertices = 0;
        __vertexAttributes = new Map();
        debugColor = new Color(1, 0, 0, 1);
        /**
         * Create a base geom.
         */
        constructor() {
            super();
            this.addVertexAttribute('positions', new Vec3f16Attribute());
        }
        /**
         * Returns the current path of the item in the tree as an array of names.
         *
         * @return - Returns an array.
         */
        getPath() {
            if (this.ownerItem == undefined)
                return [this.name];
            else if (this.ownerItem instanceof Parameter) {
                return [...this.ownerItem.getPath(), 'value'];
            }
            else {
                return [...this.ownerItem.getPath(), this.name];
            }
        }
        /**
         * The clear method.
         */
        clear() {
            this.setNumVertices(0);
        }
        /**
         * Adds a new vertex attribute to the geometry.
         *
         * @param name - The name of the vertex attribute.
         * @param dataType - The dataType value. // TODO: is any ok vs. AttrValue | number. Unsure about how dataType is used
         * @return - Returns an attribute.
         */
        addVertexAttribute(name, attr) {
            attr.setCount(this.#numVertices);
            this.__vertexAttributes.set(name, attr);
        }
        /**
         * Checks if the the geometry has an attribute with the specified name.
         *
         * @param name - The name of the vertex attribute.
         * @return - The return value.
         */
        hasVertexAttribute(name) {
            return this.__vertexAttributes.has(name);
        }
        /**
         * Returns vertex attribute with the specified name.
         *
         * @param name - The name of the vertex attribute.
         * @return - The return value.
         */
        getVertexAttribute(name) {
            return this.__vertexAttributes.get(name);
        }
        /**
         * Returns all vertex attributes in an object with their names.
         *
         * @return - The return value.
         */
        getVertexAttributes() {
            const vertexAttributes = {};
            for (const [key, attr] of this.__vertexAttributes.entries())
                vertexAttributes[key] = attr;
            return vertexAttributes;
        }
        /**
         * Returns 'positions' vertex attribute.
         */
        get positions() {
            return this.__vertexAttributes.get('positions');
        }
        /**
         * Returns the number of vertex attributes.
         *
         * @return - The return value.
         */
        numVertices() {
            return this.#numVertices;
        }
        /**
         * Returns the number of vertex attributes.
         *
         * @return - The return value.
         */
        getNumVertices() {
            return this.#numVertices;
        }
        /**
         * Sets the number of vertices the geometry has.
         *
         * @param count - The count value.
         */
        setNumVertices(count) {
            this.#numVertices = count;
            // Resizes each of the vertex attributes to match the new count.
            this.__vertexAttributes.forEach((attr) => attr.setCount(this.#numVertices));
            this.setBoundingBoxDirty();
        }
        // ////////////////////////////////////////
        // BoundingBox
        /**
         * Returns the bounding box for geometry.
         * @return - The return value.
         */
        getBoundingBox() {
            if (this.boundingBoxDirty)
                this.updateBoundingBox();
            return this.boundingBox;
        }
        /**
         * The setBoundingBoxDirty method.
         */
        setBoundingBoxDirty() {
            this.boundingBoxDirty = true;
            this.emit('boundingBoxChanged');
        }
        /**
         * The updateBoundingBox method.
         */
        updateBoundingBox() {
            const positions = this.positions;
            const bbox = new Box3();
            if (positions) {
                const numVerts = positions.getCount();
                for (let i = 0; i < numVerts; i++)
                    bbox.addPoint(positions.getValue(i));
            }
            this.boundingBox = bbox;
            this.boundingBoxDirty = false;
        }
        /**
         * Merges a separate geometry into this one. Similar to a 'union' boolean operation.
         * @param other the other geom that will be merged into this one
         * @param xfo the transformation to be applied to the other geom as it is merged in.
         */
        merge(other, xfo = new Xfo()) {
            const prevNumVerts = this.getNumVertices();
            const addedVerts = other.getNumVertices();
            for (const [attrName, attr] of this.__vertexAttributes) {
                const otherAttr = other.getVertexAttribute(attrName);
                if (otherAttr) {
                    if (attrName == 'positions')
                        attr.merge(otherAttr, xfo);
                    else if (attrName == 'normals')
                        attr.merge(otherAttr, new Xfo(new Vec3(), xfo.ori));
                }
            }
            // Note: all the attributes have already been resized, so
            // this is just a final check.
            this.setNumVertices(prevNumVerts + addedVerts);
            this.updateBoundingBox();
        }
        // ////////////////////////////////////////
        // Memory
        /**
         * Returns vertex attributes buffers and its count.
         * @return - The return value.
         */
        genBuffers(opts) {
            const attrBuffers = {};
            for (const [attrName, attr] of this.__vertexAttributes) {
                attrBuffers[attrName] = attr.genBuffer();
            }
            return {
                numVertices: this.numVertices(),
                attrBuffers,
            };
        }
        /**
         * Once the buffers have been uploaded to the GPU, we are free to release them.
         * The GLGeomLibrary may call this function to let the geometry know it can release any handles.
         */
        freeBuffers() { }
        // ////////////////////////////////////////
        // Persistence
        /**
         * Sets state of current Geometry(Including Vertices and Bounding Box) using a binary reader object.
         *
         * @param reader - The reader value.
         */
        loadBaseGeomBinary(reader, context) {
            this.name = reader.loadStr();
            const flags = reader.loadUInt8();
            this.debugColor = reader.loadRGBFloat32Color();
            const numVerts = reader.loadUInt32();
            this.boundingBox.set(reader.loadFloat32Vec3(), reader.loadFloat32Vec3());
            let normalsAttr;
            let texCoordsAttr;
            if (flags & (1 << 1)) {
                normalsAttr = this.getVertexAttribute('normals');
                if (!normalsAttr) {
                    normalsAttr = new Vec3f8Attribute();
                    this.addVertexAttribute('normals', normalsAttr);
                }
            }
            if (flags & (1 << 2)) {
                texCoordsAttr = this.getVertexAttribute('texCoords');
                if (!texCoordsAttr) {
                    texCoordsAttr = new Vec2f16Attribute();
                    this.addVertexAttribute('texCoords', texCoordsAttr);
                }
            }
            const numClusters = reader.loadUInt32();
            if (numClusters == 0) {
                const positionsAttr = this.positions;
                // From 3.12.0, vertex data is a mix of 16bit and 8 bit quanitization
                // without any k-means clustering.
                // @ts-ignore
                positionsAttr.data = reader.loadUInt16Array(numVerts * 3);
                if (normalsAttr) {
                    // @ts-ignore
                    normalsAttr.data = reader.loadInt8Array(numVerts * 3);
                }
                if (texCoordsAttr) {
                    // @ts-ignore
                    texCoordsAttr.data = reader.loadUInt16Array(numVerts * 2);
                }
                // This should be a no-op in each of the attributes as they already have
                // arrays of the appropriate size..
                this.setNumVertices(numVerts);
            }
            else if (numClusters == 1) {
                // Older zcad files store 32 bit floats. The renderer may convert these to 16bit.
                this.addVertexAttribute('positions', new Vec3Attribute());
                this.setNumVertices(numVerts);
                const positionsAttr = this.positions;
                {
                    const box3 = this.boundingBox;
                    // From 3.9.1, vertex data is a mix of 16bit and 8 bit quanitization
                    if (context.versions['zea-engine'].compare([3, 9, 1]) >= 0) {
                        const positions_quantized = reader.loadUInt16Array(numVerts * 3, false);
                        parse16BitPositionsArray([0, numVerts], box3.p0, box3.diagonal(), positions_quantized, positionsAttr);
                    }
                    else {
                        const positions_quantized = reader.loadUInt8Array(numVerts * 3, false);
                        parse8BitPositionsArray([0, numVerts], box3.p0, box3.diagonal(), positions_quantized, positionsAttr);
                    }
                }
                if (normalsAttr) {
                    const box3 = new Box3(reader.loadFloat32Vec3(), reader.loadFloat32Vec3());
                    const normals_quantized = reader.loadUInt8Array(numVerts * 3, false);
                    parse8BitNormalsArray([0, numVerts], box3.p0, box3.diagonal(), normals_quantized, normalsAttr);
                    normalsAttr.loadSplitValues(reader);
                }
                if (texCoordsAttr) {
                    const box2 = new Box2(reader.loadFloat32Vec2(), reader.loadFloat32Vec2());
                    // From 3.9.1, vertex data is a mix of 16bit and 8 bit quanitization
                    if (context.versions['zea-engine'].compare([3, 9, 1]) >= 0) {
                        const texCoords_quantized = reader.loadUInt16Array(numVerts * 2, false);
                        parse16BitTextureCoordsArray([0, numVerts], box2.p0, box2.diagonal(), texCoords_quantized, texCoordsAttr);
                    }
                    else {
                        const texCoords_quantized = reader.loadUInt8Array(numVerts * 2, false);
                        parse8BitTextureCoordsArray([0, numVerts], box2.p0, box2.diagonal(), texCoords_quantized, texCoordsAttr);
                    }
                    texCoordsAttr.loadSplitValues(reader);
                }
            }
            else {
                // Older zcad files store 32 bit floats. The renderer may convert these to 16bit.
                this.addVertexAttribute('positions', new Vec3Attribute());
                this.setNumVertices(numVerts);
                const positionsAttr = this.positions;
                const clusters = [];
                let offset = 0;
                for (let i = 0; i < numClusters; i++) {
                    const count = reader.loadUInt32();
                    const clusterData = {
                        range: [offset, offset + count],
                        bbox: new Box3(reader.loadFloat32Vec3(), reader.loadFloat32Vec3()),
                        normalsRange: new Box3(),
                        texCoordsRange: new Box2(),
                    };
                    if (normalsAttr) {
                        clusterData.normalsRange.set(reader.loadFloat32Vec3(), reader.loadFloat32Vec3());
                    }
                    if (texCoordsAttr) {
                        clusterData.texCoordsRange.set(reader.loadFloat32Vec2(), reader.loadFloat32Vec2());
                    }
                    clusters.push(clusterData);
                    offset += count;
                }
                // From 3.9.1, vertex data is a mix of 16bit and 8 bit quanitization
                let positions_quantized;
                if (context.versions['zea-engine'].compare([3, 9, 1]) >= 0) {
                    positions_quantized = reader.loadUInt16Array(numVerts * 3, false);
                }
                else {
                    positions_quantized = reader.loadUInt8Array(numVerts * 3, false);
                }
                let normals_quantized = null;
                let texCoords_quantized = null;
                if (normalsAttr) {
                    normals_quantized = reader.loadUInt8Array(numVerts * 3, false);
                }
                if (texCoordsAttr) {
                    texCoords_quantized = reader.loadUInt8Array(numVerts * 2, false);
                }
                for (let i = 0; i < numClusters; i++) {
                    {
                        const box3 = clusters[i].bbox;
                        // From 3.9.1, vertex data is a mix of 16bit and 8 bit quanitization
                        if (context.versions['zea-engine'].compare([3, 9, 1]) >= 0) {
                            parse16BitPositionsArray(clusters[i].range, box3.p0, box3.diagonal(), positions_quantized, positionsAttr);
                        }
                        else {
                            parse8BitPositionsArray(clusters[i].range, box3.p0, box3.diagonal(), positions_quantized, positionsAttr);
                        }
                    }
                    if (normals_quantized) {
                        const box3 = clusters[i].normalsRange;
                        parse8BitNormalsArray(clusters[i].range, box3.p0, box3.diagonal(), normals_quantized, normalsAttr);
                    }
                    if (texCoords_quantized) {
                        const box2 = clusters[i].texCoordsRange;
                        if (context.versions['zea-engine'].compare([3, 9, 1]) >= 0) {
                            parse16BitTextureCoordsArray([0, numVerts], box2.p0, box2.diagonal(), texCoords_quantized, texCoordsAttr);
                        }
                        else {
                            parse8BitTextureCoordsArray(clusters[i].range, box2.p0, box2.diagonal(), texCoords_quantized, texCoordsAttr);
                        }
                    }
                }
                if (normalsAttr) {
                    normalsAttr.loadSplitValues(reader);
                }
                if (texCoordsAttr) {
                    texCoordsAttr.loadSplitValues(reader);
                }
            }
            // Loading the attributes may have dirtied the bounding box.
            // we konw we already loaded the bbox, so force it to be valid.
            this.boundingBoxDirty = false;
        }
        /**
         * The toJSON method encodes this type as a json object for persistence.
         *
         * @param context - The context value.
         * @return - Returns the json object.
         */
        toJSON(context) {
            const json = super.toJSON(context);
            if (!context || !context.skipTopology) {
                json.numVertices = this.#numVertices || 0;
            }
            const vertexAttributes = {};
            for (const [key, attr] of this.__vertexAttributes.entries()) {
                if (!context || !('skipAttributes' in context) || !context.skipAttributes.includes(key))
                    vertexAttributes[key] = attr.toJSON(context);
            }
            json.vertexAttributes = vertexAttributes;
            return json;
        }
        /**
         * The fromJSON method decodes a json object for this type.
         *
         * @param json - The json object this item must decode.
         * @param context - The context value.
         */
        fromJSON(json, context) {
            this.clear();
            super.fromJSON(json, context);
            this.setNumVertices(json.numVertices);
            for (const name in json.vertexAttributes) {
                let attr = this.__vertexAttributes.get(name);
                const attrJSON = json.vertexAttributes[name];
                if (!attr || attr.dataTypeName != attrJSON.dataType) {
                    switch (attrJSON.dataType) {
                        case 'Vec2':
                            attr = new Vec2Attribute();
                            break;
                        case 'Vec2f16':
                            attr = new Vec2f16Attribute();
                            break;
                        case 'Vec3':
                            attr = new Vec3Attribute();
                            break;
                        case 'Vec3f16':
                            attr = new Vec3f16Attribute();
                            break;
                        case 'Vec3f8':
                            attr = new Vec3f8Attribute();
                            break;
                        case 'Color':
                            attr = new ColorAttribute();
                            break;
                        default:
                            throw 'Invalid Vectex Type: ' + attr.dataTypeName;
                    }
                    attr.setCount(this.#numVertices);
                    this.addVertexAttribute(name, attr);
                }
                if (attr) {
                    attr.fromJSON(attrJSON);
                }
                else {
                    console.warn('attr undefined, cannot execute fromJSON()');
                }
            }
            this.emit('geomDataTopologyChanged');
        }
        /**
         * Returns geometry data value in json format.
         *
         * @return - The return value.
         */
        toString() {
            return JSON.stringify(this.toJSON(), null, 2);
        }
    }

    /**
     * Class representing a point primitive drawing type, every vertex specified is a point.
     *
     * ```
     * const points = new Points()
     * ```
     *
     * * **Events**
     * * **boundingBoxChanged:** Triggered when the bounding box changes.
     *
     * @extends BaseGeom
     */
    class Points extends BaseGeom {
        /**
         * Create points.
         */
        constructor() {
            super();
        }
        /**
         * The clear method.
         */
        clear() {
            this.setNumVertices(0);
            this.emit('geomDataTopologyChanged');
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * Sets state of current geometry(Including line segments) using a binary reader object.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            super.loadBaseGeomBinary(reader, context);
            // this.computeVertexNormals();
            this.emit('geomDataChanged');
        }
    }
    Registry.register('Points', Points);

    /* eslint-disable @typescript-eslint/no-unused-vars */
    /**
     *
     * Class representing lines primitive drawing type, connecting vertices using the specified indices.
     * i.e. We have 4 points(vertices) but we don't know how they connect to each other,
     * and that's why we need indices(Numbers indicating which vertex connects to which).
     * In this case if we say that `indices` is `[0,1,2,3]`, it would connect the first vertex to the second,
     * and the third to the fourth.
     *
     * ```
     * const lines = new Lines()
     * ```
     *
     * **Events**
     * * **geomDataChanged:** Triggered when the data value of the geometry is set(This includes reading binary)
     *
     * @extends BaseGeom
     */
    class Lines extends BaseGeom {
        __indices;
        /**
         * Create lines.
         */
        constructor() {
            super();
            this.__indices = new Uint32Array();
        }
        /**
         * The clear method.
         */
        clear() {
            this.setNumSegments(0);
            this.setNumVertices(0);
            this.emit('geomDataTopologyChanged');
        }
        /**
         * Returns the specified indices(Vertex connectors)
         *
         * @return - The indices index array.
         */
        getIndices() {
            return this.__indices;
        }
        /**
         * Returns the number of line segments.
         *
         * @return - Returns the number of segments.
         */
        getNumSegments() {
            return this.__indices.length / 2;
        }
        /**
         * Returns the number of line segments.
         *
         * @return - Returns the number of segments.
         */
        getNumLineSegments() {
            return this.__indices.length / 2;
        }
        /**
         * Sets the number of line segments in the lines geometry.
         * **Important:** It resets indices values.
         *
         * @param numOfSegments - The count value.
         */
        setNumSegments(numOfSegments) {
            if (numOfSegments > this.getNumSegments()) {
                const indices = new Uint32Array(numOfSegments * 2);
                indices.set(this.__indices);
                this.__indices = indices;
            }
            else {
                this.__indices = this.__indices.slice(0, numOfSegments * 2);
            }
        }
        /**
         * Sets segment values in the specified index.
         *
         * @param index - The index value.
         * @param p0 - The p0 value.
         * @param p1 - The p1 value.
         */
        setSegmentVertexIndices(index, p0, p1) {
            if (index >= this.__indices.length / 2)
                throw new Error('Invalid line index:' + index + '. Num Segments:' + this.__indices.length / 2);
            this.__indices[index * 2 + 0] = p0;
            this.__indices[index * 2 + 1] = p1;
        }
        /**
         * The getSegmentVertexIndex method.
         *
         * @param line - The line value.
         * @param lineVertex - The lineVertex value.
         * @return - The return value.
         * @private
         */
        getSegmentVertexIndex(line, lineVertex) {
            const numSegments = this.getNumSegments();
            if (line < numSegments)
                return this.__indices[line * 2 + lineVertex];
            return -1;
        }
        /**
         * Merges a separate geometry into this one. Similar to a 'union' boolean operation.
         * @param other the other geom that will be merged into this one
         * @param xfo the transformation to be applied to the other geom as it is merged in.
         */
        merge(other, xfo = new Xfo()) {
            const prevNumVerts = this.getNumVertices();
            super.merge(other, xfo);
            const otheIndices = other.__indices;
            const indices = new Uint32Array(this.__indices.length + otheIndices.length);
            indices.set(this.__indices, 0);
            indices.set(otheIndices.map((index) => index + prevNumVerts), this.__indices.length);
            this.__indices = indices;
        }
        // ////////////////////////////////////////
        // Memory
        /**
         * Returns vertex attributes buffers and its count.
         *
         * @return - The return value.
         */
        genBuffers(opts) {
            const buffers = super.genBuffers();
            let indices;
            if (buffers.numVertices < Math.pow(2, 8)) {
                indices = new Uint8Array(this.__indices);
            }
            else if (buffers.numVertices < Math.pow(2, 16)) {
                indices = new Uint16Array(this.__indices);
            }
            else {
                indices = this.__indices;
            }
            buffers.indices = indices;
            return buffers;
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * Sets state of current geometry(Including line segments) using a binary reader object.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            super.loadBaseGeomBinary(reader, context);
            this.setNumSegments(reader.loadUInt32());
            const bytes = reader.loadUInt8();
            if (bytes == 1)
                this.__indices = reader.loadUInt8Array();
            else if (bytes == 2)
                this.__indices = reader.loadUInt16Array();
            else if (bytes == 4)
                this.__indices = reader.loadUInt32Array();
            this.emit('geomDataChanged');
        }
        /**
         * The toJSON method encodes this type as a json object for persistence.
         *
         * @param context - The context value.
         * @return - Returns the json object.
         */
        toJSON(context) {
            const j = super.toJSON(context);
            if (!context || !context.skipTopology)
                j.indices = Array.from(this.__indices);
            return j;
        }
        /**
         * The fromJSON method decodes a json object for this type.
         *
         * @param j - The json object this item must decode.
         * @param context - The context value.
         */
        fromJSON(j, context) {
            super.fromJSON(j, context);
            if (j.indices)
                this.__indices = Uint32Array.from(j.indices);
        }
    }
    Registry.register('Lines', Lines);

    /* eslint-disable prefer-rest-params */
    /**
     * The Mesh class provides a flexible and fast polygon mesh representation. It supports polygons of arbitrary complexity,
     * from basic triangles and quads to pentagons more.
     * It supports storing per face attributes, and per edge attributes.
     * The Mesh class handles converting its internal representation of polygons into a simpler triangles representation for rendering.
     *
     * ```
     * const mesh = new Mesh()
     * ```
     *
     * **Events**
     * * **geomDataTopologyChanged:** Triggered when the topology of the mesh has been changed.
     * * **geomDataChanged:** Triggered when the vertices of the mesh have changed, but not necessarily the topology.
     *
     * @extends BaseGeom
     */
    class Mesh extends BaseGeom {
        faceCounts;
        faceVertexIndices;
        __logTopologyWarnings;
        __edgeAttributes;
        __faceAttributes;
        numEdges;
        edgeVerts;
        edgeAngles;
        edgeVecs;
        edgeFaces;
        faceEdges;
        vertexEdges;
        /**
         * Creates an instance of Mesh.
         */
        constructor() {
            super();
            this.edgeFaces = [];
            this.faceEdges = [[]];
            this.faceCounts = [];
            this.faceVertexIndices = new Uint32Array();
            this.__logTopologyWarnings = false;
            this.__edgeAttributes = new Map();
            this.__faceAttributes = new Map();
            this.numEdges = 0;
            this.edgeVerts = [];
            this.vertexEdges = [];
            this.edgeAngles = new Float32Array();
            this.edgeVecs = [];
        }
        /**
         * The clear method.
         */
        clear() {
            super.clear();
            //this.init()
            //this.setNumVertices(0)
            // clear edge and face normals.
            this.edgeVerts = [];
            this.vertexEdges = [];
            this.numEdges = 0;
            this.edgeAngles = new Float32Array();
            this.emit('geomDataTopologyChanged');
        }
        /**
         * Adds a new vertex attribute to the geometry.
         *
         * @param name - The name of the vertex attribute.
         * @param attr - The attribute to add to the geometry
         */
        addVertexAttribute(name, attr) {
            super.addVertexAttribute(name, attr);
            attr.setMesh(this);
        }
        /**
         * The getFaceCounts method.
         * @return - The return value.
         */
        getFaceCounts() {
            return this.faceCounts;
        }
        /**
         * The getNumFaces method.
         * @return - The return value.
         */
        getNumFaces() {
            return this.faceCounts.length == 0 ? 0 : this.faceCounts.reduce((numFaces, fc) => (numFaces += fc));
        }
        /**
         * The getNumTriangles method.
         * @return {number} - The return value.
         */
        getNumTriangles() {
            let numTriangles = 0;
            let numTrisPerFace = 1;
            for (const fc of this.faceCounts) {
                numTriangles += fc * numTrisPerFace;
                numTrisPerFace++;
            }
            return numTriangles;
        }
        /**
         * Sets the number of faces on the mesh using an array specifying the counts per polygon size.
         * The first item in the array specifies the number of triangles, the second, the number of quads, the 3rd, the number of 5 sided polygons etc..
         * e.g. to specify 2 triangles, and 7 quads, we would pass [2, 7]
         * @param faceCounts - The faceCounts value.
         */
        setFaceCounts(faceCounts) {
            // let numFaces = 0
            let numFacesVertices = 0;
            let numVertsPerFace = 3;
            for (const fc of faceCounts) {
                // numFaces += fc
                numFacesVertices += fc * numVertsPerFace;
                numVertsPerFace++;
            }
            const prevNumFaces = this.getNumFaces();
            if (prevNumFaces == 0) {
                this.faceVertexIndices = new Uint32Array(numFacesVertices);
            }
            else {
                const faceVertexIndices = new Uint32Array(numFacesVertices);
                // Now we preserve the existing indices if they fit within the new faceVertexIndices array.
                let startSrc = 0;
                let startTgt = 0;
                numFacesVertices = 0;
                numVertsPerFace = 3;
                faceCounts.forEach((fc, index) => {
                    const endSrc = startSrc + Math.min(fc, this.faceCounts[index]) * numVertsPerFace;
                    faceVertexIndices.set(this.faceVertexIndices.slice(startSrc, endSrc), startTgt);
                    startSrc += this.faceCounts[index] * numVertsPerFace;
                    startTgt += fc * numVertsPerFace;
                    numVertsPerFace++;
                });
                this.faceVertexIndices = faceVertexIndices;
            }
            this.faceCounts = faceCounts;
        }
        /**
         * Returns the number of face vertices
         * @return - The return value.
         */
        getNumFaceVertices() {
            let numFaceVerts = 0;
            this.faceCounts.forEach((fc, index) => {
                numFaceVerts += fc * (index + 3);
            });
            return numFaceVerts;
        }
        /**
         * Returns the number of vertices indexed by this face
         * @param faceIndex - The faceIndex value.
         * @return - The return value.
         */
        getFaceVertexCount(faceIndex) {
            let idx = 0;
            let count = 0;
            this.faceCounts.some((fc, index) => {
                idx += fc;
                if (idx > faceIndex) {
                    count = index + 3;
                    return true;
                }
                return false;
            });
            return count;
        }
        getFaceVertexOffset(faceIndex) {
            let idx = 0;
            let offset = 0;
            this.faceCounts.some((fc, index) => {
                if (idx + fc > faceIndex) {
                    offset += (faceIndex - idx) * (index + 3);
                    return true;
                }
                idx += fc;
                offset += fc * (index + 3);
                return false;
            });
            return offset;
        }
        /**
         * The setFaceVertexIndices method.
         * @param faceIndex - The faceIndex value.
         * @param vertexIndices - The array of vertex indices for this face value.
         */
        setFaceVertexIndices(faceIndex, vertexIndices) {
            const faceVertexCount = this.getFaceVertexCount(faceIndex);
            if (vertexIndices.length != faceVertexCount) {
                throw new Error(`Invalid indices for face:${faceIndex} vertexIndices:${vertexIndices}. Expected ${faceVertexCount} indices`);
            }
            const offset = this.getFaceVertexOffset(faceIndex);
            this.faceVertexIndices.set(vertexIndices, offset);
        }
        /**
         * Adds a new face to the mesh
         * @param vertexIndices - The vertex indices of the face.
         * @return - The index of the face in the mesh.
         */
        addFace(vertexIndices) {
            const faceCounts = [...this.faceCounts];
            if (faceCounts.length <= vertexIndices.length - 3) {
                for (let i = faceCounts.length; i < vertexIndices.length - 3; i++)
                    faceCounts[i] = 0;
                faceCounts[vertexIndices.length - 3] = 1;
            }
            else {
                faceCounts[vertexIndices.length - 3]++;
            }
            this.setFaceCounts(faceCounts);
            // Calculate the offset in the faceVertexIndices of this new face.
            let faceIndex = 0;
            let offset = 0;
            this.faceCounts.some((fc, index) => {
                if (index + 3 == vertexIndices.length) {
                    faceIndex += fc - 1;
                    offset += (fc - 1) * (index + 3);
                    return true;
                }
                faceIndex += fc;
                offset += fc * (index + 3);
                return false;
            });
            this.faceVertexIndices.set(vertexIndices, offset);
            return faceIndex;
        }
        /**
         * Returns the vertex indices of the specified face.
         * @param faceIndex - The index of the specified face
         * @return - An array of indices into the vertex attributes
         */
        getFaceVertexIndices(faceIndex) {
            const vertexIndices = [];
            const offset = this.getFaceVertexOffset(faceIndex);
            const count = this.getFaceVertexCount(faceIndex);
            for (let i = 0; i < count; i++) {
                vertexIndices.push(this.faceVertexIndices[offset + i]);
            }
            return vertexIndices;
        }
        /**
         * Returns a single vertex index for a given face and faceVertex.
         * @param faceIndex - The faceIndex value.
         * @param faceVertex - The face vertex is the index within the face. So the first vertex index is 0.
         * @return - The vertex index
         */
        getFaceVertexIndex(faceIndex, faceVertex) {
            const offset = this.getFaceVertexOffset(faceIndex);
            return this.faceVertexIndices[offset + faceVertex];
        }
        // ///////////////////////////
        // Face Attributes
        /**
         * The addFaceAttribute method.
         * @param name - The name of the face attribute to add.
         * @param attr - The attr value
         */
        addFaceAttribute(name, attr) {
            attr.setCount(this.getNumFaces());
            this.__faceAttributes.set(name, attr);
            return attr;
        }
        /**
         * The hasFaceAttribute method.
         * @param name - The name of the face attribute.
         * @return - The return value.
         */
        hasFaceAttribute(name) {
            return this.__faceAttributes.has(name);
        }
        /**
         * The getFaceAttribute method.
         * @param name - The name of the face attribute.
         * @return - The return value.
         */
        getFaceAttribute(name) {
            return this.__faceAttributes.get(name);
        }
        // /////////////////////////
        // Edge Attributes
        /**
         * The addEdgeAttribute method.
         * @param name - The name of the edge attribute to add.
         * @param attr - The attr value
         */
        addEdgeAttribute(name, attr) {
            attr.setCount(this.numEdges);
            this.__edgeAttributes.set(name, attr);
        }
        /**
         * The hasEdgeAttribute method.
         * @param name - The name of the edge attribute.
         * @return - The return value.
         */
        hasEdgeAttribute(name) {
            return this.__edgeAttributes.has(name);
        }
        /**
         * The getEdgeAttribute method.
         * @param name - The name of the edge attribute.
         * @return - The return value.
         */
        getEdgeAttribute(name) {
            return this.__edgeAttributes.get(name);
        }
        // ///////////////////////////
        /**
         * The genTopologyInfo method.
         */
        genTopologyInfo() {
            let connectedVertices = {}; // acceleration structure.
            this.vertexEdges = []; // 2d array of vertex to edges.
            // this.vertexFaces = []; // 2d array of vertex to faces.
            this.edgeFaces = []; // flat array of 2 face indices per edge
            this.edgeVerts = []; // flat array of 2 vert indices per edge
            this.faceEdges = []; // the edges bordering each face.
            this.numEdges = 0;
            const positions = this.positions;
            const getEdgeIndex = (v0, v1) => {
                let tmp0 = v0;
                let tmp1 = v1;
                if (tmp1 < tmp0) {
                    const tmp = tmp0;
                    tmp0 = tmp1;
                    tmp1 = tmp;
                }
                const key = tmp0 + '>' + tmp1;
                if (key in connectedVertices) {
                    // console.log(key + ':' + connectedVertices[key] + " face:" + ( v0 < v1 ? 0 : 1) );
                    return connectedVertices[key];
                }
                const p0 = positions.getValue(tmp0);
                const p1 = positions.getValue(tmp1);
                const edgeVec = p1.subtract(p0);
                const edgeIndex = this.edgeFaces.length / 2;
                const edgeData = {
                    edgeIndex: edgeIndex,
                    edgeVec: edgeVec,
                };
                connectedVertices[key] = edgeData;
                this.edgeFaces.push(-1);
                this.edgeFaces.push(-1);
                this.edgeVerts.push(tmp0);
                this.edgeVerts.push(tmp1);
                // console.log(key + ':' + connectedVertices[key] + " face:" + ( v0 < v1 ? 0 : 1));
                this.numEdges++;
                return edgeData;
            };
            const addEdge = (v0, v1, faceIndex) => {
                // console.log('addEdge:' + v0 + " :" + v1 + " faceIndex:" + faceIndex );
                const edgeData = getEdgeIndex(v0, v1);
                const edgeIndex = edgeData.edgeIndex;
                if (v1 < v0) {
                    const edgeFaceIndex = edgeIndex * 2 + 0;
                    if (this.__logTopologyWarnings && this.edgeFaces[edgeFaceIndex] != -1)
                        console.warn('Edge poly 0 already set. Mesh is non-manifold.');
                    this.edgeFaces[edgeFaceIndex] = faceIndex;
                }
                else {
                    const edgeFaceIndex = edgeIndex * 2 + 1;
                    if (this.__logTopologyWarnings && this.edgeFaces[edgeFaceIndex] != -1)
                        console.warn('Edge poly 1 already set. Mesh is non-manifold.');
                    this.edgeFaces[edgeFaceIndex] = faceIndex;
                }
                if (!(faceIndex in this.faceEdges))
                    this.faceEdges[faceIndex] = [];
                this.faceEdges[faceIndex].push(edgeIndex);
                // Push the edge index onto both vertex edge lists.
                // We use Sets to avoid adding the same edge 2x to the same vertex.
                if (this.vertexEdges[v0] == undefined) {
                    this.vertexEdges[v0] = new Set();
                }
                if (this.vertexEdges[v1] == undefined) {
                    this.vertexEdges[v1] = new Set();
                }
                this.vertexEdges[v0].add(edgeIndex);
                this.vertexEdges[v1].add(edgeIndex);
                // if (this.vertexFaces[v0] == undefined) {
                //     this.vertexFaces[v0] = [];
                // }
                // this.vertexFaces[v0].push(faceIndex);
            };
            const numFaces = this.getNumFaces();
            for (let faceIndex = 0; faceIndex < numFaces; faceIndex++) {
                const faceVerts = this.getFaceVertexIndices(faceIndex);
                for (let j = 0; j < faceVerts.length; j++) {
                    const v0 = faceVerts[j];
                    const v1 = faceVerts[(j + 1) % faceVerts.length];
                    addEdge(v0, v1, faceIndex);
                }
            }
        }
        /**
         * Computes a normal value per face by averaging the triangle normals of the face.
         */
        computeFaceNormals() {
            const positions = this.positions;
            const faceNormals = new Vec3f8Attribute();
            this.addFaceAttribute('normals', faceNormals);
            const numFaces = this.getNumFaces();
            for (let faceIndex = 0; faceIndex < numFaces; faceIndex++) {
                const faceVerts = this.getFaceVertexIndices(faceIndex);
                const p0 = positions.getValue(faceVerts[0]);
                const p1 = positions.getValue(faceVerts[1]);
                let prev = p1;
                const faceNormal = new Vec3();
                for (let j = 2; j < faceVerts.length; j++) {
                    const pn = positions.getValue(faceVerts[j]);
                    const v0 = prev.subtract(p0);
                    const v1 = pn.subtract(p0);
                    faceNormal.addInPlace(v0.cross(v1).normalize());
                    prev = pn;
                }
                if (faceNormal.lengthSquared() < Number.EPSILON) ;
                else {
                    faceNormals.setValue(faceIndex, faceNormal.normalize());
                }
            }
        }
        /**
         * Calculates the angles at each edge between the adjoining faces
         */
        calculateEdgeAngles() {
            if (this.vertexEdges.length == 0)
                this.genTopologyInfo();
            this.computeFaceNormals();
            const positions = this.positions;
            const faceNormals = this.getFaceAttribute('normals');
            this.edgeVecs = [];
            this.edgeAngles = new Float32Array(this.numEdges);
            for (let i = 0; i < this.edgeFaces.length; i += 2) {
                const v0 = this.edgeVerts[i];
                const v1 = this.edgeVerts[i + 1];
                const edgeVec = positions.getValue(v1).subtract(positions.getValue(v0));
                edgeVec.normalizeInPlace();
                this.edgeVecs.push(edgeVec);
                const p0 = this.edgeFaces[i];
                const p1 = this.edgeFaces[i + 1];
                if (p0 == -1 || p1 == -1) {
                    // Flag the edge as a border edge....
                    this.edgeAngles[i / 2] = Math.PI * 2.0;
                    continue;
                }
                const n0 = faceNormals.getValue(p0);
                const n1 = faceNormals.getValue(p1);
                this.edgeAngles[i / 2] = n0.angleTo(n1);
            }
        }
        /**
         * Compute vertex normals.
         * @param hardAngle - The hardAngle value in radians.
         * @return - The return value.
         */
        computeVertexNormals(hardAngle = 1.0 /* radians */) {
            this.calculateEdgeAngles();
            const faceNormals = this.getFaceAttribute('normals');
            const normalsAttr = new Vec3f8Attribute();
            this.addVertexAttribute('normals', normalsAttr);
            // these methods are faster versions than using the methods
            // provided on the attributes. We cache values and use hard coded constants.
            // const faceNormalsBuffer = faceNormals.data.buffer
            const getFaceNormal = (index) => {
                return faceNormals.getValue(index);
            };
            const setVertexNormal = (index, value) => {
                normalsAttr.setValue(index, value);
            };
            const getConnectedEdgeVecs = (faceIndex, vertexIndex) => {
                let e0;
                let e1;
                const faceEdges = this.faceEdges[faceIndex];
                for (const e of faceEdges) {
                    if (this.edgeVerts[e * 2] == vertexIndex) {
                        if (!e0)
                            e0 = this.edgeVecs[e];
                        else
                            e1 = this.edgeVecs[e];
                    }
                    else if (this.edgeVerts[e * 2 + 1] == vertexIndex) {
                        if (!e0)
                            e0 = this.edgeVecs[e];
                        else
                            e1 = this.edgeVecs[e];
                    }
                }
                return [e0, e1];
            };
            for (let i = 0; i < this.vertexEdges.length; i++) {
                // If this face indexing doesn't start at 0, then the vertexEdges don't either.
                if (this.vertexEdges[i] == undefined)
                    continue;
                const edges = this.vertexEdges[i];
                // Groups of faces having a smooth normal at the current vertex.
                const faceGroups = [];
                const addFaceToGroup = (face) => {
                    let inGroup = false;
                    for (const faceGroup of faceGroups) {
                        inGroup = faceGroup.includes(face);
                        if (inGroup)
                            break;
                    }
                    if (!inGroup)
                        faceGroups.push([face]);
                };
                for (const e of edges) {
                    const f0 = this.edgeFaces[e * 2];
                    const f1 = this.edgeFaces[e * 2 + 1];
                    if (f0 != -1 && f1 != -1 && this.edgeAngles[e] < hardAngle) {
                        // if (f0 != -1 && f1 == -1 && this.edgeAngles[e] < hardAngle) {
                        let f0groupIndex = -1;
                        let f1groupIndex = -1;
                        for (let groupIndex = 0; groupIndex < faceGroups.length; groupIndex++) {
                            if (f0groupIndex == -1 && faceGroups[groupIndex].includes(f0))
                                f0groupIndex = groupIndex;
                            if (f1groupIndex == -1 && faceGroups[groupIndex].includes(f1))
                                f1groupIndex = groupIndex;
                        }
                        if (f0groupIndex == -1 && f1groupIndex == -1) {
                            faceGroups.push([f0, f1]);
                        }
                        else if (f0groupIndex != -1 && f1groupIndex != -1) {
                            if (f0groupIndex != f1groupIndex) {
                                // Merge the 2 groups that the smooth edge joins.
                                faceGroups[f0groupIndex] = faceGroups[f0groupIndex].concat(faceGroups[f1groupIndex]);
                                faceGroups.splice(f1groupIndex, 1);
                            }
                        }
                        else {
                            if (f0groupIndex == -1) {
                                faceGroups[f1groupIndex].push(f0);
                            }
                            if (f1groupIndex == -1) {
                                faceGroups[f0groupIndex].push(f1);
                            }
                        }
                        continue;
                    }
                    // This is a hard edge or a border edge... Add faces separately group.
                    if (f0 != -1)
                        addFaceToGroup(f0);
                    if (f1 != -1)
                        addFaceToGroup(f1);
                }
                // Sort the groups to have the biggest group first.
                faceGroups.sort((a, b) => (a.length < b.length ? 1 : a.length > b.length ? -1 : 0));
                let firstVertex = true;
                for (const faceGroup of faceGroups) {
                    const normal = new Vec3();
                    for (const faceIndex of faceGroup) {
                        const faceEdges = getConnectedEdgeVecs(faceIndex, i);
                        let weight;
                        if (faceEdges[0] && faceEdges[1]) {
                            weight = faceEdges[0].angleTo(faceEdges[1]);
                            normal.addInPlace(getFaceNormal(faceIndex).scale(weight));
                        }
                        else {
                            console.warn('variable weight is undefined because faceEdges[0] or faceEdges[1] is undefined');
                        }
                        // if (i == 1)
                        //     console.log("FaceNormal:" + faceIndex + ":" + getFaceNormal(faceIndex).toString());
                    }
                    normal.normalizeInPlace();
                    if (firstVertex) {
                        setVertexNormal(i, normal);
                        firstVertex = false;
                    }
                    else {
                        normalsAttr.setSplitVertexValues(i, faceGroup, normal.asArray());
                    }
                }
            }
            return normalsAttr;
        }
        /**
         * The computeHardEdgesIndices method.
         * @param hardAngle - The hardAngle value in radians.
         * @return - The return value.
         */
        computeHardEdgesIndices(hardAngle = 1.0) {
            if (this.edgeVerts.length == 0)
                this.calculateEdgeAngles();
            const hardEdges = [];
            const addEdge = (index) => {
                hardEdges.push(this.edgeVerts[index]);
                hardEdges.push(this.edgeVerts[index + 1]);
            };
            for (let i = 0; i < this.edgeAngles.length; i++) {
                if (this.edgeAngles[i] > hardAngle) {
                    addEdge(i * 2);
                }
            }
            return Uint32Array.from(hardEdges);
        }
        /**
         * Merges a separate geometry into this one. Similar to a 'union' boolean operation.
         * @param other the other geom that will be merged into this one
         * @param xfo the transformation to be applied to the other geom as it is merged in.
         */
        merge(other, xfo = new Xfo()) {
            const prevNumVerts = this.getNumVertices();
            super.merge(other, xfo);
            const otherFaceVertexIndices = other.faceVertexIndices;
            const faceVertexIndices = new Uint32Array(this.faceVertexIndices.length + otherFaceVertexIndices.length);
            const otherFaceCounts = other.getFaceCounts();
            let indexOffset = 0;
            let otherIndexOffset = 0;
            let mergedIndexOffset = 0;
            const numCounts = Math.max(this.faceCounts.length, otherFaceCounts.length);
            for (let i = 0; i < numCounts; i++) {
                if (this.faceCounts.length > i) {
                    // Add 'this' indices
                    const numIndicesThis = this.faceCounts[i] * (i + 3);
                    faceVertexIndices.set(this.faceVertexIndices.slice(indexOffset, indexOffset + numIndicesThis), mergedIndexOffset);
                    indexOffset += numIndicesThis;
                    mergedIndexOffset += numIndicesThis;
                }
                if (otherFaceCounts.length > i) {
                    // Add the 'other' indices
                    const numIndicesOther = otherFaceCounts[i] * (i + 3);
                    faceVertexIndices.set(otherFaceVertexIndices
                        .slice(otherIndexOffset, otherIndexOffset + numIndicesOther)
                        .map((index) => index + prevNumVerts), mergedIndexOffset);
                    otherIndexOffset += numIndicesOther;
                    mergedIndexOffset += numIndicesOther;
                    if (this.faceCounts.length == i)
                        this.faceCounts[i] = 0;
                    this.faceCounts[i] += otherFaceCounts[i];
                }
            }
            this.faceVertexIndices = faceVertexIndices;
            // Note: the merge does not correctly merge split values, as it is quite complex
            // and we don't have time now. We can come back to this, but the splits system is overly complex
            // and we can probably re-write.
            // for (const [attrName, attr] of this.__vertexAttributes) {
            //   const otherAttr = other.getVertexAttribute(attrName)
            //   if (otherAttr) {
            //     const thisSplits = attr.getSplits()
            //     const otherSplits = otherAttr.getSplits()
            //     // for (let key in otherSplits) {
            //     //   thisSplits[key] = otherSplits[key]
            //     // }
            //   }
            // }
        }
        // ////////////////////////////////////////
        // Rendering
        /**
         * The genBuffers method.
         * @param opts - The opts value.
         * @return - The return value.
         */
        genBuffers(opts) {
            // Compute the normals on demand.
            // if (!('normals' in this.__vertexAttributes)) {
            //     // this.geom.computeVertexNormals();
            //     this.addVertexAttribute("normals", Vec3, 0.0);
            // }
            const splitIndices = {};
            let splitCount = 0;
            for (const [, attr] of this.__vertexAttributes) {
                const attrSplits = attr.getSplits();
                for (const polygon in attrSplits) {
                    if (!(polygon in splitIndices))
                        splitIndices[polygon] = {};
                    const vertices = attrSplits[polygon];
                    for (const v in vertices) {
                        const vertex = parseInt(v);
                        if (!(vertex in splitIndices[polygon])) {
                            splitIndices[polygon][vertex] = splitCount;
                            splitCount++;
                        }
                    }
                }
            }
            const positions = this.positions;
            const numUnSplitVertices = positions.getCount();
            const totalNumVertices = numUnSplitVertices + splitCount;
            let indices;
            if (!opts || opts.includeIndices != false) {
                indices = this.generateTriangulatedIndices(totalNumVertices, numUnSplitVertices, splitIndices);
            }
            // let maxIndex;
            // if (debugAttrValues)
            //     maxIndex = Math.max(...indices);
            const attrBuffers = {};
            for (const [attrName, attr] of this.__vertexAttributes) {
                let values;
                if (splitCount == 0)
                    values = attr.asArray();
                else
                    values = attr.generateSplitValues(splitIndices, splitCount);
                const dimension = attr.stride;
                const count = values.length / dimension;
                // if (debugAttrValues) {
                //     if (count <= maxIndex)
                //         console.warn("Invalid indexing. Attr value is insufficient for indexing:" + attrName + ". Max Index:" + maxIndex + " Attr Size:" + count);
                // }
                attrBuffers[attrName] = {
                    values: values,
                    count: count,
                    dimension: dimension,
                    normalized: attrName == 'normals',
                    dataType: attr.getDataTypeName(),
                };
            }
            const result = {
                numVertices: this.numVertices(),
                numRenderVerts: totalNumVertices,
                indices,
                attrBuffers,
            };
            /* Disabled during TS migration.
            if (opts && opts.includeVertexNeighbors) {
              if (this.vertexEdges == undefined) this.genTopologyInfo()
        
              let count = 0
              for (let i = 0; i < this.vertexEdges.length; i++) {
                // If this face indexing doesn't start at 0, then the vertexEdges don't either.
                if (this.vertexEdges[i]) count += this.vertexEdges[i].size
              }
              // The array will be structured as a start+offset for each vertex, followed
              // by a 2d array of neighbor indices.
              const vertexNeighbors = new Uint32Array(this.vertexEdges.length * 2 + count)
              const sortFanEdges = (fanEdges: any) => {
                for (let i = 0; i < fanEdges.length; i++) {
                  const feA = fanEdges[i]
                  for (let j = 0; j < i; j++) {
                    const feB = fanEdges[j]
                    if (feA[0] != -1 && feA[0] == feB[1]) {
                      //  move feA after feB;
                      if (i != j + 1) {
                        fanEdges.splice(i, 1)
                        fanEdges.splice(j + 1, 0, feA)
                      }
                      break
                    }
                    if (feA[1] != -1 && feA[1] == feB[0]) {
                      //  move feA before feB;
                      fanEdges.splice(i, 1)
                      fanEdges.splice(j, 0, feA)
                      break
                    }
                  }
                }
              }
              const checkFanEdges = (fanEdges: any) => {
                // now check that the faces all build a fan. Maybe starting and ending with -1
                if (fanEdges[0][0] == -1 || fanEdges[fanEdges.length - 1][1] == -1) {
                  if (fanEdges[0][0] != -1 || fanEdges[fanEdges.length - 1][1] != -1) {
                    throw new Error('If fan starts with -1, it must also end with -1')
                  }
                }
                for (let i = 0; i < fanEdges.length; i++) {
                  const fe = fanEdges[i]
                  if (fe[0] == -1 || fe[1] == -1) {
                    if (i != 0 && i != fanEdges.length - 1) {
                      throw new Error('-1 only allowed at the beginning and end of a fan.')
                    }
                  }
                  if (fe[0] != -1) {
                    let prev = i - 1
                    if (prev < 0) prev += fanEdges.length
                    if (fe[0] != fanEdges[prev][1]) {
                      throw new Error('Faces are not sequential')
                    }
                  }
                  if (fe[1] != -1) {
                    const next = (i + 1) % fanEdges.length
                    if (fe[1] != fanEdges[next][0]) {
                      throw new Error('Faces are not sequential')
                    }
                  }
                }
              }
        
              // Populate the start and offset values.
              let offset = this.vertexEdges.length * 2
              for (let i = 0; i < this.vertexEdges.length; i++) {
                if (this.vertexEdges[i] == undefined) continue
                const edges = this.vertexEdges[i]
        
                // Build a sorted list of faces based on a fan around
                // the vertex.
                const fanEdges = []
                for (const e of edges) {
                  const v0 = this.edgeVerts[e * 2]
                  const v1 = this.edgeVerts[e * 2 + 1]
                  let f0 = this.edgeFaces[e * 2]
                  let f1 = this.edgeFaces[e * 2 + 1]
                  let neigVert
                  if (v0 == i) {
                    neigVert = v1
                  } else if (v1 == i) {
                    neigVert = v0
                    // swap the faces
                    const tmp = f0
                    f0 = f1
                    f1 = tmp
                  } else {
                    throw new Error('Invalid topology')
                  }
                  fanEdges.push([f0, f1, neigVert])
                }
                sortFanEdges(fanEdges)
                checkFanEdges(fanEdges)
                const closed = fanEdges[0][0] != -1 || fanEdges[fanEdges.length - 1][1] != -1
                let flags = 0
                if (closed) flags += 1
                vertexNeighbors[i * 2] = offset
                vertexNeighbors[i * 2 + 1] = edges.size + (flags << 8)
                for (const fe of fanEdges) {
                  vertexNeighbors[offset] = fe[2]
                  offset++
                }
              }
              ;(result as any).vertexNeighbors = vertexNeighbors
            }
            */
            return result;
        }
        /**
         * Compute the number of triangles. For higher degree polygons, they are divided into multiple triangles for rendering.
         * @return - Returns the number of triangles.
         */
        computeNumTriangles() {
            let numVertsPerFace = 3;
            let trisCount = 0;
            for (const fc of this.faceCounts) {
                trisCount += fc * (numVertsPerFace - 2);
                numVertsPerFace++;
            }
            return trisCount;
        }
        /**
         * To prepare data for rendering, the indices for the polygons is used to compute a new index buffer based on
         * only triangles. This is used during rendering and the resulting indices uploaded ot the GPU  by GLMesh class.
         *
         * @param totalNumVertices - The total number of vertices.
         * @param numUnSplitVertices - The total number of un-split vertices.
         * @param splitIndices - The splitIndices value.
         * @return - Returns a typed array containing the triangulated indices.
         */
        generateTriangulatedIndices(totalNumVertices, numUnSplitVertices, splitIndices) {
            const trisCount = this.computeNumTriangles();
            let triangulatedIndices;
            if (totalNumVertices < Math.pow(2, 8))
                triangulatedIndices = new Uint8Array(trisCount * 3);
            else if (totalNumVertices < Math.pow(2, 16))
                triangulatedIndices = new Uint16Array(trisCount * 3);
            else
                triangulatedIndices = new Uint32Array(trisCount * 3);
            let triangleVertex = 0;
            const addTriangleVertexIndex = function (vertex, faceIndex) {
                if (vertex in splitIndices && faceIndex in splitIndices[vertex])
                    vertex = numUnSplitVertices + splitIndices[vertex][faceIndex];
                triangulatedIndices[triangleVertex] = vertex;
                triangleVertex++;
            };
            const numFaces = this.getNumFaces();
            for (let faceIndex = 0; faceIndex < numFaces; faceIndex++) {
                const faceVerts = this.getFaceVertexIndices(faceIndex);
                for (let j = 0; j < faceVerts.length; j++) {
                    if (j >= 3) {
                        // For each additional triangle, we have to add 2 indices.
                        addTriangleVertexIndex(faceVerts[0], faceIndex);
                        addTriangleVertexIndex(faceVerts[j - 1], faceIndex);
                    }
                    addTriangleVertexIndex(faceVerts[j], faceIndex);
                }
            }
            return triangulatedIndices;
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * Restores mesh properties from a binary reader.
         *
         * @param reader - The reader value.
         * @param context - The context value.
         */
        readBinary(reader, context) {
            super.loadBaseGeomBinary(reader, context);
            this.setFaceCounts(Array.from(reader.loadUInt32Array()));
            const numFaces = this.getNumFaces();
            // Note: we can remove this. We can infer this from the above faceCounts array.
            // Do not clone the data, as its 'scratch memory' in any case.
            // We can avoid a lot of unnecessary temporary allocaiton by using shared buffers.
            const faceVertexCounts = reader.loadUInt8Array(numFaces, false);
            const offsetRange = reader.loadSInt32Vec2();
            const bytes = reader.loadUInt8();
            let faceVertexIndexDeltas;
            if (bytes == 1)
                faceVertexIndexDeltas = reader.loadUInt8Array(undefined, false);
            else if (bytes == 2)
                faceVertexIndexDeltas = reader.loadUInt16Array(undefined, false);
            else if (bytes == 4)
                faceVertexIndexDeltas = reader.loadUInt32Array(undefined, false);
            else {
                throw Error('faceVertexIndexDeltas undefined');
            }
            // ///////////////////////////////////////////////////
            // Note: The Mesh compression system needs a thorough review.
            // The C++ classes are not storing face indices in a sorted manner.
            // So quads precede triangles in the indexing, which isn't supposed to happen.
            // We should force the C++ code to store quads and triangles in order.
            // e.g. implement the 'addFace' method in C++ so it automatically does this.
            let numFaceVerts = 3;
            let offset = 0;
            const faceOffsetsByCount = this.faceCounts.map((fc, index) => {
                const result = offset;
                offset += fc * numFaceVerts;
                numFaceVerts++;
                return result;
            });
            let srcOffset = 0;
            let prevCount = 0;
            const faceOffsets = [];
            for (let faceIndex = 0; faceIndex < numFaces; faceIndex++) {
                const fc = faceVertexCounts[faceIndex];
                const offset = faceOffsetsByCount[fc];
                const count = fc + 3;
                faceOffsets[faceIndex] = offset;
                for (let j = 0; j < count; j++) {
                    const srcFaceVertex = srcOffset + j;
                    const faceVertex = offset + j;
                    const delta = faceVertexIndexDeltas[srcFaceVertex] + offsetRange.x;
                    if (faceIndex == 0)
                        this.faceVertexIndices[faceVertex] = delta;
                    else {
                        let prevFaceVertex = faceOffsets[faceIndex - 1];
                        prevFaceVertex += j < prevCount ? j : prevCount - 1;
                        this.faceVertexIndices[faceVertex] = this.faceVertexIndices[prevFaceVertex] + delta;
                    }
                }
                srcOffset += count;
                faceOffsetsByCount[fc] += count;
                prevCount = count;
            }
            if (!this.hasVertexAttribute('normals')) {
                this.computeVertexNormals();
            }
            // this.computeVertexNormals();
            this.emit('geomDataChanged');
        }
        /**
         * The toJSON method encodes this type as a json object for persistence.
         *
         * @param context - The context value.
         * @return - Returns the json object.
         */
        toJSON(context) {
            const j = super.toJSON(context);
            if (!context || !context.skipTopology) {
                j.faceCounts = Array.from(this.faceCounts);
                j.faceVertexIndices = Array.from(this.faceVertexIndices);
            }
            return j;
        }
        /**
         * The fromJSON method decodes a json object for this type.
         *
         * e.g. to load data into the mesh class, provide a json structure similar to the following.
         * Note: faceCounts is an array of count values, starting with the number of triangles, then the number of quads. See #setFaceCounts
         * The faceVertexIndices array should also be sorted to contain all the triangles first, followed by the quads, and then the pentagons etc..
         * ```json
         * // This code will define a mesh made up of 2 triangles and then a quad.
         * const mesh = new Mesh()
         * mesh.fromJSON({
         *   faceCounts:[2, 1],
         *   faceVertexIndices: [0, 1, 2, 0, 2, 3, 3, 2, 4, 5],
         *   numVertices: 6,
         *   vertexAttributes: {
         *     positions: {
         *       dataType: 'Vec3'
         *       defaultScalarValue: 0.0,
         *       data: [0,0,0, 0, 1, 0, 1, 1, 0, 1, 0, 0, 2, 1, 0, 2, 0, 0]
         *     }
         *   }
         * }
         * ```
         *
         * @param j - The json object this item must decode.
         * @param context - The context value.
         */
        fromJSON(j, context) {
            super.fromJSON(j, context);
            if (j.faceCounts)
                this.faceCounts = j.faceCounts;
            if (j.faceVertexIndices)
                this.faceVertexIndices = Uint32Array.from(j.faceVertexIndices);
        }
    }
    Registry.register('Mesh', Mesh);

    /* eslint-disable no-unused-vars */
    /**
     * Reads binary data in a specific encoding. Used in loading binary data such as zcad files.
     */
    class BinReader {
        __data;
        __byteOffset;
        __dataView;
        __isMobileDevice;
        utf8decoder;
        /**
         * Create a bin reader.
         *
         * @param data - The data buffer.
         * @param byteOffset - The byte offset value to start reading the buffer.
         * @param isMobileDevice - The isMobileDevice value.
         */
        constructor(data, byteOffset = 0, isMobileDevice = true) {
            this.__data = data;
            this.__byteOffset = byteOffset;
            this.__dataView = new DataView(this.__data);
            this.__isMobileDevice = isMobileDevice;
            this.utf8decoder = new TextDecoder();
        }
        /**
         * Returns state of whether or not the `BinReader` object was instantiated from a mobile device.
         *
         * @return - Returns true is a mobile device is detected.
         */
        get isMobileDevice() {
            return this.__isMobileDevice;
        }
        /**
         * Returns the data buffer we're reading from.
         *
         * @return - The data buffer we are reading from,
         */
        get data() {
            return this.__data;
        }
        /**
         * Returns the length of the buffer.
         *
         * @return - The total length of the buffer
         */
        get byteLength() {
            return this.__dataView.byteLength;
        }
        /**
         * Returns remaining length of the buffer to read.
         *
         * @return - The remaining length of the buffer to read.
         */
        get remainingByteLength() {
            return this.__dataView.byteLength - this.__byteOffset;
        }
        /**
         * Returns current byte offset in the buffer.
         * @return - The current offset in the binary buffer
         */
        pos() {
            return this.__byteOffset;
        }
        /**
         * Sets the byte offset value.
         * @param byteOffset - The byteOffset param.
         */
        seek(byteOffset) {
            this.__byteOffset = byteOffset;
        }
        /**
         * Adds offset bytes to current offset value.
         *
         * @param byteOffset - The byte Offset amount.
         */
        advance(byteOffset) {
            this.__byteOffset += byteOffset;
        }
        /**
         * Returns the unsigned Uint8 value at current byte offset position,
         * and adds one byte to the offset.
         *
         * @return - The return value.
         */
        loadUInt8() {
            const result = this.__dataView.getUint8(this.__byteOffset);
            this.__byteOffset += 1;
            return result;
        }
        /**
         * Returns the unsigned Uint16 value at current byte offset position,
         * and adds two bytes to the offset.
         *
         * @return - The return value.
         */
        loadUInt16() {
            const result = this.__dataView.getUint16(this.__byteOffset, true);
            this.__byteOffset += 2;
            return result;
        }
        /**
         * Returns the unsigned Uint32 value at current byte offset position,
         * and adds four bytes to the offset.
         *
         * @return - The return value.
         */
        loadUInt32() {
            const result = this.__dataView.getUint32(this.__byteOffset, true);
            this.__byteOffset += 4;
            return result;
        }
        /**
         * Returns the signed Int32 value at current byte offset position,
         * and adds four bytes to the offset.
         *
         * @return - The return value.
         */
        loadSInt32() {
            const result = this.__dataView.getInt32(this.__byteOffset, true);
            this.__byteOffset += 4;
            return result;
        }
        /**
         * Returns the Float16 value at current byte offset position,
         * and adds four bytes to the offset.
         *
         * @return - The return value.
         */
        loadFloat16() {
            const uint16 = this.loadUInt16();
            return MathFunctions.decode16BitFloat(uint16);
        }
        /**
         * Returns the Float16 value at current byte offset position,
         * and adds two bytes to the offset.
         *
         * @return - The return value.
         */
        loadUFloat16() {
            const result = this.loadFloat16();
            if (result < 0.0) {
                return 2048.0 - result; // Note: subtract a negative number to add it.
            }
            else {
                return result;
            }
        }
        /**
         * Returns a single signed Float16 value at current byte offset position from 2 unsigned Int8 values,
         * and adds two bytes to the offset.
         *
         * @return - The return value.
         */
        loadFloat16From2xUInt8() {
            throw Error('loadFloat16From2xUInt8 not implemented!');
            // const result = this.__dataView.getFloat16(this.__byteOffset, true)
            // const uint8s = this.loadUInt8Array(2);
            // return Math.decode16BitFloat(uint8s);
            // this.__byteOffset += 2
            // return result
        }
        /**
         * Loads and returns a single Signed integer value from 2 Unsigned Float16 values.
         * @return - The return value.
         */
        loadUInt32From2xUFloat16() {
            const partA = this.loadUFloat16();
            const partB = this.loadUFloat16();
            return partA + partB * 4096;
        }
        /**
         * Loads and returns a single Signed integer value from 2 signed Float16 values.
         * @return - The return value.
         */
        loadSInt32From2xFloat16() {
            const partA = this.loadFloat16();
            const partB = this.loadFloat16();
            return partA + partB * 2048;
        }
        /**
         * Returns the Float32 value at current byte offset position,
         * and adds four bytes to the offset.
         *
         * @return - The return value.
         */
        loadFloat32() {
            const result = this.__dataView.getFloat32(this.__byteOffset, true);
            this.__byteOffset += 4;
            return result;
        }
        /**
         * Reads buffer and return a signed Int8 array with the specified size,
         * starting from current byte offset.
         * Byte offset is increased by the specified byte size.
         *
         * @param size - The size param.
         * @param clone - The clone param.
         * @return - The return value.
         */
        loadInt8Array(size, clone = true) {
            if (size == undefined)
                size = this.loadUInt32();
            let result;
            if (clone) {
                result = new Int8Array(this.__data.slice(this.__byteOffset, this.__byteOffset + size));
                if (result.length != size)
                    console.log('broken');
            }
            else {
                result = new Int8Array(this.__data, this.__byteOffset, size);
            }
            this.__byteOffset += size;
            return result;
        }
        /**
         * Reads buffer and return an unsigned Int8 array with the specified size,
         * starting from current byte offset.
         * Byte offset is increased by the specified byte size.
         *
         * @param size - The size param.
         * @param clone - The clone param.
         * @return - The return value.
         */
        loadUInt8Array(size, clone = true) {
            if (size == undefined)
                size = this.loadUInt32();
            let result;
            if (clone) {
                result = new Uint8Array(this.__data.slice(this.__byteOffset, this.__byteOffset + size));
                if (result.length != size)
                    console.log('broken');
            }
            else {
                result = new Uint8Array(this.__data, this.__byteOffset, size);
            }
            this.__byteOffset += size;
            return result;
        }
        /**
         * Reads buffer and return an unsigned Int16 array with the specified size,
         * starting from current byte offset.
         * Byte offset is increased by the specified byte size x 2.
         *
         * @param size - The size param.
         * @param clone - The clone param.
         * @return - The return value.
         */
        loadUInt16Array(size, clone = true) {
            if (size == undefined)
                size = this.loadUInt32();
            if (size == 0)
                return new Uint16Array();
            this.readPad(2);
            let result;
            if (this.__isMobileDevice) {
                result = new Uint16Array(size);
                for (let i = 0; i < size; i++) {
                    result[i] = this.__dataView.getUint16(this.__byteOffset, true);
                    this.__byteOffset += 2;
                }
            }
            else {
                if (clone) {
                    result = new Uint16Array(this.__data.slice(this.__byteOffset, this.__byteOffset + size * 2));
                    if (result.length != size)
                        console.log('broken');
                }
                else {
                    result = new Uint16Array(this.__data, this.__byteOffset, size);
                }
                this.__byteOffset += size * 2;
            }
            return result;
        }
        /**
         * Reads buffer and return an unsigned Int32 array with the specified size,
         * starting from current byte offset.
         * Byte offset is increased by the specified byte size x 4.
         *
         * @param size - The size param.
         * @param clone - The clone param.
         * @return - The return value.
         */
        loadUInt32Array(size, clone = true) {
            if (size == undefined)
                size = this.loadUInt32();
            if (size == 0)
                return new Uint32Array();
            this.readPad(4);
            let result;
            if (this.__isMobileDevice) {
                result = new Uint32Array(size);
                for (let i = 0; i < size; i++) {
                    result[i] = this.__dataView.getUint32(this.__byteOffset, true);
                    this.__byteOffset += 4;
                }
            }
            else {
                if (clone) {
                    result = new Uint32Array(this.__data.slice(this.__byteOffset, this.__byteOffset + size * 4));
                    if (result.length != size)
                        console.log('broken');
                }
                else {
                    result = new Uint32Array(this.__data, this.__byteOffset, size);
                }
                this.__byteOffset += size * 4;
            }
            return result;
        }
        /**
         * Reads buffer and return a Float32 array with the specified size,
         * starting from current byte offset.
         * Byte offset is increased by the specified byte size x 4.
         *
         * @param size - The size param.
         * @param clone - The clone param.
         * @return - The return value.
         */
        loadFloat32Array(size, clone = true) {
            if (size == undefined)
                size = this.loadUInt32();
            if (size == 0)
                return new Float32Array();
            this.readPad(4);
            let result;
            if (this.__isMobileDevice) {
                result = new Float32Array(size);
                for (let i = 0; i < size; i++) {
                    result[i] = this.__dataView.getFloat32(this.__byteOffset, true);
                    this.__byteOffset += 4;
                }
            }
            else {
                if (clone) {
                    result = new Float32Array(this.__data.slice(this.__byteOffset, this.__byteOffset + size * 4));
                }
                else {
                    result = new Float32Array(this.__data, this.__byteOffset, size);
                }
                this.__byteOffset += size * 4;
            }
            return result;
        }
        /**
         * Returns next string.
         * First looks for the string length description in the next four bytes in the buffer(Starting from byte offset).
         *
         * @return - The return value.
         */
        loadStr() {
            const numChars = this.loadUInt32();
            const chars = new Uint8Array(this.__data, this.__byteOffset, numChars);
            this.__byteOffset += numChars;
            return this.utf8decoder.decode(chars);
        }
        /**
         * Returns an array of strings.
         * First reading the size of the array then reading each string.
         *
         * @return - The return value.
         */
        loadStrArray() {
            const size = this.loadUInt32();
            const result = [];
            for (let i = 0; i < size; i++) {
                result[i] = this.loadStr();
            }
            return result;
        }
        /**
         * Creates and returns a `Vec2` object with the next two signed Int32 values in the buffer.
         *
         * @return - Returns a Vec2.
         */
        loadSInt32Vec2() {
            const x = this.loadSInt32();
            const y = this.loadSInt32();
            return new Vec2(x, y);
        }
        /**
         * Creates and returns a `Vec2` object with the next two unsigned Int32 values in the buffer.
         * @return - Returns a Vec2.
         */
        loadUInt32Vec2() {
            const x = this.loadUInt32();
            const y = this.loadUInt32();
            return new Vec2(x, y);
        }
        /**
         * Creates and returns a `Vec2` object with the next two Float16 values in the buffer.
         *
         * @return - Returns a Vec2.
         */
        loadFloat16Vec2() {
            const x = this.loadFloat16();
            const y = this.loadFloat16();
            return new Vec2(x, y);
        }
        /**
         * Creates and returns a `Vec2` object with the next two Float32 values in the buffer.
         * @return - Returns a Vec2.
         */
        loadFloat32Vec2() {
            const x = this.loadFloat32();
            const y = this.loadFloat32();
            return new Vec2(x, y);
        }
        /**
         * Creates and returns a `Vec3` object with the next three Float16 values in the buffer.
         *
         * @return - Returns a Vec3.
         */
        loadFloat16Vec3() {
            const x = this.loadFloat16();
            const y = this.loadFloat16();
            const z = this.loadFloat16();
            return new Vec3(x, y, z);
        }
        /**
         * Creates and returns a `Vec3` object with the next three Float32 values in the buffer.
         *
         * @return - Returns a Vec3.
         */
        loadFloat32Vec3() {
            const x = this.loadFloat32();
            const y = this.loadFloat32();
            const z = this.loadFloat32();
            return new Vec3(x, y, z);
        }
        /**
         * Creates and returns a `Quat` object with the next four Float16 values in the buffer.
         *
         * @return - Returns a Quat.
         */
        loadFloat16Quat() {
            const x = this.loadFloat16();
            const y = this.loadFloat16();
            const z = this.loadFloat16();
            const w = this.loadFloat16();
            return new Quat(x, y, z, w);
        }
        /**
         * Creates and returns a `Quat` object with the next four Float32 values in the buffer.
         * @return - Returns a Quat.
         */
        loadFloat32Quat() {
            const x = this.loadFloat32();
            const y = this.loadFloat32();
            const z = this.loadFloat32();
            const w = this.loadFloat32();
            return new Quat(x, y, z, w);
        }
        /**
         * Creates and returns a `Color` object with the next three Float32 values in the buffer.
         *
         * @return - Returns a Color.
         */
        loadRGBFloat32Color() {
            const r = this.loadFloat32();
            const g = this.loadFloat32();
            const b = this.loadFloat32();
            return new Color(r, g, b);
        }
        /**
         * Creates and returns a RGBA `Color` object with the next four Float32 values in the buffer.
         * @return - Returns a Color.
         */
        loadRGBAFloat32Color() {
            const r = this.loadFloat32();
            const g = this.loadFloat32();
            const b = this.loadFloat32();
            const a = this.loadFloat32();
            return new Color(r, g, b, a);
        }
        /**
         * Creates and returns a `Color` object with the next three unsigned Int8 values in the buffer.
         * @return - Returns a Color.
         */
        loadRGBUInt8Color() {
            const r = this.loadUInt8();
            const g = this.loadUInt8();
            const b = this.loadUInt8();
            return new Color(r / 255, g / 255, b / 255);
        }
        /**
         * Creates and returns a RGBA `Color` object with the next four unsigned Int8 values in the buffer.
         * @return - Returns a Color.
         */
        loadRGBAUInt8Color() {
            const r = this.loadUInt8();
            const g = this.loadUInt8();
            const b = this.loadUInt8();
            const a = this.loadUInt8();
            return new Color(r / 255, g / 255, b / 255, a / 255);
        }
        /**
         * Creates and returns a `Box2` object with the next four Float32 values in the buffer.
         * Next four because it creates two Vec2.
         *
         * @return - Returns a Box2.
         */
        loadBox2() {
            return new Box2(this.loadFloat32Vec2(), this.loadFloat32Vec2());
        }
        /**
         * Creates and returns a `Box2` object with the next six Float32 values in the buffer.
         * Next four because it creates two Vec3.
         *
         * @return - Returns a Box3.
         */
        loadBox3() {
            return new Box3(this.loadFloat32Vec3(), this.loadFloat32Vec3());
        }
        /**
         * Given a stridee value, advance the pointer to the end of the current stride.
         * @param stride - The stride param.
         */
        readPad(stride) {
            const pad = this.__byteOffset % stride;
            if (pad != 0)
                this.__byteOffset += stride - pad;
        }
    }

    /**
     * Class designed to store version data. Widely used in the zea engine for backwards compatibility.
     */
    class Version {
        major = 0;
        minor = 0;
        patch = 0;
        branch = '';
        /**
         * Creates a version.
         * The version string should have the following structure:
         * major, minor and patch separated by a dot(`.`) and parts separated by a dash(`-`).
         *
         * @param arg - The version string value, or an array of version numbers.
         */
        constructor(arg) {
            if (typeof arg == 'string') {
                const parts = arg.split('-');
                const numbers = parts[0].split('.');
                this.major = parseInt(numbers[0]);
                this.minor = numbers.length > 1 ? parseInt(numbers[1]) : 0;
                this.patch = numbers.length > 2 ? parseInt(numbers[2]) : 0;
                if (parts.length == 2)
                    this.branch = parts[1];
            }
            else if (Array.isArray(arg)) {
                const numbers = arg;
                this.major = numbers[0];
                this.minor = numbers.length > 1 ? numbers[1] : 0;
                this.patch = numbers.length > 2 ? numbers[2] : 0;
            }
        }
        /**
         * Compare a version object against a version numbers array.
         *
         * @param numbers - An array containing 3 version numbers. [Major, Minor, Patch]
         * @return - return positive: v1 > v2, zero:v1 == v2, negative: v1 < v2
         */
        compare(numbers) {
            // https://stackoverflow.com/questions/6832596/how-to-compare-software-version-number-using-js-only-number
            // 2nd answer.
            const v1 = [this.major, this.minor, this.patch];
            for (let i = 0; i < 3; i++) {
                if (v1[i] !== numbers[i])
                    return v1[i] - numbers[i];
            }
            return 0;
        }
        /**
         * Converts the Version class instance back to an array for comparisons with other version class instances.
         * e.g.
         * ```
         *   const version1 = new Version([1, 2, 3])
         *   const version2 = new Version([1, 2, 4])
         *   const res = version1.compare(version2.asArray())
         * ```
         * @returns an array containing the major, minor and patch version numbers.
         */
        asArray() {
            return [this.major, this.minor, this.patch];
        }
        toString() {
            return `v${this.major}.${this.minor}.${this.patch}` + (this.branch != '' ? `-${this.branch}` : '');
        }
    }

    /**
     * @private
     * @extends BaseGeom
     */
    class CompoundGeomLoader extends BaseGeom {
        numSubGeoms = 0;
        indices = new Uint8Array(0);
        offsets = {};
        counts = {};
        // For each type of geom (TRIANGLES, LINES)
        // A material id, and each start and end of the block
        materialSubGeoms = {};
        subGeomOffsets = {};
        subGeomCounts = {};
        materialLibraryIndices = new Uint32Array(0);
        subGeomMaterialIndices = new Uint8Array(0);
        /**
         * Create points.
         */
        constructor() {
            super();
        }
        genBuffers() {
            const attrBuffers = {};
            for (const [attrName, attr] of this.__vertexAttributes) {
                attrBuffers[attrName] = attr.genBuffer();
            }
            const numVertices = this.numVertices();
            const result = {
                numVertices,
                numRenderVerts: numVertices,
                indices: this.indices,
                attrBuffers,
                offsets: this.offsets,
                counts: this.counts,
                numSubGeoms: this.numSubGeoms,
                subGeomOffsets: this.subGeomOffsets,
                subGeomCounts: this.subGeomCounts,
                materialLibraryIndices: this.materialLibraryIndices,
                subGeomMaterialIndices: this.subGeomMaterialIndices,
                materialSubGeoms: this.materialSubGeoms,
            };
            return result;
        }
        // ////////////////////////////////////////
        // Persistence
        /**
         * Sets state of current geometry(Including line segments) using a binary reader object.
         *
         * @param {BinReader} reader - The reader value.
         * @param {Record<string, unknown>} context - The context value.
         */
        readBinary(reader, context) {
            super.loadBaseGeomBinary(reader, context);
            const geomCountsByType = reader.loadUInt32Array(3);
            this.offsets['TRIANGLES'] = 0;
            this.counts['TRIANGLES'] = geomCountsByType[0];
            this.offsets['LINES'] = geomCountsByType[0];
            this.counts['LINES'] = geomCountsByType[1];
            this.offsets['POINTS'] = geomCountsByType[0] + geomCountsByType[1];
            this.counts['POINTS'] = geomCountsByType[2];
            const bytes = reader.loadUInt8();
            // Note: do not clone the source arrays as we will transfer the
            // entire buffer back to the main thread where it will be freed once
            // the data is uploaded to the GPU.
            if (bytes == 1)
                this.indices = reader.loadUInt8Array();
            else if (bytes == 2)
                this.indices = reader.loadUInt16Array();
            else if (bytes == 4)
                this.indices = reader.loadUInt32Array();
            // /////////////////////////////////
            // TRIANGLES subgeoms
            const bytesMeshSubGeoms = reader.loadUInt8();
            let subGeomCountsMesh;
            if (bytesMeshSubGeoms == 1)
                subGeomCountsMesh = reader.loadUInt8Array();
            else if (bytesMeshSubGeoms == 2)
                subGeomCountsMesh = reader.loadUInt16Array();
            else if (bytesMeshSubGeoms == 4)
                subGeomCountsMesh = reader.loadUInt32Array();
            else {
                throw Error('subGeomOffsets undefined');
            }
            const subGeomOffsetsMesh = new Uint32Array(subGeomCountsMesh.length);
            let offset = 0;
            for (let i = 0; i < subGeomCountsMesh.length; i++) {
                subGeomOffsetsMesh[i] = offset;
                offset += subGeomCountsMesh[i];
            }
            this.subGeomOffsets['TRIANGLES'] = subGeomOffsetsMesh;
            this.subGeomCounts['TRIANGLES'] = subGeomCountsMesh;
            // /////////////////////////////////
            // LINES subgeoms
            const bytesLinesSubGeoms = reader.loadUInt8();
            let subGeomCountsLines;
            if (bytesLinesSubGeoms == 1)
                subGeomCountsLines = reader.loadUInt8Array();
            else if (bytesLinesSubGeoms == 2)
                subGeomCountsLines = reader.loadUInt16Array();
            else if (bytesLinesSubGeoms == 4)
                subGeomCountsLines = reader.loadUInt32Array();
            else {
                throw Error('subGeomOffsets undefined');
            }
            const subGeomOffsetsLines = new Uint32Array(subGeomCountsLines.length);
            for (let i = 0; i < subGeomCountsLines.length; i++) {
                subGeomOffsetsLines[i] = offset;
                offset += subGeomCountsLines[i];
            }
            this.subGeomOffsets['LINES'] = subGeomOffsetsLines;
            this.subGeomCounts['LINES'] = subGeomCountsLines;
            // /////////////////////////////////
            // POINTS subgeoms
            const numPointsSubGeoms = reader.loadUInt32();
            const subGeomOffsetsPoints = new Uint32Array(numPointsSubGeoms);
            const subGeomCountsPoints = new Uint8Array(numPointsSubGeoms);
            for (let i = 0; i < numPointsSubGeoms; i++) {
                subGeomOffsetsPoints[i] = offset;
                subGeomCountsPoints[i] = 1;
                offset++;
            }
            this.subGeomOffsets['POINTS'] = subGeomOffsetsPoints;
            this.subGeomCounts['POINTS'] = subGeomCountsPoints;
            if (context.versions['zea-engine'].compare([3, 11, 0]) > 0) {
                // Points sub-geoms were not being exported before this release.
                this.numSubGeoms = subGeomCountsMesh.length + subGeomCountsLines.length + numPointsSubGeoms;
            }
            else {
                this.numSubGeoms = subGeomCountsMesh.length + subGeomCountsLines.length;
            }
            // /////////////////////////////////
            // Materials
            const numMaterials = reader.loadUInt32();
            if (numMaterials > 0) {
                this.materialLibraryIndices = reader.loadUInt32Array(numMaterials);
                this.subGeomMaterialIndices = reader.loadUInt8Array(this.numSubGeoms);
                // /////////////////////////////////
                // Material Groups
                let offset = 0;
                let currMaterial = -99;
                let currMaterialSubGeom = null;
                for (let i = 0; i < this.numSubGeoms; i++) {
                    let key;
                    let subGeomOffset = 0;
                    if (i < this.subGeomCounts.TRIANGLES.length) {
                        if (!this.materialSubGeoms.TRIANGLES)
                            this.materialSubGeoms.TRIANGLES = [];
                        key = 'TRIANGLES';
                    }
                    else if (i < this.subGeomCounts.TRIANGLES.length + this.subGeomCounts.LINES.length) {
                        subGeomOffset = this.subGeomCounts.TRIANGLES.length;
                        key = 'LINES';
                        if (!this.materialSubGeoms.LINES)
                            this.materialSubGeoms.LINES = [];
                    }
                    else {
                        subGeomOffset = this.subGeomCounts.TRIANGLES.length + this.subGeomCounts.LINES.length;
                        key = 'POINTS';
                        if (!this.materialSubGeoms.POINTS)
                            this.materialSubGeoms.POINTS = [];
                    }
                    const materialId = this.subGeomMaterialIndices[i];
                    if (currMaterial != materialId) {
                        currMaterial = materialId;
                        // Note: subGeomMaterialIndices is Uint8Array, and 0 means no custom
                        // material is assigned to the subGeom.
                        // Subtract 1 to get the actual material id.
                        currMaterialSubGeom = {
                            materialId: materialId - 1,
                            offset,
                            count: 0,
                        };
                        for (; i < this.numSubGeoms; i++) {
                            if (currMaterial != this.subGeomMaterialIndices[i]) {
                                break;
                            }
                            // When we get to the end og this geom type (e.g .TRIANGLES)
                            // start a new subgeom.
                            if (i - subGeomOffset == this.subGeomCounts[key].length) {
                                // Force the material index to be reset on line 162 above.
                                currMaterial = -99;
                                break;
                            }
                            currMaterialSubGeom.count += this.subGeomCounts[key][i - subGeomOffset];
                        }
                        offset += currMaterialSubGeom.count;
                        this.materialSubGeoms[key].push(currMaterialSubGeom);
                        i--;
                    }
                }
                // Note: older version of zcad files would list materials for TRIANGLES and LINES,
                // but not points, leaving the points unrendered.
                if (this.subGeomCounts.POINTS.length > 0 && !this.materialSubGeoms.POINTS) {
                    this.materialSubGeoms.POINTS = [
                        {
                            materialId: -1,
                            offset,
                            count: this.subGeomCounts.POINTS.length,
                        },
                    ];
                }
            }
            else {
                this.materialSubGeoms = {};
                let offset = 0;
                for (let key in this.counts) {
                    const count = this.counts[key];
                    if (count > 0) {
                        this.materialSubGeoms[key] = [
                            {
                                materialId: -1,
                                offset,
                                count,
                            },
                        ];
                    }
                    offset += count;
                }
            }
            this.emit('geomDataChanged', {});
        }
    }

    /* eslint-disable guard-for-in */
    // key, toc, geomIndexOffset, geomsRange, isMobileDevice, bufferSlice, genBuffersOpts, context
    const parseGeomsBinary = (data, callback) => {
        // eslint-disable-next-line guard-for-in
        for (const key in data.context.versions) {
            const v = data.context.versions[key];
            const version = new Version('');
            version.major = v.major;
            version.minor = v.minor;
            version.patch = v.patch;
            version.branch = v.branch;
            data.context.versions[key] = version;
        }
        const geomDatas = [];
        const byteOffset = data.byteOffset;
        // console.log('byteOffset:' + byteOffset, ' geomsRange:', data.geomsRange)
        const transferables = [];
        for (let i = data.geomsRange[0]; i < data.geomsRange[1]; i++) {
            const reader = new BinReader(data.bufferSlice, data.toc[i] - byteOffset, data.isMobileDevice);
            const className = reader.loadStr();
            const pos = reader.pos();
            // const name = reader.loadStr()
            // console.log(
            //   i +
            //     ':' +
            //     byteOffset +
            //     ' className:' +
            //     className +
            //     ' name:' +
            //     name /* + " pos:" + (data.toc[i] - byteOffset) + " bufferSlice.byteLength:" +  bufferSlice.byteLength*/
            // )
            let geom;
            switch (className) {
                case 'Points':
                    geom = new Points();
                    break;
                case 'Lines':
                    geom = new Lines();
                    break;
                case 'Mesh':
                    geom = new Mesh();
                    break;
                case 'CompoundGeom':
                    geom = new CompoundGeomLoader();
                    break;
                default:
                    throw new Error('Unsupported Geom type:' + className);
            }
            try {
                reader.seek(pos); // Reset the pointer to the start of the item data.
                geom.readBinary(reader, data.context);
            }
            catch (e) {
                console.warn('Error loading:' + geom.name + '\n:' + e);
                geomDatas.push({});
                continue;
            }
            const geomBuffers = geom.genBuffers(data.genBuffersOpts);
            // /////////////////////////////////
            // Transferables
            // Transfer all the attributes and indices back to the main thread.
            if (geomBuffers.indices)
                transferables.push(geomBuffers.indices.buffer);
            for (const attrName in geomBuffers.attrBuffers) {
                // Note: The type value assigned to the attribute can
                // not be transferred back to the main thread. Convert to
                // the type name here and send back as a string.
                const attrData = geomBuffers.attrBuffers[attrName];
                transferables.push(attrData.values.buffer);
            }
            // Compoound Geom buffers
            if (geomBuffers.materialLibraryIndices) {
                transferables.push(geomBuffers.materialLibraryIndices.buffer);
            }
            if (geomBuffers.subGeomMaterialIndices) {
                transferables.push(geomBuffers.subGeomMaterialIndices.buffer);
            }
            if (geomBuffers.subGeomOffsets) {
                for (let key in geomBuffers.subGeomOffsets) {
                    transferables.push(geomBuffers.subGeomOffsets[key].buffer);
                }
            }
            // if (geomBuffers.vertexNeighbors) {
            //   transferables.push(geomBuffers.vertexNeighbors.buffer)
            // }
            // /////////////////////////////////
            geomDatas.push({
                name: geom.name,
                type: className,
                geomBuffers,
                bbox: geom.getBoundingBox(),
            });
        }
        callback({
            taskId: data.taskId,
            geomLibraryId: data.geomLibraryId,
            geomFileID: data.geomFileID,
            geomIndexOffset: data.geomIndexOffset,
            geomsRange: data.geomsRange,
            geomDatas,
        }, transferables);
    };

    const handleMessage = function (srcData, postMessage) {
      parseGeomsBinary(srcData, (resultData, transferables) => {
        postMessage(resultData, transferables);
      });
    };

    // Check to see if we are running in the Worker before assigning onmessage
    if (globalThis.document === undefined) {
      globalThis.onmessage = function (event) {
        if (!event.data) {
          // Note: we see this occur when loading one large asset many times.
          // Like when loading the portafil.
          // It may be due to memory issues, but its not clear.
          console.warn('GeomLibrary worker.postMessage failed. data was lost on the way to the web worker.');
          return
        }
        // Our worker gets triggered
        if (!event.data.context) {
          return
        }
        handleMessage(event.data, self.postMessage);
      };
    }

    exports.handleMessage = handleMessage;

    Object.defineProperty(exports, '__esModule', { value: true });

    return exports;

})({});

",null,!1);class GeomParserMainThread extends EventEmitter{constructor(){super()}addTask(A,g){return new Promise((g=>{handleMessage(A,(A=>{A.eventName?this.emit(A.eventName,A):g(A)}))}))}}class GeomParserWorkerPool extends WorkerPool{constructor(){super(!0),this.poolSize=Math.max(1,Math.floor(.5*SystemDesc.hardwareConcurrency))}constructWorker(){const A=new WorkerFactory$1;return Promise.resolve(A)}}let geomParserWorkerPool;geomParserWorkerPool="Node"==SystemDesc.OS?new GeomParserMainThread:new GeomParserWorkerPool;let numGeomLibraries=0;class GeomLibrary extends EventEmitter{assetItem;listenerIDs={};streamInfos={};genBuffersOpts={};loadContext;numGeoms=-1;numGeomFiles=1;geoms=[];basePath="";loadedCount=0;constructor(A){super(),this.assetItem=A,numGeomLibraries++}isLoaded(){return this.loaded}get loaded(){return-1==this.numGeoms||this.loadedCount==this.numGeoms}loadGeomFile(A,g=!1){return g&&resourceLoader.incrementWorkload(1),new Promise((I=>{const C=this.basePath+A+".zgeoms";resourceLoader.loadFile("archive",C,!1).then((C=>{const e=C[Object.keys(C)[0]],i=this.on("streamFileParsed",(C=>{C.geomFileID==A&&(g&&resourceLoader.incrementWorkDone(1),this.off("streamFileParsed",i),I())}));this.readBinaryBuffer(A,e.buffer,this.loadContext)}))}))}loadGeomFilesStream(A,g,I){this.numGeomFiles=A.numGeomFiles?A.numGeomFiles:A.numGeomsPerFile.length,resourceLoader.incrementWorkload(this.numGeomFiles),this.numGeoms=A.numGeoms,this.basePath=g,this.loadContext=I;for(let A=0;A<this.numGeomFiles;A++)this.loadGeomFile(A,!1).finally((()=>resourceLoader.incrementWorkDone()))}setGenBufferOption(A,g){this.genBuffersOpts[A]=g}setNumGeoms(A){this.numGeoms=A}getNumGeoms(){return this.numGeoms}getGeom(A){return A>=this.geoms.length?null:this.geoms[A]}setGeom(A,g){g.libraryIndex=A,this.geoms[A]=g}readBinaryBuffer(A,g,I){const C=new BinReader(g,0,SystemDesc.isMobileDevice),e=C.loadUInt32(),i=C.loadUInt32();if(this.streamInfos[A]={total:e,done:0},0==e){const g=new StreamFileParsedEvent(A,0);return void this.emit("streamFileParsed",g)}-1==this.numGeoms&&(this.numGeoms=e);const t=C.loadUInt32Array(e);if(numGeomLibraries>1||this.numGeomFiles>1){const o=[0,e],a=0;geomParserWorkerPool.addTask({geomFileID:A,toc:t,byteOffset:a,geomIndexOffset:i,geomsRange:o,isMobileDevice:C.isMobileDevice,bufferSlice:g,genBuffersOpts:this.genBuffersOpts,context:{versions:I.versions}},[g]).then((A=>{this.receiveGeomDatas(A)}))}else{const o=2e6;let a=0;for(;a<e;){const B=t[a];let s,l,n=0,d=a;for(;d<e&&n<o;)d++,n=t[d]-B;d>=e?(s=[a,e],l=g.byteLength):(s=[a,d],l=t[s[1]]);const r=0==a&&d==e,c=r?0:t[s[0]],h=r?g:g.slice(B,l);a=d,geomParserWorkerPool.addTask({geomFileID:A,toc:t,byteOffset:c,geomIndexOffset:i,geomsRange:s,isMobileDevice:C.isMobileDevice,bufferSlice:h,genBuffersOpts:this.genBuffersOpts,context:{versions:I.versions}},[h]).then((A=>{this.receiveGeomDatas(A)}))}}}receiveGeomDatas(A){const{geomFileID:g,geomDatas:I,geomIndexOffset:C,geomsRange:e}=A,i=C+e[0],t=[i,C+e[1]];for(let A=0;A<I.length;A++){const g=I[A];if(!g.type)continue;let C;switch(g.type){case"Points":C=new PointsProxy(g);break;case"Lines":C=new LinesProxy(g);break;case"Mesh":case"Plane":case"Sphere":case"Cone":C=new MeshProxy(g);break;case"CompoundGeom":C=new CompoundGeom(g,this.assetItem.getMaterialLibrary());break;default:throw new Error("Unsupported Geom type:")}this.setGeom(i+A,C)}const o=new RangeLoadedEvent(t);this.emit("rangeLoaded",o);const a=t[1]-t[0],B=this.streamInfos[g];if(B.done+=a,B.done==B.total){const A=new StreamFileParsedEvent(g,B.done);this.emit("streamFileParsed",A),resourceLoader.emit("streamGeomsLoaded",A)}return this.loadedCount+=a,this.loadedCount==this.numGeoms&&this.emit("loaded"),this.loadedCount==this.numGeoms}toJSON(){return{numGeoms:this.geoms.length}}toString(){return JSON.stringify(this.toJSON(),null,2)}loadMetadata(A,g){const I=new BinReader(A.buffer,0,SystemDesc.isMobileDevice),C=I.loadUInt32Array();for(let A=0;A<C.length;A++)try{const e=this.geoms[A];e||console.warn("Error loading metadata for geom that was not yet loaded: ",A),e instanceof CompoundGeom&&(I.seek(C[A]),e.loadMetadata(I,g))}catch(g){console.warn("Error loading geom metadata: ",A)}}}class MaterialLibrary extends BaseItem{assetItem;images={};materials=[];__materialsMap={};constructor(A){super(),this.assetItem=A}clear(){this.images={},this.materials=[],this.__materialsMap={}}getPath(){return this.assetItem instanceof BaseItem?[...this.assetItem.getPath(),"MaterialLibrary"]:["MaterialLibrary"]}resolvePath(A,g=0){0==g&&("."!=A[0]&&"MaterialLibrary"!=A[0]||g++);const I=this.getMaterial(A[g]);if(I)return g<A.length?I.resolvePath(A,g+1):I;throw new Error(`Unable to resolve path : [${A.toString()}] after: 'MaterialLibrary' \nNo material called : "${A[g]}"`)}getNumMaterials(){return this.materials.length}getMaterials(){return this.materials}getMaterialNames(){const A=[];return this.materials.forEach((g=>{A.push(g.getName())})),A}hasMaterial(A){return A in this.__materialsMap}addMaterial(A){A.setOwner(this),A.libraryIndex=this.materials.length,this.__materialsMap[A.getName()]=this.materials.length,this.materials.push(A)}getMaterial(A){if("string"==typeof A){const g=this.__materialsMap[A];return null==g?null:this.materials[g]}if(Number.isFinite(A))return this.materials[A]}hasImage(A){return A in this.images}addImage(A){A.setOwner(this),this.images[A.getName()]=A}getImage(A,g=!0){const I=this.images[A];if(!I&&g)throw new Error("Image:"+A+" not found in library:"+this.getImageNames());return I}getImageNames(){const A=[];for(const g in this.images)A.push(g);return A}load(A){const g=new XMLHttpRequest;g.open("GET",A,!0),g.ontimeout=()=>{throw new Error("The request for "+A+" timed out.")},g.onload=()=>{4===g.readyState&&(200===g.status?this.fromJSON(JSON.parse(g.responseText)):console.warn(g.statusText))},g.send(null)}toJSON(A={}){const g={numMaterials:this.getNumMaterials(),images:{},materials:[]};for(const I in this.images)g.images[I]=this.images[I].toJSON(A);for(const I of this.materials)g.materials.push(I.toJSON(A));return g}fromJSON(A,g={}){for(const g in A.textures){const I=new FileImage(g);I.fromJSON(A.textures[g]),this.images[g]=I}for(const g in A.materials){const I=new Material(g);I.fromJSON(A.materials[g]),this.addMaterial(I)}}readBinary(A,g){A.loadStr();const I=A.loadUInt32();for(let C=0;C<I;C++){const I=A.loadStr(),C=Registry.constructClass(I);C.readBinary(A,g),this.images[C.getName()]=C}const C=A.loadUInt32();if(C>0){const I=A.loadUInt32Array(C);for(let e=0;e<C;e++){let C;switch(A.loadStr()){case"StandardMaterial":case"TransparentMaterial":case"StandardSurfaceShader":C=Registry.constructClass("StandardSurfaceMaterial");break;case"SimpleSurfaceShader":case"SimpleSurfaceMaterial":C=Registry.constructClass("SimpleSurfaceMaterial");break;case"PointsShader":case"PointsMaterial":C=Registry.constructClass("PointsMaterial");break;case"FatPointsShader":case"FatPointsMaterial":C=Registry.constructClass("FatPointsMaterial");break;case"LinesShader":case"LinesMaterial":C=Registry.constructClass("LinesMaterial");break;default:C=new Material("")}A.seek(I[e]),C.readBinary(A,g),this.materials[e]=C,C.libraryIndex=e,this.__materialsMap[C.getName()]=e}}this.emit("loaded")}toString(){return JSON.stringify(this.toJSON(),null,2)}}class AssetLoadContext extends EventEmitter{units="meters";versions={};sdk="";url="";folder="";camera=null;assetItem=null;resources=null;xrefs={};xrefLoadCallback=null;lazyLoading;postLoadCallbacks=[];promisses=[];urlStack=[];assetStack=[];addGeomToLayer;constructor(A){super(),A&&(this.units=A.units,this.sdk=A.sdk,this.camera=A.camera,this.resources=A.resources,this.xrefs=A.xrefs,this.xrefLoadCallback=A.xrefLoadCallback,this.urlStack=[...A.urlStack],this.assetStack=[...A.assetStack])}addPromise(A){this.promisses.push(A)}resolvePath(A,g,I){try{const I=this.assetItem.resolvePath(A);g(I)}catch(C){this.postLoadCallbacks.push((()=>{try{const I=this.assetItem.resolvePath(A);g(I)}catch(A){if(!I)throw new Error(A.message);I(A)}}))}}addPLCB(A){this.postLoadCallbacks.push(A)}resolvePLCBs(){for(const A of this.postLoadCallbacks)A();this.postLoadCallbacks=[]}clone(){return new AssetLoadContext(this)}}class BaseGroup extends TreeItem{itemsParam=new ItemSetParameter("Items",(A=>A instanceof TreeItem));itemsEventHandlers=[];searchRoot;constructor(A){super(A),this.addParameter(this.itemsParam),this.itemsParam.on("itemAdded",(A=>{this.bindItem(A.item,A.index),this.emit("itemAdded",A)})),this.itemsParam.on("itemRemoved",(A=>{this.unbindItem(A.item,A.index),this.emit("itemRemoved",A)}))}updateVisibility(){if(super.updateVisibility()){const A=this.isVisible();return Array.from(this.itemsParam.value).forEach((g=>{g.propagateVisibility(A?1:-1)})),!0}return!1}updateOpacity(){super.updateOpacity(),Array.from(this.itemsParam.value).forEach((A=>{A.setInheritedOpacity(this,this.opacity)}))}cleanBoundingBox(){const A=super.cleanBoundingBox();return Array.from(this.itemsParam.value).forEach((g=>{if(g.isVisible()&&g.isPickable()){const I=g.boundingBoxParam.value;I&&A.addBox3(I)}})),A}addHighlight(A,g,I=!0){super.addHighlight(A,g,I),I&&Array.from(this.itemsParam.value).forEach((I=>{I.addHighlight(A,g,!0)}))}removeHighlight(A,g=!0){super.removeHighlight(A,g),g&&Array.from(this.itemsParam.value).forEach((g=>{g.removeHighlight(A,!0)}))}setSearchRoot(A){this.searchRoot=A}setOwner(A){if(A&&!(A instanceof TreeItem))throw new Error("cannot setOwner");this.searchRoot&&this.searchRoot!=this.getOwner()||(this.searchRoot=A),super.setOwner(A)}bindItem(A,g){const I={};I.pointerDown=A.on("pointerDown",(A=>{this.onPointerDown(A)})),I.pointerUp=A.on("pointerUp",(A=>{this.onPointerUp(A)})),I.pointerMove=A.on("pointerMove",(A=>{this.onPointerMove(A)})),I.pointerEnter=A.on("pointerEnter",(A=>{this.onPointerEnter(A)})),I.pointerLeave=A.on("pointerLeave",(A=>{this.onPointerLeave(A)})),I.pointerClick=A.on("pointerClick",(A=>{this.onPointerClick(A)})),I.pointerDoubleClick=A.on("pointerDoubleClick",(A=>{this.onPointerDoubleClick(A)})),I.pointerLongPress=A.on("pointerLongPress",(A=>{this.onPointerLongPress(A)})),I["BoundingBox.valueChanged"]=A.boundingBoxParam.on("valueChanged",(()=>{this.setBoundingBoxDirty()})),this.isVisible()||A.propagateVisibility(-1),this.highlights.forEach((g=>{A.addHighlight(g,this.highlightMapping[g],!0)})),this.itemsEventHandlers.splice(g,0,I)}unbindItem(A,g){const I=this.itemsEventHandlers[g];for(let g in I){const C=g.split(".");if(C.length>1){const e=A.getParameter(C[0]);e&&e.off(C[1],I[g])}else A.off(g,I[g])}this.isVisible()||A.propagateVisibility(1),this.highlights.forEach((g=>{A.removeHighlight(g,!0)})),this.setBoundingBoxDirty(),this.itemsEventHandlers.splice(g,1)}addItem(A,g=!0){A?this.itemsParam.addItem(A,g):console.warn("Error adding item to group. Item is null")}removeItem(A,g=!0){const I=this.itemsParam.value;if(!I)return;const C=Array.from(I).indexOf(A);-1!=C&&this.itemsParam.removeItem(C,g)}clearItems(A=!0){const g=this.itemsParam.value;if(!g)return;const I=Array.from(g);for(let A=I.length-1;A>=0;A--)this.unbindItem(I[A],A);this.itemsParam.clearItems(A)}getItems(){return this.itemsParam.value}setItems(A){this.clearItems(!1),this.itemsParam.setItems(A)}}class SelectionSet extends BaseGroup{#o=!1;highlightedParam=new BooleanParameter("Highlighted",!1);highlightColorParam=new ColorParameter("HighlightColor",new Color(.5,.5,1));highlightFillParam=new NumberParameter("HighlightFill",0,[0,1]);constructor(A){super(A),this.addParameter(this.highlightedParam),this.highlightedParam.on("valueChanged",(()=>{this.updateHighlight()})),this.addParameter(this.highlightColorParam),this.highlightColorParam.on("valueChanged",(()=>{this.updateHighlight()})),this.addParameter(this.highlightFillParam),this.highlightFillParam.on("valueChanged",(()=>{this.updateHighlight()}))}updateHighlight(){this.#o||(this.#o=!0,setTimeout((()=>{this.#a(),this.#o=!1}),0))}#a(){let A,g=this.highlightedParam.value;g&&(A=this.highlightColorParam.value.clone(),A.a=this.highlightFillParam.value);const I="groupItemHighlight"+this.getId();Array.from(this.itemsParam.value).forEach((C=>{g?C.addHighlight(I,A,!0):C.removeHighlight(I,!0)}))}setSelected(A){super.setSelected(A),this.updateHighlight()}bindItem(A,g){if(super.bindItem(A,g),this.highlightedParam.value){const g=this.highlightColorParam.value;g.a=this.highlightFillParam.value,A.addHighlight("groupItemHighlight"+this.getId(),g,!0)}}unbindItem(A,g){super.unbindItem(A,g),this.highlightedParam.value&&A.removeHighlight("groupItemHighlight"+this.getId(),!0)}clone(A){const g=new SelectionSet(this.name+" clone");return g.copyFrom(this,A),g}}Registry.register("SelectionSet",SelectionSet);class GroupGlobalXfoOperator extends CalcGlobalXfoOperator{offsetXfo=new Xfo;bindXfo=new Xfo;invBindXfo=new Xfo;#B=!1;groupTransformXfo=new XfoOperatorOutput("GroupTransformXfo");constructor(A,g,I){super(A,g),this.groupTransformXfo.setParam(I),this.addOutput(this.groupTransformXfo)}set editingPivot(A){this.#B=A,this.setDirty()}get editingPivot(){return this.#B}setBindXfo(A){this.bindXfo=A,this.invBindXfo=A.inverse(),this.setDirty()}backPropagateValue(A){let g;if(g=this.parentGlobal.isConnected()?this.parentGlobal.getValue():new Xfo,this.editingPivot)this.bindXfo=A,this.invBindXfo=this.bindXfo.inverse(),this.offsetXfo=g.inverse().multiply(this.bindXfo),this.setDirty();else{const I=g.multiply(this.offsetXfo);this.localXfo.setValue(I.inverse().multiply(A))}}evaluate(){let A;if(this.parentGlobal.isConnected()){A=this.parentGlobal.getValue().multiply(this.offsetXfo)}else A=this.offsetXfo;if(this.editingPivot){const g=A;this.globalXfo.setClean(g);const I=g.multiply(this.localXfo.getValue()).multiply(this.invBindXfo);this.groupTransformXfo.setClean(I)}else{const g=A.multiply(this.localXfo.getValue());this.globalXfo.setClean(g);const I=g.multiply(this.invBindXfo);this.groupTransformXfo.setClean(I)}}}class GroupMemberXfoOperator extends Operator{#s;groupTransformXfo=new XfoOperatorInput("GroupTransformXfo");memberGlobalXfo=new XfoOperatorOutput("MemberGlobalXfo",OperatorOutputMode.OP_READ_WRITE);constructor(A,g){super(),this.groupTransformXfo.setParam(A),this.memberGlobalXfo.setParam(g),this.addInput(this.groupTransformXfo),this.addOutput(this.memberGlobalXfo),this.#s=!0}disable(){this.#s=!1,this.setDirty()}enable(){this.#s=!0,this.setDirty()}evaluate(){const A=this.memberGlobalXfo.getValue();if(this.#s){const g=this.groupTransformXfo.getValue();this.memberGlobalXfo.setClean(g.multiply(A))}else this.memberGlobalXfo.setClean(A)}backPropagateValue(A){if(this.#s){return this.groupTransformXfo.getValue().inverse().multiply(A)}return A}}const GROUP_XFO_MODES={disabled:0,manual:1,first:2,average:3,globalOri:4};class KinematicGroup extends BaseGroup{calculatingGroupXfo;memberXfoOps;#o=!1;initialXfoModeParam=new MultiChoiceParameter("InitialXfoMode",GROUP_XFO_MODES.average,["manual","first","average","global"]);groupTransformParam=new XfoParameter("GroupTransform",new Xfo);highlightedParam=new BooleanParameter("Highlighted",!1);highlightColorParam=new ColorParameter("HighlightColor",new Color(1,1,0));highlightFillParam=new NumberParameter("HighlightFill",0,[0,1]);constructor(A=""){super(A),this.calculatingGroupXfo=!1,this.memberXfoOps=[],this.addParameter(this.initialXfoModeParam),this.initialXfoModeParam.on("valueChanged",(()=>{this.calcGroupXfo()})),this.addParameter(this.groupTransformParam),this.addParameter(this.highlightedParam),this.highlightedParam.on("valueChanged",(()=>{this.#l()})),this.addParameter(this.highlightColorParam),this.highlightColorParam.on("valueChanged",(()=>{this.#l()})),this.addParameter(this.highlightFillParam),this.highlightFillParam.on("valueChanged",(()=>{this.#l()})),this.globalXfoOp=new GroupGlobalXfoOperator(this.globalXfoParam,this.localXfoParam,this.groupTransformParam)}static get INITIAL_XFO_MODES(){return GROUP_XFO_MODES}get bindXfo(){return this.globalXfoOp.bindXfo}#l(){this.#o||(this.#o=!0,setTimeout((()=>{this.#a(),this.#o=!1}),0))}#a(){let A,g=this.highlightedParam.value;g&&(A=this.highlightColorParam.value.clone(),A.a=this.highlightFillParam.value);const I="groupItemHighlight"+this.getId();Array.from(this.itemsParam.value).forEach((C=>{g?C.addHighlight(I,A,!0):C.removeHighlight(I,!0)}))}calcGroupXfo(){const A=Array.from(this.itemsParam.value);if(0==A.length)return;this.editingPivot=!0;const g=this.initialXfoModeParam.value;let I;if(g==GROUP_XFO_MODES.manual)I=this.globalXfoParam.value;else if(g==GROUP_XFO_MODES.first&&A[0])I=A[0].globalXfoParam.value;else if(g==GROUP_XFO_MODES.average){I=new Xfo,I.ori.set(0,0,0,0);let g=0;A.forEach(((A,C)=>{const e=A.globalXfoParam.value;I.tr.addInPlace(e.tr),0==C&&(I.ori=e.ori),g++})),I.tr.scaleInPlace(1/g),I.ori.normalizeInPlace()}else{if(g!=GROUP_XFO_MODES.globalOri)throw new Error("Invalid GROUP_XFO_MODES.");{I=new Xfo;let g=0;A.forEach(((A,C)=>{const e=A.globalXfoParam.value;I.tr.addInPlace(e.tr),g++})),I.tr.scaleInPlace(1/g)}}this.globalXfoParam.value=I,this.editingPivot=!1}get editingPivot(){return this.globalXfoOp.editingPivot}set editingPivot(A){this.globalXfoOp.editingPivot=A}bindItem(A,g){super.bindItem(A,g);const I=A.globalXfoParam,C=new GroupMemberXfoOperator(this.groupTransformParam,I);if(this.memberXfoOps.splice(g,0,C),this.highlightedParam.value){const g=this.highlightColorParam.value;g.a=this.highlightFillParam.value,A.addHighlight("groupItemHighlight"+this.getId(),g,!0)}}unbindItem(A,g){super.unbindItem(A,g),this.memberXfoOps[g].detach(),this.memberXfoOps.splice(g,1),this.highlightedParam.value&&A.removeHighlight("groupItemHighlight"+this.getId(),!0)}addItem(A,g=!0){super.addItem(A,g),g&&this.calcGroupXfo()}removeItem(A,g=!0){super.removeItem(A,g),g&&this.calcGroupXfo()}setItems(A){super.setItems(A),this.calcGroupXfo()}clearItems(A=!0){super.clearItems(A),this.memberXfoOps=[],A&&this.calcGroupXfo()}clone(A){const g=new KinematicGroup;return g.copyFrom(this,A),g}toJSON(A){const g=super.toJSON(A);g.bindXfo=this.globalXfoOp.bindXfo.toJSON();const I=this.globalXfoOp;return g.offsetXfo=I.offsetXfo.toJSON(),g}fromJSON(A,g){super.fromJSON(A,g);const I=new Xfo;I.fromJSON(A.bindXfo);const C=this.globalXfoOp;C.setBindXfo(I),A.offsetXfo&&C.offsetXfo.fromJSON(A.offsetXfo)}}Registry.register("KinematicGroup",KinematicGroup);class MaterialGroup extends BaseGroup{materialParam=new MaterialParameter("Material");__backupMaterials={};constructor(A){super(A),this.addParameter(this.materialParam),this.materialParam.on("valueChanged",(()=>{this.updateMaterial()}))}updateHighlight(){this.updateHighlightHelper()}updateHighlightHelper(){let A,g=!1;this.isSelected()&&(A=this.getHighlight(),g=!0,A.a=.2);const I="kinematicGroupItemHighlight"+this.getId();Array.from(this.itemsParam.value).forEach((C=>{g?C.addHighlight(I,A,!0):C.removeHighlight(I,!0)}))}setSelected(A){super.setSelected(A),this.updateHighlight()}updateOpacity(){super.updateOpacity(),Array.from(this.itemsParam.value).forEach((A=>{A.opacityParam.value=this.opacity}))}updateMaterial(){this.updateMaterialHelper()}updateMaterialHelper(){const A=this.materialParam.value;Array.from(this.itemsParam.value).forEach((g=>{g.traverse((g=>{if(g instanceof BaseGeomItem){const I=g.materialParam;if(A){const g=I.value;g==A||g&&"LinesShader"==g.getShaderName()||(this.__backupMaterials[I.getId()]=g,I.value=A)}else this.__backupMaterials[I.getId()]&&(I.value=this.__backupMaterials[I.getId()])}}))}))}bindItem(A,g){if(super.bindItem(A,g),this.isSelected()){const g=this.getHighlight();g.a=.2;const I="materialGroupItemHighlight"+this.getId();A.addHighlight(I,g,!0)}const I=this.materialParam.value;I&&A.traverse((A=>{if(A instanceof BaseGeomItem){const g=A.materialParam;if(I){const A=g.value;A==I||A&&"LinesShader"==A.getShaderName()||(this.__backupMaterials[g.getId()]=A,g.value=I)}}}),!0),A.opacityParam.value=this.opacity}unbindItem(A,g){if(super.unbindItem(A,g),this.isSelected()){const g="materialGroupItemHighlight"+this.getId();A.removeHighlight(g,!0)}}clone(A){const g=new MaterialGroup(this.name+"clone");return g.copyFrom(this,A),g}}Registry.register("MaterialGroup",MaterialGroup);class CuttingPlaneOperator extends Operator{groupGlobalXfo=new XfoOperatorInput("GroupGlobalXfo");cuttingPlane=new Vec4OperatorOutput("CuttingPlane");constructor(A,g){super(),this.groupGlobalXfo.setParam(A),this.cuttingPlane.setParam(g),this.addInput(this.groupGlobalXfo),this.addOutput(this.cuttingPlane)}evaluate(){const A=this.groupGlobalXfo.getValue(),g=A.ori.getZaxis(),I=A.tr.dot(g);this.cuttingPlane.setClean(new Vec4(g.x,g.y,g.z,-I))}}class CuttingPlane extends BaseGroup{cutPlaneOp;cutAwayEnabledParam=new BooleanParameter("CutAwayEnabled",!1);cutPlaneParam=new Vec4Parameter("CutPlane",new Vec4(1,0,0));constructor(A=""){super(A),this.addParameter(this.cutAwayEnabledParam),this.addParameter(this.cutPlaneParam),this.cutPlaneOp=new CuttingPlaneOperator(this.globalXfoParam,this.cutPlaneParam),this.cutAwayEnabledParam.on("valueChanged",(A=>{this.updateCutaway(A)})),this.cutPlaneParam.on("valueChanged",(A=>{this.updateCutaway(A)}));const g=new Material("plane","FlatSurfaceShader");g.getParameter("BaseColor").value=new Color(1,1,1,.2);const I=new GeomItem("PlaneGeom",new Plane(1,1),g);this.addChild(I);const C=new Material("border","LinesShader");C.getParameter("BaseColor").value=new Color(1,0,0,1);const e=new GeomItem("BorderGeom",new Rect(1,1),C);this.addChild(e)}updateCutaway(A){const g=this.cutAwayEnabledParam.value,I=this.cutPlaneParam.value,C=I.xyz,e=I.w;A instanceof BaseGeomItem?(A.setCutawayEnabled(g),A.setCutVector(C),A.setCutDist(e)):Array.from(this.itemsParam.value).forEach((A=>{A.traverse((A=>{A instanceof BaseGeomItem&&(A.setCutawayEnabled(g),A.setCutVector(C),A.setCutDist(e))}),!0)}))}bindItem(A,g){this.cutAwayEnabledParam.value&&this.updateCutaway(A)}unbindItem(A,g){super.unbindItem(A,g),A.traverse((A=>{A instanceof BaseGeomItem&&A.setCutawayEnabled(!1)}),!0)}clone(A){const g=new CuttingPlane;return g.copyFrom(this,A),g}}Registry.register("CuttingPlane",CuttingPlane);const getUnitsFactor=A=>{switch(A.toLowerCase()){case"millimeters":return.001;case"centimeters":return.01;case"decimeters":return.1;case"meters":return 1;case"kilometers":return 1e3;case"inches":return.0254;case"feet":return.3048;case"miles":return 1609.34}return 1};class AssetItem extends TreeItem{geomLibrary=new GeomLibrary(this);materialLibrary=new MaterialLibrary(this);loaded=!1;engineDataVersion;unitsScale=1;units="meters";constructor(A=""){super(A)}load(A){return Promise.reject(`This method is not implemented for this Asset Item: ${A}`)}isLoaded(){return this.loaded}getGeometryLibrary(){return this.geomLibrary}getMaterialLibrary(){return this.materialLibrary}readBinary(A,g){g.assetItem=this,g.units||(g.units="meters"),g.versions["zea-engine"]||(g.versions["zea-engine"]=new Version(A.loadStr())),this.engineDataVersion=g.versions["zea-engine"];const I=()=>{if(this.units=A.loadStr(),this.addParameter(new StringParameter("FileUnits",this.units)),this.units!=g.units){const A=getUnitsFactor(this.units),I=getUnitsFactor(g.units);this.unitsScale=A/I}g.units=this.units;const I=this.localXfoParam,C=I.value;C.sc.scaleInPlace(this.unitsScale),I.value=C};let C;g.versions["zea-engine"].compare([0,0,6])>0&&I();const e={};g.addGeomToLayer=(A,g)=>{if(!e[g]){C||(C=new TreeItem("Layers"),this.addChild(C,!1));const A=new BaseGroup(g);C.addChild(A,!1),e[g]=A}e[g].addItem(A)},this.materialLibrary.readBinary(A,g),super.readBinary(A,g),g.versions["zea-engine"].compare([0,0,5])>=0&&g.versions["zea-engine"].compare([0,0,7])<0&&I(),g.resolvePLCBs(),this.loaded=!0}haveGeomsLoaded(){if(!this.geomLibrary.isLoaded())return!1;let A=!0;return this.traverse((g=>g instanceof AssetItem?(g.haveGeomsLoaded()||(A=!1),!1):A),!1),A}toJSON(A={}){A.makeRelative=A=>{const g=this.getPath(),I=A.slice(0,g.length);for(let C=0;C<I.length-1;C++)if(I[C]!=g[C])return console.warn("Param Path is not relative to the asset. May not be able to be resolved at load time:"+A),A;const C=A.slice(g.length-1);return C[0]=".",C},A.assetItem=this;const g=super.toJSON(A);return g.materialLibrary=this.materialLibrary.toJSON(),g}fromJSON(A,g=new AssetLoadContext){g.assetItem=this,this.materialLibrary.fromJSON(A.materialLibrary,g),super.fromJSON(A,g),g.resolvePLCBs()}clone(A){const g=new AssetItem;return g.copyFrom(this,A),g}copyFrom(A,g){if(!(A instanceof AssetItem))throw new Error("cannot copy from src");this.geomLibrary=A.geomLibrary,this.materialLibrary=A.materialLibrary,this.units=A.units,this.loaded=A.loaded,super.copyFrom(A,g),A.loaded?(this.emit("loaded"),A.haveGeomsLoaded()?this.emit("geomsLoaded"):A.once("geomsLoaded",(()=>this.emit("geomsLoaded")))):A.once("loaded",(()=>{this.units=A.units;const I=A.localXfoParam.value,C=this.localXfoParam.value;C.sc=I.sc.clone(),this.localXfoParam.value=C,this.removeAllChildren(),A.getChildren().forEach((A=>{this.addChild(A.clone(g),!1,!1)})),this.loaded=!0,this.emit("loaded"),A.haveGeomsLoaded()?this.emit("geomsLoaded"):A.once("geomsLoaded",(()=>this.emit("geomsLoaded")))}))}}Registry.register("AssetItem",AssetItem);const BillboardAlignment={AlignedToWorld:0,AlignedToCamera:1,AlignedToCameraAndXAxis:2};class BillboardItem extends TreeItem{imageParam=new ImageParameter("Image");pixelsPerMeterParam=new NumberParameter("PixelsPerMeter",1e3);alphaParam=new NumberParameter("Alpha",1);colorParam=new ColorParameter("Color",new Color(1,1,1));alignmentParam=new MultiChoiceParameter("Alignment",0,Object.keys(BillboardAlignment));drawOnTopParam=new BooleanParameter("DrawOnTop",!1);fixedSizeOnscreenParam=new BooleanParameter("FixedSizeOnscreen",!1);frontFacingParam=new BooleanParameter("FrontFacing",!0);pivotParam=new Vec2Parameter("Pivot",new Vec2(.5,0));constructor(A,g){super(A);const I=this.addParameter(this.imageParam);g&&(I.value=g),this.addParameter(this.pixelsPerMeterParam),this.addParameter(this.alphaParam),this.addParameter(this.colorParam),this.addParameter(this.alignmentParam),this.addParameter(this.drawOnTopParam),this.addParameter(this.fixedSizeOnscreenParam),this.addParameter(this.pivotParam)}}Registry.register("BillboardItem",BillboardItem);class CADAsset extends AssetItem{sdk;url;metadataLoadPromise;metadataLoaded=!1;constructor(A){super(A)}clone(A){const g=new CADAsset;return g.copyFrom(this,A),g}copyFrom(A,g){if(!(A instanceof CADAsset))throw new Error("cannot copy from src");super.copyFrom(A,g),this.url=A.url,A.loaded||A.once("geomsLoaded",(A=>{this.emit("geomsLoaded",A)}))}readRootLevelBinary(A,g){g.versions["zea-cad"]=new Version(A.loadStr()),g.sdk=A.loadStr(),this.sdk=g.sdk,super.readBinary(A,g)}async loadZCADEntries(A,g){const I=new BinReader((A.tree2||A.tree).buffer,0,SystemDesc.isMobileDevice);g.versions={},this.removeAllChildren();const C=this.getName();if(this.readRootLevelBinary(I,g),""!=C&&this.setName(C),A.geomsdata&&(this.metadataLoadPromise=new Promise((I=>{this.geomLibrary.once("loaded",(()=>{this.geomLibrary.loadMetadata(A.geomsdata,g),this.metadataLoaded=!0,I()}))}))),A.geoms){const I=-1;this.geomLibrary.readBinaryBuffer(I,A.geoms.buffer,g)}else if(A["geomLibrary.json"]){const I=JSON.parse(new TextDecoder("utf-8").decode(A["geomLibrary.json"])),C=g.url,e=C.lastIndexOf("/")>-1?C.substring(C.lastIndexOf("/")+1):"",i=e.substring(0,e.lastIndexOf(".")),t=g.folder+i;0==I.numGeomFiles?console.error("Corrupt zcad file. Missing 'geoms':",C):this.geomLibrary.loadGeomFilesStream(I,t,g)}return new Promise(((A,I)=>{Promise.allSettled(g.promisses).then((()=>{this.loaded=!0,this.emit("loaded"),A();const g=[];this.geomLibrary.isLoaded()||g.push(new Promise((A=>this.geomLibrary.once("loaded",A))));this.traverse((A=>{A instanceof CADAsset&&!A.geomLibrary.isLoaded()&&g.push(new Promise((g=>A.once("geomsLoaded",g))))}),!1),Promise.allSettled(g).then((()=>this.emit("geomsLoaded")))})).catch((A=>{I(A)}))}))}async load(A,g=new AssetLoadContext){return(g=g.clone()).assetItem=this,new Promise(((I,C)=>{if(resourceLoader.incrementWorkload(1),"string"==typeof A){const e=A,i=e.lastIndexOf("/")>-1?e.substring(0,e.lastIndexOf("/"))+"/":"";this.url=e,g.url=e,g.folder=i,g.urlStack.push(e),g.assetStack.push(this),resourceLoader.loadFile("archive",e).then((A=>{if(!A.tree2&&!A.tree){resourceLoader.incrementWorkDone(1);const A="Corrupt zcad file. Missing 'tree':"+e;return this.emit("error",A),void C(A)}this.loadZCADEntries(A,g).then((()=>{resourceLoader.incrementWorkDone(1),I()}))}),(A=>{resourceLoader.incrementWorkDone(1),this.emit("error",A),C(A)}))}else if(A instanceof ArrayBuffer){resourceLoader.plugins.archive.extractFile(A).then((A=>{this.loadZCADEntries(A,g).then((()=>{resourceLoader.incrementWorkDone(1),I()}))}),(A=>{resourceLoader.incrementWorkDone(1),this.emit("error",A),C(A)}))}}))}loadMetadata(A=""){return this.metadataLoadPromise||(this.metadataLoadPromise=new Promise(((g,I)=>{if(this.metadataLoaded&&g(),""==A){const g=this.url,I=g.substring(0,g.lastIndexOf("."));A=I+".zmetadata"}resourceLoader.incrementWorkload(),resourceLoader.loadFile("archive",A).then((A=>{const I=new AssetLoadContext;I.versions["zea-engine"]=this.engineDataVersion,this.geomLibrary.loadMetadata(A.geomsdata,I),resourceLoader.incrementWorkDone(1),this.metadataLoaded=!0,g()}),(A=>{resourceLoader.incrementWorkDone(1),this.emit("error",A),I(A)}))}))),this.metadataLoadPromise}}Registry.register("CADAsset",CADAsset);class CADAssembly extends TreeItem{constructor(A){super(A)}clone(A){const g=new CADAssembly;return g.copyFrom(this,A),g}}Registry.register("CADAssembly",CADAssembly);class CADPart extends TreeItem{constructor(A){super(A)}clone(A){const g=new CADPart;return g.copyFrom(this,A),g}}Registry.register("CADPart",CADPart);class CADBody extends GeomItem{shattered=!1;constructor(A){super(A)}setShatterState(A){this.shattered!=A&&(this.shattered=A,this.emit("shatterStateChanged",new StateChangedEvent(A)))}clone(A){const g=new CADBody;return g.copyFrom(this,A),g}readBinary(A,g){if(g.versions["zea-engine"].compare([3,9,0])<0){if(BaseGeomItem.prototype.readBinary.call(this,A,g),A.loadSInt32(),g.versions["zea-cad"].compare([0,0,4])<0){const I=A.loadStr();let C=g.assetItem.getMaterialLibrary().getMaterial(I);C||(C=new Material(I,"SimpleSurfaceShader"),C.getParameter("BaseColor").setValue(Color.random(.25)),g.assetItem.getMaterialLibrary().addMaterial(C)),this.materialParam.setValue(C)}if(g.versions["zea-cad"].compare([0,0,2])>=0&&g.versions["zea-cad"].compare([0,0,4])<0){this.layers=A.loadStrArray();for(const A of this.layers)g.addGeomToLayer(this,A)}}else super.readBinary(A,g)}}Registry.register("CADBody",CADBody);const plane=new Cuboid(1,1,1);let planeMaterial=null;class PMIPickingPlane extends GeomItem{addHighlight(A,g,I=!0){}removeHighlight(A,g=!0){}}class PMIItem extends TreeItem{materialMapping={};constructor(A){super(A)}clone(A){const g=new PMIItem;return g.copyFrom(this,A),g}setSelected(A){super.setSelected(A),A?this.activate():this.deactivate()}activate(){}deactivate(){}addHighlight(A,g,I=!0){if(super.addHighlight(A,g,!1),I){const A=g.clone();A.a=1;const I={};this.traverse((g=>{if(g instanceof PMIPickingPlane)return!1;if(g instanceof GeomItem){const C=g.materialParam.value;if(this.materialMapping[g.getId()]=C,!(C.getId()in I)){const e=C.clone();if(e.hasParameter("BaseColor")){const g=e.getParameter("BaseColor");g instanceof MaterialColorParam&&(g.colorSpace=ColorSpace.Gamma),g.setValue(A)}if(e.hasParameter("EdgeColor")){const g=e.getParameter("EdgeColor");g instanceof MaterialColorParam&&(g.colorSpace=ColorSpace.Gamma),g.setValue(A)}e.hasParameter("Overlay")&&e.getParameter("Overlay").setValue(.85),e.__isOpaque=!1,g.materialParam.value=e,I[C.getId()]=e}g.materialParam.value=I[C.getId()]}}))}const C=this.getOwner().getOwner().getOwner();if(C){const I={},e={},i=this.getParameter("LinkedEntities");if(i){i.getValue().forEach(((i,t)=>{if(""==i)return;const o=i.split(", "),a=o.pop();try{const i=C.resolvePath(o);if(i&&i instanceof CADBody)if(0==i.getNumChildren())i.setShatterState(!0),I[i.getId()]||(I[i.getId()]=i,e[i.getId()]=[]),e[i.getId()].push(a);else{const I=i.getChildByName(a);I&&I.addHighlight(A,g,!0)}else console.log("linkedEntity.addHighlight(name, color, true):",o)}catch(A){console.log(t+":"+A.message)}}));for(let C in I){const i=I[C],t=e[C];i.addHighlight(A+":"+t.toString(),g,!0)}}}}removeHighlight(A,g=!0){super.removeHighlight(A,!1),g&&this.traverse((A=>{A instanceof GeomItem&&A.getId()in this.materialMapping&&(A.materialParam.value=this.materialMapping[A.getId()],delete this.materialMapping[A.getId()])}));const I=this.getOwner().getOwner().getOwner();if(I){const g={},C={},e=this.getParameter("LinkedEntities");if(e){e.getValue().forEach((e=>{if(""==e)return;const i=e.split(", "),t=i.pop();try{const e=I.resolvePath(i);if(e&&e instanceof CADBody)if(0==e.getNumChildren())e.setShatterState(!1),g[e.getId()]||(g[e.getId()]=e,C[e.getId()]=[]),C[e.getId()].push(t);else{const g=e.getChildByName(t);g&&g.removeHighlight(A,!0)}else console.log("linkedEntity.addHighlight(name, color, true):failed")}catch(A){console.log(A.message)}}));for(let I in g){const e=g[I],i=C[I];e.removeHighlight(A+":"+i.toString(),!0)}}}}readBinary(A,g){super.readBinary(A,g),this.traverse((A=>{if(A instanceof GeomItem){const g=A.materialParam.value;"StandardSurfaceShader"==g.getShaderName()&&g.setShaderName("FlatSurfaceShader"),A.opacity=.99}})),g.assetItem.getGeometryLibrary().once("loaded",(()=>{this.traverse((A=>{if(A.getName().startsWith("Text")){const g=[];return A.traverse((A=>{if(A instanceof GeomItem){const I=A.geomParam.value.getBoundingBox();planeMaterial||(planeMaterial=new FlatSurfaceMaterial("plane"),planeMaterial.baseColorParam.value=new Color(1,1,0,.001),planeMaterial.overlayParam.value=-.001);const C=new PMIPickingPlane("plane",plane,planeMaterial),e=A.localXfoParam.value.multiply(A.geomOffsetXfoParam.value);e.tr.addInPlace(I.center()),e.sc.multiplyInPlace(I.diagonal()),C.localXfoParam.value=e,g.push(C)}}),!1),g.forEach((g=>A.addChild(g,!1))),!1}}),!1)}))}}Registry.register("PMIItem",PMIItem);class PMIView extends PMIItem{camera;constructor(A){super(A),this.camera=null}clone(A){const g=new PMIView;return g.copyFrom(this,A),g}activate(){super.activate();let A=[];this.hasParameter("GraphicalElements")&&(A=this.getParameter("GraphicalElements").getValue());const g=(()=>{let A=this;for(;A&&!(A instanceof CADAsset);)A=A.getParentItem();return A instanceof CADAsset?A:null})(),I=this.getParentItem().getParentItem(),C=I.getParentItem();if(C){const g=[];I.traverse((A=>{A instanceof PMIView||A instanceof PMIItem&&g.push(A)})),g.forEach((g=>{const I=A.includes(g.getName());g.setVisible(I)}))}if(this.camera){const A=this.localXfoParam.value.clone(),I=this.getParameter("TargetPoint").getValue().clone(),C=this.getParameter("CameraType").getValue();A.tr.scaleInPlace(g.unitsScale),I.scaleInPlace(g.unitsScale);const e=A.tr.distanceTo(I);if(A.sc.set(1,1,1),this.camera.globalXfoParam.value=A,this.camera.setFocalDistance(e),"Camera_Orthographic"==C&&(this.camera.setIsOrthographic(1,0),this.hasParameter("CameraZoom")&&g)){const A=1/this.getParameter("CameraZoom").getValue()*g.unitsScale*2;this.camera.setFrustumHeight(A)}}if(this.hasParameter("ClippingPlaneOrigin")){const A=this.getParameter("ClippingPlaneOrigin").getValue(),I=this.getParameter("ClippingPlaneNormal").getValue(),e=!0,i=-A.dot(I)*g.unitsScale;C.traverse((A=>{if(A instanceof PMIItem)return!1;A instanceof GeomItem&&(A.setCutawayEnabled(e),A.setCutVector(I),A.setCutDist(i))}))}else C.traverse((A=>{if(A instanceof PMIItem)return!1;A instanceof GeomItem&&A.setCutawayEnabled(!1)}))}deactivate(){super.deactivate()}readBinary(A,g){super.readBinary(A,g),g.camera&&(this.camera=g.camera)}}Registry.register("PMIView",PMIView);class XRef extends CADAsset{configurationParam=new StringParameter("Configuration");constructor(A){super(A),this.addParameter(this.configurationParam)}clone(A){const g=new XRef;return g.copyFrom(this,A),g}readBinary(A,g){let I;if(g.versions["zea-engine"].compare([3,13,1])>=0)TreeItem.prototype.readBinary.call(this,A,g),I=A.loadStr();else{A.loadStr();const C=A.loadStr();this.setName(C),I=A.loadStr();const e=new Xfo;g.versions["zea-cad"].compare([3,6,2])>0?(e.tr=A.loadFloat32Vec3(),e.ori=A.loadFloat32Quat(),this.localXfoParam.value=e):""==C&&this.setName(I),g.versions["zea-engine"].compare([3,11,1])>0&&this.readBinaryParams(A,g)}const C=this.getChildByName("Assembly Features");let e;if(C&&this.removeChildByHandle(C),g.resources?g.resources[I]?e=g.resources[I]:(I.includes("/")?I=I.slice(I.lastIndexOf("/")+1):I.includes("\\")&&(I=I.slice(I.lastIndexOf("\\")+1)),g.resources[I]?e=g.resources[I]:g.xrefLoadCallback&&(e=g.xrefLoadCallback.call(g,I,this))):g.xrefLoadCallback?e=g.xrefLoadCallback.call(g,I,this):(I.includes("/")?I=I.slice(I.lastIndexOf("/")+1):I.includes("\\")&&(I=I.slice(I.lastIndexOf("\\")+1)),e=I.endsWith(".zcad")?g.folder+I:g.folder+I+".zcad"),e){if(g.urlStack.includes(e)){const A=g.urlStack.indexOf(e),I=g.assetStack[A];return void console.warn("Circular load occurring at:",I.path)}g.xrefs[e]?g.addPromise(new Promise(((A,I)=>{const i=g.xrefs[e],t=()=>{if(this.url=i.url,this.sdk=i.sdk,this.engineDataVersion=i.engineDataVersion,this.units=i.units,this.units!=g.units){const A=getUnitsFactor(this.units),I=getUnitsFactor(g.units);this.unitsScale=A/I;const C=this.localXfoParam.value;C.sc.scaleInPlace(this.unitsScale),C.tr.scaleInPlace(this.unitsScale),this.localXfoParam.value=C}const I=i.getChild(0);if(I){this.removeAllChildren();const A=new CloneContext,g=I.clone(A);this.addChild(g,!1,!1)}C&&this.applyAssemblyLevelFeatures(C),A()},o=()=>{this.emit("error"),I()};i.loaded?t():(i.once("xrefLoaded",t),i.once("error",o))}))):(g.xrefs[e]=this,g.addPromise(new Promise(((A,I)=>{this.load(e,new AssetLoadContext(g)).then((()=>{this.emit("xrefLoaded");const g=this.localXfoParam.value;g.tr.scaleInPlace(this.unitsScale),this.localXfoParam.value=g,C&&this.applyAssemblyLevelFeatures(C),A()}),(()=>{I()}))}))))}else if(console.warn(`While Loading ${this.getPath()} unable to load XRef: ${I}`),C){const A=C.getChildren();for(const g of A)this.addChild(g,!1)}}applyAssemblyLevelFeatures(A){const g=A.getChild(0),I=this.getChild(0);g&&I&&this.applyFeature(g,I)}applyFeatureToChildren(A,g){const I=[...A.getChildren()];for(const A of I){const I=g.getChildByName(A.name);I?this.applyFeature(A,I):console.log(`Feature node: ${A.path} does not match a node in the assembly tree for the following XRef:`,this.name)}}applyFeature(A,g){if(A.localXfoParam.value.isIdentity()||(g.localXfoParam.value=A.localXfoParam.value),A instanceof GeomItem){const I=A.materialParam.value;g.traverse((A=>{if(A instanceof GeomItem){A.materialParam.value=I;const g=A.geomParam.value;g instanceof CompoundGeom?g.clearMaterials():A.geomParam.once("valueChanged",(()=>{const g=A.geomParam.value;g instanceof CompoundGeom&&g.clearMaterials()}))}}),!0)}else if(A instanceof CADPart){if(g instanceof CADPart){const I=g.localXfoParam.value,C=g.parent;C.removeChildByHandle(g),C.addChild(A,!1,!1),A.localXfoParam.value=I}else(g instanceof XRef||g instanceof InstanceItem)&&(A.localXfoParam.value=new Xfo,g.removeAllChildren(),g.addChild(A,!1,!1));return}g instanceof XRef&&1==g.getNumChildren()&&(g=g.getChild(0)),g&&this.applyFeatureToChildren(A,g)}}Registry.register("XRef",XRef);const PassType={PRE:1,OPAQUE:2,TRANSPARENT:3,OVERLAY:4};class GLPass extends EventEmitter{enabled=!0;passIndex=-1;__gl=null;renderer=null;__renderer=null;constructor(){super(),this.enabled=!0,this.passIndex=0}init(A,g){if(null==g)throw new Error("Missing constructor argument.");this.__gl=A.gl,this.renderer=A,this.__renderer=A,this.passIndex=g}setPassIndex(A){this.passIndex=A}getPassType(){return PassType.OPAQUE}itemAddedToScene(A,g){throw Error(`${this.constructor.name} must implement itemAddedToScene and itemRemovedFromScene`)}itemRemovedFromScene(A,g){throw Error(`${this.constructor.name} must implement itemAddedToScene and itemRemovedFromScene`)}startPresenting(){}stopPresenting(){}draw(A){throw Error("draw not implemented on GLPass")}drawHighlightedGeoms(A){}drawGeomData(A){}getGeomItemAndDist(A){throw Error("getGeomItemAndDist not implemented on GLPass")}}class GLCADPass extends GLPass{constructor(A=!1){super(),console.warn("GLCADPass is deprecated. No need to install this pass in the renderer.")}itemAddedToScene(A,g){return!1}itemRemovedFromScene(A,g){return!1}}const placeHolderItems=new Map;let disableLoading=!1;class PlaceholderGeomItem extends GeomItem{relativePath;loadContext;parentCache;parentChildIndexCache;loading=!1;constructor(A=""){super(A)}clone(A){const g=new PlaceholderGeomItem;return g.copyFrom(this,A),g}copyFrom(A,g){if(!(A instanceof PlaceholderGeomItem))throw new Error("cannot copy from src");super.copyFrom(A),this.relativePath=A.relativePath,this.loadContext=A.loadContext}readBinary(A,g){super.readBinary(A,g),this.relativePath=A.loadStr(),this.loadContext=g}load(){if(this.loading||disableLoading)return;let A;this.loading=!0;const g=()=>{const A=this.loadContext.xrefs[this.relativePath];let g;this.parentCache=this.parent,this.parentChildIndexCache=this.parentCache.getChildIndex(this),this.parentCache.removeChild(this.parentChildIndexCache),placeHolderItems.has(this.relativePath)?(g=placeHolderItems.get(this.relativePath),g.forEach(((A,g)=>{this.parentCache.insertChild(A.clone(),this.parentChildIndexCache+g,!1)}))):(g=[...A.getChildren()],placeHolderItems.set(this.relativePath,g),g.forEach(((A,g)=>{this.parentCache.insertChild(A,this.parentChildIndexCache+g,!1)})))};if(this.relativePath){const I=this.loadContext.folder+this.relativePath;A=this.loadContext.xrefs[this.relativePath],A||(A=new XRef(this.relativePath),A.load(I),this.loadContext.xrefs[this.relativePath]=A),A.loaded&&A.geomLibrary.loaded?g():A.once("geomsLoaded",g)}}unload(){this.parentCache.removeChild(this.parentChildIndexCache),this.parentCache.insertChild(this,this.parentChildIndexCache),this.loading=!1}static disableLoading(){disableLoading=!0}}Registry.register("Placeholder",PlaceholderGeomItem);class Camera extends TreeItem{isOrthographicParam=new NumberParameter("isOrthographic",0);fovParam=new NumberParameter("fov",1);nearParam=new NumberParameter("near",.1);farParam=new NumberParameter("far",1e3);focalDistanceParam=new NumberParameter("focalDistance",5);adjustNearAndFarPlanesToFocalDist=!0;nearDistFactor=.01;farDistFactor=100;frameOnBoundingSphere=!1;viewHeight=0;intervalId=-1;constructor(A="Camera"){super(A),this.addParameter(this.isOrthographicParam),this.addParameter(this.fovParam),this.addParameter(this.nearParam),this.addParameter(this.farParam),this.addParameter(this.focalDistanceParam);const g=A=>{this.emit("projectionParamChanged",A)};this.isOrthographicParam.on("valueChanged",g),this.fovParam.on("valueChanged",g),this.nearParam.on("valueChanged",g),this.farParam.on("valueChanged",g),this.setPositionAndTarget(new Vec3(3,3,1.75),new Vec3(0,0,1)),this.setLensFocalLength("35mm")}getNear(){return this.nearParam.value}setNear(A){this.nearParam.value=A}getFar(){return this.farParam.value}setFar(A){this.farParam.value=A}getFov(){return this.fovParam.value}setFov(A){this.fovParam.value=A}getFrustumHeight(){return this.viewHeight}setFrustumHeight(A){this.viewHeight=A,this.emit("projectionParamChanged")}setLensFocalLength(A){const g={"10mm":100.4,"11mm":95,"12mm":90,"14mm":81.2,"15mm":77.3,"17mm":70.4,"18mm":67.4,"19mm":64.6,"20mm":61.9,"24mm":53.1,"28mm":46.4,"30mm":43.6,"35mm":37.8,"45mm":29.9,"50mm":27,"55mm":24.6,"60mm":22.6,"70mm":19.5,"75mm":18.2,"80mm":17.1,"85mm":16.1,"90mm":15.2,"100mm":13.7,"105mm":13,"120mm":11.4,"125mm":11,"135mm":10.2,"150mm":9.1,"170mm":8.1,"180mm":7.6,"210mm":6.5,"300mm":4.6,"400mm":3.4,"500mm":2.7,"600mm":2.3,"800mm":1.7};A in g?this.fovParam.value=MathFunctions.degToRad(g[A]):console.warn("Camera lense focal length not supported:"+A)}getFocalDistance(){return this.focalDistanceParam.value}setFocalDistance(A){A<1e-4&&console.error("Never set focal distance to zero"),this.focalDistanceParam.value=A,this.adjustNearAndFarPlanesToFocalDist&&(this.nearParam.value=A*this.nearDistFactor,this.farParam.value=A*this.farDistFactor)}isOrthographic(){return 1==this.isOrthographicParam.value}setIsOrthographic(A,g=0){if(this.intervalId&&clearInterval(this.intervalId),A>.5){const A=this.fovParam.value,g=this.focalDistanceParam.value;this.viewHeight=Math.sin(.5*A)*g*2}if(0==g)this.isOrthographicParam.value=A;else{const I=Math.round(g/20);let C=0;const e=this.isOrthographicParam.value,i=()=>{C++;const g=MathFunctions.lerp(e,A,C/I);this.isOrthographicParam.value=g,C<I?this.intervalId=window.setTimeout(i,20):(this.intervalId=-1,this.emit("movementFinished"))};i()}}setPositionAndTarget(A,g){this.setFocalDistance(A.distanceTo(g));const I=new Xfo;I.setLookAt(A,g,new Vec3(0,0,1)),this.globalXfoParam.value=I,this.emit("movementFinished")}getTargetPosition(){const A=this.focalDistanceParam.value,g=this.globalXfoParam.value,I=g.ori.getZaxis();return I.scaleInPlace(-A),I.addInPlace(g.tr),I}frameView(A,g,I,C=0,e=.1){const i=this.focalDistanceParam.value,t=this.fovParam.value,o=this.isOrthographicParam.value,a=this.globalXfoParam.value.clone(),B=A/g,s=2*Math.atan(Math.tan(.5*t)*B);let l=i;if(this.frameOnBoundingSphere){const A=new Box3;for(const g of I)A.addBox3(g.boundingBoxParam.value);if(!A.isValid())return void console.warn("Bounding box not valid.");const g=a.ori.getZaxis().scale(-i),C=a.tr.add(g),e=A.center().subtract(C);a.tr.addInPlace(e),l=A.size()/Math.tan(t)}else{const A=[];if(I.forEach((g=>{g.traverse((g=>{if(!(g instanceof TreeItem))return!1;if(g.disableBoundingBox)return!1;if(!g.isVisible()||!g.isPickable())return!1;if(g instanceof GeomItem){const I=g.geomParam.value;if(I){const C=I.getBoundingBox();if(C.isValid()){const I=g.geomMatParam.value;return A.push(I.transformVec3(C.p0)),A.push(I.transformVec3(new Vec3(C.p0.x,C.p0.y,C.p1.z))),A.push(I.transformVec3(new Vec3(C.p0.x,C.p1.y,C.p0.z))),A.push(I.transformVec3(new Vec3(C.p1.x,C.p0.y,C.p0.z))),A.push(I.transformVec3(new Vec3(C.p0.x,C.p1.y,C.p1.z))),A.push(I.transformVec3(new Vec3(C.p1.x,C.p0.y,C.p1.z))),A.push(I.transformVec3(new Vec3(C.p1.x,C.p1.y,C.p0.z))),void A.push(I.transformVec3(C.p1))}}}if(0==g.getNumChildren()||g instanceof BaseGroup){const I=g.boundingBoxParam.value;if(I.isValid())return A.push(new Vec3(I.p0.x,I.p0.y,I.p0.z)),A.push(new Vec3(I.p0.x,I.p0.y,I.p1.z)),A.push(new Vec3(I.p0.x,I.p1.y,I.p0.z)),A.push(new Vec3(I.p1.x,I.p0.y,I.p0.z)),A.push(new Vec3(I.p0.x,I.p1.y,I.p1.z)),A.push(new Vec3(I.p1.x,I.p0.y,I.p1.z)),A.push(new Vec3(I.p1.x,I.p1.y,I.p0.z)),void A.push(new Vec3(I.p1.x,I.p1.y,I.p1.z))}}),!0)})),0==A.length)return;const g=o?0:s/2,C=o?0:t/2,i={};i.XPos=new Vec3(Math.cos(g),0,Math.sin(g)),i.XNeg=new Vec3(-Math.cos(g),0,Math.sin(g)),i.YPos=new Vec3(0,Math.cos(C),Math.sin(C)),i.YNeg=new Vec3(0,-Math.cos(C),Math.sin(C)),i.ZPos=new Vec3(0,0,1),i.ZNeg=new Vec3(0,0,-1);const n={},d={};for(const A in i)n[A]=a.ori.rotateVec3(i[A]),d[A]=Number.NEGATIVE_INFINITY;const r=new Vec3;A.forEach(((A,g)=>{if(!Number.isFinite(A.x)||!Number.isFinite(A.y)||!Number.isFinite(A.z))return;const I=A.subtract(a.tr);for(const A in i){const g=I.dot(n[A]);g>d[A]&&g!=Number.POSITIVE_INFINITY&&(d[A]=g)}r.addInPlace(A)}));for(const A in d)if(d[A]==Number.POSITIVE_INFINITY)return;r.scaleInPlace(1/A.length);let c=0;if(o){const A=new Vec3(.5*(-d.XNeg+d.XPos),.5*(-d.YNeg+d.YPos),.5*(-d.ZNeg+d.ZPos)),g=d.ZNeg+d.ZPos;c=2*g,A.z=-d.ZNeg+c,a.tr.addInPlace(a.ori.rotateVec3(A)),l=2*g;const I=d.XPos+d.XNeg,C=d.YPos+d.YNeg;this.viewHeight=Math.max(C,I/B),this.viewHeight+=this.viewHeight*e}else{const A=s/2,g=t/2,I=new Vec2(Math.cos(A)*d.XPos,Math.sin(A)*d.XPos),C=I.add(new Vec2(Math.sin(A),-Math.cos(A))),i=new Vec2(-Math.cos(A)*d.XNeg,Math.sin(A)*d.XNeg),o=i.add(new Vec2(-Math.sin(A),-Math.cos(A))),B=Vec2.intersectionOfLines(I,C,i,o),n=new Vec2(Math.cos(g)*d.YPos,Math.sin(g)*d.YPos),h=n.add(new Vec2(Math.sin(g),-Math.cos(g))),G=new Vec2(-Math.cos(g)*d.YNeg,Math.sin(g)*d.YNeg),m=G.add(new Vec2(-Math.sin(g),-Math.cos(g))),b=Vec2.intersectionOfLines(n,h,G,m);if(null===B||null===b)return void console.warn("xP or yP === null");c=Math.max(B.y,b.y);const Z=new Vec3(B.x,b.x,c);a.tr.addInPlace(a.ori.rotateVec3(Z)),l=r.distanceTo(a.tr);const u=l*e;a.tr.addInPlace(a.ori.rotateVec3(new Vec3(0,0,u))),c+=u}if(this.adjustNearAndFarPlanesToFocalDist){d.ZPos-=c,d.ZNeg+=c;const A=d.ZNeg*this.nearDistFactor,g=-d.ZPos*this.farDistFactor;this.nearParam.value=A,this.farParam.value=g}}if(0==C)this.setFocalDistance(l),this.globalXfoParam.value=a,this.emit("movementFinished");else{this.intervalId>0&&window.clearTimeout(this.intervalId);const A=Math.round(C/20);let g=0;const I=this.focalDistanceParam.value,e=this.globalXfoParam.value,i=()=>{g++;const C=g/A,t=MathFunctions.lerp(I,l,C),o=e.lerp(a,C);this.globalXfoParam.value=o,this.setFocalDistance(t),g<A?this.intervalId=window.setTimeout(i,20):(this.intervalId=-1,this.emit("movementFinished"))};i()}}updateProjectionMatrix(A,g){const I=this.isOrthographicParam.value,C=this.fovParam.value,e=this.nearParam.value,i=this.farParam.value,t=new Mat4;if(I>0){const A=.5*this.viewHeight,I=-A,C=A,o=A*-g,a=A*g;t.setOrthographicMatrix(o,a,I,C,e,i)}I<1&&A.setPerspectiveMatrix(C,g,e,i),1==I?A.setFromMat4(t):I>0&&A.set(MathFunctions.lerp(A.m00,t.m00,I),MathFunctions.lerp(A.m01,t.m01,I),MathFunctions.lerp(A.m02,t.m02,I),MathFunctions.lerp(A.m03,t.m03,I),MathFunctions.lerp(A.m10,t.m10,I),MathFunctions.lerp(A.m11,t.m11,I),MathFunctions.lerp(A.m12,t.m12,I),MathFunctions.lerp(A.m13,t.m13,I),MathFunctions.lerp(A.m20,t.m20,I),MathFunctions.lerp(A.m21,t.m21,I),MathFunctions.lerp(A.m22,t.m22,I),MathFunctions.lerp(A.m23,t.m23,I),MathFunctions.lerp(A.m30,t.m30,I),MathFunctions.lerp(A.m31,t.m31,I),MathFunctions.lerp(A.m32,t.m32,I),MathFunctions.lerp(A.m33,t.m33,I))}}Registry.register("Camera",Camera);class GridTreeItem extends TreeItem{constructor(A=5,g=50,I=new Color("#DCDCDC")){super("GridTree"),this.disableBoundingBox=!0,this.pickableParam.value=!1;const C=new LinesMaterial("gridMaterial");C.baseColorParam.value=I,C.overlayParam.value=0;const e=new Grid(A,A,g,g,!0),i=new GeomItem("GridItem",e,C);i.pickableParam.value=!1,this.addChild(i,!1);const t=new Lines;t.setNumVertices(2),t.setNumSegments(1),t.setSegmentVertexIndices(0,0,1);const o=t.getVertexAttribute("positions");o.setValue(0,new Vec3(-.5*A,0,0)),o.setValue(1,new Vec3(.5*A,0,0));const a=new LinesMaterial("gridXAxisMaterial");a.baseColorParam.value=new Color(I.luminance(),0,0),a.overlayParam.value=0;const B=new GeomItem("xAxisLine",t,a);B.pickableParam.value=!1,this.addChild(B,!1);const s=new LinesMaterial("gridYAxisMaterial");s.baseColorParam.value=new Color(0,I.luminance(),0),s.overlayParam.value=0;const l=new GeomItem("yAxisLine",t,s);l.pickableParam.value=!1;const n=new Xfo;n.ori.setFromAxisAndAngle(new Vec3(0,0,1),.5*Math.PI),l.geomOffsetXfoParam.value=n,this.addChild(l,!1)}cleanBoundingBox(){return new Box3}}Registry.register("GridTreeItem",GridTreeItem);const defaultGridColor=new Color("#DCDCDC");class Scene{envMapParam=new ImageParameter("EnvMap");displayEnvMapParam=new BooleanParameter("Display EnvMap",!1);envMapLODParam=new NumberParameter("EnvMapLOD",0);root=new TreeItem("root");constructor(){}getRoot(){return this.root}getResourceLoader(){return resourceLoader}setEnvMap(A){this.envMapParam.value=A}setupGrid(A=5,g=50,I=defaultGridColor){const C=new GridTreeItem(A,g,I);return this.root.addChild(C,!1),C}}class VLAAsset extends AssetItem{constructor(A){super(A),this.geomLibrary.on("loaded",(()=>{this.emit("geomsLoaded")}))}readBinary(A,g){g.versions["zea-engine"]||(g.versions["zea-mesh"]=new Version(A.loadStr()));const I=A.loadUInt32();return super.readBinary(A,g),g.versions["zea-engine"].compare([2,1,0])<0&&A.loadFloat32Vec2(),this.geomLibrary.setNumGeoms(A.loadUInt32()),I}load(A,g=new AssetLoadContext){return new Promise(((I,C)=>{const e=A.lastIndexOf("/")>-1?A.substring(0,A.lastIndexOf("/"))+"/":"",i=A.lastIndexOf("/")>-1?A.substring(A.lastIndexOf("/")+1):"",t=i.substring(0,i.lastIndexOf("."));let o=0;g.assetItem=this,g.url=A,g.folder=e,resourceLoader.incrementWorkload(1),this.geomLibrary.on("loaded",(()=>{resourceLoader.incrementWorkDone(1)})),resourceLoader.loadFile("archive",A).then((A=>{let C;if(A.tree2)C=new BinReader(A.tree2.buffer,0,SystemDesc.isMobileDevice);else{const I=A.tree?A.tree:A[Object.keys(A)[0]];C=new BinReader(I.buffer,0,SystemDesc.isMobileDevice),g.versions["zea-engine"]=new Version}if(o=this.readBinary(C,g),this.loaded=!0,this.emit("loaded"),0==o&&A.geoms)this.geomLibrary.readBinaryBuffer(-1,A.geoms.buffer,g);else{const A=e+t,I={numGeomsPerFile:o,numGeoms:this.geomLibrary.getNumGeoms()};this.geomLibrary.loadGeomFilesStream(I,A,g)}I()}),(A=>{this.emit("error",A),C(A)}))}))}}Registry.register("VLAAsset",VLAAsset);class ObjAsset extends AssetItem{splitObjects=new BooleanParameter("splitObjects",!1);splitGroupsIntoObjects=new BooleanParameter("splitGroupsIntoObjects",!1);loadMtlFile=new BooleanParameter("loadMtlFile",!0);unitsConversion=new NumberParameter("unitsConversion",1);defaultShader=new StringParameter("defaultShader","");splitObjectsParam=new BooleanParameter("splitObjects",!1);splitGroupsIntoObjectsParam=new BooleanParameter("splitGroupsIntoObjects",!1);loadMtlFileParam=new BooleanParameter("loadMtlFile",!0);unitsConversionParam=new NumberParameter("unitsConversion",1);defaultShaderParam=new StringParameter("defaultShader","");constructor(A){super(A),this.addParameter(this.splitObjectsParam),this.addParameter(this.splitGroupsIntoObjectsParam),this.addParameter(this.loadMtlFileParam),this.addParameter(this.unitsConversionParam),this.addParameter(this.defaultShaderParam)}load(A){return this.loaded=!1,new Promise(((g,I)=>{const C=A.substring(0,A.lastIndexOf("/"))+"/",e=A=>{const g=A.split("\n"),I=/\s+/;let e;const i=function(A){if(3==A.length)return new Color(parseFloat(A[0]),parseFloat(A[1]),parseFloat(A[2]));throw new Error("Unable to parse a color from the following parts:"+A.join("_"))},t=(A,g)=>{const I=new FileImage(A);return I.load(C+g),I};for(let A=0;A<g.length;A++){let C=g[A].trim();if(C.startsWith("#"))continue;C.includes("#")&&(C=C.substring(0,C.indexOf("#")).trim());const o=C.split(I),a=o.shift(),B=o.join(" ");switch(a){case"newmtl":e=new StandardSurfaceMaterial(B),this.materialLibrary.addMaterial(e);break;case"Kd":e.baseColorParam.value=i(o),e.baseColorParam.colorSpace,ColorSpace.Linear;break;case"map_Kd":e.baseColorParam.setImage(t("map_Kd",o[0]));break;case"Ks":const A=(parseFloat(o[0])+parseFloat(o[1])+parseFloat(o[2]))/3;e.roughnessParam.value=1-A,e.reflectanceParam.value=A;break;case"map_Ks":e.roughnessParam.setImage(t("map_Ks",o[0])),e.reflectanceParam.value=.2;break;case"d":e.opacityParam.value=parseFloat(B);break;case"map_d":e.opacityParam.setImage(t("map_Kd",o[0]));break;case"map_bump":e.normalParam.setImage(t("normal",o[0]))}}},i=A=>new Promise((g=>{loadTextfile(A,(A=>{resourceLoader.incrementWorkDone(1),e(A),resourceLoader.incrementWorkDone(1),g()}))})),t=[],o=[],a=[],B={},s=async A=>{const g=A.split("\n"),I=/\s+/;let e,s,l=0;const n=A=>{if(A in B){let g=1;for(;A+String(g)in B;)g++;A+=String(g)}e={verticesRemapping:{},texCoordsRemapping:{},normalsRemapping:{},vertexIndices:[],texCoordIndices:[],normalIndices:[],numVertices:0,numTexCoords:0,numNormals:0,faceCounts:[],material:s},B[A]=e,l++};n("geom");const d=this.splitGroupsIntoObjectsParam.value;for(let A=0;A<g.length;A++){let r=g[A].trim();if(r.startsWith("#"))continue;r.includes("#")&&(r=r.substring(0,r.indexOf("#")).trim());const c=r.split(I),h=c.shift(),G=c.join(" ");switch(h){case"":case"s":continue;case"mtllib":if(!this.loadMtlFileParam.value)continue;resourceLoader.incrementWorkload(2);const A=C+G;A&&await i(A);break;case"o":n(G);break;case"usemtl":s=G,n(G+Object.keys(B).length);break;case"g":d&&n(G?c.join("_"):"Group"+l);break;case"v":t.push(c.map((A=>parseFloat(A))));break;case"vt":a.push(c.map((A=>parseFloat(A))));break;case"vn":o.push(c.map((A=>parseFloat(A))));break;case"f":{const A=[],g=[],I=[];for(let C=0,i=c.length;C<i;C++){const i=c[C].split("/").map((A=>parseInt(A)-1)),t=i[0];let o=e.verticesRemapping[t];if(null==o&&(o=e.numVertices,e.verticesRemapping[t]=o,e.numVertices++),A.push(o),i.length>1&&!isNaN(i[1])){const A=i[1];g.push(A)}if(i.length>2&&!isNaN(i[2])){const A=i[2];I.push(A)}}e.vertexIndices.push(A),I.length>0&&e.normalIndices.push(I),g.length>0&&e.texCoordIndices.push(g),null==e.faceCounts[A.length-3]&&(e.faceCounts[A.length-3]=[]),e.faceCounts[A.length-3]++;break}default:console.warn("Unhandled line:"+r)}}},l=()=>{for(const A in B)0!=B[A].numVertices&&n(A,B[A]);this.emit("loaded"),this.getGeometryLibrary().emit("loaded"),this.emit("geomsLoaded"),g()},n=(A,g)=>{for(let A=0;A<g.faceCounts.length;A++)null==g.faceCounts[A]&&(g.faceCounts[A]=0);const I=g.numVertices,C=new Mesh;C.name=A,C.setFaceCounts(g.faceCounts),C.setNumVertices(I);const e=C.getVertexAttribute("positions"),i=this.unitsConversionParam.value;for(const A in g.verticesRemapping){const I=Number.parseInt(A),C=g.verticesRemapping[I];e.setValue(C,new Vec3(t[I][0]*i,t[I][1]*i,t[I][2]*i))}let B,s;g.normalIndices.length>0&&(B=new Vec3Attribute,C.addVertexAttribute("normals",B)),g.texCoordIndices.length>0&&(s=new Vec2Attribute,C.addVertexAttribute("texCoords",s));const l=Array(g.faceCounts.length).fill(0);for(let A=0;A<g.vertexIndices.length;A++){const I=g.vertexIndices[A];let e=0;for(let A=0;A<I.length-3;++A)g.faceCounts[A]&&(e+=g.faceCounts[A]);if(e+=l[I.length-3],l[I.length-3]++,C.setFaceVertexIndices(e,I),B){const I=g.normalIndices[A];for(let A=0;A<I.length;A++){const g=new Vec3(o[I[A]][0],o[I[A]][1],o[I[A]][2]);B.setFaceVertexValue(e,A,g)}}if(s&&g.texCoordIndices.length==g.vertexIndices.length){const I=g.texCoordIndices[A];for(let A=0;A<I.length;A++){const g=new Vec2(a[I[A]][0],a[I[A]][1]);s.setFaceVertexValue(e,A,g)}}}const n=new GeomItem(A,C),d=C.getBoundingBox().center();{const A=d.negate(),g=C.getVertexAttribute("positions");for(let I=0;I<g.getCount();I++)g.setValue(I,g.getValue(I).add(A));C.setBoundingBoxDirty()}if(n.localXfoParam.value=new Xfo(d),null!=g.material&&this.materialLibrary.hasMaterial(g.material))n.materialParam.value=this.materialLibrary.getMaterial(g.material);else{const g=this.defaultShaderParam.value,I=new Material(A+" mat");I.setShaderName(""!=g?g:"StandardSurfaceShader"),this.materialLibrary.addMaterial(I),n.materialParam.value=I}this.addChild(n,!1)};(()=>{resourceLoader.incrementWorkload(2),loadTextfile(A,(A=>{resourceLoader.incrementWorkDone(1),s(A).then((()=>{l(),resourceLoader.incrementWorkDone(1)}))}),(A=>{this.emit("error",A),I(A)}))})()}))}}class BaseTool extends ParameterOwner{__activated=!1;constructor(){super()}activateTool(){this.__activated&&console.warn("Tool already active"),this.__activated=!0,this.emit("activatedChanged",{activated:this.__activated})}deactivateTool(){this.__activated=!1,this.emit("activatedChanged",{activated:this.__activated})}onPointerDown(A){}onPointerMove(A){}onPointerUp(A){}onPointerClick(A){}onPointerDoubleClick(A){}onPointerLongPress(A){}onPointerEnter(A){}onPointerLeave(A){}onWheel(A){}onKeyDown(A){}onKeyUp(A){}onTouchCancel(A){}}const MANIPULATION_MODES={pan:0,dolly:1,zoom:2,look:3,turntable:4,tumbler:5,trackball:6};class CameraManipulator extends BaseTool{appData;orbitAroundCursor=!1;zoomTowardGeomUnderCursor=!1;aimFocusOnTouchTap=2;aimFocusOnMouseClick=2;enabledWASDWalkMode=!1;defaultManipulationState=MANIPULATION_MODES.turntable;prevCursor;manipulationState;pointerDown=!1;dragging=0;keyboardMovement=!1;keysPressed=[];velocity=new Vec3;prevVelocityIntegrationTime=-1;ongoingTouches={};orbitTarget;prevMousePos;focusIntervalId;mouseWheelMovementDist=0;mouseWheelZoomCount=0;mouseWheelZoomId=-1;orbitRateParam=new NumberParameter("OrbitRate",SystemDesc.isMobileDevice?.5:1);dollySpeedParam=new NumberParameter("DollySpeed",.001);mouseWheelDollySpeedParam=new NumberParameter("MouseWheelDollySpeed",.1);walkSpeedParam=new NumberParameter("WalkSpeed",5);walkModeCollisionDetection=new BooleanParameter("WalkModeCollisionDetection",!1);constructor(A){super(),this.appData=A,this.defaultManipulationState=MANIPULATION_MODES.turntable,this.manipulationState=this.defaultManipulationState,this.addParameter(this.orbitRateParam),this.addParameter(this.dollySpeedParam),this.addParameter(this.mouseWheelDollySpeedParam),this.addParameter(this.walkSpeedParam),this.addParameter(this.walkModeCollisionDetection)}activateTool(){super.activateTool(),this.appData&&this.appData.renderer&&(this.prevCursor=this.appData.renderer.getGLCanvas().style.cursor,this.appData.renderer.getGLCanvas().style.cursor="cursor")}deactivateTool(){super.deactivateTool(),this.appData&&this.appData.renderer&&(this.appData.renderer.getGLCanvas().style.cursor=this.prevCursor)}setDefaultManipulationMode(A){if(this.defaultManipulationState="string"==typeof A?MANIPULATION_MODES[A]:A,!Object.values(MANIPULATION_MODES).includes(this.defaultManipulationState))throw new Error("Invalid Camera Manipulation Mode. Must be one of "+Object.keys(MANIPULATION_MODES))}look(A,g){const{viewport:I}=A,C=I.getCamera(),e=this.orbitRateParam.value,i=C.globalXfoParam.value,t=new Quat;t.rotateZ(g.x/I.getWidth()*Math.PI*e),i.ori=t.multiply(i.ori);const o=new Quat;o.rotateX(g.y/I.getHeight()*Math.PI*e),i.ori.multiplyInPlace(o),C.globalXfoParam.value=i}turntable(A,g){const{viewport:I}=A,C=I.getCamera(),e=this.orbitRateParam.value,i=C.globalXfoParam.value,t=i.ori.inverse().rotateVec3(i.tr.subtract(this.orbitTarget)),o=new Quat;o.rotateZ(g.x/I.getWidth()*2*Math.PI*-e),i.ori=o.multiply(i.ori);const a=new Quat;a.rotateX(g.y/I.getHeight()*Math.PI*-e),i.ori.multiplyInPlace(a),i.tr=this.orbitTarget.add(i.ori.rotateVec3(t)),C.globalXfoParam.value=i}tumbler(A,g){const{viewport:I}=A,C=I.getCamera(),e=this.orbitRateParam.value,i=C.globalXfoParam.value,t=i.ori.getXaxis(),o=i.ori.getYaxis(),a=i.ori.getZaxis(),B=t.scale(-g.x).add(o.scale(g.y)).cross(a);B.normalizeInPlace();const s=g.length(),l=i.ori.inverse().rotateVec3(i.tr.subtract(this.orbitTarget)),n=new Quat;n.setFromAxisAndAngle(B,s/I.getWidth()*Math.PI*-e),i.ori=n.multiply(i.ori),i.tr=this.orbitTarget.add(i.ori.rotateVec3(l)),C.globalXfoParam.value=i}trackball(A,g){const{viewport:I}=A,C=I.getCamera(),e=this.orbitRateParam.value,i=C.globalXfoParam.value,t=i.ori.getXaxis(),o=i.ori.getYaxis(),a=i.ori.getZaxis(),B=t.scale(-g.x).add(o.scale(g.y)).cross(a);B.normalizeInPlace();const s=g.length(),l=i.ori.inverse().rotateVec3(i.tr.subtract(this.orbitTarget)),n=new Quat;n.setFromAxisAndAngle(B,s/I.getWidth()*Math.PI*-e),i.ori=n.multiply(i.ori),i.tr=this.orbitTarget.add(i.ori.rotateVec3(l)),C.globalXfoParam.value=i}pan(A,g){const{viewport:I}=A,C=I.getCamera(),e=new Xfo,i=new Vec3(1,0,0),t=new Vec3(0,1,0);if(C.isOrthographic()){const A=C.getFrustumHeight(),o=A*(I.getWidth()/I.getHeight());e.tr=i.scale(-g.x/I.getWidth()*o),e.tr.addInPlace(t.scale(g.y/I.getHeight()*A))}else{const A=C.getFocalDistance(),o=C.getFov(),a=2*A*Math.tan(.5*o),B=a*(I.getWidth()/I.getHeight());e.tr=i.scale(-g.x/I.getWidth()*B),e.tr.addInPlace(t.scale(g.y/I.getHeight()*a))}const o=C.globalXfoParam.value;C.globalXfoParam.value=o.multiply(e)}dolly(A,g,I){const C=A.viewport,e=C.getCamera(),i=e.getFocalDistance(),t=new Vec2(.5*C.getWidth(),.5*C.getHeight()),o=g.subtract(t).normalize(),a=I.dot(o),B=()=>{const A=a*this.dollySpeedParam.value*i,g=new Xfo;g.tr.set(0,0,A);const I=e.globalXfoParam.value;e.globalXfoParam.value=I.multiply(g)},s=()=>{const A=a*this.dollySpeedParam.value,g=e.getFrustumHeight(),I=g*A;e.setFrustumHeight(g+I)};e.isOrthographic()?s():B()}zoom(A,g,I){const C=A.viewport,e=C.getCamera(),i=e.getFocalDistance(),t=new Vec2(.5*C.getWidth()/window.devicePixelRatio,.5*C.getHeight()/window.devicePixelRatio),o=g.subtract(t).normalize(),a=I.dot(o),B=()=>{const A=-a*this.dollySpeedParam.value*i,g=new Xfo;g.tr.set(0,0,A);const I=e.globalXfoParam.value;e.setFocalDistance(i+A),e.globalXfoParam.value=I.multiply(g)},s=()=>{const A=-a*this.dollySpeedParam.value,g=e.getFrustumHeight(),I=g*A;e.setFrustumHeight(g+I)};e.isOrthographic()?s():B()}initDrag(A){this.pointerDown=!0;const g=A.viewport.getCamera(),I=g.globalXfoParam.value;if(this.orbitAroundCursor)if(null!=A.intersectionData&&this.orbitAroundCursor){this.orbitTarget=A.intersectionData.intersectionPos;const C=I.inverse().transformVec3(A.intersectionData.intersectionPos);g.setFocalDistance(-C.z)}else this.orbitTarget||(A.pointerRay?this.orbitTarget=A.pointerRay.pointAtDist(g.getFocalDistance()):this.orbitTarget=I.tr.add(I.ori.getZaxis().scale(-g.getFocalDistance())));else this.orbitTarget=I.tr.add(I.ori.getZaxis().scale(-g.getFocalDistance()));this.dragging=1}endDrag(A){A.getCapture()==this&&A.releaseCapture(),this.dragging=0,this.pointerDown=!1}aimFocus(A,g,I=-1,C=400){this.focusIntervalId&&clearInterval(this.focusIntervalId);const e=Math.round(C/20),i=this.defaultManipulationState;let t=0;const o=()=>{const C=A.globalXfoParam.value,a=A.getFocalDistance(),B=g.subtract(C.tr),s=B.normalizeInPlace(),l=C.clone();if(i==MANIPULATION_MODES.turntable||i==MANIPULATION_MODES.look){{const A=C.ori.getZaxis().clone();A.z=0;const g=B.negate();g.z=0;const I=new Quat;I.setFrom2Vectors(A,g),l.ori=I.multiply(l.ori)}{const A=C.ori.getXaxis().clone(),g=C.ori.getZaxis().clone(),I=B.negate();I.subtractInPlace(A.scale(I.dot(A))),I.normalizeInPlace();const e=new Quat;g.cross(I).dot(A)>0?e.rotateX(g.angleTo(I)):e.rotateX(-g.angleTo(I)),l.ori=l.ori.multiply(e)}{const A=l.ori.getXaxis().clone(),g=A.clone();g.z=0,g.normalizeInPlace();const I=new Quat;I.setFrom2Vectors(A,g),l.ori=I.multiply(l.ori)}}else{const A=C.ori.getZaxis().clone(),g=B.negate(),I=new Quat;I.setFrom2Vectors(A,g),l.ori=I.multiply(l.ori)}const n=Math.pow(t/e,2),d=C.clone();if(d.ori=C.ori.lerp(l.ori,n),I>0){const A=B.scale(s-I);d.tr.addInPlace(A.scale(n))}A.setFocalDistance(a+(s-a)*n),A.globalXfoParam.value=d,t++,t<=e?this.focusIntervalId=setTimeout(o,20):(this.focusIntervalId=void 0,this.emit("movementFinished"),A.emit("movementFinished"))};o()}orientPointOfView(A,g,I,C=0,e=400){this.focusIntervalId&&clearInterval(this.focusIntervalId);const i=Math.round(e/20);let t=0;const o=()=>{const e=A.globalXfoParam.value,a=A.getTargetPosition(),B=Math.pow(t/i,2),s=g.subtract(e.tr),l=s.normalizeInPlace(),n=s.scale(l-C),d=e.tr.add(n.scale(B)),r=a.lerp(I,B);A.setPositionAndTarget(d,r),t++,t<=i?this.focusIntervalId=setTimeout(o,20):(this.focusIntervalId=void 0,this.emit("movementFinished"),A.emit("movementFinished"))};o()}onPointerDoubleClick(A){const g=g=>{const I=A.viewport.getCamera(),C=I.globalXfoParam.value.tr.add(g.dir.scale(A.intersectionData.dist));this.aimFocus(I,C),A.aimTarget=C,A.aimDistance=A.intersectionData.dist,this.emit("aimingFocus",A),I.emit("aimingFocus",A),A.stopPropagation()};if(A.intersectionData&&this.aimFocusOnMouseClick){if(A instanceof ZeaMouseEvent&&2==this.aimFocusOnMouseClick){const I=A;g(I.pointerRay),I.preventDefault()}if(A instanceof ZeaTouchEvent&&2==this.aimFocusOnTouchTap){g(A.pointerRay)}}}onPointerDown(A){if(A instanceof ZeaMouseEvent){1==this.dragging&&this.endDrag(A),this.initDrag(A);const g=A;this.prevMousePos=g.pointerPos,2==g.button?this.manipulationState=MANIPULATION_MODES.pan:g.ctrlKey&&g.altKey?this.manipulationState=MANIPULATION_MODES.dolly:g.ctrlKey||2==g.button?this.manipulationState=MANIPULATION_MODES.look:this.manipulationState=this.defaultManipulationState,g.preventDefault()}else A instanceof ZeaTouchEvent&&this._onTouchStart(A)}onPointerMove(A){0!=this.dragging&&(A instanceof ZeaMouseEvent&&(this._onMouseMove(A),A.preventDefault()),A instanceof ZeaTouchEvent&&this._onTouchMove(A),this.dragging=2,A.setCapture(this),A.stopPropagation())}_onMouseMove(A){if(!this.pointerDown)return;const g=A.pointerPos,I=g.subtract(this.prevMousePos);switch(this.manipulationState){case MANIPULATION_MODES.turntable:this.turntable(A,I);break;case MANIPULATION_MODES.tumbler:this.tumbler(A,I);break;case MANIPULATION_MODES.trackball:this.trackball(A,I);break;case MANIPULATION_MODES.look:this.look(A,I);break;case MANIPULATION_MODES.pan:this.pan(A,g.subtract(this.prevMousePos));break;case MANIPULATION_MODES.dolly:this.dolly(A,g,I);break;case MANIPULATION_MODES.zoom:this.zoom(A,g,I)}this.prevMousePos=g}_onTouchMove(A){const g=A.touches;if(1==g.length){const I=g[0],C=I.touchPos,e=this.ongoingTouches[I.identifier];if(!e)return;const i=C.subtract(e.pos);switch(this.defaultManipulationState){case MANIPULATION_MODES.look:i.scaleInPlace(6),this.look(A,i);break;case MANIPULATION_MODES.turntable:this.turntable(A,i);break;case MANIPULATION_MODES.tumbler:this.tumbler(A,i);break;case MANIPULATION_MODES.trackball:this.trackball(A,i);break;case MANIPULATION_MODES.pan:this.pan(A,i);break;case MANIPULATION_MODES.dolly:this.dolly(A,C,i);break;case MANIPULATION_MODES.zoom:this.zoom(A,e.pos,i)}e.pos=C}else if(2==g.length){const I=g[0],C=this.ongoingTouches[I.identifier],e=g[1],i=this.ongoingTouches[e.identifier];if(!C||!i)return;const t=I.touchPos,o=e.touchPos,a=i.pos.subtract(C.pos).length()-o.subtract(t).length(),B=t.subtract(C.pos),s=o.subtract(i.pos),l=B.add(s);l.scaleInPlace(.5);const n=.002*a,{viewport:d}=A,r=d.getCamera(),c=r.getFocalDistance(),h=r.getFov(),G=new Vec3(1,0,0),m=new Vec3(0,1,0),b=2*c*Math.tan(.5*h),Z=b*(d.getWidth()/d.getHeight()),u=new Xfo;u.tr=G.scale(-l.x/d.getWidth()*Z),u.tr.addInPlace(m.scale(l.y/d.getHeight()*b));const y=n*c;switch(r.setFocalDistance(c+y),u.tr.z+=y,this.defaultManipulationState){case MANIPULATION_MODES.tumbler:case MANIPULATION_MODES.trackball:const A=i.pos.subtract(C.pos),g=o.subtract(t);let I=A.normalize().angleTo(g.normalize());A.cross(g)<0&&(I=-I);const e=new Quat;e.rotateZ(I),u.ori.multiplyInPlace(e)}const E=r.globalXfoParam.value;r.globalXfoParam.value=E.multiply(u),C.pos=t,i.pos=o}}onPointerUp(A){if(1==this.dragging){if(this.endDrag(A),A.intersectionData&&(A instanceof ZeaMouseEvent&&1==this.aimFocusOnMouseClick||A instanceof ZeaTouchEvent&&1==this.aimFocusOnTouchTap)){const g=A.viewport.getCamera(),I=g.globalXfoParam.value,C=A.pointerRay,e=I.tr.add(C.dir.scale(A.intersectionData.dist));this.aimFocus(g,e),A.aimTarget=e,A.aimDistance=A.intersectionData.dist,this.emit("aimingFocus",A),g.emit("aimingFocus",A),A.stopPropagation(),A instanceof ZeaMouseEvent&&A.preventDefault()}}else if(2==this.dragging){if(A instanceof ZeaMouseEvent){this.endDrag(A),A.releaseCapture(),this.emit("movementFinished");A.viewport.getCamera().emit("movementFinished")}else if(A instanceof ZeaTouchEvent){const g=A,{changedTouches:I,touches:C}=g;for(let A=0;A<I.length;A++)this.__endTouch(I[A]);if(0==Object.keys(this.ongoingTouches).length){this.endDrag(A),A.releaseCapture();A.viewport.getCamera().emit("movementFinished")}g.preventDefault()}A.releaseCapture(),A.stopPropagation()}}onPointerLeave(A){this.keysPressed.length>0&&(this.keysPressed=[],this.velocity.set(0,0,0),this.keyboardMovement=!1)}onWheel(A){const g=A.viewport.getCamera(),I=this.mouseWheelDollySpeedParam.value,C=A.shiftKey?.1:.5,e=g.globalXfoParam.value;let i;if(!g.isOrthographic())if(this.zoomTowardGeomUnderCursor)if(null!=A.intersectionData){i=e.tr.subtract(A.intersectionData.intersectionPos),i.normalizeInPlace();const I=e.inverse().transformVec3(A.intersectionData.intersectionPos);g.setFocalDistance(-I.z)}else{const I=A.pointerRay.pointAtDist(g.getFocalDistance());i=e.tr.subtract(I),i.normalizeInPlace()}else i=g.globalXfoParam.value.ori.getZaxis();const t=A.deltaY<0?-1:1,o=()=>{const A=g.getFocalDistance(),I=A*this.mouseWheelMovementDist;e.tr.addInPlace(i.scale(I)),g.setFocalDistance(A+I),g.globalXfoParam.value=e,this.mouseWheelZoomCount++,this.mouseWheelZoomCount<6?this.mouseWheelZoomId=window.setTimeout(o,10):(this.mouseWheelZoomId=-1,this.emit("movementFinished"),g.emit("movementFinished"))},a=()=>{const A=50*this.mouseWheelMovementDist;(new Xfo).tr.set(0,0,A),e.tr.addInPlace(i.scale(A)),g.globalXfoParam.value=e},B=()=>{const I=g.getFrustumHeight(),C=I*this.mouseWheelMovementDist;if(g.setFrustumHeight(I+C),A.intersectionData&&this.zoomTowardGeomUnderCursor){const i=e.tr.subtract(A.intersectionData.intersectionPos),t=e.ori.getZaxis();i.subtractInPlace(t.scale(i.dot(t))),e.tr.addInPlace(i.scale(C/(I+C))),g.globalXfoParam.value=e}this.mouseWheelZoomCount++,this.mouseWheelZoomCount<6?this.mouseWheelZoomId=window.setTimeout(B,10):(this.mouseWheelZoomId=-1,this.emit("movementFinished"),g.emit("movementFinished"))};this.mouseWheelZoomId>0?(this.mouseWheelMovementDist+=t*I*C*.5/6,this.mouseWheelZoomCount=0):(this.mouseWheelMovementDist=t*I*C/6,this.mouseWheelZoomCount=0,g.isOrthographic()?B():A.ctrlKey?a():o()),A.preventDefault(),A.stopPropagation()}integrateVelocityChange(A){const{viewport:g}=A,I=g.getCamera(),C=performance.now();if(this.prevVelocityIntegrationTime>0){const A=(C-this.prevVelocityIntegrationTime)/1e3,e=this.walkSpeedParam.value;if(e>0){const C=new Xfo;C.tr=this.velocity.normalize().scale(e*A);const i=I.globalXfoParam.value.multiply(C);if(this.walkModeCollisionDetection.value){const A=1.5,I=1.5,C=.5,e=new Xfo(i.tr),t=new Ray(i.tr,new Vec3(0,0,-1)),o=g.getRenderer().raycastCluster(e,t,I,C,PassType.OPAQUE);if(o.length>0){let g=0;o.forEach((A=>{g+=A.dist})),g/=o.length,i.tr=t.start.add(t.dir.scale(g-A))}}I.globalXfoParam.value=i}}this.prevVelocityIntegrationTime=C}onKeyDown(A){if(!this.enabledWASDWalkMode)return;const g=A.key.toLowerCase();if(!this.keysPressed.includes(g)){switch(g){case"w":this.velocity.z-=1;break;case"s":this.velocity.z+=1;break;case"a":this.velocity.x-=1;break;case"d":this.velocity.x+=1;break;default:return}if(A.stopPropagation(),this.keysPressed.push(g),!this.keyboardMovement){this.keyboardMovement=!0,this.prevVelocityIntegrationTime=performance.now();const g=()=>{this.integrateVelocityChange(A),this.keyboardMovement&&window.requestAnimationFrame(g)};window.requestAnimationFrame(g)}}}onKeyUp(A){const g=A.key.toLowerCase();if(!this.keysPressed.includes(g))return;switch(g){case"w":this.velocity.z+=1;break;case"s":this.velocity.z-=1;break;case"a":this.velocity.x+=1;break;case"d":this.velocity.x-=1;break;default:return}A.stopPropagation();const I=this.keysPressed.indexOf(g);this.keysPressed.splice(I,1),0==this.keysPressed.length&&(this.keyboardMovement=!1)}__startTouch(A){this.ongoingTouches[A.identifier]={identifier:A.identifier,pos:A.touchPos}}__endTouch(A){delete this.ongoingTouches[A.identifier]}_onTouchStart(A){const g=A.changedTouches;for(let A=0;A<g.length;A++)this.__startTouch(g[A]);this.initDrag(A)}onTouchEnd(A){A.preventDefault(),A.stopPropagation();const g=A.changedTouches;for(let A=0;A<g.length;A++)this.__endTouch(g[A]);0==Object.keys(this.ongoingTouches).length&&this.endDrag(A)}onTouchCancel(A){A.preventDefault();const g=A.touches;for(let A=0;A<g.length;A++)this.__endTouch(g[A]);0==Object.keys(this.ongoingTouches).length&&this.endDrag(A)}static get MANIPULATION_MODES(){return MANIPULATION_MODES}}var GeomType$2;!function(A){A[A.TRIANGLES=0]="TRIANGLES",A[A.LINES=1]="LINES",A[A.POINTS=2]="POINTS"}(GeomType$2||(GeomType$2={}));const create3DContext=function(A,g){let I=null;if(null!=g.webglContextType)try{I=A.getContext(g.webglContextType,g),I.name=g.webglContextType}catch(A){}else{const C=["webgl2","webgl"];for(let e=0;e<C.length;e++){const i=C[e];try{I=A.getContext(i,g),I.name=i}catch(A){}if(I)break}}if(!I)throw new Error("WebGL not supported on your system");return I.sizeInBytes=function(A){switch(A){case this.BYTE:case this.UNSIGNED_BYTE:return 1;case this.SHORT:case this.UNSIGNED_SHORT:return 2;case this.INT:case this.UNSIGNED_INT:case this.FLOAT:return 4;default:throw new Error("unknown type")}},I.floatTexturesSupported=!0,I.__ext_texture_float=I.getExtension("OES_texture_float"),I.__ext_float_linear=I.getExtension("OES_texture_float_linear"),I.__ext_texture_half_float=I.getExtension("OES_texture_half_float"),I.__ext_texture_half_float_linear=I.getExtension("OES_texture_half_float_linear"),I.__ext_color_buffer_float=I.getExtension("EXT_color_buffer_float"),I.setupInstancedQuad=function(){const A=new Float32Array([0,1,2,3]),g=new Uint8Array([0,1,2,2,1,3]);this.__quadVertexIdsBuffer=this.createBuffer(),this.bindBuffer(this.ARRAY_BUFFER,this.__quadVertexIdsBuffer),this.bufferData(this.ARRAY_BUFFER,A,this.STATIC_DRAW),this.__quadIndexBuffer=this.createBuffer(),this.bindBuffer(this.ELEMENT_ARRAY_BUFFER,this.__quadIndexBuffer),this.bufferData(this.ELEMENT_ARRAY_BUFFER,g,this.STATIC_DRAW),this.__quadattrbuffers={vertexIDs:{buffer:this.__quadVertexIdsBuffer,dataType:I.FLOAT,dimension:1,elementSize:4,count:A.length,shared:!0}}},I.drawQuad=function(){this.drawElements(this.TRIANGLES,6,this.UNSIGNED_BYTE,0)},I},processTextureParams=function(A,g){if(!g.width||!g.height){if(!g.width)throw new Error("Invalid texture params. 'width' not provided");if(!g.height)throw new Error("Invalid texture params. 'height' not provided")}const I=A.getParameter(A.MAX_TEXTURE_SIZE);if(g.width<=0||g.width>I||g.height<=0||g.height>I)throw new Error("GLTextureParams: Invalid texture size. width:"+g.width+" height:"+g.height+" maxSize:"+I);const C={width:g.width,height:g.height},e=g=>isNaN(g)?A[g]:g,i=(A,I)=>{A in g?C[A]=e(g[A]):I&&(C[A]=e(I))};if(i("format"),i("internalFormat",C.format),i("type",A.UNSIGNED_BYTE),i("minFilter",g.filter?g.filter:A.LINEAR),i("magFilter",g.filter?g.filter:A.LINEAR),i("wrapS",g.wrapS?g.wrapS:A.CLAMP_TO_EDGE),i("wrapT",g.wrapT?g.wrapT:A.CLAMP_TO_EDGE),i("mipMapped",!1),i("depthInternalFormat"),i("depthFormat"),i("depthType"),g.createDepthTexture&&("webgl2"==A.name||A.__ext_WEBGL_depth_texture?(C.depthFormat=A.DEPTH_COMPONENT,C.depthType=A.UNSIGNED_INT):C.depthType=A.UNSIGNED_SHORT),C.format==A.FLOAT)C.filter!=A.LINEAR||A.__ext_float_linear||(console.warn("Floating point texture filtering not supported on result device"),C.filter=A.NEAREST);else if(C.format==A.HALF_FLOAT);else if("sRGB"==C.format&&!A.__ext_sRGB)throw new Error("EXT_sRGB is not available");return null!=C.format&&C.internalFormat==C.format&&(C.type==A.FLOAT?C.format==A.RED?C.internalFormat=A.R32F:C.format==A.RG?C.internalFormat=A.RG32F:C.format==A.RGB?C.internalFormat=A.RGB32F:C.format==A.RGBA&&(C.internalFormat=A.RGBA32F):C.type==A.HALF_FLOAT?C.format==A.RED?C.internalFormat=A.R16F:C.format==A.RGB?C.internalFormat=A.RGB16F:C.format==A.RGBA&&(C.internalFormat=A.RGBA16F):C.type==A.UNSIGNED_BYTE&&(C.format==A.RED&&(C.internalFormat=A.R8),C.format==A.RGB?C.internalFormat=A.RGB8:C.format==A.RGBA&&(C.internalFormat=A.RGBA8))),null!=C.depthFormat&&(C.depthType==A.UNSIGNED_SHORT?C.depthInternalFormat=A.DEPTH_COMPONENT16:C.depthType==A.UNSIGNED_INT&&(C.depthInternalFormat=A.DEPTH_COMPONENT24)),C};class GLTexture2D extends RefCounted{gl;width=0;height=0;textureType;textureDesc;loaded;image=null;internalFormat=0;format=0;type=0;minFilter=0;magFilter=0;wrapS=0;wrapT=0;mipMapped=!1;invert=!1;gltex=null;constructor(A,g){if(super(),this.gl=A,this.textureType=1,this.textureDesc=[0,0,0,0],this.loaded=!1,null!=g)if(g instanceof BaseImage){this.image=g;const A=()=>{const A=this.image.getParams(),g=A.width,I=A.height,C=A.data;this.bufferData(C,g,I)};this.image.on("updated",A),this.image.isLoaded()?this.configure(this.image.getParams()):this.image.on("loaded",(()=>{this.configure(this.image.getParams())}))}else this.configure(g)}configure(A){const g=this.gl;{const I=processTextureParams(g,A);this.format=I.format,this.internalFormat=I.internalFormat,this.type=I.type,this.minFilter=I.minFilter,this.magFilter=I.magFilter,this.wrapS=I.wrapS,this.wrapT=I.wrapT,this.mipMapped="mipMapped"in A&&A.mipMapped}this.textureType=1,this.textureDesc[0]=this.width,this.textureDesc[1]=this.height,1==this.textureType&&this.format==g.RGBA&&(this.textureType=2),this.gltex&&(g.deleteTexture(this.gltex),this.width=0,this.height=0),this.gltex=g.createTexture(),this.updateGLTexParams();const I=A.data;I?this.bufferData(I,A.width,A.height,!1,!1):this.resize(A.width,A.height,!1,!1),this.loaded||(this.emit("ready"),this.loaded=!0)}activateLastTextureUnit(){const A=this.gl,g=A.getParameter(A.MAX_TEXTURE_IMAGE_UNITS);A.activeTexture(A.TEXTURE0+g-1)}updateGLTexParams(){const A=this.gl;this.activateLastTextureUnit(),A.bindTexture(A.TEXTURE_2D,this.gltex),A.texParameteri(A.TEXTURE_2D,A.TEXTURE_MAG_FILTER,this.minFilter),A.texParameteri(A.TEXTURE_2D,A.TEXTURE_MIN_FILTER,this.magFilter),A.texParameteri(A.TEXTURE_2D,A.TEXTURE_WRAP_S,this.wrapS),A.texParameteri(A.TEXTURE_2D,A.TEXTURE_WRAP_T,this.wrapT)}bufferData(A,g=-1,I=-1,C=!0,e=!0){const i=this.gl;if(null!=A){if(A instanceof WebGLTexture)this.gltex=A;else if(A instanceof HTMLImageElement||A instanceof ImageData||A instanceof HTMLCanvasElement||A instanceof HTMLImageElement||A instanceof HTMLVideoElement)C&&(this.activateLastTextureUnit(),i.bindTexture(i.TEXTURE_2D,this.gltex)),i.texImage2D(i.TEXTURE_2D,0,this.internalFormat,this.format,this.type,A),this.width=A.width,this.height=A.height;else{-1==g&&(g=this.width),-1==I&&(I=this.height);const e=g*I;let t;switch(this.format){case i.RED:case i.RED_INTEGER:case i.ALPHA:case i.LUMINANCE:case i.LUMINANCE_ALPHA:t=1;break;case i.RG:t=2,i.pixelStorei(i.UNPACK_ALIGNMENT,2);break;case i.RGB:case i.RGB_INTEGER:t=3;break;case i.RGBA:case i.RGBA_INTEGER:t=4;break;default:console.warn("Reaching default case: numChannels:=1"),t=1}A.length!=e*t&&console.warn("Invalid data for Image width:"+g+" height:"+I+" Data Length:"+A.length+" Expected:"+e*t);let o=A;this.type==i.HALF_FLOAT&&A instanceof Float32Array&&(o=MathFunctions.convertFloat32ArrayToUInt16Array(A)),C&&(this.activateLastTextureUnit(),i.bindTexture(i.TEXTURE_2D,this.gltex)),i.texImage2D(i.TEXTURE_2D,0,this.internalFormat,g,I,0,this.format,this.type,o,0),this.width=g,this.height=I}this.mipMapped&&i.generateMipmap(i.TEXTURE_2D)}else C&&(this.activateLastTextureUnit(),i.bindTexture(i.TEXTURE_2D,this.gltex)),i.texImage2D(i.TEXTURE_2D,0,this.internalFormat,this.width,this.height,0,this.format,this.type,null),this.width=g,this.height=I;e&&this.emit("updated")}clear(){const A=this.gl,g=this.width*this.height;let I,C;switch(this.format){case A.RED:case A.RED_INTEGER:case A.ALPHA:case A.LUMINANCE:case A.LUMINANCE_ALPHA:I=1;break;case A.RG:I=2;break;case A.RGB:I=3;break;case A.RGBA:I=4;break;default:throw new Error("Invalid Format")}switch(this.type){case A.UNSIGNED_BYTE:C=new Uint8Array(g*I);break;case A.HALF_FLOAT:C=new Uint16Array(g*I);break;case A.FLOAT:C=new Float32Array(g*I);break;default:throw new Error("Invalid Type")}A.texImage2D(A.TEXTURE_2D,0,this.internalFormat,this.width,this.height,0,this.format,this.type,C,0)}resize(A,g,I=!1,C=!0){const e=this.gl;if(this.width!=A||this.height!=g){const i=e.getParameter(e.MAX_TEXTURE_SIZE);if(A<0||A>i||g<0||g>i)throw new Error("gl-texture2d: Invalid texture size. width:"+A+" height:"+g+" maxSize:"+i);if(I){const I=e.createTexture();this.activateLastTextureUnit(),e.bindTexture(e.TEXTURE_2D,I),e.texImage2D(e.TEXTURE_2D,0,this.internalFormat,A,g,0,this.format,this.type,null);const C=e.createFramebuffer();e.bindFramebuffer(e.FRAMEBUFFER,C),e.framebufferTexture2D(e.FRAMEBUFFER,e.COLOR_ATTACHMENT0,e.TEXTURE_2D,this.gltex,0),e.copyTexImage2D(e.TEXTURE_2D,0,this.internalFormat,0,0,this.width,this.height,0),e.bindFramebuffer(e.FRAMEBUFFER,null),e.deleteFramebuffer(C),this.gl.deleteTexture(this.gltex),this.gltex=I,this.updateGLTexParams()}else this.width>0&&this.height>0&&(this.gl.deleteTexture(this.gltex),this.gltex=e.createTexture(),this.updateGLTexParams()),e.bindTexture(e.TEXTURE_2D,this.gltex),e.texImage2D(e.TEXTURE_2D,0,this.internalFormat,A,g,0,this.format,this.type,null);if(this.width=A,this.height=g,C){const I=new ResizedEvent(A,g);this.emit("resized",I)}}}populate(A,g,I,C=0,e=0,i=!0){const t=this.gl;i&&(this.activateLastTextureUnit(),t.bindTexture(t.TEXTURE_2D,this.gltex)),t.texSubImage2D(t.TEXTURE_2D,0,C,e,g,I,this.format,this.type,A),i&&t.bindTexture(t.TEXTURE_2D,null)}getSize(){return[this.width,this.height]}get glTex(){return this.gltex}getTexHdl(){return this.gltex}preBind(A,g){return{textureTypeUnif:g[A.name+"Type"],textureDescUnif:g[A.name+"Desc"]}}bindToUniform(A,g,I){if(!this.loaded)return!1;if(!this.gltex)throw new Error("Unable to bind non-initialized or deleted texture.");if(A.bindTexture(g,this.gltex),I){const A=this.gl;I.textureTypeUnif&&A.uniform1i(I.textureTypeUnif.location,this.textureType),I.textureDescUnif&&A.uniform4fv(I.textureDescUnif.location,this.textureDesc)}return!0}destroy(){super.destroy(),this.gl.deleteTexture(this.gltex),this.gltex=null}}const WHITESPACE_RE=/\s+/;class ShaderLibrary{__shaderModules;materialTemplates;constructor(){this.__shaderModules={},this.materialTemplates={}}setShaderModule(A,g){A in this.__shaderModules||(this.__shaderModules[A]=g)}getShaderModule(A){return this.__shaderModules[A]}getShaderModuleNames(){const A=[];for(const g in this.__shaderModules)A.push(g);return A}parseAttr(A,g,I,C){const e=A[2].slice(0,A[2].length-1),i=A[1],t="int"==i||"uint"==i||"ivec2"==i||"ivec3"==i||"ivec4"==i;I.attributes[e]={glslType:i,instanced:g,integer:t},"color"==A[1]&&(A[1]="vec4",A.join(" "))}handleImport(A,g,I,C,e){if(I in this.__shaderModules){const i=this.__shaderModules[I];if(!i)throw Error("snippet not loaded or does not exists!");const t=this.parseShaderHelper(g,i,C,e);C.push(I),A.glsl=A.glsl+t.glsl,A.numLines+=t.numLines,A.uniforms={...A.uniforms,...t.uniforms},A.attributes={...A.attributes,...t.attributes}}else console.log("shaderName: "+g),console.log("SNIPPET NOT FOUND: "+I)}parseShader(A,g){return this.parseShaderHelper(A,g,[],0)}parseShaderHelper(A,g,I,C){const e=(A,g)=>{A.glsl=A.glsl+g+"\n",A.numLines++};I.push(A);const i={glsl:"",numLines:0,uniforms:{},attributes:{}},t=(g=g.toString()).split("\n");for(let g=0;g<t.length;g++){let o=t[g];const a=o.trim(),B=a.split(WHITESPACE_RE);switch(B[0]){case"<%include":case"import":{const g=a.split(/'|"|`/)[1].split("/").pop();I.includes(g)||this.handleImport(i,A,g,I,C);break}case"attribute":this.parseAttr(B,!1,i,o),e(i,o);break;case"instancedattribute":this.parseAttr(B,!0,i,o),B[0]="attribute",o=B.join(" "),e(i,o);break;case"uniform":{let A=1;4==B.length&&(A=2);const g=B[A],I=B[A+1].slice(0,B[A+1].length-1);I.includes("[")?i.uniforms[I.substring(0,I.indexOf("["))]={glslType:g}:i.uniforms[I]={glslType:g},"struct"==g&&console.log(B),"color"==B[1]&&(B[1]="vec4",o=B.join(" ")),e(i,o);break}default:e(i,o)}}return i}}const shaderLibrary=new ShaderLibrary;class RenderState{gl;renderer;stack=[];top;glShader;shaderkey;directives=[];directivesHash;attrs;unifs;drawItemsTexture;shaderInstancedGeom;glGeom;boundVao;xrviewport;passIndex;pass;vrPresenting;supportsInstancing;viewport;viewports;bindViewports;bindRendererUnifs;boundTextures;boundRendertarget;geometryMaskTextures;viewXfo;viewScale;region;depthRange;cameraMatrix;constructor(A){this.gl=A}pushGLStack(A){this.top,this.top={name:A,enabled:new Set,disabled:new Set,functions:{}},this.stack.push(this.top)}popGLStack(){const A=this.stack.pop();this.top=this.stack[this.stack.length-1],this.stack.length>0&&(A.enabled.forEach((A=>{let g;for(g=this.stack.length-1;g>=0;g--){const I=this.stack[g];if(I.enabled.has(A))break;if(I.disabled.has(A)){this.gl.disable(A);break}}g<0&&this.gl.disable(A)})),A.disabled.forEach((A=>{let g;for(g=this.stack.length-1;g>=0;g--){const I=this.stack[g];if(I.disabled.has(A))break;if(I.enabled.has(A)){this.gl.enable(A);break}}g<0&&this.gl.enable(A)})))}glEnable(A){this.gl.enable(A),this.top.enabled.add(A)}glDisable(A){this.gl.disable(A),this.top.disabled.add(A)}bindTexture(A,g){const I=this.boundTextures++,C=this.gl;C.activeTexture(C.TEXTURE0+I),C.bindTexture(C.TEXTURE_2D,g),C.uniform1i(A.location,I)}}class HighlightRenderState extends RenderState{}class ColorRenderState extends RenderState{envMap;renderMode;outlineMethod;outlineThickness;outlineColor;hiddenLineColor;screenQuad;exposure;gamma;toHighlightRenderState(){const A=new HighlightRenderState(this.gl);return A.boundRendertarget=this.boundRendertarget,A.viewXfo=this.viewXfo,A.viewScale=this.viewScale,A.region=this.region,A.cameraMatrix=this.cameraMatrix,A.viewport=this.viewport,A.vrPresenting=this.vrPresenting,A.bindViewports=this.bindViewports,A.bindRendererUnifs=this.bindRendererUnifs,A}}class GeomDataRenderState extends RenderState{geomDataFbo;floatGeomBuffer;occlusionCulling;debugGeomDataBuffer;pointAndLinePickingSize}let shaderInstanceId=0;class GLShader extends BaseItem{__gl;shaderStagesGLSL={};shaderStages={};shaderProgramHdls={};constructor(A,g){super(g),A&&(this.__gl=A),this.__id=shaderInstanceId++,this.__shaderStages=this.shaderStages}setGLContext(A){this.__gl=A}setShaderStage(A,g){this.shaderStagesGLSL[A]=g,this.clearProgramsCache()}getShaderStage(A){return this.shaderStagesGLSL[A]}clearProgramsCache(){const A=this.__gl;for(const g in this.shaderProgramHdls){const I=this.shaderProgramHdls[g];for(const g in I.shaderHdls)A.deleteShader(I.shaderHdls[g]);A.deleteProgram(I.shaderProgramHdl)}}static isOpaque(){return!0}static isOverlay(){return!1}generateNumberedLines(A,g,I,C){const e=[];for(const i in C){const t=Number.parseInt(i)-1;for(let I=Math.max(0,t-g);I<t;I++)e.push((I+1+" ").padStart(3)+"[34;107;2m"+A[I]+"[m");e.push((t+1+">").padStart(3)+"[41;93;4m"+A[t]+"[m");for(let g=t+1;g<Math.min(A.length-1,t+I);g++)e.push((g+1+" ").padStart(3)+"[34;107;2m"+A[g]+"[m");const o=C[i];for(const A of o)e.push(A)}return e}compileShaderStage(A,g,I,C){const e=this.__gl;if(C){A=C.join("\n")+"\n"+A}A=StringFunctions.replaceAll(A,"attribute","in"),A="vertexShader"==I?StringFunctions.replaceAll(A,"varying","out"):StringFunctions.replaceAll(A,"varying","in");A="#version 300 es\n"+(A=StringFunctions.replaceAll(A,"texture2D","texture"));const i=e.createShader(g);if(!i)throw Error("shaderHdl not defined");if(e.shaderSource(i,A),e.compileShader(i),!e.getShaderParameter(i,e.COMPILE_STATUS)){console.log("Errors in :"+this.constructor.name+"."+I);const g=e.getShaderInfoLog(i).split("\n"),C={};for(let A=0;A<g.length;A++){const I=g[A].split(":");if(I.length>=2){const e=parseInt(I[2]);isNaN(e)||(C[e]?C[e].push(g[A]):C[e]=[g[A]])}}const t=A.split("\n");console.groupCollapsed("ShaderError-All");this.generateNumberedLines(t,t.length,t.length,C).forEach((A=>console.info(A))),console.groupEnd(),console.group("ShaderError-Summary");throw this.generateNumberedLines(t,4,5,C).forEach((A=>console.info(A))),console.groupEnd(),new Error("An error occurred compiling the shader \n=================\n"+this.constructor.name+"."+I)}return i}createProgram(A){const g=this.__gl,I=g.createProgram();if(!I)throw Error("shaderProgramHdl not defined");const C={};{this.shaderStages.VERTEX_SHADER||(this.shaderStages.VERTEX_SHADER=shaderLibrary.parseShader("VERTEX_SHADER",this.shaderStagesGLSL.VERTEX_SHADER));const e=this.shaderStages.VERTEX_SHADER.glsl;if(null!=e){const i=this.compileShaderStage(e,g.VERTEX_SHADER,"vertexShader",A);if(!i)return;g.attachShader(I,i),C[g.VERTEX_SHADER]=i}}{this.shaderStages.FRAGMENT_SHADER||(this.shaderStages.FRAGMENT_SHADER=shaderLibrary.parseShader("FRAGMENT_SHADER",this.shaderStagesGLSL.FRAGMENT_SHADER));const e=this.shaderStages.FRAGMENT_SHADER.glsl;if(null!=e){const i=this.compileShaderStage(e,g.FRAGMENT_SHADER,"fragmentShader",A);if(!i)return;g.attachShader(I,i),C[g.FRAGMENT_SHADER]=i}}if(g.linkProgram(I),!g.getProgramParameter(I,g.LINK_STATUS)){const A=g.getProgramInfoLog(I);if(!A)throw Error("info not defined");if(A.includes("D3D shader compilation failed")){const A=g.getExtension("WEBGL_debug_shaders");if(A){const I=A.getTranslatedShaderSource(C[g.VERTEX_SHADER]);console.log(I)}}throw console.groupCollapsed("vertexShaderGLSL"),console.log(this.shaderStages.VERTEX_SHADER.glsl),console.groupEnd(),console.groupCollapsed("fragmentShaderGLSL"),console.log(this.shaderStages.FRAGMENT_SHADER.glsl),console.groupEnd(),new Error("Unable to link the shader program:"+this.constructor.name+"\n==================\n"+A)}const e=this.extractAttributeAndUniformLocations(I);return{shaderHdls:C,shaderProgramHdl:I,unifs:e.unifs,attrs:e.attrs}}extractAttributeAndUniformLocations(A){const g=this.__gl,I=this.getAttributes(),C={attrs:{},unifs:{}};for(const e in I){const i=g.getAttribLocation(A,e);if(null==i){console.warn("Shader attribute not found:"+e);continue}const t=I[e];C.attrs[e]={name:e,location:i,glslType:t.glslType,instanced:t.instanced,integer:t.integer}}const e=this.getUniforms();for(let I in e){const i=e[I],t=g.getUniformLocation(A,I);null!=t&&(C.unifs[I]={name:I,location:t,glslType:i.glslType})}return C}getAttributes(){const A={};for(const g in this.shaderStages){const I=this.shaderStages[g];for(const g in I.attributes)A[g]=I.attributes[g]}return A}getUniforms(){const A={};for(const g in this.shaderStages){const I=this.shaderStages[g];for(const g in I.uniforms)A[g]=I.uniforms[g]}return A}isCompiledForTarget(A){const g=A+this.getId()+"";return null!=this.shaderProgramHdls[g]}compileForTarget(A,g){const I=A+this.getId()+"";let C=this.shaderProgramHdls[I];return C||(C=this.createProgram(g||[]),C&&(C.shaderkey=A),this.shaderProgramHdls[I]=C,C)}bind(A,g){const I=this.__gl;if(A.glShader!=this){const C=this.compileForTarget(g,A.directives);if(!C)return console.warn(this.constructor.name+" is not compiled for "+g),!1;const e=C.shaderProgramHdl;if(I.useProgram(e),A.glShader=this,A.shaderkey=C.shaderkey,A.unifs=C.unifs,A.attrs=C.attrs,A.boundTextures=0,A.glGeom=void 0,A.bindRendererUnifs&&A.bindRendererUnifs(C.unifs),A.geometryMaskTextures&&A.unifs.frontDepthTexture){const{frontDepthTexture:g,backDepthTexture:C,viewportSize:e}=A.unifs;A.bindTexture(g,A.geometryMaskTextures[0]),A.bindTexture(C,A.geometryMaskTextures[1]),I.uniform2f(e.location,A.region[2]-A.region[0],A.region[3]-A.region[1])}if(A instanceof GeomDataRenderState){const{floatGeomBuffer:g,occlusionCulling:C,debugGeomDataBuffer:e}=A.unifs;g&&I.uniform1i(g.location,A.floatGeomBuffer?1:0),C&&I.uniform1i(C.location,A.occlusionCulling),e&&I.uniform1i(e.location,A.debugGeomDataBuffer?1:0)}}return A.supportsInstancing=!0,!0}unbind(A){return A.glShader=null,A.shaderkey="",A.unifs={},A.attrs={},!0}getGeomDataShaderName(){return""}getSelectedShaderName(){return""}static supportsInstancing(){return!0}static getPackedMaterialData(A){return new Float32Array(4)}static getMaterialTemplate(){throw new Error("Shader does not provide a material template.")}destroy(){const A=this.__gl;for(const g in this.shaderProgramHdls){const I=this.shaderProgramHdls[g];A.deleteProgram(I.shaderProgramHdl)}this.shaderProgramHdls={}}}class GLFbo{colorTextureResizeEventId=-1;__gl;__colorTexture;__createDepthTexture;__clearColor;__depthTexture=null;__fbo=null;__prevBoundFbo=null;constructor(A,g,I=!1){this.__gl=A,this.__colorTexture=g,this.__createDepthTexture=I,this.__clearColor=new Color(0,0,0,0),this.__colorTexture&&(this.colorTextureResizeEventId=this.__colorTexture.on("resized",(()=>{console.warn("This code path will be deprecated. Instead, resize the GLFbo and it will resize the texture."),this.resize(this.__colorTexture.width,this.__colorTexture.height,!1)}))),this.setup()}setClearColor(A){this.__clearColor=A}getWidth(){return this.__colorTexture.width}getHeight(){return this.__colorTexture.height}getSize(){return[this.__colorTexture.width,this.__colorTexture.height]}getColorTexture(){return this.__colorTexture}getDepthTextureGL(){return this.__depthTexture}get width(){return this.__colorTexture.width}get height(){return this.__colorTexture.height}get size(){return[this.__colorTexture.width,this.__colorTexture.height]}get colorTexture(){return this.__colorTexture}setColorTexture(A){const g=this.__gl;this.__colorTexture=A,g.framebufferTexture2D(g.FRAMEBUFFER,g.COLOR_ATTACHMENT0,g.TEXTURE_2D,this.__colorTexture.glTex,0)}get depthTextureGL(){return this.__depthTexture}setup(){const A=this.__gl;this.__fbo=A.createFramebuffer(),A.bindFramebuffer(A.DRAW_FRAMEBUFFER,this.__fbo),this.__colorTexture&&A.framebufferTexture2D(A.DRAW_FRAMEBUFFER,A.COLOR_ATTACHMENT0,A.TEXTURE_2D,this.__colorTexture.glTex,0),this.__createDepthTexture&&this.createDepthTexture(),checkFramebuffer(A,this.width,this.height),A.bindFramebuffer(A.DRAW_FRAMEBUFFER,null)}createDepthTexture(){const A=this.__gl;if("webgl2"==A.name||A.__ext_WEBGL_depth_texture)A.activeTexture(A.TEXTURE0),this.__depthTexture=A.createTexture(),A.bindTexture(A.TEXTURE_2D,this.__depthTexture),A.texParameteri(A.TEXTURE_2D,A.TEXTURE_MAG_FILTER,A.LINEAR),A.texParameteri(A.TEXTURE_2D,A.TEXTURE_MIN_FILTER,A.LINEAR),A.texParameteri(A.TEXTURE_2D,A.TEXTURE_WRAP_S,A.CLAMP_TO_EDGE),A.texParameteri(A.TEXTURE_2D,A.TEXTURE_WRAP_T,A.CLAMP_TO_EDGE),A.texImage2D(A.TEXTURE_2D,0,A.DEPTH_COMPONENT24,this.width,this.height,0,A.DEPTH_COMPONENT,A.UNSIGNED_INT,null),A.framebufferTexture2D(A.DRAW_FRAMEBUFFER,A.DEPTH_ATTACHMENT,A.TEXTURE_2D,this.__depthTexture,0);else{const g=A.createRenderbuffer();A.bindRenderbuffer(A.RENDERBUFFER,g),A.renderbufferStorage(A.RENDERBUFFER,A.DEPTH_COMPONENT16,this.width,this.height),A.framebufferRenderbuffer(A.FRAMEBUFFER,A.DEPTH_ATTACHMENT,A.RENDERBUFFER,g)}}resize(A,g,I=!0){const C=this.__gl;C.bindFramebuffer(C.FRAMEBUFFER,null),C.deleteFramebuffer(this.__fbo),I&&this.__colorTexture.resize(A,g,!1,!1),this.__fbo=C.createFramebuffer(),C.bindFramebuffer(C.DRAW_FRAMEBUFFER,this.__fbo),C.framebufferTexture2D(C.FRAMEBUFFER,C.COLOR_ATTACHMENT0,C.TEXTURE_2D,this.__colorTexture.glTex,0),this.__depthTexture&&(C.deleteTexture(this.__depthTexture),this.createDepthTexture()),checkFramebuffer(C,this.width,this.height),C.bindFramebuffer(C.DRAW_FRAMEBUFFER,null)}bindForWriting(A){A&&(this.__prevBoundFbo=A.boundRendertarget,A.boundRendertarget=this.__fbo);const g=this.__gl;g.bindFramebuffer(g.DRAW_FRAMEBUFFER,this.__fbo),g.viewport(0,0,this.width,this.height)}unbindForWriting(A){A&&(A.boundRendertarget=this.__prevBoundFbo);const g=this.__gl;g.bindFramebuffer(g.DRAW_FRAMEBUFFER,this.__prevBoundFbo)}bind(A){this.bindForWriting(A)}unbind(A){if(A)this.unbindForWriting(A);else{const A=this.__gl;A.bindFramebuffer(A.FRAMEBUFFER,null)}}bindForReading(A){const g=this.__gl;g.bindFramebuffer(g.READ_FRAMEBUFFER,this.__fbo)}unbindForReading(){const A=this.__gl;A.bindFramebuffer(A.READ_FRAMEBUFFER,null)}clear(){const A=this.__gl;A.colorMask(!0,!0,!0,!0);const g=this.__clearColor.asArray();A.clearColor(g[0],g[1],g[2],g[3]),this.__createDepthTexture?A.clear(A.COLOR_BUFFER_BIT|A.DEPTH_BUFFER_BIT):A.clear(A.COLOR_BUFFER_BIT)}bindAndClear(A){this.bind(A),this.clear()}destroy(){const A=this.__gl;A.bindFramebuffer(A.FRAMEBUFFER,null),A.deleteFramebuffer(this.__fbo),this.__fbo=null,this.__colorTexture.off("resized",this.colorTextureResizeEventId)}}function checkFramebuffer(A,g,I){let C;if(C=A.checkFramebufferStatus(A.DRAW_FRAMEBUFFER),C!==A.FRAMEBUFFER_COMPLETE)switch(A.bindTexture(A.TEXTURE_2D,null),A.bindFramebuffer(A.DRAW_FRAMEBUFFER,null),console.warn("Error creating Fbo width:",g,", height:",I),C){case A.FRAMEBUFFER_INCOMPLETE_ATTACHMENT:throw new Error("The attachment types are mismatched or not all framebuffer attachment points are framebuffer attachment complete.");case A.FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT:throw new Error("There is no attachment.");case A.FRAMEBUFFER_INCOMPLETE_DIMENSIONS:throw new Error("Height and width of the attachment are not the same.");case A.FRAMEBUFFER_INCOMPLETE_DIMENSIONS:throw new Error("The format of the attachment is not supported or if depth and stencil attachments are not the same renderbuffer.");case 36061:throw new Error("The framebuffer is unsupported");default:throw new Error("Incomplete Frambuffer")}}class GLRenderTarget extends EventEmitter{gl;textureTargets;depthTexture;frameBuffer;textureDesc;type;format;internalFormat;depthFormat;depthInternalFormat;depthType;minFilter;magFilter;wrapS=0;wrapT=0;width=0;height=0;clearColor;colorMask;textureType;prevBoundFbo;constructor(A,g){super(),this.gl=A,this.textureTargets=[],this.depthTexture=null,this.textureDesc=[0,0,0,0],this.clearColor=new Color(0,0,0,0),this.colorMask=[!0,!0,!0,!0],g&&this.configure(g)}configure(A){const g=this.gl;{const I=processTextureParams(g,A);this.type=I.type,this.format=I.format,this.internalFormat=I.internalFormat,this.depthType=I.depthType,this.depthFormat=I.depthFormat,this.depthInternalFormat=I.depthInternalFormat,this.minFilter=I.minFilter?I.minFilter:I.filter,this.magFilter=I.magFilter?I.magFilter:I.filter,this.wrapS=I.wrapS,this.wrapT=I.wrapT,this.width=I.width,this.height=I.height}this.textureType=1,this.textureDesc[0]=this.width,this.textureDesc[1]=this.height,this.textureTargets.forEach((A=>{g.deleteTexture(A)})),this.textureTargets=[],this.depthTexture&&(g.deleteTexture(this.depthTexture),this.depthTexture=null),this.frameBuffer&&g.deleteFramebuffer(this.frameBuffer);const I=null!=A.numColorChannels?A.numColorChannels:null!=this.format?1:0;for(let A=0;A<I;A++){g.activeTexture(g.TEXTURE0+1);const A=g.createTexture();g.bindTexture(g.TEXTURE_2D,A),g.texParameteri(g.TEXTURE_2D,g.TEXTURE_WRAP_S,this.wrapS),g.texParameteri(g.TEXTURE_2D,g.TEXTURE_WRAP_T,this.wrapT),g.texParameteri(g.TEXTURE_2D,g.TEXTURE_MIN_FILTER,this.minFilter),g.texParameteri(g.TEXTURE_2D,g.TEXTURE_MAG_FILTER,this.magFilter),g.texImage2D(g.TEXTURE_2D,0,this.internalFormat,this.width,this.height,0,this.format,this.type,null),this.textureTargets.push(A)}if(this.depthFormat){if("webgl"==g.name&&!g.__ext_WEBGL_depth_texture)throw new Error("Depth textures not support on this device");g.activeTexture(g.TEXTURE0),this.depthTexture=g.createTexture(),g.bindTexture(g.TEXTURE_2D,this.depthTexture),g.texParameteri(g.TEXTURE_2D,g.TEXTURE_WRAP_S,this.wrapS),g.texParameteri(g.TEXTURE_2D,g.TEXTURE_WRAP_T,this.wrapT),g.texParameteri(g.TEXTURE_2D,g.TEXTURE_MIN_FILTER,this.minFilter),g.texParameteri(g.TEXTURE_2D,g.TEXTURE_MAG_FILTER,this.magFilter),g.texImage2D(g.TEXTURE_2D,0,this.depthInternalFormat,this.width,this.height,0,this.depthFormat,this.depthType,null)}if(this.frameBuffer=g.createFramebuffer(),this.bindForWriting(),this.textureTargets.length>0){this.textureTargets.length>1&&"webgl"==g.name&&g.drawBuffers;const A=[];for(let I=0;I<this.textureTargets.length;I++)g.framebufferTexture2D(g.DRAW_FRAMEBUFFER,g.COLOR_ATTACHMENT0+I,g.TEXTURE_2D,this.textureTargets[I],0),A.push(g.COLOR_ATTACHMENT0+I);this.textureTargets.length>1&&g.drawBuffers(A)}this.depthTexture&&g.framebufferTexture2D(g.DRAW_FRAMEBUFFER,g.DEPTH_ATTACHMENT,g.TEXTURE_2D,this.depthTexture,0),this.checkFramebuffer()}checkFramebuffer(){this.bindForWriting();const A=this.gl,g=A.checkFramebufferStatus(A.DRAW_FRAMEBUFFER);if(g!=A.FRAMEBUFFER_COMPLETE)switch(g){case A.FRAMEBUFFER_INCOMPLETE_ATTACHMENT:throw new Error("The attachment types are mismatched or not all framebuffer attachment points are framebuffer attachment complete.");case A.FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT:throw new Error("There is no attachment.");case A.FRAMEBUFFER_INCOMPLETE_DIMENSIONS:throw new Error("Height and width of the attachment are not the same.");case A.FRAMEBUFFER_INCOMPLETE_DIMENSIONS:throw new Error("The format of the attachment is not supported or if depth and stencil attachments are not the same renderbuffer.");case 36061:throw new Error("The framebuffer is unsupported");default:throw new Error("Incomplete Frambuffer")}this.unbindForWriting()}bindForWriting(A,g=!1){A&&(this.prevBoundFbo=A.boundRendertarget,A.boundRendertarget=this.frameBuffer);const I=this.gl;I.bindFramebuffer(I.DRAW_FRAMEBUFFER,this.frameBuffer),I.viewport(0,0,this.width,this.height),g&&this.clear()}unbindForWriting(A){A&&(A.boundRendertarget=this.prevBoundFbo);const g=this.gl;g.bindFramebuffer(g.DRAW_FRAMEBUFFER,this.prevBoundFbo),this.prevBoundFbo=null}clear(A=!0){const g=this.gl,I=this.colorMask;g.colorMask(I[0],I[1],I[2],I[3]);const C=this.clearColor.asArray();g.clearColor(C[0],C[1],C[2],C[3]);let e=0;this.textureTargets.length>0&&(e|=g.COLOR_BUFFER_BIT),this.depthTexture&&(e|=g.DEPTH_BUFFER_BIT),g.clear(e)}bindForReading(){const A=this.gl;A.bindFramebuffer(A.READ_FRAMEBUFFER,this.frameBuffer)}unbindForReading(){const A=this.gl;A.bindFramebuffer(A.READ_FRAMEBUFFER,null)}bindColorTexture(A,g,I=0){const C=this.gl,e=A.boundTextures++;return C.uniform1i(g.location,e),C.activeTexture(C.TEXTURE0+e),C.bindTexture(C.TEXTURE_2D,this.textureTargets[I]),!0}bindDepthTexture(A,g){const I=this.gl,C=A.boundTextures++;return I.uniform1i(g.location,C),I.activeTexture(I.TEXTURE0+C),I.bindTexture(I.TEXTURE_2D,this.depthTexture),!0}unbind(A){this.unbindForWriting(A)}resize(A,g,I=!1){const C=this.gl;if(this.width!=A||this.height!=g){const e=C.getParameter(C.MAX_TEXTURE_SIZE);if(A<0||A>e||g<0||g>e)throw new Error(`GLRenderTarget: Invalid texture size. width: ${A} height: ${g} maxSize: ${e}`);I&&this.bindForReading();for(let e=0;e<this.textureTargets.length;e++){const i=C.createTexture();C.bindTexture(C.TEXTURE_2D,i),C.texParameteri(C.TEXTURE_2D,C.TEXTURE_WRAP_S,this.wrapS),C.texParameteri(C.TEXTURE_2D,C.TEXTURE_WRAP_T,this.wrapT),C.texParameteri(C.TEXTURE_2D,C.TEXTURE_MIN_FILTER,this.minFilter),C.texParameteri(C.TEXTURE_2D,C.TEXTURE_MAG_FILTER,this.magFilter),C.texImage2D(C.TEXTURE_2D,0,this.internalFormat,A,g,0,this.format,this.type,null),I&&C.copyTexImage2D(C.TEXTURE_2D,0,this.internalFormat,0,0,Math.min(A,this.width),Math.min(g,this.height),0),C.deleteTexture(this.textureTargets[e]),this.textureTargets[e]=i}if(this.depthFormat){if("webgl"==C.name&&!C.__ext_WEBGL_depth_texture)throw new Error("Depth textures not support on this device");C.activeTexture(C.TEXTURE0);const e=C.createTexture();C.bindTexture(C.TEXTURE_2D,e),C.texParameteri(C.TEXTURE_2D,C.TEXTURE_WRAP_S,this.wrapS),C.texParameteri(C.TEXTURE_2D,C.TEXTURE_WRAP_T,this.wrapT),C.texParameteri(C.TEXTURE_2D,C.TEXTURE_MIN_FILTER,this.minFilter),C.texParameteri(C.TEXTURE_2D,C.TEXTURE_MAG_FILTER,this.magFilter),C.texImage2D(C.TEXTURE_2D,0,this.depthInternalFormat,A,g,0,this.depthFormat,this.depthType,null),I&&C.copyTexImage2D(C.TEXTURE_2D,0,this.internalFormat,0,0,Math.min(A,this.width),Math.min(g,this.height),0),C.deleteTexture(this.depthTexture),this.depthTexture=e}if(I&&this.unbindForReading(),this.width=A,this.height=g,this.frameBuffer&&C.deleteFramebuffer(this.frameBuffer),this.frameBuffer=C.createFramebuffer(),this.bindForWriting(),this.textureTargets.length>0){this.textureTargets.length>1&&"webgl"==C.name&&C.drawBuffers;const A=[];for(let g=0;g<this.textureTargets.length;g++)C.framebufferTexture2D(C.DRAW_FRAMEBUFFER,C.COLOR_ATTACHMENT0+g,C.TEXTURE_2D,this.textureTargets[g],0),A.push(C.COLOR_ATTACHMENT0+g);this.textureTargets.length>1&&C.drawBuffers(A)}this.depthTexture&&C.framebufferTexture2D(C.DRAW_FRAMEBUFFER,C.DEPTH_ATTACHMENT,C.TEXTURE_2D,this.depthTexture,0),this.checkFramebuffer()}}bindToUniform(A,g,I){const C=A.boundTextures++,e=this.gl.TEXTURE0+C,i=this.gl;return i.activeTexture(e),i.bindTexture(i.TEXTURE_2D,this.textureTargets[0]),i.uniform1i(g.location,C),I&&(I.textureTypeUnif&&i.uniform1i(I.textureTypeUnif.location,this.textureType),I.textureDescUnif&&this.gl.uniform4fv(I.textureDescUnif.location,this.textureDesc)),!0}destroy(){const A=this.gl;this.textureTargets.forEach((g=>{A.deleteTexture(g)})),this.textureTargets=[],this.depthTexture&&(A.deleteTexture(this.depthTexture),this.depthTexture=null),this.frameBuffer&&A.deleteFramebuffer(this.frameBuffer)}}var computeViewNormal="#define GLSLIFY 1\n  \n#ifdef ENABLE_ES3\nvec3 computeViewNormal(vec3 viewPos) {\n  vec3 fdx = dFdx(viewPos);\n  vec3 fdy = dFdy(viewPos);\n  return normalize(cross(fdx, fdy));\n}\n#else \nvec3 computeViewNormal(vec3 viewPos) {\n  return vec3(0.0, 0.0, 0.0);\n}\n#endif\n",calcFatLinesViewPos="#define GLSLIFY 1\nimport 'GLSLUtils.glsl'\nvec3 calcFatLinesViewPos(int vertexID, mat4 modelViewMatrix, inout vec3 viewNormal, inout vec2 texCoord, inout vec3 pos) {\n\n  int seqentialIndex_0 = int(mod(segmentIndices.x, 2.));\n  int seqentialIndex_1 = int(mod(segmentIndices.y, 2.));\n  int index_0 = int(segmentIndices.x) / 2;\n  int index_1 = int(segmentIndices.y) / 2;\n\n  vec3 viewPos;\n  vec4 data_0 = fetchTexel(positionsTexture, positionsTextureSize, index_0);\n  vec4 data_1 = fetchTexel(positionsTexture, positionsTextureSize, index_1);\n\n  // During XR sessions, there is a scaling applied to the view matrix\n  // which causes a distortion to the line width. We extract that scale here\n  // and use to correct the distortion.\n  // See also: FatPointsShader\n  vec3 viewZ = modelViewMatrix[2].xyz;\n  float viewScale = length(viewZ);\n\n  vec4 pos_0 = modelViewMatrix * vec4(data_0.xyz, 1.0);\n  vec4 pos_1 = modelViewMatrix * vec4(data_1.xyz, 1.0);\n  // Note: multiply the per-vertex line thickness with the line thickness uniform value;\n  float lineThickness_0 = LineThickness * data_0.w * viewScale;\n  float lineThickness_1 = LineThickness * data_1.w * viewScale;\n\n  if (vertexID < 2) {\n    pos = data_0.xyz;\n    viewPos = pos_0.xyz;\n  }\n  else {\n    pos = data_1.xyz;\n    viewPos = pos_1.xyz;\n  }\n  if (pos_1 != pos_0) {\n    vec3 segmentDir = normalize(pos_1.xyz - pos_0.xyz);\n    vec3 viewVector = normalize(viewPos);\n\n    if (vertexID < 2) {\n      vec3 segmentStartDir = segmentDir;\n      if (seqentialIndex_0 != 0) {\n        //if index_0 == 0, get the last index in the line as previous\n        int index_prev = (index_0 > 0) ? (index_0-1) : (positionsTextureSize-1);\n        vec4 data_prev = fetchTexel(positionsTexture, positionsTextureSize, index_prev);\n        vec4 pos_prev = modelViewMatrix * vec4(data_prev.xyz, 1.0);\n        segmentStartDir = normalize(segmentDir + normalize(pos_0.xyz - pos_prev.xyz));\n        // segmentStartDir = segmentDir;\n      }\n      // vec3 startBiTangent = normalize(cross(segmentStartDir, viewVector));\n      // viewNormal = normalize(cross(segmentStartDir, startBiTangent));\n      vec3 startBiTangent = normalize(vec3(-segmentStartDir.y, segmentStartDir.x, 0.0));\n      viewNormal = normalize(-viewVector);\n      // Move the endpoints to overlap a bit more.\n      //viewPos -= vec3(segmentStartDir * lineThickness_0 * 0.25);\n      if (mod(vertexIDs, 2.0) == 0.0) {\n        viewPos += vec3(startBiTangent * lineThickness_0);\n        texCoord.x = 1.0;\n      }\n      else {\n        viewPos -= vec3(startBiTangent * lineThickness_0);\n        texCoord.x = 0.0;\n      }\n      texCoord.y = 0.0;\n    }\n    else {\n      vec3 segmentEndDir = segmentDir;\n      if (seqentialIndex_1 != 0) {\n        //if index_1 == numPoints-1, get the first index in the line as next\n        int index_next = (index_1 < (positionsTextureSize-1)) ? (index_1+1) : 0;\n        vec4 data_next = fetchTexel(positionsTexture, positionsTextureSize, index_next);\n        vec4 pos_next = modelViewMatrix * vec4(data_next.xyz, 1.0);\n        segmentEndDir = normalize(segmentDir + normalize(pos_next.xyz - pos_1.xyz));\n        // segmentEndDir = segmentDir;\n      }\n      // vec3 endBiTangent = normalize(cross(segmentEndDir, viewVector));\n      // viewNormal = normalize(cross(segmentEndDir, endBiTangent));\n      vec3 endBiTangent = normalize(vec3(-segmentEndDir.y, segmentEndDir.x, 0.0));\n      viewNormal = normalize(-viewVector);\n      // Move the endpoints to overlap a bit more.\n      //viewPos += vec3(segmentEndDir * lineThickness_1 * 0.25);\n      if (mod(vertexIDs, 2.0) == 0.0) {\n        viewPos += vec3(endBiTangent * lineThickness_1);\n        texCoord.x = 1.0;\n      }\n      else {\n        viewPos -= vec3(endBiTangent * lineThickness_1);\n        texCoord.x = 0.0;\n      }\n      texCoord.y = 1.0;\n    }\n\n    // Move the line towards the viewer by the line thickness.\n    // this is to avoid depth issues when lines are rendered over meshes. \n    viewPos.z += (lineThickness_0 + lineThickness_1) * 0.5;\n  }\n\n  return viewPos;\n}\n\n",constants="#define GLSLIFY 1\n#define PI 3.141592653589793\n#define TwoPI (2.0 * PI)\n#define HalfPI (0.5 * PI)\n\n",convolveHelpers="#define GLSLIFY 1\n\n#ifdef ENVMAP_CUBE\n\nuniform samplerCube envMap;\n\nvec4 sampleEnvMap(vec3 dir) {\n  return texture(envMap, dir);\n}\n\n#else \n\nuniform sampler2D   envMap;\n\nimport 'envmap-octahedral.glsl'\n\nvec4 sampleEnvMap(vec3 dir) {\n  vec2 uv = dirToSphOctUv(dir);\n  vec4 texel = texture2D(envMap, vec2(uv.x, 1.0 - uv.y));\n  return vec4(texel.rgb/texel.a, 1.0); // TODO: Check this line. Do we need it?\n}\n\n#endif \n\nvec3 cubeFaceUvToDir(float u, float v, int faceId) {\n\n  // normalize into [-1, 1] range\n  float n_u = 2.0 * u - 1.0;\n  float n_v = 2.0 * v - 1.0;\n\n  vec3 dir;\n  switch (faceId)\n  {\n  case 0: //TEXTURE_CUBE_MAP_POSITIVE_X:\n    dir.x = 1.0f;\n    dir.y = n_v;\n    dir.z = -n_u;\n    break;\n  case 1: //TEXTURE_CUBE_MAP_NEGATIVE_X:\n    dir.x = -1.0f;\n    dir.y = n_v;\n    dir.z = n_u;\n    break;\n  case 3: //TEXTURE_CUBE_MAP_POSITIVE_Y:\n    dir.x = n_u;\n    dir.y = 1.0f;\n    dir.z = -n_v;\n    break;\n  case 2: //TEXTURE_CUBE_MAP_NEGATIVE_Y:\n    dir.x = n_u;\n    dir.y = -1.0f;\n    dir.z = n_v;\n    break;\n  case 4: //TEXTURE_CUBE_MAP_POSITIVE_Z:\n    dir.x = n_u;\n    dir.y = n_v;\n    dir.z = 1.0f;\n    break;\n  case 5: //TEXTURE_CUBE_MAP_NEGATIVE_Z:\n    dir.x = -n_u;\n    dir.y = n_v;\n    dir.z = -1.0f;\n    break;\n  }\n  return normalize(dir);\n}\n\n",cutaways="#define GLSLIFY 1\nuniform color cutColor;\n\n#ifdef ENABLE_FLOAT_TEXTURES\n  vec4 getCutaway(int id) {\n    return fetchTexel(instancesTexture, instancesTextureSize, (id * pixelsPerItem) + 5);\n  }\n\n#else\n\n  uniform vec4 cutawayData;\n\n  vec4 getCutaway(int id) {\n    return cutawayData;\n  }\n\n#endif\n\n#define RAY_EPS 0.0000001\nstruct Ray {\n  vec3 start;\n  vec3 dir;\n};\n\nfloat intersectRayPlane(Ray ray, Ray plane) {\n  vec3 w = ray.start - plane.start;\n  float D = dot(plane.dir, ray.dir);\n  float N = dot(-plane.dir, w);\n\n  if (abs(D) < RAY_EPS) {\n    // segment is parallel to plane\n    if (N == 0.0)\n      return -1.0; // segment lies in plane\n    else\n      return -1.0; // no intersection\n  }\n  // they are not parallel\n  // compute intersect param\n  float sI = N / D;\n  if (sI < -RAY_EPS) {\n    return -1.0; // no intersection\n  }\n  return sI;\n}\n\nbool cutaway(vec3 worldPos, vec3 planeNormal, float planeDist) {\n\n  vec3 planePos = planeNormal * planeDist;\n  vec3 planeDir = worldPos + planePos;\n  float planeOffset = dot(planeDir, planeNormal);\n  if (planeOffset > 0.0) {\n    return true;\n  }\n  return false;\n}\n",debugColors="#define GLSLIFY 1\nimport 'GLSLUtils.glsl'\n\nvec3 getDebugColor(float id) {\n  int sel = int(round(mod(round(id), 16.0)));\n  \n  if (sel==0)\n    return vec3(0.0, 1.0, 1.0);\n  else if (sel==1)\n    return vec3(0.0, 1.0, 0.0);\n  else if (sel==2)\n    return vec3(1.0, 0.0, 1.0);\n  else if (sel==3)\n    return vec3(0.75, 0.75, 0.0);\n  else if (sel==4)\n    return vec3(0.0, 0.75, 0.75);\n  else if (sel==5)\n    return vec3(0.75, 0.0, 0.75);\n  else if (sel==6)\n    return vec3(0.45, 0.95, 0.0);\n  else if (sel==7)\n    return vec3(0.0, 0.45, 0.95);\n  else if (sel==8)\n    return vec3(0.95, 0.0, 0.45);\n  else if (sel==9)\n    return vec3(0.95, 0.45, 0.0);\n  else if (sel==10)\n    return vec3(0.0, 0.95, 0.45);\n  else if (sel==11)\n    return vec3(0.45, 0.0, 0.95);\n  else if (sel==12)\n    return vec3(0.45, 0.45, 0.95);\n  else if (sel==13)\n    return vec3(0.0, 0.0, 0.45);\n  else if (sel==14)\n    return vec3(0.0, 0.45, 0.45);\n  else if (sel==15)\n    return vec3(0.45, 0.0, 0.45);\n  else return vec3(0.2, 0.2, 0.2);\n}\n\n",geomItemId="#define GLSLIFY 1\n\n#ifdef ENABLE_MULTI_DRAW\n\n// On some mobile GPUs the sampler2D defaults to lowp, which implies a maximum\n// resolution of 255. This caused rendering artifacts on larger scenes on mobile\n// devices.\nuniform highp sampler2D drawIdsTexture;\n\n#ifdef EMULATE_MULTI_DRAW\nuniform int drawId;\n#endif // EMULATE_MULTI_DRAW\n\nint getGeomItemId() {\n#ifndef EMULATE_MULTI_DRAW\n  int drawId = gl_DrawID;\n#endif // EMULATE_MULTI_DRAW\n\n  ivec2 texSize = textureSize(drawIdsTexture, 0);\n  ivec2 texelCoords = ivec2(drawId % texSize.x, drawId / texSize.x);\n  return int(texelFetch(drawIdsTexture, texelCoords, 0).r + 0.5);\n}\n\nvec4 getDrawItemIds() {\n#ifndef EMULATE_MULTI_DRAW\n  int drawId = gl_DrawID;\n#endif // EMULATE_MULTI_DRAW\n\n  ivec2 texSize = textureSize(drawIdsTexture, 0);\n  ivec2 texelCoords = ivec2(drawId % texSize.x, drawId / texSize.x);\n  vec4 color = texelFetch(drawIdsTexture, texelCoords, 0);\n  // Note: A 0 value in the texture means no sub-geom index is being rendered.\n  // subtract off 1 to get the true sub-geom index.\n  return vec4(color.r, color.g - 1.0, color.b, color.a);\n}\n\n#else // ENABLE_MULTI_DRAW\n\nuniform int geomItemId;\n\nattribute float instancedIds;    // instanced attribute..\nuniform int instancedDraw;\n\nint getGeomItemId() {\n  if (instancedDraw == 0) {\n    return geomItemId;\n  }\n  else {\n    return int(instancedIds);\n  }\n}\n\nvec4 getDrawItemIds() {\n  if (instancedDraw == 0) {\n    return vec4(float(geomItemId), 0.0, -1.0, -1.0);\n  }\n  else {\n    return vec4(float(instancedIds), 0.0, -1.0, -1.0);\n  }\n}\n\n#endif // ENABLE_MULTI_DRAW\n\n// For backwards compatibility with older plugins (UX.HandleShader)\nint getDrawItemId() {\n  return getGeomItemId();\n}\n",geomItemFlags="#define GLSLIFY 1\nconst int GEOMITEM_FLAG_CUTAWAY = 2; // 1<<1;\nconst int GEOMITEM_INVISIBLE_IN_GEOMDATA = 4; // 1<<2;\nconst int GEOMITEM_TRANSPARENT = 8; // 1<<3;\nconst int GEOMITEM_PLACEHOLDER = 16; // 1<<4;\nconst int GEOMITEM_MASK = 32; // 1<<5;\n",geomType="#define GLSLIFY 1\n\nconst int TRIANGLES = 0;\nconst int LINES = 1;\nconst int POINTS = 2;",geometryMask="#define GLSLIFY 1\nuniform sampler2D frontDepthTexture;\nuniform sampler2D backDepthTexture;\n\nuniform vec2 depthRange;\nuniform vec2 viewportSize;\n\nbool testGeometryMask(vec3 viewPos) {\n  float depth = gl_FragCoord.z;\n  vec2 texCoord = gl_FragCoord.xy / viewportSize;\n  float front = texture2D(frontDepthTexture, texCoord).r;\n  float back = texture2D(backDepthTexture, texCoord).r;\n\n  return depth < front || depth > back;\n}\n",drawItemTexture="#define GLSLIFY 1\nimport 'GLSLUtils.glsl'\n\n// The texture populated by the GLGeomItemLibrary\nuniform sampler2D instancesTexture;\nuniform highp int instancesTextureSize;\n\n// See also: src\\Renderer\\GLSLConstants.js\nconst int pixelsPerItem = 8;\n\nvec4 getInstanceData(int id) {\n  return fetchTexel(instancesTexture, instancesTextureSize, (id * pixelsPerItem) + 0);\n}\n\n",envmapDualfisheye="#define GLSLIFY 1\n\nvec2 dualfisheyeUVsFromDir(vec3 dir) {\n  vec2 result;\n  float angle = 0.465;\n  if (dir.x < 0.0) {\n    result = vec2(((dir.z * -angle) + 0.5) * 0.5, (dir.y * angle) + 0.5);\n  }\n  else {\n    result = vec2( 0.5 + ((dir.z * angle) + 0.5) * 0.5, (dir.y * angle) + 0.5);\n  }\n  return result;\n}\n\n",envmapEquirect="#define GLSLIFY 1 \nimport 'constants.glsl'\n\nvec2 latLongUVsFromDir(vec3 dir) {\n  // Math function taken from...\n  // http://gl.ict.usc.edu/Data/HighResProbes/\n  // Note: Scaling from u=[0,2], v=[0,1] to u=[0,1], v=[0,1]\n  float phi = acos(dir.z);\n  float theta = atan(dir.x, dir.y);\n  return vec2((1.0 + theta / PI) / 2.0, phi / PI);\n}\n\n// Note: when u == 0.5 z = 1.0\nvec3 dirFromLatLongUVs(float u, float v) {\n  // http://gl.ict.usc.edu/Data/HighResProbes/\n  float theta = PI*((u * 2.0) - 1.0);\n  float phi = PI*v;\n  return vec3(sin(phi)*sin(theta), sin(phi)*cos(theta), cos(phi));\n}\n\nvec3 dirFromPolar(vec2 polar) {\n  float u = polar.x / (PI * 2.0);\n  float v = polar.y / PI;\n  return dirFromLatLongUVs(u, v);\n}\n\n",envmapOctahedral="#define GLSLIFY 1\nimport 'constants.glsl'\nimport 'GLSLUtils.glsl'\n#define sectorize(value) step(0.0, (value))*2.0-1.0\n#define sum(value) dot(clamp((value), 1.0, 1.0), (value))\n\nvec2 dirToSphOctUv(vec3 normal) {\n  normal = normalize(normal);\n  vec3 aNorm = abs(normal);\n  vec3 sNorm = sectorize(normal);\n  \n  vec2 dir = aNorm.xy;\n  float orient = atan(dir.x, max(dir.y,0.0000000000000001))/HalfPI;\n\n  dir = vec2(aNorm.z, length(aNorm.xy));\n  float pitch = atan(dir.y, dir.x)/HalfPI;\n\n  vec2 uv = vec2(sNorm.x*orient, sNorm.y*(1.0-orient))*pitch;\n\n  if (normal.z < 0.0) {\n    uv = sNorm.xy - abs(uv.ts)*sNorm.xy;\n  }\n  vec2 res = uv*0.5+0.5;\n  // Flip-v\n  // return res;\n  return vec2(res.x, 1.0 - res.y);\n}\n\nvec3 sphOctUvToDir(vec2 uv) {\n  uv = uv*2.0-1.0;\n  // Flip-v\n  uv.y = -uv.y;\n  vec2 suv = sectorize(uv);\n  float sabsuv = sum(abs(uv));\n  float pitch = sabsuv*HalfPI;\n\n  if (pitch <= 0.0) {\n    return vec3(0.0, 0.0, 1.0);\n  }\n  if (abs(pitch - PI) < 0.000001) {\n    return vec3(0.0, 0.0, -1.0);\n  }\n  if (sabsuv > 1.0) {\n    uv = (1.0-abs(uv.ts))*suv;\n  }\n\n  float orient = (abs(uv.s)/sabsuv)*HalfPI;\n  float sOrient = sin(orient);\n  float cOrient = cos(orient);\n  float sPitch = sin(pitch);\n  float cPitch = cos(pitch);\n\n  return vec3(\n    sOrient*suv.s*sPitch,\n    cOrient*suv.t*sPitch,\n    cPitch\n  );\n}\n\n",GLSLBits='#define GLSLIFY 1\n    \n/////////////////////////////////////////////////////////////////\n// http://concord-consortium.github.io/lab/experiments/webgl-gpgpu/script.js\nfloat shift_right(float v, float amt) {\n  v = floor(v) + 0.5;\n  return floor(v / exp2(amt));\n}\nfloat shift_left(float v, float amt) {\n  return floor(v * exp2(amt) + 0.5);\n}\n\nfloat mask_last(float v, float bits) {\n  return mod(v, shift_left(1.0, bits));\n}\nfloat extract_bits(float num, float from, float to) {\n  from = floor(from + 0.5);\n  to = floor(to + 0.5);\n  return mask_last(shift_right(num, from), to - from);\n}\n\n/////////////////////////////////////////////////////////////////\n// https://stackoverflow.com/questions/18453302/how-do-you-pack-one-32bit-int-into-4-8bit-ints-in-glsl-webgl\n\nconst vec4 bitEnc = vec4(1.,255.,65025.,16581375.);\nconst vec4 bitDec = 1./bitEnc;\nvec4 EncodeFloatRGBA (float v) {\n  vec4 enc = bitEnc * v;\n  enc = fract(enc);\n  enc -= enc.yzww * vec2(1./255., 0.).xxxy;\n  return enc;\n}\nfloat DecodeFloatRGBA (vec4 v) {\n  return dot(v, bitDec);\n}\n\n/////////////////////////////////////////////////////////////////\n// https://gist.github.com/Flexi23/1713774\n// \nvec2 encode16BitFloatInto2xUInt8(float v) {\n  vec2 c = vec2(0.);\n\n  int signum = (v >= 0.) ? 128 : 0;\n  v = abs(v);\n  int exponent = 15;\n  float limit = 1024.; // considering the bias from 2^-5 to 2^10 (==1024)\n  for(int exp = 15; exp > 0; exp--) {\n    if ( v < limit) {\n      limit /= 2.;\n      exponent--;\n    }\n  }\n\n  float rest;\n  if (exponent == 0) {\n    rest = v / limit / 2.;      // "subnormalize" implicite preceding 0. \n  } \n  else {\n    rest = (v - limit)/limit;   // normalize accordingly to implicite preceding 1.\n  }\n\n  int mantissa = int(rest * 2048.);   // 2048 = 2^11 for the (split) 11 bit mantissa\n  int msb = mantissa / 256;           // the most significant 3 bits go into the lower part of the first byte\n  int lsb = mantissa - msb * 256;     // there go the other 8 bit of the lower significance\n\n  c.x = float(signum + exponent * 8 + msb) / 255.;    // color normalization for texture2D\n  c.y = float(lsb) / 255.;\n\n  if (v >= 2048.) {\n    c.y = 1.;\n  }\n\n  return c;\n}\n\nfloat decode16BitFloatFrom2xUInt8(vec2 c) {\n  float v = 0.;\n\n  int ix = int(c.x*255.); // 1st byte: 1 bit signum, 4 bits exponent, 3 bits mantissa (MSB)\n  int iy = int(c.y*255.); // 2nd byte: 8 bit mantissa (LSB)\n\n  int s = (c.x >= 0.5) ? 1 : -1;\n  ix = (s > 0) ? ix - 128 : ix;   // remove the signum bit from exponent\n  int iexp = ix / 8;              // cut off the last 3 bits of the mantissa to select the 4 exponent bits\n  int msb = ix - iexp * 8;        // subtract the exponent bits to select the 3 most significant bits of the mantissa\n\n  int norm = (iexp == 0) ? 0 : 2048;          // distinguish between normalized and subnormalized numbers\n  int mantissa = norm + msb * 256 + iy;       // implicite preceding 1 or 0 added here\n  norm = (iexp == 0) ? 1 : 0;                 // normalization toggle\n  float exponent = pow( 2., float(iexp + norm) - 16.); // -5 for the the exponent bias from 2^-5 to 2^10 plus another -11 for the normalized 12 bit mantissa \n  v = float( s * mantissa ) * exponent;\n\n  return v;\n}\n\n// TODO : Encoding Float32 to 4x UInt8\n// http://concord-consortium.github.io/lab/experiments/webgl-gpgpu/script.js\n// http://ultraist.hatenablog.com/entry/20110608/1307539319\n\n',GLSLUtils="#define GLSLIFY 1\n\nint ftoi(float val) {\n  return int(floor(val + 0.5));\n}\nivec2 ftoi(vec2 v2) {\n  return ivec2(ftoi(v2.x), ftoi(v2.y));\n}\nivec3 ftoi(vec3 v4) {\n  return ivec3(ftoi(v4.x), ftoi(v4.y), ftoi(v4.z));\n}\nivec4 ftoi(vec4 v4) {\n  return ivec4(ftoi(v4.x), ftoi(v4.y), ftoi(v4.z), ftoi(v4.w));\n}\n\n#ifdef ENABLE_ES3\n\nint imod(int x, int y) {\n  return x % y;\n}\n\nvoid setFlag(inout int flags, int flag) {\n  flags |= flag;\n}\n\nvoid clearFlag(inout int flags, int flag) {\n  flags &= ~flag;\n}\n\nbool testFlag(int flags, int flag) {\n  return (flags & flag) != 0;\n}\n\n// private function: Mangle me...\nivec2 _pixelIndexToUV(int index, int textureWidth) {\n  return ivec2(index % textureWidth, index / textureWidth);\n}\n\nvec4 fetchTexel(sampler2D texture, int textureWidth, int index) {\n  return texelFetch(texture, _pixelIndexToUV(index, textureWidth), 0);\n}\n\nvec4 fetchTexel(sampler2D texture, ivec2 textureSize, ivec2 texCoord) {\n  return texelFetch(texture, texCoord, 0);\n}\n\nvec4 fetchTexel(sampler2D texture, ivec2 textureSize, int index) {\n    return texelFetch(texture, _pixelIndexToUV(index, textureSize.x), 0);\n}\n\n#else\n\n// TODO: integrate: https://gist.github.com/mattatz/70b96f8c57d4ba1ad2cd\n\nint max(int a, int b) {\n  return a > b ? a : b;\n}\nint min(int a, int b) {\n  return a < b ? a : b;\n}\n\nfloat round(float val) {\n  return floor(val + 0.4);\n}\n\nint imod(int x, int y) {\n  return x-y*(x/y);\n}\n\nvoid setFlag(inout int flags, int flag) {\n  flags += flag;\n}\nvoid clearFlag(inout int flags, int flag) {\n  flags -= flag;\n}\n\nbool testFlag(int flags, int flag) {\n  return imod(flags / flag, 2) != 0;\n}\n\n// private function: Mangle me...\nvec2 _pixelIndexToUV(int index, int textureSize) {\n  float flTexSize = float(textureSize);\n  float x = (float(imod(index, textureSize))+0.5)/flTexSize;\n  float y = (floor(float(index / textureSize))+0.5)/flTexSize;\n  return vec2(x, y);\n}\n\nvec4 fetchTexel(sampler2D texture, int textureSize, int index) {\n  vec2 texCoord = _pixelIndexToUV(index, textureSize);\n  return texture2D(texture, texCoord);\n}\n\nvec4 fetchTexel(sampler2D texture, ivec2 textureSize, ivec2 texCoord) {\n  vec2 ftextureSize = vec2(textureSize);\n  return texture2D(texture, (vec2(texCoord) + 0.5) / ftextureSize);\n}\n\n#endif // ENABLE_ES3\n\nint uvToPixelIndex(vec2 uv, int textureSize) {\n  return int(uv.x * float(textureSize)) + (int(floor(uv.y * float(textureSize))) * textureSize);\n}\n\n",Hammersley="#define GLSLIFY 1\nfloat RadicalInverse_VdC(uint bits) \n{\n  bits = (bits << 16u) | (bits >> 16u);\n  bits = ((bits & 0x55555555u) << 1u) | ((bits & 0xAAAAAAAAu) >> 1u);\n  bits = ((bits & 0x33333333u) << 2u) | ((bits & 0xCCCCCCCCu) >> 2u);\n  bits = ((bits & 0x0F0F0F0Fu) << 4u) | ((bits & 0xF0F0F0F0u) >> 4u);\n  bits = ((bits & 0x00FF00FFu) << 8u) | ((bits & 0xFF00FF00u) >> 8u);\n  return float(bits) * 2.3283064365386963e-10; // / 0x100000000\n}\n// ----------------------------------------------------------------------------\nvec2 Hammersley(uint i, uint N)\n{\n  return vec2(float(i)/float(N), RadicalInverse_VdC(i));\n} \n",ImportanceSampleGGX="#define GLSLIFY 1\nimport 'constants.glsl'\n\nvec3 ImportanceSampleGGX(vec2 Xi, vec3 N, float roughness)\n{\n  float a = roughness*roughness;\n\n  float phi = 2.0 * PI * Xi.x;\n  float cosTheta = sqrt((1.0 - Xi.y) / (1.0 + (a*a - 1.0) * Xi.y));\n  float sinTheta = sqrt(1.0 - cosTheta*cosTheta);\n\n  // from spherical coordinates to cartesian coordinates\n  vec3 H = vec3(cos(phi) * sinTheta, sin(phi) * sinTheta, cosTheta);\n\n  // from tangent-space vector to world-space sample vector\n  vec3 up        = abs(N.z) < 0.999 ? vec3(0.0, 0.0, 1.0) : vec3(1.0, 0.0, 0.0);\n  vec3 tangent   = normalize(cross(up, N));\n  vec3 bitangent = cross(N, tangent);\n\n  vec3 sampleVec = tangent * H.x + bitangent * H.y + N * H.z;\n  return normalize(sampleVec);\n} \n",materialparams="#define GLSLIFY 1\nimport 'GLSLUtils.glsl'\nimport 'gamma.glsl'\n\nconst int FatPointsShaderId = 1;\nconst int FlatSurfaceShaderId = 2;\nconst int LinesShaderId = 3;\nconst int PointsShaderId = 4;\nconst int ScreenSpaceShaderId = 5;\nconst int SimpleSurfaceShaderId = 6;\nconst int StandardSurfaceShaderId = 7;\n\nconst int HeaderId = 0;\nconst int BaseColorId = 1;\n \nuniform sampler2D materialsTexture;\nuniform highp ivec2 materialsTextureSize;\n\nvec4 getMaterialValue(vec2 materialCoords, int valueIndex) {\n  int index = ftoi(materialCoords.x) + valueIndex;\n  ivec2 texelCoords = ivec2(imod(index, materialsTextureSize.x), index / materialsTextureSize.x);\n  \n  return fetchTexel(materialsTexture, materialsTextureSize, texelCoords);\n}\n\n////////////////////////\n// Material Param Helpers.\n\nvoid getTextureColorValue(inout vec4 value, sampler2D tex, int texType, vec2 texCoord) {\n  if (texType == 1 || texType == 2) {\n    // Note: we assume textures are always in gamma space, and must be converted\n    // to linear. I cann't find evidence that 8-bit textures can be in linear space.\n    // TODO: Use SRGB textures.\n    value = toLinear(texture2D(tex, texCoord));\n  }\n  else if (texType == 3) {\n    // Float HDR Texture. We assume these textures are in linear space.\n    value = texture2D(tex, texCoord);\n  }\n}\n\nvec4 getColorParamValue(in vec4 value, sampler2D tex, int texType, vec2 texCoord) {\n  // for backwards compatiblity with zea-ux shader\n  vec4 tmp = value;\n  getTextureColorValue(tmp, tex, texType, texCoord);\n  return tmp;\n}\n\nfloat luminanceFromRGB(vec3 rgb) {\n  return 0.2126*rgb.r + 0.7152*rgb.g + 0.0722*rgb.b;\n}\n\nvoid getTextureLuminanceValue(inout float value, sampler2D tex, int texType, vec2 texCoord) {\n  if (texType != 0)\n    value = luminanceFromRGB(texture2D(tex, texCoord).rgb);\n}\n\nfloat getLuminanceParamValue(in float value, sampler2D tex, int texType, vec2 texCoord) {\n  // for backwards compatiblity with zea-ux shader\n  float tmp = value;\n  getTextureLuminanceValue(tmp, tex, texType, texCoord);\n  // for backwards compatiblity with zea-ux shader\n  return tmp;\n}\n\n",modelMatrix="#define GLSLIFY 1\n#ifdef ENABLE_FLOAT_TEXTURES\nimport 'GLSLUtils.glsl'\nimport 'transpose.glsl'\nmat4 getMatrix(sampler2D texture, int textureSize, int index) {\n  // Unpack 3 x 4 matrix columns into a 4 x 4 matrix.\n  vec4 col0 = fetchTexel(texture, textureSize, (index * pixelsPerItem) + 1);\n  vec4 col1 = fetchTexel(texture, textureSize, (index * pixelsPerItem) + 2);\n  vec4 col2 = fetchTexel(texture, textureSize, (index * pixelsPerItem) + 3);\n  mat4 result = transpose(mat4(col0, col1, col2, vec4(0.0, 0.0, 0.0, 1.0)));\n  return result;\n}\n\nmat4 getModelMatrix(int id) {\n  return getMatrix(instancesTexture, instancesTextureSize, id);\n}\n\n#else\n\nuniform mat4 modelMatrix;\n\nmat4 getModelMatrix(int id) {\n  return modelMatrix;\n}\n\n#endif\n\n",PBRSurfaceRadiance="#define GLSLIFY 1\nimport 'GLSLUtils.glsl'\nconst int ENVMAP_FLAG_HEADLIGHT =  1; // 1<<0;\n\nstruct MaterialParams {\n  vec3 baseColor;\n  float ambientOcclusion;\n  float metallic;\n  float roughness;\n  float reflectance;\n  float opacity;\n  float emission;\n};\n\n#ifndef ENABLE_PBR\n\nvec4 pbrSurfaceRadiance(in MaterialParams material, vec3 normal, in vec3 viewVector) {\n  vec3 irradiance = vec3(dot(normal, viewVector));\n  float ao = material.ambientOcclusion; \n  return vec4(material.baseColor * ao * irradiance + (material.emission * material.baseColor), material.opacity);\n\n  // return vec4(material.baseColor * ao * irradiance , material.opacity);\n}\n\n#else\n\nuniform int envMapFlags;\nuniform samplerCube irradianceMap;\nuniform samplerCube prefilterMap;\nuniform sampler2D brdfLUT;\n\nvec3 sampleIrradiance(vec3 dir) {\n  return texture(irradianceMap, dir).rgb;\n}\n\nvec3 fresnelSchlickRoughness(float cosTheta, vec3 F0, float roughness) {\n  return F0 + (max(vec3(1.0 - roughness), F0) - F0) * pow(max(1.0 - cosTheta, 0.0), 5.0);\n}\n\nfloat luminance(vec3 color) {\n  return 0.2126 * color.r + 0.7152 * color.g + 0.0722 * color.b;\n}\n\nvec4 pbrSurfaceRadiance(in MaterialParams material, vec3 normal, in vec3 viewVector) {\n  if (envMapFlags == -1) {\n    vec3 irradiance = vec3(dot(normal, viewVector));\n    float ao = material.ambientOcclusion; \n    return vec4(material.baseColor * ao * irradiance + (material.emission * material.baseColor), material.opacity);\n  }\n\n  vec3 N = normal;\n  vec3 V = viewVector;\n  vec3 R = reflect(-V, N);\n  float roughness = material.roughness * material.roughness;\n  vec3 diffuseColor = (1.0 - material.metallic) * material.baseColor;\n\n  // Note: The specular reflectance of metallic surfaces is chromatic\n  // https://google.github.io/filament/Filament.html#listing_fnormal\n  vec3 F0 = 0.16 * material.reflectance * material.reflectance * (1.0 - material.metallic) + material.baseColor * material.metallic;\n\n  float NdotV = dot(N, V);\n\n  vec3 F = fresnelSchlickRoughness(max(NdotV, 0.0), F0, roughness);\n\n  vec3 kS = F;\n  vec3 kD = 1.0 - kS;\n  kD *= 1.0 - material.metallic;\n  float ao = material.ambientOcclusion; \n  \n  vec3 irradiance;\n  vec3 irradianceSampleDir = normal;\n  \n  bool headLightMode = testFlag(envMapFlags, ENVMAP_FLAG_HEADLIGHT);\n  if (headLightMode) {\n    irradianceSampleDir = viewVector;\n  }\n  irradiance = sampleIrradiance(irradianceSampleDir);\n  // vec3 irradiance = shGetIrradianceAt(shCoefficients, N);\n  vec3 diffuse    = irradiance * diffuseColor;\n  \n  const float MAX_REFLECTION_LOD = 4.0;\n  vec3 prefilteredColor = textureLod(prefilterMap, R,  roughness * MAX_REFLECTION_LOD).rgb;   \n  vec2 envBRDF  = texture(brdfLUT, vec2(max(NdotV, 0.0), roughness)).rg;\n  vec3 specular = prefilteredColor * (F * envBRDF.x + envBRDF.y);\n  \n  vec3 radiance = (kD * diffuse + specular) * ao;\n  \n  // Now handle semi-transparent objects. We need to be able to linearly interpolate\n  // opacity to make objects disappear, so we need a continuous change.\n  float opacity = material.opacity;\n  vec4 transparent = vec4((radiance * opacity) + specular, opacity + luminance(specular) + luminance(F));\n  vec4 result = mix(transparent, vec4(radiance, 1.0), opacity);\n\n  // Add emission on as the final component.\n  // Note: emission allows a material to blend off its specular component, \n  // which can also be used to make an object completely disappear if also transparent.\n  return mix(result, vec4(material.baseColor, opacity), material.emission);\n}\n\n#endif // ENABLE_PBR\n",SHCoeffs="#define GLSLIFY 1\nuniform vec3 shCoeffs[9];\n\nvec3 sampleSHCoeffs(vec3 dir) {\n  // dir is assumed to have unit length\n  float x = dir.x, y = dir.y, z = dir.z;\n  // band 0\n  vec3 result = shCoeffs[ 0 ] * 0.886227;\n  // band 1\n  result += shCoeffs[ 1 ] * 2.0 * 0.511664 * y;\n  result += shCoeffs[ 2 ] * 2.0 * 0.511664 * z;\n  result += shCoeffs[ 3 ] * 2.0 * 0.511664 * x;\n  // band 2\n  result += shCoeffs[ 4 ] * 2.0 * 0.429043 * x * y;\n  result += shCoeffs[ 5 ] * 2.0 * 0.429043 * y * z;\n  result += shCoeffs[ 6 ] * ( 0.743125 * z * z - 0.247708 );\n  result += shCoeffs[ 7 ] * 2.0 * 0.429043 * x * z;\n  result += shCoeffs[ 8 ] * 0.429043 * ( x * x - y * y );\n  return result;\n}\n",gamma="#define GLSLIFY 1\nconst float gamma_const = 2.2;\n\nfloat toLinear(float v) {\n  return pow(v, gamma_const);\n}\n\nvec2 toLinear(vec2 v) {\n  return pow(v, vec2(gamma_const));\n}\n\nvec3 toLinear(vec3 v) {\n  return pow(v, vec3(gamma_const));\n}\n\nvec4 toLinear(vec4 v) {\n  return vec4(toLinear(v.rgb), v.a);\n}\n\nfloat toGamma(float v) {\n  return pow(v, 1.0 / gamma_const);\n}\n\nvec2 toGamma(vec2 v) {\n  return pow(v, vec2(1.0 / gamma_const));\n}\n\nvec3 toGamma(vec3 v) {\n  return pow(v, vec3(1.0 / gamma_const));\n}\n\nvec4 toGamma(vec4 v) {\n  return vec4(toGamma(v.rgb), v.a);\n}\n\nfloat toGamma(float v, float gamma) {\n  return pow(v, 1.0 / gamma);\n}\n\nvec2 toGamma(vec2 v, float gamma) {\n  return pow(v, vec2(1.0 / gamma));\n}\n\nvec3 toGamma(vec3 v, float gamma) {\n  return pow(v, vec3(1.0 / gamma));\n}\n\nvec4 toGamma(vec4 v, float gamma) {\n  return vec4(toGamma(v.rgb, gamma), v.a);\n}\n\n",inverse="#define GLSLIFY 1\n\n#ifndef ENABLE_ES3\n\nfloat inverse(float m) {\n  return 1.0 / m;\n}\n\nmat2 inverse(mat2 m) {\n  return mat2(m[1][1],-m[0][1],\n             -m[1][0], m[0][0]) / (m[0][0]*m[1][1] - m[0][1]*m[1][0]);\n}\n\nmat3 inverse(mat3 m) {\n  float a00 = m[0][0], a01 = m[0][1], a02 = m[0][2];\n  float a10 = m[1][0], a11 = m[1][1], a12 = m[1][2];\n  float a20 = m[2][0], a21 = m[2][1], a22 = m[2][2];\n\n  float b01 = a22 * a11 - a12 * a21;\n  float b11 = -a22 * a10 + a12 * a20;\n  float b21 = a21 * a10 - a11 * a20;\n\n  float det = a00 * b01 + a01 * b11 + a02 * b21;\n\n  return mat3(b01, (-a22 * a01 + a02 * a21), (a12 * a01 - a02 * a11),\n              b11, (a22 * a00 - a02 * a20), (-a12 * a00 + a02 * a10),\n              b21, (-a21 * a00 + a01 * a20), (a11 * a00 - a01 * a10)) / det;\n}\n\nmat4 inverse(mat4 m) {\n  float\n      a00 = m[0][0], a01 = m[0][1], a02 = m[0][2], a03 = m[0][3],\n      a10 = m[1][0], a11 = m[1][1], a12 = m[1][2], a13 = m[1][3],\n      a20 = m[2][0], a21 = m[2][1], a22 = m[2][2], a23 = m[2][3],\n      a30 = m[3][0], a31 = m[3][1], a32 = m[3][2], a33 = m[3][3],\n\n      b00 = a00 * a11 - a01 * a10,\n      b01 = a00 * a12 - a02 * a10,\n      b02 = a00 * a13 - a03 * a10,\n      b03 = a01 * a12 - a02 * a11,\n      b04 = a01 * a13 - a03 * a11,\n      b05 = a02 * a13 - a03 * a12,\n      b06 = a20 * a31 - a21 * a30,\n      b07 = a20 * a32 - a22 * a30,\n      b08 = a20 * a33 - a23 * a30,\n      b09 = a21 * a32 - a22 * a31,\n      b10 = a21 * a33 - a23 * a31,\n      b11 = a22 * a33 - a23 * a32,\n\n      det = b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;\n\n  return mat4(\n      a11 * b11 - a12 * b10 + a13 * b09,\n      a02 * b10 - a01 * b11 - a03 * b09,\n      a31 * b05 - a32 * b04 + a33 * b03,\n      a22 * b04 - a21 * b05 - a23 * b03,\n      a12 * b08 - a10 * b11 - a13 * b07,\n      a00 * b11 - a02 * b08 + a03 * b07,\n      a32 * b02 - a30 * b05 - a33 * b01,\n      a20 * b05 - a22 * b02 + a23 * b01,\n      a10 * b10 - a11 * b08 + a13 * b06,\n      a01 * b08 - a00 * b10 - a03 * b06,\n      a30 * b04 - a31 * b02 + a33 * b00,\n      a21 * b02 - a20 * b04 - a23 * b00,\n      a11 * b07 - a10 * b09 - a12 * b06,\n      a00 * b09 - a01 * b07 + a02 * b06,\n      a31 * b01 - a30 * b03 - a32 * b00,\n      a20 * b03 - a21 * b01 + a22 * b00) / det;\n}\n\n#endif\n\n",transpose="#define GLSLIFY 1\n\n#ifndef ENABLE_ES3\n\nfloat transpose(float m) {\n  return m;\n}\n\nmat2 transpose(mat2 m) {\n  return mat2(m[0][0], m[1][0],\n              m[0][1], m[1][1]);\n}\n\nmat3 transpose(mat3 m) {\n  return mat3(m[0][0], m[1][0], m[2][0],\n              m[0][1], m[1][1], m[2][1],\n              m[0][2], m[1][2], m[2][2]);\n}\n\nmat4 transpose(mat4 m) {\n  return mat4(m[0][0], m[1][0], m[2][0], m[3][0],\n              m[0][1], m[1][1], m[2][1], m[3][1],\n              m[0][2], m[1][2], m[2][2], m[3][2],\n              m[0][3], m[1][3], m[2][3], m[3][3]);\n}\n\n#endif\n\n",quadVertexFromID="#define GLSLIFY 1\nattribute float vertexIDs;\n\nvec2 getQuadVertexPositionFromID() {\n  int vertexID = int(vertexIDs);\n  if (vertexID == 0)\n    return vec2(-0.5, -0.5);\n  else if (vertexID == 1)\n    return vec2(0.5, -0.5);\n  else if (vertexID == 2)\n    return vec2(-0.5, 0.5);\n  else if (vertexID == 3)\n    return vec2(0.5, 0.5);\n  return vec2(0,0);\n}\n",unpackHDR="#define GLSLIFY 1\n\nvec3 decodeHDR(const in vec3 ldrPixel, const in float cdmAlpha) {\n  float avg = (cdmAlpha * 16.0 - 8.0);\n  float scl = 1.0;\n  vec3 color;\n  color.x = (tan((ldrPixel.x-0.5)*1.5)/scl)+avg;\n  color.y = (tan((ldrPixel.y-0.5)*1.5)/scl)+avg;\n  color.z = (tan((ldrPixel.z-0.5)*1.5)/scl)+avg;\n\n  // convert from logarithmic curve to linear curve.\n  // subtract the epsilon that was added during encoding.\n  const float eps = 0.001;\n  color.x = pow(10.0, color.x) - eps;\n  color.y = pow(10.0, color.y) - eps;\n  color.z = pow(10.0, color.z) - eps;\n  return color;\n}\n\nvec3 decodeHDR(sampler2D ldrSampler, sampler2D cdmSampler, vec2 texCoord) {\n#ifdef ENABLE_ES3\n  float cdm = texture2D(cdmSampler, texCoord).r;\n#else\n  float cdm = texture2D(cdmSampler, texCoord).a;\n#endif\n  return decodeHDR(texture2D(ldrSampler, texCoord).rgb, cdm);\n}\n\n",surfaceGeomData="#define GLSLIFY 1\n\n  uniform int floatGeomBuffer; // Remove once we update UX. \n  uniform int passId;\n  uniform int occlusionCulling;\n  uniform int debugGeomDataBuffer;\n\n  import 'geomItemFlags.glsl'\n  import 'GLSLBits.glsl'\n  import 'geomType.glsl'\n  \n  vec4 setFragColor_geomData(vec3 v_viewPos, float geomItemId, float elemItemId, int isOrthographic, int flags, int geomType){\n    vec4 fragColor;\n  \n    if (occlusionCulling != 0) {\n      // Transparent geoms do not render to the occlusion buffer\n      if (testFlag(flags, GEOMITEM_TRANSPARENT)) {\n        discard;\n        return fragColor;\n      }\n\n      // Calculate a simple stochastic transparency to reduce the cost of the reduction shader.\n      // We only need one pixel to be visible to consider the geometry visible, so here we \n      // keep only one in 7x7 (49) pixels. This signficatly reduces the cost of reducing the scene\n      // to the reduction data buffer.\n      // Note: this is not a stochastic transparency, as we are simply setting black pixels to \n      // reduce the number of reduction points for this geometry. The black pixels (not discarded)\n      // still occlude other geometries, but during reduction count to nothing.\n      int x = int(gl_FragCoord.x * 1000.0);\n      int y = int(gl_FragCoord.y * 1000.0);\n      if (x % 7 != 0 || y % 7 != 0) {\n        fragColor = vec4(0.0, 0.0, 0.0, 1.0);\n      } else {\n        // Write out just the green channel for the ReductionShader\n        fragColor.r = float(geomItemId);\n\n        // Write out a value to the Green channel to indicate this is not a BBox.\n        // We propagate this value in the ReductionShader to the output histogram\n        // to determine if we should trigger loading of placeholders. (BBox should not)\n        fragColor.g = 1.0;\n\n        \n        // Note: the alpha value is simply so we can debug the buffer over the top of the viewport.\n        fragColor.a = 1.0;\n      }\n    }\n    else {\n      // When not occlusion culling, we don't render non-selectable objects.\n      //When occlusion culling, we want to draw all the geom data for all objects.\n      if (testFlag(flags, GEOMITEM_INVISIBLE_IN_GEOMDATA)) {\n        discard;\n        return fragColor;\n      }\n\n      float viewDist;\n      if (isOrthographic > 0) {\n        viewDist = abs(v_viewPos.z);\n      } else {\n        viewDist = length(v_viewPos);\n      }\n      \n      fragColor.r = float(passId); \n      fragColor.g = float(geomItemId);\n      fragColor.b = elemItemId;\n      fragColor.a = viewDist;\n      \n      if (debugGeomDataBuffer == 1) {\n        if (geomType == TRIANGLES) {\n          fragColor = vec4(1.0, 0.0, 0.0, 1.0); \n        }\n        else if (geomType == LINES) {\n          fragColor = vec4(0.0, 1.0, 0.0, 1.0); \n        } \n        else if (geomType == POINTS) {\n          fragColor = vec4(0.0, 0.0, 1.0, 1.0); \n        } \n      }\n\n    }\n\n    return fragColor;\n  }\n  vec4 setFragColor_geomData(vec3 v_viewPos, float geomItemId, float elemItemId, int isOrthographic, int flags){\n    return setFragColor_geomData(v_viewPos, geomItemId, 0.0, isOrthographic, flags, TRIANGLES);\n  }\n\n  vec4 setFragColor_geomData(vec3 v_viewPos, int floatGeomBuffer, int passId, float geomItemId, int isOrthographic){\n    return setFragColor_geomData(v_viewPos, geomItemId, 0.0, isOrthographic, 0);\n  }\n  vec4 setFragColor_geomData(vec3 v_viewPos, int floatGeomBuffer, int passId, float geomItemId, float elemItemId, int isOrthographic){\n    return setFragColor_geomData(v_viewPos, geomItemId, elemItemId, isOrthographic, 0);\n  }",surfaceHighlight="#define GLSLIFY 1\nimport 'GLSLUtils.glsl'\nimport 'drawItemTexture.glsl'\n\nvec4 getHighlightColor(int id) {\n  return fetchTexel(instancesTexture, instancesTextureSize, (id * pixelsPerItem) + 4);\n}\n\nvec4 setFragColor_highlight(float geomItemId){\n  vec4 fragColor = getHighlightColor(int(round(geomItemId)));\n  return fragColor;\n}\n",imageAtlas="#define GLSLIFY 1\n// Note: On mobile, I can't seem to pass around a stuct containing sampler2D.\n// I have to unpack the struct and pass its members. :(\n// struct ImageAtlas {\n//     sampler2D layout;\n//     sampler2D image;\n//     vec4 desc;\n// };\n\nimport 'GLSLUtils.glsl'\n\nvec4 getSubImageLayout(int index, in sampler2D atlasLayout, in vec4 atlasDesc){\n    return fetchTexel(atlasLayout, int(floor(atlasDesc.z+0.5)), index);\n}\nvec2 calcSubImageTexCoords(vec2 texCoord, int index, in sampler2D atlasLayout, in vec4 atlasDesc){\n    vec4 layoutData = fetchTexel(atlasLayout, int(floor(atlasDesc.z+0.5)), index);\n    // The following line is a hack to fix artifacts in our PBR lighting\n    // We were seeing loads of lighting garbage on some sufaces that were orthogonal\n    // to the world. The UV coordinates would have been landing right on the edges\n    // of our subimages and were often sampling outside the image. This couuld\n    // have been because of filtering, or an error in the uv coords. \n    texCoord = clamp(texCoord, vec2(0.01, 0.01), vec2(0.99, 0.99));\n    vec2 subimageTexel = texCoord * layoutData.zw;\n    // subimageTexel = clamp(subimageTexel, vec2(0.0, 0.0), vec2(1.0, 1.0));\n    return subimageTexel + layoutData.xy;\n}\nvec4 sampleSubImage(vec2 texCoord, int index, in sampler2D atlasLayout, in sampler2D atlasImage, in vec4 atlasDesc){\n    vec4 layoutData = fetchTexel(atlasLayout, int(floor(atlasDesc.z+0.5)), index);\n    vec2 atlasCoords = calcSubImageTexCoords(texCoord, index, atlasLayout, atlasDesc);\n    return texture2D(atlasImage, atlasCoords);\n}";shaderLibrary.setShaderModule("imageAtlas.glsl",imageAtlas),shaderLibrary.setShaderModule("surfaceGeomData.glsl",surfaceGeomData),shaderLibrary.setShaderModule("surfaceHighlight.glsl",surfaceHighlight),shaderLibrary.setShaderModule("computeViewNormal.glsl",computeViewNormal),shaderLibrary.setShaderModule("calcFatLinesViewPos.glsl",calcFatLinesViewPos),shaderLibrary.setShaderModule("constants.glsl",constants),shaderLibrary.setShaderModule("convolve-helpers.glsl",convolveHelpers),shaderLibrary.setShaderModule("cutaways.glsl",cutaways),shaderLibrary.setShaderModule("debugColors.glsl",debugColors),shaderLibrary.setShaderModule("drawItemId.glsl",geomItemId),shaderLibrary.setShaderModule("geomItemId.glsl",geomItemId),shaderLibrary.setShaderModule("geomItemFlags.glsl",geomItemFlags),shaderLibrary.setShaderModule("geomType.glsl",geomType),shaderLibrary.setShaderModule("geometryMask.glsl",geometryMask),shaderLibrary.setShaderModule("drawItemTexture.glsl",drawItemTexture),shaderLibrary.setShaderModule("envmap-dualfisheye.glsl",envmapDualfisheye),shaderLibrary.setShaderModule("envmap-equirect.glsl",envmapEquirect),shaderLibrary.setShaderModule("envmap-octahedral.glsl",envmapOctahedral),shaderLibrary.setShaderModule("GLSLBits.glsl",GLSLBits),shaderLibrary.setShaderModule("GLSLUtils.glsl",GLSLUtils),shaderLibrary.setShaderModule("Hammersley.glsl",Hammersley),shaderLibrary.setShaderModule("ImportanceSampleGGX.glsl",ImportanceSampleGGX),shaderLibrary.setShaderModule("materialparams.glsl",materialparams),shaderLibrary.setShaderModule("modelMatrix.glsl",modelMatrix),shaderLibrary.setShaderModule("PBRSurfaceRadiance.glsl",PBRSurfaceRadiance),shaderLibrary.setShaderModule("SHCoeffs.glsl",SHCoeffs),shaderLibrary.setShaderModule("gamma.glsl",gamma),shaderLibrary.setShaderModule("inverse.glsl",inverse),shaderLibrary.setShaderModule("transpose.glsl",transpose),shaderLibrary.setShaderModule("quadVertexFromID.glsl",quadVertexFromID),shaderLibrary.setShaderModule("unpackHDR.glsl",unpackHDR);var frag$i=" \nprecision highp float;\n#define GLSLIFY 1\n\nvarying vec2 v_texCoord;\nuniform sampler2D ldrSampler;\nuniform sampler2D cdmSampler;\nuniform vec4 srcRegion; // pos, and size of the source region\n\nimport 'unpackHDR.glsl'\n\n#ifdef ENABLE_ES3\n  out vec4 fragColor;\n#endif\nvoid main(void) {\n\n#ifndef ENABLE_ES3\n  vec4 fragColor;\n#endif\n\n  vec2 srcUv = srcRegion.xy + (v_texCoord * srcRegion.zw);\n\n  fragColor = vec4(decodeHDR(ldrSampler, cdmSampler, srcUv), 1.0);\n\n#ifndef ENABLE_ES3\n  gl_FragColor = fragColor;\n#endif\n}\n\n",vert$j="\nprecision highp float;\n#define GLSLIFY 1\n\nimport 'quadVertexFromID.glsl'\n\n/* VS Outputs */\nvarying vec2 v_texCoord;\n \nvoid main()\n{\n  vec2 position = getQuadVertexPositionFromID();\n  v_texCoord = position+0.5;\n  gl_Position = vec4(position*2.0, 0.0, 1.0);\n}\n";class UnpackHDRShader extends GLShader{constructor(A){super(A,"UnpackHDRShader"),this.setShaderStage("VERTEX_SHADER",vert$j),this.setShaderStage("FRAGMENT_SHADER",frag$i)}}const genDataTypeDesc=(A,g)=>{let I,C,e;switch(g){case"UInt8":I=1,C=4,e=A.UNSIGNED_BYTE;break;case"SInt8":I=1,C=1,e=A.BYTE;break;case"UInt16":I=1,C=2,e=A.UNSIGNED_SHORT;break;case"SInt16":I=1,C=2,e=A.SHORT;break;case"UInt32":I=1,C=4,e=A.UNSIGNED_INT;break;case"SInt32":I=1,C=4,e=A.INT;break;case"Float32":I=1,C=4,e=A.FLOAT;break;case"Vec2f8":I=2,C=1,e=A.BYTE;break;case"Vec2f16":I=2,C=2,e=A.HALF_FLOAT;break;case"Vec2":I=2,C=4,e=A.FLOAT;break;case"Vec3f8":I=3,C=1,e=A.BYTE;break;case"Vec3f16":I=3,C=2,e=A.HALF_FLOAT;break;case"Vec3":I=3,C=4,e=A.FLOAT;break;case"Vec4":case"Color":I=4,C=4,e=A.FLOAT;break;case"Vec4f8":I=4,C=1,e=A.BYTE;break;case"Vec4f16":I=4,C=2,e=A.HALF_FLOAT;break;case"RGBA":I=4,C=1,e=A.UNSIGNED_BYTE;break;default:throw"Unhandled Type:"+g}return{name:g,dimension:I,elementSize:C,dataType:e}};class IGeomShaderBinding{}class GeomShaderBinding extends IGeomShaderBinding{gl;shaderAttrs;glattrbuffers;indexBuffer;constructor(A,g,I,C){super(),this.gl=A,this.shaderAttrs=g,this.glattrbuffers=I,this.indexBuffer=C}bind(A){const g=this.gl;for(const A in this.shaderAttrs){if("instancedIds"==A)continue;const I=this.shaderAttrs[A],C=I.location;if(-1==C)continue;const e=this.glattrbuffers[A];if(!e){g.disableVertexAttribArray(C);continue}const i=e.dimension,t=e.dataType,o=e.normalized,a=I.integer,B=i*e.elementSize,s=null!=e.offset?e.offset*i*e.elementSize:0,l=I.instanced;g.enableVertexAttribArray(C),g.bindBuffer(g.ARRAY_BUFFER,e.buffer),a?g.vertexAttribIPointer(C,i,t,B,s):g.vertexAttribPointer(C,i,t,o,B,s),1==l?g.vertexAttribDivisor(C,1):g.vertexAttribDivisor(C,0)}return this.indexBuffer&&g.bindBuffer(g.ELEMENT_ARRAY_BUFFER,this.indexBuffer),!0}unbind(A){const g=this.gl;for(const A in this.shaderAttrs){const I=this.shaderAttrs[A],C=I.location;-1==C&&g.enableVertexAttribArray(C),I.instanced&&g.vertexAttribDivisor(C,0)}g.bindBuffer(g.ELEMENT_ARRAY_BUFFER,null)}destroy(){}}class VAOGeomShaderBinding extends IGeomShaderBinding{vao;gl;indexBuffer;constructor(A,g,I,C){super(),this.gl=A,this.vao=A.createVertexArray(),A.bindVertexArray(this.vao);for(const C in g){if("instancedIds"==C)continue;const e=g[C],i=e.location;if(-1==i)continue;let t=I[C],o=0;if(!t&&(C.endsWith("Next")&&(t=I[C.substring(0,C.length-4)],o=1*t.dimension*t.elementSize),!t)){A.disableVertexAttribArray(i);continue}const a=t.dimension,B=t.dataType,s=t.dimension*t.elementSize,l=e.integer,n=e.instanced;if(A.enableVertexAttribArray(i),A.bindBuffer(A.ARRAY_BUFFER,t.buffer),l)A.vertexAttribIPointer(i,a,B,s,o);else{const g=1==t.normalized;A.vertexAttribPointer(i,a,B,g,s,o)}A.vertexAttribDivisor&&(1==n?A.vertexAttribDivisor(i,1):A.vertexAttribDivisor(i,0))}this.indexBuffer=C,this.indexBuffer&&A.bindBuffer(A.ELEMENT_ARRAY_BUFFER,this.indexBuffer)}bind(A){const g=this.gl;return g.bindVertexArray(this.vao),this.indexBuffer&&g.bindBuffer(g.ELEMENT_ARRAY_BUFFER,this.indexBuffer),!0}unbind(A){const g=this.gl;g.bindVertexArray(null),this.indexBuffer&&g.bindBuffer(g.ELEMENT_ARRAY_BUFFER,null)}destroy(){const A=this.gl;this.indexBuffer&&(A.bindVertexArray(this.vao),A.bindBuffer(A.ELEMENT_ARRAY_BUFFER,null)),A.deleteVertexArray(this.vao)}}function generateShaderGeomBinding(A,g,I,C){return null==A.createVertexArray?new GeomShaderBinding(A,g,I,C):new VAOGeomShaderBinding(A,g,I,C)}class GLHDRImage extends GLTexture2D{listenerIDs={};hdrImage;fbo=null;srcLDRTex=null;srcCDMTex=null;unpackHDRShader=null;shaderBinding=null;constructor(A,g){super(A),this.hdrImage=g;const I=()=>{this.__unpackHDRImage(this.hdrImage.getParams())};this.listenerIDs.updated=this.hdrImage.on("updated",I),this.hdrImage.isLoaded()?I():this.listenerIDs.loaded=this.hdrImage.on("loaded",I)}getImage(){return this.hdrImage}__unpackHDRImage(A){const g=this.gl,I=A.data.ldr,C=A.data.cdm;if(this.fbo)this.srcLDRTex.bufferData(I),this.srcCDMTex.bufferData(C);else{this.configure({format:this.gl.RGBA,type:this.gl.HALF_FLOAT,width:I.width,height:I.height,filter:this.gl.LINEAR,wrap:this.gl.CLAMP_TO_EDGE}),this.fbo=new GLFbo(this.gl,this),this.fbo.setClearColor(new Color(0,0,0,0)),this.srcLDRTex=new GLTexture2D(this.gl,{format:this.gl.RGB,type:this.gl.UNSIGNED_BYTE,width:I.width,height:I.height,filter:this.gl.NEAREST,mipMapped:!1,wrap:this.gl.CLAMP_TO_EDGE,data:I}),this.srcCDMTex=new GLTexture2D(this.gl,{format:this.gl.RED,type:this.gl.UNSIGNED_BYTE,width:I.width,height:I.height,filter:this.gl.NEAREST,mipMapped:!1,wrap:this.gl.CLAMP_TO_EDGE,data:C}),this.unpackHDRShader=new UnpackHDRShader(this.gl);const A=this.unpackHDRShader.compileForTarget("GLHDRImage",["#define ENABLE_ES3"]);this.shaderBinding=generateShaderGeomBinding(this.gl,A.attrs,g.__quadattrbuffers,g.__quadIndexBuffer)}const e=new RenderState(this.gl);this.fbo.bindForWriting(e),this.fbo.clear(),this.unpackHDRShader.bind(e,"GLHDRImage"),this.shaderBinding.bind(e);const i=e.unifs;this.srcLDRTex.bindToUniform(e,i.ldrSampler),this.srcCDMTex.bindToUniform(e,i.cdmSampler),g.uniform4fv(i.srcRegion.location,[0,0,1,1]),g.drawQuad(),this.fbo.unbindForWriting(e),this.emit("updated")}bindToUniform(A,g,I){return super.bindToUniform(A,g,I)}destroy(){super.destroy(),this.fbo&&(this.fbo.destroy(),this.srcLDRTex.destroy(),this.srcCDMTex.destroy()),this.unpackHDRShader&&this.unpackHDRShader.destroy(),this.shaderBinding&&this.shaderBinding.destroy(),"loaded"in this.listenerIDs&&this.hdrImage.off("loaded",this.listenerIDs.loaded),this.hdrImage.off("updated",this.listenerIDs.updated)}}class GLGeom extends RefCounted{__gl;geom;numVertices=-1;glattrbuffers={};shaderBindings={};buffersDirty=!0;genBufferOpts={};indexBuffer=null;constructor(A,g){super(),this.__gl=A,this.geom=g;this.geom.on("geomDataChanged",(A=>{this.dirtyBuffers(A)}));this.geom.on("geomDataTopologyChanged",(A=>{this.clearBuffers(),this.dirtyBuffers(A)}))}getGeom(){return this.geom}dirtyBuffers(A){this.genBufferOpts=A,this.buffersDirty=!0,this.emit("updated")}genBuffers(A){const g=this.__gl,I=this.geom.genBuffers();for(const C in I.attrBuffers){if(!A.attrs[C])continue;this.glattrbuffers[C]&&this.glattrbuffers[C].buffer&&g.deleteBuffer(this.glattrbuffers[C].buffer);const e=I.attrBuffers[C],i=genDataTypeDesc(g,e.dataType),t=g.createBuffer();g.bindBuffer(g.ARRAY_BUFFER,t),g.bufferData(g.ARRAY_BUFFER,e.values,g.STATIC_DRAW),this.glattrbuffers[C]={dataType:i.dataType,name:C,dimension:e.dimension,elementSize:i.elementSize,normalized:!1,shared:!1,numValues:e.count,buffer:t}}this.numVertices=this.geom.getNumVertices(),this.buffersDirty=!1}updateBuffers(A){if(this.numVertices!=this.geom.getNumVertices())return void this.genBuffers(A);const g=this.__gl,I=this.geom.genBuffers({includeIndices:!1});for(const A in I.attrBuffers){const C=I.attrBuffers[A],e=this.glattrbuffers[A];g.bindBuffer(g.ARRAY_BUFFER,e.buffer),g.bufferData(g.ARRAY_BUFFER,C.values,g.STATIC_DRAW)}this.buffersDirty=!1}bind(A){this.buffersDirty&&this.updateBuffers(A);let g=this.shaderBindings[A.shaderkey];if(!g){g=generateShaderGeomBinding(this.__gl,A.attrs,this.glattrbuffers,this.indexBuffer),this.shaderBindings[A.shaderkey]=g}g.bind(A)}unbind(A){const g=this.shaderBindings[A.shaderkey];g&&g.unbind(A)}draw(A){throw new Error("Not implemented. Implement this method in a derived class.")}drawInstanced(A,g){throw new Error("Not implemented. Implement this method in a derived class.")}bindAndDraw(A){this.bind(A),this.draw(A)}clearBuffers(){const A=this.__gl;for(const g in this.glattrbuffers){const I=this.glattrbuffers[g];I.shared||A.deleteBuffer(I.buffer)}this.glattrbuffers={};for(const A in this.shaderBindings){this.shaderBindings[A].destroy()}this.shaderBindings={}}destroy(){this.clearBuffers(),super.destroy()}}class GLMesh extends GLGeom{numTriIndices=0;indexDataType=0;numTriangles=0;constructor(A,g){super(A,g)}genBuffers(A){super.genBuffers(A);const g=this.__gl,I=this.geom.genBuffers(),C=I.indices;C instanceof Uint8Array&&(this.indexDataType=this.__gl.UNSIGNED_BYTE),C instanceof Uint16Array&&(this.indexDataType=this.__gl.UNSIGNED_SHORT),C instanceof Uint32Array&&(this.indexDataType=this.__gl.UNSIGNED_INT),this.indexBuffer&&g.deleteBuffer(this.indexBuffer),this.indexBuffer=g.createBuffer(),g.bindBuffer(g.ELEMENT_ARRAY_BUFFER,this.indexBuffer),g.bufferData(g.ELEMENT_ARRAY_BUFFER,I.indices,g.STATIC_DRAW),this.numTriIndices=I.indices.length,this.numTriangles=C.length/3,this.glattrbuffers.textureCoords&&(this.glattrbuffers.texCoords=this.glattrbuffers.textureCoords,delete this.glattrbuffers.textureCoords)}clearBuffers(){this.__gl.deleteBuffer(this.indexBuffer),this.indexBuffer=null,super.clearBuffers()}bind(A){super.bind(A);const{geomType:g}=A.unifs;g&&this.__gl.uniform1i(g.location,0)}draw(A){this.__gl.drawElements(this.__gl.TRIANGLES,this.numTriIndices,this.indexDataType,0)}drawInstanced(A,g){this.__gl.drawElementsInstanced(this.__gl.TRIANGLES,this.numTriIndices,this.indexDataType,0,g)}destroy(){super.destroy();this.__gl.deleteBuffer(this.indexBuffer),this.indexBuffer=null}}const FRAMEBUFFER={MSAA_RENDERBUFFER:0,COLORBUFFER:1,DEPTHBUFFER:2};class GLBaseViewport extends ParameterOwner{__gl;renderer;__renderer;__fbo=null;quad;offscreenBuffer=null;depthTexture=null;highlightedGeomsBufferFbo;__backgroundTexture=null;__backgroundGLTexture=null;offscreenBufferFbo=null;width=0;height=0;__canvasWidth=0;__canvasHeight=0;fb=null;colorRenderbuffer;depthBuffer=null;EXT_frag_depth=null;manipulator;depthRange=[0,0];region=[0,0,0,0];backgroundColorParam=new ColorParameter("BackgroundColor",new Color("#eeeeee"));clickTime=200;doubleClickTime=200;longPressTime=1e3;pointerClickTolerance=5;constructor(A){super(),this.renderer=A,this.__renderer=A;const g=this.__renderer.gl;this.__gl=g,this.quad=new GLMesh(g,new Plane(1,1)),this.highlightedGeomsBufferFbo=new GLRenderTarget(g,{type:g.UNSIGNED_BYTE,format:g.RGBA,filter:g.NEAREST,width:4,height:4,createDepthTexture:!0}),this.highlightedGeomsBufferFbo.clearColor=new Color(0,0,0,0);const I=()=>{const A=this.backgroundColorParam.value;A instanceof BaseImage?A instanceof HDRImage?(this.__backgroundTexture=A,this.__backgroundGLTexture=new GLHDRImage(g,A)):(this.__backgroundTexture=A,this.__backgroundGLTexture=new GLTexture2D(g,A)):A instanceof Color?(this.__backgroundGLTexture&&(this.__backgroundGLTexture.destroy(),this.__backgroundGLTexture=null,this.__backgroundTexture=null),this.offscreenBufferFbo&&(this.offscreenBufferFbo.clearColor=A)):console.warn("Invalid background:"+A),this.emit("updated")};I(),this.backgroundColorParam.on("valueChanged",I)}getRenderer(){return this.renderer}getWidth(){return this.width}getHeight(){return this.height}resize(A,g){if(this.__canvasWidth==A&&this.__canvasHeight==g)return;this.__canvasWidth=A,this.__canvasHeight=g,this.width=A,this.height=g,this.region=[0,0,this.width,this.height],this.resizeRenderTargets(A,g);const I=new ResizedEvent(this.width,this.height);this.emit("resized",I)}resizeRenderTargets(A,g){this.highlightedGeomsBufferFbo&&this.highlightedGeomsBufferFbo.resize(A/devicePixelRatio,g/devicePixelRatio);const I=this.__renderer.gl;if(this.renderer.outlineThickness>0&&"image"==this.renderer.outlineMethod){if("Safari"==SystemDesc.browserName||"webgl"==I.name)return void console.warn("Disabling outlines on Safari due to a regression in WebKit",SystemDesc);this.fb&&(I.deleteFramebuffer(this.fb[FRAMEBUFFER.MSAA_RENDERBUFFER]),I.deleteFramebuffer(this.fb[FRAMEBUFFER.COLORBUFFER]),I.deleteFramebuffer(this.fb[FRAMEBUFFER.DEPTHBUFFER]),this.colorRenderbuffer&&I.deleteRenderbuffer(this.colorRenderbuffer),this.depthBuffer&&I.deleteRenderbuffer(this.depthBuffer)),this.offscreenBuffer&&this.depthTexture?(this.offscreenBuffer.resize(A,g),this.depthTexture.resize(A,g)):(this.offscreenBuffer=new GLTexture2D(I,{type:"UNSIGNED_BYTE",format:"RGBA",filter:"LINEAR",width:4,height:4}),this.depthTexture=new GLTexture2D(I,{type:I.UNSIGNED_INT_24_8,format:I.DEPTH_STENCIL,internalFormat:"webgl"==I.name?I.DEPTH24_STENCIL8:I.DEPTH_COMPONENT,filter:I.NEAREST,wrap:I.CLAMP_TO_EDGE,width:4,height:4})),this.fb=[],this.fb[FRAMEBUFFER.MSAA_RENDERBUFFER]=I.createFramebuffer(),I.bindFramebuffer(I.FRAMEBUFFER,this.fb[FRAMEBUFFER.MSAA_RENDERBUFFER]),this.colorRenderbuffer=I.createRenderbuffer(),I.bindRenderbuffer(I.RENDERBUFFER,this.colorRenderbuffer),this.renderer.multiSampledScreenBuffer?I.renderbufferStorage(I.RENDERBUFFER,I.RGBA8,A,g):I.renderbufferStorageMultisample(I.RENDERBUFFER,4,I.RGBA8,A,g),I.framebufferRenderbuffer(I.FRAMEBUFFER,I.COLOR_ATTACHMENT0,I.RENDERBUFFER,this.colorRenderbuffer),this.depthBuffer=I.createRenderbuffer(),I.bindRenderbuffer(I.RENDERBUFFER,this.depthBuffer),this.renderer.multiSampledScreenBuffer?I.renderbufferStorage(I.RENDERBUFFER,I.DEPTH24_STENCIL8,A,g):I.renderbufferStorageMultisample(I.RENDERBUFFER,4,I.DEPTH24_STENCIL8,A,g),I.framebufferRenderbuffer(I.FRAMEBUFFER,I.DEPTH_ATTACHMENT,I.RENDERBUFFER,this.depthBuffer),this.fb[FRAMEBUFFER.COLORBUFFER]=I.createFramebuffer(),I.bindFramebuffer(I.FRAMEBUFFER,this.fb[FRAMEBUFFER.COLORBUFFER]),I.framebufferTexture2D(I.FRAMEBUFFER,I.COLOR_ATTACHMENT0,I.TEXTURE_2D,this.offscreenBuffer.glTex,0),I.bindFramebuffer(I.FRAMEBUFFER,null),this.fb[FRAMEBUFFER.DEPTHBUFFER]=I.createFramebuffer(),I.bindFramebuffer(I.FRAMEBUFFER,this.fb[FRAMEBUFFER.DEPTHBUFFER]),I.framebufferTexture2D(I.FRAMEBUFFER,I.DEPTH_ATTACHMENT,I.TEXTURE_2D,this.depthTexture.glTex,0),I.bindFramebuffer(I.FRAMEBUFFER,null);const C=I.checkFramebufferStatus("webgl"==I.name?I.DRAW_FRAMEBUFFER:I.FRAMEBUFFER);if(C!==I.FRAMEBUFFER_COMPLETE)switch(C){case I.FRAMEBUFFER_INCOMPLETE_ATTACHMENT:throw new Error("The attachment types are mismatched or not all framebuffer attachment points are framebuffer attachment complete.");case I.FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT:throw new Error("There is no attachment.");case I.FRAMEBUFFER_INCOMPLETE_DIMENSIONS:throw new Error("Height and width of the attachment are not the same.");case I.FRAMEBUFFER_INCOMPLETE_DIMENSIONS:throw new Error("The format of the attachment is not supported or if depth and stencil attachments are not the same renderbuffer.");case 36061:throw new Error("The framebuffer is unsupported");default:throw new Error("Incomplete Frambuffer")}}}draw(A){const g=this.__renderer.gl;A.pushGLStack("GLBaseViewport.draw");const I=A.boundRendertarget;if(this.renderer.outlineThickness>0&&"image"==this.renderer.outlineMethod){if("Safari"==SystemDesc.browserName||"webgl"==g.name)console.warn("Disabling outlines on Safari due to a regression in WebKit",SystemDesc);else{this.fb||this.resizeRenderTargets(this.width,this.height);const I=this.fb[FRAMEBUFFER.MSAA_RENDERBUFFER];g.bindFramebuffer("webgl"==g.name?g.DRAW_FRAMEBUFFER:g.FRAMEBUFFER,I),A.boundRendertarget=I}}else A.boundRendertarget||g.bindFramebuffer(g.FRAMEBUFFER,null);A.glEnable(g.DEPTH_TEST),this.__renderer.drawScene(A);const C=A.toHighlightRenderState();if(this.drawHighlights(C),0!=C.stack.length&&console.warn(" corrupt highlightRenderState.stack.length:",C.stack.length),this.fb){g.bindFramebuffer(g.READ_FRAMEBUFFER,this.fb[FRAMEBUFFER.MSAA_RENDERBUFFER]),g.bindFramebuffer(g.DRAW_FRAMEBUFFER,this.fb[FRAMEBUFFER.COLORBUFFER]),g.clearBufferfv(g.COLOR,0,[0,0,0,0]),g.blitFramebuffer(0,0,this.width,this.height,0,0,this.width,this.height,g.COLOR_BUFFER_BIT,g.LINEAR),g.bindFramebuffer(g.DRAW_FRAMEBUFFER,I),A.boundRendertarget=I,g.viewport(this.region[0],this.region[1],this.region[2],this.region[3]),g.disable(g.DEPTH_TEST);const C=this.__renderer.screenQuad;C.bindShader(A),C.draw(A,this.offscreenBuffer),g.enable(g.DEPTH_TEST)}A.popGLStack(),this.renderer.emit("redrawOccurred")}drawSilhouettes(A){A.pushGLStack("GLBaseViewport.drawSilhouettes");const g=this.__renderer.gl;g.bindFramebuffer(g.READ_FRAMEBUFFER,this.fb[FRAMEBUFFER.MSAA_RENDERBUFFER]),g.bindFramebuffer(g.DRAW_FRAMEBUFFER,this.fb[FRAMEBUFFER.DEPTHBUFFER]),g.clearBufferfv(g.COLOR,0,[1,1,1,1]),g.blitFramebuffer(0,0,this.width,this.height,0,0,this.width,this.height,g.DEPTH_BUFFER_BIT,g.NEAREST),g.bindFramebuffer(g.DRAW_FRAMEBUFFER,this.fb[FRAMEBUFFER.MSAA_RENDERBUFFER]),A.boundRendertarget=this.fb[FRAMEBUFFER.MSAA_RENDERBUFFER],g.viewport(this.region[0],this.region[1],this.region[2],this.region[3]),A.glEnable(g.BLEND),A.glDisable(g.DEPTH_TEST),g.blendEquation(g.FUNC_ADD),g.blendFuncSeparate(g.SRC_ALPHA,g.ONE_MINUS_SRC_ALPHA,g.ZERO,g.ONE),g.depthMask(!1),this.renderer.silhouetteShader.bind(A);const I=A.unifs;this.depthTexture.bindToUniform(A,I.depthTexture),g.uniform2f(I.screenSize.location,this.width,this.height),g.uniform1f(I.outlineThickness.location,this.renderer.outlineThickness*window.devicePixelRatio);const C=this.renderer.outlineColor.asArray();g.uniform4f(I.outlineColor.location,C[0],C[1],C[2],C[3]),g.uniform1f(I.outlineSensitivity.location,this.renderer.outlineSensitivity),g.uniform1f(I.outlineDepthBias.location,this.renderer.outlineDepthBias),g.uniform2f(I.depthRange.location,this.depthRange[0],this.depthRange[1]),this.quad.bindAndDraw(A),g.depthMask(!0),A.popGLStack()}drawHighlights(A){if(this.highlightedGeomsBufferFbo){const g=this.__renderer.gl;A.pushGLStack("GLBaseViewport.drawHighlights"),this.highlightedGeomsBufferFbo.bindForWriting(A,!0),A.glEnable(g.CULL_FACE),A.glEnable(g.DEPTH_TEST),A.glDisable(g.BLEND),g.depthFunc(g.LESS),g.depthMask(!0),A.glShader=null,this.__renderer.drawHighlightedGeoms(A),this.highlightedGeomsBufferFbo.unbindForWriting(A),g.viewport(this.region[0],this.region[1],this.region[2],this.region[3]);{A.pushGLStack("GLBaseViewport.drawHighlights.highlightsShader"),this.renderer.highlightsShader.bind(A),A.glEnable(g.BLEND),g.blendEquation(g.FUNC_ADD),g.blendFuncSeparate(g.SRC_ALPHA,g.ONE_MINUS_SRC_ALPHA,g.ONE,g.ONE);const I=A.unifs;g.uniform1f(I.outlineThickness.location,this.renderer.highlightOutlineThickness),this.highlightedGeomsBufferFbo.bindColorTexture(A,I.highlightDataTexture),this.quad.bindAndDraw(A),A.popGLStack()}A.popGLStack()}}getManipulator(){return this.manipulator}setManipulator(A){this.manipulator!=A&&(this.manipulator&&this.manipulator.deactivateTool&&this.manipulator.deactivateTool(),this.manipulator=A,this.manipulator.activateTool&&this.manipulator.activateTool())}onPointerDown(A){console.warn("@GLBaseViewport#onPointerDown - Implement me!")}onPointerUp(A){console.warn("@GLBaseViewport#onPointerUp - Implement me!")}onPointerMove(A){console.warn("@GLBaseViewport#onPointerMove - Implement me!")}onPointerEnter(A){console.warn("@GLBaseViewport#onPointerEnter - Implement me!")}onPointerLeave(A){console.warn("@GLBaseViewport#onPointerLeave - Implement me!")}onMouseLeave(A){}onKeyDown(A){}onKeyUp(A){}}class GLViewport extends GLBaseViewport{debugGeomDataBuffer=!1;pointAndLinePickingSize=3;debugOcclusionBuffer=!1;debugReductionBuffer=!1;debugHighlightedGeomsBuffer=!1;mousePointerSearchArea=5;touchPointerSearchArea=45;projectionMatrix;viewportFrustum;camera;bottomLeftCoords=new Vec2(0,0);topRightCoords=new Vec2(1,1);posX=0;posY=0;pointerDownTime=[];pointerDownButton=-1;pointerDownPos=new Vec2(-1,-1);prevClickTime;prevClickPos=new Vec2(-1,-1);prevClickButton=-1;geomDataBufferSizeFactor=window.devicePixelRatio;geomDataRenderTarget;geomDataBufferInvalid=!0;cameraXfo=new Xfo;cameraMat=new Mat4;viewMatrix=new Mat4;screenPos=null;intersectionData;pointerOverItem;longPressId;constructor(A,g,I,C){super(A),this.name=g,this.projectionMatrix=new Mat4,this.viewportFrustum=new Vec4,this.bottomLeftCoords=new Vec2(0,0),this.topRightCoords=new Vec2(1,1),this.prevClickTime=0,SystemDesc.isMobileDevice&&(this.geomDataBufferSizeFactor=8);const e=this.__renderer.gl;this.geomDataRenderTarget=new GLRenderTarget(e,{type:A.floatGeomBuffer?e.FLOAT:e.UNSIGNED_BYTE,format:e.RGBA,filter:e.NEAREST,width:I<=1?1:Math.floor(I/this.geomDataBufferSizeFactor),height:C<=1?1:Math.floor(C/this.geomDataBufferSizeFactor),createDepthTexture:!0}),this.geomDataRenderTarget.clearColor=new Color(0,0,0,0),this.camera=new Camera("DefaultCamera"),this.setCamera(this.camera),this.setManipulator(new CameraManipulator({renderer:A})),this.resize(I,C)}getBl(){return this.bottomLeftCoords}setBl(A){this.bottomLeftCoords=A,this.#n()}getTr(){return this.topRightCoords}setTr(A){this.topRightCoords=A,this.#n()}getPosX(){return this.posX}getPosY(){return this.posY}resize(A,g){this.__canvasWidth=A,this.__canvasHeight=g,this.#n()}#n(){this.posX=Math.round(this.__canvasWidth*this.bottomLeftCoords.x),this.posY=Math.round(this.__canvasWidth*this.bottomLeftCoords.y),this.width=Math.round(this.__canvasWidth*this.topRightCoords.x-this.__canvasWidth*this.bottomLeftCoords.x),this.height=Math.round(this.__canvasHeight*this.topRightCoords.y-this.__canvasHeight*this.bottomLeftCoords.y),this.region=[this.posX,this.posY,this.width,this.height],this.camera&&this.updateProjectionMatrix(),this.resizeRenderTargets(this.width,this.height);const A=new ResizedEvent(this.width,this.height);this.emit("resized",A)}resizeRenderTargets(A,g){super.resizeRenderTargets(A,g),this.geomDataRenderTarget&&(this.geomDataRenderTarget.resize(Math.floor(this.width/this.geomDataBufferSizeFactor),Math.floor(this.height/this.geomDataBufferSizeFactor)),this.invalidateGeomDataBuffer())}getCamera(){return this.camera}setCamera(A){this.camera=A,this.depthRange=[this.camera.getNear(),this.camera.getFar()];const g=A.globalXfoParam,I=()=>{this.cameraXfo=g.value,this.cameraMat=this.cameraXfo.toMat4(),this.viewMatrix=this.cameraMat.inverse()};I(),g.on("valueChanged",(()=>{I(),this.emit("updated"),this.invalidateGeomDataBuffer();const A=new ViewChangedEvent("CameraAndPointer",this.cameraXfo);this.emit("viewChanged",A)})),this.camera.on("projectionParamChanged",(()=>{this.updateProjectionMatrix(),this.depthRange=[this.camera.getNear(),this.camera.getFar()],this.emit("updated")})),this.updateProjectionMatrix()}updateProjectionMatrix(){const A=this.width/this.height;this.camera.updateProjectionMatrix(this.projectionMatrix,A);const g=this.camera.getFocalDistance(),I=this.camera.getFov(),C=this.camera.isOrthographicParam.value,e=Math.tan(I/2)*g*2,i=e*A,t=this.camera.getFrustumHeight(),o=e*A;this.viewportFrustum.set(MathFunctions.lerp(i,o,C),MathFunctions.lerp(e,t,C),g,I)}getProjectionMatrix(){return this.projectionMatrix}getViewMatrix(){return this.viewMatrix}frameView(A,g=0,I=.1){this.width>0&&this.height>0?this.camera.frameView(this.width,this.height,A,g,I):this.once("resized",(()=>this.frameView(A,g,I)))}calcScreenPosFromWorldPos(A){const g=this.projectionMatrix.multiply(this.viewMatrix).transformVec4(new Vec4(A.x,A.y,A.z,1));return g.x/=g.w,g.y/=g.w,new Vec2((.5*g.x+.5)*this.width,(-.5*g.y+.5)*this.height)}calcRayFromScreenPos(A){const g=A.x*window.devicePixelRatio,I=A.y*window.devicePixelRatio,C=this.__canvasHeight*(1-this.topRightCoords.y),e=(g-this.posX)/this.width*2-1,i=(I-C)/this.height*2-1,t=this.cameraMat,o=this.projectionMatrix.inverse();if(null==o)return console.warn(`Unable to generate Ray from screen pos:${A.toString()} in region ${this.region}`),new Ray;let a,B;if(this.camera.isOrthographic()){const A=o.transformVec3(new Vec3(e,-i,-1));A.z=0,a=t.transformVec3(A),B=new Vec3(0,0,-1)}else a=t.translation,B=o.transformVec3(new Vec3(e,-i,-1));return B=t.rotateVec3(B).normalize(),new Ray(a,B)}renderGeomDataFbo(){if(this.geomDataRenderTarget){const A=new GeomDataRenderState(this.renderer.__gl);A.pushGLStack("GLViewport.renderGeomDataFbo"),this.bindGLViewport(A),A.geomDataFbo=this.geomDataRenderTarget,A.debugGeomDataBuffer=this.debugGeomDataBuffer,A.pointAndLinePickingSize=this.pointAndLinePickingSize,this.geomDataRenderTarget.bindForWriting(A,!0),this.__renderer.drawSceneGeomData(A),this.geomDataRenderTarget.unbindForWriting(),A.popGLStack(),this.geomDataBufferInvalid=!1}}invalidateGeomDataBuffer(){this.geomDataBufferInvalid=!0}getGeomDataAtPos(A,g,I=5){if(this.geomDataRenderTarget){if(this.geomDataBufferInvalid&&(this.renderGeomDataFbo(),this.screenPos=null),A===this.screenPos)return this.intersectionData;this.screenPos=A,this.intersectionData=null;const C=this.__renderer.gl;C.finish(),this.geomDataRenderTarget.bindForReading();const e=A.x*window.devicePixelRatio,i=A.y*window.devicePixelRatio,t=I*I,o=this.geomDataRenderTarget.width,a=this.geomDataRenderTarget.height,B=Math.floor(e*(o/this.width))-.5*I,s=Math.floor(a-i*(a/this.height)-1)-.5*I,l=new Float32Array(4*t);let n;C.readPixels(B,s,I,I,C.RGBA,C.FLOAT,l),this.geomDataRenderTarget.unbindForReading();let d=null;const r=new Map,c=A=>{const g=63&Math.round(l[4*A+0]),I=this.__renderer.getPass(g);return I?(n=l.slice(4*A,4*(A+1)),d=I.getGeomItemAndDist(n),!!d):(console.warn("Geom data buffer returns invalid pass id:",g),null)},h=I/2;for(let A=0;A<h&&!d;A++)for(let g=-A;g<=A&&!d;g++)for(let C=-A;C<=A&&!d;C++){if(c(h+g+(h+g)*I))break}if(g||(g=this.calcRayFromScreenPos(A)),d){const I=g.start.add(g.dir.scale(d.dist));this.intersectionData=new IntersectionData(A,g,I,n,d),r.has(d.geomItem)&&(this.intersectionData.componentIds=Array.from(r.get(d.geomItem)))}return this.intersectionData}return null}getGeomVisiblity(A,g){if(this.geomDataRenderTarget){this.geomDataBufferInvalid&&(this.renderGeomDataFbo(),this.screenPos=null);const I=this.__renderer.gl;I.finish();const C=this.geomDataRenderTarget.width,e=this.geomDataRenderTarget.height,i=C/this.width,t=e/this.height,o=Math.round(A.x*i),a=Math.round(A.y*t),B=Math.round(g.x*i),s=Math.round(g.y*t),l=Math.round(e-s),n=Math.round(o),d=Math.round(B-o),r=Math.round(s-a),c=d*r;let h;this.geomDataRenderTarget.bindForReading(),this.__renderer.floatGeomBuffer?(h=new Float32Array(4*c),I.readPixels(n,l,d,r,I.RGBA,I.FLOAT,h)):(h=new Uint8Array(4*c),I.readPixels(n,l,d,r,I.RGBA,I.UNSIGNED_BYTE,h)),this.geomDataRenderTarget.unbindForReading();const G=[],m=[],b=new Map;for(let A=0;A<c;A++){let g;const I=h.subarray(4*A,4*(A+1));if(this.__renderer.floatGeomBuffer){if(0==I[3])continue;g=63&Math.round(I[0])}else{if(0==I[0]&&0==I[1])continue;g=Math.floor(I[1]/64)}const C=I[0]+"-"+I[1],e=b.get(C);if(null!=e){m[e]++;continue}const i=this.__renderer.getPass(g)?.getGeomItemAndDist(I);i&&(b.set(C,m.length),m.push(1),G.push(i.geomItem))}const Z=[];for(let A=0;A<m.length;A++)Z[A]=A;return Z.sort(((A,g)=>m[A]>m[g]?-1:1)),{geomItems:Z.map((A=>G[A])),histogram:Z.map((A=>m[A]))}}return null}getGeomItemsInRect(A,g){const I=this.getGeomVisiblity(A,g),C=new Set;return I.geomItems.forEach((A=>{C.add(A)})),C}__getPointerPos(A,g){return new Vec2(A-this.getPosX(),g-this.getPosY())}prepareUIEvent(A){A.viewport=this}onPointerDown(A){if(this.prepareUIEvent(A),A instanceof ZeaMouseEvent)A.pointerPos=this.__getPointerPos(A.rendererX,A.rendererY),A.pointerRay=this.calcRayFromScreenPos(A.pointerPos),this.pointerDownButton=A.button,this.pointerDownPos=A.pointerPos,this.pointerDownTime[A.button]=Date.now();else if(A instanceof ZeaTouchEvent){const g=A.touches[0];A.pointerPos=this.__getPointerPos(g.rendererX,g.rendererY),A.pointerRay=this.calcRayFromScreenPos(A.pointerPos),this.pointerDownButton=0,this.pointerDownPos=A.pointerPos,this.pointerDownTime[0]=Date.now()}A.propagating&&A.getCapture()&&A.getCapture().onPointerDown(A),A.propagating&&(A.intersectionData=this.getGeomDataAtPos(A.pointerPos,A.pointerRay,this.mousePointerSearchArea),A.intersectionData&&A.intersectionData.geomItem.onPointerDown(A)),A.propagating&&this.emit("pointerDown",A),A.propagating&&this.manipulator&&this.manipulator.onPointerDown(A),null!=this.longPressId&&clearTimeout(this.longPressId),(A instanceof ZeaMouseEvent||A instanceof ZeaTouchEvent&&1==A.changedTouches.length)&&(this.longPressId=setTimeout((()=>{this.longPressId=void 0,A.propagating=!0,A.intersectionData&&A.intersectionData.geomItem&&A.intersectionData.geomItem.onPointerLongPress(A),this.emit("pointerLongPress",A),A.propagating&&this.manipulator&&this.manipulator.onPointerLongPress(A)}),this.longPressTime))}onPointerUp(A){if(this.prepareUIEvent(A),A.pointerType===POINTER_TYPES.mouse){const g=A;g.pointerPos=this.__getPointerPos(g.rendererX,g.rendererY),g.pointerRay=this.calcRayFromScreenPos(g.pointerPos)}else if(A.pointerType===POINTER_TYPES.touch){const g=A;if(1==g.changedTouches.length){const A=g.changedTouches[0];g.pointerPos=this.__getPointerPos(A.rendererX,A.rendererY),g.pointerRay=this.calcRayFromScreenPos(g.pointerPos)}}null!=this.longPressId&&clearTimeout(this.longPressId),A.getCapture()&&A.getCapture().onPointerUp(A),A.propagating&&(A.intersectionData=this.getGeomDataAtPos(A.pointerPos,A.pointerRay,this.mousePointerSearchArea),A.intersectionData&&A.intersectionData.geomItem.onPointerUp(A)),A.propagating&&this.emit("pointerUp",A),A.propagating&&this.manipulator&&this.manipulator.onPointerUp(A);const g=A instanceof ZeaMouseEvent?A.button:0,I=Date.now();if((A instanceof ZeaMouseEvent||A instanceof ZeaTouchEvent&&1==A.changedTouches.length)&&I-this.pointerDownTime[g]<this.clickTime&&A.pointerPos.distanceTo(this.pointerDownPos)<this.pointerClickTolerance&&this.pointerDownButton==g){const C=I;A.propagating=!0,g==this.prevClickButton&&C-this.prevClickTime<this.doubleClickTime?(A.intersectionData&&A.intersectionData.geomItem&&A.intersectionData.geomItem.onPointerDoubleClick(A),A.propagating&&A.propagating&&this.emit("pointerDoubleClick",A),A.propagating&&this.manipulator&&this.manipulator.onPointerDoubleClick(A)):(this.prevClickButton=g,this.prevClickPos=A.pointerPos,this.prevClickTime=C,A.intersectionData&&A.intersectionData.geomItem&&A.intersectionData.geomItem.onPointerClick(A),A.propagating&&this.emit("pointerClick",A),A.propagating&&this.manipulator&&this.manipulator.onPointerClick(A))}}onPointerMove(A){if(this.prepareUIEvent(A),A.pointerType===POINTER_TYPES.mouse){const g=A,I=this.__getPointerPos(g.rendererX,g.rendererY);g.pointerPos=I,g.pointerRay=this.calcRayFromScreenPos(I)}else if(A.pointerType===POINTER_TYPES.touch){const g=A;for(let A=0;A<g.touches.length;A++){const I=g.touches[A];I.touchPos=this.__getPointerPos(I.rendererX,I.rendererY),I.touchRay=this.calcRayFromScreenPos(I.touchPos)}g.pointerPos=g.touches[0].touchPos,g.pointerRay=g.touches[0].touchRay}null!=this.longPressId&&A.pointerPos.distanceTo(this.pointerDownPos)>this.pointerClickTolerance&&(clearTimeout(this.longPressId),this.longPressId=void 0),A.getCapture()&&A.getCapture().onPointerMove(A),A.propagating&&(A.intersectionData=this.getGeomDataAtPos(A.pointerPos,A.pointerRay,this.mousePointerSearchArea),A.intersectionData?(A.intersectionData.geomItem!=this.pointerOverItem&&(this.pointerOverItem&&(A.leftGeometry=this.pointerOverItem,this.pointerOverItem.onPointerLeave(A),A.propagating&&this.emit("pointerLeaveGeom",A)),A.propagating=!0,this.pointerOverItem=A.intersectionData.geomItem,this.pointerOverItem.onPointerEnter(A),A.propagating&&this.emit("pointerEnterGeom",A)),A.propagating=!0,A.intersectionData.geomItem.onPointerMove(A)):A.propagating&&this.pointerOverItem&&(A.leftGeometry=this.pointerOverItem,this.pointerOverItem.onPointerLeave(A),this.pointerOverItem=null,this.emit("pointerLeaveGeom",A))),A.propagating&&this.emit("pointerMove",A),A.propagating&&this.manipulator&&this.manipulator.onPointerMove(A)}onPointerEnter(A){this.prepareUIEvent(A),this.emit("pointerEnter",A),A.propagating&&this.manipulator&&this.manipulator.onPointerEnter&&this.manipulator.onPointerEnter(A)}onPointerLeave(A){this.prepareUIEvent(A),this.emit("pointerLeave",A),A.propagating&&(!this.manipulator||!this.manipulator.onPointerLeave||(this.manipulator.onPointerLeave(A),A.propagating))}onKeyDown(A){this.prepareUIEvent(A),this.manipulator&&(this.manipulator.onKeyDown(A),!A.propagating)||this.emit("keyDown",A)}onKeyUp(A){this.prepareUIEvent(A),this.manipulator&&(this.manipulator.onKeyUp(A),!A.propagating)||this.emit("keyUp",A)}onWheel(A){this.prepareUIEvent(A),A.pointerPos=this.__getPointerPos(A.rendererX,A.rendererY),A.pointerRay=this.calcRayFromScreenPos(A.pointerPos),A.intersectionData=this.getGeomDataAtPos(A.pointerPos,A.pointerRay,this.mousePointerSearchArea),(null==A.intersectionData||(A.intersectionData.geomItem.onWheel(A),A.propagating))&&(this.manipulator?this.manipulator.onWheel(A):this.emit("mouseWheel",A))}onTouchCancel(A){this.prepareUIEvent(A),A.getCapture()&&(A.getCapture().onTouchCancel(A),!A.propagating)||this.manipulator&&(this.manipulator.onTouchCancel(A),!A.propagating)||this.emit("touchCancel",A)}bindGLViewport(A){A.viewXfo=this.cameraXfo,A.viewScale=1,A.region=this.region,A.depthRange=this.depthRange,A.cameraMatrix=this.cameraMat,A.viewport=this;const g=this.__renderer.gl;A.bindRendererUnifs=I=>{const{cameraMatrix:C,viewMatrix:e,projectionMatrix:i,eye:t,isOrthographic:o,viewportFrustum:a}=I;C&&g.uniformMatrix4fv(C.location,!1,A.cameraMatrix.asArray()),e&&g.uniformMatrix4fv(e.location,!1,this.viewMatrix.asArray()),i&&g.uniformMatrix4fv(i.location,!1,this.projectionMatrix.asArray()),t&&g.uniform1i(t.location,0),o&&g.uniform1i(o.location,this.camera.isOrthographic()?1:0),a&&this.viewportFrustum&&g.uniform4f(a.location,this.viewportFrustum.x,this.viewportFrustum.y,this.viewportFrustum.z,this.viewportFrustum.w)},A.bindViewports=(A,g)=>g()}draw(A){if(this.debugGeomDataBuffer){this.renderGeomDataFbo();const A=this.__renderer.gl;A.viewport(this.region[0],this.region[1],this.region[2],this.region[3]);const g=this.backgroundColorParam.value.asArray();A.clearColor(g[0],g[1],g[2],g[3]);const I=new ColorRenderState(this.__renderer.gl),C=this.__renderer.screenQuad;return C.bindShader(I),void C.draw(I,this.geomDataRenderTarget.textureTargets[0])}const g=this.__renderer.gl;this.bindGLViewport(A),A.pushGLStack("GLViewport.draw"),A.glDisable(g.BLEND),A.glEnable(g.DEPTH_TEST),A.glEnable(g.CULL_FACE),g.viewport(this.region[0],this.region[1],this.region[2],this.region[3]);const I=this.backgroundColorParam.value.asArray();if(g.clearColor(I[0],I[1],I[2],I[3]),g.colorMask(!0,!0,!0,!0),g.clear(g.COLOR_BUFFER_BIT|g.DEPTH_BUFFER_BIT),super.draw(A),A.popGLStack(),this.debugHighlightedGeomsBuffer){const A=new ColorRenderState(this.__renderer.gl),g=this.__renderer.screenQuad;g.bindShader(A),g.draw(A,this.highlightedGeomsBufferFbo.textureTargets[0])}if(this.debugOcclusionBuffer){A.pushGLStack("debugOcclusionBuffer"),g.clear(g.COLOR_BUFFER_BIT|g.DEPTH_BUFFER_BIT);const I=this.__renderer.glGeomItemLibrary.occlusionDataBuffer;A.glEnable(g.BLEND),A.glDisable(g.DEPTH_TEST),g.blendEquation(g.FUNC_ADD),g.blendFuncSeparate(g.SRC_ALPHA,g.ONE_MINUS_SRC_ALPHA,g.ONE,g.ONE);const C=this.__renderer.screenQuad;C.bindShader(A),C.draw(A,I.textureTargets[0]),A.popGLStack()}if(this.debugReductionBuffer){const g=this.__renderer.glGeomItemLibrary.reductionDataBuffer,I=this.__renderer.screenQuad;I.bindShader(A);const C=A.unifs.image;g.bindColorTexture(A,C);const e=this.__renderer.gl;e.uniform1i(A.unifs.isTextured.location,1),e.enable(e.BLEND),e.blendEquation(e.FUNC_ADD),e.blendFuncSeparate(e.SRC_ALPHA,e.ONE_MINUS_SRC_ALPHA,e.ONE,e.ONE),I.draw(A,null),e.disable(e.BLEND)}}}var frag$h="\nprecision highp float;\n#define GLSLIFY 1\n\nuniform sampler2D image;\nuniform vec4 color;\nuniform int isTextured;\n\nvarying vec2 v_texCoord;\n\n#ifdef ENABLE_ES3\n  out vec4 fragColor;\n#endif\nvoid main(void) {\n#ifndef ENABLE_ES3\n  vec4 fragColor;\n#endif\n\n  if (isTextured != 0)\n    fragColor = texture2D(image, v_texCoord);\n  else \n    fragColor = color;\n\n#ifndef ENABLE_ES3\n  gl_FragColor = fragColor;\n#endif\n}\n",vert$i="\nprecision highp float;\n#define GLSLIFY 1\n\nimport 'quadVertexFromID.glsl'\n\nuniform vec2 pos;\nuniform vec2 size;\n\n/* VS Outputs */\nvarying vec2 v_texCoord;\n \nvoid main()\n{\n  vec2 position = getQuadVertexPositionFromID();\n  v_texCoord = position+0.5;\n  gl_Position = vec4(vec2(-1.0, -1.0) + (pos * 2.0) + (v_texCoord * abs(size) * 2.0), 0.0, 1.0);\n  if (size.x < 0.0)\n    v_texCoord.x = 1.0 - v_texCoord.x;\n  if (size.y < 0.0)\n    v_texCoord.y = 1.0 - v_texCoord.y;\n}\n";class ScreenQuadShader extends GLShader{constructor(A){super(A,"ScreenQuadShader"),this.setShaderStage("VERTEX_SHADER",vert$i),this.setShaderStage("FRAGMENT_SHADER",frag$h)}}class GLScreenQuad{__gl;__pos;__size;__glshader;__quadBinding;ready;constructor(A,g){this.__gl=A,this.__pos=[0,0],this.__size=[1,1],this.__glshader=new ScreenQuadShader(A),A.__quadVertexIdsBuffer||A.setupInstancedQuad();const I=this.__glshader.compileForTarget("GLScreenQuad",g);this.__quadBinding=generateShaderGeomBinding(this.__gl,I.attrs,A.__quadattrbuffers,A.__quadIndexBuffer)}bind(A,g,I,C){const e=A.unifs,i=this.__gl;if(g instanceof GLTexture2D)i.uniform1i(e.isTextured.location,1),g.bindToUniform(A,A.unifs.image);else if(g instanceof WebGLTexture){i.uniform1i(e.isTextured.location,1);const I=A.boundTextures++;i.activeTexture(i.TEXTURE0+I),i.bindTexture(i.TEXTURE_2D,g),i.uniform1i(e.image.location,I)}else g&&g instanceof Color&&(i.uniform1i(e.isTextured.location,0),i.uniform4fv(e.color.location,g.asArray()));{const A=e.pos;if(A){let g=I?I instanceof Vec2?I.asArray():I:this.__pos;i.uniform2fv(A.location,g)}}{const A=e.size;if(A){let g=C?C instanceof Vec2?C.asArray():C:this.__size;i.uniform2fv(A.location,g)}}this.__quadBinding.bind(A)}bindShader(A){return this.__glshader.bind(A,"GLScreenQuad")}draw(A,g,I,C){this.bind(A,g,I,C);const e=this.__gl;e.drawElements(e.TRIANGLES,6,e.UNSIGNED_BYTE,0)}}const gltextureCache=new Map;class ParamUniformBinding extends BaseClass{unif;dirty=!1;constructor(A){super(),this.unif=A}bind(A){}unbind(A){}destroy(){}}class Sampler2DBinding extends ParamUniformBinding{boundImage;textureType;textureTypeUnif;dirty=!1;param;texBinding;gltexture=null;constructor(A,g,I,C,e){super(C);const i=I.getName();this.param=I,this.textureTypeUnif=e[i+"Type"];const t=I=>{let C=gltextureCache.get(I);C||(C="HDR"===I.type?new GLHDRImage(A,I):new GLTexture2D(A,I),gltextureCache.set(I,C)),this.texBinding=C.preBind(this.unif,e),C.on("updated",(()=>{g.emit("updated")})),this.gltexture=C,this.gltexture.addRef(this),this.textureType=1,g.emit("updated")};let o;const a=A=>{A.isLoaded()?t(A):o=A.on("loaded",(()=>{t(this.boundImage)})),this.boundImage=A},B=()=>{gltextureCache.get(this.boundImage).removeRef(this),this.texBinding=null,this.gltexture=null,this.textureType=-1,o&&this.boundImage.off("loaded",o),this.boundImage=null,o=null,g.emit("updated")};I.getImage()&&a(I.getImage()),I.on("textureConnected",(()=>{a(I.getImage())})),I.on("textureDisconnected",(()=>{B()})),this.dirty=!0,I.on("valueChanged",(()=>{this.dirty=!0,g.emit("updated")}))}bind(A){this.gltexture&&this.gltexture.bindToUniform(A,this.unif,this.texBinding)}}class SimpleUniformBinding extends ParamUniformBinding{param;textureUnif;textureTypeUnif;texBinding;gltexture=null;textureType=-1;val;uniform1i;uniformXX;update;constructor(A,g,I,C,e){super(C);const i=I.getName();switch(this.param=I,this.unif=C,this.textureUnif=e[i+"Tex"],this.textureTypeUnif=e[i+"TexType"],this.uniform1i=A.uniform1i.bind(A),this.unif.glslType){case"bool":case"int":this.uniformXX=A.uniform1i.bind(A);break;case"uint":this.uniformXX=A.uniform1ui.bind(A);break;case"float":this.uniformXX=A.uniform1f.bind(A)}this.bind=this.bindValue;const t=I=>{let C=gltextureCache.get(I);C||(C="HDR"===I.type?new GLHDRImage(A,I):new GLTexture2D(A,I),gltextureCache.set(I,C)),this.texBinding=C.preBind(this.textureUnif,e),C.on("updated",(()=>{g.emit("updated")})),this.gltexture=C,this.gltexture.addRef(this),this.textureType=1,this.bind=this.bindTexture,g.emit("updated")};let o,a;if(this.update=()=>{try{o||("boolean"==typeof I.value?this.val=I.value?1:0:this.val=I.value)}catch(A){}g.emit("updated")},this.textureUnif&&I instanceof MaterialFloatParam){const A=A=>{A.isLoaded()?t(A):a=A.on("loaded",(()=>{t(o)})),o=A},C=()=>{gltextureCache.get(o).removeRef(this),this.texBinding=null,this.gltexture=null,this.textureType=-1,this.bind=this.bindValue,a&&o.off("loaded",a),o=null,a=null,g.emit("updated")};I.getImage()&&A(I.getImage()),I.on("textureConnected",(()=>{A(I.getImage())})),I.on("textureDisconnected",(()=>{C()}))}this.dirty=!0,I.on("valueChanged",(()=>{this.dirty=!0,g.emit("updated")}))}bindValue(A){this.dirty&&(this.update(),this.dirty=!1),this.unif&&this.uniformXX(this.unif.location,this.val),this.textureTypeUnif&&this.uniform1i(this.textureTypeUnif.location,0)}bindTexture(A){this.dirty&&(this.update(),this.dirty=!1),this.gltexture.bindToUniform(A,this.textureUnif,this.texBinding)}}class ComplexUniformBinding extends ParamUniformBinding{param;values;uniformXX;constructor(A,g,I,C){switch(super(C),this.param=I,this.unif.glslType){case"vec2":this.uniformXX=A.uniform2fv.bind(A);break;case"vec3":this.uniformXX=A.uniform3fv.bind(A);break;case"vec4":this.uniformXX=A.uniform4fv.bind(A)}this.dirty=!0,I.on("valueChanged",(()=>{this.dirty=!0,g.emit("updated")}))}bind(A){this.dirty&&(this.values=this.param.value.asArray(),this.dirty=!1),this.uniformXX(this.unif.location,this.values)}unbind(){}destroy(){}}class MatrixUniformBinding extends ParamUniformBinding{param;uniformMatrixXXX;values=[];constructor(A,g,I,C){switch(super(C),this.param=I,this.unif.glslType){case"mat3":this.uniformMatrixXXX=A.uniformMatrix3fv.bind(A);break;case"mat4":this.uniformMatrixXXX=A.uniformMatrix4fv.bind(A)}this.dirty=!0,I.on("valueChanged",(()=>{this.dirty=!0,g.emit("updated")}))}bind(A){this.dirty&&(this.values=this.param.value.asArray(),this.dirty=!1),this.uniformMatrixXXX(this.unif.location,!1,this.values)}unbind(){}destroy(){}}class ColorUniformBinding extends ParamUniformBinding{param;textureUnif;textureTypeUnif;values=[];gltexture;textureType;texBinding;uniform1i;uniform4fv;update;constructor(A,g,I,C,e){super(C);const i=I.getName();let t,o;if(this.param=I,this.textureUnif=e[i+"Tex"],this.textureTypeUnif=e[i+"TexType"],this.values=[0,0,0,0],this.bind=this.bindValue,this.update=()=>{try{t||this.unif&&(I instanceof MaterialColorParam&&I.colorSpace==ColorSpace.Gamma?this.values=I.value.toLinear().asArray():this.values=I.value.asArray())}catch(A){}g.emit("updated")},this.textureUnif&&I instanceof MaterialColorParam){const C=I=>{t=I;let C=gltextureCache.get(I);C||(C="FLOAT"===I.type?new GLHDRImage(A,I):new GLTexture2D(A,I),gltextureCache.set(I,C)),this.texBinding=C.preBind(this.textureUnif,e),C.on("updated",(()=>{g.emit("updated")})),this.gltexture=C,this.gltexture.addRef(this),this.textureType=1,this.bind=this.bindTexture,g.emit("updated")},i=A=>{A.isLoaded()?C(A):o=A.once("loaded",(()=>{C(A)}))},a=()=>{this.gltexture.removeRef(this),this.gltexture=null,this.texBinding=null,this.textureType=null,o&&t.off("loaded",o),this.bind=this.bindValue,t=null,o=null,g.emit("updated")};I.getImage()&&i(I.getImage()),I.on("textureConnected",(()=>{i(I.getImage())})),I.on("textureDisconnected",(()=>{a()}))}this.dirty=!0,I.on("valueChanged",(()=>{this.dirty=!0})),this.uniform1i=A.uniform1i.bind(A),this.uniform4fv=A.uniform4fv.bind(A)}bindValue(A){this.unif&&(this.dirty&&(this.update(),this.dirty=!1),this.unif&&this.uniform4fv(this.unif.location,this.values),this.textureTypeUnif&&this.uniform1i(this.textureTypeUnif.location,0))}bindTexture(A){this.dirty&&(this.update(),this.dirty=!1),this.gltexture.bindToUniform(A,this.textureUnif,this.texBinding)}}class MaterialShaderBinding{uniformBindings=[];constructor(A,g,I,C){const e=C=>{const e=C.getName(),i=I[e];if(i)switch(i.glslType){case"bool":case"uint":case"int":case"float":this.uniformBindings.push(new SimpleUniformBinding(A,g,C,i,I));break;case"vec2":case"vec3":case"vec4":this.uniformBindings.push(new ComplexUniformBinding(A,g,C,i));break;case"color":this.uniformBindings.push(new ColorUniformBinding(A,g,C,i,I));break;case"mat4":this.uniformBindings.push(new MatrixUniformBinding(A,g,C,i));break;default:return void console.warn("Param :"+e+" has unhandled data type:"+i.glslType)}else{const i=I[e+"Tex"];i&&"sampler2D"==i.glslType&&(C instanceof MaterialColorParam||C instanceof MaterialFloatParam)&&C.getImage()&&i&&"sampler2D"==i.glslType&&this.uniformBindings.push(new Sampler2DBinding(A,g,C,i,I))}},i=g.material.getParameters();for(const A of i)e(A)}bind(A){for(const g of this.uniformBindings)g.bind(A);return!0}unbind(A){for(const g of this.uniformBindings)g.unbind(A)}destroy(){for(const A of this.uniformBindings)A.destroy()}}class GLMaterial extends EventEmitter{gl;material;glShader;shaderBindings;boundTexturesBeforeMaterial;constructor(A,g,I){super(),this.gl=A,this.material=g,this.glShader=I,this.shaderBindings={},g.on("parameterValueChanged",(()=>this.emit("updated")))}bind(A,g){this.boundTexturesBeforeMaterial=A.boundTextures;let I=this.shaderBindings[A.shaderkey];if(!I){const C=this.gl;I=new MaterialShaderBinding(C,this,A.unifs,g),this.shaderBindings[A.shaderkey]=I}I.bind(A)}unbind(A){A.boundTextures=this.boundTexturesBeforeMaterial}}class GLMaterialLibrary extends EventEmitter{renderer;materials=[];materialBindings=new Map;freeIndices=[];dirtyIndices=new Set;materialsAllocator=new Allocator1D;materialsTexture;constructor(A){super(),this.renderer=A,this.materialsAllocator.on("dataReallocated",(A=>{const g=A.id;this.dirtyIndices.add(g)}))}addMaterial(A){const g=this.materialBindings.get(A);if(null!=g)return g.refCount++,g.index;const I=this.freeIndices.length>0?this.freeIndices.pop():this.materials.length;this.materials[I]=A;const C={},e=A.getShaderClass().getPackedMaterialData(A);this.materialsAllocator.allocate(I,e.length/4);return C.parameterValueChanged=A.on("parameterValueChanged",(()=>{this.dirtyIndices.add(I),this.emit("updated")})),this.materialBindings.set(A,{index:I,refCount:1,listenerIDs:C}),this.dirtyIndices.add(I),I}getGLMaterial(A){this.addMaterial(A);const g=this.materialBindings.get(A);if(g.glMaterial)return g.glMaterial;const I=this.renderer.getOrCreateShader(A.getShaderName()),C=this.renderer.gl,e=new GLMaterial(C,A,I);return g.listenerIDs.updated=e.on("updated",(()=>{this.renderer.requestRedraw()})),g.glMaterial=e,e}getMaterialAllocation(A){const g=this.materialBindings.get(A);if(null!=g)return this.materialsAllocator.getAllocation(g.index)}removeMaterial(A){const g=this.materialBindings.get(A);g.refCount--,g.refCount>0||(this.freeIndices.push(g.index),this.materialsAllocator.deallocate(g.index),this.materials[g.index]=null,this.materialBindings.delete(A),g.listenerIDs.updated&&g.glMaterial&&g.glMaterial.off("updated",g.listenerIDs.updated),g.listenerIDs.parameterValueChanged&&A.off("parameterValueChanged",g.listenerIDs.parameterValueChanged),this.dirtyIndices.has(g.index)&&this.dirtyIndices.delete(g.index))}uploadMaterials(A){const g=this.renderer.__gl,I=MathFunctions.nextPow2(Math.ceil(Math.sqrt(this.materialsAllocator.reservedSpace))),C=A.boundTextures++;if(g.activeTexture(g.TEXTURE0+C),this.materialsTexture){if(this.materialsTexture.width<I){this.materialsTexture.resize(I,I);for(let A=0;A<this.materials.length;A++)this.materialsAllocator.getAllocation(A)&&this.dirtyIndices.add(A)}}else this.materialsTexture=new GLTexture2D(this.renderer.__gl,{format:"RGBA",type:"FLOAT",width:I,height:I,filter:"NEAREST",wrap:"CLAMP_TO_EDGE",mipMapped:!1}),this.materialsTexture.clear();const e=this.materialsTexture,i=this.materialsTexture.width;g.bindTexture(g.TEXTURE_2D,e.glTex),this.dirtyIndices.forEach((A=>{const I=this.materialsAllocator.getAllocation(A),C=this.materials[A],t=C.getShaderClass().getPackedMaterialData(C),o=I.start%i,a=Math.ceil((o+I.size)/i);let B=0,s=I.size,l=o;for(let A=0;A<a;A++){let A;l+s>i?(A=i-l,l=0):A=s;const C=(I.start+B)%i,o=Math.floor((I.start+B)/i),a=t.subarray(4*B,4*(B+A));g.texSubImage2D(g.TEXTURE_2D,0,C,o,A,1,e.format,e.type,a),B+=A,s-=A}})),this.dirtyIndices=new Set,g.bindTexture(g.TEXTURE_2D,null),A.boundTextures--}update(A){this.dirtyIndices.size>0&&this.uploadMaterials(A)}bind(A){if(this.dirtyIndices.size>0&&this.uploadMaterials(A),!this.materialsTexture)return!1;const{materialsTexture:g,materialsTextureSize:I}=A.unifs;if(g&&(this.materialsTexture.bindToUniform(A,g),I)){this.renderer.gl.uniform2i(I.location,this.materialsTexture.width,this.materialsTexture.height)}return!0}}class GLPoints extends GLGeom{constructor(A,g){super(A,g)}draw(A){const g=this.__gl;A.shaderInstancedGeom?g.drawElementsInstanced(g.TRIANGLES,A.shaderInstancedGeom.numTriIndices,A.shaderInstancedGeom.indexDataType,0,this.numVertices):g.drawArrays(g.POINTS,0,this.numVertices)}drawInstanced(A,g){this.__gl.drawArraysInstanced(this.__gl.POINTS,0,this.numVertices,g)}}const convertValue=(A,g)=>A instanceof Int16Array||A instanceof Uint16Array||A instanceof Int16Array||A instanceof Uint16Array?MathFunctions.decode16BitFloat(A[g]):A[g],convertBuffer=(A,g,I)=>{const C=g.values,e=g.count*I.dimension;switch(I.dataType){case A.BYTE:if(C instanceof Int8Array)return C;const i=new Int8Array(e);return C instanceof Float32Array&&C.forEach(((A,C)=>{const e=C%g.dimension+Math.floor(C/g.dimension)*I.dimension;i[e]=MathFunctions.remap(A,-1,1,-127,127)})),i;case A.UNSIGNED_BYTE:{if(C instanceof Uint8Array)return C;const A=new Uint8Array(e);return C instanceof Float32Array&&C.forEach(((C,e)=>{const i=e%g.dimension+Math.floor(e/g.dimension)*I.dimension;A[i]=MathFunctions.remap(C,0,1,0,255)})),A}case A.UNSIGNED_SHORT:{if(C instanceof Uint16Array)return C;const A=new Uint16Array(e);for(let e=0;e<g.count;e++)A.set(C.subarray(e*g.dimension,(e+1)*g.dimension),e*I.dimension);return A}case A.SHORT:{if(C instanceof Int16Array)return C;const A=new Int16Array(e);for(let e=0;e<g.count;e++)A.set(C.subarray(e*g.dimension,(e+1)*g.dimension),e*I.dimension);return A}case A.HALF_FLOAT:if(C instanceof Uint16Array)return C;if(C instanceof Float32Array)return MathFunctions.convertFloat32ArrayToUInt16Array(C);throw"Unable to convert from src data array";case A.FLOAT:{if(C instanceof Float32Array)return C;const A=new Float32Array(e);if(C instanceof Uint16Array)for(let g=0;g<C.length;g++)A[g]=MathFunctions.decode16BitFloat(C[g]);return A}default:throw`Unhandled attribute type: ${I.dataType} for ${C.constructor.name}`}};class GLLines extends GLGeom{numSegIndices=-1;fatBuffersNeedUpload=!0;numFatVertices=-1;fatBuffers=null;indexDataType=0;constructor(A,g){super(A,g)}dirtyBuffers(A){super.dirtyBuffers(A),this.fatBuffersNeedUpload=!0,this.emit("updated")}clearBuffers(){const A=this.__gl;if(A.deleteBuffer(this.indexBuffer),this.indexBuffer=null,this.fatBuffers&&this.fatBuffers.positionsTexture){this.fatBuffers.positionsTexture&&(this.fatBuffers.positionsTexture.destroy(),this.fatBuffers.positionsTexture=null);const g=this.fatBuffers.glattrbuffers.segmentIndices;g.buffer&&(A.deleteBuffer(g.buffer),this.fatBuffers.glattrbuffers.segmentIndices=null)}super.clearBuffers()}genFatBuffers(A){const g=this.__gl,I=this.geom.genBuffers(),C=I.indices,e=I.numVertices!=this.numFatVertices;g.__quadVertexIdsBuffer||g.setupInstancedQuad(),this.fatBuffers||(this.fatBuffers={drawCount:0,positionsTexture:null,glattrbuffers:{}},this.fatBuffers.glattrbuffers.vertexIDs=g.__quadattrbuffers.vertexIDs);const i=A.boundTextures++;g.activeTexture(this.__gl.TEXTURE0+i),this.fatBuffers.drawCount=C.length/2;const t=I.attrBuffers.positions,o=I.attrBuffers.lineThickness,a=new Float32Array(4*t.count);for(let A=0;A<t.count;A++)a[4*A+0]=convertValue(t.values,3*A+0),a[4*A+1]=convertValue(t.values,3*A+1),a[4*A+2]=convertValue(t.values,3*A+2),a[4*A+3]=o?convertValue(o.values,A):1;e&&this.fatBuffers.positionsTexture&&(this.fatBuffers.positionsTexture.destroy(),this.fatBuffers.positionsTexture=null),this.fatBuffers.positionsTexture?this.fatBuffers.positionsTexture.bufferData(a,t.count,1):this.fatBuffers.positionsTexture=new GLTexture2D(this.__gl,{format:"RGBA",type:"FLOAT",width:t.count,height:1,filter:"NEAREST",wrap:"CLAMP_TO_EDGE",data:a,mipMapped:!1});const B=()=>{const A=new Float32Array(C.length);for(let g=0;g<C.length;g++){let I;I=g%2==0?g>0?C[g]==C[g-1]:C[g]==C[C.length-1]:g<C.length-1?C[g]==C[g+1]:C[g]==C[0],A[g]=(I?1:0)+2*C[g]}return A};if(this.fatBuffers.glattrbuffers.segmentIndices)(!this.genBufferOpts||this.genBufferOpts&&this.genBufferOpts.topologyChanged)&&(g.bindBuffer(g.ARRAY_BUFFER,this.fatBuffers.glattrbuffers.segmentIndices.buffer),g.bufferData(g.ARRAY_BUFFER,B(),g.STATIC_DRAW));else{const A=g.createBuffer();g.bindBuffer(g.ARRAY_BUFFER,A),g.bufferData(g.ARRAY_BUFFER,B(),g.STATIC_DRAW),this.fatBuffers.glattrbuffers.segmentIndices={dataType:g.FLOAT,name:"segmentIndices",dimension:2,elementSize:4,normalized:!1,shared:!1,numValues:C.length,buffer:A}}g.bindTexture(g.TEXTURE_2D,null),A.boundTextures--,this.numSegIndices=C.length,this.numFatVertices=I.numVertices,this.fatBuffersNeedUpload=!1}genBuffers(A){super.genBuffers(A);const g=this.__gl,I=this.geom.genBuffers(),C=I.indices;this.numSegIndices!=C.length&&(g.deleteBuffer(this.indexBuffer),this.indexBuffer=g.createBuffer()),g.bindBuffer(g.ELEMENT_ARRAY_BUFFER,this.indexBuffer),g.bufferData(g.ELEMENT_ARRAY_BUFFER,C,g.STATIC_DRAW),this.numSegIndices=C.length,this.numVertices=I.numVertices,C instanceof Uint8Array&&(this.indexDataType=this.__gl.UNSIGNED_BYTE),C instanceof Uint16Array&&(this.indexDataType=this.__gl.UNSIGNED_SHORT),C instanceof Uint32Array&&(this.indexDataType=this.__gl.UNSIGNED_INT)}bind(A){const g=this.__gl,I=A.unifs,{LineThickness:C,geomType:e}=A.unifs;if(e&&this.__gl.uniform1i(e.location,1),C&&g.floatTexturesSupported){this.fatBuffersNeedUpload&&this.genFatBuffers(A);const C=this.fatBuffers;let e=this.shaderBindings[A.shaderkey];return e||(e=generateShaderGeomBinding(this.__gl,A.attrs,C.glattrbuffers,g.__quadIndexBuffer),this.shaderBindings[A.shaderkey]=e),e.bind(A),I.positionsTexture&&(C.positionsTexture.bindToUniform(A,I.positionsTexture),g.uniform1i(I.positionsTextureSize.location,C.positionsTexture.width)),!0}return super.bind(A),!0}drawPoints(){this.__gl.drawArrays(this.__gl.POINTS,0,this.geom.getNumVertices())}draw(A){const g=this.__gl;A.unifs.LineThickness&&g.floatTexturesSupported?g.drawElementsInstanced(g.TRIANGLES,6,g.UNSIGNED_BYTE,0,this.fatBuffers.drawCount):g.drawElements(this.__gl.LINES,this.numSegIndices,this.indexDataType,0)}drawInstanced(A,g){const I=this.__gl,{occluded:C}=A.unifs;C&&I.uniform1i(C.location,0),I.drawElementsInstanced(this.__gl.LINES,this.numSegIndices,this.indexDataType,0,g),C&&(I.uniform1i(C.location,1),I.depthFunc(I.GREATER),I.drawElementsInstanced(this.__gl.LINES,this.numSegIndices,this.indexDataType,0,g),I.depthFunc(I.LEQUAL))}}class GLCompoundGeom extends GLGeom{indexDataType=0;drawCounts={};constructor(A,g){super(A,g)}genBuffers(A){super.genBuffers(A);const g=this.__gl,I=this.geom.genBuffers(),C=I.indices;let e=0;C instanceof Uint8Array?(this.indexDataType=this.__gl.UNSIGNED_BYTE,e=1):C instanceof Uint16Array?(this.indexDataType=this.__gl.UNSIGNED_SHORT,e=2):C instanceof Uint32Array&&(this.indexDataType=this.__gl.UNSIGNED_INT,e=4),this.numVertices=this.geom.getNumVertices(),this.indexBuffer&&g.deleteBuffer(this.indexBuffer),this.indexBuffer=g.createBuffer(),g.bindBuffer(g.ELEMENT_ARRAY_BUFFER,this.indexBuffer),g.bufferData(g.ELEMENT_ARRAY_BUFFER,I.indices,g.STATIC_DRAW),this.updateDrawIds(A,I,e),this.buffersDirty=!1}updateDrawIds(A,g,I){g.materials;g.materialSubGeoms;for(let A in g.counts)if(g.counts[A]>0){const C={geomType:GeomType$2[A],offsets:new Int32Array([g.offsets[A]*I]),counts:new Int32Array([g.counts[A]]),materialIds:new Uint8Array(g.counts.length)};this.drawCounts[A]=C}}updateBuffers(A){if(this.__gl,this.numVertices!=this.geom.getNumVertices())return void this.genBuffers(A);super.updateBuffers(A);let g=0;this.indexDataType==this.__gl.UNSIGNED_BYTE&&(g=1),this.indexDataType==this.__gl.UNSIGNED_SHORT&&(g=2),this.indexDataType==this.__gl.UNSIGNED_INT&&(g=4);const I=this.geom.genBuffers({includeIndices:!1});this.updateDrawIds(A,I,g)}clearBuffers(){this.__gl.deleteBuffer(this.indexBuffer),this.indexBuffer=null,super.clearBuffers()}draw(A){this.drawInstanced(A,1)}drawInstanced(A,g){A.pushGLStack("GLCompoundGeom.drawInstanced");const I=this.__gl,{drawIds:C,geomType:e,outlineThickness:i,viewportSize:t,occluded:o}=A.unifs,a=I.getParameter(I.DEPTH_FUNC);let B=null,s=!1,l=!1;if(A instanceof ColorRenderState){B=A.renderMode;const g="flat-noedges"!=B&&"shaded-noedges"!=B&&"pbr-noedges"!=B;s=i&&t&&"geometry"==A.outlineMethod&&A.outlineThickness>0&&g,l="hiddenline"==B&&null!=o}if(this.drawCounts.TRIANGLES){e&&I.uniform1i(e.location,GeomType$2.TRIANGLES),"hiddenline"==B&&I.colorMask(!1,!1,!1,!1),i&&I.uniform1f(i.location,0);const o=this.drawCounts.TRIANGLES,a=o.counts.map((()=>g));if(I.multiDrawElementsInstanced)I.multiDrawElementsInstanced(I.TRIANGLES,o.counts,0,this.indexDataType,o.offsets,0,a,0,o.counts.length);else for(let A=0;A<o.counts.length;A++)C&&I.uniform2i(C.location,A,o.materialIds[A]),this.__gl.drawElementsInstanced(I.TRIANGLES,o.counts[A],this.indexDataType,o.offsets[A],g);if(s){if(A.pushGLStack("GLCompoundGeom.drawingOutlines"),A.glEnable(I.CULL_FACE),I.cullFace(I.FRONT),I.uniform1f(i.location,A.outlineThickness*window.devicePixelRatio),I.uniform2f(t.location,A.region[2]-A.region[0],A.region[3]-A.region[1]),"hiddenline"==B&&I.colorMask(!0,!0,!0,!0),I.multiDrawElementsInstanced)I.multiDrawElementsInstanced(I.TRIANGLES,o.counts,0,this.indexDataType,o.offsets,0,a,0,o.counts.length);else for(let A=0;A<o.counts.length;A++)C&&I.uniform2i(C.location,A,o.materialIds[A]),this.__gl.drawElementsInstanced(I.TRIANGLES,o.counts[A],this.indexDataType,o.offsets[A],g);A.popGLStack(),I.cullFace(I.BACK)}}if(A instanceof ColorRenderState&&(A.glEnable(I.BLEND),I.blendEquation(I.FUNC_ADD),I.blendFuncSeparate(I.SRC_ALPHA,I.ONE_MINUS_SRC_ALPHA,I.ONE,I.ONE)),this.drawCounts.LINES){e&&I.uniform1i(e.location,GeomType$2.LINES);const i=this.drawCounts.LINES,t=i.counts.map((()=>g));if(I.multiDrawElementsInstanced)I.multiDrawElementsInstanced(I.LINES,i.counts,0,this.indexDataType,i.offsets,0,t,0,i.counts.length);else for(let A=0;A<i.counts.length;A++)C&&I.uniform2i(C.location,A,i.materialIds[A]),I.drawElementsInstanced(I.LINES,i.counts[A],this.indexDataType,i.offsets[A],g);if(l){const{hiddenLineColor:e}=A.unifs;if(I.uniform1i(o.location,1),I.uniform4fv(e.location,A.hiddenLineColor.asArray()),I.depthFunc(I.GREATER),I.depthMask(!1),I.multiDrawElementsInstanced)I.multiDrawElementsInstanced(I.LINES,i.counts,0,this.indexDataType,i.offsets,0,t,0,i.counts.length);else for(let A=0;A<i.counts.length;A++)C&&I.uniform2i(C.location,A,i.materialIds[A]),I.drawElementsInstanced(I.LINES,i.counts[A],this.indexDataType,i.offsets[A],g);I.depthFunc(a),I.depthMask(!0),I.uniform1i(o.location,0)}}if(this.drawCounts.POINTS){e&&I.uniform1i(e.location,GeomType$2.POINTS);const C=this.drawCounts.POINTS,i=C.counts.map((()=>g));if(I.multiDrawElementsInstanced(I.POINTS,C.counts,0,this.indexDataType,C.offsets,0,i,0,C.counts.length),l){const{hiddenLineColor:g}=A.unifs;I.uniform1i(o.location,1),I.uniform4fv(g.location,A.hiddenLineColor.asArray()),I.depthFunc(I.GREATER),I.depthMask(!1),I.multiDrawElementsInstanced(I.POINTS,C.counts,0,this.indexDataType,C.offsets,0,i,0,C.counts.length),I.depthFunc(a),I.depthMask(!0),I.uniform1i(o.location,0)}}A.popGLStack()}destroy(){super.destroy();this.__gl.deleteBuffer(this.indexBuffer),this.indexBuffer=null}}class GLImageAtlas extends GLTexture2D{subImages=[];layoutNeedsRegeneration=!1;asyncCount=0;layout=[];freeIndices=[];atlasLayoutTexture;layoutVec4s=[];constructor(A,g,I){super(A),this.format=g,this.type=I}incAsyncCount(A=1){this.asyncCount+=A}decAsyncCount(){this.asyncCount>0&&(this.asyncCount--,0==this.asyncCount&&this.emit("loaded"))}isLoaded(){return 0==this.asyncCount}get needsUpdate(){return this.layoutNeedsRegeneration}addSubImage(A){const g=this.freeIndices.length?this.freeIndices.pop():this.subImages.length;return A.isLoaded()||(this.incAsyncCount(),A.on("loaded",(()=>{A.getParams(),this.decAsyncCount()}))),A.on("updated",(()=>{this.layoutNeedsRegeneration=!0})),this.subImages[g]=A,this.layoutNeedsRegeneration=!0,g}removeSubImage(A){const g=this.subImages.indexOf(A);this.freeIndices.push(g),this.subImages[g]=null,this.layoutNeedsRegeneration=!0}getSubImage(A){return this.subImages[A]}numSubImages(){return this.layout?this.layout.length:this.subImages.length-this.freeIndices.length}generateAtlasLayout(){if(this.subImages.length-this.freeIndices.length==0)return void(this.layoutNeedsRegeneration=!1);const A=[];this.subImages.forEach(((g,I)=>{g&&A.push({w:g.width+4,h:g.height+4,area:g.width*g.height,index:I})})),A.sort(((A,g)=>A.area>g.area?-1:A.area<g.area?1:0));const g=new GrowingPacker;g.fit(A),this.layout=[],A.forEach((A=>{A.fit?this.layout[A.index]={pos:new Vec2(A.fit.x+2,A.fit.y+2),size:new Vec2(A.w,A.h)}:console.warn("Unable to fit image")}));const I=g.root.w,C=g.root.h,e=this.gl;this.glTex?this.resize(I,C):this.configure({width:I,height:C,format:this.format,type:this.type,filter:e.LINEAR});let i=Math.round(Math.sqrt(1*this.layout.length)+.5);if(i=MathFunctions.nextPow2(i),i%1!=0&&(i+=1-i%1),e.floatTexturesSupported){const A=new Float32Array(i*i*4);this.layout.forEach(((g,e)=>{if(!g)return;const i=A.subarray(4*e,4*(e+1));i[0]=g.pos.x/I,i[1]=g.pos.y/C,i[2]=g.size.x/I,i[3]=g.size.y/C})),this.atlasLayoutTexture&&this.atlasLayoutTexture.width==i&&this.atlasLayoutTexture.height==i?this.atlasLayoutTexture.bufferData(A,i,i):(this.atlasLayoutTexture&&this.atlasLayoutTexture.destroy(),this.atlasLayoutTexture=new GLTexture2D(e,{format:e.RGBA,type:e.FLOAT,filter:e.NEAREST,wrap:e.CLAMP_TO_EDGE,mipMapped:!1,width:i,height:i,data:A}))}else this.layoutVec4s=[],this.layout.forEach(((A,g)=>{A&&(this.layoutVec4s[g]=[A.pos.x/I,A.pos.y/C,A.size.x/I,A.size.y/C])}));this.textureDesc[0]=this.width,this.textureDesc[1]=this.height,this.textureDesc[2]=this.atlasLayoutTexture.width,this.layoutNeedsRegeneration=!1}getLayoutData(A){return this.layoutVec4s[A]}renderAtlas(A=!1,g=0){if(this.subImages.length-this.freeIndices.length==0)return;this.layoutNeedsRegeneration&&this.generateAtlasLayout();const I=this.gl;I.bindTexture(I.TEXTURE_2D,this.glTex);for(let A=g;A<this.subImages.length;A++){const g=this.subImages[A];if(!g)continue;const C=this.layout[A],e=g.getParams().data;I.texSubImage2D(I.TEXTURE_2D,0,C.pos.x,C.pos.y,g.width,g.height,this.format,this.type,e)}A&&this.cleanup(),this.emit("updated")}isReady(){return null!=this.atlasLayoutTexture}bindToUniform(A,g){super.bindToUniform(A,g);const I=A.unifs;if(this.atlasLayoutTexture){const C=I[g.name+"_layout"];C&&this.atlasLayoutTexture.bindToUniform(A,C);const e=I[g.name+"_desc"];e&&this.gl.uniform4fv(e.location,this.textureDesc)}else{const A=I[g.name+"_desc"];A&&this.gl.uniform4f(A.location,0,0,0,0)}return!0}cleanup(){this.subImages=[],this.destroy()}destroy(){this.cleanup(),super.destroy()}}class GLFatPoints extends GLPoints{indexArray=new Int32Array(0);distances=new Float32Array(0);pointsAttributesTexture=null;texelsPerPoint=1;prevSortCameraPos=new Vec3;threshold=0;atlas=null;spriteCoords=null;constructor(A,g){super(A,g)}genBuffers(A){const g=this.__gl,I=this.geom,C=this.geom.genBuffers();this.numVertices=this.geom.getNumVertices(),this.texelsPerPoint=1;const e=C.attrBuffers.positions,i=C.attrBuffers.colors,t=C.attrBuffers.sizes;i&&(this.texelsPerPoint=2);const o=C.attrBuffers.spriteIndices,a=I.getParameter("Sprites");if(o&&a&&a.value.length>0){this.texelsPerPoint=3,this.atlas=new GLImageAtlas(g,g.RGBA,g.UNSIGNED_BYTE);const A=A=>this.emit("updated",A);this.atlas.on("loaded",A),this.atlas.on("updated",A);a.value.forEach((A=>this.atlas.addSubImage(A))),this.atlas.isLoaded()?this.atlas.renderAtlas():this.atlas.on("loaded",(()=>{this.atlas.renderAtlas()}))}const B=MathFunctions.nextPow2(Math.round(Math.sqrt(this.numVertices)+.5)),s=new Float32Array(B*this.texelsPerPoint*B*4),l=4*this.texelsPerPoint;for(let A=0;A<e.count;A++)s[A*l+0]=convertValue(e.values,3*A+0),s[A*l+1]=convertValue(e.values,3*A+1),s[A*l+2]=convertValue(e.values,3*A+2),s[A*l+3]=t?t.values[A]:1,i&&s.set(i.values.subarray(4*A,4*(A+1)),A*l+4),o&&(s[A*l+8]=o.values[A]);const n=B*this.texelsPerPoint,d=B;this.pointsAttributesTexture=new GLTexture2D(g,{format:g.RGBA,type:g.FLOAT,width:n,height:d,data:s,filter:g.NEAREST,wrap:g.CLAMP_TO_EDGE,mipMapped:!1}),this.numVertices=C.numVertices,this.buffersDirty=!1}updateBuffers(A){if(this.numVertices!=this.geom.getNumVertices())return void this.genBuffers(A);const g=this.geom.genBuffers(),I=g.attrBuffers.positions,C=g.attrBuffers.sizes,e=g.attrBuffers.colors,i=g.attrBuffers.spriteIndices,t=MathFunctions.nextPow2(Math.round(Math.sqrt(g.numVertices)+.5)),o=new Float32Array(t*this.texelsPerPoint*t*4),a=4*this.texelsPerPoint;for(let A=0;A<I.count;A++)o[A*a+0]=convertValue(I.values,3*A+0),o[A*a+1]=convertValue(I.values,3*A+1),o[A*a+2]=convertValue(I.values,3*A+2),o[A*a+3]=C?C.values[A]:1,e&&o.set(e.values.subarray(4*A,4*(A+1)),A*a+4),i&&(o[A*a+8]=i.values[A]);const B=t*this.texelsPerPoint,s=t;this.pointsAttributesTexture.populate(o,B,s),this.numVertices=g.numVertices,this.buffersDirty=!1}sort(A){const g=this.geom.positions;this.distances=new Float32Array(g.count);let I=!1;this.indexArray.length!=this.distances.length&&(this.indexArray=new Int32Array(this.distances.length),I=!0);for(let I=0;I<g.count;I++){const C=g.getValue(I);this.distances[I]=C.distanceTo(A),this.indexArray[I]=I}this.indexArray.sort(((A,g)=>this.distances[A]>this.distances[g]?-1:1));const C=this.__gl;this.glattrbuffers.drawIndices?I&&(this.glattrbuffers.drawIndices.buffer&&C.deleteBuffer(this.glattrbuffers.drawIndices.buffer),this.glattrbuffers.drawIndices.buffer=C.createBuffer()):this.glattrbuffers.drawIndices={name:"drawIndices",dimension:1,elementSize:4,buffer:C.createBuffer(),dataType:C.INT,normalized:!1,shared:!1,numValues:g.count},C.bindBuffer(C.ARRAY_BUFFER,this.glattrbuffers.drawIndices.buffer),C.bufferData(C.ARRAY_BUFFER,this.indexArray,C.STATIC_DRAW),this.threshold=0}bind(A){if(this.buffersDirty&&this.updateBuffers(A),A.attrs.drawIndices){const g=A.viewXfo.tr;if(g.distanceTo(this.prevSortCameraPos)>this.threshold)if(this.sort(g),this.prevSortCameraPos=g.clone(),this.distances.length>1){const A=this.indexArray[this.indexArray.length-1],g=this.distances[A];this.threshold=.25*g}else this.threshold=9999}let g=this.shaderBindings[A.shaderkey];if(!g){const I=Object.assign(this.glattrbuffers,A.shaderInstancedGeom.attrBuffers);g=generateShaderGeomBinding(this.__gl,A.attrs,I,A.shaderInstancedGeom.indexBuffer),this.shaderBindings[A.shaderkey]=g}g.bind(A);const{pointsAttributes:I,texelsPerPoint:C,atlasSprites:e}=A.unifs;I&&(this.pointsAttributesTexture.bindToUniform(A,I),this.__gl.uniform1i(C.location,this.texelsPerPoint));const{geomType:i}=A.unifs;return i&&this.__gl.uniform1i(i.location,2),e&&this.atlas&&this.atlas.bindToUniform(A,e),!0}draw(A){const g=this.__gl;A.shaderInstancedGeom?g.drawElementsInstanced(g.TRIANGLES,A.shaderInstancedGeom.numTriIndices,A.shaderInstancedGeom.indexDataType,0,this.numVertices):g.drawArrays(g.POINTS,0,this.numVertices)}}const resizeIntArray=(A,g)=>{const I=new Int32Array(g);return I.set(A),I};class GLGeomLibrary extends EventEmitter{renderer;__gl;freeGeomIndices=[];geoms=[];geomRefCounts=[];geomsDict=new Map;glGeomsDict=new Map;geomBuffersTmp=[];shaderAttrSpec={};glattrbuffers={};shaderBindings={};attributesBufferNeedsRealloc=!1;attributesBufferNeedsAlloc=[];attributesAllocator=new Allocator1D;dirtyGeomIndices=new Set;geomVertexOffsets=new Int32Array(1);geomVertexCounts=new Int32Array(1);numIndices=0;indicesBufferNeedsRealloc=!1;indicesAllocator=new Allocator1D;indicesCounts=new Int32Array(1);indicesOffsets=new Int32Array(1);indexBuffer=null;freeDataAfterUpload=!1;__destroyed=!1;geomEventHandlerIds=[];constructor(A){super(),this.renderer=A,this.__gl=A.gl,this.attributesAllocator.on("resized",(()=>{this.attributesBufferNeedsRealloc=!0})),this.attributesAllocator.on("dataReallocated",(A=>{const g=A.id,I=A.allocation;this.dirtyGeomIndices.add(g),this.geomVertexOffsets[g]=I.start,this.geomVertexCounts[g]=I.size})),this.freeGeomIndices.push(0),this.indicesAllocator.on("resized",(()=>{if(Math.log2(this.indicesAllocator.reservedSpace)>=29){if(Math.log2(this.indicesAllocator.allocatedSpace)>=29)throw"Indices buffer too big. WebGL cannot allocate index buffers more than 2Gb";this.indicesAllocator.reservedSpace=536870911}this.indicesBufferNeedsRealloc=!0})),this.indicesAllocator.on("dataReallocated",(A=>{const g=A.id;this.dirtyGeomIndices.add(g)}));const g=2<<19;this.reserveSpace(g,2*g)}reserveSpace(A,g){this.attributesAllocator.reservedSpace=A,this.indicesAllocator.reservedSpace=g,this.attributesBufferNeedsRealloc=!0,this.indicesBufferNeedsRealloc=!0}constructGLGeom(A){let g=this.glGeomsDict.get(A);if(null!=g)return g;const I=this.__gl;if(A instanceof Mesh||A instanceof MeshProxy)g=new GLMesh(I,A);else if(A instanceof Lines||A instanceof LinesProxy)g=new GLLines(I,A);else if(A instanceof FatPoints)g=new GLFatPoints(I,A);else if(A instanceof Points||A instanceof PointsProxy)g=new GLPoints(I,A);else{if(!(A instanceof CompoundGeom))throw new Error("Unsupported geom type:"+A.constructor.name);g=new GLCompoundGeom(I,A)}return this.glGeomsDict.set(A,g),g.on("updated",(()=>{this.renderer.requestRedraw()})),g.addRef(this),g}addGeom(A){let g=this.geomsDict.get(A);if(null!=g)return this.geomRefCounts[g]++,g;if(0==this.freeGeomIndices.length){const A=this.geomVertexCounts.length,g=2*A;this.geomVertexCounts=resizeIntArray(this.geomVertexCounts,g),this.geomVertexOffsets=resizeIntArray(this.geomVertexOffsets,g),this.indicesCounts=resizeIntArray(this.indicesCounts,g),this.indicesOffsets=resizeIntArray(this.indicesOffsets,g);for(let I=g-1;I>=A;I--)this.freeGeomIndices.push(I)}g=this.freeGeomIndices.pop(),this.geoms[g]=A,this.geomRefCounts[g]=1,this.geomsDict.set(A,g),this.dirtyGeomIndices.add(g),this.geomVertexCounts[g]=0,this.geomVertexOffsets[g]=0,this.indicesCounts[g]=0,this.indicesOffsets[g]=0;const I={};if(I.geomDataChanged=A.on("geomDataChanged",(()=>{this.dirtyGeomIndices.add(g),this.emit("updated")})),I.geomDataTopologyChanged=A.on("geomDataTopologyChanged",(()=>{this.dirtyGeomIndices.add(g),this.emit("updated")})),A instanceof CompoundGeom){const C=this.renderer.glMaterialLibrary;A.materials.forEach((A=>{A&&C.addMaterial(A)})),I.materialsChanged=A.on("materialsChanged",(()=>{A.materials.forEach((A=>{A&&C.addMaterial(A)})),this.emit("geomDataChanged",new IndexEvent(g)),this.emit("updated")}))}return this.geomEventHandlerIds[g]=I,g}removeGeom(A){let g;if(A instanceof BaseGeom||A instanceof BaseProxy?g=this.geomsDict.get(A):(g=A,A=this.geoms[g]),null==g)return void console.warn("geom does not exist in the GLGeomLibrary");if(this.geomRefCounts[g]--,this.geomRefCounts[g]>0)return;this.attributesAllocator.getAllocation(g)&&this.attributesAllocator.deallocate(g),this.indicesAllocator.getAllocation(g)&&this.indicesAllocator.deallocate(g),this.dirtyGeomIndices.has(g)&&this.dirtyGeomIndices.delete(g),this.geomVertexCounts[g]=0,this.geomVertexOffsets[g]=0,this.geoms[g]=null,this.freeGeomIndices.push(g),this.geomsDict.delete(A),delete this.geomBuffersTmp[g],this.indicesCounts[g]=0,this.indicesOffsets[g]=0;const I=this.geomEventHandlerIds[g];A.off("geomDataChanged",I.geomDataChanged),A.off("geomDataTopologyChanged",I.geomDataTopologyChanged),A instanceof CompoundGeom&&A.off("materialsChanged",I.materialsChanged)}getGeom(A){return this.geoms[A]}getGeomOffsetAndCount(A){return[this.indicesOffsets[A],this.indicesCounts[A]]}getGeomBuffers(A){return this.geomBuffersTmp[A]}allocateBuffers(A){const g=this.geoms[A];if(!g)return;const I=g.genBuffers(),C=I.numRenderVerts?I.numRenderVerts:I.numVertices;if(this.geomVertexCounts[A]!=C)if(0==C)this.attributesAllocator.deallocate(A),this.geomVertexOffsets[A]=0,this.geomVertexCounts[A]=0;else{const g=this.attributesAllocator.allocate(A,C);this.geomVertexOffsets[A]=g.start,this.geomVertexCounts[A]=g.size}for(const A in I.attrBuffers)if("textureCoords"!=A&&"texCoords"!=A&&!this.shaderAttrSpec[A]){const g=I.attrBuffers[A];this.shaderAttrSpec[A]=genDataTypeDesc(this.__gl,g.dataType),this.shaderAttrSpec[A].normalized=g.normalized,this.attributesBufferNeedsAlloc.push(A)}if(I.indices){const g=I.indices.length;if(this.indicesCounts[A]!=g)if(0==g)this.indicesAllocator.deallocate(A),this.indicesOffsets[A]=0,this.indicesCounts[A]=0;else{const I=this.indicesAllocator.allocate(A,g),C=4;this.indicesOffsets[A]=I.start*C,this.indicesCounts[A]=I.size}}else this.indicesOffsets[A]=this.geomVertexOffsets[A],this.indicesCounts[A]=this.geomVertexCounts[A];this.geomBuffersTmp[A]=I}genAttributesBuffers(){for(const A in this.shaderAttrSpec)this.genAttributesBuffer(A);this.attributesBufferNeedsAlloc=[]}genAttributesBuffer(A){if("textureCoords"==A||"texCoords"==A)return;const g=this.attributesAllocator.reservedSpace,I=this.__gl;{const C=this.shaderAttrSpec[A],e=g*C.dimension,i=I.createBuffer();I.bindBuffer(I.ARRAY_BUFFER,i);const t=e*C.elementSize;I.bufferData(I.ARRAY_BUFFER,t,I.STATIC_DRAW),this.glattrbuffers[A]&&this.glattrbuffers[A].buffer&&(I.bindBuffer(I.COPY_WRITE_BUFFER,i),I.bindBuffer(I.COPY_READ_BUFFER,this.glattrbuffers[A].buffer),I.copyBufferSubData(I.COPY_READ_BUFFER,I.COPY_WRITE_BUFFER,0,0,this.glattrbuffers[A].numValues*C.elementSize),I.deleteBuffer(this.glattrbuffers[A].buffer)),this.glattrbuffers[A]={name:A,elementSize:C.elementSize,buffer:i,dataType:C.dataType,normalized:C.normalized,numValues:e,dimension:C.dimension,shared:!1}}}genIndicesBuffers(){const A=this.indicesAllocator.reservedSpace;if(this.numIndices!=A){const g=this.__gl,I=g.createBuffer();g.bindBuffer(g.ELEMENT_ARRAY_BUFFER,I);const C=4,e=A*C;g.bufferData(g.ELEMENT_ARRAY_BUFFER,e,g.STATIC_DRAW),this.indexBuffer&&(g.bindBuffer(g.COPY_WRITE_BUFFER,I),g.bindBuffer(g.COPY_READ_BUFFER,this.indexBuffer),g.copyBufferSubData(g.COPY_READ_BUFFER,g.COPY_WRITE_BUFFER,0,0,this.numIndices*C),g.deleteBuffer(this.indexBuffer)),this.indexBuffer=I,this.numIndices=A}}uploadBuffers(A){const g=this.__gl;let I=this.geomBuffersTmp[A];if(!I){const g=this.geoms[A];if(!g)return;I=g.genBuffers(),this.geomBuffersTmp[A]=I}const C=this.geomVertexCounts[A],e=I.numRenderVerts?I.numRenderVerts:I.numVertices;if(C!=e)throw new Error("Invalid allocation for this geom");if(0!=e){for(const C in I.attrBuffers){const e=this.shaderAttrSpec[C],i=I.attrBuffers[C],t=this.glattrbuffers[C];if(!i||!t)continue;g.bindBuffer(g.ARRAY_BUFFER,t.buffer);const o=e.elementSize,a=this.geomVertexOffsets[A]*o*e.dimension,B=convertBuffer(g,i,e);g.bufferSubData(g.ARRAY_BUFFER,a,B)}if(g.bindBuffer(g.ARRAY_BUFFER,null),I.indices&&I.indices.length>0){const g=I.indices,C=this.indicesAllocator.getAllocation(A);if(C.size!=g.length)throw new Error("Invalid allocation for this geom");const e=this.attributesAllocator.getAllocation(A),i=new Uint32Array(C.size);for(let A=0;A<g.length;A++)i[A]=I.indices[A]+e.start;const t=this.__gl;t.bindBuffer(t.ELEMENT_ARRAY_BUFFER,this.indexBuffer);const o=4,a=C.start*o;t.bufferSubData(t.ELEMENT_ARRAY_BUFFER,a,i),t.bindBuffer(t.ELEMENT_ARRAY_BUFFER,null)}if(this.freeDataAfterUpload){this.geoms[A].freeBuffers()}this.emit("geomDataChanged",new IndexEvent(A))}else{const g=new IndexEvent(A);this.emit("geomDataChanged",g)}}cleanGeomBuffers(){this.dirtyGeomIndices.forEach((A=>{this.allocateBuffers(A)})),this.attributesBufferNeedsRealloc||this.indicesBufferNeedsRealloc?(this.resetBindings(),this.attributesBufferNeedsRealloc&&(this.genAttributesBuffers(),this.attributesBufferNeedsRealloc=!1),this.indicesBufferNeedsRealloc&&(this.genIndicesBuffers(),this.indicesBufferNeedsRealloc=!1)):this.attributesBufferNeedsAlloc.length>0&&(this.attributesBufferNeedsAlloc.forEach((A=>{this.genAttributesBuffer(A)})),this.attributesBufferNeedsAlloc=[]),this.dirtyGeomIndices.forEach((A=>{this.uploadBuffers(A)})),this.dirtyGeomIndices=new Set}calcMemoryAllocation(){const A={attrs:{},indices:{}},g=1<<20;for(const I in this.shaderAttrSpec){const C=this.shaderAttrSpec[I],e=this.attributesAllocator.reservedSpace,i=this.attributesAllocator.allocatedSpace,t=e*C.dimension*C.elementSize;A.attrs[I]={count:i,MB:t/g}}const I=this.indicesAllocator.reservedSpace,C=this.indicesAllocator.allocatedSpace,e=4*I;return A.indices={count:C,MB:e/g},A}resetBindings(){for(const A in this.shaderBindings){this.shaderBindings[A].destroy()}this.shaderBindings={}}bind(A){this.dirtyGeomIndices.size>0&&this.cleanGeomBuffers();const g=A.glShader.getId()+A.shaderkey;let I=this.shaderBindings[g];if(I)I.bind(A);else{if(I=generateShaderGeomBinding(this.__gl,A.attrs,this.glattrbuffers,this.indexBuffer),this.shaderBindings[g]=I,"Safari"==SystemDesc.browserName&&"15.4"==SystemDesc.fullVersion){const A=this.__gl;A.drawElements(A.POINTS,1,A.UNSIGNED_INT,0),A.drawElements(A.LINES,2,A.UNSIGNED_INT,0),A.drawElements(A.TRIANGLES,3,A.UNSIGNED_INT,0);const g=new Uint32Array(1),I=4,C=this.indicesAllocator.allocatedSpace*I;A.bufferSubData(A.ELEMENT_ARRAY_BUFFER,C,g)}}return!0}unbind(A){const g=A.glShader.getId()+A.shaderkey,I=this.shaderBindings[g];I&&I.unbind(A)}clearBuffers(){const A=this.__gl;for(const g in this.glattrbuffers){const I=this.glattrbuffers[g];I.shared||A.deleteBuffer(I.buffer)}this.glattrbuffers={},this.indexBuffer&&(A.deleteBuffer(this.indexBuffer),this.indexBuffer=null),this.resetBindings()}destroy(){this.clearBuffers(),this.__destroyed=!0,this.emit("destructing")}}const GLGeomItemChangeType={GEOMITEM_CHANGED:0,GEOM_CHANGED:1,VISIBILITY_CHANGED:2,HIGHLIGHT_CHANGED:3},GLGeomItemFlags={GEOMITEM_FLAG_CUTAWAY:2,GEOMITEM_INVISIBLE_IN_GEOMDATA:4,GEOMITEM_TRANSPARENT:8};class GLGeomItem extends EventEmitter{listenerIDs={};GLGeomItemSet;GLShaderGeomSets;gl;geomItem;geomItemId;geomId;materialId;supportInstancing;visible;shattered=!1;culled=!1;geomMatrixDirty=!1;modelMatrixArray;geomData=[];highlightSubIndex=-1;constructor(A,g,I,C,e,i=!1){if(super(),this.gl=A,this.geomItem=g,this.geomItemId=I,this.geomId=C,this.materialId=e,this.supportInstancing=i,this.visible=this.geomItem.isVisible(),this.listenerIDs.visibilityChanged=this.geomItem.on("visibilityChanged",(A=>{const g=!this.culled&&this.visible;this.visible=A.state;const I=!this.culled&&this.visible;g!=I&&this.emit("visibilityChanged",new StateChangedEvent(I))})),g instanceof CADBody&&(this.shattered=g.shattered),this.listenerIDs.shatterStateChanged=this.geomItem.on("shatterStateChanged",(A=>{this.shattered=A.state,this.emit("shatterStateChanged",A)})),!this.supportInstancing){const A=A=>{if(A.name){const g=A.name,I=g.indexOf(":");let C=[];-1!=I&&(C=g.substring(I+1).split(",").map((A=>Number.parseInt(A))),this.highlightSubIndex=C[0])}else this.highlightSubIndex=-1};this.listenerIDs.highlightChanged=this.geomItem.on("highlightChanged",A)}}isVisible(){return!this.culled&&this.visible}setCulled(A){const g=!this.culled&&this.visible;this.culled=A;const I=!this.culled&&this.visible;g!=I&&this.emit("visibilityChanged",new StateChangedEvent(I))}bind(A){const g=this.gl,{highlightSubIndex:I,geomItemId:C}=A.unifs;return I&&g.uniform1i(I.location,this.highlightSubIndex),C&&g.uniform1i(C.location,this.geomItemId),!0}destroy(){this.geomItem.off("visibilityChanged",this.listenerIDs.visibilityChanged),this.supportInstancing||this.geomItem.off("highlightChanged",this.listenerIDs.highlightChanged)}}class ReductionShader extends GLShader{constructor(A){super(A,"ReductionShader"),this.setShaderStage("VERTEX_SHADER","\n\nprecision highp float;\n\nuniform int reductionTextureWidth;\nuniform sampler2D geomDataTexture;\n\nivec2 texelCoordFromVertexId(){\n  ivec2 texSize = textureSize(geomDataTexture, 0);\n  return ivec2(\n    gl_VertexID % texSize.x, \n    gl_VertexID / texSize.x\n    );\n}\n\nvec2 pointPositionFromGeomItemId(int geomItemId){\n  vec2 result = vec2(\n    (float(geomItemId % reductionTextureWidth) + 0.5) / float(reductionTextureWidth), \n    (float(geomItemId / reductionTextureWidth) + 0.5) / float(reductionTextureWidth)\n    );\n  return vec2(-1.0, -1.0) + (result * 2.0);\n}\n\n/* VS Outputs */\nvarying vec4 v_geomDataPixel;\n\nvoid main()\n{\n  // Get the texel coordinate in the source geomdata buffer.\n  // there is one point for every pixel in the geomdata texture.\n  ivec2 texelCoord = texelCoordFromVertexId();\n  v_geomDataPixel = texelFetch(geomDataTexture, texelCoord, 0);\n  int geomItemId = int(v_geomDataPixel.r + 0.5);\n\n  if (geomItemId > 0) {\n    vec2 position = pointPositionFromGeomItemId(geomItemId);\n    gl_Position = vec4(position, 0.0, 1.0);\n    gl_PointSize = 1.0;\n  } else {\n    // Move it off screen\n    gl_Position = vec4(-2.0, -2.0, 0.0, 1.0);\n    gl_PointSize = 0.0;\n  }\n}\n\n"),this.setShaderStage("FRAGMENT_SHADER","\nprecision highp float;\n\n\n/* VS Outputs */\nvarying vec4 v_geomDataPixel;\n\n#ifdef ENABLE_ES3\n  out vec4 fragColor;\n#endif\n\nvoid main(void) {\n#ifndef ENABLE_ES3\n  vec4 fragColor;\n#endif\n\n  fragColor.r = 1.0/255.0;\n\n  // Note: the green channel in the histogram image indicates if the source\n  // pixel contained a real geom or a bounding box.\n  fragColor.g = v_geomDataPixel.g;\n\n#ifndef ENABLE_ES3\n  gl_FragColor = fragColor;\n#endif\n}\n\n")}}var frag$g="\nprecision highp float;\n#define GLSLIFY 1\n\n/* VS Outputs */\nvarying vec4 v_color;\n\n#ifdef ENABLE_ES3\n  out vec4 fragColor;\n#endif\nvoid main(void) {\n#ifndef ENABLE_ES3\n  vec4 fragColor;\n#endif\n\n  int drawItemId = int(v_color.g);\n  fragColor = v_color;\n\n#ifndef ENABLE_ES3\n  gl_FragColor = fragColor;\n#endif\n}\n",vert$h="\nprecision highp float;\n#define GLSLIFY 1\n\nattribute vec4 positions;\n\nuniform mat4 viewMatrix;\nuniform mat4 projectionMatrix;\nuniform mat4 cameraMatrix;\n\nuniform highp int occlusionCulling;\nuniform sampler2D reductionDataTexture;\n\nimport 'transpose.glsl'\nimport 'GLSLUtils.glsl'\nimport 'stack-gl/transpose.glsl'\nimport 'stack-gl/inverse.glsl'\nimport 'geomItemId.glsl'\nimport 'drawItemTexture.glsl'\nimport 'modelMatrix.glsl'\n\nconst int GEOMITEM_INVISIBLE_IN_GEOMDATA = 2; // 1<<1;\n\n/* VS Outputs */\nvarying vec4 v_color;\n\nvoid main(void) {\n\n  int drawItemId = getGeomItemId();\n  vec4 geomItemData  = getInstanceData(drawItemId);\n  int flags = int(geomItemData.r + 0.5);\n\n  if (occlusionCulling != 0) {\n    // Check if in the reduction texture, this item is already flagged as visible.\n    // Note: we only draw bboxes for those that have been flagged as invisible, but might\n    // be just off screen, or onscreen, but were culled in the previous update.\n    float isVisible = fetchTexel(reductionDataTexture, textureSize(reductionDataTexture, 0), drawItemId).r;\n    if (isVisible > 0.0) {\n      return;\n    }\n  }\n\n  vec4 bboxMin = fetchTexel(instancesTexture, instancesTextureSize, (drawItemId * pixelsPerItem) + 6);\n  vec4 bboxMax = fetchTexel(instancesTexture, instancesTextureSize, (drawItemId * pixelsPerItem) + 7);\n  mat4 viewProjectionMatrix = projectionMatrix * viewMatrix;\n\n  if (occlusionCulling != 0) {\n    // Note: write the drawItemId to the red channel.\n    // this aligns with src\\Renderer\\Shaders\\GLSL\\surfaceGeomData.glsl:39\n    // The ReductionShader reads the drawItemId from the red channel, and if the texel\n    // contained a bbox from the green channel. (in this case, it does not)\n    // Note: the alpha value is simply so we can debug the buffer over the top of the viewport.\n    v_color = vec4(float(drawItemId), 0.0, 0.0, 1.0);\n  } else {\n    v_color = fetchTexel(instancesTexture, instancesTextureSize, (drawItemId * pixelsPerItem) + 4);\n  }\n\n  vec4 pos = positions;\n  if (pos.x < 0.0) pos.x = bboxMin.x;\n  else if (pos.x > 0.0) pos.x = bboxMax.x;\n  if (pos.y < 0.0) pos.y = bboxMin.y;\n  else if (pos.y > 0.0) pos.y = bboxMax.y;\n  if (pos.z < 0.0) pos.z = bboxMin.z;\n  else if (pos.z > 0.0) pos.z = bboxMax.z;\n\n  gl_Position = viewProjectionMatrix * pos;\n}\n";class BoundingBoxShader extends GLShader{constructor(A){super(A,"BoundingBoxShader"),this.setShaderStage("VERTEX_SHADER",vert$h),this.setShaderStage("FRAGMENT_SHADER",frag$g)}}var WorkerFactory=createBase64WorkerFactory("/* rollup-plugin-web-worker-loader */
var worker_code = (function (exports) {
  'use strict';

  /* eslint-disable camelcase */
  const vec3_subtract = (vec1, vec2) => {
    return [vec1[0] - vec2[0], vec1[1] - vec2[1], vec1[2] - vec2[2]]
  };
  const vec3_length = (vec) => {
    return Math.sqrt(vec[0] * vec[0] + vec[1] * vec[1] + vec[2] * vec[2])
  };
  const vec2_scale = (vec, scl) => {
    return [vec[0] * scl, vec[1] * scl]
  };
  const vec2_length = (vec) => {
    return Math.sqrt(vec[0] * vec[0] + vec[1] * vec[1])
  };

  const quat_conjugate = (quat) => {
    return [-quat[0], -quat[1], -quat[2], quat[3]]
  };
  const quat_multiply = (quat1, quat2) => {
    const ax = quat1[0];
    const ay = quat1[1];
    const az = quat1[2];
    const aw = quat1[3];
    const bx = quat2[0];
    const by = quat2[1];
    const bz = quat2[2];
    const bw = quat2[3];

    return [
      ax * bw + aw * bx + ay * bz - az * by,
      ay * bw + aw * by + az * bx - ax * bz,
      az * bw + aw * bz + ax * by - ay * bx,
      aw * bw - ax * bx - ay * by - az * bz,
    ]
  };
  const quat_rotateVec3 = (quat, vec3) => {
    const vq = [vec3[0], vec3[1], vec3[2], 0.0];
    const pq = quat_multiply(quat_multiply(quat, vq), quat_conjugate(quat));
    return [pq[0], pq[1], pq[2]]
  };

  // ///////////////////////////////////////////////
  // Frustum Culling data.
  let enableOcclusionCulling = false;
  const geomItemsData = [];
  const outOfFrustum = [];
  let frustumCulledCount = 0;
  let newlyCulled = [];
  let newlyUnCulled = [];

  let visibleCount = 0;
  const totalGeomStats = {
    triangles: 0,
    lines: 0,
    points: 0,
  };
  const visibleGeomStats = {
    triangles: 0,
    lines: 0,
    points: 0,
  };
  const geomStats_addTotal = (geomStats) => {
    // console.log('geomStats_add:', geomStats.triangles, visibleGeomStats.triangles)
    totalGeomStats.triangles += geomStats.triangles;
    totalGeomStats.lines += geomStats.lines;
    totalGeomStats.points += geomStats.points;
  };
  const geomStats_subtractTotal = (geomStats) => {
    // console.log('geomStats_subtract:', geomStats.triangles, visibleGeomStats.triangles)
    totalGeomStats.triangles -= geomStats.triangles;
    totalGeomStats.lines -= geomStats.lines;
    totalGeomStats.points -= geomStats.points;
  };
  const geomStats_add = (geomStats) => {
    // console.log('geomStats_add:', geomStats.triangles, visibleGeomStats.triangles)
    visibleCount++;
    visibleGeomStats.triangles += geomStats.triangles;
    visibleGeomStats.lines += geomStats.lines;
    visibleGeomStats.points += geomStats.points;
  };
  const geomStats_subtract = (geomStats) => {
    // console.log('geomStats_subtract:', geomStats.triangles, visibleGeomStats.triangles)
    visibleCount--;
    visibleGeomStats.triangles -= geomStats.triangles;
    visibleGeomStats.lines -= geomStats.lines;
    visibleGeomStats.points -= geomStats.points;
  };

  let cameraPos;
  let cameraInvOri;
  let isOrthographic = false;
  let frustumHeight = 0;
  let frustumWidth = 0;
  let frustumHalfAngleX = 0;
  let frustumHalfAngleY = 0;
  let solidAngleLimit = 0.004;
  let placeholderLoadThreshold = 200;

  const cull = (index) => {
    if (!outOfFrustum[index]) {
      outOfFrustum[index] = true;
      frustumCulledCount++;
      newlyCulled.push(index);

      // We could be culling something that was already
      // not visible in the occlusion buffer.
      if (!enableOcclusionCulling || !occluded[index]) {
        geomStats_subtract(geomItemsData[index].geomStats);
      }
    }
  };
  const unCull = (index) => {
    if (outOfFrustum[index]) {
      outOfFrustum[index] = false;
      frustumCulledCount--;
      newlyUnCulled.push(index);

      // Occlusion culling can only determine if something is visible
      // meaning that we assume it is not, until it shows up in the occlusion buffer.
      // Once it appears in the occlusion buffer, we start rendering it again.
      // So, when an item comes back into the frustum, we start rendering its bbox.
      // Then if the bbox is seen, we then start showing it.
      if (enableOcclusionCulling && !geomItemsData[index].transparent) {
        occluded[index] = true;
      } else {
        geomStats_add(geomItemsData[index].geomStats);
      }
    }
  };

  const checkGeomItem = (geomItemData) => {
    if (!geomItemData || !cameraPos) return
    if (!geomItemData.visible) {
      return
    }

    // Some items, like Handles and the grid, or or the VR controllers that should not be culled.
    if (!geomItemData.cullable) {
      unCull(geomItemData.id);
      return
    }
    const boundingRadius = geomItemData.boundingRadius;

    if (isOrthographic) {
      // Cull very small items
      // Note: when in VR, the FoV becomes very wide and the pixel
      // height varies. It seems more consistent to just use solidAngle
      // which is resolution invariant.
      const vheight = boundingRadius / frustumHeight;
      if (solidAngleLimit > 0 && vheight < solidAngleLimit) {
        cull(geomItemData.id);
        return
      }

      // Now we check if the item is within the view frustum.
      // We need the solid angle of the item for each axis (X & Y)
      // This is because at the corners of the screen, the object is slightly
      // further away, so the solid angle calculated above gets smaller.
      // This was causing items with big bounding spheres to be culled too early
      // at the corner of the screen.
      const vec = vec3_subtract(geomItemData.pos, cameraPos);
      const viewPos = quat_rotateVec3(cameraInvOri, vec);
      if (
        Math.abs(viewPos[0]) - boundingRadius > frustumWidth * 0.5 ||
        Math.abs(viewPos[1]) - boundingRadius > frustumHeight * 0.5
      ) {
        cull(geomItemData.id);
        return
      }
    } else {
      const vec = vec3_subtract(geomItemData.pos, cameraPos);
      const dist = vec3_length(vec);
      // unCull items close to the view.
      if (dist < boundingRadius) {
        unCull(geomItemData.id);
        return
      }
      // Cull very small items
      // Note: when in VR, the FoV becomes very wide and the pixel
      // height varies. It seems more consistent to just use solidAngle
      // which is resolution invariant.
      const solidAngle = Math.asin(boundingRadius / dist);
      if (solidAngleLimit > 0 && solidAngle < solidAngleLimit) {
        cull(geomItemData.id);
        return
      }

      // Now we check if the item is within the view frustum.
      // We need the solid angle of the item for each axis (X & Y)
      // This is because at the corners of the screen, the object is slightly
      // further away, so the solid angle calculated above gets smaller.
      // This was causing items with big bounding spheres to be culled too early
      // at the corner of the screen.
      const viewPos = quat_rotateVec3(cameraInvOri, vec);
      const viewVecXZ = [viewPos[0], viewPos[2]];
      const viewVecYZ = [viewPos[1], viewPos[2]];
      const distX = vec2_length(viewVecXZ);
      const distY = vec2_length(viewVecYZ);
      const solidAngleXZ = Math.asin(boundingRadius / distX);
      const solidAngleYZ = Math.asin(boundingRadius / distY);
      const viewVecNormXZ = vec2_scale(viewVecXZ, 1 / distX);
      const viewVecNormYZ = vec2_scale(viewVecYZ, 1 / distY);

      let viewAngle;
      // If an item is behind the viewer
      if (viewPos[2] > 0) {
        viewAngle = [
          Math.PI - Math.abs(Math.asin(viewVecNormXZ[0])) - solidAngleXZ,
          Math.PI - Math.abs(Math.asin(viewVecNormYZ[0])) - solidAngleYZ,
        ];
      } else {
        viewAngle = [
          Math.abs(Math.asin(viewVecNormXZ[0])) - solidAngleXZ,
          Math.abs(Math.asin(viewVecNormYZ[0])) - solidAngleYZ,
        ];
      }
      // console.log(geomItemData.id, 'angle To Item:', frustumHalfAngleX, viewAngle[0], frustumHalfAngleY, viewAngle[1])
      if (viewAngle[0] > frustumHalfAngleX || viewAngle[1] > frustumHalfAngleY) {
        cull(geomItemData.id);
        return
      }
    }

    unCull(geomItemData.id);
  };

  const onViewPortChanged = (data, postMessage) => {
    if (data.isOrthographic) {
      isOrthographic = true;
      frustumHeight = data.frustumHeight;
      frustumWidth = data.frustumWidth;
    } else {
      isOrthographic = false;
      frustumHalfAngleX = data.frustumHalfAngleX;
      frustumHalfAngleY = data.frustumHalfAngleY;
    }

    solidAngleLimit = data.solidAngleLimit;
    if (cameraPos && cameraInvOri) {
      geomItemsData.forEach(checkGeomItem);
      onDoneFrustumCull(postMessage);
    }
  };

  const onViewChanged = (data, postMessage) => {
    cameraPos = data.cameraPos;
    cameraInvOri = quat_conjugate(data.cameraOri);
    solidAngleLimit = data.solidAngleLimit;
    if (geomItemsData.length > 0) {
      geomItemsData.forEach(checkGeomItem);
      onDoneFrustumCull(postMessage);
    }
  };

  let inFrustumDrawIdsBufferPopulated = false;
  const generateInFrustumIndices = () => {
    let offset = 0;
    outOfFrustum.forEach((value, index) => {
      if (index > 0 && !value && geomItemsData[index].visible && !geomItemsData[index].transparent) offset++;
    });
    // Create a float array that can be used as an instances
    // attribute to pass into the drawing of the bounding boxes.
    const inFrustumIndices = new Float32Array(offset);
    offset = 0;
    outOfFrustum.forEach((value, index) => {
      if (index > 0 && !value && geomItemsData[index].visible && !geomItemsData[index].transparent) {
        inFrustumIndices[offset] = index;
        offset++;
      }
    });
    return inFrustumIndices
  };

  const onDoneFrustumCull = (postMessage) => {
    if (!enableOcclusionCulling) {
      const countInFrustum = geomItemsData.length - 1 - frustumCulledCount;

      postMessage({
        type: 'CullResults',
        newlyCulled,
        newlyUnCulled,
        visible: countInFrustum,
        total: geomItemsData.length - 1,
        visibleGeomStats,
        totalGeomStats,
      });
    } else {
      // console.log('FrustumCullResults:', 'newlyCulled:', newlyCulled, 'newlyUnCulled:', newlyUnCulled, outOfFrustum)
      // const countInFrustum = geomItemsData.length - 1 - frustumCulledCount

      // if (countInFrustum > 300) {
      //   console.log('countInFrustum:', countInFrustum)
      // }
      if (newlyCulled.length > 0 || newlyUnCulled.length > 0 || !inFrustumDrawIdsBufferPopulated) {
        const inFrustumIndices = generateInFrustumIndices();

        // When occlusion culling is on, we only uncull items after they
        // are detected in the occlusion buffer. Transparent items are not
        // rendered to the occlusion buffer, so must be unculled immediately.
        const newlyUnCulled_transparent = [];
        newlyUnCulled.forEach((index) => {
          if (index > 0 && geomItemsData[index] && geomItemsData[index].visible && geomItemsData[index].transparent) {
            newlyUnCulled_transparent.push(index);
          }
        });
        postMessage(
          {
            type: 'InFrustumIndices',
            newlyCulled,
            newlyUnCulled: newlyUnCulled_transparent,
            /* When occlusion culling is running,we only consider
            culling completed after the occlusion culling is done.
            visible: visibleCount,
            total: geomItemsData.length - 1,
            visibleGeomStats,
            totalGeomStats,*/
            inFrustumIndices,
          },
          [inFrustumIndices.buffer]
        );
        inFrustumDrawIdsBufferPopulated = true;
      } else {
        // Note: the inFrustumDrawIdsBuffer is already up to date we can skip this.
        postMessage({
          type: 'InFrustumIndices',
          newlyCulled: [],
          newlyUnCulled: [],
          visible: visibleCount,
          total: geomItemsData.length - 1,
          visibleGeomStats,
          totalGeomStats,
        });
      }
    }

    newlyCulled = [];
    newlyUnCulled = [];
  };

  // ///////////////////////////////////////////////
  // Occlusion Culling data.
  const occluded = [];
  const processOcclusionData = (data) => {
    const reductionDataArray = data.reductionDataArray; // Uint8Array

    const newlyCulled = [];
    const newlyUnCulled = [];
    const visiblePlaceholders = [];
    for (let i = 0; i < reductionDataArray.length; i += 2) {
      const index = i / 2;
      const value = reductionDataArray[i];

      if (index == 0) continue
      if (index >= geomItemsData.length) {
        break
      }

      const geomItemData = geomItemsData[index];
      if (!geomItemData || !geomItemData.cullable || !geomItemData.visible || geomItemData.transparent) {
        continue
      }

      if (geomItemData.isPlaceholder && value > placeholderLoadThreshold) {
        const isNotBBox = reductionDataArray[i + 1] > 0;
        if (isNotBBox) {
          visiblePlaceholders.push({ index, count: value });
        }
      }

      if (!outOfFrustum[index]) {
        if (value == 0) {
          // Not transparent object can not be occlusion culled, because we do not render them to the
          // occlusion buffer. This means they cannot occlude, or be considered occluded.
          if (!occluded[index]) {
            occluded[index] = true;
            newlyCulled.push(index);
            geomStats_subtract(geomItemData.geomStats);
          }
        } else {
          if (occluded[index]) {
            occluded[index] = false;
            newlyUnCulled.push(index);
            geomStats_add(geomItemData.geomStats);
          }
        }
      }
    }
    // Note: even if occlusion culling did nt filter out any more items
    // we must emit this message to the mains worker so it knows that culling is completed.
    postMessage({
      type: 'CullResults',
      newlyCulled,
      newlyUnCulled,
      visible: visibleCount,
      total: geomItemsData.length - 1,
      visibleGeomStats,
      totalGeomStats,
      visiblePlaceholders,
    });
  };

  // ///////////////////////////////////////////////
  // Messaging
  const handleMessage = (data, postMessage) => {
    if (data.type == 'Init') {
      console.log('Initialized Culling Worker');
      enableOcclusionCulling = data.enableOcclusionCulling;
      placeholderLoadThreshold = data.placeholderLoadThreshold;
      postMessage({
        type: 'WorkerReady',
      });
    } else if (data.type == 'ViewportChanged') {
      onViewPortChanged(data, postMessage);
    } else if (data.type == 'ViewChanged') {
      onViewChanged(data, postMessage);
    } else if (data.type == 'UpdateGeomItems') {
      data.removedItemIndices.forEach((index) => {
        const geomItemData = geomItemsData[index];
        if (geomItemData && geomItemData.visible) {
          geomStats_subtractTotal(geomItemData.geomStats);
          if (!enableOcclusionCulling || !occluded[index]) {
            geomStats_subtract(geomItemData.geomStats);
          }
        }
        geomItemsData[index] = null;
        outOfFrustum[index] = true;
      });
      data.geomItems.forEach((geomItemData) => {
        const index = geomItemData.id;
        // New geoms default to being un-culled
        // Existing geoms that may be changing state, like changing
        // visibility or transformations should simply update.
        if (!geomItemsData[index]) {
          outOfFrustum[index] = false;
          if (geomItemData.visible) {
            geomStats_addTotal(geomItemData.geomStats);
            geomStats_add(geomItemData.geomStats);
          }
        } else {
          // are either adding a new item, or unhiding an existing item.
          const becomingVisible = !geomItemsData[index].visible && geomItemData.visible;
          const becomingInVisible = geomItemsData[index].visible && !geomItemData.visible;
          if (becomingVisible) {
            geomStats_addTotal(geomItemData.geomStats);
            outOfFrustum[index] = true;
            occluded[index] = false;
            // if (!outOfFrustum[index] && !occluded[index]) {
            //   geomStats_add(geomItemData.geomStats)
            // }
          } else if (becomingInVisible) {
            geomStats_subtractTotal(geomItemData.geomStats);
            if (!outOfFrustum[index] && (!enableOcclusionCulling || !occluded[index])) {
              geomStats_subtract(geomItemData.geomStats);
            }
            outOfFrustum[index] = true;
            occluded[index] = false;
          } else {
            const becomingTransparent = !geomItemsData[index].transparent && geomItemData.transparent;
            const becomingOpaque = geomItemsData[index].transparent && !geomItemData.transparent;
            if (becomingTransparent) {
              if (enableOcclusionCulling && !outOfFrustum[index] && occluded[index]) {
                // Items becoming transparent must be unculled immediately.
                newlyUnCulled.push(index);
                geomStats_add(geomItemData.geomStats);
              }
            } else if (becomingOpaque) {
              // console.log('Becoming opaque:', index)
              // Items becoming transparent must be unculled immediately.
              // Occlusion culling can only determine if something is visible
              // meaning that we assume it is not, until it shows up in the occlusion buffer.
              // Once it appears in the occlusion buffer, we start rendering it again.
              // So, when an item comes back into the frustum, we start rendering its bbox.
              // Then if the bbox is seen, we then start showing it.
              if (enableOcclusionCulling && !outOfFrustum[index]) {
                occluded[index] = true;
                geomStats_subtract(geomItemData.geomStats);
              }
            }
          }
        }
        geomItemsData[index] = geomItemData;
        checkGeomItem(geomItemsData[index]);
      });
      inFrustumDrawIdsBufferPopulated = false;
      onDoneFrustumCull(postMessage);
    } else if (data.type == 'OcclusionData') {
      processOcclusionData(data);
    }
  };

  self.onmessage = function (event) {
    handleMessage(event.data, self.postMessage);
  };

  exports.handleMessage = handleMessage;

  Object.defineProperty(exports, '__esModule', { value: true });

  return exports;

})({});

",null,!1);const pixelsPerItem$1=8;class AbortController{__abort=!1;abort(){this.__abort=!0}}function clientWaitAsync(A,g,I,C,e){return new Promise(((i,t)=>{const o=setInterval((()=>{if(e.__abort)return clearInterval(o),void i();const C=A.clientWaitSync(g,I,0);if(C==A.WAIT_FAILED)return clearInterval(o),void t();C!=A.TIMEOUT_EXPIRED&&(clearInterval(o),i())}),C)}))}async function getBufferSubDataAsync(A,g,I,C,e,i,t=0,o=0){const a=A.fenceSync(A.SYNC_GPU_COMMANDS_COMPLETE,0);return A.flush(),await clientWaitAsync(A,a,0,10,i),i.__abort||(A.bindBuffer(g,I),A.getBufferSubData(g,C,e,t,o),A.bindBuffer(g,null)),A.deleteSync(a),e}async function readPixelsAsync(A,g,I,C,e,i,t,o,a){const B=A.createBuffer();return A.bindBuffer(A.PIXEL_PACK_BUFFER,B),A.bufferData(A.PIXEL_PACK_BUFFER,o.byteLength,A.STREAM_READ),A.readPixels(g,I,C,e,i,t,0),A.bindBuffer(A.PIXEL_PACK_BUFFER,null),await getBufferSubDataAsync(A,A.PIXEL_PACK_BUFFER,B,0,o,a),A.deleteBuffer(B),o}class BBoxOcclusionLinesCuboid extends Lines{constructor(){super(),this.setNumVertices(8);const A=this.getVertexAttribute("positions");A.setValue(0,new Vec3(.5,-.5,.5)),A.setValue(1,new Vec3(.5,.5,.5)),A.setValue(2,new Vec3(-.5,.5,.5)),A.setValue(3,new Vec3(-.5,-.5,.5)),A.setValue(4,new Vec3(.5,-.5,-.5)),A.setValue(5,new Vec3(.5,.5,-.5)),A.setValue(6,new Vec3(-.5,.5,-.5)),A.setValue(7,new Vec3(-.5,-.5,-.5)),this.setNumSegments(34),this.setSegmentVertexIndices(0,0,1),this.setSegmentVertexIndices(1,0,2),this.setSegmentVertexIndices(2,0,3),this.setSegmentVertexIndices(3,0,4),this.setSegmentVertexIndices(4,0,5),this.setSegmentVertexIndices(5,0,6),this.setSegmentVertexIndices(6,0,7),this.setSegmentVertexIndices(7,1,0),this.setSegmentVertexIndices(8,1,2),this.setSegmentVertexIndices(9,1,3),this.setSegmentVertexIndices(10,1,4),this.setSegmentVertexIndices(11,1,5),this.setSegmentVertexIndices(12,1,6),this.setSegmentVertexIndices(13,1,7),this.setSegmentVertexIndices(14,2,0),this.setSegmentVertexIndices(15,2,1),this.setSegmentVertexIndices(16,2,3),this.setSegmentVertexIndices(17,2,4),this.setSegmentVertexIndices(18,2,5),this.setSegmentVertexIndices(19,2,6),this.setSegmentVertexIndices(20,2,7),this.setSegmentVertexIndices(21,3,1),this.setSegmentVertexIndices(22,3,2),this.setSegmentVertexIndices(23,3,0),this.setSegmentVertexIndices(24,3,4),this.setSegmentVertexIndices(25,3,5),this.setSegmentVertexIndices(26,3,6),this.setSegmentVertexIndices(27,3,7),this.setSegmentVertexIndices(28,4,5),this.setSegmentVertexIndices(29,4,6),this.setSegmentVertexIndices(30,4,7),this.setSegmentVertexIndices(31,5,6),this.setSegmentVertexIndices(32,5,7),this.setSegmentVertexIndices(33,6,7)}}class XRViewport extends GLBaseViewport{projectionMatricesUpdated;tick;stageTreeItem;stageScale;viewXfo=new Xfo;stageXfo=new Xfo;invStageXfo=new Xfo;invStageMatrix=new Mat4;region=[];session=null;refSpace;projectionMatrices=[];viewMatrices=[];cameraMatrices=[];sessionMode="immersive-vr";controllersMap={};controllers=[];controllerPointerDownTime=[];constructor(A,g){super(A),this.clickTime=300,this.doubleClickTime=300,this.sessionMode=g,this.stageTreeItem=new TreeItem("VRStage"),this.stageTreeItem.setVisible(!1),this.__renderer.addTreeItem(this.stageTreeItem);const I=new Xfo;I.ori.setFromAxisAndAngle(new Vec3(1,0,0),.5*Math.PI),this.setXfo(I)}getRenderer(){return this.renderer}getTreeItem(){return this.stageTreeItem}getXfo(){return this.stageXfo}setXfo(A){this.stageXfo=A,this.stageTreeItem.globalXfoParam.value=A,this.invStageXfo=A.inverse(),this.invStageMatrix=this.invStageXfo.toMat4(),this.stageScale=A.sc.x}getControllers(){return this.controllers}isPresenting(){return null!=this.session}startSession(){const A=(g,I)=>{this.session&&(this.session.requestAnimationFrame(A),this.drawXRFrame(I))};this.session.requestAnimationFrame(A)}stopPresenting(){this.session&&this.session.end()}togglePresenting(){this.session?this.stopPresenting():this.startPresenting()}bindCullingViewport(A,g,I){A.viewXfo=this.viewXfo,A.viewScale=1,A.region=this.region,A.cameraMatrix=A.viewXfo.toMat4();const C=A.cameraMatrix.inverse();A.viewport=this;const e=this.__renderer.gl;A.bindRendererUnifs=g=>{const{cameraMatrix:i,viewMatrix:t,projectionMatrix:o,eye:a,isOrthographic:B}=g;i&&e.uniformMatrix4fv(i.location,!1,A.cameraMatrix.asArray()),t&&e.uniformMatrix4fv(t.location,!1,C.asArray()),o&&e.uniformMatrix4fv(o.location,!1,I.asArray()),a&&e.uniform1i(a.location,0),B&&e.uniform1i(B.location,0)},A.bindViewports=(A,g)=>g()}bindXRViewport(A,g){A.viewXfo=this.viewXfo,A.viewScale=1,A.region=this.region,A.cameraMatrix=A.viewXfo.toMat4(),A.viewport=this;const I=this.__renderer.gl;A.bindRendererUnifs=g=>{const{cameraMatrix:C}=g;C&&I.uniformMatrix4fv(C.location,!1,A.cameraMatrix.asArray())},A.bindViewports=(A,C)=>{const{isOrthographic:e}=A;e&&I.uniform1i(e.location,0),g.forEach(((g,e)=>{let i=g.region;I.viewport(i[0],i[1],i[2],i[3]);const{viewMatrix:t,projectionMatrix:o,eye:a}=A;t&&I.uniformMatrix4fv(t.location,!1,g.viewMatrix.asArray()),o&&I.uniformMatrix4fv(o.location,!1,g.projectionMatrix.asArray()),a&&I.uniform1i(a.location,e),C()}))}}}class XRHead{xrvp;treeItem;mat4;localXfo;hmdGeomItem;constructor(A,g){this.xrvp=A,this.treeItem=new TreeItem("XRHead"),g.addChild(this.treeItem),this.mat4=new Mat4,this.localXfo=new Xfo}setVisible(A){if(A&&!this.hmdGeomItem){const A=this.xrvp.getAsset();if(!A)return;const g=A.getChildByName("HMD");if(!g)return;if(this.hmdGeomItem=g.clone({assetItem:A}),this.hmdGeomItem){const A=new Quat;A.setFromAxisAndAngle(new Vec3(0,1,0),Math.PI),this.hmdGeomItem.localXfoParam.value=new Xfo(new Vec3(0,-.035,-.03),A,new Vec3(.001,.001,.001)),this.treeItem.addChild(this.hmdGeomItem,!1)}}this.hmdGeomItem&&(this.hmdGeomItem.visibleParam.value=A)}update(A){this.mat4.fromArray(A.transform.matrix),this.localXfo.setFromMat4(this.mat4),this.treeItem.localXfoParam.value=this.localXfo}getTreeItem(){return this.treeItem}getXfo(){return this.localXfo}}class XRController extends EventEmitter{id;buttonPressed;xrvp;inputSource;pressedButtons=[];treeItem;tipItem;raycastTick=5;raycastArea=.005;raycastDist=.04;pointerRay=new Ray;mat4=new Mat4;xfo=new Xfo;raycastAreaCache=0;raycastDistCache=0;rayCastRenderTargetProjMatrix=new Mat4;tick;hitTested;pointerOverItem;intersectionData;constructor(A,g,I){if(super(),this.xrvp=A,this.inputSource=g,this.id=I,this.buttonPressed=!1,this.inputSource.gamepad.buttons.forEach(((A,g)=>{0!=g&&(this.pressedButtons[g]=A.pressed)})),this.treeItem=new TreeItem("XRController:"+g.handedness+I),!SystemDesc.isMobileDevice&&(this.tipItem=new TreeItem("Tip"),this.tipItem.localXfoParam.value.tr.set(0,-.05,-.13),this.tipItem.localXfoParam.value.ori.setFromAxisAndAngle(new Vec3(1,0,0),.8*Math.PI),this.treeItem.addChild(this.tipItem,!1),A.stageTreeItem.addChild(this.treeItem),"tracked-pointer"==g.targetRayMode)){switch(g.profiles[0]){case"htc-vive":localStorage.setItem("ZeaEngine_XRDevice","Vive");break;case"oculus-touch":case"oculus-touch-v2":case"oculus-touch-v3":localStorage.setItem("ZeaEngine_XRDevice","Oculus")}A.loadHMDResources().then((A=>{if(!A)return;const I=new Xfo;let C;if(I.ori.setFromAxisAndAngle(new Vec3(0,1,0),Math.PI),I.sc.set(.001,.001,.001),"htc-vive"==g.profiles[0])C=A.getChildByName("Controller"),I.tr.set(0,-.035,-.085);else switch(g.handedness){case"left":C=A.getChildByName("LeftController"),I.tr.set(0,-.035,-.085),I.sc.scaleInPlace(.85);break;case"right":C=A.getChildByName("RightController"),I.tr.set(0,-.035,-.085),I.sc.scaleInPlace(.85);break;default:C=A.getChildByName("Controller")}if(C){const g=new CloneContext;g.assetItem=A;const e=C.clone(g);e.localXfoParam.value=I,this.treeItem.addChild(e,!1)}}))}this.tick=0}getHandedness(){return this.inputSource.handedness}getId(){return this.id}getTreeItem(){return this.treeItem}getTipItem(){return this.tipItem}getTipXfo(){return this.tipItem.globalXfoParam.value}isButtonPressed(){return this.buttonPressed}getControllerStageLocalXfo(){return this.xfo}getControllerTipStageLocalXfo(){return this.xfo.multiply(this.tipItem.localXfoParam.value)}updatePose(A,g,I){const C=g.getPose(I.gripSpace,A);if(!C||!C.transform)return;this.mat4.fromArray(C.transform.matrix),this.xfo.setFromMat4(this.mat4),this.treeItem.localXfoParam.value=this.xfo;const e=this.tipItem.globalXfoParam.value;this.pointerRay.start=e.tr,this.pointerRay.dir=e.ori.getZaxis(),this.hitTested=!1;let i=!1;if(this.raycastDist>0&&this.raycastTick>0&&this.tick%this.raycastTick==0){const A=this.getGeomItemAtTip();if(i=!0,null!=A){const g=new XRControllerEvent(this.xrvp,this,0,this.buttonPressed?1:0);g.intersectionData=A,g.pointerRay=this.pointerRay,A.geomItem!=this.pointerOverItem&&(this.pointerOverItem&&(g.leftGeometry=this.pointerOverItem,this.pointerOverItem.onPointerLeave(g),g.propagating&&this.xrvp.emit("pointerLeaveGeom",g)),g.propagating=!0,this.pointerOverItem=A.geomItem,this.pointerOverItem.onPointerEnter(g),g.propagating&&this.xrvp.emit("pointerOverGeom",g)),A.geomItem.onPointerMove(g)}else if(this.pointerOverItem){const A=new XRControllerEvent(this.xrvp,this,0,this.buttonPressed?1:0);A.pointerRay=this.pointerRay,A.leftGeometry=this.pointerOverItem,this.pointerOverItem.onPointerLeave(A),this.pointerOverItem=null}}return this.inputSource.gamepad.buttons.forEach(((A,g)=>{if(0!=g)if(A.pressed&&!this.pressedButtons[g]){this.pressedButtons[g]=!0;const A=new XRControllerEvent(this.xrvp,this,g,1);A.intersectionData=this.getGeomItemAtTip(),A.pointerRay=this.pointerRay,this.emit("buttonPressed",A),i=!0}else if(!A.pressed&&this.pressedButtons[g]){this.pressedButtons[g]=!1;const A=new XRControllerEvent(this.xrvp,this,g,0);this.emit("buttonReleased",A)}})),this.tick++,i}getGeomItemAtTip(){if(this.hitTested)return this.intersectionData;if(this.hitTested=!0,0==this.raycastDist)return null;const A=this.raycastDist*this.xrvp.stageScale,g=this.raycastArea*this.xrvp.stageScale;A==this.raycastDistCache&&g==this.raycastAreaCache||(this.rayCastRenderTargetProjMatrix.setOrthographicMatrix(-.5*g,.5*g,-.5*g,.5*g,0,-A),this.raycastDistCache=A,this.raycastAreaCache=g);const I=this.xrvp.getRenderer(),C=this.tipItem.globalXfoParam.value.clone();return C.sc.set(1,1,1),this.intersectionData=I.raycastWithProjection(C,this.rayCastRenderTargetProjMatrix,this.pointerRay),this.intersectionData}}class VRController extends XRController{}class XRViewManipulator extends BaseTool{enableViewScale=!0;listenerIDs={};controllerTriggersHeld=[];xrvp;vrControllerToolTip;vrControllerToolTipMat;grabPos;stageXfo__GrabStart;grabDir;grabDist;constructor(A){super(),this.xrvp=A,this.vrControllerToolTip=new Cross(.05),this.vrControllerToolTipMat=new Material("VRController Cross","LinesShader"),this.vrControllerToolTipMat.getParameter("BaseColor").value=new Color("#03E3AC")}activateTool(){super.activateTool();const A=A=>{const g=new GeomItem("XRViewManipulatorToolTip",this.vrControllerToolTip,this.vrControllerToolTipMat);g.pickableParam.value=!1,A.getTipItem().removeAllChildren(),A.getTipItem().addChild(g,!1)};for(const g of this.xrvp.getControllers())A(g);this.listenerIDs.controllerAdded=this.xrvp.on("controllerAdded",(g=>{A(g.controller)}))}deactivateTool(){super.deactivateTool();const A=A=>{const g=A.getTipItem().getChildByName("XRViewManipulatorToolTip");A.getTipItem().removeChildByHandle(g)};for(const g of this.xrvp.getControllers())A(g);this.xrvp.off("controllerAdded",this.listenerIDs.controllerAdded)}initMoveStage(){if(1==this.controllerTriggersHeld.length)this.grabPos=this.controllerTriggersHeld[0].getControllerTipStageLocalXfo().tr.clone(),this.stageXfo__GrabStart=this.xrvp.getXfo().clone();else if(2==this.controllerTriggersHeld.length){const A=this.controllerTriggersHeld[0].getControllerTipStageLocalXfo().tr,g=this.controllerTriggersHeld[1].getControllerTipStageLocalXfo().tr;this.grabDir=g.subtract(A),this.grabPos=A.lerp(g,.5),this.grabDir.y=0,this.grabDist=this.grabDir.length(),this.grabDir.scaleInPlace(1/this.grabDist),this.stageXfo__GrabStart=this.xrvp.getXfo().clone()}}onXRControllerButtonDown(A){if(0!=A.button)return;-1==this.controllerTriggersHeld.indexOf(A.controller)&&(this.controllerTriggersHeld.push(A.controller),this.initMoveStage(),A.stopPropagation(),A.setCapture(this))}onXRControllerButtonUp(A){if(0!=A.button)return;const g=this.controllerTriggersHeld.indexOf(A.controller);-1!=g&&(this.controllerTriggersHeld.splice(g,1),this.initMoveStage(),A.stopPropagation(),0==this.controllerTriggersHeld.length&&A.releaseCapture())}onVRControllerDoubleClicked(A){console.log("onVRControllerDoubleClicked:",this.controllerTriggersHeld.length);const g=this.xrvp.getXfo().clone();g.sc.set(1,1,1),this.xrvp.setXfo(g)}onVRPoseChanged(A){if(1==this.controllerTriggersHeld.length){const A=this.controllerTriggersHeld[0].getControllerTipStageLocalXfo().tr,g=new Xfo;g.tr=this.grabPos.subtract(A);const I=this.stageXfo__GrabStart.multiply(g);this.xrvp.setXfo(I)}else if(2==this.controllerTriggersHeld.length){const A=this.controllerTriggersHeld[0].getControllerTipStageLocalXfo().tr,g=this.controllerTriggersHeld[1].getControllerTipStageLocalXfo().tr,I=A.lerp(g,.5),C=g.subtract(A);C.y=0;const e=C.length();if(e<1e-4)return;C.scaleInPlace(1/e);const i=new Xfo;let t=this.grabDir.angleTo(C);this.grabDir.cross(C).y>0&&(t=-t),i.ori.rotateY(t);const o=this.grabPos.subtract(I),a=i.ori.rotateVec3(this.grabPos);if(i.tr.addInPlace(this.grabPos.subtract(a)),this.enableViewScale){const A=Math.max(Math.min(this.grabDist/e,10),.1);i.sc.set(A,A,A);const g=this.grabPos.scale(1-A);i.tr.addInPlace(i.ori.rotateVec3(g)),o.scaleInPlace(A)}i.tr.addInPlace(i.ori.rotateVec3(o));const B=this.stageXfo__GrabStart.multiply(i);this.xrvp.setXfo(B)}}onPointerDown(A){A instanceof XRControllerEvent&&this.onXRControllerButtonDown(A)}onPointerMove(A){A instanceof XRPoseEvent&&this.onVRPoseChanged(A)}onPointerUp(A){A instanceof XRControllerEvent&&this.onXRControllerButtonUp(A)}onPointerDoubleClick(A){A instanceof XRControllerEvent&&this.onVRControllerDoubleClicked(A)}}class VRViewport extends XRViewport{vrAsset;hmd="";hmdAssetPromise;xrhead;constructor(A,g){super(A,g),this.xrhead=new XRHead(this,this.stageTreeItem),this.setManipulator(new XRViewManipulator(this))}getAsset(){return this.vrAsset}getVRHead(){return this.xrhead}getHMDName(){return this.hmd}loadHMDResources(){if(SystemDesc.isMobileDevice)return Promise.resolve(null);let A=localStorage.getItem("ZeaEngine_XRDevice");if(A||(A="Oculus",localStorage.setItem("ZeaEngine_XRDevice",A)),this.hmd!=A)this.hmdAssetPromise=void 0;else if(this.hmdAssetPromise)return this.hmdAssetPromise;return this.hmd=A,this.hmdAssetPromise=new Promise(((g,I)=>{{let I;switch(A){case"Vive":default:I="ZeaEngine/Vive.vla";break;case"Oculus":I="ZeaEngine/Oculus.vla"}if(!resourceLoader.commonResources[I]){const A=new VLAAsset(I);A.load(resourceLoader.systemUrls[I]),resourceLoader.commonResources[I]=A}this.vrAsset=resourceLoader.getCommonResource(I);const C=()=>{const A=this.vrAsset.getMaterialLibrary(),I=A.getMaterialNames();for(const g of I){const I=A.getMaterial(g);I&&I.setShaderName("SimpleSurfaceShader")}this.vrAsset.traverse((A=>{A.pickableParam.value=!1})),g(this.vrAsset)};this.vrAsset.isLoaded()?C():this.vrAsset.once("loaded",C)}})),this.hmdAssetPromise}startPresenting(){return new Promise(((A,g)=>{(()=>{navigator.xr.requestSession("immersive-vr",{requiredFeatures:["local-floor"],optionalFeatures:["bounded-floor"]}).then((I=>{const C=this.__renderer.getViewport();if(C){const A=C.getCamera().globalXfoParam.value,g=new Xfo;g.tr=A.tr.clone(),g.tr.z-=1.3;const I=A.ori.getZaxis();I.z=0,I.normalizeInPlace(),g.ori.setFromDirectionAndUpvector(I,new Vec3(0,0,1)),this.setXfo(g)}I.addEventListener("end",(()=>{this.stageTreeItem.setVisible(!1),this.session=null;const A=this.__renderer.gl;A.bindFramebuffer(A.FRAMEBUFFER,null),this.emit("presentingChanged",new StateChangedEvent(!1))}));const e=A=>{if(this.controllersMap[A.handedness])return void(this.controllersMap[A.handedness].inputSource=A);const g=this.controllers.length;console.log("creating controller:",A.handedness,A.profiles);const I=new XRController(this,A,g);this.controllersMap[A.handedness]=I,this.controllers[g]=I,this.controllerPointerDownTime[g]=[],I.on("buttonPressed",(A=>{this.onPointerDown(A)})),I.on("buttonReleased",(A=>{this.onPointerUp(A)}));const C=new ControllerAddedEvent(I);return this.emit("controllerAdded",C),I};I.addEventListener("selectstart",(A=>{const g=this.controllersMap[A.inputSource.handedness];g&&(g.buttonPressed=!0,this.onPointerDown(new XRControllerEvent(this,g,0,1)))})),I.addEventListener("selectend",(A=>{const g=this.controllersMap[A.inputSource.handedness];g&&(g.buttonPressed=!1,this.onPointerUp(new XRControllerEvent(this,g,0,0)))})),I.addEventListener("inputsourceschange",(A=>{for(const g of A.added)0!=g.profiles.length&&("generic-touchscreen"==g.profiles[0]||e(g))})),this.session=I;const i=new XRWebGLLayer(I,this.__gl);I.updateRenderState({baseLayer:i}),this.width=i.framebufferWidth,this.height=i.framebufferHeight,this.region=[0,0,this.width,this.height],this.depthRange=[I.renderState.depthNear,I.renderState.depthFar],this.resizeRenderTargets(this.width,this.height);const t=g=>{this.refSpace=g,this.stageTreeItem.setVisible(!0),this.emit("presentingChanged",new StateChangedEvent(!0)),this.loadHMDResources().then((()=>{this.startSession(),A()}))};I.requestReferenceSpace("bounded-floor").catch((A=>{console.warn(A.message),console.log("Falling back to local-floor reference space"),I.requestReferenceSpace("local-floor").then(t)})).then((A=>{A&&t(A)})).catch((A=>{console.warn(A.message),g(new Error("Unable to start XR Session:"+A.message))}))})).catch((A=>{console.warn(A.message)}))})()}))}updateControllers(A){let g=!1;const I=this.session.inputSources;for(let C=0;C<I.length;C++){const e=I[C];if(0==e.profiles.length)return;if(!this.controllers[C]){console.warn("Missing controller");continue}const i=this.controllers[C].updatePose(this.refSpace,A,e);g=g||i}return g}drawXRFrame(A){const g=A.session,I=g.renderState.baseLayer,C=A.getViewerPose(this.refSpace);if(!C)return;const e=C.views;if(!this.projectionMatricesUpdated){this.projectionMatrices=[],this.viewMatrices=[],this.cameraMatrices=[];for(let A=0;A<e.length;A++){const g=e[A],I=new Mat4;I.fromArray(g.projectionMatrix),this.projectionMatrices[A]=I,this.viewMatrices[A]=new Mat4,this.cameraMatrices[A]=new Mat4}this.projectionMatricesUpdated=!0}const i=this.__renderer.gl;this.depthRange=[g.renderState.depthNear,g.renderState.depthFar];const t=new ColorRenderState(this.__renderer.gl);t.pushGLStack("VRViewport.drawXRFrame"),t.boundRendertarget=I.framebuffer,t.region=this.region,t.depthRange=this.depthRange,t.viewport=this,t.xrviewport=this;const o=[];for(let A=0;A<e.length;A++){const g=e[A];this.viewMatrices[A].fromArray(g.transform.inverse.matrix),this.viewMatrices[A].multiplyInPlace(this.invStageMatrix);const C=I.getViewport(g);o.push({viewMatrix:this.viewMatrices[A],projectionMatrix:this.projectionMatrices[A],region:[C.x,C.y,C.width,C.height],isOrthographic:0})}this.xrhead.update(C);const a=this.xrhead.getTreeItem().globalXfoParam.value,B=a.toMat4();this.viewXfo=a,t.viewXfo=a,t.cameraMatrix=B,t.viewScale=this.stageScale,t.region=this.region,t.vrPresenting=!0,this.updateControllers(A);const s=new XRPoseEvent(this,this.viewXfo,this.controllers);s.getCapture()&&s.getCapture().onPointerMove(s),this.manipulator&&s.propagating&&this.manipulator.onPointerMove(s),this.bindXRViewport(t,o),i.bindFramebuffer(i.DRAW_FRAMEBUFFER,t.boundRendertarget);let l=this.backgroundColorParam.value.asArray();i.clearColor(l[0],l[1],l[2],l[3]),i.colorMask(!0,!0,!0,!0),i.clear(i.COLOR_BUFFER_BIT|i.DEPTH_BUFFER_BIT),this.draw(t),t.popGLStack(),0!=t.stack.length&&console.warn(" corrupt renderstate.stack.length:",t.stack.length);const n=new XRViewChangedEvent(t.viewXfo);n.hmd=this.hmd,n.controllers=this.controllers,n.viewport=this,n.xrviewport=this,this.emit("viewChanged",n),this.tick++}onPointerDown(A){A.intersectionData=A.controller.getGeomItemAtTip(),A.pointerRay=A.controller.pointerRay;const g=Date.now();if(g-this.controllerPointerDownTime[A.controller.id][A.button]<this.doubleClickTime){if(this.emit("pointerDoubleClick",A),!A.propagating)return;if(this.manipulator&&(this.manipulator.onPointerDoubleClick(A),!A.propagating))return}this.controllerPointerDownTime[A.controller.id][A.button]=g,A.getCapture()&&(A.getCapture().onPointerDown(A),!A.propagating)||(null==A.intersectionData||(A.intersectionData.geomItem.onPointerDown(A),A.propagating))&&(this.emit("pointerDown",A),A.propagating&&this.manipulator&&this.manipulator.onPointerDown(A))}onPointerUp(A){A.pointerRay=A.controller.pointerRay,A.intersectionData=A.controller.getGeomItemAtTip(),A.getCapture()&&A.getCapture().onPointerUp(A),A.propagating&&null!=A.intersectionData&&A.intersectionData.geomItem.onPointerUp(A),A.propagating&&this.emit("pointerUp",A),A.propagating&&this.manipulator&&this.manipulator.onPointerUp(A);const g=this.controllerPointerDownTime[A.controller.id][A.button];Date.now()-g<this.clickTime&&(A.propagating=!0,null!=A.intersectionData&&A.intersectionData.geomItem.onPointerClick(A),A.propagating&&this.emit("pointerClick",A),A.propagating&&this.manipulator&&this.manipulator.onPointerClick(A)),this.controllerPointerDownTime[A.controller.id][A.button]=0}}class ARViewport extends XRViewport{stageXfoSet=!1;xrViewerSpace;xrHitTestSource;reticle;constructor(A,g){super(A,g);const I=new Disc(.1,24),C=new FlatSurfaceMaterial("reticle");C.baseColorParam.value=new Color(.75,.75,.75,.4),this.reticle=new GeomItem("reticle",I,C);const e=new Xfo;e.ori.setFromAxisAndAngle(new Vec3(1,0,0),.5*Math.PI),this.reticle.geomOffsetXfoParam.value=e,this.stageTreeItem.addChild(this.reticle)}startPresenting(){return new Promise(((A,g)=>{(()=>{navigator.xr.requestSession("immersive-ar",{requiredFeatures:["local","hit-test","anchors"]}).then((g=>{g.addEventListener("end",(()=>{this.stageTreeItem.setVisible(!1),this.session=null,this.emit("presentingChanged",new StateChangedEvent(!1))})),this.reticle.setVisible(!0);g.addEventListener("select",(A=>{const g=A.frame.getHitTestResults(this.xrHitTestSource);if(g.length>0){const I=g[0].getPose(this.refSpace);if(this.reticle.isVisible())if(this.stageXfoSet){const I=new XRPointerEvent(this,this.reticle.globalXfoParam.value,A,g);this.emit("pointerPressed",I)}else this.stageXfo.tr=this.reticle.globalXfoParam.value.tr.negate(),this.setXfo(this.stageXfo),this.stageXfoSet=!0;else{const C=new Xfo;C.setFromMat4(new Mat4(...I.transform.matrix));const e=new XRPointerEvent(this,this.stageXfo.multiply(C),A,g);this.emit("pointerPressed",e)}}})),this.session=g;const I=new XRWebGLLayer(g,this.__gl);g.updateRenderState({baseLayer:I}),this.width=I.framebufferWidth,this.height=I.framebufferHeight,this.region=[0,0,this.width,this.height],this.depthRange=[g.renderState.depthNear,g.renderState.depthFar],this.resizeRenderTargets(this.width,this.height),g.requestReferenceSpace("viewer").then((A=>{this.xrViewerSpace=A,g.requestHitTestSource({space:this.xrViewerSpace}).then((A=>{this.xrHitTestSource=A}))})),g.requestReferenceSpace("local").then((g=>{this.refSpace=g,this.stageTreeItem.setVisible(!0),this.emit("presentingChanged",new StateChangedEvent(!0)),this.startSession(),A()}))})).catch((A=>{console.warn(A.message)}))})()}))}drawXRFrame(A){const g=A.session,I=g.renderState.baseLayer,C=A.getViewerPose(this.refSpace);if(!C)return;if(this.xrHitTestSource&&this.reticle.isVisible()){const g=A.getHitTestResults(this.xrHitTestSource);if(g.length>0){const A=g[0].getPose(this.refSpace),I=new Xfo;I.setFromMat4(new Mat4(A.transform.matrix)),this.reticle.localXfoParam.value=I}}const e=C.views;if(!this.projectionMatricesUpdated){this.projectionMatrices=[],this.viewMatrices=[],this.cameraMatrices=[];for(let A=0;A<e.length;A++){const g=e[A],I=new Mat4;I.fromArray(g.projectionMatrix),this.projectionMatrices[A]=I,this.viewMatrices[A]=new Mat4,this.cameraMatrices[A]=new Mat4}this.projectionMatricesUpdated=!0}this.depthRange=[g.renderState.depthNear,g.renderState.depthFar];const i=new ColorRenderState(this.__renderer.gl);i.boundRendertarget=I.framebuffer,i.region=this.region,i.viewport=this,i.xrviewport=this;const t=[];for(let A=0;A<e.length;A++){const g=e[A];this.viewMatrices[A].fromArray(g.transform.inverse.matrix),this.viewMatrices[A].multiplyInPlace(this.invStageMatrix);const C=I.getViewport(g);t.push({viewMatrix:this.viewMatrices[A],projectionMatrix:this.projectionMatrices[A],region:[C.x,C.y,C.width,C.height],isOrthographic:0})}this.viewXfo.setFromMat4(new Mat4(C.transform.matrix)),i.viewXfo=this.stageXfo.multiply(this.viewXfo),i.cameraMatrix=i.viewXfo.toMat4(),i.viewScale=this.stageScale,i.region=this.region,i.vrPresenting=!0,this.bindXRViewport(i,t),this.draw(i);{const A=new XRPoseEvent(this,this.viewXfo);A.getCapture()&&A.getCapture().onPointerMove(A),this.manipulator&&A.propagating&&this.manipulator.onPointerMove(A)}const o=new XRViewChangedEvent(i.viewXfo);o.viewport=this,o.xrviewport=this,this.emit("viewChanged",o),this.tick++}}class GLGeomItemLibrary extends EventEmitter{renderer;glGeomItems=[null];glGeomItemEventHandlers=[];glGeomItemsMap=new Map;glGeomItemsIndexFreeList=[];dirtyItemIndices=new Set;dirtyWorkerItemIndices=new Set;removedItemIndices=[];glGeomItemsTexture=null;enableFrustumCulling;xrViewport;xrPresenting=!1;xrFovY=0;xrProjectionMatrix=new Mat4;enableOcclusionCulling;debugOcclusionBuffer=!1;occlusionDataBuffer;occlusionImage;occlusionImageItem;reductionDataBuffer;reductionDataArray;bbox;reductionShader;boundingBoxShader;inFrustumIndicesCount=0;inFrustumDrawIdsBuffer;abortController=null;cameraMovementInProgress=!1;occlusionDataBufferSizeFactor=window.devicePixelRatio;occlusionDataBufferSizeFactorXR=2;placeholderLoadThreshold=5;timer_query_ext=null;worker;constructor(A,g){super(),this.renderer=A,this.enableFrustumCulling=g.enableFrustumCulling||g.enableOcclusionCulling,this.enableOcclusionCulling=g.enableOcclusionCulling,this.debugOcclusionBuffer=A.getViewport().debugOcclusionBuffer,this.enableFrustumCulling&&this.setupCullingWorker(A)}setupCullingWorker(A){this.worker=new WorkerFactory;let g=!1;this.worker.postMessage({type:"Init",enableOcclusionCulling:this.enableOcclusionCulling,placeholderLoadThreshold:this.placeholderLoadThreshold}),this.worker.onmessage=A=>{"WorkerReady"==A.data.type?(I(),g=!0):"InFrustumIndices"==A.data.type?this.enableOcclusionCulling?(A.data.newlyCulled&&this.applyCullResults(A.data),this.calculateOcclusionCulling(A.data.inFrustumIndices)):(this.applyCullResults(A.data),this.emitCullingUpdateData(A.data),g=!0):"CullResults"==A.data.type?(this.applyCullResults(A.data),this.emitCullingUpdateData(A.data),g=!0):"Done"==A.data.type&&this.renderer.emit("CullingUpdated"),g=!0};const I=()=>{const g=A.getViewport(),I=A.getViewport().getCamera(),C=g.getWidth()/g.getHeight();if(I.isOrthographic()){const g=I.getFrustumHeight(),e=g*C;this.worker.postMessage({type:"ViewportChanged",frustumHeight:g,frustumWidth:e,isOrthographic:!0,solidAngleLimit:A.solidAngleLimit})}else{const g=.5*I.getFov(),e=Math.atan(Math.tan(g)*C);this.worker.postMessage({type:"ViewportChanged",frustumHalfAngleX:e,frustumHalfAngleY:g,isOrthographic:!1,solidAngleLimit:A.solidAngleLimit})}};A.on("resized",I);const C=A.getViewport().getCamera();C.on("projectionParamChanged",(A=>{C.isOrthographic()&&I()})),A.once("xrViewportSetup",(g=>{this.xrViewport=g.xrViewport;const C=g.xrViewport;C.on("presentingChanged",(g=>{if(this.xrPresenting=g.state,g.state){let g,I;if(i=10,C instanceof VRViewport)g=MathFunctions.degToRad(62),I=MathFunctions.degToRad(50);else if(C instanceof ARViewport){const A=C.getWidth()/C.getHeight();I=62*MathFunctions.degToRad(.5),g=Math.atan(Math.tan(I)*A)}this.xrFovY=2*I;const e=I/g,t=C.depthRange[0],o=C.depthRange[1];this.xrProjectionMatrix.setPerspectiveMatrix(this.xrFovY,e,t,o),this.worker.postMessage({type:"ViewportChanged",frustumHalfAngleX:g,frustumHalfAngleY:I,isOrthographic:!1,solidAngleLimit:4*A.solidAngleLimit}),this.occlusionDataBuffer.resize(Math.ceil(C.getWidth()/this.occlusionDataBufferSizeFactorXR),Math.ceil(C.getWidth()/this.occlusionDataBufferSizeFactorXR))}else i=5,I(),t()}))}));let e=0,i=5;A.on("viewChanged",(I=>{if(this.cameraMovementInProgress=!0,g){if(e%i==0){g=!1;const C=I.viewXfo.tr,e=I.viewXfo.ori;this.worker.postMessage({type:"ViewChanged",cameraPos:C.asArray(),cameraOri:e.asArray(),solidAngleLimit:A.solidAngleLimit})}e++}}));const t=()=>{this.cameraMovementInProgress=!1;const g=A.getViewport().getCamera().globalXfoParam.value,I=g.tr,C=g.ori;this.worker.postMessage({type:"ViewChanged",cameraPos:I.asArray(),cameraOri:C.asArray(),solidAngleLimit:A.solidAngleLimit})};if(A.getViewport().getCamera().on("movementFinished",t),t(),this.enableOcclusionCulling){const A=this.renderer.gl,g=Math.ceil(this.renderer.getWidth()/this.occlusionDataBufferSizeFactor),I=Math.ceil(this.renderer.getHeight()/this.occlusionDataBufferSizeFactor);this.occlusionDataBuffer=new GLRenderTarget(A,{type:A.FLOAT,format:A.RGBA,minFilter:A.NEAREST,magFilter:A.NEAREST,width:g,height:I,depthType:A.UNSIGNED_INT,depthFormat:A.DEPTH_COMPONENT,depthInternalFormat:A.DEPTH_COMPONENT24}),this.renderer.on("resized",(A=>{this.xrPresenting||this.occlusionDataBuffer.resize(Math.ceil(A.width/this.occlusionDataBufferSizeFactor),Math.ceil(A.height/this.occlusionDataBufferSizeFactor))})),this.reductionDataBuffer=new GLRenderTarget(A,{type:A.UNSIGNED_BYTE,format:A.RG,internalFormat:A.RG8,minFilter:A.NEAREST,magFilter:A.NEAREST,width:1,height:1}),this.bbox=new GLLines(A,new BBoxOcclusionLinesCuboid),this.reductionShader=new ReductionShader(A),this.boundingBoxShader=new BoundingBoxShader(A),this.boundingBoxShader.compileForTarget("GLGeomItemLibrary",this.renderer.directives),this.inFrustumIndicesCount=0}}applyCullResults(A){A.newlyCulled&&A.newlyCulled.forEach((A=>{const g=this.glGeomItems[A];g&&g.setCulled(!0)})),A.newlyUnCulled&&A.newlyUnCulled.forEach((A=>{const g=this.glGeomItems[A];g&&g.setCulled(!1)})),A.visiblePlaceholders&&A.visiblePlaceholders.length>0&&!this.cameraMovementInProgress&&A.visiblePlaceholders.forEach((A=>{const g=this.glGeomItems[A.index]?.geomItem;g instanceof PlaceholderGeomItem&&!g.loading&&g.load()})),this.renderer.requestRedraw()}emitCullingUpdateData(A){this.renderer.emit("CullingUpdated",{culled:A.newlyCulled?.length,unCulled:A.newlyUnCulled?.length,visible:A.visible,total:A.total,visibleGeomStats:A.visibleGeomStats,totalGeomStats:A.totalGeomStats})}updateCulledDrawIDsBuffer(A){const g=this.renderer.gl;if(!g.floatTexturesSupported)return;this.inFrustumDrawIdsBuffer&&this.inFrustumIndicesCount!=A.length&&(g.deleteBuffer(this.inFrustumDrawIdsBuffer),this.inFrustumDrawIdsBuffer=null),this.inFrustumDrawIdsBuffer||(this.inFrustumDrawIdsBuffer=g.createBuffer(),g.bindBuffer(g.ARRAY_BUFFER,this.inFrustumDrawIdsBuffer)),g.bindBuffer(g.ARRAY_BUFFER,this.inFrustumDrawIdsBuffer),g.bufferData(g.ARRAY_BUFFER,A,g.STATIC_DRAW),this.inFrustumIndicesCount=A.length;const I=Math.max(4,MathFunctions.nextPow2(Math.round(Math.sqrt(this.glGeomItems.length)+.5)));this.reductionDataBuffer.width!=I&&(this.reductionDataBuffer.resize(I,I),this.reductionDataArray=new Uint8Array(I*I*2))}calculateOcclusionCulling(A){if(A&&A.length>0&&this.updateCulledDrawIDsBuffer(A),this.abortController&&this.abortController.abort(),this.abortController=new AbortController,0==this.inFrustumIndicesCount)return;const g=this.renderer.gl,I=new GeomDataRenderState(g);I.pushGLStack("calculateOcclusionCulling"),this.renderer.bindGLRenderer(I),I.directives=[...this.renderer.directives,"#define DRAW_GEOMDATA"],I.floatGeomBuffer=!0,I.occlusionCulling=1,this.xrPresenting?this.xrViewport.bindCullingViewport(I,this.xrFovY,this.xrProjectionMatrix):this.renderer.getViewport().bindGLViewport(I);const C=this.occlusionDataBuffer.width*this.occlusionDataBuffer.height,e=this.timer_query_ext,i=(A,I,i)=>{this.reductionDataBuffer.bindForWriting(A,I),g.enable(g.BLEND),g.blendEquation(g.FUNC_ADD),g.blendFunc(g.ONE,g.ONE),this.reductionShader.bind(A);const{geomDataTexture:t,reductionTextureWidth:o}=A.unifs;t&&this.occlusionDataBuffer.bindToUniform(A,t),o&&g.uniform1i(o.location,this.reductionDataBuffer.width),e&&g.beginQuery(e.TIME_ELAPSED_EXT,i),g.drawArrays(g.POINTS,0,C),e&&g.endQuery(e.TIME_ELAPSED_EXT),g.disable(g.BLEND),this.reductionDataBuffer.unbindForWriting(A)};let t,o,a,B;e&&(t=g.createQuery(),g.beginQuery(e.TIME_ELAPSED_EXT,t)),(A=>{this.occlusionDataBuffer.bindForWriting(A,!0),A.glDisable(g.BLEND),A.glDisable(g.CULL_FACE),A.glEnable(g.DEPTH_TEST),g.depthFunc(g.LESS),g.depthMask(!0);this.renderer.getPass(0).drawGeomData(A);this.renderer.getPass(1).drawGeomData(A),this.occlusionDataBuffer.unbindForWriting(A)})(I),e&&g.endQuery(e.TIME_ELAPSED_EXT),e&&(o=g.createQuery()),i(I,!0,o),e&&(a=g.createQuery(),g.beginQuery(e.TIME_ELAPSED_EXT,a)),(()=>{this.occlusionDataBuffer.bindForWriting(I,!1),this.debugOcclusionBuffer||(g.colorMask(!0,!0,!0,!0),g.clearColor(0,0,0,0),g.clear(g.COLOR_BUFFER_BIT)),this.boundingBoxShader.bind(I,"GLGeomItemLibrary"),this.bbox.bind(I);const{instancesTexture:A,instancesTextureSize:C,instancedDraw:e,reductionDataTexture:i,occlusionCulling:t}=I.unifs;this.glGeomItemsTexture.bindToUniform(I,A),g.uniform1i(C.location,this.glGeomItemsTexture.width),g.uniform1i(e.location,1),this.reductionDataBuffer.bindColorTexture(I,i);const o=I.attrs.instancedIds.location;g.enableVertexAttribArray(o),g.bindBuffer(g.ARRAY_BUFFER,this.inFrustumDrawIdsBuffer),g.vertexAttribPointer(o,1,g.FLOAT,!1,4,0),g.vertexAttribDivisor(o,1),I.bindViewports(I.unifs,(()=>{this.bbox.drawInstanced(I,this.inFrustumIndicesCount)})),this.occlusionDataBuffer.unbindForWriting(I)})(),e&&g.endQuery(e.TIME_ELAPSED_EXT),e&&(B=g.createQuery()),i(I,!1,B),I.popGLStack();const s={numReductionPoints:C},l=(A,I)=>{const C=g.getQueryParameter(I,g.QUERY_RESULT_AVAILABLE),i=g.getParameter(e.GPU_DISJOINT_EXT);if(C&&!i){const C=g.getQueryParameter(I,g.QUERY_RESULT);s[A]=C/1e6,g.deleteQuery(I)}},n=this.reductionDataBuffer.width,d=this.reductionDataBuffer.height,r=g.RG,c=g.UNSIGNED_BYTE;this.reductionDataBuffer.bindForReading();const h=this.abortController;readPixelsAsync(g,0,0,n,d,r,c,this.reductionDataArray,h).then((()=>{this.reductionDataBuffer.unbindForReading(),h.__abort||(e&&(l("queryDrawScene",t),l("queryDrawCulledBBoxes",a),l("queryReduceSceneGeoms",o),l("queryReduceBBoxes",B),this.renderer.emit("occlusionCullingProfilingData",s)),this.worker.postMessage({type:"OcclusionData",reductionDataArray:this.reductionDataArray}),this.abortController=null)}))}addGeomItem(A){let g=this.glGeomItemsMap.get(A);if(null!=g)return this.glGeomItems[g];this.glGeomItemsIndexFreeList.length>0?g=this.glGeomItemsIndexFreeList.pop():(g=this.glGeomItems.length,this.glGeomItems.push(null)),this.removedItemIndices.includes(g)&&this.removedItemIndices.splice(this.removedItemIndices.indexOf(g),1),this.dirtyItemIndices.add(g);const I=A.materialParam;let C=I.value;const e=this.renderer.glMaterialLibrary.addMaterial(C),i=()=>{this.renderer.glMaterialLibrary.removeMaterial(C),C=I.value,a.materialId=C?this.renderer.glMaterialLibrary.addMaterial(C):-1,s(),B()};I.on("valueChanged",i);const t=this.renderer.gl,o=C.getShaderClass().supportsInstancing(),a=new GLGeomItem(t,A,g,-1,e,o),B=()=>{if(this.dirtyItemIndices.has(g))return;this.dirtyItemIndices.add(g);const I=A.isPickable();this.renderer.drawItemChanged(I)};A.geomMatParam.on("valueChanged",B),A.on("cutAwayChanged",B),A.on("highlightChanged",B),A.on("selectabilityChanged",B),A.on("opacityChanged",B),A.on("pickabilityChanged",B);const s=()=>{if(this.enableFrustumCulling&&!this.dirtyWorkerItemIndices.has(g)){this.dirtyWorkerItemIndices.add(g);const I=A.isPickable();this.renderer.drawItemChanged(I)}};return this.enableFrustumCulling&&this.dirtyWorkerItemIndices.add(g),A.on("visibilityChanged",s),A.on("opacityChanged",s),A.geomMatParam.on("valueChanged",s),A.geomParam.on("boundingBoxChanged",s),this.glGeomItems[g]=a,this.glGeomItemEventHandlers[g]={geomItemChanged:B,materialChanged:i,workerItemDataChanged:s},this.glGeomItemsMap.set(A,g),this.renderer.requestRedraw(),a}removeGeomItem(A){const g=this.glGeomItemsMap.get(A);if(null==g)return null;const I=this.glGeomItems[g],C=A.materialParam.value;this.renderer.glMaterialLibrary.removeMaterial(C);const e=this.glGeomItemEventHandlers[g];return A.materialParam.off("valueChanged",e.materialChanged),A.geomMatParam.off("valueChanged",e.geomItemChanged),A.off("cutAwayChanged",e.geomItemChanged),A.off("highlightChanged",e.geomItemChanged),A.off("selectabilityChanged",e.geomItemChanged),A.off("opacityChanged",e.geomItemChanged),A.off("pickabilityChanged",e.geomItemChanged),A.off("visibilityChanged",e.workerItemDataChanged),A.geomMatParam.off("valueChanged",e.workerItemDataChanged),A.off("opacityChanged",e.workerItemDataChanged),A.geomParam.off("boundingBoxChanged",e.workerItemDataChanged),this.glGeomItems[g]=null,this.glGeomItemEventHandlers[g]=null,this.glGeomItemsIndexFreeList.push(g),this.glGeomItemsMap.delete(A),this.removedItemIndices.push(g),this.dirtyWorkerItemIndices.has(g)&&this.dirtyWorkerItemIndices.delete(g),this.renderer.requestRedraw(),I}getGeomItem(A){if(!(A>=this.glGeomItems.length))return this.glGeomItems[A]?.geomItem;console.warn("Invalid Draw Item id:"+A+" NumItems:"+(this.glGeomItems.length-1))}getGLGeomItem(A){const g=this.glGeomItemsMap.get(A);return null!=g?this.glGeomItems[g]:null}populateDrawItemDataArray(A,g,I){const C=this.glGeomItems[A];if(!C)return;const{geomItem:e,geomId:i}=C,t=e.materialParam.value,o=32*g;let a=0;e.isCutawayEnabled()&&(a|=GLGeomItemFlags.GEOMITEM_FLAG_CUTAWAY),e.isPickable()||(a|=GLGeomItemFlags.GEOMITEM_INVISIBLE_IN_GEOMDATA),t.isOpaque()&&e.isOpaque()||(a|=GLGeomItemFlags.GEOMITEM_TRANSPARENT);const B=new Float32Array(I.buffer,4*(o+0),4);B[0]=a,B[1]=e.opacity;const s=this.renderer.glMaterialLibrary.getMaterialAllocation(t);s&&(B[2]=s.start),B[4]=i;const l=e.geomMatParam.value,n=new Float32Array(I.buffer,4*(o+4),4),d=new Float32Array(I.buffer,4*(o+8),4),r=new Float32Array(I.buffer,4*(o+12),4);n.set([l.xAxis.x,l.yAxis.x,l.zAxis.x,l.translation.x]),d.set([l.xAxis.y,l.yAxis.y,l.zAxis.y,l.translation.y]),r.set([l.xAxis.z,l.yAxis.z,l.zAxis.z,l.translation.z]);const c=new Float32Array(I.buffer,4*(o+16),4);if(e.isHighlighted()){const A=e.getHighlight();c.set([A.r,A.g,A.b,A.a])}const h=new Float32Array(I.buffer,4*(o+20),4);if(e.isCutawayEnabled()){const A=e.getCutVector(),g=e.getCutDist();h.set([A.x,A.y,A.z,g])}const G=e.boundingBoxParam.value,m=new Float32Array(I.buffer,4*(o+24)),b=new Float32Array(I.buffer,4*(o+28));m.set([G.p0.x,G.p0.y,G.p0.z,0]),b.set([G.p1.x,G.p1.y,G.p1.z,0])}getCullingWorkerData(A,g){const I=A.boundingBoxParam.value,C=.5*I.size(),e=I.center(),i=A.materialParam.value,t=A.cullable&&!A.isOverlay()&&!i.getShaderClass().isOverlay(),o=!A.isOpaque()||!i.isOpaque(),a=A instanceof PlaceholderGeomItem,B={triangles:0,lines:0,points:0},s=A.geomParam.value;if(s instanceof CompoundGeom)B.triangles+=s.getNumTriangles(),B.lines+=s.getNumLineSegments(),B.points+=s.getNumPoints();else if(s instanceof Mesh||s instanceof MeshProxy)B.triangles+=s.getNumTriangles();else if(s instanceof Lines||s instanceof LinesProxy)B.lines+=s.getNumLineSegments();else{if(!(s instanceof Points||s instanceof PointsProxy))throw new Error("Unsupported geom type:"+s.constructor.name);B.points+=s.getNumVertices()}return{id:g,boundingRadius:C,pos:e.asArray(),cullable:t,visible:A.isVisible(),transparent:o,isPlaceholder:a,geomStats:B}}uploadGeomItemsToWorker(){if(this.enableFrustumCulling){const A=[];this.dirtyWorkerItemIndices.forEach((g=>{const I=this.glGeomItems[g];if(!I)return;const{geomItem:C}=I,e=this.getCullingWorkerData(C,g);e&&A.push(e)})),this.worker.postMessage({type:"UpdateGeomItems",geomItems:A,removedItemIndices:this.removedItemIndices}),this.dirtyWorkerItemIndices.clear(),this.removedItemIndices=[]}}uploadGeomItems(A){const g=this.renderer.gl;if(!g.floatTexturesSupported)return;let I=Math.round(Math.sqrt(8*this.glGeomItems.length)+.5);I=MathFunctions.nextPow2(I),I%8!=0&&(I+=8-I%8),this.glGeomItemsTexture?this.glGeomItemsTexture.width!=I&&(this.glGeomItemsTexture.resize(I,I),this.dirtyItemIndices=new Set(Array(I*I/8).fill(0).map(((A,g)=>g)))):(this.glGeomItemsTexture=new GLTexture2D(g,{format:"RGBA",type:"FLOAT",width:I,height:I,filter:"NEAREST",wrap:"CLAMP_TO_EDGE",mipMapped:!1}),this.glGeomItemsTexture.clear()),g.bindTexture(g.TEXTURE_2D,this.glGeomItemsTexture.glTex);const C=this.glGeomItemsTexture.type,e=Array.from(this.dirtyItemIndices);for(let A=0;A<e.length;A++){const i=e[A],t=Math.floor(8*i/I);let o=i+1;for(let g=A+1;g<e.length;g++){const A=e[g];if(Math.floor(8*A/I)!=t)break;if(A!=o)break;o++}const a=o-i,B=8*i%I,s=8*a,l=1,n=new Float32Array(32*a);for(let A=i;A<o;A++)this.populateDrawItemDataArray(A,A-i,n);if(C==g.FLOAT)this.glGeomItemsTexture.populate(n,s,l,B,t,!1);else{const A=MathFunctions.convertFloat32ArrayToUInt16Array(n);this.glGeomItemsTexture.populate(A,s,l,B,t,!1)}A+=a-1}this.removedItemIndices=[],this.dirtyItemIndices=new Set}bind(A){(this.dirtyWorkerItemIndices.size>0||this.removedItemIndices.length>0)&&this.uploadGeomItemsToWorker(),this.dirtyItemIndices.size>0&&this.uploadGeomItems(A);const g=this.renderer.gl,{instancesTexture:I,instancesTextureSize:C}=A.unifs;I&&(this.glGeomItemsTexture.bindToUniform(A,I),g.uniform1i(C.location,this.glGeomItemsTexture.width))}}var frag$f="\nprecision highp float;\n#define GLSLIFY 1\n\nuniform float outlineThickness;\nuniform sampler2D highlightDataTexture;\n\nvarying vec2 v_texCoord;\n\nfloat M_PI = 3.141592653589793;\nfloat diff(vec4 pixelA, vec4 pixelB)\n{\n  return abs(pixelA.r - pixelB.r) + abs(pixelA.g - pixelB.g) + abs(pixelA.b - pixelB.b) + abs(pixelA.a - pixelB.a);\n}\n\n// find the first pixel which is not the same as the center pixel.\nvec4 RadialSearch(vec2 uv)\n{\n  vec2 texSize = vec2(textureSize(highlightDataTexture, 0));\n  ivec2 pixelCenterCoord = ivec2(uv * texSize);\n  vec4 pixelCenter = texelFetch(highlightDataTexture, pixelCenterCoord, 0);\n\n  vec3 offset = vec3((1.0 / texSize.x), (1.0 / texSize.y), 0.0);\n\n  vec4 result = pixelCenter;\n  float weights = 1.0;\n  float radius = 0.0;\n  int differentPixels = 0;\n  while (radius <= outlineThickness + 0.001) {\n    radius += 1.0;\n    \n    int samples = int(2.0 * M_PI * radius);\n    for (int i =0; i<samples; i++) {\n      float theta = (float(i) / float(samples)) * 2.0 * M_PI;\n      vec2 dir = vec2(radius * cos(theta) * offset.x, radius * sin(theta) * offset.y);\n      \n      ivec2 pixelCoord = ivec2((uv + dir) * texSize);\n      vec4 pixel = texelFetch(highlightDataTexture, pixelCoord, 0);\n      if ((pixel.r > 0.0 || pixel.g > 0.0 || pixel.b > 0.0)) {\n        if (diff(pixel, pixelCenter) > 0.1) differentPixels++;\n        // Blend the outer ring of pixels.\n        // Note: disabled because I ran out of time. We can blend off the highlight towards\n        // the edges to get a nicely anti-aliazed outline. \n        // float dist = length(vec2(pixelCoord) - vec2(pixelCenterCoord));\n        // float blendStart = max(1.0, outlineThickness - 0.5);\n        pixel.a = 1.0; //smoothstep(1.0, 0.0, dist - blendStart);\n\n        result += pixel;\n        weights += pixel.a;\n      }\n    }\n  }\n\n  // Note: at the boundary between 2 highlighted objects, we get a nice blending effect. \n  if (weights > 1.0) {\n    result = result / weights;\n  }\n\n  // If all the pixels found are the same as the center pixel, we just\n  // return the center pixel.\n  if (differentPixels == 0) {\n    return pixelCenter;\n  }\n\n  return result;\n}\n\n#ifdef ENABLE_ES3\nout vec4 fragColor;\n#endif\n\nvoid main(void) {\n    \n  vec4 outlineColor = RadialSearch(v_texCoord);\n  \n  if (outlineColor.a > 0.0001) {\n#ifndef ENABLE_ES3\n    gl_FragColor = outlineColor;\n#else\n    fragColor = outlineColor;\n#endif\n  }\n  else {\n    discard;\n  }\n}\n\n",vert$g="\nprecision highp float;\n#define GLSLIFY 1\n\nattribute vec3 positions;    //(location = 0)\n\n/* VS Outputs */\nvarying vec2 v_texCoord;\n \nvoid main()\n{\n  v_texCoord = positions.xy+0.5;\n  gl_Position = vec4(positions.xy*2.0, 0.0, 1.0);\n}\n\n";class HighlightsShader extends GLShader{constructor(A){super(A,"HighlightsShader"),this.setShaderStage("VERTEX_SHADER",vert$g),this.setShaderStage("FRAGMENT_SHADER",frag$f)}}var frag$e="\n#ifndef ENABLE_ES3\n#extension GL_EXT_frag_depth: enable\n#endif\nprecision highp float;\n#define GLSLIFY 1\n\nuniform sampler2D colorTexture;\nuniform sampler2D depthTexture;\nuniform vec2 screenSize;\nuniform vec2 depthRange;\n\nuniform float outlineThickness;\nuniform color outlineColor;\nuniform float outlineSensitivity;\nuniform float outlineDepthBias;\n\nvarying vec2 v_texCoord;\n\n// http://web.archive.org/web/20130416194336/http://olivers.posterous.com/linear-depth-in-glsl-for-real\nfloat LinearEyeDepth(float z_b) {\n  float z_n = 2.0 * z_b - 1.0;\n  float z_e = 2.0 * depthRange.x * depthRange.y / (depthRange.y + depthRange.x - z_n * (depthRange.y - depthRange.x));\n  return z_e;\n}\n\nfloat LogEyeDepth(float z_b) {\n  return depthRange.x + ((depthRange.y - depthRange.x) * z_b);\n}\n\n// https://www.vertexfragment.com/ramblings/unity-postprocessing-sobel-outline/#depth-based-outline\n// https://github.com/ssell/UnitySobelOutline/blob/2e1f4a5b4e703ae2c96aaf08d5518ce58abbaab9/Assets/Resources/Shaders/SobelOutlineHLSL.shader\n\nfloat SobelDepth(float ldc, float ldl, float ldr, float ldu, float ldd)\n{\n  return abs(ldl - ldc) +\n      abs(ldr - ldc) +\n      abs(ldu - ldc) +\n      abs(ldd - ldc);\n}\n\nfloat SobelSampleDepth(vec2 uv, vec3 offset)\n{\n  float pixelCenter = LinearEyeDepth(texture2D(depthTexture, uv).r);\n  float pixelLeft   = LinearEyeDepth(texture2D(depthTexture, uv - offset.xz).r);\n  float pixelRight  = LinearEyeDepth(texture2D(depthTexture, uv + offset.xz).r);\n  float pixelUp     = LinearEyeDepth(texture2D(depthTexture, uv + offset.zy).r);\n  float pixelDown   = LinearEyeDepth(texture2D(depthTexture, uv - offset.zy).r);\n\n  float  outlineDepthMultiplier = (1.0 / pixelCenter) * outlineSensitivity;\n\n  return SobelDepth(pixelCenter, pixelLeft, pixelRight, pixelUp, pixelDown) * outlineDepthMultiplier;\n}\n\n#ifdef ENABLE_ES3\n  out vec4 fragColor;\n#endif\nvoid main(void) {\n#ifndef ENABLE_ES3\n  vec4 fragColor;\n#endif\n\n  vec3 offset = vec3((1.0 / screenSize.x), (1.0 / screenSize.y), 0.0) * outlineThickness;\n  float sobelDepth = SobelSampleDepth(v_texCoord, offset);\n  float sobelValue = pow(sobelDepth, outlineDepthBias);\n\n  float minEdgeValue = 0.25;\n  float maxEdgeValue = 0.55;\n  sobelDepth = smoothstep(minEdgeValue, maxEdgeValue, sobelValue);\n\n#ifdef ENABLE_ES3\n  fragColor = vec4(outlineColor.rgb, sobelDepth);\n#else\n  fragColor = vec4(mix(texture2D(colorTexture, v_texCoord).rgb, outlineColor.rgb, sobelDepth), 1.0);\n#ifdef  GL_EXT_frag_depth\n  gl_FragDepthEXT = texture2D(depthTexture, v_texCoord).r;\n#endif\n#endif\n\n  // float z = texture2D(depthTexture, v_texCoord).r;\n  // float near = depthRange.x * 2.0;    // the near plane\n  // float far = depthRange.y / 2.0;     // the far plane\n  // float c = (2.0 * near) / (far + near - z * (far - near));  // convert to linear values \n  // fragColor = vec4(vec3(c), 1.0);\n\n  \n\n#ifndef ENABLE_ES3\n  gl_FragColor = fragColor;\n#endif\n}\n",vert$f="\nprecision highp float;\n#define GLSLIFY 1\nattribute vec3 positions;    //(location = 0)\n\n/* VS Outputs */\nvarying vec2 v_texCoord;\n \nvoid main()\n{\n  v_texCoord = positions.xy+0.5;\n  gl_Position = vec4(positions.xy*2.0, 0.0, 1.0);\n}\n";class SilhouetteShader extends GLShader{constructor(A){super(A,"SilhouetteShader"),this.setShaderStage("VERTEX_SHADER",vert$f),this.setShaderStage("FRAGMENT_SHADER",frag$e)}}var freeGlobal="object"==typeof global&&global&&global.Object===Object&&global,freeGlobal$1=freeGlobal,freeSelf="object"==typeof self&&self&&self.Object===Object&&self,root=freeGlobal$1||freeSelf||Function("return this")(),root$1=root,Symbol$1=root$1.Symbol,Symbol$2=Symbol$1,objectProto$1=Object.prototype,hasOwnProperty=objectProto$1.hasOwnProperty,nativeObjectToString$1=objectProto$1.toString,symToStringTag$1=Symbol$2?Symbol$2.toStringTag:void 0;function getRawTag(A){var g=hasOwnProperty.call(A,symToStringTag$1),I=A[symToStringTag$1];try{A[symToStringTag$1]=void 0;var C=!0}catch(A){}var e=nativeObjectToString$1.call(A);return C&&(g?A[symToStringTag$1]=I:delete A[symToStringTag$1]),e}var objectProto=Object.prototype,nativeObjectToString=objectProto.toString;function objectToString(A){return nativeObjectToString.call(A)}var nullTag="[object Null]",undefinedTag="[object Undefined]",symToStringTag=Symbol$2?Symbol$2.toStringTag:void 0;function baseGetTag(A){return null==A?void 0===A?undefinedTag:nullTag:symToStringTag&&symToStringTag in Object(A)?getRawTag(A):objectToString(A)}function isObjectLike(A){return null!=A&&"object"==typeof A}var symbolTag="[object Symbol]";function isSymbol(A){return"symbol"==typeof A||isObjectLike(A)&&baseGetTag(A)==symbolTag}var reWhitespace=/\s/;function trimmedEndIndex(A){for(var g=A.length;g--&&reWhitespace.test(A.charAt(g)););return g}var reTrimStart=/^\s+/;function baseTrim(A){return A?A.slice(0,trimmedEndIndex(A)+1).replace(reTrimStart,""):A}function isObject(A){var g=typeof A;return null!=A&&("object"==g||"function"==g)}var NAN=NaN,reIsBadHex=/^[-+]0x[0-9a-f]+$/i,reIsBinary=/^0b[01]+$/i,reIsOctal=/^0o[0-7]+$/i,freeParseInt=parseInt;function toNumber(A){if("number"==typeof A)return A;if(isSymbol(A))return NAN;if(isObject(A)){var g="function"==typeof A.valueOf?A.valueOf():A;A=isObject(g)?g+"":g}if("string"!=typeof A)return 0===A?A:+A;A=baseTrim(A);var I=reIsBinary.test(A);return I||reIsOctal.test(A)?freeParseInt(A.slice(2),I?2:8):reIsBadHex.test(A)?NAN:+A}var now=function(){return root$1.Date.now()},now$1=now,FUNC_ERROR_TEXT$1="Expected a function",nativeMax=Math.max,nativeMin=Math.min;function debounce(A,g,I){var C,e,i,t,o,a,B=0,s=!1,l=!1,n=!0;if("function"!=typeof A)throw new TypeError(FUNC_ERROR_TEXT$1);function d(g){var I=C,i=e;return C=e=void 0,B=g,t=A.apply(i,I)}function r(A){var I=A-a;return void 0===a||I>=g||I<0||l&&A-B>=i}function c(){var A=now$1();if(r(A))return h(A);o=setTimeout(c,function(A){var I=g-(A-a);return l?nativeMin(I,i-(A-B)):I}(A))}function h(A){return o=void 0,n&&C?d(A):(C=e=void 0,t)}function G(){var A=now$1(),I=r(A);if(C=arguments,e=this,a=A,I){if(void 0===o)return function(A){return B=A,o=setTimeout(c,g),s?d(A):t}(a);if(l)return clearTimeout(o),o=setTimeout(c,g),d(a)}return void 0===o&&(o=setTimeout(c,g)),t}return g=toNumber(g)||0,isObject(I)&&(s=!!I.leading,i=(l="maxWait"in I)?nativeMax(toNumber(I.maxWait)||0,g):i,n="trailing"in I?!!I.trailing:n),G.cancel=function(){void 0!==o&&clearTimeout(o),B=0,C=a=e=o=void 0},G.flush=function(){return void 0===o?t:h(now$1())},G}var FUNC_ERROR_TEXT="Expected a function";function throttle(A,g,I){var C=!0,e=!0;if("function"!=typeof A)throw new TypeError(FUNC_ERROR_TEXT);return isObject(I)&&(C="leading"in I?!!I.leading:C,e="trailing"in I?!!I.trailing:e),debounce(A,g,{leading:C,maxWait:g,trailing:e})}let activeGLRenderer,pointerIsDown=!1,pointerLeft=!1;const registeredPasses=[];let requestPostAnimationFrame;requestPostAnimationFrame=globalThis.requestPostAnimationFrame?window.requestPostAnimationFrame:function(A){requestAnimationFrame((()=>{setTimeout(A,0)}))};class GLBaseRenderer extends ParameterOwner{#d=new Map;directives=[];directivesHash;solidAngleLimit=.004;__gl;glcanvas=null;#r=null;#c={};#h=!1;#G={};#m={};#b=new Map;#Z=[];#u=[];#y=void 0;#E=!1;#Q=!1;#V=0;#R=0;#W=0;floatGeomBuffer=!0;multiSampledScreenBuffer=!1;xrViewportPresenting=!1;xrViewport=void 0;#p;glMaterialLibrary;glGeomItemLibrary;glGeomLibrary;highlightsShader;silhouetteShader;screenQuad=null;#S;#F;constructor(A,g={}){if(super(),!SystemDesc.gpuDesc)throw new Error("Unable to create renderer. WebGL not Supported");this.requestRedraw=this.requestRedraw.bind(this),this.__gl=this.setupWebGL(A,g);const I=this.__gl;this.highlightsShader=new HighlightsShader(I),this.silhouetteShader=new SilhouetteShader(I),this.screenQuad=new GLScreenQuad(this.__gl,this.directives),this.bindEventHandlers();const C=this.addViewport("main");C.debugGeomDataBuffer=g.debugGeomDataBuffer,C.debugOcclusionBuffer=g.debugOcclusionBuffer,this.glMaterialLibrary=new GLMaterialLibrary(this),this.glMaterialLibrary.on("updated",(()=>{this.requestRedraw()})),this.glGeomLibrary=new GLGeomLibrary(this),this.glGeomLibrary.on("updated",(()=>{this.requestRedraw()})),this.glGeomItemLibrary=new GLGeomItemLibrary(this,g),this.glGeomItemLibrary.on("updated",(()=>{this.requestRedraw()})),registeredPasses.forEach((A=>{const g=new A.cls;this.addPass(g,A.passType)})),this.#p=new Promise(((A,I)=>{if((g.supportXR??1)&&navigator.xr){const C="AR"==(g.xrMode??"VR")?"immersive-ar":"immersive-vr",e=()=>{this.xrViewport=this.setupXRViewport(C);let g=new XrViewportEvent(this.xrViewport);this.emit("xrViewportSetup",g),A(this.xrViewport)};navigator.xr.isSessionSupported(C).then((A=>{A&&e()})).catch((A=>{console.warn("Unable to setup XR:"+A),I("Unable to setup XR:"+A)}))}}))}setShaderPreprocessorDirective(A,g){this.#c[A]=g;const I=[];let C="";for(const A in this.#c){const g=this.#c[A];I.push(g),C+=g}this.directives=I,this.directivesHash=""+StringFunctions.hashStr(C)}getWidth(){return this.glcanvas.width}getHeight(){return this.glcanvas.height}addViewport(A){const g=new GLViewport(this,A,this.getWidth(),this.getHeight());return g.on("updated",(()=>{this.requestRedraw()})),g.on("viewChanged",(A=>{this.xrViewportPresenting||this.emit("viewChanged",A)})),this.#u.push(g),this.#y=g,g}getViewport(A=0){return this.#u[A]}getViewportAtPos(A,g){for(const I of this.#u){const C=I.getPosX(),e=I.getPosY(),i=I.getWidth(),t=I.getHeight();if(A>=C&&g>=e&&A<=i+C&&g<=t+e)return I}}activateViewport(A){this.#y!=A&&(this.#y=A)}activateViewportAtPos(A,g){if(this.xrViewportPresenting)return;const I=this.getViewportAtPos(A,g);I&&I!=this.#y&&this.activateViewport(I)}getActiveViewport(){return this.#y}suspendDrawing(){this.#V++}resumeDrawing(){this.#V--,0==this.#V&&(this.renderGeomDataFbos(),this.requestRedraw())}renderGeomDataFbos(){if(1==this.#h)return;this.#h=!0;requestPostAnimationFrame((()=>{for(const A of this.#u)A.invalidateGeomDataBuffer();this.#h=!1}))}getScene(){return this.#r}setScene(A){this.#r=A,this.addTreeItem(this.#r.getRoot());let g=new SceneSetEvent(this.#r);this.emit("sceneSet",g)}addTreeItem(A){if(!(A instanceof TreeItem))return;const g={};let I=!0;if(A instanceof GeomItem){const C=A.geomParam;if(null==C.value){const I=()=>{delete g["Geometry.valueChanged"],this.assignTreeItemToGLPass(A)};g["Geometry.valueChanged"]=C.once("valueChanged",I)}else I=this.assignTreeItemToGLPass(A)}else I=this.assignTreeItemToGLPass(A);if(I){for(const g of A.getChildren())g&&this.addTreeItem(g);g.childAdded=A.on("childAdded",(A=>{this.addTreeItem(A.childItem)})),g.childRemoved=A.on("childRemoved",(A=>{this.removeTreeItem(A.childItem)}))}g.visibilityChanged=A.on("visibilityChanged",(()=>{this.renderGeomDataFbos()})),g.pickabilityChanged=A.on("pickabilityChanged",(()=>{this.renderGeomDataFbos()})),this.#d.set(A,g),this.renderGeomDataFbos()}assignTreeItemToGLPass(A){if(A instanceof GeomItem){const g=A;this.glGeomItemLibrary.addGeomItem(g)}for(let g=this.#Z.length-1;g>=0;g--){const I=this.#Z[g],C={continueInSubTree:!0};if(I.itemAddedToScene(A,C))return this.#b.set(A,I),C.continueInSubTree}return!0}removeTreeItem(A){if(!(A instanceof TreeItem))return;const g=this.#d.get(A);this.#d.delete(A),A.off("visibilityChanged",g.visibilityChanged),A.off("pickabilityChanged",g.pickabilityChanged);const I=this.#b.get(A);if(null!=I){const g={continueInSubTree:!0};I.itemRemovedFromScene(A,g),this.#b.delete(A)}if(null!=g.childAdded&&null!=g.childRemoved){A.off("childAdded",g.childAdded),A.off("childRemoved",g.childRemoved);for(const g of A.getChildren())g&&this.removeTreeItem(g)}if(A instanceof GeomItem){const I=A;if(g["Geometry.valueChanged"]){A.geomParam.off("valueChanged",g["Geometry.valueChanged"])}this.glGeomItemLibrary.removeGeomItem(I)}this.renderGeomDataFbos()}get gl(){return this.__gl}getGL(){return this.__gl}handleResize(A,g){if(this.xrViewportPresenting)return;const I=Math.round(Math.max(4,A)*window.devicePixelRatio),C=Math.round(Math.max(4,g)*window.devicePixelRatio);if(this.glcanvas.width!=I||this.glcanvas.height!=C){this.glcanvas.width=I,this.glcanvas.height=C,this.#u.forEach((A=>{A.resize(I,C)}));const A=new ResizedEvent(I,C);this.emit("resized",A),this.requestRedraw()}}getDiv(){return this.glcanvas.parentElement}setupWebGL(A,g={}){const{tagName:I}=A;if(!["DIV","CANVAS"].includes(I))throw new Error("Only CANVAS and DIV are valid root elements.");const C="DIV"===I;this.glcanvas=A,C?(console.warn("@GLBaseRenderer#setupWebGL.","Using a DIV as root element is deprecated.","Use a CANVAS instead.","See: https://docs.zea.live/zea-engine/#/getting-started/get-started-with-engine?id=basic-setup"),this.glcanvas=document.createElement("canvas"),A.appendChild(this.glcanvas)):this.glcanvas=A,this.glcanvas.style["touch-action"]="none",this.glcanvas.style["user-select"]="none",this.glcanvas.style["-webkit-user-select"]="none",this.glcanvas.style["-webkit-touch-callout"]="none",this.glcanvas.parentElement.style.position="relative",this.glcanvas.parentElement.style.overflow="hidden",this.glcanvas.style.width="100%",this.glcanvas.style.height="100%",this.glcanvas.style.position="absolute";const e=throttle((A=>{if(this.__gl)if(this.__gl.isContextLost())console.warn("WebGL Context Lost");else if(Array.isArray(A)&&A.length)for(const g of A){if(!g.contentRect)return;const A=Math.round(g.contentRect.width),I=Math.round(g.contentRect.height);this.handleResize(A,I)}}),500);this.#F=()=>{const A=[{contentRect:{width:this.glcanvas.parentElement.clientWidth,height:this.glcanvas.parentElement.clientHeight}}];e(A)},window.addEventListener("resize",this.#F),this.#S=new ResizeObserver(e);try{this.#S.observe(this.glcanvas.parentNode,{box:"device-pixel-content-box"})}catch(A){this.#S.observe(this.glcanvas.parentNode,{box:"content-box"})}this.handleResize(this.glcanvas.parentElement.clientWidth,this.glcanvas.parentElement.clientHeight);const i="macOS"===SystemDesc.OS&&"Chrome"===SystemDesc.browserName,t=SystemDesc.isIOSDevice,o={preserveDrawingBuffer:!0};o.antialias=!i&&!t&&(g.antialias??!0),o.depth=!0,o.stencil=!0,o.alpha=g.alpha??!0,o.premultipliedAlpha=g.premultipliedAlpha??!1,(navigator.xr||g.supportXR)&&(o.xrCompatible=!0),this.multiSampledScreenBuffer=o.antialias,o.powerPreference=g.powerPreference||"high-performance";const a=create3DContext(this.glcanvas,o);a||alert("Unable to create WebGL context. WebGL not supported."),this.setShaderPreprocessorDirective("ENABLE_ES3","#define ENABLE_ES3"),a.floatTexturesSupported&&this.setShaderPreprocessorDirective("ENABLE_FLOAT_TEXTURES","#define ENABLE_FLOAT_TEXTURES");{const A=a.getExtension("WEBGL_multi_draw");A&&!g.disableMultiDraw?(a.multiDrawArrays=A.multiDrawArraysWEBGL.bind(A),a.multiDrawElements=A.multiDrawElementsWEBGL.bind(A),a.multiDrawElementsInstanced=A.multiDrawElementsInstancedWEBGL.bind(A),a.multiDrawArraysInstanced=A.multiDrawArraysInstancedWEBGL.bind(A)):this.setShaderPreprocessorDirective("EMULATE_MULTI_DRAW","#define EMULATE_MULTI_DRAW")}return"Safari"==SystemDesc.browserName&&"webgl"==a.name?this.floatGeomBuffer=!1:this.floatGeomBuffer=null!=g.floatGeomBuffer?g.floatGeomBuffer:a.floatTexturesSupported,a.floatGeomBuffer=this.floatGeomBuffer,a}bindEventHandlers(){const A=()=>this.getWidth()>0&&this.getHeight(),g=A=>!!SystemDesc.isMobileDevice;this.glcanvas.addEventListener("mousedown",(A=>{if(g())return;const I=new ZeaMouseEvent(A,this.glcanvas.getBoundingClientRect());pointerIsDown=!0,activeGLRenderer=this,this.activateViewportAtPos(I.rendererX,I.rendererY);const C=this.getActiveViewport();C&&C.onPointerDown(I),pointerLeft=!1})),document.addEventListener("mouseup",(I=>{if(g())return;if(activeGLRenderer!=this||!A())return;const C=new ZeaMouseEvent(I,this.glcanvas.getBoundingClientRect());pointerIsDown=!1;const e=this.getActiveViewport();e&&e.onPointerUp(C),pointerLeft&&(e&&e.onPointerLeave(C),activeGLRenderer=void 0)})),document.addEventListener("mousemove",(I=>{if(g())return;if(activeGLRenderer!=this||!A())return;const C=new ZeaMouseEvent(I,this.glcanvas.getBoundingClientRect());pointerIsDown||this.activateViewportAtPos(C.rendererX,C.rendererY);const e=this.getActiveViewport();e&&e.onPointerMove(C)})),this.glcanvas.addEventListener("mouseenter",(A=>{if(!g()&&!pointerIsDown){activeGLRenderer=this;const g=new ZeaMouseEvent(A,this.glcanvas.getBoundingClientRect());if(this.activateViewportAtPos(g.rendererX,g.rendererY),!pointerIsDown){const A=this.getActiveViewport();A&&A.onPointerEnter(g)}pointerLeft=!1}})),this.glcanvas.addEventListener("mouseleave",(I=>{if(g())return;if(activeGLRenderer!=this||!A())return;const C=new ZeaMouseEvent(I,this.glcanvas.getBoundingClientRect());if(pointerIsDown)pointerLeft=!0;else{const A=this.getActiveViewport();A&&A.onPointerLeave(C),activeGLRenderer=void 0}})),document.addEventListener("contextmenu",(g=>{activeGLRenderer==this&&A()&&(g.preventDefault(),g.stopPropagation())})),this.glcanvas.addEventListener("touchstart",(A=>{activeGLRenderer=this;const g=this.getActiveViewport(),I=new ZeaTouchEvent(A,this.glcanvas.getBoundingClientRect());g.onPointerDown(I)}),{passive:!0}),this.glcanvas.addEventListener("touchend",(A=>{const g=this.getActiveViewport(),I=new ZeaTouchEvent(A,this.glcanvas.getBoundingClientRect());g.onPointerUp(I)}),{passive:!0}),this.glcanvas.addEventListener("touchmove",(A=>{const g=this.getActiveViewport(),I=new ZeaTouchEvent(A,this.glcanvas.getBoundingClientRect());g.onPointerMove(I)}),{passive:!0}),this.glcanvas.addEventListener("touchcancel",(A=>{const g=this.getActiveViewport(),I=new ZeaTouchEvent(A,this.glcanvas.getBoundingClientRect());g.onTouchCancel(I)}),{passive:!0});window.addEventListener("wheel",(g=>{if(activeGLRenderer==this&&A()&&activeGLRenderer){const A=new ZeaWheelEvent(g,this.glcanvas.getBoundingClientRect()),I=activeGLRenderer.getActiveViewport();I&&I.onWheel(A)}}),{passive:!1}),document.addEventListener("keydown",(g=>{if(activeGLRenderer!=this||!A())return;const I=new ZeaKeyboardEvent(g),C=activeGLRenderer.getActiveViewport();C&&C.onKeyDown(I)})),document.addEventListener("keyup",(g=>{if(activeGLRenderer!=this||!A())return;const I=new ZeaKeyboardEvent(g),C=activeGLRenderer.getActiveViewport();C&&C.onKeyUp(I)}))}getGLCanvas(){return this.glcanvas}frameAll(A=0,g=0,I=.1){this.#u[A].frameView([this.#r.getRoot()],g,I)}getOrCreateShader(A){let g=this.#G[A];return g||(g=Registry.constructClass(A),g||console.error("@GLBaseRenderer#getOrCreateShader - Shader not registered with the Registry:",A),g.setGLContext(this.__gl),this.#G[A]=g),g}addPass(A,g=-1){-1==g&&(g=A.getPassType()),this.#m[g]||(this.#m[g]=[]);let I=0;for(const A in this.#m){if(A==g.toString())break;I+=this.#m[A].length}I+=this.#m[g].length,A.on("updated",(A=>{this.requestRedraw(),this.renderGeomDataFbos()})),A.init(this,I),this.#m[g].push(A);let C=0;for(const A in this.#m){const g=this.#m[A];g.forEach(((A,g)=>{A.setPassIndex(C+g)})),C+=g.length}return this.#Z.push(A),this.requestRedraw(),I}getPass(A){let g=0;for(const I in this.#m){const C=this.#m[I];if(A-g<C.length)return C[A-g];g+=C.length}}findPassIndex(A){let g=0;for(const I in this.#m){const C=this.#m[I];for(let I=0;I<C.length;I++){if(C[I]instanceof A)return g;g++}}return-1}setupXRViewport(A){const g="immersive-ar"==A?new ARViewport(this,A):new VRViewport(this,A),I=A=>{this.emit("viewChanged",A)};return g.on("presentingChanged",(A=>{const C=A.state;if(this.xrViewportPresenting!=C)if(this.xrViewportPresenting=C,C){for(const A in this.#m){const g=this.#m[A];for(const A of g)A.startPresenting()}g.on("viewChanged",I)}else{g.off("viewChanged",I),this.emit("updated");for(const A in this.#m){const g=this.#m[A];for(const A of g)A.stopPresenting()}const A=this.getViewport().getCamera().globalXfoParam.value,C=new ViewChangedEvent("CameraAndPointer",A);this.emit("viewChanged",C),this.requestRedraw()}})),g}getVRViewport(){return this.xrViewport}getXRViewport(){return this.#p}isXRViewportPresenting(){return this.xrViewportPresenting}isContinuouslyDrawing(){return this.#E}startContinuousDrawing(){if(this.isContinuouslyDrawing()||this.xrViewportPresenting)return;const A=()=>{const g=new ColorRenderState(this.gl);this.#E&&!this.xrViewportPresenting&&window.requestAnimationFrame(A);for(const A of this.#u)A.draw(g)};this.#E=!0,requestPostAnimationFrame(A)}stopContinuousDrawing(){this.#E=!1}toggleContinuousDrawing(){this.#E?this.stopContinuousDrawing():this.startContinuousDrawing()}drawItemChanged(A){if(A)for(const A of this.#u)A.invalidateGeomDataBuffer();this.requestRedraw()}requestRedraw(){if(!this.__gl||this.#Q||this.#E||this.xrViewportPresenting||this.#V>0)return!1;if(this.__gl.isContextLost())return console.warn("WebGL Context Lost"),!1;const A=()=>{this.#Q=!1;const A=performance.now(),g=new ColorRenderState(this.gl);for(const A of this.#u)A.draw(g);0!=g.stack.length&&console.warn(" corrupt renderstate.stack.length:",g.stack.length);const I=performance.now();this.#R=I,this.#W=I-A};return requestAnimationFrame((()=>{const g=performance.now()-this.#R,I=Math.max(0,1e3/60-g-this.#W);setTimeout(A,I)})),this.#Q=!0,!0}forceRender(){if(!this.#Q)return void console.warn("@GlBaseRenderer#forceRender - Scene is not dirty");this.#Q=!1;const A=new ColorRenderState(this.gl);for(const g of this.#u)g.draw(A)}bindGLRenderer(A){A.gl=this.__gl,A.renderer=this,A.directives=this.directives,A.directivesHash=this.directivesHash}drawScene(A){A.directives=[...this.directives,"#define DRAW_COLOR"];for(const g in this.#m){const I=this.#m[g];for(const g of I)g.enabled&&g.draw(A)}}drawHighlightedGeoms(A){this.bindGLRenderer(A),A.directives=[...this.directives,"#define DRAW_HIGHLIGHT"];for(const g in this.#m){const I=this.#m[g];for(const g of I)g.enabled&&g.drawHighlightedGeoms(A)}}drawSceneGeomData(A,g=255){A.pushGLStack("GLBaseRenderer.drawSceneGeomData"),A.glEnable(this.__gl.DEPTH_TEST),A.glEnable(this.__gl.CULL_FACE),this.bindGLRenderer(A),A.directives=[...this.directives,"#define DRAW_GEOMDATA"],A.floatGeomBuffer=this.floatGeomBuffer;for(const I in this.#m){if(!(Number.parseInt(I)&g))continue;const C=this.#m[I];for(const g of C)g.enabled&&g.drawGeomData(A)}A.popGLStack()}static registerPass(A,g){registeredPasses.push({cls:A,passType:g})}destroy(A=!0){if(this.#S.disconnect(),window.removeEventListener("resize",this.#F),A){const A=this.__gl.getExtension("WEBGL_lose_context");A&&A.loseContext()}}}var frag$d="\nprecision highp float;\n#define GLSLIFY 1\n\nvarying vec2 v_texCoord;\n\nimport 'GLSLUtils.glsl'\nimport 'Hammersley.glsl'\nimport 'ImportanceSampleGGX.glsl'\n\nfloat GeometrySchlickGGX(float NdotV, float roughness)\n{\n  float a = roughness;\n  float k = (a * a) / 2.0;\n\n  float nom   = NdotV;\n  float denom = NdotV * (1.0 - k) + k;\n\n  return nom / denom;\n}\n// ----------------------------------------------------------------------------\nfloat GeometrySmith(vec3 N, vec3 V, vec3 L, float roughness)\n{\n  float NdotV = max(dot(N, V), 0.0);\n  float NdotL = max(dot(N, L), 0.0);\n  float ggx2 = GeometrySchlickGGX(NdotV, roughness);\n  float ggx1 = GeometrySchlickGGX(NdotL, roughness);\n\n  return ggx1 * ggx2;\n}\n\nvec2 IntegrateBRDF(float NdotV, float roughness)\n{\n  vec3 V;\n  V.x = sqrt(1.0 - NdotV*NdotV);\n  V.y = 0.0;\n  V.z = NdotV;\n\n  float A = 0.0;\n  float B = 0.0;\n\n  vec3 N = vec3(0.0, 0.0, 1.0);\n\n  for(uint i = 0u; i < SAMPLE_COUNT; ++i)\n  {\n    vec2 Xi = Hammersley(i, SAMPLE_COUNT);\n    vec3 H  = ImportanceSampleGGX(Xi, N, roughness);\n    vec3 L  = normalize(2.0 * dot(V, H) * H - V);\n\n    float NdotL = max(L.z, 0.0);\n    float NdotH = max(H.z, 0.0);\n    float VdotH = max(dot(V, H), 0.0);\n\n    if (NdotL > 0.0)\n    {\n      float G = GeometrySmith(N, V, L, roughness);\n      float G_Vis = (G * VdotH) / (NdotH * NdotV);\n      float Fc = pow(1.0 - VdotH, 5.0);\n\n      A += (1.0 - Fc) * G_Vis;\n      B += Fc * G_Vis;\n    }\n  }\n  A /= float(SAMPLE_COUNT);\n  B /= float(SAMPLE_COUNT);\n  return vec2(A, B);\n}\n\nout vec2 fragColor;\nvoid main(void) {\n  vec2 integratedBRDF = IntegrateBRDF(v_texCoord.x, v_texCoord.y);\n  fragColor = integratedBRDF;\n}\n\n",vert$e="\nprecision highp float;\n#define GLSLIFY 1\n\nimport 'quadVertexFromID.glsl'\n\n/* VS Outputs */\nvarying vec2 v_texCoord;\n \nvoid main()\n{\n  vec2 position = getQuadVertexPositionFromID();\n  v_texCoord = position+0.5;\n  gl_Position = vec4(position*2.0, 0.0, 1.0);\n}\n";class PreComputeBRDFShader extends GLShader{constructor(A){super(A,"PreComputeBRDFShader"),this.setShaderStage("VERTEX_SHADER",vert$e),this.setShaderStage("FRAGMENT_SHADER",frag$d)}}var frag$c="\nprecision highp float;\n#define GLSLIFY 1\nimport 'constants.glsl'\nimport 'convolve-helpers.glsl'\n\nuniform int faceId;\nvarying vec2 v_texCoord;\n\nout vec4 fragColor;\nvoid main(void) {\n\n  vec3 N = cubeFaceUvToDir(v_texCoord.x, v_texCoord.y, faceId);   \n\n  vec3 irradiance = vec3(0.0);\n\n  vec3 up        = abs(N.z) < 0.999 ? vec3(0.0, 0.0, 1.0) : vec3(1.0, 0.0, 0.0);\n  vec3 tangent   = normalize(cross(up, N));\n  vec3 bitangent = cross(N, tangent);\n\n  float nrSamples = 0.0; \n  for(float phi = 0.0; phi < 2.0 * PI; phi += SAMPLE_DELTA)\n  {\n    for(float theta = 0.0; theta < 0.5 * PI; theta += SAMPLE_DELTA)\n    {\n      // spherical to cartesian (in tangent space)\n      // from spherical coordinates to cartesian coordinates\n      vec3 H = vec3(cos(phi) * sin(theta), sin(phi) * sin(theta), cos(theta));\n      // tangent space to world\n      vec3 sampleVec = tangent * H.x + bitangent * H.y + N * H.z;\n\n      irradiance += sampleEnvMap(normalize(sampleVec)).rgb * cos(theta) * sin(theta);\n      nrSamples++;\n    }\n  }\n  irradiance = PI * irradiance * (1.0 / float(nrSamples));\n\n  fragColor = vec4(irradiance, 1.0);\n}\n",vert$d="\nprecision highp float;\n#define GLSLIFY 1\n\nimport 'quadVertexFromID.glsl'\n\n/* VS Outputs */\nvarying vec2 v_texCoord;\n \nvoid main()\n{\n  vec2 position = getQuadVertexPositionFromID();\n  v_texCoord = position+0.5;\n  gl_Position = vec4(position*2.0, 0.0, 1.0);\n}\n";class ConvolveIrradianceShader extends GLShader{constructor(A){super(A,"ConvolveIrradianceShader"),this.setShaderStage("VERTEX_SHADER",vert$d),this.setShaderStage("FRAGMENT_SHADER",frag$c)}}var vert$c="\nprecision highp float;\n#define GLSLIFY 1\n\nimport 'quadVertexFromID.glsl'\n\n/* VS Outputs */\nvarying vec2 v_texCoord;\n \nvoid main()\n{\n  vec2 position = getQuadVertexPositionFromID();\n  v_texCoord = position+0.5;\n  gl_Position = vec4(position*2.0, 0.0, 1.0);\n}\n",frag$b="precision highp float;\nprecision highp int;\n#define GLSLIFY 1\n\nimport 'GLSLUtils.glsl'\nimport 'ImportanceSampleGGX.glsl'\nimport 'convolve-helpers.glsl'\nimport 'Hammersley.glsl'\n  \n\nuniform int faceId;\nuniform float roughness;\n\nvarying vec2 v_texCoord;\n\nout vec4 fragColor;\nvoid main(void) {\n\n  vec3 N = cubeFaceUvToDir(v_texCoord.x, v_texCoord.y, faceId);   \n\n  vec3 R = N;\n  vec3 V = R;\n\n  float totalWeight = 0.0;   \n  vec3 prefilteredColor = vec3(0.0);     \n  for(uint i = 0u; i < SAMPLE_COUNT; ++i)\n  {\n    vec2 Xi = Hammersley(i, SAMPLE_COUNT);\n    vec3 H  = ImportanceSampleGGX(Xi, N, roughness);\n    vec3 L  = normalize(2.0 * dot(V, H) * H - V);\n\n    float NdotL = max(dot(N, L), 0.0);\n    if (NdotL > 0.0)\n    {\n      prefilteredColor += sampleEnvMap(L).rgb * NdotL;\n      totalWeight      += NdotL;\n    }\n  }\n  prefilteredColor = prefilteredColor / totalWeight;\n\n  fragColor = vec4(prefilteredColor, 1.0);\n}\n";class ConvolveSpecularShader extends GLShader{constructor(A){super(A,"ConvolveSpecularShader"),this.setShaderStage("VERTEX_SHADER",vert$c),this.setShaderStage("FRAGMENT_SHADER",frag$b)}}class GLProbe extends EventEmitter{gl;textureDesc;maxFragmentShaderTextureUnits;textureType;convolved=!1;brdfLUTTexture;irradianceCubeTex;specularCubetex;constructor(A,g){super(),this.gl=A,this.maxFragmentShaderTextureUnits=A.getParameter(this.gl.MAX_TEXTURE_IMAGE_UNITS),this.gl.__quadVertexIdsBuffer||this.gl.setupInstancedQuad(),this.textureType=1,this.textureDesc=[0,0,0,0],this.convolved=!1}convolveProbe(A){const g=this.gl,I=new RenderState(g);I.directives=["#define ENABLE_ES3","#define ENABLE_FLOAT_TEXTURES"],"Low"==SystemDesc.deviceCategory?(I.directives.push("#define SAMPLE_DELTA 0.1"),I.directives.push("#define SAMPLE_COUNT 64u")):"Medium"==SystemDesc.deviceCategory?(I.directives.push("#define SAMPLE_DELTA 0.08"),I.directives.push("#define SAMPLE_COUNT 256u")):(I.directives.push("#define SAMPLE_DELTA 0.025"),I.directives.push("#define SAMPLE_COUNT 1024u")),this.brdfLUTTexture=g.createTexture(),g.bindTexture(g.TEXTURE_2D,this.brdfLUTTexture),g.texImage2D(g.TEXTURE_2D,0,g.RG16F,512,512,0,g.RG,g.HALF_FLOAT,null),g.texParameteri(g.TEXTURE_2D,g.TEXTURE_WRAP_S,g.CLAMP_TO_EDGE),g.texParameteri(g.TEXTURE_2D,g.TEXTURE_WRAP_T,g.CLAMP_TO_EDGE),g.texParameteri(g.TEXTURE_2D,g.TEXTURE_MIN_FILTER,g.LINEAR),g.texParameteri(g.TEXTURE_2D,g.TEXTURE_MAG_FILTER,g.LINEAR);const C=new PreComputeBRDFShader(this.gl),e=C.compileForTarget("GLProbe",I.directives),i=generateShaderGeomBinding(this.gl,e.attrs,g.__quadattrbuffers,g.__quadIndexBuffer),t=g.createFramebuffer();g.bindFramebuffer(g.DRAW_FRAMEBUFFER,t),g.framebufferTexture2D(g.DRAW_FRAMEBUFFER,g.COLOR_ATTACHMENT0,g.TEXTURE_2D,this.brdfLUTTexture,0),C.bind(I),i.bind(I),g.clear(g.COLOR_BUFFER_BIT|g.DEPTH_BUFFER_BIT),g.viewport(0,0,512,512),g.drawQuad(),g.bindFramebuffer(g.DRAW_FRAMEBUFFER,null),g.deleteFramebuffer(t),C.unbind(I),C.destroy();{const C=new ConvolveIrradianceShader(this.gl),e=C.compileForTarget("GLProbe",I.directives),i=generateShaderGeomBinding(this.gl,e.attrs,g.__quadattrbuffers,g.__quadIndexBuffer);C.bind(I,"GLProbe"),i.bind(I);const t=I.unifs;A.bindToUniform(I,t.envMap);const o=64;this.irradianceCubeTex=g.createTexture(),g.bindTexture(g.TEXTURE_CUBE_MAP,this.irradianceCubeTex),g.texParameteri(g.TEXTURE_CUBE_MAP,g.TEXTURE_MAG_FILTER,g.LINEAR),g.texParameteri(g.TEXTURE_CUBE_MAP,g.TEXTURE_MIN_FILTER,g.LINEAR_MIPMAP_LINEAR),g.texParameteri(g.TEXTURE_CUBE_MAP,g.TEXTURE_WRAP_S,g.CLAMP_TO_EDGE),g.texParameteri(g.TEXTURE_CUBE_MAP,g.TEXTURE_WRAP_T,g.CLAMP_TO_EDGE),g.texParameteri(g.TEXTURE_CUBE_MAP,g.TEXTURE_WRAP_R,g.CLAMP_TO_EDGE);for(let A=0;A<6;A++)g.texImage2D(g.TEXTURE_CUBE_MAP_POSITIVE_X+A,0,g.RGBA16F,o,o,0,g.RGBA,g.HALF_FLOAT,null);const a=g.createFramebuffer();g.bindFramebuffer(g.DRAW_FRAMEBUFFER,a);for(let A=0;A<6;++A)g.uniform1i(t.faceId.location,A),g.framebufferTexture2D(g.DRAW_FRAMEBUFFER,g.COLOR_ATTACHMENT0,g.TEXTURE_CUBE_MAP_POSITIVE_X+A,this.irradianceCubeTex,0),g.viewport(0,0,o,o),g.clearColor(1,0,0,1),g.clear(g.COLOR_BUFFER_BIT|g.DEPTH_BUFFER_BIT),g.drawQuad();g.bindFramebuffer(g.DRAW_FRAMEBUFFER,null),g.deleteFramebuffer(a),g.generateMipmap(g.TEXTURE_CUBE_MAP)}{const C=new ConvolveSpecularShader(this.gl),e=C.compileForTarget("GLProbe",I.directives),i=generateShaderGeomBinding(this.gl,e.attrs,g.__quadattrbuffers,g.__quadIndexBuffer);C.bind(I,"GLProbe"),i.bind(I);const t=I.unifs;A.bindToUniform(I,t.envMap),this.specularCubetex=g.createTexture(),g.bindTexture(g.TEXTURE_CUBE_MAP,this.specularCubetex),g.texParameteri(g.TEXTURE_CUBE_MAP,g.TEXTURE_MAG_FILTER,g.LINEAR),g.texParameteri(g.TEXTURE_CUBE_MAP,g.TEXTURE_MIN_FILTER,g.LINEAR_MIPMAP_LINEAR),g.texParameteri(g.TEXTURE_CUBE_MAP,g.TEXTURE_WRAP_S,g.CLAMP_TO_EDGE),g.texParameteri(g.TEXTURE_CUBE_MAP,g.TEXTURE_WRAP_T,g.CLAMP_TO_EDGE),g.texParameteri(g.TEXTURE_CUBE_MAP,g.TEXTURE_WRAP_R,g.CLAMP_TO_EDGE);const o=256;for(let A=0;A<6;A++)g.texImage2D(g.TEXTURE_CUBE_MAP_POSITIVE_X+A,0,g.RGBA16F,o,o,0,g.RGBA,g.HALF_FLOAT,null);g.generateMipmap(g.TEXTURE_CUBE_MAP);const a=5;for(let A=0;A<a;++A){const I=o*Math.pow(.5,A),C=o*Math.pow(.5,A),e=g.createFramebuffer();g.bindFramebuffer(g.DRAW_FRAMEBUFFER,e),g.viewport(0,0,I,C);const i=A/(a-1);g.uniform1f(t.roughness.location,i);for(let I=0;I<6;++I)g.uniform1i(t.faceId.location,I),g.framebufferTexture2D(g.DRAW_FRAMEBUFFER,g.COLOR_ATTACHMENT0,g.TEXTURE_CUBE_MAP_POSITIVE_X+I,this.specularCubetex,A),g.drawQuad();g.bindFramebuffer(g.DRAW_FRAMEBUFFER,null),g.deleteFramebuffer(e)}C.destroy()}this.convolved=!0}bind(A){const g=this.gl,{irradianceMap:I,prefilterMap:C,brdfLUT:e,envMapFlags:i}=A.unifs;if(!this.convolved)return I&&g.uniform1i(I.location,this.maxFragmentShaderTextureUnits-1),C&&g.uniform1i(C.location,this.maxFragmentShaderTextureUnits-1),i&&g.uniform1i(i.location,-1),!1;if(e){const I=A.boundTextures++;g.activeTexture(this.gl.TEXTURE0+I),g.bindTexture(g.TEXTURE_2D,this.brdfLUTTexture),g.uniform1i(e.location,I)}if(I){const C=A.boundTextures++,e=this.gl.TEXTURE0+C;g.activeTexture(e),g.bindTexture(g.TEXTURE_CUBE_MAP,this.irradianceCubeTex),g.uniform1i(I.location,C)}if(C){const I=A.boundTextures++,e=this.gl.TEXTURE0+I;g.activeTexture(e),g.bindTexture(g.TEXTURE_CUBE_MAP,this.specularCubetex),g.uniform1i(C.location,I)}return i&&g.uniform1i(i.location,0),!0}destroy(){}}var frag$a="\nprecision highp float;\n#define GLSLIFY 1\n\nimport 'GLSLUtils.glsl'\nimport 'gamma.glsl'\nimport 'constants.glsl'\n\nuniform float focus;\nuniform float exposure;\n\n/* VS Outputs */\nvarying vec3 v_worldDir;\nvarying vec2 v_texCoord;\n\n#define ENABLE_INLINE_GAMMACORRECTION\n\n#define ENV_MAP_LATLONG 0\n#define ENV_MAP_OCT 1\n#define ENV_MAP_CUBE 2\n#define ENV_MAP_irradianceMap 8\n#define ENV_MAP_prefilterMap 3\n#define ENV_MAP_STEREO_LATLONG 4\n#define ENV_MAP_DUALFISHEYE 5\n#define ENV_MAP_SH 6\n#define ENV_MAP_BRDF_LUT 7\n\n#define ENV_MAPTYPE ENV_MAP_OCT\n\n#if (ENV_MAPTYPE == ENV_MAP_LATLONG)  \n\nimport 'envmap-equirect.glsl'\n\nuniform sampler2D backgroundImage;\n\nvec4 sampleEnvMap(vec3 dir) {\n  vec2 uv = latLongUVsFromDir(normalize(dir));\n  vec4 texel = texture2D(backgroundImage, uv) * exposure;\n  return vec4(texel.rgb/texel.a, 1.0);\n}\n\n#elif (ENV_MAPTYPE == ENV_MAP_OCT)  \n\nimport 'envmap-octahedral.glsl'\n\nuniform sampler2D   envMap;\n\nvec4 sampleEnvMap(vec3 dir) {\n  vec2 uv = dirToSphOctUv(normalize(dir));\n  if (false) {\n    vec4 texel = texture2D(envMap, uv);\n    return vec4(texel.rgb/texel.a, 1.0);\n  }\n  else {\n    return texture2D(envMap, uv) * exposure;\n  }\n}\n\n#elif (ENV_MAPTYPE == ENV_MAP_CUBE)\n\nuniform samplerCube cubeMap;\n\nvec4 sampleEnvMap(vec3 dir) {\n  return texture(cubeMap, dir, 0.0);// * exposure;\n  // return textureLod(cubeMap, dir, exposure);\n}\n\n#elif (ENV_MAPTYPE == ENV_MAP_irradianceMap)\n\nuniform samplerCube irradianceMap;\n\nvec4 sampleEnvMap(vec3 dir) {\n  return textureLod(irradianceMap, dir, exposure);\n}\n\n#elif (ENV_MAPTYPE == ENV_MAP_prefilterMap)\n\nuniform samplerCube prefilterMap;\n\nvec4 sampleEnvMap(vec3 dir) {\n  return textureLod(prefilterMap, dir, exposure);\n}\n\n#elif (ENV_MAPTYPE == ENV_MAP_STEREO_LATLONG)  \n\nimport 'envmap-equirect.glsl'\nuniform int eye;// L = 0, R = 1;\nuniform sampler2D backgroundImage;\n\nvec4 sampleEnvMap(vec3 dir) {\n  vec2 uv = latLongUVsFromDir(normalize(v_worldDir));\n  uv.y *= 0.5;\n  if (eye == 1) {\n    uv.y += 0.5;\n  }\n  vec4 texel = texture2D(backgroundImage, uv) * exposure;\n  fragColor = vec4(texel.rgb/texel.a, 1.0);\n}\n\n#elif (ENV_MAPTYPE == ENV_MAP_DUALFISHEYE)\n\nimport 'envmap-dualfisheye.glsl'\n\nvec4 sampleEnvMap(vec3 dir) {\n  vec2 uv = dualfisheyeUVsFromDir(dir);\n  return texture2D(backgroundImage, uv) * exposure;\n}\n\n#elif (ENV_MAPTYPE == ENV_MAP_SH)\n\nimport 'SHCoeffs.glsl'\n\nvec4 sampleEnvMap(vec3 dir) {\n\treturn vec4(sampleSHCoeffs(dir) * exposure, 1.0);\n}\n\n#elif (ENV_MAPTYPE == ENV_MAP_BRDF_LUT)\n\nuniform sampler2D brdfLUT;\n\nvec4 sampleEnvMap(vec3 dir) {\n  return texture2D(brdfLUT, v_texCoord);\n}\n#endif\n\n#ifdef ENABLE_ES3\n  out vec4 fragColor;\n#endif\n\nvoid main(void) {\n#ifndef ENABLE_ES3\n  vec4 fragColor;\n#endif\n\n  fragColor = sampleEnvMap(normalize(v_worldDir));\n  fragColor.a = 1.0;\n\n#ifdef ENABLE_INLINE_GAMMACORRECTION\n  fragColor.rgb = toGamma(fragColor.rgb);\n#endif\n\n#ifndef ENABLE_ES3\n  gl_FragColor = fragColor;\n#endif\n}\n",vert$b="\nprecision highp float;\n#define GLSLIFY 1\n\nimport 'quadVertexFromID.glsl'\n\nuniform mat4 projectionMatrix;\nuniform mat4 viewMatrix;\n\nimport 'inverse.glsl'\nimport 'transpose.glsl'\n\n/* VS Outputs */\nvarying vec3 v_worldDir;\nvarying vec2 v_texCoord;\n \nvoid main()\n{\n  vec2 position = getQuadVertexPositionFromID() * 2.0;\n  v_texCoord = position * 0.5 + 0.5;\n\n  mat4 inverseProjection = inverse(projectionMatrix);\n  mat3 inverseModelview = transpose(mat3(viewMatrix));\n\n  // transform from the normalized device coordinates back to the view space\n  vec3 unprojected = (inverseProjection * vec4(position, 0, 1)).xyz;\n\n  // transfrom from the view space back to the world space\n  // and use it as a sampling vector\n  v_worldDir = inverseModelview * unprojected;\n\n  gl_Position = vec4(position, 0, 1);\n}\n";class EnvMapShader extends GLShader{constructor(A){super(A,"EnvMapShader"),this.setShaderStage("VERTEX_SHADER",vert$b),this.setShaderStage("FRAGMENT_SHADER",frag$a)}}class GLEnvMap extends GLProbe{renderer;envMap;backgroundFocus;srcGLTex=null;envMapShader=null;envMapShaderBinding=null;constructor(A,g){super(A.gl,"EnvMap"),this.renderer=A,this.envMap=g,this.backgroundFocus=0,this.envMap.isLoaded()?this.init():this.envMap.once("loaded",(A=>{this.init()}))}get glTex(){return this.srcGLTex}init(){const A=this.renderer.gl;A.__quadVertexIdsBuffer||A.setupInstancedQuad(),this.srcGLTex=new GLHDRImage(A,this.envMap),this.envMapShader=new EnvMapShader(A);const g=this.envMapShader.compileForTarget("GLEnvMap",["#define ENABLE_ES3"]);this.envMapShaderBinding=generateShaderGeomBinding(A,g.attrs,A.__quadattrbuffers,A.__quadIndexBuffer);const I=this.envMap.headlightModeParam,C=()=>{I.value?this.textureDesc[3]|=1:this.textureDesc[3]&=-2};C(),I.on("valueChanged",(()=>{C(),this.emit("updated")})),this.convolveProbe(this.srcGLTex),this.emit("updated")}getEnvMap(){return this.envMap}getBackgroundFocus(){return this.backgroundFocus}setBackgroundFocus(A){this.backgroundFocus=A,this.renderer.requestRedraw()}draw(A){if(this.envMap.isLoaded()){const g=this.gl;{this.envMapShader.bind(A,"GLEnvMap");const I=A.unifs,{envMap:C,focus:e,exposure:i}=A.unifs;C&&this.srcGLTex.bindToUniform(A,C),e&&g.uniform1f(e.location,this.backgroundFocus),i&&g.uniform1f(i.location,A.exposure),this.envMapShaderBinding.bind(A),g.depthMask(!1),A.bindViewports(I,(()=>{g.drawQuad()})),g.depthMask(!0)}}}destroy(){super.destroy(),this.srcGLTex&&this.srcGLTex.destroy()}}const ALL_PASSES=PassType.OPAQUE|PassType.TRANSPARENT|PassType.OVERLAY;class GLRenderer extends GLBaseRenderer{#w=1;#H=2.2;#X=null;#Y;#K;highlightOutlineThickness=1;renderMode="pbr";outlineMethod="geometry";outlineThickness=0;outlineColor=new Color(.15,.15,.15,1);hiddenLineColor=new Color(.15,.15,.15,0);outlineSensitivity=2;outlineDepthBias=.7;#N;#v;#J=null;#f=null;#L=null;#M=new Mat4;constructor(A,g={}){super(A,g),this.#w=1,this.#H=2.2,this.#K=!0,this.#N=0,this.#v=0,this.setShaderPreprocessorDirective("ENABLE_GAMMACORRECTION","#define ENABLE_INLINE_GAMMACORRECTION"),g.disableTextures||this.setShaderPreprocessorDirective("ENABLE_TEXTURES","#define ENABLE_TEXTURES"),g.debugGeomIds&&this.setShaderPreprocessorDirective("DEBUG_GEOM_ID","#define DEBUG_GEOM_ID")}__bindEnvMap(A){const g=this.__gl;if(!(A instanceof EnvMap)){const I=A;if("HDR"===I.type?this.#Y=new GLHDRImage(this.__gl,I):this.#Y=new GLTexture2D(this.__gl,I),this.#Y.on("loaded",(()=>{this.requestRedraw()})),this.#Y.on("updated",(()=>{this.requestRedraw()})),!this.#f){g.__quadVertexIdsBuffer||g.setupInstancedQuad(),this.#f=new EnvMapShader(this.__gl);const A=this.#f.compileForTarget("key",[]);this.#L=generateShaderGeomBinding(this.__gl,A.attrs,g.__quadattrbuffers,g.__quadIndexBuffer)}return}if("webgl2"!==g.name)return;"FLOAT"===A.type&&(this.setShaderPreprocessorDirective("ENABLE_PBR","#define ENABLE_PBR"),this.#X=new GLEnvMap(this,A)),this.#X.on("loaded",(A=>{this.requestRedraw()})),this.#X.on("updated",(A=>{this.requestRedraw()}));const I=new EnvMapAssignedEvent(this.#X);this.emit("envMapAssigned",I)}setScene(A){const g=A.envMapParam;null!=g.value&&this.__bindEnvMap(g.value),g.on("valueChanged",(()=>{this.__bindEnvMap(g.value)}));const I=A.displayEnvMapParam;this.#K=I.value,I.on("valueChanged",(()=>{this.#K=I.value,this.requestRedraw()})),super.setScene(A)}addViewport(A){return super.addViewport(A)}get exposure(){return this.#w}set exposure(A){this.#w=A,this.requestRedraw()}get gamma(){return this.#H}set gamma(A){this.#H=A,this.requestRedraw()}get displayEnvironment(){return this.#K}set displayEnvironment(A){this.#K=A,this.requestRedraw()}bindRaycastViewport(A,g,I){A.viewXfo=g,A.cameraMatrix=g.toMat4(),A.viewScale=1,A.region=[0,0,3,3];const C=A.cameraMatrix.inverse(),e=this.gl;A.bindRendererUnifs=g=>{const{cameraMatrix:i,viewMatrix:t,projectionMatrix:o,eye:a,isOrthographic:B,viewportFrustum:s}=g;i&&e.uniformMatrix4fv(i.location,!1,A.cameraMatrix.asArray()),t&&e.uniformMatrix4fv(t.location,!1,C.asArray()),o&&e.uniformMatrix4fv(o.location,!1,I.asArray()),a&&e.uniform1i(a.location,0),B&&e.uniform1i(B.location,1)},A.bindViewports=(A,g)=>g()}raycastWithRay(A,g,I=.01,C=ALL_PASSES){const e=new Xfo;return e.setLookAt(A.start,A.start.add(A.dir),new Vec3(0,0,1)),this.raycast(e,A,g,I,C)}raycastWithXfo(A,g,I=.01,C=ALL_PASSES){const e=new Ray(A.tr,A.ori.getZaxis().negate());return this.raycast(A,e,g,I,C)}raycast(A,g,I,C=.01,e=ALL_PASSES){return this.#N==I&&this.#v==C||(this.#M.setOrthographicMatrix(-.5*C,.5*C,-.5*C,.5*C,0,I),this.#N=I,this.#v=C),this.raycastWithProjection(A,this.#M,g,e)}raycastWithProjection(A,g,I,C=ALL_PASSES){const e=new GeomDataRenderState(this.__gl);this.bindRaycastViewport(e,A,g);const i=this.__gl;this.#J||(this.#J=new GLRenderTarget(i,{type:i.FLOAT,format:i.RGBA,filter:i.NEAREST,createDepthTexture:!0,width:3,height:3,numColorChannels:1})),this.#J.bindForWriting(e,!0),i.enable(i.CULL_FACE),i.enable(i.DEPTH_TEST),i.depthFunc(i.LEQUAL),i.depthMask(!0),this.drawSceneGeomData(e,C),i.finish(),this.#J.unbindForWriting(),this.#J.bindForReading();const t=new Float32Array(36);i.readPixels(0,0,3,3,i.RGBA,i.FLOAT,t),this.#J.unbindForReading();const o=[4,3,5,1,7];let a;for(const A of o)if(0!=t[4*A+3]){a=t.subarray(4*A,4*A+4);break}if(!a)return null;const B=63&Math.round(a[0]),s=this.getPass(B)?.getGeomItemAndDist(a);if(s){const A=I.start.add(I.dir.scale(s.dist));return{pointerRay:I,intersectionPos:A,geomItem:s.geomItem,componentId:-1,dist:s.dist,geomData:a}}return null}raycastCluster(A,g,I,C=.01,e=ALL_PASSES){const i=this.__gl,t=new GeomDataRenderState(this.__gl);this.bindRaycastViewport(t,A,this.#M),this.#J||(this.#J=new GLRenderTarget(i,{type:"FLOAT",format:"RGBA",filter:"NEAREST",createDepthTexture:!0,width:3,height:3,numColorChannels:1}),this.#M=new Mat4),this.#N==I&&this.#v==C||(this.#M.setOrthographicMatrix(-.5*C,.5*C,-.5*C,.5*C,0,I),this.#N=I,this.#v=C),this.#J.bindForWriting(t,!0),i.enable(i.CULL_FACE),i.enable(i.DEPTH_TEST),i.depthFunc(i.LEQUAL),i.depthMask(!0),this.drawSceneGeomData(t,e),i.finish(),this.#J.unbindForWriting(),this.#J.bindForReading();const o=new Float32Array(36);i.readPixels(0,0,3,3,i.RGBA,i.FLOAT,o),this.#J.unbindForReading();const a=[];for(let A=0;A<9;A++)if(0!=o[4*A+3]){const I=o.subarray(4*A,4*A+4),C=63&Math.round(I[0]),e=this.getPass(C);if(e){const A=e.getGeomItemAndDist(I);if(A){const C=g.start.add(g.dir.scale(A.dist));a.push({pointerRay:g,intersectionPos:C,geomItem:A.geomItem,componentId:-1,dist:A.dist,geomData:I})}}}return a}drawBackground(A){if(this.#Y&&this.#f&&this.#L){if(!this.#Y.isLoaded())return;const g=this.__gl;g.depthMask(!1),this.#f.bind(A);const I=A.unifs;this.#Y.bindToUniform(A,I.backgroundImage),this.#L.bind(A),g.drawQuad()}else this.#X&&this.#X.draw&&this.#X.draw(A)}bindGLRenderer(A){super.bindGLRenderer(A),A instanceof ColorRenderState&&(A.envMap=this.#X,A.exposure=this.#w,A.renderMode=this.renderMode,A.outlineThickness=this.outlineThickness,A.outlineColor=this.outlineColor,A.hiddenLineColor=this.hiddenLineColor,A.outlineMethod=this.outlineMethod,A.screenQuad=this.screenQuad)}drawScene(A){this.bindGLRenderer(A),this.#K&&this.drawBackground(A),super.drawScene(A)}}class GLGeomItemSetMultiDraw extends EventEmitter{renderer;gl;glGeomItems=[];glGeomIdsMapping={};glgeomItemEventHandlers=[];freeIndices=[];viewPos=new Vec3;drawElementCounts=new Int32Array(0);drawElementOffsets=new Int32Array(0);highlightElementCounts=new Int32Array(0);highlightElementOffsets=new Int32Array(0);drawOrderToIndex=[];indexToDrawIndex=[];drawIdsArray=new Float32Array(0);drawIdsBufferDirty=!0;drawIdsTexture=null;highlightedItems=[];highlightedIdsArray=null;highlightedIdsTexture=null;highlightedIdsBufferDirty=!1;dirtyGeomIndices=new Set;constructor(A){super(),this.renderer=A,this.gl=A.gl,this.renderer.glGeomLibrary.on("geomDataChanged",(A=>{null!=this.glGeomIdsMapping[A.index]&&this.dirtyGeomIndices.add(A.index)}))}addGLGeomItem(A){const g=this.freeIndices.length>0?this.freeIndices.pop():this.glGeomItems.length;this.indexToDrawIndex[g]=this.drawOrderToIndex.length,this.drawOrderToIndex.push(g);const I={};I.visibilityChanged=I=>{const C=this.indexToDrawIndex[g];if(I.state){const g=this.renderer.glGeomLibrary.getGeomOffsetAndCount(A.geomId);this.drawElementCounts[C]=g[1]}else this.drawElementCounts[C]=0;this.emit("updated")},A.on("visibilityChanged",I.visibilityChanged),A.geomItem.isHighlighted()&&(this.highlightedItems.push(A),this.highlightedIdsBufferDirty=!0),I.highlightChanged=g=>{if(g&&g.name){if(this.highlightedItems.includes(A))return;this.highlightedItems.push(A)}else this.highlightedItems.splice(this.highlightedItems.indexOf(A),1);this.highlightedIdsBufferDirty=!0,this.emit("updated")},A.geomItem.on("highlightChanged",I.highlightChanged);let C=A.geomItem.geomParam.value;A.geomId=this.renderer.glGeomLibrary.addGeom(C),this.glGeomIdsMapping[A.geomId]?this.glGeomIdsMapping[A.geomId].push(g):this.glGeomIdsMapping[A.geomId]=[g],this.glGeomItems[g]=A,this.glgeomItemEventHandlers[g]=I,this.drawIdsBufferDirty=!0,this.emit("updated")}removeGLGeomItem(A){const g=this.glGeomItems.indexOf(A),I=this.renderer.glGeomLibrary.getGeom(A.geomId);this.renderer.glGeomLibrary.removeGeom(I);const C=this.glGeomIdsMapping[A.geomId];C.splice(C.indexOf(g),1),0==C.length&&(delete this.glGeomIdsMapping[A.geomId],this.dirtyGeomIndices.has(A.geomId)&&this.dirtyGeomIndices.delete(A.geomId));const e=this.glgeomItemEventHandlers[g];A.geomItem.off("highlightChanged",e.highlightChanged),A.off("visibilityChanged",e.visibilityChanged),this.glGeomItems[g]=null,this.glgeomItemEventHandlers[g]=null,this.drawIdsArray[g]=0,this.freeIndices.push(g);const i=this.drawOrderToIndex.indexOf(g);if(this.drawOrderToIndex.splice(i,1),this.indexToDrawIndex[g]=-1,this.drawElementOffsets[i]=0,this.drawElementCounts[i]=0,this.drawIdsBufferDirty=!0,A.geomItem.isHighlighted()){const g=this.highlightedItems.indexOf(A);this.highlightedItems.splice(g,1),this.highlightedIdsBufferDirty=!0}this.emit("updated")}cleanGeomIds(){this.dirtyGeomIndices.forEach((A=>{const g=this.glGeomIdsMapping[A];if(null!=g){const I=this.renderer.glGeomLibrary.getGeomOffsetAndCount(A);g.forEach((A=>{const g=this.glGeomItems[A];if(g.isVisible()){const C=this.indexToDrawIndex[A];this.drawElementOffsets[C]=I[0],this.drawElementCounts[C]=I[1],this.drawIdsArray[C]=g.geomItemId;const e=this.highlightedItems.indexOf(g);-1!=e&&(this.highlightElementOffsets[e]=I[0],this.highlightElementCounts[e]=I[1])}}))}})),this.dirtyGeomIndices=new Set}updateDrawIDsBuffer(A){(!this.drawIdsArray||this.drawOrderToIndex.length>this.drawIdsArray.length)&&(this.drawIdsArray=new Float32Array(this.drawOrderToIndex.length),this.drawElementOffsets=new Int32Array(this.drawOrderToIndex.length+1),this.drawElementCounts=new Int32Array(this.drawOrderToIndex.length+1)),this.drawOrderToIndex.forEach(((A,g)=>{const I=this.glGeomItems[A];if(!I)return;const C=this.renderer.glGeomLibrary.getGeomOffsetAndCount(I.geomId);this.drawElementOffsets[g]=C[0],this.drawElementCounts[g]=I.isVisible()?C[1]:0,this.drawIdsArray[g]=I.geomItemId,this.indexToDrawIndex[A]=g})),this.dirtyGeomIndices=new Set;const g=this.renderer.gl,I=A.boundTextures++;g.activeTexture(g.TEXTURE0+I);const C=2*MathFunctions.nextPow2(Math.ceil(Math.sqrt(this.drawOrderToIndex.length)));this.drawIdsTexture?(this.drawIdsTexture.width<C||this.drawIdsTexture.height<C)&&this.drawIdsTexture.resize(C,C):this.drawIdsTexture=new GLTexture2D(this.gl,{format:this.gl.RED,type:this.gl.FLOAT,width:C,height:C,filter:this.gl.NEAREST,wrap:this.gl.CLAMP_TO_EDGE,mipMapped:!1});{const A=this.drawIdsTexture,I=this.drawIdsTexture.width;g.bindTexture(g.TEXTURE_2D,A.glTex);const C=0,e=0,i=1,t=A.format,o=A.type,a=Math.ceil((e+this.drawOrderToIndex.length)/I);let B=0,s=this.drawOrderToIndex.length,l=e;for(let A=0;A<a;A++){let A;l+s>I?(A=I-l,l=0):A=s;const e=B%I,a=Math.floor(B/I),n=this.drawIdsArray.subarray(B,B+A);g.texSubImage2D(g.TEXTURE_2D,C,e,a,A,i,t,o,n),B+=A,s-=A}}g.bindTexture(g.TEXTURE_2D,null),A.boundTextures--,this.drawIdsBufferDirty=!1}updateHighlightedIDsBuffer(A){if(this.highlightedIdsBufferDirty){(!this.highlightedIdsArray||this.highlightedItems.length>this.highlightedIdsArray.length)&&(this.highlightedIdsArray=new Float32Array(this.highlightedItems.length),this.highlightElementOffsets=new Int32Array(this.highlightedItems.length+1),this.highlightElementCounts=new Int32Array(this.highlightedItems.length+1)),this.highlightedItems.forEach(((A,g)=>{this.highlightedIdsArray[g]=A.geomItemId;const I=this.renderer.glGeomLibrary.getGeomOffsetAndCount(A.geomId);this.highlightElementOffsets[g]=I[0],this.highlightElementCounts[g]=I[1]}));for(let A=this.highlightedItems.length;A<this.highlightElementCounts.length;A++)this.highlightElementOffsets[A]=0,this.highlightElementCounts[A]=0;this.highlightedIdsBufferDirty=!1}const g=this.renderer.gl,I=A.boundTextures++;g.activeTexture(g.TEXTURE0+I);const C=MathFunctions.nextPow2(Math.ceil(Math.sqrt(this.highlightedItems.length)));this.highlightedIdsTexture?(this.highlightedIdsTexture.width<C||this.highlightedIdsTexture.height<C)&&this.highlightedIdsTexture.resize(C,C):this.highlightedIdsTexture=new GLTexture2D(this.gl,{format:this.gl.RED,type:this.gl.FLOAT,width:C,height:C,filter:this.gl.NEAREST,wrap:this.gl.CLAMP_TO_EDGE,mipMapped:!1});{const A=this.highlightedIdsTexture,I=this.highlightedIdsTexture.width;g.bindTexture(g.TEXTURE_2D,A.glTex);const C=0,e=0,i=1,t=A.format,o=A.type,a=Math.ceil((e+this.highlightedIdsArray.length)/I);let B=0,s=this.highlightedIdsArray.length,l=e;for(let A=0;A<a;A++){let A;l+s>I?(A=I-l,l=0):A=s;const e=B%I,a=Math.floor(B/I),n=this.highlightedIdsArray.subarray(B,B+A);g.texSubImage2D(g.TEXTURE_2D,C,e,a,A,i,t,o,n),B+=A,s-=A}}g.bindTexture(g.TEXTURE_2D,null),A.boundTextures--}draw(A){if(this.drawIdsBufferDirty?this.updateDrawIDsBuffer(A):this.dirtyGeomIndices.size>0&&this.cleanGeomIds(),0!=this.drawIdsArray.length){if(this.drawIdsTexture){const{drawIdsTexture:g}=A.unifs;this.drawIdsTexture.bindToUniform(A,g)}this.bindAndRender(A,this.drawElementCounts,this.drawElementOffsets,this.drawOrderToIndex.length)}}drawHighlighted(A){if(0!=this.highlightedItems.length){if(this.highlightedIdsBufferDirty&&this.updateHighlightedIDsBuffer(A),this.highlightedIdsTexture){const{drawIdsTexture:g}=A.unifs;this.highlightedIdsTexture.bindToUniform(A,g)}this.bindAndRender(A,this.highlightElementCounts,this.highlightElementOffsets,this.highlightedItems.length)}}drawGeomData(A){if(this.drawIdsBufferDirty&&this.updateDrawIDsBuffer(A),0!=this.drawOrderToIndex.length){if(this.drawIdsTexture){const{drawIdsTexture:g}=A.unifs;this.drawIdsTexture.bindToUniform(A,g)}this.bindAndRender(A,this.drawElementCounts,this.drawElementOffsets,this.drawOrderToIndex.length)}}bindAndRender(A,g,I,C){const e=this.gl,i=A.unifs;i.instancedDraw&&e.uniform1i(A.unifs.instancedDraw.location,1),A.bindViewports(i,(()=>{this.multiDraw(A,g,I,C)}))}sortItems(A){const g=new Float32Array(this.drawOrderToIndex.length);this.drawOrderToIndex.forEach((I=>{const C=this.glGeomItems[I];if(C){const e=C.geomItem.boundingBoxParam.value,i=e.center(),t=e.size(),o=i.distanceTo(A)-t;g[I]=o}})),this.drawOrderToIndex.sort(((A,I)=>g[I]-g[A])),this.drawOrderToIndex.forEach(((A,g)=>{const I=this.glGeomItems[A];I&&(this.drawIdsArray[g]=I.geomItemId,this.indexToDrawIndex[A]=g)})),this.drawIdsBufferDirty=!0,this.viewPos=A}destroy(){this.drawIdsTexture&&this.drawIdsTexture.destroy(),this.highlightedIdsTexture&&this.highlightedIdsTexture.destroy(),this.emit("destructing")}}class GLPointsItemSet extends GLGeomItemSetMultiDraw{multiDraw(A,g,I,C){const e=this.gl;if(e.multiDrawArrays)e.multiDrawArrays(e.POINTS,I,0,g,0,C);else{const{drawId:i}=A.unifs;for(let A=0;A<C;A++)e.uniform1i(i.location,A),e.drawArrays(e.POINTS,I[A],g[A])}}}class GLLinesItemSet extends GLGeomItemSetMultiDraw{multiDraw(A,g,I,C){const{occluded:e,hiddenLineColor:i}=A.unifs,t=A.hiddenLineColor&&A.hiddenLineColor.a>0&&e&&i,o=this.gl;if(o.multiDrawArrays)o.multiDrawElements(o.LINES,g,0,o.UNSIGNED_INT,I,0,C),t&&(o.uniform1i(e.location,1),o.uniform4fv(i.location,A.hiddenLineColor.asArray()),o.depthFunc(o.GREATER),o.depthMask(!1),o.multiDrawElements(o.LINES,g,0,o.UNSIGNED_INT,I,0,C),o.depthFunc(o.LEQUAL),o.depthMask(!0),o.uniform1i(e.location,0));else{const{drawId:a}=A.unifs;for(let A=0;A<C;A++)o.uniform1i(a.location,A),o.drawElements(o.LINES,g[A],o.UNSIGNED_INT,I[A]);if(t){o.uniform1i(e.location,1),o.uniform4fv(i.location,A.hiddenLineColor.asArray()),o.depthFunc(o.GREATER),o.depthMask(!1);for(let A=0;A<C;A++)o.uniform1i(a.location,A),o.drawElements(o.LINES,g[A],o.UNSIGNED_INT,I[A]);o.depthFunc(o.LEQUAL),o.depthMask(!0),o.uniform1i(e.location,0)}}}}var GeomType$1;!function(A){A[A.TRIANGLES=0]="TRIANGLES",A[A.LINES=1]="LINES",A[A.POINTS=2]="POINTS"}(GeomType$1||(GeomType$1={}));class GLMeshItemSet extends GLGeomItemSetMultiDraw{multiDraw(A,g,I,C){const e=this.gl,i=()=>{if(e.multiDrawElements)e.multiDrawElements(e.TRIANGLES,g,0,e.UNSIGNED_INT,I,0,C);else{const{drawId:i}=A.unifs;for(let A=0;A<C;A++)e.uniform1i(i.location,A),e.drawElements(e.TRIANGLES,g[A],e.UNSIGNED_INT,I[A])}},{geomType:t,outlineThickness:o,viewportSize:a,renderMode:B}=A.unifs,s=A instanceof ColorRenderState&&B?A.renderMode:null,l=A instanceof ColorRenderState&&o&&a&&"geometry"==A.outlineMethod&&A.outlineThickness>0&&"flat-noedges"!=s&&"pbr-noedges"!=s,n=l&&"wireframe"==s;if(n?(e.enable(e.STENCIL_TEST),e.clearStencil(0),e.clear(e.STENCIL_BUFFER_BIT),e.stencilOpSeparate(e.FRONT,e.DECR_WRAP,e.DECR_WRAP,e.DECR_WRAP),e.stencilOpSeparate(e.BACK,e.INCR_WRAP,e.INCR_WRAP,e.INCR_WRAP),e.stencilFunc(e.ALWAYS,0,255),e.enable(e.CULL_FACE),e.cullFace(e.BACK),e.disable(e.DEPTH_TEST),e.depthMask(!1),e.colorMask(!1,!1,!1,!1)):"hiddenline"==s&&e.colorMask(!1,!1,!1,!0),t&&e.uniform1i(t.location,GeomType$1.TRIANGLES),o&&e.uniform1f(o.location,0),i(),l){const g=A;if(e.enable(e.CULL_FACE),e.cullFace(e.FRONT),e.uniform1f(o.location,g.outlineThickness*window.devicePixelRatio),e.uniform2f(a.location,A.region[2]-A.region[0],A.region[3]-A.region[1]),"hiddenline"==s&&e.colorMask(!0,!0,!0,!0),n||(e.enable(e.BLEND),e.blendEquation(e.FUNC_ADD),e.blendFuncSeparate(e.SRC_ALPHA,e.ONE_MINUS_SRC_ALPHA,e.ONE,e.ONE)),i(),e.disable(e.CULL_FACE),e.cullFace(e.BACK),n){e.enable(e.DEPTH_TEST),e.depthMask(!0),e.colorMask(!0,!0,!0,!0),e.stencilFunc(e.NOTEQUAL,0,255),e.stencilOp(e.KEEP,e.KEEP,e.KEEP),e.enable(e.BLEND),e.blendEquation(e.FUNC_ADD),e.blendFuncSeparate(e.SRC_ALPHA,e.ONE_MINUS_SRC_ALPHA,e.ONE,e.ONE);const A=g.glShader,I=g.shaderkey,C=g.screenQuad;C.bindShader(g),C.draw(g,g.outlineColor),A.bind(g,I),this.renderer.glGeomItemLibrary.bind(g),this.renderer.glGeomLibrary.bind(g),this.renderer.glMaterialLibrary.bind(g),e.disable(e.STENCIL_TEST)}}}}class GLGeomItemSet extends EventEmitter{gl;glGeom;id;glGeomItems;glgeomItems_freeIndices;glgeomItemEventHandlers;drawIdsArray=null;drawIdsBuffer=null;drawIdsBufferDirty;highlightedIdsArray=null;highlightedIdsBuffer=null;highlightedIdsBufferDirty;visibleItems;highlightedItems;constructor(A,g){super(),this.gl=A,this.glGeom=g,this.id=g?g.getGeom().getId():this.getId(),this.glGeomItems=[],this.glgeomItems_freeIndices=[],this.glgeomItemEventHandlers=[],this.drawIdsArray=null,this.drawIdsBuffer=null,this.drawIdsBufferDirty=!0,this.highlightedIdsArray=null,this.highlightedIdsBuffer=null,this.highlightedIdsBufferDirty=!0,this.visibleItems=[],this.highlightedItems=[]}getGLGeom(){return this.glGeom}getDrawCount(){return this.visibleItems.length}addGLGeomItem(A){let g;if(this.glgeomItems_freeIndices.length>0?g=this.glgeomItems_freeIndices.pop():(g=this.glGeomItems.length,this.glGeomItems.push(null)),A.geomItem.isVisible()){this.visibleItems.push(g);const A=new CountChangedEvent(1,this.visibleItems.length);this.emit("drawCountChanged",A)}A.geomItem.isHighlighted()&&(this.highlightedItems.push(g),this.highlightedIdsBufferDirty=!0);const I={};I.highlightChanged=I=>{if(A.geomItem.isHighlighted()){if(this.highlightedItems.includes(g))return;this.highlightedItems.push(g);const A=new CountChangedEvent(1,this.highlightedItems.length);this.emit("highlightedCountChanged",A)}else{this.highlightedItems.splice(this.highlightedItems.indexOf(g),1);const A=new CountChangedEvent(-1,this.highlightedItems.length);this.emit("highlightedCountChanged",A)}this.highlightedIdsBufferDirty=!0},A.geomItem.on("highlightChanged",I.highlightChanged),I.visibilityChanged=A=>{if(A.state){this.visibleItems.push(g);const A=new CountChangedEvent(1,this.visibleItems.length);this.emit("drawCountChanged",A)}else{this.visibleItems.splice(this.visibleItems.indexOf(g),1);const A=new CountChangedEvent(-1,this.visibleItems.length);this.emit("drawCountChanged",A)}this.drawIdsBufferDirty=!0},A.geomItem.on("visibilityChanged",I.visibilityChanged),this.glGeomItems[g]=A,this.glgeomItemEventHandlers[g]=I,this.drawIdsBufferDirty=!0,A.GLGeomItemSet=this}removeGLGeomItem(A){const g=this.glGeomItems.indexOf(A),I=this.glgeomItemEventHandlers[g];if(A.geomItem.off("highlightChanged",I.highlightChanged),A.geomItem.off("visibilityChanged",I.visibilityChanged),this.glGeomItems[g]=null,this.glgeomItemEventHandlers[g]=null,A.GLGeomItemSet=null,this.glgeomItems_freeIndices.push(g),A.geomItem.isVisible()){this.visibleItems.splice(this.visibleItems.indexOf(g),1);const A=new CountChangedEvent(-1,this.visibleItems.length);this.emit("drawCountChanged",A)}if(A.geomItem.isHighlighted()){this.highlightedItems.splice(this.highlightedItems.indexOf(g),1);const A=new CountChangedEvent(-1,this.highlightedItems.length);this.emit("highlightedCountChanged",A)}this.drawIdsBufferDirty=!0,this.glGeomItems.length==this.glgeomItems_freeIndices.length&&this.destroy()}updateDrawIDsBuffer(){const A=this.gl;A.floatTexturesSupported?(this.drawIdsBuffer&&this.glGeomItems.length!=this.drawIdsArray.length&&(this.gl.deleteBuffer(this.drawIdsBuffer),this.drawIdsBuffer=null),this.drawIdsBuffer||(this.drawIdsBuffer=A.createBuffer(),A.bindBuffer(A.ARRAY_BUFFER,this.drawIdsBuffer)),A.bindBuffer(A.ARRAY_BUFFER,this.drawIdsBuffer),A.bufferData(A.ARRAY_BUFFER,this.getDrawIdsArray(),A.STATIC_DRAW),this.drawIdsBufferDirty=!1):this.drawIdsBufferDirty=!1}getDrawIdsArray(){return this.drawIdsBufferDirty&&(this.drawIdsArray&&this.glGeomItems.length==this.drawIdsArray.length||(this.drawIdsArray=new Float32Array(this.glGeomItems.length)),this.visibleItems.forEach(((A,g)=>{this.drawIdsArray[g]=this.glGeomItems[A].geomItemId})),this.drawIdsBufferDirty=!1),this.drawIdsArray}updateHighlightedIDsBuffer(){const A=this.gl;A.floatTexturesSupported?(this.highlightedIdsBuffer&&this.glGeomItems.length>this.highlightedIdsArray.length&&(this.gl.deleteBuffer(this.highlightedIdsBuffer),this.highlightedIdsBuffer=null),this.highlightedIdsBuffer||(this.highlightedIdsBuffer=A.createBuffer()),A.bindBuffer(A.ARRAY_BUFFER,this.highlightedIdsBuffer),A.bufferData(A.ARRAY_BUFFER,this.getHighlightedIdsArray(),A.STATIC_DRAW),this.highlightedIdsBufferDirty=!1):this.highlightedIdsBufferDirty=!1}getHighlightedIdsArray(){return this.highlightedIdsBufferDirty&&((!this.highlightedIdsArray||this.highlightedItems.length>this.highlightedIdsArray.length)&&(this.highlightedIdsArray=new Float32Array(this.glGeomItems.length)),this.highlightedItems.forEach(((A,g)=>{this.highlightedIdsArray[g]=this.glGeomItems[A].geomItemId})),this.highlightedIdsBufferDirty=!1),this.highlightedIdsArray}draw(A){0!=this.visibleItems.length&&(this.drawIdsBufferDirty&&this.updateDrawIDsBuffer(),this.bindAndRender(A,this.visibleItems,this.drawIdsBuffer))}drawHighlighted(A){0!=this.highlightedItems.length&&(this.highlightedIdsBufferDirty&&this.updateHighlightedIDsBuffer(),this.bindAndRender(A,this.highlightedItems,this.highlightedIdsBuffer))}drawGeomData(A){0!=this.visibleItems.length&&(this.drawIdsBufferDirty&&this.updateDrawIDsBuffer(),this.bindAndRender(A,this.visibleItems,this.drawIdsBuffer))}bindAndRender(A,g,I){const C=this.gl,e=A.unifs;if(A.glGeom!=this.glGeom&&(this.glGeom.bind(A),A.glGeom=this.glGeom),C.floatTexturesSupported&&C.drawElementsInstanced&&A.supportsInstancing){A.unifs.instancedDraw&&C.uniform1i(A.unifs.instancedDraw.location,1);const i=A.attrs.instancedIds.location;C.enableVertexAttribArray(i),C.bindBuffer(C.ARRAY_BUFFER,I),C.vertexAttribPointer(i,1,C.FLOAT,!1,4,0),C.vertexAttribDivisor(i,1),A.bindViewports(e,(()=>{this.glGeom.drawInstanced(A,g.length)}))}else A.unifs.instancedDraw&&C.uniform1i(A.unifs.instancedDraw.location,0),g.forEach((g=>{this.glGeomItems[g].bind(A),A.bindViewports(e,(()=>{this.glGeom.draw(A)}))}))}destroy(){this.drawIdsBuffer&&(this.gl.deleteBuffer(this.drawIdsBuffer),this.drawIdsBuffer=null),this.highlightedIdsBuffer&&(this.gl.deleteBuffer(this.highlightedIdsBuffer),this.highlightedIdsBuffer=null),this.emit("destructing")}}var vert$a="\nprecision highp float;\n#define GLSLIFY 1\nattribute vec3 positions;  //(location = 0)\n\n/* VS Outputs */\nvarying vec2 v_texCoord;\n \nvoid main()\n{\n  v_texCoord = positions.xy+0.5;\n  gl_Position = vec4(positions.xy*2.0, -1.0, 1.0);\n}\n",frag$9="\nprecision highp float;\n#define GLSLIFY 1\n\nuniform sampler2D colorTexture;\nuniform sampler2D depthTexture;\nuniform vec2 screenSize;\n\nuniform int pointAndLinePickingSize;\n\nvarying vec2 v_texCoord;\n\nbool sampleNeiPixel(vec2 fragCoord, inout vec4 res) {\n  res = texture2D(colorTexture, fragCoord/screenSize);\n  if (res.a > 0.0) {\n    return true;\n  }\n  return false;\n}\n\nvec4 samplePixels(vec2 fragCoord, int dist) {\n  vec4 res = vec4(0.0);\n  \n  \n  // Search surrounding pixels for geoms\n  if (sampleNeiPixel(fragCoord+vec2( dist, 0), res)) return res; // E\n  if (sampleNeiPixel(fragCoord+vec2(-dist, 0), res)) return res; // W\n  if (sampleNeiPixel(fragCoord+vec2( 0, dist), res)) return res; // N\n  if (sampleNeiPixel(fragCoord+vec2( 0,-dist), res)) return res; // S\n  if (sampleNeiPixel(fragCoord+vec2( dist, dist), res)) return res; // NW\n  if (sampleNeiPixel(fragCoord+vec2(-dist, dist), res)) return res; // NE\n  if (sampleNeiPixel(fragCoord+vec2( dist,-dist), res)) return res; // SW\n  if (sampleNeiPixel(fragCoord+vec2(-dist,-dist), res)) return res; // SE\n  \n  return res;\n}\n\n#ifdef ENABLE_ES3\n  out vec4 fragColor;\n#endif\nvoid main(void) {\n#ifndef ENABLE_ES3\n  vec4 fragColor;\n#endif\n\n  sampleNeiPixel(gl_FragCoord.xy, fragColor);\n  if (fragColor.a < 0.0001 && pointAndLinePickingSize > 1) {\n    for (int i=1; i<pointAndLinePickingSize; i++) {\n      fragColor = samplePixels(gl_FragCoord.xy, i);\n      if (fragColor.a > 0.0001) {\n        break;\n      } \n    }\n  }\n\n  if (fragColor.a < 0.0001) {\n    discard; \n  }\n\n#ifndef ENABLE_ES3\n  gl_FragColor = fragColor;\n#endif\n}\n";class FattenLinesShader extends GLShader{constructor(A){super(A,"FattenLinesShader"),this.setShaderStage("VERTEX_SHADER",vert$a),this.setShaderStage("FRAGMENT_SHADER",frag$9)}}const deepEquals=(A,g)=>A.length==g.length&&!A.some(((A,I)=>A!=g[I]));var GeomType;!function(A){A[A.TRIANGLES=0]="TRIANGLES",A[A.LINES=1]="LINES",A[A.POINTS=2]="POINTS"}(GeomType||(GeomType={}));class GLGeomItemSetMultiDrawCompoundGeom extends EventEmitter{renderer;gl;glGeomItems=[];glGeomIdsMapping={};glgeomItemEventHandlers=[];freeIndices=[];dirtyGeomItems=new Set;drawIdsBufferDirty=!0;drawIdsArraysAllocators={};drawIdsArrays={};drawIdsTextures={};drawElementCounts={};drawElementOffsets={};drawOrderToIndex=[];indexToDrawIndex=[];highlightedItems={};highlightedIdsArraysAllocators={};highlightElementCounts={};highlightElementOffsets={};highlightedIdsArray={};highlightedIdsTextures={};dirtyHighlightedGeomItems=new Set;highlightedIdsBufferDirty=!0;linesGeomDataBuffer=null;fattenLinesShader=null;quad=null;fbo=null;constructor(A){super(),this.renderer=A,this.gl=A.gl,this.renderer.glGeomLibrary.on("geomDataChanged",(A=>{const g=this.glGeomIdsMapping[A.index];null!=g&&g.forEach((A=>{this.dirtyGeomItems.add(A),this.drawIdsBufferDirty||(this.drawIdsBufferDirty=!0,this.emit("updated"))}))}))}addItemToHighlight(A,g){const I=g.indexOf(":");let C=[];-1!=I&&(C=g.substring(I+1).split(",").map((A=>Number.parseInt(A)))),this.highlightedItems[A]&&deepEquals(this.highlightedItems[A],C)||(this.highlightedItems[A]=C,this.highlightedIdsBufferDirty=!0,this.emit("updated"))}addGLGeomItem(A){const g=this.freeIndices.length>0?this.freeIndices.pop():this.glGeomItems.length;this.indexToDrawIndex[g]=this.drawOrderToIndex.length,this.drawOrderToIndex.push(g),this.dirtyGeomItems.add(g);const I={};I.visibilityChanged=A=>{this.dirtyGeomItems.add(g),this.drawIdsBufferDirty=!0,this.emit("updated")},A.on("visibilityChanged",I.visibilityChanged),A.geomItem.isHighlighted()&&this.addItemToHighlight(g,A.geomItem.getHighlightName()),I.highlightChanged=A=>{A&&A.name?this.addItemToHighlight(g,A.name):(delete this.highlightedItems[g],this.highlightedIdsBufferDirty=!0,this.emit("updated"))},A.geomItem.on("highlightChanged",I.highlightChanged),I.shatterStateChanged=A=>{this.dirtyGeomItems.add(g),this.drawIdsBufferDirty=!0,this.renderer.renderGeomDataFbos()},A.on("shatterStateChanged",I.shatterStateChanged);let C=A.geomItem.geomParam.value;A.geomId=this.renderer.glGeomLibrary.addGeom(C),this.glGeomIdsMapping[A.geomId]?this.glGeomIdsMapping[A.geomId].push(g):this.glGeomIdsMapping[A.geomId]=[g],this.glGeomItems[g]=A,this.glgeomItemEventHandlers[g]=I,this.drawIdsBufferDirty=!0,this.emit("updated")}removeGLGeomItem(A){const g=this.glGeomItems.indexOf(A),I=this.glGeomIdsMapping[A.geomId];I.splice(I.indexOf(g),1),0==I.length&&delete this.glGeomIdsMapping[A.geomId],this.renderer.glGeomLibrary.removeGeom(A.geomId);const C=this.glgeomItemEventHandlers[g];A.geomItem.off("highlightChanged",C.highlightChanged),A.off("visibilityChanged",C.visibilityChanged),this.glGeomItems[g]=null,this.glgeomItemEventHandlers[g]=null,this.freeIndices.push(g),this.dirtyGeomItems.has(g)&&this.dirtyGeomItems.delete(g);for(let A in this.drawIdsArraysAllocators){const I=this.drawIdsArraysAllocators[A].getAllocation(g);if(I){for(let g=0;g<I.size;g++)this.drawElementOffsets[A][I.start+g]=0,this.drawElementCounts[A][I.start+g]=0;this.drawIdsArraysAllocators[A].deallocate(g)}}const e=this.drawOrderToIndex.indexOf(g);this.drawOrderToIndex.splice(e,1),this.indexToDrawIndex[g]=-1,this.drawIdsBufferDirty=!0,A.geomItem.isHighlighted()&&(delete this.highlightedItems[g],this.highlightedIdsBufferDirty=!0),this.emit("updated")}updateDrawIDsBuffer(A){this.dirtyGeomItems.forEach((A=>{const g=this.glGeomItems[A];if(g){const I=this.renderer.glGeomLibrary.getGeom(g.geomId).genBuffers();let C={};if(g.shattered)for(let A in I.subGeomCounts)C[A]=I.subGeomCounts[A].length;else for(let A in I.materialSubGeoms)C[A]=I.materialSubGeoms[A].length;for(let g in C){const I=C[g];let e=this.drawIdsArraysAllocators[g];if(e){const C=e.getAllocation(A);if(C){if(C.size==I)continue;for(let A=0;A<C.size;A++)this.drawElementCounts[g][C.start+A]=0}}else e=new Allocator1D,this.drawIdsArraysAllocators[g]=e;e.allocate(A,I)}}}));for(let A in this.drawIdsArraysAllocators){const g=this.drawIdsArraysAllocators[A];if(!this.drawElementCounts[A]||g.reservedSpace>this.drawElementCounts[A].length-1){const I=new Float32Array(4*g.reservedSpace),C=new Int32Array(g.reservedSpace+1),e=new Int32Array(g.reservedSpace+1);this.drawElementCounts[A]&&(I.set(this.drawIdsArrays[A],0),C.set(this.drawElementOffsets[A],0),e.set(this.drawElementCounts[A],0)),this.drawIdsArrays[A]=I,this.drawElementOffsets[A]=C,this.drawElementCounts[A]=e}}this.dirtyGeomItems.forEach((A=>{const g=this.glGeomItems[A];if(!g)return;const I=this.renderer.glGeomLibrary.getGeomOffsetAndCount(g.geomId),C=this.renderer.glGeomLibrary.getGeomBuffers(g.geomId),e=g.isVisible();if(g.shattered){let i=0;const t=(t,o,a)=>{const B=this.drawIdsArraysAllocators[a],s=this.drawIdsArrays[a],l=this.drawElementOffsets[a],n=this.drawElementCounts[a],d=B.getAllocation(A);if(!d)return;const r=C.materials;for(let A=0;A<t.length;A++){const a=d.start+A;l[a]=I[0]+4*t[A],n[a]=e?o[A]:0,s[4*a+0]=g.geomItemId,s[4*a+1]=i+1;const B=C.subGeomMaterialIndices[i];if(r.length>0&&B>0&&B<=C.materials.length){const A=C.materials[B-1],g=this.renderer.glMaterialLibrary.getMaterialAllocation(A);s[4*a+2]=g.start}else s[4*a+2]=0;s[4*a+3]=0,i++}};t(C.subGeomOffsets.TRIANGLES,C.subGeomCounts.TRIANGLES,"TRIANGLES"),t(C.subGeomOffsets.LINES,C.subGeomCounts.LINES,"LINES"),t(C.subGeomOffsets.POINTS,C.subGeomCounts.POINTS,"POINTS")}else{const i=(i,t)=>{const o=this.drawIdsArraysAllocators[t],a=this.drawIdsArrays[t],B=this.drawElementOffsets[t],s=this.drawElementCounts[t],l=o.getAllocation(A);if(!l)return;const n=C.materials;for(let A=0;A<i.length;A++){const t=i[A],o=l.start+A;if(B[o]=I[0]+4*t.offset,s[o]=e?t.count:0,a[4*o+0]=g.geomItemId,a[4*o+1]=0,n.length>0&&t.materialId>=0&&C.materials[t.materialId]){const A=C.materials[t.materialId],g=this.renderer.glMaterialLibrary.getMaterialAllocation(A);a[4*o+2]=g.start}else a[4*o+2]=0;a[4*o+3]=0}};for(let g in C.materialSubGeoms){if(!this.drawIdsArraysAllocators[g].getAllocation(A))continue;i(C.materialSubGeoms[g],g)}}}));const g=this.renderer.gl;let I=!1;const C=C=>{const e=this.drawIdsArrays[C];let i=this.drawIdsTextures[C];const t=this.drawIdsArraysAllocators[C].reservedSpace,o=A.boundTextures++;g.activeTexture(g.TEXTURE0+o);const a=MathFunctions.nextPow2(Math.ceil(Math.sqrt(t)));if(i){if(i.width<a||i.height<a){i.resize(a,a);for(let A=0;A<this.drawOrderToIndex.length;A++)this.dirtyGeomItems.add(A);I=!0}}else i=new GLTexture2D(this.gl,{format:"RGBA",type:"FLOAT",width:a,height:a,filter:"NEAREST",wrap:"CLAMP_TO_EDGE",mipMapped:!1}),this.drawIdsTextures[C]=i,I=!0;{const A=i,t=i.width;g.bindTexture(g.TEXTURE_2D,A.glTex);const o=0,a=0,B=1,s=A.format,l=A.type;if(I){const A=this.drawIdsArraysAllocators[C].allocatedSpace,I=Math.ceil((a+A)/t);let i=0,n=A,d=a;for(let A=0;A<I;A++){let A;d+n>t?(A=t-d,d=0):A=n;const I=i%t,C=Math.floor(i/t),a=e.subarray(4*i,4*(i+A));if(a.length!=4*A)throw new Error("Invalid drawIds subarray :"+a.length+" width:"+A);g.texSubImage2D(g.TEXTURE_2D,o,I,C,A,B,s,l,a),i+=A,n-=A}}else{const A=this.drawIdsArraysAllocators[C];this.dirtyGeomItems.forEach((I=>{const C=A.getAllocation(I);if(!C)return;const i=C.start,a=C.size,n=i%t,d=Math.ceil((n+a)/t);let r=0,c=a,h=n;for(let A=0;A<d;A++){let A;h+c>t?(A=t-h,h=0):A=c;const I=(i+r)%t,C=Math.floor((i+r)/t),a=e.subarray(4*(i+r),4*(i+r+A));if(a.length!=4*A)throw new Error("Invalid drawIds subarray :"+a.length+" width:"+A);g.texSubImage2D(g.TEXTURE_2D,o,I,C,A,B,s,l,a),r+=A,c-=A}}))}}g.bindTexture(g.TEXTURE_2D,null),A.boundTextures--};for(let A in this.drawIdsArrays)C(A);this.dirtyGeomItems=new Set,this.drawIdsBufferDirty=!1}updateHighlightedIDsBuffer(A){if(this.highlightedIdsBufferDirty){this.highlightedIdsArraysAllocators={},this.highlightedIdsArray={},this.highlightElementOffsets={},this.highlightElementCounts={};for(let A in this.highlightedItems){const g=Number.parseInt(A),I=this.highlightedItems[A],C=this.glGeomItems[g],e=this.renderer.glGeomLibrary.getGeomBuffers(C.geomId);let i={TRIANGLES:0,LINES:0,POINTS:0};if(I.length>0)I.forEach((A=>{if(A<e.subGeomCounts.TRIANGLES.length)i.TRIANGLES++;else{const g=A-e.subGeomCounts.TRIANGLES.length;if(g<e.subGeomCounts.LINES.length)i.LINES++;else{g-e.subGeomCounts.LINES.length<e.subGeomCounts.POINTS.length&&i.POINTS++}}}));else for(let A in e.counts)e.counts[A]>0&&(i[A]=1);for(let A in i){const I=i[A];0!=I&&(this.highlightedIdsArraysAllocators[A]||(this.highlightedIdsArraysAllocators[A]=new Allocator1D),this.highlightedIdsArraysAllocators[A].allocate(g,I))}}for(let A in this.highlightedIdsArraysAllocators){const g=this.highlightedIdsArraysAllocators[A];(!this.highlightElementCounts[A]||g.reservedSpace>this.highlightElementCounts[A].length-1)&&(this.highlightedIdsArray[A]=new Float32Array(4*g.reservedSpace),this.highlightElementOffsets[A]=new Int32Array(g.reservedSpace+1),this.highlightElementCounts[A]=new Int32Array(g.reservedSpace+1))}const A=4;for(let g in this.highlightedItems){const I=Number.parseInt(g),C=this.highlightedItems[g],e=this.glGeomItems[I],i=this.renderer.glGeomLibrary.getGeomOffsetAndCount(e.geomId),t=this.renderer.glGeomLibrary.getGeomBuffers(e.geomId);if(0!=C.length){const g={TRIANGLES:0,LINES:0,POINTS:0};C.forEach((C=>{const o=(g,t,o,a,B)=>{const s=this.highlightedIdsArraysAllocators[a],l=this.highlightedIdsArray[a],n=this.highlightElementOffsets[a],d=this.highlightElementCounts[a],r=s.getAllocation(I).start+g;n[r]=i[0]+t[B]*A,d[r]=o[B],l[4*r+0]=e.geomItemId,l[4*r+1]=C+1,l[4*r+2]=0,l[4*r+3]=0};if(C<t.subGeomCounts.TRIANGLES.length)o(g.TRIANGLES,t.subGeomOffsets.TRIANGLES,t.subGeomCounts.TRIANGLES,"TRIANGLES",C),g.TRIANGLES++;else{const A=C-t.subGeomCounts.TRIANGLES.length;if(A<t.subGeomCounts.LINES.length)o(g.LINES,t.subGeomOffsets.LINES,t.subGeomCounts.LINES,"LINES",A),g.LINES++;else{const I=A-t.subGeomCounts.LINES.length;I<t.subGeomCounts.POINTS.length&&(o(g.POINTS,t.subGeomOffsets.POINTS,t.subGeomCounts.POINTS,"POINTS",I),g.POINTS++)}}}))}else for(let g in t.offsets){const C=t.counts[g];if(0==C)continue;const o=t.offsets[g],a=this.highlightedIdsArraysAllocators[g].getAllocation(I).start;this.highlightElementOffsets[g][a]=i[0]+o*A,this.highlightElementCounts[g][a]=C,this.highlightedIdsArray[g][4*a+0]=e.geomItemId}}this.highlightedIdsBufferDirty=!1}const g=this.renderer.gl,I=I=>{const C=this.highlightedIdsArray[I];if(!C||0==C.length)return;let e=this.highlightedIdsTextures[I];const i=this.highlightedIdsArraysAllocators[I].reservedSpace,t=A.boundTextures++;g.activeTexture(g.TEXTURE0+t);const o=MathFunctions.nextPow2(Math.ceil(Math.sqrt(i)));e?(e.width<o||e.height<o)&&e.resize(o,o):(e=new GLTexture2D(this.gl,{format:"RGBA",type:"FLOAT",width:o,height:o,filter:"NEAREST",wrap:"CLAMP_TO_EDGE",mipMapped:!1}),this.highlightedIdsTextures[I]=e);{const A=e,i=e.width;g.bindTexture(g.TEXTURE_2D,A.glTex);const t=0,o=0,a=1,B=A.format,s=A.type,l=this.highlightedIdsArraysAllocators[I].allocatedSpace,n=Math.ceil((o+l)/i);let d=0,r=l,c=o;for(let A=0;A<n;A++){let A;c+r>i?(A=i-c,c=0):A=r;const I=d%i,e=Math.floor(d/i),o=C.subarray(4*d,4*(d+A));if(o.length!=4*A)throw new Error("Invalid drawIds subarray :"+o.length+" width:"+A);g.texSubImage2D(g.TEXTURE_2D,t,I,e,A,a,B,s,o),d+=A,r-=A}}g.bindTexture(g.TEXTURE_2D,null),A.boundTextures--};for(let A in this.highlightedIdsArray)I(A)}draw(A){if(0==this.drawOrderToIndex.length)return;this.drawIdsBufferDirty&&this.updateDrawIDsBuffer(A),A.pushGLStack("GLGeomItemSetMultiDrawCompoundGeom.draw");const g=this.drawIdsArrays,I=this.drawElementCounts,C=this.drawElementOffsets,e=this.drawIdsTextures,i=this.drawIdsArraysAllocators,t=this.gl,o=A.unifs,a=t.getParameter(t.DEPTH_FUNC),{drawIdsTexture:B,geomType:s,outlineThickness:l,viewportSize:n,occluded:d,renderMode:r}=A.unifs,c=A instanceof ColorRenderState&&r?A.renderMode:null,h="flat-noedges"!=c&&"shaded-noedges"!=c&&"pbr-noedges"!=c,G=A instanceof ColorRenderState&&l&&n&&"geometry"==A.outlineMethod&&A.outlineThickness>0&&h,m=G&&"wireframe"==c,b=A.hiddenLineColor&&A.hiddenLineColor.a>0&&d;if(m?(t.enable(t.STENCIL_TEST),t.clearStencil(0),t.clear(t.STENCIL_BUFFER_BIT),t.stencilOpSeparate(t.FRONT,t.DECR_WRAP,t.DECR_WRAP,t.DECR_WRAP),t.stencilOpSeparate(t.BACK,t.INCR_WRAP,t.INCR_WRAP,t.INCR_WRAP),t.stencilFunc(t.ALWAYS,0,255),t.enable(t.CULL_FACE),t.cullFace(t.BACK),t.disable(t.DEPTH_TEST),t.depthMask(!1),t.colorMask(!1,!1,!1,!1)):"hiddenline"==c?t.colorMask(!1,!1,!1,!0):t.disable(t.CULL_FACE),g.TRIANGLES&&i.TRIANGLES.allocatedSpace>0){if(e.TRIANGLES.bindToUniform(A,B),s&&t.uniform1i(s.location,GeomType.TRIANGLES),t.enable(t.POLYGON_OFFSET_FILL),t.polygonOffset(1,1),l&&t.uniform1f(l.location,0),A.bindViewports(o,(()=>{this.multiDrawMeshes(A,I.TRIANGLES,C.TRIANGLES,i.TRIANGLES.allocatedSpace)})),G){if(t.enable(t.CULL_FACE),t.cullFace(t.FRONT),t.uniform1f(l.location,this.renderer.outlineThickness*window.devicePixelRatio),t.uniform2f(n.location,A.region[2]-A.region[0],A.region[3]-A.region[1]),"hiddenline"==c&&t.colorMask(!0,!0,!0,!0),m||(A.glEnable(t.BLEND),t.blendEquation(t.FUNC_ADD),t.blendFuncSeparate(t.SRC_ALPHA,t.ONE_MINUS_SRC_ALPHA,t.ONE,t.ONE)),A.bindViewports(o,(()=>{this.multiDrawMeshes(A,I.TRIANGLES,C.TRIANGLES,i.TRIANGLES.allocatedSpace)})),t.disable(t.CULL_FACE),t.cullFace(t.BACK),m){t.enable(t.DEPTH_TEST),t.depthMask(!0),t.colorMask(!0,!0,!0,!0),t.stencilFunc(t.NOTEQUAL,0,255),t.stencilOp(t.KEEP,t.KEEP,t.KEEP),A.glEnable(t.BLEND),t.blendEquation(t.FUNC_ADD),t.blendFuncSeparate(t.SRC_ALPHA,t.ONE_MINUS_SRC_ALPHA,t.ONE,t.ONE);const g=A.glShader,I=A.shaderkey,C=this.renderer.screenQuad;C.bindShader(A),C.draw(A,this.renderer.outlineColor),g.bind(A,I),this.renderer.glGeomItemLibrary.bind(A),this.renderer.glGeomLibrary.bind(A),this.renderer.glMaterialLibrary.bind(A),t.disable(t.STENCIL_TEST)}}else A.glEnable(t.BLEND),t.blendEquation(t.FUNC_ADD),t.blendFuncSeparate(t.SRC_ALPHA,t.ONE_MINUS_SRC_ALPHA,t.ONE,t.ONE);t.disable(t.POLYGON_OFFSET_FILL)}if(h&&g.LINES&&i.LINES.allocatedSpace>0&&(e.LINES.bindToUniform(A,B),s&&t.uniform1i(s.location,GeomType.LINES),A.bindViewports(o,(()=>{this.multiDrawLines(A,I.LINES,C.LINES,i.LINES.allocatedSpace)})),b)){const{hiddenLineColor:g}=A.unifs;t.uniform1i(d.location,1),t.uniform4fv(g.location,A.hiddenLineColor.asArray()),t.depthFunc(t.GREATER),t.depthMask(!1),A.bindViewports(o,(()=>{this.multiDrawLines(A,I.LINES,C.LINES,i.LINES.allocatedSpace)})),t.depthFunc(a),t.depthMask(!0),t.uniform1i(d.location,0)}if(g.POINTS&&i.POINTS.allocatedSpace>0&&(e.POINTS.bindToUniform(A,B),s&&t.uniform1i(s.location,GeomType.POINTS),A.bindViewports(o,(()=>{this.multiDrawPoints(A,I.POINTS,C.POINTS,i.POINTS.allocatedSpace)})),b)){const{hiddenLineColor:g}=A.unifs;t.uniform1i(d.location,1),t.uniform4fv(g.location,A.hiddenLineColor.asArray()),t.depthFunc(t.GREATER),t.depthMask(!1),A.bindViewports(o,(()=>{this.multiDrawPoints(A,I.POINTS,C.POINTS,i.POINTS.allocatedSpace)})),t.depthFunc(a),t.depthMask(!0),t.uniform1i(d.location,0)}s&&t.uniform1i(s.location,0),A.popGLStack()}drawGeomData(A){this.drawIdsBufferDirty&&this.updateDrawIDsBuffer(A),A.pushGLStack("GLGeomItemSetMultiDrawCompoundGeom.drawGeomData");const g=this.renderer.gl,I=A.unifs,{drawIdsTexture:C,geomType:e}=I,i=this.drawElementCounts,t=this.drawElementOffsets,o=this.drawIdsTextures,a=this.drawIdsArraysAllocators,B=this.drawIdsArrays;"wireframe"!=this.renderer.renderMode&&A.bindViewports(I,(()=>{B.TRIANGLES&&a.TRIANGLES.allocatedSpace>0&&(o.TRIANGLES.bindToUniform(A,C),e&&g.uniform1i(e.location,GeomType.TRIANGLES),g.enable(g.POLYGON_OFFSET_FILL),g.polygonOffset(1,1),this.multiDrawMeshes(A,i.TRIANGLES,t.TRIANGLES,a.TRIANGLES.allocatedSpace),g.disable(g.POLYGON_OFFSET_FILL))}));if(A.geomDataFbo){this.linesGeomDataBuffer||(this.linesGeomDataBuffer=new GLTexture2D(g,{type:this.renderer.floatGeomBuffer?"FLOAT":"UNSIGNED_BYTE",format:"RGBA",filter:"NEAREST",width:1,height:2}),this.fattenLinesShader=new FattenLinesShader(g),this.quad=new GLMesh(g,new Plane(1,1)));const I=A.geomDataFbo,C=I.width,e=I.height;if(this.linesGeomDataBuffer.width!=C||this.linesGeomDataBuffer.height!=e){this.fbo&&(g.deleteFramebuffer(this.fbo),this.fbo=null),this.linesGeomDataBuffer.resize(C,e),this.fbo=g.createFramebuffer();const A=this.linesGeomDataBuffer.glTex,i=I.depthTexture;g.bindFramebuffer(g.DRAW_FRAMEBUFFER,this.fbo),g.framebufferTexture2D(g.DRAW_FRAMEBUFFER,g.COLOR_ATTACHMENT0,g.TEXTURE_2D,A,0),g.framebufferTexture2D(g.DRAW_FRAMEBUFFER,g.DEPTH_ATTACHMENT,g.TEXTURE_2D,i,0),checkFramebuffer(g,C,e)}else g.bindFramebuffer(g.DRAW_FRAMEBUFFER,this.fbo);g.colorMask(!0,!0,!0,!0),g.clearColor(0,0,0,0),g.clear(g.COLOR_BUFFER_BIT)}if(A.bindViewports(I,(()=>{B.LINES&&a.LINES.allocatedSpace>0&&(o.LINES.bindToUniform(A,C),e&&g.uniform1i(e.location,GeomType.LINES),this.multiDrawLines(A,i.LINES,t.LINES,a.LINES.allocatedSpace)),B.POINTS&&a.POINTS.allocatedSpace>0&&(o.POINTS.bindToUniform(A,C),e&&g.uniform1i(e.location,GeomType.POINTS),this.multiDrawPoints(A,i.POINTS,t.POINTS,a.POINTS.allocatedSpace))})),this.linesGeomDataBuffer&&A.geomDataFbo){A.boundRendertarget=null,A.geomDataFbo.bindForWriting(A);const I=A.glShader,C=A.shaderkey;this.fattenLinesShader.bind(A),g.disable(g.DEPTH_TEST);const{colorTexture:e,screenSize:i,pointAndLinePickingSize:t}=A.unifs;this.linesGeomDataBuffer.bindToUniform(A,e);const o=A.geomDataFbo;g.uniform2f(i.location,o.width,o.height),g.uniform1i(t.location,A.pointAndLinePickingSize),this.quad.bindAndDraw(A),g.enable(g.DEPTH_TEST),I.bind(A,C),this.renderer.glGeomItemLibrary.bind(A),this.renderer.glGeomLibrary.bind(A),this.renderer.glMaterialLibrary.bind(A)}A.popGLStack()}drawHighlighted(A){if(0==Object.keys(this.highlightedItems).length)return;this.highlightedIdsBufferDirty&&this.updateHighlightedIDsBuffer(A),A.pushGLStack("GLGeomItemSetMultiDrawCompoundGeom.drawHighlighted");const g=this.highlightedIdsArray,I=this.highlightElementCounts,C=this.highlightElementOffsets,e=this.highlightedIdsTextures,i=this.highlightedIdsArraysAllocators,t=A.unifs,o=this.renderer.gl;A.glDisable(o.CULL_FACE);const{drawIdsTexture:a,geomType:B}=A.unifs;A.bindViewports(t,(()=>{g.TRIANGLES&&i.TRIANGLES.allocatedSpace>0&&(e.TRIANGLES.bindToUniform(A,a),o.enable(o.POLYGON_OFFSET_FILL),o.polygonOffset(1,1),B&&o.uniform1i(B.location,GeomType.TRIANGLES),this.multiDrawMeshes(A,I.TRIANGLES,C.TRIANGLES,i.TRIANGLES.allocatedSpace),o.disable(o.POLYGON_OFFSET_FILL)),g.LINES&&i.LINES.allocatedSpace>0&&(e.LINES.bindToUniform(A,a),B&&o.uniform1i(B.location,GeomType.LINES),this.multiDrawLines(A,I.LINES,C.LINES,i.LINES.allocatedSpace)),g.POINTS&&i.POINTS.allocatedSpace>0&&(e.POINTS.bindToUniform(A,a),B&&o.uniform1i(B.location,GeomType.POINTS),this.multiDrawPoints(A,I.POINTS,C.POINTS,i.POINTS.allocatedSpace))})),B&&o.uniform1i(B.location,0),A.popGLStack()}multiDrawMeshes(A,g,I,C){const e=this.gl;if(e.multiDrawElements)e.multiDrawElements(e.TRIANGLES,g,0,e.UNSIGNED_INT,I,0,C);else{const{drawId:i}=A.unifs;for(let A=0;A<C;A++)e.uniform1i(i.location,A),e.drawElements(e.TRIANGLES,g[A],e.UNSIGNED_INT,I[A])}}multiDrawLines(A,g,I,C){const e=this.gl;if(e.multiDrawElements)e.multiDrawElements(e.LINES,g,0,e.UNSIGNED_INT,I,0,C);else{const{drawId:i}=A.unifs;for(let A=0;A<C;A++)e.uniform1i(i.location,A),e.drawElements(e.LINES,g[A],e.UNSIGNED_INT,I[A])}}multiDrawPoints(A,g,I,C){const e=this.gl;if(e.multiDrawElements)e.multiDrawElements(e.POINTS,g,0,e.UNSIGNED_INT,I,0,C);else{const{drawId:i}=A.unifs;for(let A=0;A<C;A++)e.uniform1i(i.location,A),e.drawElements(e.POINTS,g[A],e.UNSIGNED_INT,I[A])}}sortItems(A){this.drawIdsBufferDirty}destroy(){for(let A in this.drawIdsTextures)this.drawIdsTextures[A].destroy();for(let A in this.highlightedIdsTextures)this.highlightedIdsTextures[A].destroy();this.emit("destructing")}}class GLShaderGeomSets extends EventEmitter{renderer;gl;glShader;glGeomItemSets={};constructor(A,g,I){super(),this.renderer=A,this.gl=g,this.glShader=I}getOrCreateGLGeomItemSet(A){let g;if(A instanceof CompoundGeom){if(this.glGeomItemSets.CompoundGeom)return this.glGeomItemSets.CompoundGeom;g=new GLGeomItemSetMultiDrawCompoundGeom(this.renderer),this.glGeomItemSets.CompoundGeom=g}else if(A instanceof Mesh||A instanceof MeshProxy){if(this.glGeomItemSets.GLMesh)return this.glGeomItemSets.GLMesh;g=new GLMeshItemSet(this.renderer),this.glGeomItemSets.GLMesh=g}else if(A instanceof Lines||A instanceof LinesProxy){if(this.glGeomItemSets.GLLines)return this.glGeomItemSets.GLLines;g=new GLLinesItemSet(this.renderer),this.glGeomItemSets.GLLines=g}else{if(!(A instanceof Points||A instanceof PointsProxy))throw new Error("Unsupported geom type:"+A.constructor.name);if(this.glGeomItemSets.GLPoints)return this.glGeomItemSets.GLPoints;g=new GLPointsItemSet(this.renderer),this.glGeomItemSets.GLPoints=g}return g.on("updated",(()=>{this.emit("updated")})),g}addGLGeomItem(A){const g=A.geomItem.geomParam.value,I=this.getOrCreateGLGeomItemSet(g);A.GLGeomItemSet=I,I.addGLGeomItem(A)}removeGLGeomItem(A){A.GLGeomItemSet.removeGLGeomItem(A),A.GLGeomItemSet=null}bindShader(A,g,I){const C=this.gl;if(A.isCompiledForTarget(I)||(C.multiDrawElements?g.directives.push("#define ENABLE_MULTI_DRAW\n#extension GL_ANGLE_multi_draw : enable"):g.directives.push("#define ENABLE_MULTI_DRAW"),A.compileForTarget(I,g.directives),g.directives.pop()),!A.bind(g,I))throw new Error("Unable to bind shader:"+A);this.renderer.glGeomItemLibrary.bind(g),this.renderer.glGeomLibrary.bind(g),this.renderer.glMaterialLibrary.bind(g)}draw(A){this.bindShader(this.glShader,A,"multidraw-draw");for(const g in this.glGeomItemSets)this.glGeomItemSets[g].draw(A);this.glShader.unbind(A)}drawHighlightedGeoms(A){this.bindShader(this.glShader,A,"multidraw-highlight");for(const g in this.glGeomItemSets)this.glGeomItemSets[g].drawHighlighted(A);this.glShader.unbind(A)}drawGeomData(A){this.bindShader(this.glShader,A,"multidraw-geomdata");const g=A.gl,{passId:I}=A.unifs;I&&g.uniform1i(I.location,A.passIndex);for(const g in this.glGeomItemSets)this.glGeomItemSets[g].drawGeomData(A);this.glShader.unbind(A)}sortItems(A){for(const g in this.glGeomItemSets)this.glGeomItemSets[g].sortItems(A)}}class GLMaterialGeomItemSets extends EventEmitter{pass;gl;glMaterial;glGeomItemSets=new Map;listenerIds={};drawCount=0;constructor(A,g){if(super(),this.pass=A,this.gl=A.renderer.gl,g){this.glMaterial=g;const A=g.material,I=()=>{A.off("opacityChanged",I),this.glGeomItemSets.forEach((A=>{for(const g of A.glGeomItems){const A=g.geomItem;this.pass.removeGeomItem(A),this.pass.renderer.assignTreeItemToGLPass(A)}}))};this.listenerIds["material.opacityChanged"]=A.on("opacityChanged",I)}}addGLGeomItem(A,g){let I=this.glGeomItemSets.get(g);I||(I=new GLGeomItemSet(this.gl,g),this.addGeomItemSet(I)),I.addGLGeomItem(A)}drawCountChanged(A){this.drawCount+=A.change,this.emit("updated")}addGeomItemSet(A){this.glGeomItemSets.set(A.getGLGeom(),A);const g=A.on("drawCountChanged",(A=>{this.drawCountChanged(A)}));A.once("destructing",(()=>{if(A.off("drawCountChanged",g),this.glGeomItemSets.delete(A.getGLGeom()),0==this.glGeomItemSets.size&&this.glMaterial){this.glMaterial.material.off("opacityChanged",this.listenerIds["material.opacityChanged"]),this.emit("destructing")}}))}draw(A){if(0==this.drawCount)return;this.glMaterial&&this.glMaterial.bind(A,!0),this.glGeomItemSets.forEach((g=>{g.draw(A)})),this.glMaterial&&this.glMaterial.unbind(A)}drawHighlighted(A){this.glMaterial&&this.glMaterial.bind(A,!1),this.glGeomItemSets.forEach((g=>{g.drawHighlighted(A)})),this.glMaterial&&this.glMaterial.unbind(A)}drawGeomData(A){this.glMaterial&&this.glMaterial.bind(A,!1),this.glGeomItemSets.forEach((g=>{g.drawGeomData(A)})),this.glMaterial&&this.glMaterial.unbind(A)}}class GLShaderMaterials extends EventEmitter{gl;pass;glShader;glShaderKey="";glShaderGeomDataKey="";glShaderHighlightKey="";glMaterialGeomItemSets=new Map;constructor(A,g,I){super(),this.gl=A,this.pass=g,this.glShader=I,this.glShaderKey="draw",this.glShaderGeomDataKey="geomdata",this.glShaderHighlightKey="highlight"}getOrCreateGLMaterialGeomItemSets(A){let g=this.glMaterialGeomItemSets.get(A);return g||(g=new GLMaterialGeomItemSets(this.pass,A),this.addMaterialGeomItemSets(A,g)),g}addGLGeomItem(A,g,I){if(I.material.isTextured()){this.getOrCreateGLMaterialGeomItemSets(I).addGLGeomItem(A,g)}else{this.getOrCreateGLMaterialGeomItemSets(null).addGLGeomItem(A,g)}}addMaterialGeomItemSets(A,g){this.glMaterialGeomItemSets.set(A,g);const I=()=>{this.emit("updated")},C=()=>{g.off("updated",I),g.off("destructing",C),this.glMaterialGeomItemSets.delete(A),0==this.glMaterialGeomItemSets.size&&this.emit("destructing")};g.on("updated",I),g.on("destructing",C)}removeMaterialGeomItemSets(A){this.glMaterialGeomItemSets.delete(A)}draw(A){const g=this.glShader;this.glShader.bind(A,this.glShaderKey)&&(this.pass.renderer.glGeomItemLibrary.bind(A),this.pass.renderer.glMaterialLibrary.bind(A),this.glMaterialGeomItemSets.forEach((g=>{g.draw(A)})),g.unbind(A))}drawHighlightedGeoms(A){this.glShader.bind(A,this.glShaderHighlightKey)&&(this.pass.renderer.glGeomItemLibrary.bind(A),this.pass.renderer.glMaterialLibrary.bind(A),this.glMaterialGeomItemSets.forEach((g=>{g.drawHighlighted(A)})),this.glShader.unbind(A))}drawGeomData(A){if(!this.glShader.bind(A,this.glShaderGeomDataKey))return;this.pass.renderer.glGeomItemLibrary.bind(A),this.pass.renderer.glMaterialLibrary.bind(A);const g=this.gl,{passId:I}=A.unifs;I&&g.uniform1i(I.location,A.passIndex),this.glMaterialGeomItemSets.forEach((g=>{g.drawGeomData(A)})),this.glShader.unbind(A)}}var frag$8="\nprecision highp float;\nprecision highp int;\n#define GLSLIFY 1\n\nimport 'imageAtlas.glsl'\n\nuniform sampler2D atlasBillboards;\n\n/* VS Outputs */\nvarying float v_instanceID;\nvarying vec2 v_texCoord;\nvarying vec4 v_layoutData;\nvarying float v_alpha;\nvarying vec4 v_tint;\nvarying vec3 v_viewPos;\nvarying float v_frontFacing;\n\nuniform sampler2D instancesTexture;\nuniform int instancesTextureSize;\n\n#if defined(DRAW_GEOMDATA)\n  uniform int isOrthographic;\n  import 'surfaceGeomData.glsl'\n#endif // DRAW_GEOMDATA\n\nconst int cols_per_instance = 7;\n\nvec4 getHilightColor(int id) {\n  return fetchTexel(instancesTexture, instancesTextureSize, (id * cols_per_instance) + 6);\n}\n\n#ifdef ENABLE_ES3\n  out vec4 fragColor;\n#endif\nvoid main(void) {\n#ifndef ENABLE_ES3\n  vec4 fragColor;\n#endif\n\n  int instanceID = int(v_instanceID);\n\n  vec2 texCoord = v_texCoord;\n\n  bool frontFacing = bool(v_frontFacing);\n  if (frontFacing && !gl_FrontFacing) {\n    texCoord.x = 1.0 - texCoord.x;\n  }\n  texCoord *= v_layoutData.zw;\n  texCoord += v_layoutData.xy;\n\n  vec4 imageColor = texture2D(atlasBillboards, texCoord) * v_tint;\n  imageColor.a *= v_alpha;\n  if(imageColor.a < 0.1)\n    discard;\n\n#if defined(DRAW_COLOR)\n  fragColor = imageColor;\n  // fragColor.r = 1.0;\n  // fragColor.a = 1.0;\n#elif defined(DRAW_GEOMDATA)\n  fragColor = setFragColor_geomData(v_viewPos, v_instanceID, 0.0, isOrthographic, 0, TRIANGLES);\n#elif defined(DRAW_HIGHLIGHT)\n  fragColor = getHilightColor(instanceID);\n  // Skip unhilighting labels.\n  if(fragColor.r < 0.001 && fragColor.g < 0.001 && fragColor.b < 0.001)\n    discard;\n#endif // DRAW_HIGHLIGHT\n  \n#ifndef ENABLE_ES3\n  gl_FragColor = fragColor;\n#endif\n}\n",vert$9="\nprecision highp float;\nprecision highp int;\n#define GLSLIFY 1\n\nimport 'quadVertexFromID.glsl'\n\nuniform mat4 viewMatrix;\nuniform mat4 projectionMatrix;\nuniform mat4 cameraMatrix;\nuniform int isOrthographic;\nuniform vec4 viewportFrustum;\n\nimport 'GLSLUtils.glsl'\n\n#ifdef ENABLE_FLOAT_TEXTURES\n\n// A sorted attribute of instance Ids so we draw from back to front.\ninstancedattribute float instanceIds;\n\nimport 'transpose.glsl'\nimport 'imageAtlas.glsl'\n\nuniform sampler2D atlasBillboards_layout;\nuniform vec4 atlasBillboards_desc;\n\nuniform sampler2D instancesTexture;\nuniform int instancesTextureSize;\n\nconst int cols_per_instance = 7;\n\nconst int ALIGN_TO_CAMERA_FLAG = 1 << 2;\nconst int ALIGN_TO_CAMERA_AND_AXIS_FLAG = 1 << 3;\nconst int DRAW_ON_TOP_FLAG = 1 << 4;\nconst int FIXED_SIZE_ON_SCREEN_FLAG = 1 << 5;\nconst int FRONT_FACING_FLAG = 1 << 6;\n\nmat4 getMatrix(sampler2D texture, int textureSize, int index) {\n  // Unpack 3 x 4 matix columns into a 4 x 4 matrix.\n  vec4 col0 = fetchTexel(texture, textureSize, (index * cols_per_instance) + 0);\n  vec4 col1 = fetchTexel(texture, textureSize, (index * cols_per_instance) + 1);\n  vec4 col2 = fetchTexel(texture, textureSize, (index * cols_per_instance) + 2);\n  mat4 result = mat4(col0, col1, col2, vec4(0.0, 0.0, 0.0, 1.0));\n  return transpose(result);\n  // return mat4(1.0);\n}\n\nmat4 getModelMatrix(int id) {\n  return getMatrix(instancesTexture, instancesTextureSize, id);\n}\nvec4 getInstanceData(int id) {\n  return fetchTexel(instancesTexture, instancesTextureSize, (id * cols_per_instance) + 3);\n}\nvec4 getPivot(int id) {\n  return fetchTexel(instancesTexture, instancesTextureSize, (id * cols_per_instance) + 4);\n}\nvec4 getTintColor(int id) {\n  return fetchTexel(instancesTexture, instancesTextureSize, (id * cols_per_instance) + 5);\n}\n\n#else\n\nuniform vec4 atlasBillboards_desc;\n\nuniform mat4 modelMatrix;\nuniform vec2 pivot;\nuniform vec4 billboardData;\nuniform vec4 tintColor;\nuniform vec4 layoutData;\n\n#endif\n\nuniform int inVR;\n\nmat4 calcAlignToCameraMatrix(mat4 modelMatrix, mat4 cameraMatrix) {\n  vec3 cameraY = vec3(cameraMatrix[1][0], cameraMatrix[1][1], cameraMatrix[1][2]);\n  vec3 cameraZ = vec3(cameraMatrix[2][0], cameraMatrix[2][1], cameraMatrix[2][2]);\n\n  vec3 modelOrigin = vec3(modelMatrix[3][0], modelMatrix[3][1], modelMatrix[3][2]);\n\n  vec3 zAxis = normalize(cameraZ);\n  vec3 xAxis = normalize(cross(cameraY, zAxis));\n  vec3 yAxis = normalize(cross(zAxis, xAxis));\n\n  return mat4(vec4(xAxis, 0.0), vec4(yAxis, 0.0), vec4(zAxis, 0.0), vec4(modelOrigin, 1.0));\n}\n  \n\nmat4 calcAlignToCameraAndXAxisMatrix(mat4 modelMatrix, mat4 cameraMatrix) {\n  vec3 cameraPos = vec3(cameraMatrix[3][0], cameraMatrix[3][1], cameraMatrix[3][2]);\n  vec3 cameraY = vec3(cameraMatrix[1][0], cameraMatrix[1][1], cameraMatrix[1][2]);\n  vec3 cameraZ = vec3(cameraMatrix[2][0], cameraMatrix[2][1], cameraMatrix[2][2]);\n\n  vec3 modelOrigin = vec3(modelMatrix[3][0], modelMatrix[3][1], modelMatrix[3][2]);\n  vec3 modelXAxis = normalize(vec3(modelMatrix[0][0], modelMatrix[0][1], modelMatrix[0][2]));\n  vec3 modelToCamera = normalize(cameraPos - modelOrigin);\n\n  vec3 zAxis;\n  if (isOrthographic > 0) {\n    zAxis = normalize(cameraZ);\n  } else {\n    zAxis = normalize(modelToCamera);\n  }\n  vec3 yAxis = normalize(cross(zAxis, modelXAxis));\n  vec3 xAxis = normalize(cross(zAxis, yAxis));\n\n  if (dot(cameraY, yAxis) < 0.0) {\n    yAxis = -yAxis;\n  }\n\n  return mat4(vec4(xAxis, 0.0), vec4(yAxis, 0.0), vec4(zAxis, 0.0), vec4(modelOrigin, 1.0));\n}\n\nfloat map(float value, float inMin, float inMax, float outMin, float outMax) {\n  return outMin + (outMax - outMin) * (value - inMin) / (inMax - inMin);\n}\n\n/* VS Outputs */\nvarying float v_instanceID;\nvarying vec2 v_texCoord;\nvarying vec4 v_layoutData;\nvarying float v_alpha;\nvarying vec4 v_tint;\nvarying vec3 v_viewPos;\nvarying float v_frontFacing;\n\nvoid main(void) {\n\n#ifdef ENABLE_FLOAT_TEXTURES\n\n  int instanceID = int(instanceIds);\n  v_instanceID = float(instanceID) + 0.25;\n\n  mat4 modelMatrix = getModelMatrix(instanceID);\n  vec2 pivot = getPivot(instanceID).xy;\n  vec4 billboardData = getInstanceData(instanceID);\n  vec4 layoutData = fetchTexel(atlasBillboards_layout, int(atlasBillboards_desc.z), int(billboardData.z));\n  v_tint = getTintColor(instanceID);\n\n#else\n\n  v_tint = tintColor;\n\n#endif\n\n  vec2 quadVertex = getQuadVertexPositionFromID();\n  \n  vec2 pos = quadVertex + vec2(0.5, 0.0) - pivot;\n\n  v_layoutData = layoutData;\n  v_texCoord = vec2(quadVertex.x, -quadVertex.y) + 0.5;\n  v_alpha = billboardData.w;\n\n  float scl = billboardData.x;\n  float width = layoutData.z * atlasBillboards_desc.x * scl;\n  float height = layoutData.w * atlasBillboards_desc.y * scl;\n  int flags = int(billboardData.y);\n\n  // Use cross platform bit flags methods\n  bool alignedToCamera = testFlag(flags, ALIGN_TO_CAMERA_FLAG);\n  bool alignedToCameraAndXAxis = testFlag(flags, ALIGN_TO_CAMERA_AND_AXIS_FLAG);\n  bool drawOnTop = testFlag(flags, DRAW_ON_TOP_FLAG);\n  bool fixedSizeOnscreen = testFlag(flags, FIXED_SIZE_ON_SCREEN_FLAG);\n  bool frontFacing = testFlag(flags, FRONT_FACING_FLAG);\n\n  v_frontFacing = float(frontFacing);\n\n  mat4 modelViewMatrix = viewMatrix * modelMatrix;\n\n  // Note: items in front of the camera will have a negative value here.\n  float sc = 1.0;\n  if (fixedSizeOnscreen) {\n    if (isOrthographic > 0){\n      // At a distance of 1, we should match the orthographic and perspective sizes.\n      // Calculate the size of the handle in perpective projection at 1 meter.\n      // With a 24mm camera, the Fov is 46.4 degrees, or 0.8098327729253689 radians.\n      // (0.81 / 2.0) * focalDist * 2.0\n      // The frustum height at 1m is 0.81.\n      sc = viewportFrustum.y / 0.8098327;\n    } else {\n      sc = -modelViewMatrix[3][2];\n      \n      if (inVR == 1) {\n        // During XR sessions, there is a scaling applied to the view matrix\n        // which causes a distortion to the line width. We extract that scale here\n        // and use to correct the distortion.\n        // See also: FatPointsShader\n        vec3 viewZ = modelViewMatrix[2].xyz;\n        float viewScale = length(viewZ);\n        sc /= viewScale;\n      }\n    }\n  }\n  \n  float aspect = viewportFrustum.x / viewportFrustum.y;\n  if (alignedToCamera || alignedToCameraAndXAxis) {\n    if (alignedToCamera) {\n      mat4 lookAt = calcAlignToCameraMatrix(modelMatrix, cameraMatrix);\n      modelViewMatrix = viewMatrix * lookAt;\n    }\n    else if (alignedToCameraAndXAxis) {\n      // Align the billboard to the camera, while preserving the X axis in the modelMatrix;\n      mat4 lookAt = calcAlignToCameraAndXAxisMatrix(modelMatrix, cameraMatrix);\n      modelViewMatrix = viewMatrix * lookAt;\n    }\n  }\n  v_viewPos =  (modelViewMatrix * vec4(pos.x * width * sc, (pos.y + 0.5) * height * sc, 0.0, 1.0)).xyz;\n  gl_Position = projectionMatrix * vec4(v_viewPos, 1.0);\n  \n\n  // Use cross platform bit flags methods\n  if (drawOnTop) {\n    if (isOrthographic > 0){\n      gl_Position.z = mix(gl_Position.z, -1.0, 0.5);\n    } else {\n      gl_Position.z = mix(gl_Position.z, -gl_Position.z, 0.5);\n    }\n  }\n}\n";class BillboardShader extends GLShader{constructor(A){super(A,"BillboardShader"),this.setShaderStage("VERTEX_SHADER",vert$9),this.setShaderStage("FRAGMENT_SHADER",frag$8)}}var vert$8="\nprecision highp float;\n#define GLSLIFY 1\n\nattribute vec3 positions;    //(location = 0)\n\nuniform mat4 viewMatrix;\nuniform mat4 projectionMatrix;\nuniform vec3 projectionCenter;\n\nimport 'transpose.glsl'\nimport 'GLSLUtils.glsl'\nimport 'stack-gl/transpose.glsl'\nimport 'stack-gl/inverse.glsl'\nimport 'geomItemId.glsl'\nimport 'drawItemTexture.glsl'\nimport 'modelMatrix.glsl'\n\n/* VS Outputs */\nvarying vec3 v_worldDir;\n \nvoid main()\n{\n  int geomItemId = getGeomItemId();\n  vec4 pos = vec4(positions, 1.);\n  mat4 modelMatrix = getModelMatrix(geomItemId);\n  mat4 modelViewProjectionMatrix = projectionMatrix * viewMatrix * modelMatrix;\n\n  gl_Position = modelViewProjectionMatrix * pos;\n\n  vec4 worldPos = modelMatrix * pos;\n  v_worldDir = worldPos.xyz - projectionCenter;\n}\n\n",OctahedralEnvProjectionFrag="\nprecision highp float;\n#define GLSLIFY 1\n\nimport 'GLSLUtils.glsl'\nimport 'envmap-octahedral.glsl'\nimport 'gamma.glsl'\nimport 'materialparams.glsl'\n\nuniform sampler2D envMap;\nuniform int envMapLoaded;\n\n#ifdef ENABLE_INLINE_GAMMACORRECTION\nuniform float exposure;\n#endif\n\nvec4 sampleEnvMap(vec3 dir) {\n  vec2 uv = dirToSphOctUv(dir);\n  vec4 texel = texture2D(envMap, vec2(uv.x, 1.0 - uv.y));\n  return vec4(texel.rgb/texel.a, 1.0); // TODO: Check this line. Do we need it?\n}\n\n/* VS Outputs */\nvarying vec3 v_worldDir;\n\n#ifdef ENABLE_ES3\n  out vec4 fragColor;\n#endif\nvoid main(void) {\n#ifndef ENABLE_ES3\n  vec4 fragColor;\n#endif\n\n  if (envMapLoaded == 0)  {\n    discard;\n    return;\n  }\n\n  fragColor = sampleEnvMap(normalize(v_worldDir));\n\n#ifdef ENABLE_INLINE_GAMMACORRECTION\n  fragColor.rgb = toGamma(fragColor.rgb * exposure);\n#endif\n\n#ifndef ENABLE_ES3\n  gl_FragColor = fragColor;\n#endif\n}\n";class EnvProjectionShader extends GLShader{constructor(A){super(A,"EnvProjectionShader"),this.setShaderStage("VERTEX_SHADER",vert$8),this.setShaderStage("FRAGMENT_SHADER",OctahedralEnvProjectionFrag)}bind(A,g){if(super.bind(A,g),A instanceof ColorRenderState){const g=A,I=this.__gl,{envMap:C,envMapLoaded:e,exposure:i}=A.unifs;g.envMap&&g.envMap.glTex?(g.envMap.glTex.bindToUniform(A,C),I.uniform1i(e.location,1)):I.uniform1i(e.location,0),i&&I.uniform1f(i.location,g.exposure)}return!0}static supportsInstancing(){return!1}static getMaterialTemplate(){return material$9}}const material$9=new EnvProjectionMaterial;Registry.register("EnvProjectionShader",EnvProjectionShader);var frag$7="\nprecision highp float;\n#define GLSLIFY 1\n\nuniform color BaseColor;\nuniform float Opacity;\nuniform mat4 cameraMatrix;\n\n#ifdef ENABLE_ES3\n  out vec4 fragColor;\n#endif\n\n/* VS Outputs */\nvarying vec3 v_viewPos;\nvarying vec3 v_viewNormal;\nvarying vec2 v_texCoord;\nvarying float v_geomItemId;\nvarying vec4 v_geomItemData;\nvarying float v_drawItemID;\nvarying vec3 v_worldPos;\n\nimport 'GLSLUtils.glsl'\nimport 'drawItemTexture.glsl'\nimport 'cutaways.glsl'\nimport 'geometryMask.glsl'\nimport 'GLSLBits.glsl'\nimport 'geomItemFlags.glsl'\n\nuniform int passId;\n\n#if defined(DRAW_HIGHLIGHT)\n  import 'surfaceHighlight.glsl'\n#endif\n\nvoid main(void) {\n#ifndef ENABLE_ES3\n  vec4 fragColor;\n#endif\n  int geomItemId = int(round(v_geomItemId));\n  int flags = int(round(v_geomItemData.x));\n  float treeItemOpacity = v_geomItemData.y;\n\n  // Cutaways\n  if (testFlag(flags, GEOMITEM_FLAG_CUTAWAY)) {\n    vec4 cutAwayData  = getCutaway(geomItemId);\n    vec3 planeNormal = cutAwayData.xyz;\n    float planeDist = cutAwayData.w;\n    if (length(planeNormal) > 0.5) {\n      if (cutaway(v_worldPos, planeNormal, planeDist)) {\n          discard;\n          return;\n      }\n    } else {\n      if (testGeometryMask(v_viewPos)) {\n        discard;\n        return;\n      }\n    }\n  }\n\n#if defined(DRAW_COLOR)\n  int debugLevel = 0;\n  if (debugLevel == 0) {\n\n    vec3 viewVector = mat3(cameraMatrix) * normalize(-v_viewPos);\n    vec3 normal = mat3(cameraMatrix) * v_viewNormal;\n    float NdotV = dot(normalize(normal), normalize(viewVector));\n\n    // Modulate the lighting using the texture coord so the line looks round.\n    NdotV *= cos((v_texCoord.x - 0.5) * 2.0);\n\n    vec4 color = BaseColor * NdotV;\n    fragColor = vec4(color.rgb, BaseColor.a * Opacity * treeItemOpacity);\n  }\n  else {\n    fragColor = vec4(v_texCoord.x, 0.0, 0.0, 1.0);\n  }\n#elif defined(DRAW_GEOMDATA)\n\n  if (testFlag(flags, GEOMITEM_INVISIBLE_IN_GEOMDATA)) {\n    discard;\n    return;\n  }\n\n  float dist = length(v_viewPos);\n  fragColor.r = float(passId); \n  fragColor.g = float(v_drawItemID);\n  fragColor.b = 0.0;// TODO: store poly-id or something.\n  fragColor.a = dist;\n    \n#elif defined(DRAW_HIGHLIGHT)\n  fragColor = setFragColor_highlight(v_geomItemId);\n#endif\n\n#ifndef ENABLE_ES3\n  gl_FragColor = fragColor;\n#endif\n}\n",vert$7="\nprecision highp float;\n#define GLSLIFY 1\n\ninstancedattribute vec2 segmentIndices;\nattribute float vertexIDs;\n\nuniform mat4 viewMatrix;\nuniform mat4 projectionMatrix;\nuniform int isOrthographic;\n\nimport 'drawItemTexture.glsl'\nimport 'modelMatrix.glsl'\n\nuniform int geomItemId;\nint getGeomItemId() {\n  return geomItemId;\n}\n\nuniform sampler2D positionsTexture;\nuniform int positionsTextureSize;\n\nuniform float LineThickness;\nuniform float Overlay;\n\nimport 'calcFatLinesViewPos.glsl'\n\nvarying vec3 v_viewPos;\nvarying vec3 v_viewNormal;\nvarying vec2 v_texCoord;\n\nvarying float v_geomItemId;\nvarying vec4 v_geomItemData;\nvarying float v_drawItemID;\nvarying vec3 v_worldPos;\n\nvoid main(void) {\n\n  int geomItemId = getGeomItemId();\n  v_geomItemId = float(geomItemId);\n  v_geomItemData = getInstanceData(geomItemId);\n\n  int vertexID = int(vertexIDs);\n\n  mat4 modelMatrix = getModelMatrix(geomItemId);\n  mat4 modelViewMatrix = viewMatrix * modelMatrix;\n\n  vec3 pos;\n  v_viewPos       = calcFatLinesViewPos(vertexID, modelViewMatrix, v_viewNormal, v_texCoord, pos);\n  v_worldPos      = (modelMatrix * vec4(pos, 1.0)).xyz;\n  v_drawItemID    = float(getGeomItemId());\n\n  gl_Position     = projectionMatrix * vec4(v_viewPos, 1.0);\n\n  if (isOrthographic > 0){\n    gl_Position.z -= mix(gl_Position.z, -1.0, Overlay);\n  } else {\n    gl_Position.z = mix(gl_Position.z, -gl_Position.z, Overlay);\n  }\n\n}\n";class FatLinesShader extends GLShader{constructor(A){super(A,"FatLinesShader"),this.setShaderStage("VERTEX_SHADER",vert$7),this.setShaderStage("FRAGMENT_SHADER",frag$7)}bind(A,g){return!!super.bind(A,g)&&(A.supportsInstancing=!1,!0)}static supportsInstancing(){return!1}static getMaterialTemplate(){return material$8}}const material$8=new FatLinesMaterial("FatLinesShader_template");Registry.register("FatLinesShader",FatLinesShader);var vert$6="\nprecision highp float;\nprecision highp int;\n#define GLSLIFY 1\n\nattribute vec3 positions;\n#ifdef ENABLE_TEXTURES\nattribute vec2 texCoords;\n#endif\n\nuniform mat4 viewMatrix;\nuniform mat4 projectionMatrix;\nuniform int isOrthographic;\n\nimport 'GLSLUtils.glsl'\nimport 'geomItemId.glsl'\nimport 'drawItemTexture.glsl'\nimport 'modelMatrix.glsl'\nimport 'geomItemFlags.glsl'\nimport 'materialparams.glsl'\n\n/* VS Outputs */\nvarying vec4 v_drawItemIds;\nvarying vec4 v_geomItemData;\nvarying vec3 v_viewPos;\n#ifdef ENABLE_TEXTURES\nvarying vec2 v_textureCoord;\n#endif\nvarying vec3 v_worldPos;\n\nvoid main(void) {\n  v_drawItemIds = getDrawItemIds();\n  int geomItemId = int(round(v_drawItemIds.x));\n  v_geomItemData  = getInstanceData(geomItemId);\n\n  mat4 modelMatrix = getModelMatrix(geomItemId);\n  mat4 modelViewMatrix = viewMatrix * modelMatrix;\n\n  vec4 pos = vec4(positions, 1.);\n  vec4 viewPos = (modelViewMatrix * pos);\n  gl_Position = projectionMatrix * viewPos;\n\n  v_viewPos = viewPos.xyz;\n#ifdef ENABLE_TEXTURES\n  v_textureCoord = texCoords;\n  v_textureCoord.y = 1.0 - v_textureCoord.y;// Flip y\n#endif\n\n  //////////////////////////////////////////////\n  // Overlay\n  vec2 materialCoords = v_geomItemData.zw;\n  vec4 materialValue1 = getMaterialValue(materialCoords, 2);\n  float overlay = materialValue1.x;\n\n  if (isOrthographic > 0){\n    gl_Position.z -= overlay;\n  } else {\n    gl_Position.z = mix(gl_Position.z, -gl_Position.z, overlay);\n  }\n\n  //////////////////////////////////////////////\n  \n  v_worldPos      = (modelMatrix * pos).xyz;\n}\n",frag$6="\nprecision highp float;\nprecision highp int;\n#define GLSLIFY 1\n\nimport 'GLSLUtils.glsl'\nimport 'drawItemTexture.glsl'\nimport 'cutaways.glsl'\nimport 'geometryMask.glsl'\nimport 'gamma.glsl'\nimport 'materialparams.glsl'\nimport 'geomItemFlags.glsl'\n\n#ifdef DEBUG_GEOM_ID\nimport 'debugColors.glsl'\n#endif\n\n#ifdef ENABLE_TEXTURES\nuniform sampler2D BaseColorTex;\nuniform int BaseColorTexType;\n#endif\n\n/* VS Outputs */\nvarying vec4 v_drawItemIds;\nvarying vec4 v_geomItemData;\nvarying vec3 v_viewPos;\n#ifdef ENABLE_TEXTURES\nvarying vec2 v_textureCoord;\n#endif\nvarying vec3 v_worldPos;\n\n#ifdef ENABLE_ES3\nout vec4 fragColor;\n#endif\n\n#if defined(DRAW_GEOMDATA)\n  uniform int isOrthographic;\n  import 'surfaceGeomData.glsl'\n#elif defined(DRAW_HIGHLIGHT)\n  import 'surfaceHighlight.glsl'\n#endif // DRAW_HIGHLIGHT\n\nvoid main(void) {\n#ifndef ENABLE_ES3\n  vec4 fragColor;\n#endif\n\n  int geomItemId = int(round(v_drawItemIds.x));\n  int flags = int(round(v_geomItemData.x));\n  float treeItemOpacity = v_geomItemData.y;\n\n  // We can make geoms invisible to hide them. \n  // Avoid drawing GeomData for geoms that are completely transparent.\n  if (treeItemOpacity < 0.001) {\n    discard;\n    return;\n  }\n  \n  // Cutaways\n  if (testFlag(flags, GEOMITEM_FLAG_CUTAWAY)) \n  {\n    vec4 cutAwayData   = getCutaway(geomItemId);\n    vec3 planeNormal = cutAwayData.xyz;\n    float planeDist = cutAwayData.w;\n    if (length(planeNormal) > 0.5) {\n      if (cutaway(v_worldPos, planeNormal, planeDist)) {\n          discard;\n          return;\n      }\n    } else {\n      if (testGeometryMask(v_viewPos)) {\n        discard;\n        return;\n      }\n    }\n  }\n\n  //////////////////////////////////////////////\n  // Material\n\n  vec2 materialCoords = v_geomItemData.zw;\n  if (v_drawItemIds.z > 0.5) {\n    materialCoords.x = v_drawItemIds.z;\n  }\n  vec4 baseColor = getMaterialValue(materialCoords, 1);\n\n#ifdef ENABLE_TEXTURES\n  getTextureColorValue(baseColor, BaseColorTex, BaseColorTexType, v_textureCoord);\n#endif // ENABLE_TEXTURES\n\n  baseColor.a *= treeItemOpacity;\n\n#if defined(DRAW_COLOR)\n\n  //////////////////////////////////////////////\n  fragColor = baseColor;\n\n#ifdef DEBUG_GEOM_ID\n  // ///////////////////////\n  // Debug Draw ID (this correlates to GeomID within a GLGeomSet)\n  float geomId = v_geomItemData.w;\n  fragColor.rgb = getDebugColor(geomId);\n  // ///////////////////////\n#endif\n\n#ifdef ENABLE_INLINE_GAMMACORRECTION\n  fragColor.rgb = toGamma(fragColor.rgb);\n#endif\n\n#elif defined(DRAW_GEOMDATA)\n\n  fragColor = setFragColor_geomData(v_viewPos, v_drawItemIds.x, v_drawItemIds.y, isOrthographic, flags, TRIANGLES);\n#elif defined(DRAW_HIGHLIGHT)\n  fragColor = setFragColor_highlight(v_drawItemIds.x);\n#endif // DRAW_HIGHLIGHT\n\n#ifndef ENABLE_ES3\n  gl_FragColor = fragColor;\n#endif\n}\n";class FlatSurfaceShader extends GLShader{constructor(A){super(A,"FlatSurfaceShader"),this.setShaderStage("VERTEX_SHADER",vert$6),this.setShaderStage("FRAGMENT_SHADER",frag$6)}bind(A,g){super.bind(A,g),A.pushGLStack("FlatSurfaceShader.bind");const I=this.__gl;return A.glDisable(I.CULL_FACE),!0}unbind(A){return super.unbind(A),A.popGLStack(),!0}static getPackedMaterialData(A){const g=new Float32Array(12);let I=0;g[4*I+0]=2;const C=A.getParameter("BaseColor");let e;e=C instanceof MaterialColorParam&&C.colorSpace==ColorSpace.Gamma?C.value.toLinear():C.value,I++,g[4*I+0]=e.r,g[4*I+1]=e.g,g[4*I+2]=e.b,g[4*I+3]=e.a,I++;const i=A.getParameter("Overlay");return i&&(g[4*I+0]=i.getValue()),g}static getMaterialTemplate(){return material$7}}const material$7=new FlatSurfaceMaterial("FlatSurfaceShader_template");Registry.register("FlatSurfaceShader",FlatSurfaceShader);var vert$5="\nprecision highp float;\nprecision highp int;\n#define GLSLIFY 1\n\nattribute vec3 positions;\n\nimport 'GLSLUtils.glsl'\nimport 'geomItemId.glsl'\nimport 'drawItemTexture.glsl'\nimport 'modelMatrix.glsl'\nimport 'materialparams.glsl'\n\nuniform mat4 viewMatrix;\nuniform mat4 projectionMatrix;\nuniform int isOrthographic;\n\n/* VS Outputs */\nvarying float v_geomItemId;\nvarying vec4 v_geomItemData;\nvarying vec3 v_viewPos;\nvarying vec3 v_worldPos;\n\nvoid main(void) {\n  int geomItemId = getGeomItemId();\n  v_geomItemId = float(geomItemId);\n  v_geomItemData  = getInstanceData(geomItemId);\n\n  mat4 modelMatrix = getModelMatrix(geomItemId);\n  mat4 modelViewMatrix = viewMatrix * modelMatrix;\n  vec4 viewPos = modelViewMatrix * vec4(positions, 1.0);\n\n  v_viewPos = viewPos.xyz;\n  gl_Position = projectionMatrix * viewPos;\n\n  //////////////////////////////////////////////\n  // Overlay\n  vec2 materialCoords = v_geomItemData.zw;\n  vec4 materialValue1 = getMaterialValue(materialCoords, 2);\n  float overlay = materialValue1.y;\n   \n#if defined(DRAW_GEOMDATA)\n  float _overlay = mix(overlay, 1.0, 0.0001);\n#else\n  float _overlay = overlay;\n#endif\n\n  if (isOrthographic > 0){\n    gl_Position.z -= _overlay;\n  } else {\n    gl_Position.z = mix(gl_Position.z, -gl_Position.z, _overlay);\n  }\n  \n  vec4 pos = vec4(positions, 1.);\n  v_worldPos      = (modelMatrix * pos).xyz;\n}\n",frag$5="\nprecision highp float;\nprecision highp int;\n#define GLSLIFY 1\n\nimport 'GLSLUtils.glsl'\nimport 'drawItemTexture.glsl'\nimport 'cutaways.glsl'\nimport 'geometryMask.glsl'\nimport 'materialparams.glsl'\nimport 'geomItemFlags.glsl'\n\n#if defined(DRAW_COLOR)\n\nuniform int occluded;\nuniform vec4 hiddenLineColor;\n\n#elif defined(DRAW_GEOMDATA)\n\nuniform int isOrthographic;\nimport 'surfaceGeomData.glsl'\n\n#elif defined(DRAW_HIGHLIGHT)\n\n#ifdef ENABLE_FLOAT_TEXTURES\nvec4 getHighlightColor(int id) {\n  return fetchTexel(instancesTexture, instancesTextureSize, (id * pixelsPerItem) + 4);\n}\n#else // ENABLE_FLOAT_TEXTURES\n\nuniform vec4 highlightColor;\n\nvec4 getHighlightColor() {\n  return highlightColor;\n}\n\n#endif // ENABLE_FLOAT_TEXTURES\n\n#endif // DRAW_HIGHLIGHT\n\n/* VS Outputs */\nvarying float v_geomItemId;\nvarying vec4 v_geomItemData;\nvarying vec3 v_viewPos;\nvarying vec3 v_worldPos;\n\n#ifdef ENABLE_ES3\n  out vec4 fragColor;\n#endif\n\nvoid main(void) {\n#ifndef ENABLE_ES3\n  vec4 fragColor;\n#endif\n\n  int geomItemId = int(round(v_geomItemId));\n  int flags = int(round(v_geomItemData.x));\n  float treeItemOpacity = v_geomItemData.y;\n\n  // Cutaways\n  if (testFlag(flags, GEOMITEM_FLAG_CUTAWAY)) \n  {\n    vec4 cutAwayData   = getCutaway(geomItemId);\n    vec3 planeNormal = cutAwayData.xyz;\n    float planeDist = cutAwayData.w;\n    if (length(planeNormal) > 0.5) {\n      if (cutaway(v_worldPos, planeNormal, planeDist)) {\n          discard;\n          return;\n      }\n    } else {\n      if (testGeometryMask(v_viewPos)) {\n        discard;\n        return;\n      }\n    }\n  }\n\n  //////////////////////////////////////////////\n  // Material\n  vec2 materialCoords = v_geomItemData.zw;\n  vec4 BaseColor = getMaterialValue(materialCoords, 1);\n  vec4 matValue1 = getMaterialValue(materialCoords, 2);\n  vec4 matValue2 = getMaterialValue(materialCoords, 3);\n  float Opacity  = matValue1.r;\n\n  //////////////////////////////////////////////\n  // Color\n#if defined(DRAW_COLOR)\n\n  fragColor = BaseColor;\n  \n  if (occluded == 1) {\n    fragColor = hiddenLineColor;\n  }\n\n  fragColor.a *= Opacity * treeItemOpacity;\n\n  //////////////////////////////////////////////\n  // GeomData\n#elif defined(DRAW_GEOMDATA)\n\n  fragColor = setFragColor_geomData(v_viewPos, v_geomItemId, 0.0, isOrthographic, flags, LINES);\n  \n  //////////////////////////////////////////////\n  // Highlight\n#elif defined(DRAW_HIGHLIGHT)\n  \n  fragColor = getHighlightColor(geomItemId);\n\n#endif // DRAW_HIGHLIGHT\n\n#ifndef ENABLE_ES3\n  gl_FragColor = fragColor;\n#endif\n}\n";class LinesShader extends GLShader{constructor(A){super(A,"LinesShader"),this.setShaderStage("VERTEX_SHADER",vert$5),this.setShaderStage("FRAGMENT_SHADER",frag$5)}static getPackedMaterialData(A){const g=new Float32Array(12);let I=0;g[4*I+0]=3;const C=A.getParameter("BaseColor");let e;return e=C instanceof MaterialColorParam&&C.colorSpace==ColorSpace.Gamma?C.value.toLinear():C.value,I++,g[4*I+0]=e.r,g[4*I+1]=e.g,g[4*I+2]=e.b,g[4*I+3]=e.a,I++,g[4*I+0]=A.getParameter("Opacity").value,g[4*I+1]=A.getParameter("Overlay").value,g}static getMaterialTemplate(){return material$6}}const material$6=new LinesMaterial("LinesShader_template");Registry.register("LinesShader",LinesShader);var frag$4="\nprecision highp float;\n#define GLSLIFY 1\n\nimport 'GLSLUtils.glsl'\nimport 'drawItemTexture.glsl'\nimport 'cutaways.glsl'\nimport 'geometryMask.glsl'\nimport 'materialparams.glsl'\nimport 'geomItemFlags.glsl'\nimport 'surfaceGeomData.glsl'\n\n#if defined(DRAW_GEOMDATA)\n\nimport 'GLSLBits.glsl'\nuniform highp int isOrthographic;\n\n#elif defined(DRAW_HIGHLIGHT)\n\n#ifdef ENABLE_FLOAT_TEXTURES\nvec4 getHighlightColor(int id) {\n  return fetchTexel(instancesTexture, instancesTextureSize, (id * pixelsPerItem) + 4);\n}\n#else // ENABLE_FLOAT_TEXTURES\n\nuniform vec4 highlightColor;\n\nvec4 getHighlightColor() {\n  return highlightColor;\n}\n\n#endif // ENABLE_FLOAT_TEXTURES\n\n#endif // DRAW_HIGHLIGHT\n\n/* VS Outputs */\nvarying vec2 v_drawItemIds;\nvarying vec4 v_geomItemData;\nvarying vec3 v_viewPos;\n/* VS Outputs */\n\n#ifdef ENABLE_ES3\nout vec4 fragColor;\n#endif\n\nvoid main(void) {\n\n#ifndef ENABLE_ES3\n  vec4 fragColor;\n#endif\n\n  //////////////////////////////////////////////\n  // Color\n#if defined(DRAW_COLOR)\n\n  vec2 materialCoords = v_geomItemData.zw;\n  vec4 baseColor = getMaterialValue(materialCoords, 1);\n  vec4 matValue1 = getMaterialValue(materialCoords, 2);\n  float pointSize     = baseColor.a * matValue1.r;\n  float overlay       = matValue1.g;\n\n  fragColor = baseColor;\n\n  //////////////////////////////////////////////\n  // GeomData\n#elif defined(DRAW_GEOMDATA)\n\n  fragColor = setFragColor_geomData(v_viewPos, v_drawItemIds.x, v_drawItemIds.y, isOrthographic, 0, POINTS);\n\n  //////////////////////////////////////////////\n  // Highlight\n#elif defined(DRAW_HIGHLIGHT)\n  \n  int geomItemId = int(round(v_drawItemIds.x));\n  fragColor = getHighlightColor(geomItemId);\n\n#endif // DRAW_HIGHLIGHT\n\n#ifndef ENABLE_ES3\n  gl_FragColor = fragColor;\n#endif\n}\n",vert$4="\nprecision highp float;\n#define GLSLIFY 1\n\nattribute vec3 positions;\n\nimport 'GLSLUtils.glsl'\nimport 'geomItemId.glsl'\nimport 'drawItemTexture.glsl'\nimport 'modelMatrix.glsl'\nimport 'materialparams.glsl'\n\nuniform mat4 viewMatrix;\nuniform mat4 projectionMatrix;\nuniform int isOrthographic;\n\n/* VS Outputs */\nvarying vec2 v_drawItemIds;\nvarying vec4 v_geomItemData;\nvarying vec3 v_viewPos;\n\nvoid main(void) {\n  int geomItemId = getGeomItemId();\n  v_drawItemIds.x = float(geomItemId);\n  v_drawItemIds.y = float(gl_VertexID);\n  v_geomItemData  = getInstanceData(geomItemId);\n\n  mat4 modelMatrix = getModelMatrix(geomItemId);\n  mat4 modelViewMatrix = viewMatrix * modelMatrix;\n  \n  vec4 viewPos = modelViewMatrix * vec4(positions, 1.);\n  gl_Position = projectionMatrix * viewPos;\n  \n\n  //////////////////////////////////////////////\n  // Material\n  vec2 materialCoords = v_geomItemData.zw;\n  vec4 materialValue1 = getMaterialValue(materialCoords, 2);\n  float pointSize = materialValue1.x;\n  float overlay = materialValue1.y;\n\n  //////////////////////////////////////////////\n\n  // Note: as of 22/01/2021 gl_PointSize has stopped working again...\n  // It is working again now in Chrome\n  gl_PointSize = pointSize * 1.0 / length(viewPos.xyz);\n\n  if (isOrthographic > 0){\n    gl_Position.z -= overlay;\n  } else {\n    gl_Position.z = mix(gl_Position.z, -gl_Position.z, overlay);\n  }\n  \n  v_viewPos = -viewPos.xyz;\n}\n";class PointsShader extends GLShader{constructor(A){super(A,"PointsShader"),this.setShaderStage("VERTEX_SHADER",vert$4),this.setShaderStage("FRAGMENT_SHADER",frag$4)}static getPackedMaterialData(A){const g=new Float32Array(12);let I=0;g[4*I+0]=4;const C=A.getParameter("BaseColor");let e;return e=C instanceof MaterialColorParam&&C.colorSpace==ColorSpace.Gamma?C.value.toLinear():C.value,I++,g[4*I+0]=e.r,g[4*I+1]=e.g,g[4*I+2]=e.b,g[4*I+3]=e.a,I++,g[4*I+0]=A.getParameter("PointSize").value,g[4*I+1]=A.getParameter("Overlay").value,g}static getMaterialTemplate(){return material$5}}const material$5=new PointsMaterial("PointsShader_template");Registry.register("PointsShader",PointsShader);var vert$3="\nprecision highp float;\nprecision highp int;\n#define GLSLIFY 1\n\ninstancedattribute int drawIndices;\nuniform sampler2D pointsAttributes;\nuniform int texelsPerPoint;\n\nuniform mat4 viewMatrix;\nuniform mat4 projectionMatrix;\nuniform int isOrthographic;\n\nimport 'GLSLUtils.glsl' \nimport 'inverse.glsl'\nimport 'drawItemTexture.glsl'\nimport 'geomItemId.glsl'\nimport 'modelMatrix.glsl'\nimport 'quadVertexFromID.glsl'\nimport 'materialparams.glsl'\n\n/* VS Outputs */\nvarying vec4 v_drawItemIds;\nvarying vec4 v_geomItemData;\nvarying vec3 v_texCoord;\nvarying vec3 v_viewPos;\nvarying vec4 v_color;\n\nvoid main(void) {\n  int geomItemId = getGeomItemId();\n  v_geomItemData  = getInstanceData(geomItemId);\n\n  vec2 quadPointPos = getQuadVertexPositionFromID();\n  v_texCoord.xy = vec2(quadPointPos.x, -quadPointPos.y) + 0.5;\n\n  mat4 modelMatrix = getModelMatrix(geomItemId);\n  mat4 modelViewMatrix = viewMatrix * modelMatrix;\n\n  ivec2 texSize = textureSize(pointsAttributes, 0);\n  int texelIndex = drawIndices * texelsPerPoint;\n  ivec2 texelCoord = ivec2(\n    texelIndex % texSize.x, \n    texelIndex / texSize.x\n  );\n  vec4 texel0 = texelFetch(pointsAttributes, texelCoord, 0);\n\n  vec3 pos = texel0.xyz;\n  float size = texel0.w;\n\n  if (texelsPerPoint >= 2) {\n    vec4 texel1 = texelFetch(pointsAttributes, ivec2(texelCoord.x + 1, texelCoord.y), 0);\n    v_color = texel1;\n  } else {\n    v_color = vec4(1., 1., 1., 1.);\n  }\n\n  if (texelsPerPoint >= 3) {\n    vec4 texel2 = texelFetch(pointsAttributes, ivec2(texelCoord.x + 2, texelCoord.y), 0);\n    v_texCoord.z = texel2.x;\n  } else {\n    v_texCoord.z = -1.0;\n  }\n  \n  //////////////////////////////////////////////\n  // Material\n  vec2 materialCoords = v_geomItemData.zw;\n  vec4 materialValue1 = getMaterialValue(materialCoords, 1);\n  float pointSize = materialValue1.r * size;\n  float overlay = materialValue1.b;\n\n  vec4 viewPos = modelViewMatrix * vec4(pos, 1.0);\n\n  // During XR sessions, there is a scaling applied to the view matrix\n  // which causes a distortion to the line width. We extract that scale here\n  // and use to correct the distortion.\n  // See also: FatLinesShader\n  vec3 viewZ = modelViewMatrix[2].xyz;\n  float viewScale = length(viewZ);\n  viewPos += vec4(vec3(quadPointPos, 0.0) * pointSize * viewScale, 0.);\n\n  // Generate a quad which is 0.5 * PointSize closer towards\n  // us. This allows points to be visualized even if snug on \n  // a surface. (else they get fully clipped)\n  viewPos.z += 0.5 * pointSize;\n\n  v_drawItemIds.x = float(geomItemId);\n  v_drawItemIds.y = float(drawIndices);\n  v_viewPos = -viewPos.xyz;\n  \n  gl_Position = projectionMatrix * viewPos;\n\n  if (isOrthographic > 0){\n    gl_Position.z -= overlay;\n  } else {\n    gl_Position.z = mix(gl_Position.z, -gl_Position.z, overlay);\n  }\n}\n",frag$3="\nprecision highp float;\nprecision highp int;\n#define GLSLIFY 1\n\nimport 'constants.glsl'\nimport 'GLSLUtils.glsl'\nimport 'materialparams.glsl'\n\nuniform sampler2D BaseColorTex;\nuniform int BaseColorTexType;\nuniform int highlightSubIndex;\n\nuniform sampler2D atlasSprites;\nuniform sampler2D atlasSprites_layout;\nuniform vec4 atlasSprites_desc;\n\n/* VS Outputs */\nvarying vec4 v_drawItemIds;\nvarying vec4 v_geomItemData;\nvarying vec3 v_texCoord;\nvarying vec3 v_viewPos;\nvarying vec4 v_color;\n\nvec4 sampleSpriteAtlas(int spriteCoord) {\n  ivec2 atlasSpritesSize = textureSize(atlasSprites_layout, 0);\n  // round to an integer, and avoid floating point issues by adding a small value\n  ivec2 texelCoord = ivec2(\n    spriteCoord % atlasSpritesSize.x, \n    spriteCoord / atlasSpritesSize.x\n  );\n  vec4 layoutData = fetchTexel(atlasSprites_layout, atlasSpritesSize, spriteCoord);\n  vec2 spriteTexCoord = v_texCoord.xy;\n  spriteTexCoord *= layoutData.zw;\n  spriteTexCoord += layoutData.xy;\n  return texture2D(atlasSprites, spriteTexCoord);\n}\n\n#ifdef ENABLE_ES3\nout vec4 fragColor;\n#endif\n\n#if defined(DRAW_GEOMDATA)\n  uniform int isOrthographic;\n  import 'surfaceGeomData.glsl'\n#elif defined(DRAW_HIGHLIGHT)\n  import 'surfaceHighlight.glsl'\n#endif // DRAW_HIGHLIGHT\n\nvoid main(void) {\n\n#ifndef ENABLE_ES3\n  vec4 fragColor;\n#endif\n\n#if defined(DRAW_COLOR)\n\n  vec2 materialCoords = v_geomItemData.zw;\n  vec4 baseColor = getMaterialValue(materialCoords, 0);\n  vec4 matValue1 = getMaterialValue(materialCoords, 1);\n  float pointSize     = matValue1.r;\n  float borderWidth   = matValue1.g;\n\n  if (v_texCoord.z > -0.5) {\n    int spriteCoord = int(v_texCoord.z + 0.1);\n    baseColor = sampleSpriteAtlas(spriteCoord) * v_color;\n    if (baseColor.a < 0.0001) discard;\n  } else {\n    \n    float dist = length(v_texCoord.xy - 0.5);\n    if (dist > 0.5) discard;\n\n    if (dist > 0.5 - (borderWidth * 0.5))\n      baseColor = vec4(0.,0.,0.,1.);\n    else {\n      baseColor = baseColor * v_color;\n      ivec2 texSize = textureSize(BaseColorTex, 0);\n      if (texSize.x == 1 && texSize.y > 1) {\n        vec4 gradient = texture2D(BaseColorTex, vec2(0.5, dist * 2.0));\n        baseColor *= gradient;\n      }\n    }\n  }\n\n  fragColor = baseColor;\n#elif defined(DRAW_GEOMDATA)\n  fragColor = setFragColor_geomData(v_viewPos, v_drawItemIds.x, v_drawItemIds.y, isOrthographic, 0, POINTS);\n#elif defined(DRAW_HIGHLIGHT)\n  if (highlightSubIndex != -1 && int(round(v_drawItemIds.y)) != highlightSubIndex) discard;\n  fragColor = setFragColor_highlight(v_drawItemIds.x);\n#endif // DRAW_HIGHLIGHT\n\n#ifndef ENABLE_ES3\n  gl_FragColor = fragColor;\n#endif\n}\n";class FatPointsShader extends GLShader{constructor(A){super(A,"FatPointsShader"),this.setShaderStage("VERTEX_SHADER",vert$3),this.setShaderStage("FRAGMENT_SHADER",frag$3)}bind(A,g){if(super.bind(A,g)){A.supportsInstancing=!1;const g=this.__gl;return g.__quadVertexIdsBuffer||g.setupInstancedQuad(),A.shaderInstancedGeom={attrBuffers:g.__quadattrbuffers,indexBuffer:g.__quadIndexBuffer,indexDataType:g.UNSIGNED_BYTE,numVertices:4,numTriIndices:6},A.supportsInstancing=!1,!0}return!1}static getPackedMaterialData(A){const g=new Float32Array(8),I=A.getParameter("BaseColor");let C;C=I instanceof MaterialColorParam&&I.colorSpace==ColorSpace.Gamma?I.value.toLinear():I.value;let e=0;return g[4*e+0]=C.r,g[4*e+1]=C.g,g[4*e+2]=C.b,g[4*e+3]=C.a,e++,g[4*e+0]=A.getParameter("PointSize").value,g[4*e+1]=A.getParameter("BorderWidth").value,g[4*e+2]=A.getParameter("Overlay").value,g}static supportsInstancing(){return!1}static getMaterialTemplate(){return material$4}}const material$4=new FatPointsMaterial("FatPointsShader_template");Registry.register("FatPointsShader",FatPointsShader);var frag$2="precision highp float;\n#define GLSLIFY 1\n\nimport 'GLSLUtils.glsl'\nimport 'drawItemTexture.glsl'\nimport 'cutaways.glsl'\nimport 'geometryMask.glsl'\nimport 'gamma.glsl'\nimport 'materialparams.glsl'\nimport 'geomItemFlags.glsl'\n\n#ifdef DEBUG_GEOM_ID\nimport 'debugColors.glsl'\n#endif\n\n/* VS Outputs */\nvarying vec4 v_drawItemIds;\nvarying vec4 v_geomItemData;\nvarying vec3 v_viewPos;\nvarying vec3 v_viewNormal;\n#ifdef ENABLE_TEXTURES\nvarying vec2 v_textureCoord;\n#endif\nvarying vec3 v_worldPos;\n/* VS Outputs */\n\nuniform mat4 cameraMatrix;\nuniform int isOrthographic;\n\n#ifdef ENABLE_ES3\n    out vec4 fragColor;\n#endif\n\n#if defined(DRAW_COLOR)\n\n#ifdef ENABLE_TEXTURES\nuniform sampler2D BaseColorTex;\nuniform int BaseColorTexType;\nuniform sampler2D OpacityTex;\nuniform int OpacityTexType;\nuniform sampler2D EmissiveStrengthTex;\nuniform int EmissiveStrengthTexType;\n#endif // ENABLE_TEXTURES\n\nimport 'computeViewNormal.glsl'\n  \n// end DRAW_COLOR\n#elif defined(DRAW_GEOMDATA)\n  import 'surfaceGeomData.glsl'\n#elif defined(DRAW_HIGHLIGHT)\n  import 'surfaceHighlight.glsl'\n#endif // DRAW_HIGHLIGHT\n\nvoid main(void) {\n#ifndef ENABLE_ES3\n  vec4 fragColor;\n#endif\n  int geomItemId = int(round(v_drawItemIds.x));\n  int elemId = int(round(v_drawItemIds.y));\n  int perFaceMaterialId = int(round(v_drawItemIds.z));\n  int flags = int(round(v_geomItemData.x));\n  float treeItemOpacity = v_geomItemData.y;\n\n  // We can make geoms invisible to hide them. \n  // Avoid drawing GeomData for geoms that are completely transparent.\n  if (treeItemOpacity < 0.001 && !testFlag(flags, GEOMITEM_MASK)) {\n    discard;\n    return;\n  }\n  \n  // Cutaways\n  if (testFlag(flags, GEOMITEM_FLAG_CUTAWAY)) \n  {\n    vec4 cutAwayData   = getCutaway(geomItemId);\n    vec3 planeNormal = cutAwayData.xyz;\n    float planeDist = cutAwayData.w;\n    if (length(planeNormal) > 0.5) {\n      if (cutaway(v_worldPos, planeNormal, planeDist)) {\n          discard;\n          return;\n      }\n    } else {\n      if (testGeometryMask(v_viewPos)) {\n        discard;\n        return;\n      }\n    }\n  }\n\n#if defined(DRAW_COLOR)\n\n  //////////////////////////////////////////////\n  // Normals\n  \n  vec3 viewNormal;\n  if (length(v_viewNormal) < 0.1) {\n    viewNormal = computeViewNormal(v_viewPos);\n  } else {\n    viewNormal = normalize(v_viewNormal);\n  }\n  vec3 normal = normalize(mat3(cameraMatrix) * viewNormal);\n  \n  vec3 viewVector;\n  if (isOrthographic == 0)\n    viewVector = normalize(mat3(cameraMatrix) * normalize(v_viewPos));\n  else \n    viewVector = vec3(-cameraMatrix[2][0], -cameraMatrix[2][1], -cameraMatrix[2][2]);\n  \n  //////////////////////////////////////////////\n  // Material\n\n  vec2 materialCoords = v_geomItemData.zw;\n  if (perFaceMaterialId > 0) {\n    materialCoords.x = float(perFaceMaterialId);\n  }\n  vec4 baseColor      = getMaterialValue(materialCoords, 1);\n  vec4 matValue1      = getMaterialValue(materialCoords, 2);\n  float opacity       = baseColor.a * matValue1.r;\n  float emission      = matValue1.g;\n\n#ifdef ENABLE_TEXTURES\n  getTextureColorValue(baseColor, BaseColorTex, BaseColorTexType, v_textureCoord);\n  getTextureLuminanceValue(opacity, OpacityTex, OpacityTexType, v_textureCoord);\n  getTextureLuminanceValue(emission, EmissiveStrengthTex, EmissiveStrengthTexType, v_textureCoord);\n#endif\n\n  // Cutaways\n  if (testFlag(flags, GEOMITEM_FLAG_CUTAWAY) && !gl_FrontFacing) {\n    fragColor = baseColor;\n  } else {\n    // Hacky simple irradiance. \n    float ndotv = dot(normal, viewVector);\n    if (ndotv < 0.0) {\n      normal = -normal;\n      ndotv = dot(normal, viewVector);\n\n      // Note: these 2 lines can be used to debug inverted meshes.\n      //baseColor = vec4(1.0, 0.0, 0.0, 1.0);\n      //ndotv = 1.0;\n    }\n\n    fragColor = vec4((ndotv * baseColor.rgb) + (emission * baseColor.rgb), opacity);\n  }\n\n  // Note: the 'treeItemOpacity' is not an input to the lighting, \n  // as we want to also blend off the specular reflections to make an object\n  // fade away to nothing. (not become a transparent glass object).\n  fragColor.a *= treeItemOpacity;\n\n#ifdef DEBUG_GEOM_ID\n  // ///////////////////////\n  // Debug Draw ID (this correlates to GeomID within a GLGeomSet)\n  float geomId = v_geomItemData.w;\n  fragColor.rgb = getDebugColor(geomId);\n  // ///////////////////////\n#endif\n\n#ifdef ENABLE_INLINE_GAMMACORRECTION\n  fragColor.rgb = toGamma(fragColor.rgb);\n#endif\n\n#elif defined(DRAW_GEOMDATA)\n  fragColor = setFragColor_geomData(v_viewPos, v_drawItemIds.x, v_drawItemIds.y, isOrthographic, flags, TRIANGLES);\n#elif defined(DRAW_HIGHLIGHT)\n  fragColor = setFragColor_highlight(v_drawItemIds.x);\n#endif // DRAW_HIGHLIGHT\n\n#ifndef ENABLE_ES3\n  gl_FragColor = fragColor;\n#endif\n}",vert$2="precision highp float;\n#define GLSLIFY 1\n\nattribute vec3 positions;\nattribute vec3 normals;\n#ifdef ENABLE_TEXTURES\nattribute vec2 texCoords;\n#endif\n\nuniform mat4 viewMatrix;\nuniform mat4 projectionMatrix;\n\nimport 'GLSLUtils.glsl'\nimport 'transpose.glsl'\nimport 'inverse.glsl'\nimport 'geomItemId.glsl'\nimport 'drawItemTexture.glsl'\nimport 'modelMatrix.glsl'\n\n/* VS Outputs */\nvarying vec4 v_drawItemIds;\nvarying vec4 v_geomItemData;\nvarying vec3 v_viewPos;\nvarying vec3 v_viewNormal;\n#ifdef ENABLE_TEXTURES\nvarying vec2 v_textureCoord;\n#endif\nvarying vec3 v_worldPos;\n\nvoid main(void) {\n  v_drawItemIds = getDrawItemIds();\n  int geomItemId = int(round(v_drawItemIds.x));\n  v_geomItemData  = getInstanceData(geomItemId);\n\n  mat4 modelMatrix = getModelMatrix(geomItemId);\n  mat4 modelViewMatrix = viewMatrix * modelMatrix;\n\n  vec4 pos = vec4(positions, 1.);\n  vec4 viewPos    = modelViewMatrix * pos;\n  gl_Position     = projectionMatrix * viewPos;\n\n  mat3 normalMatrix = mat3(transpose(inverse(modelViewMatrix)));\n  v_viewPos       = -viewPos.xyz;\n  v_viewNormal    = normalMatrix * normals; // Note: we normalize in the fragment shader.\n\n#ifdef ENABLE_TEXTURES\n  v_textureCoord = texCoords;\n  v_textureCoord.y = 1.0 - v_textureCoord.y;// Flip y\n#endif\n\n  v_worldPos      = (modelMatrix * pos).xyz;\n}\n";class SimpleSurfaceShader extends GLShader{constructor(A){super(A,"SimpleSurfaceShader"),this.setShaderStage("VERTEX_SHADER",vert$2),this.setShaderStage("FRAGMENT_SHADER",frag$2)}static getPackedMaterialData(A){const g=new Float32Array(12);let I=0;g[4*I+0]=6;const C=A.getParameter("BaseColor");let e;return e=C instanceof MaterialColorParam&&C.colorSpace==ColorSpace.Gamma?C.value.toLinear():C.value,I++,g[4*I+0]=e.r,g[4*I+1]=e.g,g[4*I+2]=e.b,g[4*I+3]=e.a,I++,g[4*I+0]=A.getParameter("Opacity").value,g[4*I+1]=A.getParameter("EmissiveStrength").value,g}static getMaterialTemplate(){return material$3}}const material$3=new SimpleSurfaceMaterial("SimpleSurfaceShader_template");Registry.register("SimpleSurfaceShader",SimpleSurfaceShader);var vert$1="\nprecision highp float;\nprecision highp int;\n#define GLSLIFY 1\n\nattribute vec3 positions;\nattribute vec3 normals;\n#ifdef ENABLE_TEXTURES\nattribute vec2 texCoords;\n#endif\n\nuniform mat4 viewMatrix;\nuniform mat4 projectionMatrix;\nuniform int isOrthographic;\n\n// Now that we render multiple types of geometry from a single shader\n// we need to know what kind of geometry it is...\nuniform int geomType;\n\nuniform float outlineThickness;\nuniform vec2 viewportSize;\n\n// should be imported by bottom 3\nimport 'GLSLUtils.glsl'\nimport 'transpose.glsl'\nimport 'inverse.glsl'\n\nimport 'geomItemId.glsl'\nimport 'geomType.glsl'\nimport 'drawItemTexture.glsl'\nimport 'modelMatrix.glsl'\nimport 'geomItemFlags.glsl'\nimport 'materialparams.glsl'\n\n/* VS Outputs */\nvarying vec4 v_drawItemIds;\nvarying vec4 v_geomItemData;\nvarying vec3 v_viewPos;\nvarying vec3 v_viewNormal;\n#ifdef ENABLE_TEXTURES\nvarying vec2 v_textureCoord;\n#endif\nvarying vec3 v_worldPos;\n/* VS Outputs */\n\n#if defined(DRAW_COLOR)\n#elif defined(DRAW_GEOMDATA)\n#elif defined(DRAW_HIGHLIGHT)\n#endif // DRAW_HIGHLIGHT\n\nvoid main(void) {\n\n  v_drawItemIds = getDrawItemIds();\n  int geomItemId = int(round(v_drawItemIds.x));\n  v_geomItemData = getInstanceData(geomItemId);\n  mat4 modelMatrix = getModelMatrix(geomItemId);\n\n  vec4 pos = vec4(positions, 1.);\n  mat4 modelViewMatrix = viewMatrix * modelMatrix;\n  vec4 viewPos    = modelViewMatrix * pos;\n  gl_Position     = projectionMatrix * viewPos;\n  v_viewPos       = -viewPos.xyz;\n  v_worldPos      = (modelMatrix * pos).xyz;\n\n  mat3 normalMatrix = mat3(transpose(inverse(modelViewMatrix)));\n  v_viewNormal    = normalMatrix * normals; // Note: we normalize in the fragment shader.\n  \n  // offset slightly the lines and points to make them clearly defined.\n  // This ensures that lines drawn over surfaces are solid and not clipped\n  // at all by the surface.\n  if (geomType == TRIANGLES) {\n    if (outlineThickness > 0.00001) {\n      vec2 screenNormal = v_viewNormal.xy;\n      gl_Position.xy += normalize(screenNormal) * ((2.0 / viewportSize) * outlineThickness) * gl_Position.w;\n    }\n  }\n  \n  //////////////////////////////////////////////\n  // Overlay\n  float viewDepth = v_viewPos.z;\n    \n  vec2 materialCoords = v_geomItemData.zw;\n  vec4 materialValue2 = getMaterialValue(materialCoords, 3);\n  vec4 materialValue5 = getMaterialValue(materialCoords, 6);\n  float overlay = materialValue2.b;\n  float pointSize = materialValue5.r;\n\n  if (geomType == POINTS) { // start 'POINTS'\n    // Fixed size on screen points.\n    gl_PointSize = pointSize;\n    overlay += 0.000005 / viewDepth;\n  }  // end 'POINTS'\n\n  if (isOrthographic > 0){\n    gl_Position.z -= overlay;\n  } else {\n    gl_Position.z = mix(gl_Position.z, -gl_Position.z, overlay);\n  }\n\n#ifdef ENABLE_TEXTURES\n  v_textureCoord = texCoords;\n  v_textureCoord.y = 1.0 - v_textureCoord.y;// Flip y\n#endif\n\n}\n",frag$1="precision highp float;\nprecision highp int;\n#define GLSLIFY 1\nimport 'GLSLUtils.glsl'\nimport 'drawItemTexture.glsl' \nimport 'cutaways.glsl'\nimport 'geometryMask.glsl'\nimport 'gamma.glsl'\nimport 'materialparams.glsl'\nimport 'GLSLBits.glsl'\nimport 'geomItemFlags.glsl'\n\n/* VS Outputs */\nvarying vec4 v_drawItemIds;\nvarying vec4 v_geomItemData;\nvarying vec3 v_viewPos;\nvarying vec3 v_viewNormal;\n#ifdef ENABLE_TEXTURES\nvarying vec2 v_textureCoord;\n#endif\nvarying vec3 v_worldPos;\n/* VS Outputs */\n\n#ifdef ENABLE_ES3\n  out vec4 fragColor;\n#endif\n\n// Now that we render multiple types of geometry from a single shader\n// we need to know what kind of geometry it is...\nuniform int geomType;\nimport 'geomType.glsl'\n\nuniform int isOrthographic;\n\n#if defined(DRAW_COLOR)\n\nuniform int renderMode;\nuniform int occluded;\nuniform float outlineThickness;\nuniform vec4 hiddenLineColor;\n\n#ifdef ENABLE_INLINE_GAMMACORRECTION\nuniform float exposure;\n#endif\n\nuniform mat4 cameraMatrix;\n\n#ifdef ENABLE_TEXTURES\nuniform sampler2D BaseColorTex;\nuniform int BaseColorTexType;\n\nuniform sampler2D AmbientOcclusionTex;\nuniform int AmbientOcclusionTexType;\n\n#ifdef ENABLE_PBR\nuniform sampler2D RoughnessTex;\nuniform int RoughnessTexType;\n\nuniform sampler2D MetallicTex;\nuniform int MetallicTexType;\n\nuniform sampler2D ReflectanceTex;\nuniform int ReflectanceTexType;\n\nuniform sampler2D NormalTex;\nuniform int NormalTexType;\n#endif // ENABLE_PBR\n\nuniform sampler2D EmissiveStrengthTex;\nuniform int EmissiveStrengthTexType;\n\n#endif // ENABLE_TEXTURES\n\nimport 'PBRSurfaceRadiance.glsl'\n\n#ifdef ENABLE_PBR\nmat3 cotangentFrame( in vec3 normal, in vec3 pos, in vec2 texCoord ) {\n  // https://stackoverflow.com/questions/5255806/how-to-calculate-tangent-and-binormal\n  vec3 n = normal;\n  // derivations of the fragment position\n  vec3 pos_dx = dFdx( pos );\n  vec3 pos_dy = dFdy( pos );\n  // derivations of the texture coordinate\n  vec2 texC_dx = dFdx( texCoord );\n  vec2 texC_dy = dFdy( texCoord );\n  // tangent vector and binormal vector\n  vec3 t = -(texC_dy.y * pos_dx - texC_dx.y * pos_dy);\n  vec3 b = -(texC_dx.x * pos_dy - texC_dy.x * pos_dx);\n\n  t = t - n * dot( t, n ); // orthonormalization ot the tangent vectors\n  b = b - n * dot( b, n ); // orthonormalization of the binormal vectors to the normal vector\n  b = b - t * dot( b, t ); // orthonormalization of the binormal vectors to the tangent vector\n  mat3 tbn = mat3( normalize(t), normalize(b), n );\n\n  return tbn;\n}\n#endif\n\nimport 'computeViewNormal.glsl'\n\n#ifdef DEBUG_GEOM_ID\nimport 'debugColors.glsl'\n#endif\n\n// end DRAW_COLOR\n#elif defined(DRAW_GEOMDATA)\n\nimport 'surfaceGeomData.glsl'\n\n#elif defined(DRAW_HIGHLIGHT)\nimport 'surfaceHighlight.glsl'\n#endif // DRAW_HIGHLIGHT\n\nvoid main(void) {\n  #ifndef ENABLE_ES3\n    vec4 fragColor;\n  #endif\n  \n  int geomItemId = int(round(v_drawItemIds.x));\n  int elemId = int(round(v_drawItemIds.y));\n  int perElementMaterialId = int(round(v_drawItemIds.z));\n  int flags = int(round(v_geomItemData.x));\n  float treeItemOpacity = v_geomItemData.y;\n\n  // We can make geoms invisible to hide them. \n  // Avoid drawing GeomData for geoms that are completely transparent.\n  if (treeItemOpacity < 0.001) {\n    discard;\n    return;\n  }\n  \n  if (testFlag(flags, GEOMITEM_FLAG_CUTAWAY)) {\n    vec4 cutAwayData   = getCutaway(geomItemId);\n    vec3 planeNormal = cutAwayData.xyz;\n    float planeDist = cutAwayData.w;\n    if (length(planeNormal) > 0.5) {\n      if (cutaway(v_worldPos, planeNormal, planeDist)) {\n          discard;\n          return;\n      }\n    } else {\n      if (testGeometryMask(v_viewPos)) {\n        discard;\n        return;\n      }\n    }\n  }\n  vec2 materialCoords = v_geomItemData.zw;\n  if (perElementMaterialId > 0) {\n    materialCoords.x = float(perElementMaterialId);\n  }\n\n  \n  vec4 materialHeader      = getMaterialValue(materialCoords, 0);\n  int materialId = int(materialHeader.x + 0.5);\n  \n#if defined(DRAW_COLOR)\n\n  if (geomType == TRIANGLES) { // start 'TRIANGLES'\n\n    //////////////////////////////////////////////\n    // Normals\n    vec3 viewNormal;\n    if (length(v_viewNormal) < 0.1) {\n      viewNormal = computeViewNormal(v_viewPos);\n    } else {\n      viewNormal = normalize(v_viewNormal);\n    }\n    vec3 normal = normalize(mat3(cameraMatrix) * viewNormal);\n    \n    vec3 viewVector;\n    if (isOrthographic == 0)\n      viewVector = normalize(mat3(cameraMatrix) * normalize(v_viewPos));\n    else \n      viewVector = vec3(cameraMatrix[2][0], cameraMatrix[2][1], cameraMatrix[2][2]);\n      \n    if (dot(normal, viewVector) < 0.0) {\n      normal = -normal;\n      // Note: this line can be used to debug inverted meshes.\n      //material.baseColor = vec3(1.0, 0.0, 0.0);\n    }\n\n    //////////////////////////////////////////////\n    // Material\n\n    MaterialParams material;\n\n    vec4 matValue0      = getMaterialValue(materialCoords, 1);\n    vec4 matValue1      = getMaterialValue(materialCoords, 2);\n    vec4 matValue2      = getMaterialValue(materialCoords, 3);\n\n    vec4 baseColor = matValue0;\n    float opacity = matValue2.g;\n    material.baseColor     = baseColor.rgb;\n    material.ambientOcclusion = matValue1.r;\n    material.metallic      = matValue1.g;\n    material.roughness     = matValue1.b;\n    material.reflectance   = matValue1.a;\n\n    material.emission      = matValue2.r;\n    material.opacity       = opacity * baseColor.a;\n\n#ifdef ENABLE_TEXTURES\n\n    // Planar YZ projection for texturing, repeating every meter.\n    // vec2 texCoord       = v_worldPos.xz * 0.2;\n    vec2 texCoord          = v_textureCoord;\n\n    getTextureColorValue(baseColor, BaseColorTex, BaseColorTexType, texCoord);\n    getTextureLuminanceValue(material.ambientOcclusion, AmbientOcclusionTex, AmbientOcclusionTexType, texCoord);\n    \n    material.baseColor     = baseColor.rgb;\n    material.opacity       = opacity * baseColor.a;\n\n#ifdef ENABLE_PBR\n\n    getTextureLuminanceValue(material.metallic, MetallicTex, MetallicTexType, texCoord);\n    getTextureLuminanceValue(material.roughness, RoughnessTex, RoughnessTexType, texCoord);\n\n    // TODO: Communicate that this tex contains the roughness as well.\n    if (MetallicTexType != 0) {\n      vec4 metallicRoughness = texture2D(MetallicTex, texCoord);\n      material.roughness = metallicRoughness.g;\n      material.metallic = metallicRoughness.b;\n    }\n\n    getTextureLuminanceValue(material.reflectance, ReflectanceTex, ReflectanceTexType, texCoord);\n#endif // ENABLE_PBR\n\n    getTextureLuminanceValue(material.emission, EmissiveStrengthTex, EmissiveStrengthTexType, texCoord);\n#endif // ENABLE_TEXTURES\n\n#ifdef ENABLE_TEXTURES\n#ifdef ENABLE_PBR\n    if (NormalTexType != 0) {\n      mat3 tbn = cotangentFrame(normal, viewVector, texCoord);\n      normal = normalize(tbn * (texture2D(NormalTex, texCoord).rgb * 2.0 - 1.0));\n    }\n#endif // ENABLE_PBR\n#endif // ENABLE_TEXTURES\n\n    if (outlineThickness > 0.00001) {\n      vec4 edgeColor      = getMaterialValue(materialCoords, 4);\n      vec4 matValue2      = getMaterialValue(materialCoords, 3);\n      float opacity       = matValue2.g;\n\n      edgeColor.a = edgeColor.a * opacity * treeItemOpacity;\n      \n      fragColor = edgeColor;\n    } else {\n      \n      if (testFlag(flags, GEOMITEM_FLAG_CUTAWAY) && !gl_FrontFacing) {\n        fragColor = vec4(material.baseColor, material.opacity);\n      }\n      else if (renderMode == 1) { // Flat\n        fragColor = vec4(material.baseColor, material.opacity);\n      } else if (renderMode == 2) { // Shaded\n        // simple irradiance. \n        float ndotv = dot(normal, viewVector);\n        fragColor = vec4((ndotv * material.baseColor) + (material.emission * material.baseColor), material.opacity);\n      } else if (renderMode == 3) { // PBR\n        fragColor = pbrSurfaceRadiance(material, normal, viewVector);\n      }\n      \n      // Note: the 'treeItemOpacity' is not an input to the PBR lighting, \n      // as we want to also blend off the specular reflections to make an object\n      // fade away to nothing. (not become a transparent glass object).\n      fragColor.a *= treeItemOpacity;\n\n      // Debugging code to help understand what might be happening in the shader.\n      // fragColor = vec4(texture2D(NormalTex, texCoord).rgb, 1.0);\n      // fragColor = metallicRoughness;\n      // fragColor = vec4(material.baseColor, 1.0);;\n      // fragColor = vec4(vec3(material.metallic), 1.0);;\n      // fragColor = vec4(vec3(material.roughness), 1.0);;\n      // fragColor = vec4(vec3(material.ambientOcclusion), 1.0);\n    }\n\n  } // end 'TRIANGLES'\n  else if (geomType == LINES) { // start 'LINES'\n    if (occluded == 1) {\n      vec4 matValue2      = getMaterialValue(materialCoords, 3);\n      float opacity       = matValue2.g;\n      fragColor = hiddenLineColor;\n      fragColor.a = hiddenLineColor.a * opacity * treeItemOpacity;\n    } else {\n      vec4 edgeColor;\n      // Note: if a custom material has been assigned to an edge, we use the base color, not the edge color.\n      // The ZeaSpatialBridge actually assigns 'LineColor' to the edges, so they only have a 'baseColor'.\n      \n      if (materialId == StandardSurfaceShaderId) {\n        vec4 matValue2      = getMaterialValue(materialCoords, 3);\n        edgeColor           = getMaterialValue(materialCoords, 4);\n        float opacity       = matValue2.g;\n        float edgeWeight    = matValue2.a;\n        edgeColor.a = edgeColor.a * edgeWeight * opacity * treeItemOpacity;\n      } else {\n        vec4 baseColor      = getMaterialValue(materialCoords, 1);\n        baseColor.a = baseColor.a * treeItemOpacity;\n        edgeColor = baseColor;\n      }\n      fragColor = edgeColor;\n    }\n  } // end 'LINES'\n  else if (geomType == POINTS) { // start 'POINTS'\n    if (materialId == StandardSurfaceShaderId) {\n      vec4 matValue2      = getMaterialValue(materialCoords, 3);\n      vec4 pointColor     = getMaterialValue(materialCoords, 5);\n      float opacity       = matValue2.g;\n      pointColor.a = pointColor.a * opacity * treeItemOpacity;\n      fragColor = pointColor;\n    } else {\n      vec4 baseColor      = getMaterialValue(materialCoords, 1);\n      baseColor.a = baseColor.a * treeItemOpacity;\n      fragColor = baseColor;\n    }\n  }  // end 'POINTS'\n  \n#ifdef DEBUG_GEOM_ID\n  // ///////////////////////\n  // Debug Draw ID (this correlates to GeomID within a GLGeomSet)\n  float geomId = v_geomItemData.w;\n  fragColor.rgb = getDebugColor(geomId);\n  // ///////////////////////\n#endif\n\n#ifdef ENABLE_INLINE_GAMMACORRECTION\n  fragColor.rgb = toGamma(fragColor.rgb * exposure);\n#endif\n\n// end DRAW_COLOR\n#elif defined(DRAW_GEOMDATA)\n\n  fragColor = setFragColor_geomData(v_viewPos, v_drawItemIds.x, v_drawItemIds.y, isOrthographic, flags, geomType);\n   \n#elif defined(DRAW_HIGHLIGHT)\n\n  // we don't render highlighs for non-selectable objects.\n  if (testFlag(flags, GEOMITEM_INVISIBLE_IN_GEOMDATA)) {\n    discard;\n  }\n\n  fragColor = getHighlightColor(geomItemId);\n#endif // DRAW_HIGHLIGHT\n\n#ifndef ENABLE_ES3\n  gl_FragColor = fragColor;\n#endif\n\n}";class StandardSurfaceShader extends GLShader{constructor(A){super(A,"StandardSuraceShader"),this.setShaderStage("VERTEX_SHADER",vert$1),this.setShaderStage("FRAGMENT_SHADER",frag$1)}bind(A,g){if(super.bind(A,g),A instanceof ColorRenderState){const g=A,I=this.__gl;g.envMap&&g.envMap.bind(g);const{exposure:C,renderMode:e}=g.unifs;C&&I.uniform1f(C.location,g.exposure),g.renderMode&&e&&("flat"==g.renderMode||"flat-noedges"==g.renderMode?I.uniform1i(e.location,1):"shaded"==g.renderMode||"shaded-noedges"==g.renderMode?I.uniform1i(e.location,2):"pbr"!=g.renderMode&&"pbr-noedges"!=g.renderMode||I.uniform1i(e.location,3))}return!0}static getPackedMaterialData(A){const g=new Float32Array(28);let I=0;g[4*I+0]=7;const C=A.getParameter("BaseColor");let e;e=C instanceof MaterialColorParam&&C.colorSpace==ColorSpace.Gamma?C.value.toLinear():C.value,I++,g[4*I+0]=e.r,g[4*I+1]=e.g,g[4*I+2]=e.b,g[4*I+3]=e.a,I++,g[4*I+0]=A.getParameter("AmbientOcclusion").value,g[4*I+1]=A.getParameter("Metallic").value,g[4*I+2]=A.getParameter("Roughness").value,g[4*I+3]=A.getParameter("Reflectance").value,I++,g[4*I+0]=A.getParameter("EmissiveStrength").value,g[4*I+1]=A.getParameter("Opacity").value,g[4*I+2]=A.getParameter("Overlay").value,g[4*I+3]=A.getParameter("EdgeWeight").value,I++;const i=A.getParameter("EdgeColor").value;g[4*I+0]=i.r,g[4*I+1]=i.g,g[4*I+2]=i.b,g[4*I+3]=i.a,I++;const t=A.getParameter("PointColor").value;return g[4*I+0]=t.r,g[4*I+1]=t.g,g[4*I+2]=t.b,g[4*I+3]=t.a,I++,g[4*I+0]=A.getParameter("PointSize").value,g}static getMaterialTemplate(){return material$2}}const material$2=new StandardSurfaceMaterial("StandardSurfaceShader_template");Registry.register("StandardSurfaceShader",StandardSurfaceShader),Registry.register("TransparentSurfaceShader",StandardSurfaceShader);class VertexColorShader extends GLShader{constructor(A){super(A,"VertexColorShader"),this.setShaderStage("VERTEX_SHADER","precision highp float;\n\n\n    attribute vec3 positions;\n    attribute vec3 normals;\n    #ifdef ENABLE_TEXTURES\n    attribute vec2 texCoords;\n    #endif\n    attribute vec4 colors;\n    \n    uniform mat4 viewMatrix;\n    uniform mat4 projectionMatrix;\n    \n    import 'GLSLUtils.glsl'\n    import 'transpose.glsl'\n    import 'inverse.glsl'\n    import 'geomItemId.glsl'\n    import 'drawItemTexture.glsl'\n    import 'modelMatrix.glsl'\n    \n    /* VS Outputs */\n    varying vec4 v_drawItemIds;\n    varying vec4 v_geomItemData;\n    varying vec3 v_viewPos;\n    varying vec3 v_viewNormal;\n    varying vec4 v_vertexColors;\n    #ifdef ENABLE_TEXTURES\n    varying vec2 v_textureCoord;\n    #endif\n    varying vec3 v_worldPos;\n    \n    void main(void) {\n      v_drawItemIds = getDrawItemIds();\n      int geomItemId = int(round(v_drawItemIds.x));\n      v_geomItemData  = getInstanceData(geomItemId);\n      v_vertexColors = colors;\n      \n      mat4 modelMatrix = getModelMatrix(geomItemId);\n      mat4 modelViewMatrix = viewMatrix * modelMatrix;\n    \n      vec4 pos = vec4(positions, 1.);\n      vec4 viewPos    = modelViewMatrix * pos;\n      gl_Position     = projectionMatrix * viewPos;\n    \n      mat3 normalMatrix = mat3(transpose(inverse(modelViewMatrix)));\n      v_viewPos       = -viewPos.xyz;\n      v_viewNormal    = normalMatrix * normals; // Note: we normalize in the fragment shader.\n    \n    #ifdef ENABLE_TEXTURES\n      v_textureCoord  = texCoords;\n      // v_textureCoord.y = 1.0 - v_textureCoord.y;// Flip y\n    #endif\n    \n      v_worldPos      = (modelMatrix * pos).xyz;\n    }\n    "),this.setShaderStage("FRAGMENT_SHADER","precision highp float;\n\n    import 'GLSLUtils.glsl'\n    import 'drawItemTexture.glsl'\n    import 'cutaways.glsl'\n    import 'geometryMask.glsl'\n    import 'gamma.glsl'\n    import 'materialparams.glsl'\n    import 'geomItemFlags.glsl'\n    \n    #ifdef DEBUG_GEOM_ID\n    import 'debugColors.glsl'\n    #endif\n    \n    /* VS Outputs */\n    varying vec4 v_drawItemIds;\n    varying vec4 v_geomItemData;\n    varying vec3 v_viewPos;\n    varying vec3 v_viewNormal;\n    varying vec4 v_vertexColors;\n    #ifdef ENABLE_TEXTURES\n    varying vec2 v_textureCoord;\n    #endif\n    varying vec3 v_worldPos;\n    /* VS Outputs */\n    \n    uniform mat4 cameraMatrix;\n    uniform int isOrthographic;\n    \n    #ifdef ENABLE_ES3\n        out vec4 fragColor;\n    #endif\n    \n    \n    #if defined(DRAW_COLOR)\n    \n    import 'computeViewNormal.glsl'\n      \n    // end DRAW_COLOR\n    #elif defined(DRAW_GEOMDATA)\n      import 'surfaceGeomData.glsl'\n    #elif defined(DRAW_HIGHLIGHT)\n      import 'surfaceHighlight.glsl'\n    #endif // DRAW_HIGHLIGHT\n    \n    \n    void main(void) {\n    #ifndef ENABLE_ES3\n      vec4 fragColor;\n    #endif\n      int geomItemId = int(round(v_drawItemIds.x));\n      int elemId = int(round(v_drawItemIds.y));\n      int perFaceMaterialId = int(round(v_drawItemIds.z));\n      int flags = int(round(v_geomItemData.x));\n      float treeItemOpacity = v_geomItemData.y;\n    \n      // We can make geoms invisible to hide them. \n      // Avoid drawing GeomData for geoms that are completely transparent.\n      if (treeItemOpacity < 0.001) {\n        discard;\n        return;\n      }\n      \n      // Cutaways\n      if (testFlag(flags, GEOMITEM_FLAG_CUTAWAY)) \n      {\n        vec4 cutAwayData   = getCutaway(geomItemId);\n        vec3 planeNormal = cutAwayData.xyz;\n        float planeDist = cutAwayData.w;\n        if (length(planeNormal) > 0.5) {\n          if (cutaway(v_worldPos, planeNormal, planeDist)) {\n              discard;\n              return;\n          }\n        } else {\n          if (testGeometryMask(v_viewPos)) {\n            discard;\n            return;\n          }\n        }\n      }\n    \n    #if defined(DRAW_COLOR)\n    \n      if (testFlag(flags, GEOMITEM_FLAG_CUTAWAY) && !gl_FrontFacing) {\n        fragColor = cutColor;\n        return;\n      } else {\n\n        //////////////////////////////////////////////\n        // Normals\n        \n        vec3 viewNormal;\n        if (length(v_viewNormal) < 0.1) {\n          viewNormal = computeViewNormal(v_viewPos);\n        } else {\n          viewNormal = normalize(v_viewNormal);\n        }\n        vec3 normal = normalize(mat3(cameraMatrix) * viewNormal);\n        \n        vec3 viewVector;\n        if (isOrthographic == 0)\n          viewVector = normalize(mat3(cameraMatrix) * normalize(v_viewPos));\n        else \n          viewVector = vec3(-cameraMatrix[2][0], -cameraMatrix[2][1], -cameraMatrix[2][2]);\n        \n        //////////////////////////////////////////////\n        // Material\n      \n        vec4 baseColor = v_vertexColors;\n      \n        // Hacky simple irradiance. \n        float ndotv = dot(normal, viewVector);\n        if (ndotv < 0.0) {\n          normal = -normal;\n          ndotv = dot(normal, viewVector);\n      \n          // Note: these 2 lines can be used to debug inverted meshes.\n          //baseColor = vec4(1.0, 0.0, 0.0, 1.0);\n          //ndotv = 1.0;\n        }\n\n        fragColor = vec4((ndotv * baseColor.rgb), 1.0);\n      }\n    \n      // Note: the 'treeItemOpacity' is not an input to the lighting, \n      // as we want to also blend off the specular reflections to make an object\n      // fade away to nothing. (not become a transparent glass object).\n      fragColor.a *= treeItemOpacity;\n    \n    \n    #ifdef DEBUG_GEOM_ID\n      // ///////////////////////\n      // Debug Draw ID (this correlates to GeomID within a GLGeomSet)\n      float geomId = v_geomItemData.w;\n      fragColor.rgb = getDebugColor(geomId);\n      // ///////////////////////\n    #endif\n    \n    #ifdef ENABLE_INLINE_GAMMACORRECTION\n      fragColor.rgb = toGamma(fragColor.rgb);\n    #endif\n    \n    #elif defined(DRAW_GEOMDATA)\n      fragColor = setFragColor_geomData(v_viewPos, v_drawItemIds.x, v_drawItemIds.y, isOrthographic, 0, TRIANGLES);\n    #elif defined(DRAW_HIGHLIGHT)\n      fragColor = setFragColor_highlight(v_drawItemIds.x);\n    #endif // DRAW_HIGHLIGHT\n    \n    #ifndef ENABLE_ES3\n      gl_FragColor = fragColor;\n    #endif\n    }")}bind(A,g){if(super.bind(A,g),A instanceof ColorRenderState){const g=this.__gl,{cutColor:I}=A.unifs;I&&g.uniform4f(I.location,.3,0,0,1)}return!0}static getMaterialTemplate(){return material$1}static supportsInstancing(){return!1}}const material$1=new VertexColorMaterial("VertexColorShader_template");Registry.register("VertexColorShader",VertexColorShader);var frag="\nprecision highp float;\n#define GLSLIFY 1\n\nimport 'GLSLUtils.glsl'\nimport 'drawItemTexture.glsl'\n\nimport 'gamma.glsl'\nimport 'materialparams.glsl'\nimport 'geomItemFlags.glsl'\n\n#if defined(DRAW_COLOR)\n\n#ifdef ENABLE_TEXTURES\nuniform sampler2D BaseColorTex;\nuniform int BaseColorTexType;\n#endif\n\n#endif // DRAW_COLOR\n\n/* VS Outputs */\nvarying float v_geomItemId;\nvarying vec4 v_geomItemData;\n#ifdef ENABLE_TEXTURES\nvarying vec2 v_textureCoord;\n#endif\n\n#if defined(DRAW_GEOMDATA)\n  uniform int isOrthographic;\n  import 'surfaceGeomData.glsl'\n#elif defined(DRAW_HIGHLIGHT)\n  import 'surfaceHighlight.glsl'\n#endif // DRAW_HIGHLIGHT\n\n#ifdef ENABLE_ES3\n  out vec4 fragColor;\n#endif\n\nvoid main(void) {\n  \n#ifndef ENABLE_ES3\n  vec4 fragColor;\n#endif\n\n  //////////////////////////////////////////////\n  // Color\n#if defined(DRAW_COLOR)\n\n    vec2 materialCoords = v_geomItemData.zw;\n    vec4 baseColor = getMaterialValue(materialCoords, 0);\n\n  #ifdef ENABLE_TEXTURES\n    getTextureColorValue(baseColor, BaseColorTex, BaseColorTexType, v_textureCoord);\n  #endif\n\n    fragColor = baseColor;\n\n  #ifdef ENABLE_INLINE_GAMMACORRECTION\n    fragColor.rgb = toGamma(fragColor.rgb);\n  #endif\n\n  //////////////////////////////////////////////\n  // GeomData\n#elif defined(DRAW_GEOMDATA)\n  fragColor = setFragColor_geomData(vec3(0,0,0), v_geomItemId, 0.0, isOrthographic, 0, TRIANGLES);\n#elif defined(DRAW_HIGHLIGHT)\n  fragColor = setFragColor_highlight(v_geomItemId);\n#endif // DRAW_HIGHLIGHT\n\n#ifndef ENABLE_ES3\n  gl_FragColor = fragColor;\n#endif\n}\n",vert="\nprecision highp float;\n#define GLSLIFY 1\n\nattribute vec3 positions;\n#ifdef ENABLE_TEXTURES\nattribute vec2 texCoords;\n#endif\n\nimport 'GLSLUtils.glsl'\nimport 'geomItemId.glsl'\nimport 'drawItemTexture.glsl'\nimport 'modelMatrix.glsl'\n\n/* VS Outputs */\nvarying float v_geomItemId;\nvarying vec4 v_geomItemData;\n#ifdef ENABLE_TEXTURES\nvarying vec2 v_textureCoord;\n#endif\n\nvoid main(void) {\n  int geomItemId = getGeomItemId();\n  v_geomItemId = float(geomItemId);\n  v_geomItemData  = getInstanceData(geomItemId);\n\n  mat4 modelMatrix = getModelMatrix(geomItemId);\n\n  gl_Position = (modelMatrix * vec4(positions, 1.0));\n\n  v_textureCoord = texCoords;\n  v_textureCoord.y = 1.0 - v_textureCoord.y;// Flip y\n}\n";class ScreenSpaceShader extends GLShader{constructor(A){super(A,"ScreenSpaceShader"),this.setShaderStage("VERTEX_SHADER",vert),this.setShaderStage("FRAGMENT_SHADER",frag)}static isOverlay(){return!0}static getPackedMaterialData(A){const g=new Float32Array(8);const I=A.getParameter("BaseColor");let C;return C=I instanceof MaterialColorParam&&I.colorSpace==ColorSpace.Gamma?I.value.toLinear():I.value,g[0]=C.r,g[1]=C.g,g[2]=C.b,g[3]=C.a,g}static getMaterialTemplate(){return material}}const material=new ScreenSpaceMaterial("ScreenSpaceShader_template");Registry.register("ScreenSpaceShader",ScreenSpaceShader);class GLStandardGeomsPass extends GLPass{materials=new Map;listenerIDs=new Map;constructor(){super()}init(A,g){super.init(A,g)}itemAddedToScene(A,g){if(A instanceof GeomItem){const g=A;return!!this.filterGeomItem(g)&&(this.addGeomItem(g),!0)}return!1}itemRemovedFromScene(A,g){return A instanceof GeomItem&&(this.removeGeomItem(A),!0)}filterGeomItem(A){return!0}checkMaterial(A){return!0}addGeomItem(A){const g={};this.listenerIDs.set(A,g);const I=()=>{this.removeGeomItem(A),this.renderer.assignTreeItemToGLPass(A)};g["materialParam.valueChanged"]=A.materialParam.on("valueChanged",I),g["geomParam.valueChanged"]=A.geomParam.on("valueChanged",I);const C=A=>{A.isOpaqueStateChanged&&I()},e=A.materialParam.value;this.materials.set(A,e),g["geomItem.opacityChanged"]=A.on("opacityChanged",C),g["material.opacityChanged"]=e.on("opacityChanged",C)}removeGeomItem(A){const g=this.listenerIDs.get(A);this.listenerIDs.delete(A),A.materialParam.off("valueChanged",g["materialParam.valueChanged"]),A.geomParam.off("valueChanged",g["geomParam.valueChanged"]);const I=this.materials.get(A);this.materials.delete(A),A.off("opacityChanged",g["geomItem.opacityChanged"]),I.off("opacityChanged",g["material.opacityChanged"])}constructShader(A){return this.__renderer.getOrCreateShader(A)}getGeomItemAndDist(A){let g,I,C=-1;A instanceof Float32Array?(g=Math.round(A[1]),C=Math.round(A[2]),I=A[3]):(g=A[0]+((63&A[1])<<8),I=MathFunctions.decode16BitFloatFrom2xUInt8(A.slice(2,3)));const e=this.renderer.glGeomItemLibrary.getGeomItem(g);if(e)return{geomItem:e,componentId:C,dist:I}}}class GLOpaqueGeomsPass extends GLStandardGeomsPass{glShaderMaterials=new Map;glShaderGeomSets=new Map;constructor(){super()}getPassType(){return PassType.OPAQUE}filterGeomItem(A){const g=A.materialParam.value;return A.isOpaque()&&g.isOpaque()}checkMaterial(A){return A.isOpaque()}removeAndReAddGeomItem(A){this.removeGeomItem(A),this.__renderer.assignTreeItemToGLPass(A)}addGeomItem(A){super.addGeomItem(A);const g=A.materialParam.value,I=this.renderer.glGeomItemLibrary.getGLGeomItem(A);this.__gl.multiDrawElementsInstanced&&I.supportInstancing&&!g.isTextured()?this.addGeomItemToMultiDraw(A,I):this.addGeomItemToConventionalDraw(A,I)}addGeomItemToMultiDraw(A,g){const I=A.materialParam.value.getShaderName(),C=this.__renderer.getOrCreateShader(I);let e=this.glShaderGeomSets.get(C);e||(e=new GLShaderGeomSets(this.__renderer,this.__gl,C),e.on("updated",(()=>{this.__renderer.requestRedraw()})),this.glShaderGeomSets.set(C,e)),e.addGLGeomItem(g),g.GLShaderGeomSets=e,this.emit("updated")}addGeomItemToConventionalDraw(A,g){const I=A.materialParam.value,C=this.renderer.glGeomLibrary.constructGLGeom(A.geomParam.value),e=I.getShaderName(),i=this.renderer.glMaterialLibrary.getGLMaterial(I),t=i.glShader;let o=this.glShaderMaterials.get(t);if(!o){const A=this.constructShader(e);o=new GLShaderMaterials(this.__gl,this,A),this.glShaderMaterials.set(t,o),o.on("updated",(()=>{this.__renderer.requestRedraw()}))}o.addGLGeomItem(g,C,i)}removeGeomItem(A){super.removeGeomItem(A);const g=this.renderer.glGeomItemLibrary.getGLGeomItem(A);if(g.GLShaderGeomSets){return g.GLShaderGeomSets.removeGLGeomItem(g),void(g.GLShaderGeomSets=null)}if(g.GLGeomItemSet){return g.GLGeomItemSet.removeGLGeomItem(g),void(g.GLGeomItemSet=null)}}removeMaterial(A){const g=this.renderer.glMaterialLibrary.getGLMaterial(A),I=g.glShader;this.glShaderMaterials.get(I).removeMaterialGeomItemSets(g)}traverseTreeAndDraw(A){this.glShaderGeomSets.forEach((g=>{g.draw(A)})),this.glShaderMaterials.forEach((g=>{g.draw(A)})),A.glGeom&&A.glGeom.unbind(A)}draw(A){const g=this.__gl;A.pushGLStack("GLOpaqueGeomsPass.drawGeomData"),A.glDisable(g.BLEND),A.glEnable(g.DEPTH_TEST),A.glDisable(g.CULL_FACE),g.depthFunc(g.LEQUAL),g.depthMask(!0),this.traverseTreeAndDraw(A);const I=this.renderer;I.outlineThickness>0&&"image"==I.outlineMethod&&A.viewport.drawSilhouettes(A),A.popGLStack()}drawHighlightedGeoms(A){const g=this.__gl;g.disable(g.CULL_FACE),this.glShaderGeomSets.forEach((g=>{g.drawHighlightedGeoms(A)})),this.glShaderMaterials.forEach((g=>{g.drawHighlightedGeoms(A)})),A.glGeom&&A.glGeom.unbind(A)}drawGeomData(A){A.passIndex=this.passIndex;const g=this.__gl;g.disable(g.BLEND),g.disable(g.CULL_FACE),g.enable(g.DEPTH_TEST),g.depthFunc(g.LEQUAL),g.depthMask(!0),this.glShaderGeomSets.forEach((g=>{g.drawGeomData(A)})),this.glShaderMaterials.forEach((g=>{g.drawGeomData(A)})),A.glGeom&&A.glGeom.unbind(A)}}GLRenderer.registerPass(GLOpaqueGeomsPass,PassType.OPAQUE);class GLLinesPass extends GLOpaqueGeomsPass{linesGeomDataBuffer=null;fattenLinesShader=null;quad=null;fbo=null;constructor(){super()}init(A,g){super.init(A,g)}filterGeomItem(A){const g=A.geomParam.value;return g instanceof Lines||g instanceof LinesProxy||g instanceof Points||g instanceof PointsProxy}draw(A){const g=this.__gl;A.pushGLStack("GLLinesPass.draw"),A.glEnable(g.BLEND),A.glEnable(g.DEPTH_TEST),g.blendFuncSeparate(g.SRC_ALPHA,g.ONE_MINUS_SRC_ALPHA,g.ONE,g.ONE),g.depthFunc(g.LEQUAL),g.depthMask(!0),this.traverseTreeAndDraw(A),A.popGLStack()}drawGeomData(A){const g=this.__gl;if(A.geomDataFbo&&!A.occlusionCulling){A.pushGLStack("GLLinesPass.drawGeomData"),this.linesGeomDataBuffer||(this.linesGeomDataBuffer=new GLTexture2D(g,{type:this.__renderer.floatGeomBuffer?"FLOAT":"UNSIGNED_BYTE",format:"RGBA",filter:"NEAREST",width:1,height:2}),this.fattenLinesShader=new FattenLinesShader(g),this.quad=new GLMesh(g,new Plane(1,1)));const I=A.geomDataFbo,C=I.width,e=I.height;if(this.linesGeomDataBuffer.width!=C||this.linesGeomDataBuffer.height!=e){this.fbo&&(g.deleteFramebuffer(this.fbo),this.fbo=null),this.linesGeomDataBuffer.resize(C,e),this.fbo=g.createFramebuffer();const A=this.linesGeomDataBuffer.glTex,i=I.depthTexture;g.bindFramebuffer(g.DRAW_FRAMEBUFFER,this.fbo),g.framebufferTexture2D(g.DRAW_FRAMEBUFFER,g.COLOR_ATTACHMENT0,g.TEXTURE_2D,A,0),g.framebufferTexture2D(g.DRAW_FRAMEBUFFER,g.DEPTH_ATTACHMENT,g.TEXTURE_2D,i,0),checkFramebuffer(g,C,e)}else g.bindFramebuffer(g.DRAW_FRAMEBUFFER,this.fbo);g.colorMask(!0,!0,!0,!0),g.clearColor(0,0,0,0),g.clear(g.COLOR_BUFFER_BIT)}if(super.drawGeomData(A),A.geomDataFbo&&!A.occlusionCulling){A.popGLStack(),A.boundRendertarget=null,A.geomDataFbo.bindForWriting(A),this.fattenLinesShader.bind(A);const{colorTexture:I,screenSize:C,pointAndLinePickingSize:e}=A.unifs;this.linesGeomDataBuffer.bindToUniform(A,I),g.uniform1i(e.location,A.pointAndLinePickingSize);const i=A.geomDataFbo;g.uniform2f(C.location,i.width,i.height),this.quad.bindAndDraw(A)}}}GLRenderer.registerPass(GLLinesPass,PassType.OPAQUE);class GLTransparentGeomsPass extends GLStandardGeomsPass{itemCount=0;glShaderGeomSets={};transparentItems=[];transparentItemIndices=new Map;freeList=[];visibleItems=[];prevSortCameraPos=new Vec3(999,999,999);sortCameraMovementDistance=.25;reSort=!1;constructor(){super()}init(A,g){super.init(A,g)}getPassType(){return PassType.TRANSPARENT}filterGeomItem(A){const g=A.geomParam.value;if(g instanceof Lines||g instanceof Points||g instanceof PointsProxy||g instanceof LinesProxy)return!1;const I=A.materialParam.value;return!A.isOpaque()||!I.isOpaque()}resortNeeded(){this.reSort=!0}addGeomItem(A){super.addGeomItem(A),this.itemCount++;const g=this.listenerIDs.get(A),I=A.materialParam.value,C=I.getShaderName(),e=this.constructShader(C);if(!I.isTextured()&&I.getShaderClass().supportsInstancing()){let I=this.glShaderGeomSets[C];I||(I=new GLShaderGeomSets(this.__renderer,this.__gl,e),I.on("updated",(()=>{this.renderer.requestRedraw()})),this.glShaderGeomSets[C]=I);const i=this.renderer.glGeomItemLibrary.getGLGeomItem(A);return I.addGLGeomItem(i),g["glGeomItem.visibilityChanged"]=i.on("visibilityChanged",(()=>{this.resortNeeded()})),this.emit("updated"),i.GLShaderGeomSets=I,void(this.reSort=!0)}const i=this.renderer.glGeomLibrary.constructGLGeom(A.geomParam.value),t=this.renderer.glGeomItemLibrary.getGLGeomItem(A);if(!t)throw new Error("glGeomItem not found for geomItem:"+A.getName());const o=this.renderer.glMaterialLibrary.getGLMaterial(I);g["glGeomItem.visibilityChanged"]=t.on("visibilityChanged",(A=>{if(A.state)this.visibleItems.push(a);else{const A=this.visibleItems.indexOf(a);this.visibleItems.splice(A,1)}this.reSort=!0})),g["GeomMat.valueChanged"]=A.geomMatParam.on("valueChanged",(()=>{this.reSort=!0}));const a={geomItem:A,glShader:e,glGeom:i,glMaterial:o,glGeomItem:t,material:I,dist:0};let B;B=this.freeList.length>0?this.freeList.pop():this.transparentItems.length,this.transparentItems[B]=a,this.transparentItemIndices.set(A,B),A.isVisible()&&this.visibleItems.push(a),this.reSort=!0}removeGeomItem(A){this.itemCount--;const g=this.listenerIDs.get(A);super.removeGeomItem(A);const I=this.renderer.glGeomItemLibrary.getGLGeomItem(A);if(!I)throw new Error("glGeomItem not found for geomItem:"+A.getName());if(I.off("visibilityChanged",g["glGeomItem.visibilityChanged"]),I.GLShaderGeomSets){I.GLShaderGeomSets.removeGLGeomItem(I),I.GLShaderGeomSets=null}else{const g=this.transparentItemIndices.get(A),I=this.transparentItems[g];this.transparentItemIndices.delete(A),this.transparentItems[g]=null,this.freeList.push(g);const C=this.visibleItems.indexOf(I);-1!=C&&this.visibleItems.splice(C,1)}return this.emit("updated"),!0}sortItems(A){for(const g in this.glShaderGeomSets)this.glShaderGeomSets[g].sortItems(A);for(const g of this.visibleItems){const I=g.glGeomItem.geomItem.geomMatParam.value;g.dist=I.translation.distanceTo(A)}this.visibleItems.sort(((A,g)=>A.dist>g.dist?-1:A.dist<g.dist?1:0)),this.reSort=!1}drawItem(A,g,I){I.currentGLMaterial!=g.glMaterial&&(I.currentGLMaterial=g.glMaterial,I.currentGLMaterial.bind(A,!1)),I.currentGLGeom!=g.glGeom&&(I.currentGLGeom=g.glGeom,I.currentGLGeom.bind(A));g.glGeomItem.bind(A),A.bindViewports(A.unifs,(()=>{I.currentGLGeom.draw(A)}))}_drawItems(A){for(const g in this.glShaderGeomSets)this.glShaderGeomSets[g].draw(A);const g={currentglShader:null,currentGLMaterial:null,currentGLGeom:null};for(const I of this.visibleItems){const C=I.glShader;if(g.currentglShader!=C){if(!C.bind(A,"color"))continue;const I=this.__gl,e=A.unifs;e.instancedDraw&&I.uniform1i(e.instancedDraw.location,0),this.renderer.glGeomItemLibrary.bind(A),this.renderer.glGeomLibrary.bind(A),this.renderer.glMaterialLibrary.bind(A),g.currentglShader=C}this.drawItem(A,I,g)}g.currentglShader&&g.currentglShader.unbind(A)}draw(A){if(0==this.itemCount)return;const g=this.__gl,I=A.viewXfo.tr;if(this.reSort||I.distanceTo(this.prevSortCameraPos)>this.sortCameraMovementDistance)if(this.sortItems(I),this.prevSortCameraPos=I,A.xrviewport)this.sortCameraMovementDistance=.2*A.viewScale;else if(A.viewport){const g=A.viewport.getCamera();this.sortCameraMovementDistance=.3*g.getFocalDistance()}A.pushGLStack("GLTransparentGeomsPass.draw"),A.glEnable(g.BLEND),A.glEnable(g.DEPTH_TEST),A.glEnable(g.CULL_FACE),g.depthFunc(g.LESS),g.blendEquation(g.FUNC_ADD),A.pass="ADD",g.blendFuncSeparate(g.SRC_ALPHA,g.ONE_MINUS_SRC_ALPHA,g.ONE,g.ONE),g.cullFace(g.BACK),this._drawItems(A),A.popGLStack()}drawHighlightedGeoms(A){const g=this.__gl;g.disable(g.CULL_FACE);for(const g in this.glShaderGeomSets)this.glShaderGeomSets[g].drawHighlightedGeoms(A);const I={currentglShader:null,currentGLMaterial:null,currentGLGeom:null};for(const C of this.visibleItems){if(!C.geomItem.isHighlighted())continue;const e=C.glShader;if(I.currentglShader!=e){if(!e.bind(A,"highlight"))continue;I.currentglShader=e;const{floatGeomBuffer:C,passId:i,instancedDraw:t}=A.unifs;C&&g.uniform1i(C.location,g.floatGeomBuffer?1:0),i&&g.uniform1i(i.location,this.passIndex),t&&g.uniform1i(t.location,0),this.renderer.glGeomItemLibrary.bind(A)}this.drawItem(A,C,I)}I.currentglShader&&I.currentglShader.unbind(A),I.currentGLGeom&&I.currentGLGeom.unbind(A)}drawGeomData(A){const g=this.__gl;A.pushGLStack("GLTransparentGeomsPass.drawGeomData"),A.glEnable(g.DEPTH_TEST),A.glEnable(g.CULL_FACE);for(const g in this.glShaderGeomSets)this.glShaderGeomSets[g].drawGeomData(A);const I={currentglShader:null,currentGLMaterial:null,currentGLGeom:null};for(const C of this.visibleItems){if(!C.glGeomItem.geomItem.isPickable())continue;const e=C.glShader;if(e){if(I.currentglShader!=e){if(!e.bind(A,"geomdata"))continue;I.currentglShader=e;const{floatGeomBuffer:C,passId:i,instancedDraw:t}=A.unifs;C&&g.uniform1i(C.location,g.floatGeomBuffer?1:0),i&&g.uniform1i(i.location,this.passIndex),t&&g.uniform1i(t.location,0),this.renderer.glGeomItemLibrary.bind(A)}this.drawItem(A,C,I)}}I.currentGLGeom&&I.currentGLGeom.unbind(A),I.currentglShader&&I.currentglShader.unbind(A),A.popGLStack()}}GLRenderer.registerPass(GLTransparentGeomsPass,PassType.TRANSPARENT);const pixelsPerItem=7,ALIGN_TO_CAMERA_FLAG=4,ALIGN_TO_CAMERA_AND_AXIS_FLAG=8,DRAW_ON_TOP_FLAG=16,FIXED_SIZE_ON_SCREEN_FLAG=32,FRONT_FACING_FLAG=64;class GLBillboardsPass extends GLPass{billboards=[];billboardIndices=new Map;dirtyBillboards=new Set;freeIndices=[];drawCount=0;threshold=0;prevSortCameraPos=new Vec3;atlas=null;indexArrayUpdateNeeded=!1;instanceIdsBuffer=null;indexArray=new Float32Array(0);glshader=null;shaderBinding=null;modelMatrixArray=[];billboardDataArray=[];tintColorArray=[];drawItemsTexture=null;constructor(){super()}init(A,g){super.init(A,g);const I=this.renderer.gl;this.atlas=new GLImageAtlas(I,I.RGBA,I.UNSIGNED_BYTE);const C=()=>this.emit("updated");this.atlas.on("loaded",C),this.atlas.on("updated",C)}getPassType(){return PassType.TRANSPARENT}itemAddedToScene(A,g){return A instanceof BillboardItem&&(this.addBillboard(A),!0)}itemRemovedFromScene(A,g){return A instanceof BillboardItem&&(this.removeBillboard(A),!0)}addBillboard(A){const g=A.imageParam,I=g.value;if(!I)return void g.on("valueChanged",(()=>this.addBillboard(A)));let C;C=this.freeIndices.length>0?this.freeIndices.pop():this.billboards.length;const e=this.atlas.addSubImage(I);this.billboardIndices.set(A,C);const i=()=>{A.isVisible()?(this.drawCount++,this.dirtyBillboards.add(C)):this.drawCount--,this.reqUpdateIndexArray()};A.on("visibilityChanged",i);const t=()=>{A.isVisible()&&(this.dirtyBillboards.add(C),this.emit("updated"))};A.on("parameterValueChanged",t),A.on("highlightChanged",t),A.isVisible()&&this.drawCount++,this.billboards[C]={billboard:A,imageIndex:e,dist:0,visibilityChanged:i,updateBillboard:t},this.dirtyBillboards.add(C),this.indexArrayUpdateNeeded=!0,this.emit("updated")}removeBillboard(A){const g=this.billboardIndices.get(A);if(-1==g)return void console.warn("Billboard already removed.");const I=this.billboards[g],C=I.billboard.imageParam.value;this.atlas.removeSubImage(C),A.off("visibilityChanged",I.visibilityChanged),A.off("highlightChanged",I.updateBillboard),A.off("parameterValueChanged",I.updateBillboard),A.isVisible()&&this.drawCount--,this.billboards[g]=null,this.billboardIndices.delete(A),this.freeIndices.push(g),this.dirtyBillboards.has(g)&&this.dirtyBillboards.delete(g),this.indexArrayUpdateNeeded=!0,this.emit("updated")}populateBillboardDataArray(A,g,I){const C=A.billboard,e=C.globalXfoParam.value.toMat4(),i=C.pixelsPerMeterParam.value,t=C.pivotParam.value,o=1/i;let a=0;switch(C.alignmentParam.value){case 0:break;case 1:a|=4;break;case 2:a|=8}C.drawOnTopParam.value&&(a|=16),C.fixedSizeOnscreenParam.value&&(a|=32),C.frontFacingParam.value&&(a|=64);const B=C.alphaParam.value,s=C.colorParam.value,l=7*g*4,n=new Float32Array(I.buffer,4*l,4),d=new Float32Array(I.buffer,4*(l+4),4),r=new Float32Array(I.buffer,4*(l+8),4),c=new Float32Array(I.buffer,4*(l+12),4);n.set([e.xAxis.x,e.yAxis.x,e.zAxis.x,e.translation.x]),d.set([e.xAxis.y,e.yAxis.y,e.zAxis.y,e.translation.y]),r.set([e.xAxis.z,e.yAxis.z,e.zAxis.z,e.translation.z]),c.set([o,a,A.imageIndex,B]);new Float32Array(I.buffer,4*(l+16),4).set([t.x,t.y,0,0]);if(new Float32Array(I.buffer,4*(l+20),4).set([s.r,s.g,s.b,s.a]),C.isHighlighted()){const A=C.getHighlight();new Float32Array(I.buffer,4*(l+24),4).set([A.r,A.g,A.b,A.a])}}reqUpdateIndexArray(){this.indexArrayUpdateNeeded||(this.indexArrayUpdateNeeded=!0,this.emit("updated"))}updateIndexArray(){const A=this.__gl;this.indexArray&&this.indexArray.length!=this.drawCount&&(A.deleteBuffer(this.instanceIdsBuffer),this.instanceIdsBuffer=null),this.indexArray=new Float32Array(this.drawCount);let g=0;for(let A=0;A<this.billboards.length;A++)this.billboards[A]&&this.billboards[A].billboard.isVisible()&&(this.indexArray[g]=A,g++);this.instanceIdsBuffer||(this.instanceIdsBuffer=A.createBuffer()),A.bindBuffer(A.ARRAY_BUFFER,this.instanceIdsBuffer),A.bufferData(A.ARRAY_BUFFER,this.indexArray,A.STATIC_DRAW),this.indexArrayUpdateNeeded=!1}updateBillboards(A){const g=new Set(this.dirtyBillboards);this.dirtyBillboards.clear();const I=()=>{this.indexArrayUpdateNeeded&&this.updateIndexArray();const I=this.__gl;if(!this.glshader){I.__quadVertexIdsBuffer||I.setupInstancedQuad(),this.glshader=new BillboardShader(I);const g=this.glshader.compileForTarget("GLBillboardsPass",A.directives);this.shaderBinding=generateShaderGeomBinding(I,g.attrs,I.__quadattrbuffers,I.__quadIndexBuffer)}if(this.atlas.needsUpdate&&this.atlas.renderAtlas(),!I.floatTexturesSupported||!I.drawElementsInstanced)return this.modelMatrixArray=[],this.billboardDataArray=[],this.tintColorArray=[],void g.forEach((A=>{const g=this.billboards[A],I=g.billboard,C=I.globalXfoParam.value.toMat4(),e=1/I.pixelsPerMeterParam.value;let i=0;switch(I.alignmentParam.value){case 0:break;case 1:i|=4;break;case 2:i|=8}I.drawOnTopParam.value&&(i|=8),I.fixedSizeOnscreenParam.value&&(i|=16);const t=I.alphaParam.value,o=I.colorParam.value;this.modelMatrixArray[A]=C.asArray(),this.billboardDataArray[A]=[e,i,g.imageIndex,t],this.tintColorArray[A]=[o.r,o.g,o.b,o.a]}));let C=Math.ceil(Math.sqrt(7*this.billboards.length));if(C%7!=0&&(C+=7-C%7),this.drawItemsTexture)this.drawItemsTexture.width!=C&&(this.drawItemsTexture.resize(C,C),this.billboards.forEach(((A,g)=>{this.dirtyBillboards.add(g)})));else{const A={format:I.RGBA,type:I.FLOAT,width:C,height:C,filter:I.NEAREST,wrap:I.CLAMP_TO_EDGE,mipMapped:!1};this.drawItemsTexture=new GLTexture2D(I,A),this.drawItemsTexture.clear()}g.forEach((A=>{this.updateBillboard(A)}))};this.atlas.isLoaded()?I():this.atlas.on("loaded",I)}updateBillboard(A){if(!this.billboards[A])return;const g=this.billboards[A];if(!g.billboard.isVisible())return;const I=this.__gl,C=new Float32Array(28);this.populateBillboardDataArray(g,0,C),I.bindTexture(I.TEXTURE_2D,this.drawItemsTexture.glTex);const e=7*A%this.drawItemsTexture.width,i=Math.floor(7*A/this.drawItemsTexture.width),t=this.drawItemsTexture.type,o=this.drawItemsTexture.format;if(t==I.FLOAT)I.texSubImage2D(I.TEXTURE_2D,0,e,i,7,1,o,t,C);else{const A=MathFunctions.convertFloat32ArrayToUInt16Array(C);I.texSubImage2D(I.TEXTURE_2D,0,e,i,7,1,o,t,A)}}sort(A){for(const g of this.billboards){if(!g)continue;const{billboard:I}=g;if(I&&I.isVisible()){const C=I.globalXfoParam.value;g.dist=C.tr.distanceTo(A)}}this.indexArray.sort(((A,g)=>-1==A?1:-1==g||this.billboards[A].dist>this.billboards[g].dist?-1:this.billboards[A].dist<this.billboards[g].dist?1:0));const g=this.__gl;g.floatTexturesSupported&&this.instanceIdsBuffer&&(g.bindBuffer(g.ARRAY_BUFFER,this.instanceIdsBuffer),g.bufferData(g.ARRAY_BUFFER,this.indexArray,g.STATIC_DRAW))}__draw(A,g){const I=this.__gl;if(!this.glshader)return;this.glshader.bind(A,g),this.shaderBinding.bind(A);const C=A.unifs,{atlasBillboards:e,passId:i,floatGeomBuffer:t,inVR:o}=A.unifs;if(e&&this.atlas.bindToUniform(A,e),t&&A instanceof GeomDataRenderState&&I.uniform1i(t.location,A.floatGeomBuffer?1:0),i&&I.uniform1i(i.location,this.passIndex),o&&I.uniform1i(o.location,A.vrPresenting?1:0),I.floatTexturesSupported&&I.drawElementsInstanced){const{instancesTexture:g,instancesTextureSize:e}=A.unifs;this.drawItemsTexture.bindToUniform(A,g),I.uniform1i(e.location,this.drawItemsTexture.width);{const g=A.attrs.instanceIds.location;I.enableVertexAttribArray(g),I.bindBuffer(I.ARRAY_BUFFER,this.instanceIdsBuffer),I.vertexAttribPointer(g,1,I.FLOAT,!1,4,0),I.vertexAttribDivisor(g,1)}A.bindViewports(C,(()=>{I.drawElementsInstanced(I.TRIANGLES,6,I.UNSIGNED_BYTE,0,this.drawCount)}))}else{const{modelMatrix:g,billboardData:e,tintColor:i,layoutData:t}=A.unifs,o=this.indexArray.length;for(let a=0;a<o;a++)I.uniformMatrix4fv(g.location,!1,this.modelMatrixArray[a]),I.uniform4fv(e.location,this.billboardDataArray[a]),I.uniform4fv(i.location,this.tintColorArray[a]),I.uniform4fv(t.location,this.atlas.getLayoutData(this.billboards[a].imageIndex)),A.bindViewports(C,(()=>{I.drawQuad()}))}}draw(A){if(0==this.drawCount)return;if(this.dirtyBillboards.size>0&&this.updateBillboards(A),this.indexArrayUpdateNeeded&&this.updateIndexArray(),!this.glshader)return;const g=A.viewXfo.tr;if(g.distanceTo(this.prevSortCameraPos)>this.threshold)if(this.sort(g),this.prevSortCameraPos=g.clone(),this.drawCount>1){const A=this.indexArray[this.indexArray.length-1],g=this.indexArray[this.indexArray.length-2],I=this.billboards[A].billboard,C=this.billboards[g].billboard,e=I.globalXfoParam.value.tr,i=C.globalXfoParam.value.tr;this.threshold=e.distanceTo(i)}else this.threshold=9999;const I=this.__gl;I.depthMask(!1),I.disable(I.CULL_FACE),I.enable(I.BLEND),I.blendEquation(I.FUNC_ADD),I.blendFuncSeparate(I.SRC_ALPHA,I.ONE_MINUS_SRC_ALPHA,I.ONE,I.ONE),this.__draw(A,"DRAW_COLOR"),I.disable(I.BLEND),I.depthMask(!0)}drawHighlightedGeoms(A){0!=this.drawCount&&this.__draw(A,"DRAW_HIGHLIGHT")}drawGeomData(A){0!=this.drawCount&&this.__draw(A,"DRAW_GEOMDATA")}getGeomItemAndDist(A){let g,I;if(A instanceof Float32Array?(g=Math.round(A[1]),I=A[3]):(g=A[0]+((63&A[1])<<8),I=MathFunctions.decode16BitFloatFrom2xUInt8(A.slice(2,3))),!(g>=this.billboards.length))return{geomItem:this.billboards[g].billboard,componentId:0,dist:I};console.warn("Invalid Draw Item id:"+g+" NumBillboards:"+(this.billboards.length-1))}}GLRenderer.registerPass(GLBillboardsPass,PassType.TRANSPARENT);class GLOverlayPass extends GLOpaqueGeomsPass{constructor(){super()}getPassType(){return PassType.OVERLAY}filterGeomItem(A){if(A.isOverlay())return!0;const g=A.materialParam.value.getShaderClass();return!(!g||!g.isOverlay())}draw(A){const g=this.__gl;g.clear(g.DEPTH_BUFFER_BIT),g.enable(g.CULL_FACE),g.cullFace(g.BACK),g.enable(g.BLEND),g.blendEquation(g.FUNC_ADD),A.pass="ADD",g.blendFuncSeparate(g.SRC_ALPHA,g.ONE_MINUS_SRC_ALPHA,g.ONE,g.ONE),this.traverseTreeAndDraw(A),g.disable(g.BLEND)}drawGeomData(A){const g=this.__gl;g.clear(g.DEPTH_BUFFER_BIT),g.enable(g.CULL_FACE),g.cullFace(g.BACK),g.enable(g.BLEND),g.blendEquation(g.FUNC_ADD),A.pass="ADD",g.blendFuncSeparate(g.SRC_ALPHA,g.ONE_MINUS_SRC_ALPHA,g.ONE,g.ONE),super.drawGeomData(A),g.disable(g.BLEND),g.enable(g.DEPTH_TEST)}}GLRenderer.registerPass(GLOverlayPass,PassType.OVERLAY),Registry.register("GLOverlayPass",GLOverlayPass);class GLBoundingBoxPass extends GLPass{boxes=[];dirtyBoxes=new Set;freeIndices=[];idToIndex=new Map;drawCount=0;indexArrayUpdateNeeded=!1;__updateRequested=!1;glgeom;glshader;__modelMatrixArray=[];__treeItemDataArray=[];__tintColorArray=[];__instanceIdsBuffer;__indexArray=new Float32Array(0);__drawItemsTexture;width=0;constructor(){super()}getPassType(){return PassType.OPAQUE}init(A,g){super.init(A,g);const I=this.__renderer.gl;this.glgeom=new GLLines(I,new LinesCuboid(1,1,1)),this.glshader=new BoundingBoxShader(I)}itemAddedToScene(A,g){return!1}itemRemovedFromScene(A,g){return!1}addTreeItem(A,g=!0){if(A instanceof TreeItem&&(this.bindTreeItem(A),g)){for(const g of A.getChildren())g&&this.addTreeItem(g);A.on("childAdded",(A=>{this.addTreeItem(A.childItem)})),A.on("childRemoved",(A=>{this.unbindTreeItem(A.childItem)}))}}bindTreeItem(A){let g,I=this.freeIndices.pop();g=I||this.boxes.length,this.idToIndex.set(A,g);const C=()=>{A.isVisible()?(this.drawCount++,this.dirtyBoxes.add(g)):this.drawCount--,this.indexArrayUpdateNeeded=!0};A.on("visibilityChanged",C);const e=()=>{A.isVisible()&&(this.dirtyBoxes.add(g),this.emit("updated"))};A.globalXfoParam.on("valueChanged",e),A.boundingBoxParam.on("valueChanged",e),A.isVisible()&&this.drawCount++,this.boxes[g]={treeItem:A,visibilityChanged:C,xfoChanged:e},this.indexArrayUpdateNeeded=!0,this.__updateRequested=!0,this.emit("updated")}unbindTreeItem(A){if(!this.idToIndex.has(A))return void console.warn("Billboard already removed.");const g=this.idToIndex.get(A),I=this.boxes[g];A.off("visibilityChanged",I.visibilityChanged),A.globalXfoParam.off("valueChanged",I.xfoChanged),A.boundingBoxParam.off("valueChanged",I.xfoChanged),this.boxes[g]=null,this.freeIndices.push(g),A.isVisible()&&this.drawCount--,this.indexArrayUpdateNeeded=!0,this.__updateRequested=!0,this.__updateBoxes(),this.emit("updated")}__populateBoxesDataArray(A,g,I){const C=A.treeItem;let e,i;C instanceof GeomItem?(e=new Color(1,0,0,1),i=C.geomMatParam.value):(e=new Color(0,0,1,1),i=C.globalXfoParam.value.toMat4());const t=C.boundingBoxParam.value,o=8*g*4,a=new Float32Array(I.buffer,4*o,4),B=new Float32Array(I.buffer,4*(o+4),4),s=new Float32Array(I.buffer,4*(o+8),4),l=new Float32Array(I.buffer,4*(o+12),4),n=new Float32Array(I.buffer,4*(o+16),4),d=new Float32Array(I.buffer,4*(o+24),4),r=new Float32Array(I.buffer,4*(o+28),4);a.set([0,0,0,0]),B.set([i.xAxis.x,i.yAxis.x,i.zAxis.x,i.translation.x]),s.set([i.xAxis.y,i.yAxis.y,i.zAxis.y,i.translation.y]),l.set([i.xAxis.z,i.yAxis.z,i.zAxis.z,i.translation.z]),n.set([e.r,e.g,e.b,e.a]),d.set([t.p0.x,t.p0.y,t.p0.z,0]),r.set([t.p1.x,t.p1.y,t.p1.z,0])}__updateIndexArray(){const A=this.__gl;this.__indexArray&&this.__indexArray.length!=this.drawCount&&(A.deleteBuffer(this.__instanceIdsBuffer),this.__instanceIdsBuffer=void 0),this.__indexArray=new Float32Array(this.drawCount);let g=0;for(let A=0;A<this.boxes.length;A++)this.boxes[A]&&this.boxes[A].treeItem.isVisible()&&(this.__indexArray[g]=A,g++);this.__instanceIdsBuffer||(this.__instanceIdsBuffer=A.createBuffer()),A.bindBuffer(A.ARRAY_BUFFER,this.__instanceIdsBuffer),A.bufferData(A.ARRAY_BUFFER,this.__indexArray,A.STATIC_DRAW),this.indexArrayUpdateNeeded=!1}__updateBoxes(){this.indexArrayUpdateNeeded&&this.__updateIndexArray();const A=this.__renderer.gl;let g=Math.round(Math.sqrt(8*(this.boxes.length-this.freeIndices.length))+.5);g%8!=0&&(g+=8-g%8),this.width=g,this.__drawItemsTexture?this.__drawItemsTexture.resize(g,g):(this.__drawItemsTexture=new GLTexture2D(A,{format:"RGBA",type:"FLOAT",width:g,height:g,filter:"NEAREST",wrap:"CLAMP_TO_EDGE",mipMapped:!1}),this.__drawItemsTexture.clear()),this.__indexArray.forEach((A=>{-1!=A&&this.__updateBox(A)})),this.__updateRequested=!1}__updateBox(A){if(0==this.drawCount||!this.__drawItemsTexture)return;const g=this.boxes[A];if(!g.treeItem.isVisible())return;const I=this.__gl,C=new Float32Array(32);this.__populateBoxesDataArray(g,0,C),I.bindTexture(I.TEXTURE_2D,this.__drawItemsTexture.glTex);const e=8*A%this.width,i=Math.floor(8*A/this.width),t=this.__drawItemsTexture.type,o=this.__drawItemsTexture.format;if(t==I.FLOAT)I.texSubImage2D(I.TEXTURE_2D,0,e,i,8,1,o,t,C);else{const A=MathFunctions.convertFloat32ArrayToUInt16Array(C);I.texSubImage2D(I.TEXTURE_2D,0,e,i,8,1,o,t,A)}}draw(A){if(0==this.drawCount)return;this.__updateRequested&&this.__updateBoxes(),this.dirtyBoxes.size>0&&(this.dirtyBoxes.forEach((A=>{this.__updateBox(A)})),this.dirtyBoxes.clear()),this.indexArrayUpdateNeeded&&this.__updateIndexArray();const g=this.__gl;this.glshader.bind(A),this.glgeom.bind(A);const I=A.unifs;if(g.floatTexturesSupported&&g.drawElementsInstanced){this.__drawItemsTexture.bindToUniform(A,I.instancesTexture),g.uniform1i(I.instancesTextureSize.location,this.width);{const I=A.attrs.instancedIds.location;g.enableVertexAttribArray(I),g.bindBuffer(g.ARRAY_BUFFER,this.__instanceIdsBuffer),g.vertexAttribPointer(I,1,g.FLOAT,!1,4,0),g.vertexAttribDivisor(I,1)}g.uniform1i(I.instancedDraw.location,1),A.bindViewports(I,(()=>{this.glgeom.drawInstanced(A,this.drawCount)}))}else{const C=this.__indexArray.length;for(let e=0;e<C;e++)A.bindViewports(I,(()=>{g.drawQuad()}))}}}class GLMaskGeomsPass extends GLOpaqueGeomsPass{frontRenderTarget;backRenderTarget;itemCount=0;drawMaskGeoms=!1;init(A,g){super.init(A,g);const I=A.gl;this.frontRenderTarget=new GLRenderTarget(I,{numColorChannels:0,minFilter:I.NEAREST,magFilter:I.NEAREST,width:4,height:4,depthType:I.UNSIGNED_SHORT,depthFormat:I.DEPTH_COMPONENT,depthInternalFormat:I.DEPTH_COMPONENT16}),this.frontRenderTarget.clearColor=new Color(0,0,0,0),this.backRenderTarget=new GLRenderTarget(I,{numColorChannels:0,minFilter:I.NEAREST,magFilter:I.NEAREST,width:4,height:4,depthType:I.UNSIGNED_SHORT,depthFormat:I.DEPTH_COMPONENT,depthInternalFormat:I.DEPTH_COMPONENT16}),this.backRenderTarget.clearColor=new Color(0,0,0,0)}filterGeomItem(A){return A.materialParam.value instanceof MaskMaterial}addGeomItem(A){super.addGeomItem(A),this.itemCount++}removeGeomItem(A){super.removeGeomItem(A),this.itemCount--}draw(A){if(0==this.itemCount)return;A.pushGLStack("GLMaskGeomsPass.draw");const g=this.__gl;this.drawMaskGeoms&&(A.glEnable(g.BLEND),g.blendEquation(g.FUNC_ADD),g.blendFuncSeparate(g.SRC_ALPHA,g.ONE_MINUS_SRC_ALPHA,g.ONE,g.ONE),this.traverseTreeAndDraw(A),A.glDisable(g.BLEND),g.clear(g.DEPTH_BUFFER_BIT));const I=A.region[2],C=A.region[3];I==this.frontRenderTarget.width&&C==this.frontRenderTarget.height||(this.frontRenderTarget.resize(I,C),this.backRenderTarget.resize(I,C)),g.depthFunc(g.LESS),g.depthMask(!0),A.glEnable(g.DEPTH_TEST),this.frontRenderTarget.bindForWriting(A,!0),this.traverseTreeAndDraw(A),this.frontRenderTarget.unbindForWriting(A),this.backRenderTarget.bindForWriting(A,!0),g.depthFunc(g.GREATER),g.clearDepth(0),g.clear(g.DEPTH_BUFFER_BIT),g.cullFace(g.FRONT),this.traverseTreeAndDraw(A),this.backRenderTarget.unbindForWriting(A),g.clearDepth(1),g.depthFunc(g.LESS),g.cullFace(g.BACK),A.geometryMaskTextures=[this.frontRenderTarget.depthTexture,this.backRenderTarget.depthTexture],A.popGLStack()}drawGeomData(A){}}GLRenderer.registerPass(GLMaskGeomsPass,PassType.PRE),console.log(`Zea Engine v${version}`);const libsRegistry=new LibsRegistry(version);export{Allocation1D,Allocator1D,AngleParameter,AssetItem,AssetLoadContext,Attribute,BaseClass,BaseEvent,BaseGeom,BaseGeomItem,BaseGroup,BaseImage,BaseItem,BaseProxy,BaseTool,BillboardAlignment,BillboardItem,BillboardShader,BinReader,BinWriter,BooleanOperatorInput,BooleanOperatorOutput,BooleanParameter,Box2,Box2Parameter,Box3,Box3Parameter,CADAssembly,CADAsset,CADBody,CADPart,Camera,CameraManipulator,ChildAddedEvent,Circle,CloneContext,Color,ColorAttribute,ColorOperatorInput,ColorOperatorOutput,ColorParameter,ColorRenderState,ColorSpace,CompoundGeom,Cone,ControllerAddedEvent,CountChangedEvent,Cross,Cuboid,CuttingPlane,Cylinder,DataImage,Disc,EnvMap,EnvMapAssignedEvent,EnvMapShader,EnvProjectionMaterial,EnvProjectionShader,EulerAngles,EulerAnglesAxisOrder,EventEmitter,FRAMEBUFFER,FatLinesMaterial,FatLinesShader,FatPoints,FatPointsMaterial,FatPointsShader,FileImage,FileImage2D,FlatSurfaceMaterial,FlatSurfaceShader,Float32,Float32ArrayParameter,Frustum,GIFImage,GLBaseRenderer,GLBaseViewport,GLBillboardsPass,GLBoundingBoxPass,GLCADPass,GLFbo,GLGeom,GLGeomItem,GLGeomItemChangeType,GLGeomItemFlags,GLGeomItemSet,GLGeomItemSetMultiDraw,GLGeomItemSetMultiDrawCompoundGeom,GLLines,GLLinesItemSet,GLLinesPass,GLMaskGeomsPass,GLMaterial,GLMaterialGeomItemSets,GLMesh,GLMeshItemSet,GLOpaqueGeomsPass,GLOverlayPass,GLPass,GLPoints,GLPointsItemSet,GLRenderTarget,GLRenderer,GLShader,GLShaderGeomSets,GLShaderMaterials,GLStandardGeomsPass,GLTexture2D,GLTransparentGeomsPass,GLViewport,GROUP_XFO_MODES,GeomDataRenderState,GeomItem,GeomLibrary,GeomType$2 as GeomType,GeometryParameter,Grid,GridTreeItem,GrowingPacker,HDRImage,HighlightRenderState,IGeomShaderBinding,ImageParameter,IndexDBCache,IndexEvent,InstanceItem,IntersectionData,ItemEvent,ItemSetParameter,KeyboardEvent,KinematicGroup,LDRImage,LDRVideo,Label,LabelManager,Lines,LinesCuboid,LinesCylinder,LinesMaterial,LinesProxy,LinesShader,LinesSphere,ListParameter,MaskMaterial,Mat3,Mat3OperatorInput,Mat3OperatorOutput,Mat3Parameter,Mat4,Mat4OperatorInput,Mat4OperatorOutput,Mat4Parameter,Material,MaterialColorParam,MaterialFloatParam,MaterialGroup,MaterialLibrary,MaterialParameter,MathFunctions,Mesh,MeshProxy,MultiChoiceParameter,NameChangedEvent,NumberOperatorInput,NumberOperatorOutput,NumberParameter,ObjAsset,OpacityStateChangedEvent,Operator,OperatorInput,OperatorOutput,OperatorOutputMode,PMIItem,PMIView,POINTER_TYPES,Parameter,ParameterAddedEvent,ParameterOwner,ParameterRemovedEvent,PassType,PlaceholderGeomItem,Plane,PlaneType,PointGrid,Points,PointsMaterial,PointsProxy,PointsShader,ProceduralLines,ProceduralMesh,ProceduralPoints,ProgressEvent,Quat,QuatOperatorInput,QuatOperatorOutput,QuatParameter,RGBA,RangeLoadedEvent,Ray,Rect,RefCounted,Registry,RenderState,ResizedEvent,ResourceLoader,SInt16,SInt32,SInt8,Scene,SceneSetEvent,ScreenQuadShader,ScreenSpaceMaterial,ScreenSpaceShader,SelectabilityChangedEvent,SelectedEvent,SelectionSet,ShaderLibrary,ShaderNameChangedEvent,SimpleSurfaceMaterial,SimpleSurfaceShader,Sphere,SphereType,StandardSurfaceMaterial,StandardSurfaceShader,StateChangedEvent,StreamFileParsedEvent,StringFunctions,StringListParameter,StringParameter,StructParameter,SystemDesc,TexturedChangedEvent,Torus,Touch,TreeItem,TreeItemParameter,UInt16,UInt32,UInt8,UnpackHDRShader,VLAAsset,VRController,VRViewport,Vec2,Vec2Attribute,Vec2OperatorInput,Vec2OperatorOutput,Vec2Parameter,Vec3,Vec3Attribute,Vec3OperatorInput,Vec3OperatorOutput,Vec3Parameter,Vec4,Vec4Attribute,Vec4OperatorInput,Vec4OperatorOutput,Vec4Parameter,Version,VertexColorMaterial,VertexColorShader,VideoStreamImage2D,ViewChangedEvent,XRController,XRControllerEvent,XRHead,XRPointerEvent,XRPoseEvent,XRViewChangedEvent,XRViewManipulator,XRViewport,XRef,Xfo,XfoOperatorInput,XfoOperatorOutput,XfoParameter,XrViewportEvent,ZeaKeyboardEvent,ZeaMouseEvent,ZeaPointerEvent,ZeaTouchEvent,ZeaUIEvent,ZeaWheelEvent,archiveUnpackerWorkerPool,checkFramebuffer,create3DContext,genDataTypeDesc,generateShaderGeomBinding,geomParserWorkerPool,getFileFolder,labelManager,libsRegistry,loadBinfile,loadJSONfile,loadTextfile,loadXMLfile,resourceLoader,shaderLibrary};
//# sourceMappingURL=/sm/148b49714453fb395ea07554986a021f8d82c877aecf55dae3d67ec13c232b22.map