diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..e4e7572 --- /dev/null +++ b/.gitignore @@ -0,0 +1,4 @@ +package-lock.json +node_modules +.idea +.DS* \ No newline at end of file diff --git a/dist/flexgl.js b/dist/flexgl.js new file mode 100644 index 0000000..7c061bc --- /dev/null +++ b/dist/flexgl.js @@ -0,0 +1,2 @@ +!function(e){var t={};function r(n){if(t[n])return t[n].exports;var i=t[n]={i:n,l:!1,exports:{}};return e[n].call(i.exports,i,i.exports,r),i.l=!0,i.exports}r.m=e,r.c=t,r.d=function(e,t,n){r.o(e,t)||Object.defineProperty(e,t,{enumerable:!0,get:n})},r.r=function(e){"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},r.t=function(e,t){if(1&t&&(e=r(e)),8&t)return e;if(4&t&&"object"==typeof e&&e&&e.__esModule)return e;var n=Object.create(null);if(r.r(n),Object.defineProperty(n,"default",{enumerable:!0,value:e}),2&t&&"string"!=typeof e)for(var i in e)r.d(n,i,function(t){return e[t]}.bind(null,i));return n},r.n=function(e){var t=e&&e.__esModule?function(){return e.default}:function(){return e};return r.d(t,"a",t),t},r.o=function(e,t){return Object.prototype.hasOwnProperty.call(e,t)},r.p="",r(r.s=0)}([function(e,t,r){"use strict";function n(e,t,r,i){var a=this instanceof n?this:{},u=e;function o(e){var t=[];return e.forEach(function(e){t=t.concat(e)}),t}return a.create=function(e,t,r){Array.isArray(r)&&(r.filter(function(e){return Array.isArray(e)})&&(r=o(r)));return a[e]={type:t,name:e,data:r,location:null,size:parseInt(t.slice(3,4))||parseInt(t.slice(4,5))||1},a[e].link=function(e){return void 0!==this.data&&null!==this.data&&(this.location=u.getUniformLocation(e,this.name),function(){var e,t=this.type,r=this.location,n=this.size,i=this.data;Array.isArray(i)&&i.filter(function(e){return Array.isArray(e)})&&(i=o(i)),"float"!=t&&"int"!=t||Array.isArray(i)||ArrayBuffer.isView(i)||(i=[i]),"vec"==t.slice(0,3)||"float"==t?(e=new Float32Array(i),u["uniform"+n+"fv"](r,e)):"ivec"==t.slice(0,4)||"int"==t?(e=new Int32Array(i),u["uniform"+n+"iv"](r,e)):"mat"==t.slice(0,3)?(e=new Float32Array(i),u["uniformMatrix"+n+"fv"](r,!1,e)):"sampler2D"==t&&i.hasOwnProperty("resourceType")&&"texture"==i.resourceType&&(u.activeTexture(u.TEXTURE0+i.index),u.bindTexture(u.TEXTURE_2D,i.ptr),u.uniform1i(r,i.index))}.call(this)),this},a[e].load=function(e){return this.data=e,this},a[e].header=function(){var e="uniform "+this.type+" "+this.name,r=0;return"sampler2D"!=this.type&&(r=this.data.length/this.size),r>1&&"mat4"!=t&&(e+="["+r+"]"),e+";\n"},a[e]},a}function i(e){var t=this instanceof i?this:{},r=e,a=0;function u(e,n){var i=r[t[e].type.toUpperCase()],a=r[t[e].channel.toUpperCase()],u=t[e].dim[0],o=t[e].dim[1];t[e].data=n,r.bindTexture(r.TEXTURE_2D,t[e].ptr),r.texImage2D(r.TEXTURE_2D,0,a,u,o,0,a,i,n),r.texParameteri(r.TEXTURE_2D,r.TEXTURE_MIN_FILTER,r.NEAREST),r.texParameteri(r.TEXTURE_2D,r.TEXTURE_MAG_FILTER,r.NEAREST),r.texParameteri(r.TEXTURE_2D,r.TEXTURE_WRAP_S,r.CLAMP_TO_EDGE),r.texParameteri(r.TEXTURE_2D,r.TEXTURE_WRAP_T,r.CLAMP_TO_EDGE),r.bindTexture(r.TEXTURE_2D,null)}return t.create=function(i,o,s,f,c,l){var d=t.hasOwnProperty(i)?t[i].index:a++;return t[i]={name:i,index:d,type:o||"float",dim:s||[512,512],channel:f||"alpha",data:null,location:null,sampler:l||null,ptr:r.createTexture()},u(i,c),null===t[i].sampler?t[i].sampler=n(r).create(i,"sampler2D",t[i]):t[i].sampler.data=t[i],t[i].link=function(e){return void 0!==this.sampler.data&&null!==this.sampler.data||(this.sampler.data=t[i]),this.sampler.link(e),this},t[i].load=function(e){return u(this.name,e),this},t[i].copyFromFBO=function(){r.bindTexture(r.TEXTURE_2D,this.ptr),r.copyTexImage2D(r.TEXTURE_2D,0,r.RGBA,0,0,this.dim[0],this.dim[1],0),r.bindTexture(r.TEXTURE_2D,null)},t[i].update=function(e,n,i){return function(e,n,i,a){var u=r[t[e].type.toUpperCase()],o=r[t[e].channel.toUpperCase()],s=a[0]||t[e].dim[0],f=a[1]||t[e].dim[1];r.bindTexture(r.TEXTURE_2D,t[e].ptr),r.texSubImage2D(r.TEXTURE_2D,0,i[0],i[1],s,f,o,u,n),r.bindTexture(r.TEXTURE_2D,null)}(this.name,e,n,i),this},t[i].resize=function(e,t){this.dim=e,u(this.name,t)},t[i].delete=function(){e.deleteTexture(this.ptr)},t[i].header=function(){return this.name==this.sampler.name?"uniform sampler2D "+this.sampler.name+";\n":""},t[i]},t}function a(e){var t=this instanceof a?this:{},r={};t.uniform=new n(e),t.attribute=new function e(t){var r=this instanceof e?this:{},n=t,i=0;function a(e,t){(Array.isArray(t)||ArrayBuffer.isView(t))&&(ArrayBuffer.isView(t)||(t=new Float32Array(t)),r[e].data=t,n.bindBuffer(n.ARRAY_BUFFER,r[e].ptr),n.bufferData(n.ARRAY_BUFFER,t,n.STATIC_DRAW))}return r.create=function(e,t,u){return r[e]={name:e,type:t||"float",data:null,location:i++,ptr:n.createBuffer(),size:parseInt(t.slice(3,4))||1},null!==u&&u.length&&a(e,u),r[e].link=function(e){return n.bindBuffer(n.ARRAY_BUFFER,this.ptr),this.location=n.getAttribLocation(e,this.name),n.vertexAttribPointer(this.location,this.size,n.FLOAT,!1,0,0),n.enableVertexAttribArray(this.location),this},r[e].load=function(e){return a(this.name,e),this},r[e].header=function(){return"attribute "+this.type+" "+this.name+";\n"},r[e].delete=function(){n.deleteBuffer(this.ptr)},r[e]},r}(e),t.texture=new i(e),t.varying=new function e(t){var r=this instanceof e?this:{};return r.create=function(e,t,n){return r[e]={name:e,type:t||"float",size:n||1},r[e].link=function(){},r[e].header=function(){var e="varying "+this.type+" "+this.name;return this.size>1&&(e+="["+this.size+"]"),e+";\n"},r[e]},r}(e),t.subroutine=new function e(){var t=this instanceof e?this:{};return t.create=function(e,r,n){return t[e]={name:e,type:r||"float",fn:n,resourceType:"subroutine"},t[e].link=function(e){return this},t[e].load=function(r){return t[e].fn=r,this},t[e].header=function(){return this.fn.toString()},t[e]},t};var u=["uniform","attribute","texture","varying","subroutine"];return t.allocate=function(e){if(-1===u.indexOf(e))throw Error("Error: Invalid resource type: "+e);var n=t[e].create.apply(null,Array.prototype.slice.call(arguments,1));return n.resourceType=e,r[n.name]=n,r.hasOwnProperty(n.name)||Object.defineProperty(r,n.name,{get:function(){return r[n.name]},set:function(e){r[n.name].load(e)}}),n},t.link=function(e,t){(Array.isArray(t)?t:Object.keys(r)).forEach(function(t){r.hasOwnProperty(t)&&r[t].link(e)})},t.get=function(e){return r[e]},t.create=t.allocate,t}function u(e,t){var r={},n=e,i={},a=new function e(t){var r=this instanceof e?this:{},n=t;return r.create=function(e,t,i){function a(e,t,r){var n=e.createShader(t);if(e.shaderSource(n,r),e.compileShader(n),e.getShaderParameter(n,e.COMPILE_STATUS))return n;console.log(e.getShaderInfoLog(n)),e.deleteShader(n)}function u(e,t){var r=/\s*(attribute|uniform)\s+\w+\s+(\w+)/;e.split("\n").forEach(function(e){var n=r.exec(e);n&&t.push(n[2])})}r[e]={name:e,vertex_shader_source:t,fragment_shader_source:i,vs:a(n,n.VERTEX_SHADER,t),fs:a(n,n.FRAGMENT_SHADER,i)},r[e].vs.deps=[],u(t,r[e].vs.deps),r[e].fs.deps=[],u(i,r[e].vs.deps)},r}(e);return r.create=function(e,t,r){e=e||"default",t=t||"default",r=r||"default";var a=[];if(i.hasOwnProperty(e)&&this.delete(e),i[e]=n.createProgram(),i[e].vs=t,i[e].fs=r,n.attachShader(i[e],i[e].vs),n.attachShader(i[e],i[e].fs),n.linkProgram(i[e]),!n.getProgramParameter(i[e],n.LINK_STATUS))throw"Error in program linking:"+n.getProgramInfoLog(i[e]);a=(a=a.concat(i[e].vs.deps)).concat(i[e].fs.deps),i[e].deps=a},r.use=function(e,r,u){return i.hasOwnProperty(e)?(n.useProgram(i[e]),t.link(i[e],i[e].deps),i[e]):(a.create(e,r,u),this.create(e,a[e].vs,a[e].fs),n.useProgram(i[e]),t.link(i[e],i[e].deps),i[e])},r.delete=function(e){i.hasOwnProperty(e)&&(n.detachShader(i[e],i[e].vs),n.detachShader(i[e],i[e].fs),n.deleteProgram(i[e]),delete i[e])},r}function o(e){var t=this instanceof o?this:{},r=e||{},n=r.container||null,i=r.canvas||document.createElement("canvas"),s=r.width||null,f=r.height||null,c=r.padding||{left:0,right:0,top:0,bottom:0},l=r.context||r.ctx||null;r.sharedFunction;"string"==typeof i&&(i="#"==i[0]?document.getElementById(cavnas.substring(1)):document.getElementById(cavnas)),n&&(n="string"==typeof n?document.getElementById(n):n,null===s&&(s=n.clientWidth),null===f&&(f=n.clientHeight)),i.width=s,i.height=f,i.style.position="absolute",i.style.marginLeft=c.left+"px",i.style.marginTop=c.top+"px",null===l&&(l=function(e){for(var t=["webgl","experimental-webgl"],r=null,n=0;nd.attribute[e],set(t){d.attribute[e].load(t)}}),t},t.uniform=function(e,r,n){return d.allocate("uniform",e,r,n),t.uniform.hasOwnProperty(e)||Object.defineProperty(t.uniform,e,{get:function(){return d.uniform[e]},set:function(t){d.uniform[e].load(t),l.isProgram(m)&&d.uniform[e].link(m)}}),t},t.uniform.serialize=function(e){var t=[];return e.forEach(function(e){t=t.concat(e)}),t},t.texture=function(e,r,n,i,a,u){return d.allocate("texture",e,r,i,a,n,u),Object.defineProperty(t.texture,e,{get:function(){return d.texture[e]},set:function(t){d.texture[e].load(t)}}),t},t.texture.update=function(e,t,r,n){d.texture[e].update(t,r,n)},t.varying=function(e,r,n){return d.allocate("varying",e,r,n),t},t.framebuffer=function(e,r,n,i){i=i||d.allocate("texture",e,r,n,"rgba",null);return h.create(e,r,n,i),t.framebuffer.hasOwnProperty(e)||Object.defineProperty(t.framebuffer,e,{get:function(){return h[e]}}),t},t.framebuffer.enableRead=function(e,t){h[e].enableRead(t)},t.bindFramebuffer=function(e){null===e?l.bindFramebuffer(l.FRAMEBUFFER,null):l.bindFramebuffer(l.FRAMEBUFFER,h[e].ptr)},t.subroutine=function(e,r,n){return d.allocate("subroutine",e,r,n),t},t.parameter=function(e){return Object.keys(e).forEach(function(t){if(l._dict[t]=e[t],Array.isArray(l._dict[t])){var r=0;Object.defineProperty(l._dict,t,{get:function(){return e[t][r++]},set:function(e){r=0,l._dict[t]=e}})}else if("object"==typeof l._dict[t]){var n=Object.keys(l._dict[t]);fxgl.uniform("dict"+t,"float",n.map(e=>l._dict[t][e]))}}),t},t.dictionary=t.parameter,t.shader=p.shader,t.app=function(e,r){return function(n){return m=p.use(e,r.vertex_shader_source,r.fragment_shader_source),r.render.call(t,n)}},t.dimension=function(){return[i.width,i.height]},t}r.r(t),"undefined"!=typeof window&&(window.FlexGL=o)}]); +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///webpack/bootstrap","webpack:///./src/uniform.js","webpack:///./src/texture.js","webpack:///./src/resource.js","webpack:///./src/attribute.js","webpack:///./src/varying.js","webpack:///./src/subroutine.js","webpack:///./src/program.js","webpack:///./src/shader.js","webpack:///./src/flexgl.js","webpack:///./src/framebuffer.js","webpack:///./src/bundle.js"],"names":["installedModules","__webpack_require__","moduleId","exports","module","i","l","modules","call","m","c","d","name","getter","o","Object","defineProperty","enumerable","get","r","Symbol","toStringTag","value","t","mode","__esModule","ns","create","key","bind","n","object","property","prototype","hasOwnProperty","p","s","Uniform","glContext","type","data","uniform","this","ctx","serializeArray","arrayOfArray","sa","forEach","a","concat","Array","isArray","filter","location","size","parseInt","slice","link","program","getUniformLocation","buf","ArrayBuffer","isView","Float32Array","Int32Array","resourceType","activeTexture","TEXTURE0","index","bindTexture","TEXTURE_2D","ptr","uniform1i","load","header","len","length","Texture","texture","textureID","setTexture","texData","toUpperCase","format","channel","width","dim","height","texImage2D","texParameteri","TEXTURE_MIN_FILTER","NEAREST","TEXTURE_MAG_FILTER","TEXTURE_WRAP_S","CLAMP_TO_EDGE","TEXTURE_WRAP_T","sampler","texIndex","createTexture","copyFromFBO","copyTexImage2D","RGBA","update","offset","texSubImage2D","updateTexture","resize","delete","deleteTexture","Resource","resource","gpuResources","attribute","Attribute","attributeID","setAttribute","bindBuffer","ARRAY_BUFFER","bufferData","STATIC_DRAW","createBuffer","getAttribLocation","vertexAttribPointer","FLOAT","enableVertexAttribArray","arrayBuffer","deleteBuffer","varying","Varying","subroutine","Subroutine","fn","toString","resourceTypes","allocate","indexOf","Error","res","apply","arguments","set","resource_names","keys","resourceName","Program","resources","kernels","shader","Shader","vertex_shader_source","fragment_shader_source","createShader","gl","source","shaderSource","compileShader","getShaderParameter","COMPILE_STATUS","console","log","getShaderInfoLog","deleteShader","addDeps","deps","re","split","v","result","exec","push","vs","VERTEX_SHADER","fs","FRAGMENT_SHADER","createProgram","attachShader","linkProgram","getProgramParameter","LINK_STATUS","getProgramInfoLog","use","useProgram","detachShader","deleteProgram","FlexGL","arg","flexgl","options","container","canvas","document","createElement","padding","left","right","top","bottom","context","sharedFunction","getElementById","cavnas","substring","clientWidth","clientHeight","style","position","marginLeft","marginTop","names","getContext","e","setupWebGL","_dict","env","dict","dictionary","framebuffers","Framebuffer","framebuffer","createFramebuffer","bindFramebuffer","FRAMEBUFFER","framebufferTexture2D","COLOR_ATTACHMENT0","enableRead","bindRenderbuffer","RENDERBUFFER","deleteRenderbuffer","renderbuffer","deleteFramebuffer","realProgram","blendExt","getExtension","enableExtension","extensions","extension","extProps","ep","ext","MAX_EXT","MIN_EXT","appendChild","[object Object]","isProgram","serialize","aoa","fbName","parameter","keyValuePairs","newArray","dictKeys","fxgl","map","app","args","render","dimension","window"],"mappings":"aACA,IAAAA,KAGA,SAAAC,EAAAC,GAGA,GAAAF,EAAAE,GACA,OAAAF,EAAAE,GAAAC,QAGA,IAAAC,EAAAJ,EAAAE,IACAG,EAAAH,EACAI,GAAA,EACAH,YAUA,OANAI,EAAAL,GAAAM,KAAAJ,EAAAD,QAAAC,IAAAD,QAAAF,GAGAG,EAAAE,GAAA,EAGAF,EAAAD,QAKAF,EAAAQ,EAAAF,EAGAN,EAAAS,EAAAV,EAGAC,EAAAU,EAAA,SAAAR,EAAAS,EAAAC,GACAZ,EAAAa,EAAAX,EAAAS,IACAG,OAAAC,eAAAb,EAAAS,GAA0CK,YAAA,EAAAC,IAAAL,KAK1CZ,EAAAkB,EAAA,SAAAhB,GACA,oBAAAiB,eAAAC,aACAN,OAAAC,eAAAb,EAAAiB,OAAAC,aAAwDC,MAAA,WAExDP,OAAAC,eAAAb,EAAA,cAAiDmB,OAAA,KAQjDrB,EAAAsB,EAAA,SAAAD,EAAAE,GAEA,GADA,EAAAA,IAAAF,EAAArB,EAAAqB,IACA,EAAAE,EAAA,OAAAF,EACA,KAAAE,GAAA,iBAAAF,QAAAG,WAAA,OAAAH,EACA,IAAAI,EAAAX,OAAAY,OAAA,MAGA,GAFA1B,EAAAkB,EAAAO,GACAX,OAAAC,eAAAU,EAAA,WAAyCT,YAAA,EAAAK,UACzC,EAAAE,GAAA,iBAAAF,EAAA,QAAAM,KAAAN,EAAArB,EAAAU,EAAAe,EAAAE,EAAA,SAAAA,GAAgH,OAAAN,EAAAM,IAAqBC,KAAA,KAAAD,IACrI,OAAAF,GAIAzB,EAAA6B,EAAA,SAAA1B,GACA,IAAAS,EAAAT,KAAAqB,WACA,WAA2B,OAAArB,EAAA,SAC3B,WAAiC,OAAAA,GAEjC,OADAH,EAAAU,EAAAE,EAAA,IAAAA,GACAA,GAIAZ,EAAAa,EAAA,SAAAiB,EAAAC,GAAsD,OAAAjB,OAAAkB,UAAAC,eAAA1B,KAAAuB,EAAAC,IAGtD/B,EAAAkC,EAAA,GAIAlC,IAAAmC,EAAA,kCClFA,SAAAC,EAAAC,EAAA1B,EAAA2B,EAAAC,GAEA,IAAAC,EAAAC,gBAAAL,EAAAK,QACAC,EAAAL,EAEA,SAAAM,EAAAC,GACA,IAAAC,KAIA,OAHAD,EAAAE,QAAA,SAAAC,GACAF,IAAAG,OAAAD,KAEAF,EAsFA,OAhDAL,EAAAd,OAAA,SAAAf,EAAA2B,EAAAC,GAEAU,MAAAC,QAAAX,KACAA,EAAAY,OAAA,SAAAzC,GAAmD,OAAAuC,MAAAC,QAAAxC,OAEnD6B,EAAAI,EAAAJ,KAuCA,OApCAC,EAAA7B,IACA2B,OACA3B,OACA4B,OACAa,SAAA,KACAC,KAAAC,SAAAhB,EAAAiB,MAAA,OAAAD,SAAAhB,EAAAiB,MAAA,UAGAf,EAAA7B,GAAA6C,KAAA,SAAAC,GAKA,YAJA,IAAAhB,KAAAF,MAAA,OAAAE,KAAAF,OACAE,KAAAW,SAAAV,EAAAgB,mBAAAD,EAAAhB,KAAA9B,MArDA,WACA,IAcAgD,EAdArB,EAAAG,KAAAH,KACAc,EAAAX,KAAAW,SACAC,EAAAZ,KAAAY,KACAd,EAAAE,KAAAF,KAEAU,MAAAC,QAAAX,IACAA,EAAAY,OAAA,SAAAzC,GAAmD,OAAAuC,MAAAC,QAAAxC,OAEnD6B,EAAAI,EAAAJ,IAGA,SAAAD,GAAA,OAAAA,GAAAW,MAAAC,QAAAX,IAAAqB,YAAAC,OAAAtB,KACAA,OAGA,OAAAD,EAAAiB,MAAA,eAAAjB,GACAqB,EAAA,IAAAG,aAAAvB,GACAG,EAAA,UAAAW,EAAA,MAAAD,EAAAO,IACS,QAAArB,EAAAiB,MAAA,aAAAjB,GACTqB,EAAA,IAAAI,WAAAxB,GACAG,EAAA,UAAAW,EAAA,MAAAD,EAAAO,IACS,OAAArB,EAAAiB,MAAA,MACTI,EAAA,IAAAG,aAAAvB,GACAG,EAAA,gBAAAW,EAAA,MAAAD,GAAA,EAAAO,IACS,aAAArB,GACTC,EAAAN,eAAA,4BAAAM,EAAAyB,eAEAtB,EAAAuB,cAAAvB,EAAAwB,SAAA3B,EAAA4B,OACAzB,EAAA0B,YAAA1B,EAAA2B,WAAA9B,EAAA+B,KACA5B,EAAA6B,UAAAnB,EAAAb,EAAA4B,SAwBA5D,KAAAkC,OAEAA,MAGAD,EAAA7B,GAAA6D,KAAA,SAAAjC,GAEA,OADAE,KAAAF,OACAE,MAGAD,EAAA7B,GAAA8D,OAAA,WACA,IAAAA,EAAA,WAAAhC,KAAAH,KAAA,IAAAG,KAAA9B,KACA+D,EAAA,EAUA,MARA,aAAAjC,KAAAH,OACAoC,EAAAjC,KAAAF,KAAAoC,OAAAlC,KAAAY,MAIAqB,EAAA,WAAApC,IACAmC,GAAA,IAAAC,EAAA,KAEAD,EAAA,OAGAjC,EAAA7B,IAIA6B,EC9FA,SAAAoC,EAAAvC,GAEA,IAAAwC,EAAApC,gBAAAmC,EAAAnC,QACAC,EAAAL,EACAyC,EAAA,EAEA,SAAAC,EAAApE,EAAAqE,GAEA,IAAA1C,EAAAI,EAAAmC,EAAAlE,GAAA2B,KAAA2C,eACAC,EAAAxC,EAAAmC,EAAAlE,GAAAwE,QAAAF,eACAG,EAAAP,EAAAlE,GAAA0E,IAAA,GACAC,EAAAT,EAAAlE,GAAA0E,IAAA,GAEAR,EAAAlE,GAAA4B,KAAAyC,EAEAtC,EAAA0B,YAAA1B,EAAA2B,WAAAQ,EAAAlE,GAAA2D,KACA5B,EAAA6C,WAAA7C,EAAA2B,WAAA,EAAAa,EAAAE,EAAAE,EAAA,EAAAJ,EAAA5C,EAAA0C,GACAtC,EAAA8C,cAAA9C,EAAA2B,WAAA3B,EAAA+C,mBAAA/C,EAAAgD,SACAhD,EAAA8C,cAAA9C,EAAA2B,WAAA3B,EAAAiD,mBAAAjD,EAAAgD,SACAhD,EAAA8C,cAAA9C,EAAA2B,WAAA3B,EAAAkD,eAAAlD,EAAAmD,eACAnD,EAAA8C,cAAA9C,EAAA2B,WAAA3B,EAAAoD,eAAApD,EAAAmD,eACAnD,EAAA0B,YAAA1B,EAAA2B,WAAA,MA+GA,OAnFAQ,EAAAnD,OAAA,SAAAf,EAAA2B,EAAA+C,EAAAF,EAAA5C,EAAAwD,GACA,IAAAC,EAAAnB,EAAA5C,eAAAtB,GAAAkE,EAAAlE,GAAAwD,MAAAW,IA+EA,OA9EAD,EAAAlE,IACAA,OACAwD,MAAA6B,EACA1D,QAAA,QACA+C,QAAA,SACAF,WAAA,QACA5C,KAAA,KACAa,SAAA,KACA2C,WAAA,KACAzB,IAAA5B,EAAAuD,iBAIAlB,EAAApE,EAAA4B,GAEA,OAAAsC,EAAAlE,GAAAoF,QACAlB,EAAAlE,GAAAoF,QAAA3D,EAAAM,GAAAhB,OAAAf,EAAA,YAAAkE,EAAAlE,IAEAkE,EAAAlE,GAAAoF,QAAAxD,KAAAsC,EAAAlE,GAGAkE,EAAAlE,GAAA6C,KAAA,SAAAC,GAaA,YAPA,IAAAhB,KAAAsD,QAAA,aAAAtD,KAAAsD,QAAAxD,OACAE,KAAAsD,QAAAxD,KAAAsC,EAAAlE,IAEA8B,KAAAsD,QAAAvC,KAAAC,GAIAhB,MAGAoC,EAAAlE,GAAA6D,KAAA,SAAAQ,GAEA,OADAD,EAAAtC,KAAA9B,KAAAqE,GACAvC,MAGAoC,EAAAlE,GAAAuF,YAAA,WACAxD,EAAA0B,YAAA1B,EAAA2B,WAAA5B,KAAA6B,KACA5B,EAAAyD,eACAzD,EAAA2B,WACA,EACA3B,EAAA0D,KACA,EACA,EACA3D,KAAA4C,IAAA,GACA5C,KAAA4C,IAAA,GACA,GAEA3C,EAAA0B,YAAA1B,EAAA2B,WAAA,OAGAQ,EAAAlE,GAAA0F,OAAA,SAAArB,EAAAsB,EAAAjB,GAEA,OAtFA,SAAA1E,EAAAqE,EAAAsB,EAAAjB,GACA,IAAA/C,EAAAI,EAAAmC,EAAAlE,GAAA2B,KAAA2C,eACAC,EAAAxC,EAAAmC,EAAAlE,GAAAwE,QAAAF,eACAG,EAAAC,EAAA,IAAAR,EAAAlE,GAAA0E,IAAA,GACAC,EAAAD,EAAA,IAAAR,EAAAlE,GAAA0E,IAAA,GAEA3C,EAAA0B,YAAA1B,EAAA2B,WAAAQ,EAAAlE,GAAA2D,KACA5B,EAAA6D,cAAA7D,EAAA2B,WAAA,EAAAiC,EAAA,GAAAA,EAAA,GAAAlB,EAAAE,EAAAJ,EAAA5C,EAAA0C,GACAtC,EAAA0B,YAAA1B,EAAA2B,WAAA,MA6EAmC,CAAA/D,KAAA9B,KAAAqE,EAAAsB,EAAAjB,GACA5C,MAGAoC,EAAAlE,GAAA8F,OAAA,SAAApB,EAAA9C,GACAE,KAAA4C,MACAN,EAAAtC,KAAA9B,KAAA4B,IAGAsC,EAAAlE,GAAA+F,OAAA,WACArE,EAAAsE,cAAAlE,KAAA6B,MAGAO,EAAAlE,GAAA8D,OAAA,WACA,OAAAhC,KAAA9B,MAAA8B,KAAAsD,QAAApF,KACA,qBAAA8B,KAAAsD,QAAApF,KAAA,MAEA,IAGAkE,EAAAlE,IAGAkE,EChIA,SAAA+B,EAAAvE,GAEA,IAAAwE,EAAApE,gBAAAmE,EAAAnE,QACAqE,KAEAD,EAAArE,QAAA,IAAAJ,EAAAC,GACAwE,EAAAE,UAAA,ICZA,SAAAC,EAAA3E,GAEA,IAAA0E,EAAAtE,gBAAAuE,EAAAvE,QACAC,EAAAL,EACA4E,EAAA,EAEA,SAAAC,EAAAvG,EAAA4B,IAEAU,MAAAC,QAAAX,IAAAqB,YAAAC,OAAAtB,MAEAqB,YAAAC,OAAAtB,KAEAA,EAAA,IAAAuB,aAAAvB,IAEAwE,EAAApG,GAAA4B,OACAG,EAAAyE,WAAAzE,EAAA0E,aAAAL,EAAApG,GAAA2D,KACA5B,EAAA2E,WAAA3E,EAAA0E,aAAA7E,EAAAG,EAAA4E,cA8CA,OAxCAP,EAAArF,OAAA,SAAAf,EAAA2B,EAAAC,GAqCA,OAnCAwE,EAAApG,IACAA,OACA2B,QAAA,QACAC,KAAA,KACAa,SAAA6D,IACA3C,IAAA5B,EAAA6E,eACAlE,KAAAC,SAAAhB,EAAAiB,MAAA,UAGA,OAAAhB,KAAAoC,QACauC,EAAAvG,EAAA4B,GAEbwE,EAAApG,GAAA6C,KAAA,SAAAC,GAMA,OAJAf,EAAAyE,WAAAzE,EAAA0E,aAAA3E,KAAA6B,KACA7B,KAAAW,SAAAV,EAAA8E,kBAAA/D,EAAAhB,KAAA9B,MACA+B,EAAA+E,oBAAAhF,KAAAW,SAAAX,KAAAY,KAAAX,EAAAgF,OAAA,OACAhF,EAAAiF,wBAAAlF,KAAAW,UACAX,MAGAsE,EAAApG,GAAA6D,KAAA,SAAAoD,GAGA,OADAV,EAAAzE,KAAA9B,KAAAiH,GACAnF,MAGAsE,EAAApG,GAAA8D,OAAA,WACa,mBAAAhC,KAAAH,KAAA,IAAAG,KAAA9B,KAAA,OAGboG,EAAApG,GAAA+F,OAAA,WACAhE,EAAAmF,aAAApF,KAAA6B,MAGAyC,EAAApG,IAGAoG,EDlDA,CAAA1E,GACAwE,EAAAhC,QAAA,IAAAD,EAAAvC,GACAwE,EAAAiB,QAAA,IEdA,SAAAC,EAAA1F,GAEA,IAAAyF,EAAArF,gBAAAsF,EAAAtF,QAsBA,OAnBAqF,EAAApG,OAAA,SAAAf,EAAA2B,EAAAe,GAgBA,OAfAyE,EAAAnH,IACAA,OACA2B,QAAA,QACAe,QAAA,GAGAyE,EAAAnH,GAAA6C,KAAA,aAEAsE,EAAAnH,GAAA8D,OAAA,WACA,IAAAA,EAAA,WAAAhC,KAAAH,KAAA,IAAAG,KAAA9B,KAGA,OAFA8B,KAAAY,KAAA,IACAoB,GAAA,IAAAhC,KAAAY,KAAA,KACAoB,EAAA,OAGAqD,EAAAnH,IAGAmH,EFVA,CAAAzF,GACAwE,EAAAmB,WAAA,IGfA,SAAAC,IAEA,IAAAD,EAAAvF,gBAAAwF,EAAAxF,QA0BA,OAxBAuF,EAAAtG,OAAA,SAAAf,EAAA2B,EAAA4F,GAqBA,OApBAF,EAAArH,IACAA,OACA2B,QAAA,QACA4F,KACAlE,aAAA,cAGAgE,EAAArH,GAAA6C,KAAA,SAAAC,GACA,OAAAhB,MAGAuF,EAAArH,GAAA6D,KAAA,SAAA0D,GAEA,OADAF,EAAArH,GAAAuH,KACAzF,MAGAuF,EAAArH,GAAA8D,OAAA,WACA,OAAAhC,KAAAyF,GAAAC,YAGAH,EAAArH,IAGAqH,GHXA,IAAAI,GAAA,wDAkCA,OAhCAvB,EAAAwB,SAAA,SAAA/F,GAEA,QAAA8F,EAAAE,QAAAhG,GAAkD,MAAAiG,MAAA,iCAAAjG,GAElD,IAAAkG,EAAA3B,EAAAvE,GAAAZ,OAAA+G,MAAA,KAAAxF,MAAAjB,UAAAuB,MAAAhD,KAAAmI,UAAA,IAWA,OAVAF,EAAAxE,aAAA1B,EACAwE,EAAA0B,EAAA7H,MAAA6H,EAEA1B,EAAA7E,eAAAuG,EAAA7H,OAEAG,OAAAC,eAAA+F,EAAA0B,EAAA7H,MACAM,IAAA,WAAiC,OAAA6F,EAAA0B,EAAA7H,OACjCgI,IAAA,SAAApG,GAAqCuE,EAAA0B,EAAA7H,MAAA6D,KAAAjC,MAGrCiG,GAGA3B,EAAArD,KAAA,SAAAC,EAAAmF,IACA3F,MAAAC,QAAA0F,KAAA9H,OAAA+H,KAAA/B,IACAhE,QAAA,SAAAgG,GACAhC,EAAA7E,eAAA6G,IACAhC,EAAAgC,GAAAtF,KAAAC,MAKAoD,EAAA5F,IAAA,SAAAN,GACA,OAAAmG,EAAAnG,IAEAkG,EAAAnF,OAAAmF,EAAAwB,SAEAxB,EIjDA,SAAAkC,EAAA1G,EAAA2G,GAEA,IAAAvF,KACAf,EAAAL,EACA4G,KACAC,EAAA,ICPA,SAAAC,EAAA9G,GAEA,IAAA6G,EAAAzG,gBAAA0G,EAAA1G,QACAC,EAAAL,EA+CA,OA3CA6G,EAAAxH,OAAA,SAAAf,EAAAyI,EAAAC,GAmBA,SAAAC,EAAAC,EAAAjH,EAAAkH,GACA,IAAAN,EAAAK,EAAAD,aAAAhH,GAIA,GAHAiH,EAAAE,aAAAP,EAAAM,GACAD,EAAAG,cAAAR,GACAK,EAAAI,mBAAAT,EAAAK,EAAAK,gBAEA,OAAAV,EAGAW,QAAAC,IAAAP,EAAAQ,iBAAAb,IACAK,EAAAS,aAAAd,GAGA,SAAAe,EAAAT,EAAAU,GACA,IAAAC,EAAA,uCACAX,EAAAY,MAAA,MAAAtH,QAAA,SAAAuH,GACA,IAAAC,EAAAH,EAAAI,KAAAF,GACAC,GACAJ,EAAAM,KAAAF,EAAA,MApCApB,EAAAvI,IACAA,OACAyI,uBACAC,yBACAoB,GAAAnB,EAAA5G,IAAAgI,cAAAtB,GACAuB,GAAArB,EAAA5G,IAAAkI,gBAAAvB,IAQAH,EAAAvI,GAAA8J,GAAAP,QACAD,EAAAb,EAAAF,EAAAvI,GAAA8J,GAAAP,MACAhB,EAAAvI,GAAAgK,GAAAT,QACAD,EAAAZ,EAAAH,EAAAvI,GAAA8J,GAAAP,OA0BAhB,ED3CA,CAAA7G,GAwDA,OAtDAoB,EAAA/B,OAAA,SAAAf,EAAA8J,EAAAE,GACAhK,KAAA,UACA8J,KAAA,UACAE,KAAA,UAFA,IAGAT,KAcA,GAZAjB,EAAAhH,eAAAtB,IACA8B,KAAAiE,OAAA/F,GAGAsI,EAAAtI,GAAA+B,EAAAmI,gBACA5B,EAAAtI,GAAA8J,KACAxB,EAAAtI,GAAAgK,KAEAjI,EAAAoI,aAAA7B,EAAAtI,GAAAsI,EAAAtI,GAAA8J,IACA/H,EAAAoI,aAAA7B,EAAAtI,GAAAsI,EAAAtI,GAAAgK,IACAjI,EAAAqI,YAAA9B,EAAAtI,KACA+B,EAAAsI,oBAAA/B,EAAAtI,GAAA+B,EAAAuI,aAGA,iCADAvI,EAAAwI,kBAAAjC,EAAAtI,IAMAuJ,GADAA,IAAAlH,OAAAiG,EAAAtI,GAAA8J,GAAAP,OACAlH,OAAAiG,EAAAtI,GAAAgK,GAAAT,MACAjB,EAAAtI,GAAAuJ,QAIAzG,EAAA0H,IAAA,SAAAxK,EAAAyI,EAAAC,GACA,OAAAJ,EAAAhH,eAAAtB,IACA+B,EAAA0I,WAAAnC,EAAAtI,IACAqI,EAAAxF,KAAAyF,EAAAtI,GAAAsI,EAAAtI,GAAAuJ,MACAjB,EAAAtI,KAGAuI,EAAAxH,OAAAf,EAAAyI,EAAAC,GACA5G,KAAAf,OAAAf,EAAAuI,EAAAvI,GAAA8J,GAAAvB,EAAAvI,GAAAgK,IACAjI,EAAA0I,WAAAnC,EAAAtI,IACAqI,EAAAxF,KAAAyF,EAAAtI,GAAAsI,EAAAtI,GAAAuJ,MACAjB,EAAAtI,KAIA8C,EAAAiD,OAAA,SAAA/F,GACAsI,EAAAhH,eAAAtB,KACA+B,EAAA2I,aAAApC,EAAAtI,GAAAsI,EAAAtI,GAAA8J,IACA/H,EAAA2I,aAAApC,EAAAtI,GAAAsI,EAAAtI,GAAAgK,IACAjI,EAAA4I,cAAArC,EAAAtI,WACAsI,EAAAtI,KAIA8C,EE1DA,SAAA8H,EAAAC,GAEA,IAAAC,EAAAhJ,gBAAA8I,EAAA9I,QAGAiJ,EAAAF,MACAG,EAAAD,EAAAC,WAAA,KACAC,EAAAF,EAAAE,QAAAC,SAAAC,cAAA,UACA1G,EAAAsG,EAAAtG,OAAA,KACAE,EAAAoG,EAAApG,QAAA,KACAyG,EAAAL,EAAAK,UACAC,KAAA,EACAC,MAAA,EACAC,IAAA,EACAC,OAAA,GAEAzJ,EAAAgJ,EAAAU,SAAAV,EAAAhJ,KAAA,KAEAgJ,EAAAW,eAIA,qBACAT,EAAA,KAAAA,EAAA,GAAAC,SAAAS,eAAAC,OAAAC,UAAA,IACAX,SAAAS,eAAAC,SAEAZ,IACAA,EAAA,mBAAAE,SAAAS,eAAAX,KACA,OAAAvG,MAAAuG,EAAAc,aACA,OAAAnH,MAAAqG,EAAAe,eAIAd,EAAAxG,QACAwG,EAAAtG,SACAsG,EAAAe,MAAAC,SAAA,WACAhB,EAAAe,MAAAE,WAAAd,EAAAC,KAAA,KACAJ,EAAAe,MAAAG,UAAAf,EAAAG,IAAA,KAGA,OAAAxJ,IACAA,EA4BA,SAAAkJ,GAGA,IAFA,IAAAmB,GAAA,8BACAxD,EAAA,KACAnJ,EAAA,EAAuBA,EAAA2M,EAAApI,SAAkBvE,EAAA,CACzC,IACAmJ,EAAAqC,EAAAoB,WAAAD,EAAA3M,IACa,MAAA6M,IACb,GAAA1D,EAAA,MAEA,OAAAA,EArCA2D,CAAAtB,IACAH,EAAA/I,MACA+I,EAAAG,SACAH,EAAAzC,YACAtG,EAAAyK,MAAAzB,EAAA0B,KAAA1B,EAAA2B,MAAA3B,EAAA4B,eAEA,IAAAtE,EAAA,IAAApC,EAAAlE,GACA6K,EAAA,ICnDA,SAAAC,EAAAnL,GAEA,IAAAoL,EAAAhL,gBAAA+K,EAAA/K,QACAC,EAAAL,EAoEA,OAlEAoL,EAAA/L,OAAA,SAAAf,EAAA2B,EAAA+C,EAAAR,GA+DA,OA7DA4I,EAAA9M,IACA2D,IAAA5B,EAAAgL,oBACA/M,OACA2B,QAAA,QACA8C,MAAAC,EAAA,SACAC,OAAAD,EAAA,SACAR,WAAA,MAYAnC,EAAAiL,gBAAAjL,EAAAkL,YAAAH,EAAA9M,GAAA2D,KAQA5B,EAAAmL,qBACAnL,EAAAkL,YACAlL,EAAAoL,kBACApL,EAAA2B,WACAoJ,EAAA9M,GAAAkE,QAAAP,IACA,GASA5B,EAAAiL,gBAAAjL,EAAAkL,YAAA,MAEAH,EAAA9M,GAAAoN,WAAA,SAAAtK,GACAf,EAAAuB,cAAAvB,EAAAwB,SAAAzB,KAAAoC,QAAAV,OACAzB,EAAA0B,YAAA1B,EAAA2B,WAAA5B,KAAAoC,QAAAP,KACA7B,KAAAoC,QAAAzB,SAAAV,EAAAgB,mBAAAD,EAAAhB,KAAAoC,QAAAlE,MACA+B,EAAA6B,UAAA9B,KAAAoC,QAAAzB,SAAAX,KAAAoC,QAAAV,QAOAsJ,EAAA9M,GAAA+F,OAAA,WACAhE,EAAAsL,iBAAAzE,GAAA0E,aAAA,MACAvL,EAAAiL,gBAAApE,GAAAqE,YAAA,MACAlL,EAAAwL,mBAAAzL,KAAA0L,cACAzL,EAAAiE,cAAAlE,KAAAoC,QAAAP,KACA5B,EAAA0L,kBAAA3L,KAAA6B,MAGAmJ,EAAA9M,IAGA8M,EDpBA,CAAA/K,GACAe,EAAA,IAAAsF,EAAArG,EAAAsG,GACAqF,EAAA,KAGAC,EAAA5L,EAAA6L,aAAA,oBA4BA,SAAAC,EAAAC,GACAxL,MAAAC,QAAAuL,YACAA,EAAA3L,QAAA,SAAA4L,GACA,IAAAC,EAAAjM,EAAA6L,aAAAG,GACA,OAAAC,GACA7N,OAAA+H,KAAA8F,GAAA7L,QAAA,SAAA8L,GACAC,IAAA5M,eAAA2M,KACAlM,EAAAmM,IAAAD,GAAAD,EAAAC,QAiLA,OAnNAN,IACA5L,EAAAoM,QAAAR,EAAAQ,QACApM,EAAAqM,QAAAT,EAAAS,SAEArM,EAAAmM,IAAAnM,EAAA6L,aAAA,0BACAC,GACA,oBACA,6BAIA7C,GACAA,EAAAqD,YAAApD,GA4BAH,EAAA+C,kBASA/C,EAAA1E,UAAA,SAAApG,EAAA2B,EAAAC,GAYA,OAXAyG,EAAAX,SAAA,YAAA1H,EAAA2B,EAAAC,GACAkJ,EAAA1E,UAAA9E,eAAAtB,IACAG,OAAAC,eAAA0K,EAAA1E,UAAApG,GACAM,IAAA,IACA+H,EAAAjC,UAAApG,GAEAsO,IAAA1M,GACAyG,EAAAjC,UAAApG,GAAA6D,KAAAjC,MAIAkJ,GAWAA,EAAAjJ,QAAA,SAAA7B,EAAA2B,EAAAC,GAcA,OAbAyG,EAAAX,SAAA,UAAA1H,EAAA2B,EAAAC,GACAkJ,EAAAjJ,QAAAP,eAAAtB,IACAG,OAAAC,eAAA0K,EAAAjJ,QAAA7B,GACAM,IAAA,WACA,OAAA+H,EAAAxG,QAAA7B,IAEAgI,IAAA,SAAApG,GACAyG,EAAAxG,QAAA7B,GAAA6D,KAAAjC,GACAG,EAAAwM,UAAAb,IACArF,EAAAxG,QAAA7B,GAAA6C,KAAA6K,MAIA5C,GAGAA,EAAAjJ,QAAA2M,UAAA,SAAAC,GACA,IAAAvM,KAIA,OAHAuM,EAAAtM,QAAA,SAAAC,GACAF,IAAAG,OAAAD,KAEAF,GAaA4I,EAAA5G,QAAA,SAAAlE,EAAA2B,EAAAC,EAAA8C,EAAAF,EAAAY,GAUA,OATAiD,EAAAX,SAAA,UAAA1H,EAAA2B,EAAA+C,EAAAF,EAAA5C,EAAAwD,GACAjF,OAAAC,eAAA0K,EAAA5G,QAAAlE,GACAM,IAAA,WACA,OAAA+H,EAAAnE,QAAAlE,IAEAgI,IAAA,SAAApG,GACAyG,EAAAnE,QAAAlE,GAAA6D,KAAAjC,MAGAkJ,GAGAA,EAAA5G,QAAAwB,OAAA,SAAA1F,EAAA4B,EAAA+D,EAAAjB,GACA2D,EAAAnE,QAAAlE,GAAA0F,OAAA9D,EAAA+D,EAAAjB,IAUAoG,EAAA3D,QAAA,SAAAnH,EAAA2B,EAAAe,GAEA,OADA2F,EAAAX,SAAA,UAAA1H,EAAA2B,EAAAe,GACAoI,GAWAA,EAAAgC,YAAA,SAAA9M,EAAA2B,EAAA+C,EAAAR,GACAA,KAAAmE,EAAAX,SAAA,UAAA1H,EAAA2B,EAAA+C,EAAA,aAUA,OATAkI,EAAA7L,OAAAf,EAAA2B,EAAA+C,EAAAR,GAEA4G,EAAAgC,YAAAxL,eAAAtB,IACAG,OAAAC,eAAA0K,EAAAgC,YAAA9M,GACAM,IAAA,WACA,OAAAsM,EAAA5M,MAIA8K,GAGAA,EAAAgC,YAAAM,WAAA,SAAApN,EAAA8C,GACA8J,EAAA5M,GAAAoN,WAAAtK,IAGAgI,EAAAkC,gBAAA,SAAA0B,GACA,OAAAA,EACA3M,EAAAiL,gBAAAjL,EAAAkL,YAAA,MAEAlL,EAAAiL,gBAAAjL,EAAAkL,YAAAL,EAAA8B,GAAA/K,MAGAmH,EAAAzD,WAAA,SAAArH,EAAA2B,EAAA4F,GAEA,OADAc,EAAAX,SAAA,aAAA1H,EAAA2B,EAAA4F,GACAuD,GAGAA,EAAA6D,UAAA,SAAAC,GAmBA,OAlBAzO,OAAA+H,KAAA0G,GAAAzM,QAAA,SAAAnB,GAEA,GADAe,EAAAyK,MAAAxL,GAAA4N,EAAA5N,GACAsB,MAAAC,QAAAR,EAAAyK,MAAAxL,IAAA,CACA,IAAAvB,EAAA,EACAU,OAAAC,eAAA2B,EAAAyK,MAAAxL,GACAV,IAAA,WACA,OAAAsO,EAAA5N,GAAAvB,MAEAuI,IAAA,SAAA6G,GACApP,EAAA,EACAsC,EAAAyK,MAAAxL,GAAA6N,UAGa,oBAAA9M,EAAAyK,MAAAxL,GAAA,CACb,IAAA8N,EAAA3O,OAAA+H,KAAAnG,EAAAyK,MAAAxL,IACA+N,KAAAlN,QAAA,OAAAb,EAAA,QAAA8N,EAAAE,IAAAjP,GAAAgC,EAAAyK,MAAAxL,GAAAjB,QAGA+K,GAGAA,EAAA6B,WAAA7B,EAAA6D,UACA7D,EAAAvC,OAAAzF,EAAAyF,OAGAuC,EAAAmE,IAAA,SAAAjP,EAAA+K,GACA,gBAAAmE,GAEA,OADAxB,EAAA5K,EAAA0H,IAAAxK,EAAA+K,EAAAtC,qBAAAsC,EAAArC,wBACAqC,EAAAoE,OAAAvP,KAAAkL,EAAAoE,KAIApE,EAAAsE,UAAA,WACA,OAAAnE,EAAAxG,MAAAwG,EAAAtG,SAGAmG,SE3QA,oBAAAuE,SACAA,OAAAzE","file":"flexgl.js","sourcesContent":[" \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, { enumerable: true, get: getter });\n \t\t}\n \t};\n\n \t// define __esModule on exports\n \t__webpack_require__.r = function(exports) {\n \t\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n \t\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n \t\t}\n \t\tObject.defineProperty(exports, '__esModule', { value: true });\n \t};\n\n \t// create a fake namespace object\n \t// mode & 1: value is a module id, require it\n \t// mode & 2: merge all properties of value into the ns\n \t// mode & 4: return value when already ns object\n \t// mode & 8|1: behave like require\n \t__webpack_require__.t = function(value, mode) {\n \t\tif(mode & 1) value = __webpack_require__(value);\n \t\tif(mode & 8) return value;\n \t\tif((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;\n \t\tvar ns = Object.create(null);\n \t\t__webpack_require__.r(ns);\n \t\tObject.defineProperty(ns, 'default', { enumerable: true, value: value });\n \t\tif(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));\n \t\treturn ns;\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = 0);\n","export default function Uniform(glContext, name, type, data) {\n\n    var uniform = (this instanceof Uniform) ? this : {},\n        ctx = glContext;\n\n    function serializeArray(arrayOfArray) {\n        var sa = [];\n        arrayOfArray.forEach(function(a){\n            sa = sa.concat(a);\n        })\n        return sa;\n    }\n\n    function setUniform() {\n        var type = this.type,\n            location = this.location,\n            size = this.size,\n            data = this.data;\n\n        if(Array.isArray(data)) {\n            var hasArray = data.filter(function(d){return Array.isArray(d);});\n            if(hasArray)\n                data = serializeArray(data);\n        }\n\n        if((type == 'float' || type == 'int') && !Array.isArray(data) && !ArrayBuffer.isView(data))\n            data = [data];\n\n        var buf;\n        if (type.slice(0,3) == 'vec' || type == 'float') {\n            buf = new Float32Array(data);\n            ctx['uniform' + size + 'fv'](location, buf);\n        } else if(type.slice(0,4) == 'ivec' || type == 'int'){\n            buf = new Int32Array(data);\n            ctx['uniform' + size + 'iv'](location, buf);\n        } else if(type.slice(0,3) == 'mat') {\n            buf = new Float32Array(data);\n            ctx['uniformMatrix' + size + 'fv'](location, false, buf);\n        } else if(type == 'sampler2D') {\n            if(data.hasOwnProperty('resourceType') && data.resourceType == 'texture') {\n                // console.log('bind ' + data.index);\n                ctx.activeTexture(ctx.TEXTURE0 + data.index);\n                ctx.bindTexture(ctx.TEXTURE_2D, data.ptr);\n                ctx.uniform1i(location, data.index);\n            }\n        }\n    }\n\n    uniform.create = function(name, type, data) {\n\n        if(Array.isArray(data)) {\n            var hasArray = data.filter(function(d){return Array.isArray(d);});\n            if(hasArray)\n                data = serializeArray(data);\n        }\n\n        uniform[name] = {\n            type: type,\n            name: name,\n            data: data,\n            location: null,\n            size: parseInt(type.slice(3,4)) || parseInt(type.slice(4,5)) || 1\n        };\n\n        uniform[name].link = function(program) {\n            if(typeof this.data !== 'undefined' && this.data !== null) {\n                this.location = ctx.getUniformLocation(program, this.name);\n                setUniform.call(this);\n            }\n            return this;\n        };\n\n        uniform[name].load = function(data) {\n            this.data = data;\n            return this;\n        };\n\n        uniform[name].header = function() {\n            var header = 'uniform ' + this.type + ' ' + this.name,\n                len = 0;\n\n            if(this.type != 'sampler2D') {\n                len = this.data.length / this.size;\n            }\n\n            //TODO: fix declaration for matrix\n            if(len > 1 && type != 'mat4') {\n                header += '[' + len + ']';\n            }\n            return header + ';\\n';\n        };\n\n        return uniform[name];\n    }\n\n\n    return uniform;\n}\n","import Uniform from \"./uniform\";\n\nexport default function Texture(glContext) {\n\n    var texture = (this instanceof Texture) ? this : {},\n        ctx = glContext,\n        textureID = 0;\n\n    function setTexture(name, texData) \n    {\n        var type = ctx[texture[name].type.toUpperCase()],\n            format = ctx[texture[name].channel.toUpperCase()],\n            width = texture[name].dim[0],\n            height = texture[name].dim[1];\n\n        texture[name].data = texData;\n\n        ctx.bindTexture(ctx.TEXTURE_2D, texture[name].ptr);\n        ctx.texImage2D(ctx.TEXTURE_2D, 0, format, width, height, 0, format, type, texData);\n        ctx.texParameteri(ctx.TEXTURE_2D, ctx.TEXTURE_MIN_FILTER, ctx.NEAREST);\n        ctx.texParameteri(ctx.TEXTURE_2D, ctx.TEXTURE_MAG_FILTER, ctx.NEAREST);\n        ctx.texParameteri(ctx.TEXTURE_2D, ctx.TEXTURE_WRAP_S, ctx.CLAMP_TO_EDGE);\n        ctx.texParameteri(ctx.TEXTURE_2D, ctx.TEXTURE_WRAP_T, ctx.CLAMP_TO_EDGE);\n        ctx.bindTexture(ctx.TEXTURE_2D, null);\n    }\n\n    function updateTexture(name, texData, offset, dim) {\n        var type = ctx[texture[name].type.toUpperCase()],\n            format = ctx[texture[name].channel.toUpperCase()],\n            width = dim[0] || texture[name].dim[0],\n            height = dim[1] || texture[name].dim[1];\n\n        ctx.bindTexture(ctx.TEXTURE_2D, texture[name].ptr);\n        ctx.texSubImage2D(ctx.TEXTURE_2D, 0, offset[0], offset[1], width, height, format, type, texData);\n        ctx.bindTexture(ctx.TEXTURE_2D, null);\n    }\n\n    // TODO: Add support for texture compression\n    // function compressTexture(texData) {\n    //\n    //     var ext = (\n    //       ctx.getExtension(\"WEBGL_compressed_texture_s3tc\") ||\n    //       ctx.getExtension(\"MOZ_WEBGL_compressed_texture_s3tc\") ||\n    //       ctx.getExtension(\"WEBKIT_WEBGL_compressed_texture_s3tc\")\n    //     );\n    //\n    //     ctx.compressedTexImage2D(ctx.TEXTURE_2D, 0, ext.COMPRESSED_RGBA_S3TC_DXT3_EXT, texture[name].dim[0], texture[name].dim[1], 0, texData);\n    //     ctx.texParameteri(ctx.TEXTURE_2D, ctx.TEXTURE_MAG_FILTER, ctx.LINEAR);\n    //     ctx.texParameteri(ctx.TEXTURE_2D, ctx.TEXTURE_MIN_FILTER, ctx.LINEAR);\n    // }\n\n    texture.create = function(name, type, dim, channel, data, sampler) {\n        var texIndex = (texture.hasOwnProperty(name)) ? texture[name].index : textureID++;\n        texture[name] = {\n            name: name,\n            index: texIndex,\n            type: type || \"float\",\n            dim: dim || [512, 512],\n            channel: channel || \"alpha\",\n            data: null,\n            location: null,\n            sampler: sampler || null,\n            ptr: ctx.createTexture()\n        };\n\n        // if(data !== null && data.length)\n        setTexture(name, data);\n\n        if (texture[name].sampler === null) {\n            texture[name].sampler = Uniform(ctx).create(name, 'sampler2D', texture[name]);\n        } else {\n            texture[name].sampler.data = texture[name];\n        }\n\n        texture[name].link = function(program) {\n            // if (this.data !== null) {\n                // ctx.activeTexture(ctx.TEXTURE0 + this.index);\n                // ctx.bindTexture(ctx.TEXTURE_2D, this.ptr);\n                // this.location = ctx.getUniformLocation(program, this.name);\n                // ctx.uniform1i(this.location, this.index);\n                if (typeof(this.sampler.data) == 'undefined' || this.sampler.data === null)\n                    this.sampler.data = texture[name];\n\n                this.sampler.link(program);\n            // }else{\n            //     console.log('texture data is null!!!')\n            // }\n            return this;\n        }\n\n        texture[name].load = function(texData) {\n            setTexture(this.name, texData);\n            return this;\n        }\n\n        texture[name].copyFromFBO = function() {\n            ctx.bindTexture(ctx.TEXTURE_2D, this.ptr);\n            ctx.copyTexImage2D(\n                ctx.TEXTURE_2D,\n                0,\n                ctx.RGBA,\n                0,\n                0,\n                this.dim[0],\n                this.dim[1],\n                0\n            );\n            ctx.bindTexture(ctx.TEXTURE_2D, null);\n        }\n\n        texture[name].update = function(texData, offset, dim) {\n            updateTexture(this.name, texData, offset, dim);\n            return this;\n        }\n        \n        texture[name].resize = function(dim, data) {\n            this.dim = dim;\n            setTexture(this.name, data);\n        }\n\n        texture[name].delete = function() {\n            glContext.deleteTexture(this.ptr);\n        }\n\n        texture[name].header = function() {\n            if (this.name == this.sampler.name)\n                return 'uniform sampler2D ' + this.sampler.name + ';\\n';\n            else\n                return '';\n        }\n\n        return texture[name];\n    }\n\n    return texture;\n}\n","import Uniform from './uniform';\nimport Attribute from './attribute';\nimport Texture from './texture';\nimport Varying from './varying';\nimport Subroutine from './subroutine';\n\nexport default function Resource(glContext) \n{\n    var resource = (this instanceof Resource) ? this : {},\n        gpuResources = {};\n\n    resource.uniform = new Uniform(glContext);\n    resource.attribute = new Attribute(glContext);\n    resource.texture = new Texture(glContext);\n    resource.varying = new Varying(glContext);\n    resource.subroutine = new Subroutine();\n\n    var resourceTypes = ['uniform', 'attribute', 'texture', 'varying', 'subroutine'];\n\n    resource.allocate = function(type) \n    {\n        if (resourceTypes.indexOf(type) === -1)  { throw Error(\"Error: Invalid resource type: \" + type); }\n        \n        var res = resource[type].create.apply(null, Array.prototype.slice.call(arguments, 1));\n        res.resourceType = type;\n        gpuResources[res.name] = res;\n\n        if (!gpuResources.hasOwnProperty(res.name)) \n        {\n            Object.defineProperty(gpuResources, res.name, {\n                get: function() { return gpuResources[res.name];},\n                set: function(data) { gpuResources[res.name].load(data);}\n            });\n        }\n        return res;\n    };\n\n    resource.link = function(program, resource_names) {\n        var requiredResourceNames = (Array.isArray(resource_names)) ? resource_names : Object.keys(gpuResources);\n        requiredResourceNames.forEach(function(resourceName) {\n            if (gpuResources.hasOwnProperty(resourceName)) {\n                gpuResources[resourceName].link(program);\n            }\n        })\n    };\n\n    resource.get = function(name) {\n        return gpuResources[name];\n    }\n    resource.create = resource.allocate;\n\n    return resource;\n};\n","export default function Attribute(glContext) \n{\n    var attribute = (this instanceof Attribute) ? this : {},\n        ctx = glContext,\n        attributeID = 0;\n\n    function setAttribute(name, data) \n    {\n        if( Array.isArray(data) || ArrayBuffer.isView(data) )\n        {\n            if(!ArrayBuffer.isView(data)) \n            {\n                data = new Float32Array(data);\n            }\n            attribute[name].data = data;\n            ctx.bindBuffer(ctx.ARRAY_BUFFER, attribute[name].ptr);\n            ctx.bufferData(ctx.ARRAY_BUFFER, data, ctx.STATIC_DRAW);\n            // console.log(attribute[name].ptr === attribute[name].old_ptr);\n            // attribute[name].old_ptr = attribute[name].ptr;\n        }\n    }\n\n    attribute.create = function(name, type, data) \n    {\n        attribute[name] = {\n            name: name,\n            type: type || 'float',\n            data: null,\n            location: attributeID++,\n            ptr: ctx.createBuffer(),\n            size: parseInt(type.slice(3,4)) || 1\n        };\n\n        if(data !== null && data.length) \n            {setAttribute(name, data);}\n\n        attribute[name].link = function(program) \n        {\n            ctx.bindBuffer(ctx.ARRAY_BUFFER, this.ptr);\n            this.location = ctx.getAttribLocation(program, this.name);\n            ctx.vertexAttribPointer(this.location, this.size, ctx.FLOAT, false, 0, 0);\n            ctx.enableVertexAttribArray(this.location);\n            return this;\n        }\n\n        attribute[name].load = function(arrayBuffer) \n        {\n            setAttribute(this.name, arrayBuffer);\n            return this;\n        }\n\n        attribute[name].header = function() \n            {return 'attribute ' + this.type + ' ' + this.name + ';\\n';}\n        \n\n        attribute[name].delete = function() {\n            ctx.deleteBuffer(this.ptr);\n        }\n\n        return attribute[name];\n    };\n\n    return attribute;\n}\n","export default function Varying(glContext) {\n\n    var varying = (this instanceof Varying) ? this : {},\n        ctx = glContext;\n\n    varying.create = function(name, type, size) {\n        varying[name] = {\n            name: name,\n            type: type || 'float',\n            size: size || 1,\n        };\n\n        varying[name].link = function() {};\n\n        varying[name].header = function() {\n            var header = 'varying ' + this.type + ' ' + this.name;\n            if(this.size > 1)\n                header += '[' + this.size + ']';\n            return header + ';\\n';\n        }\n\n        return varying[name];\n    }\n\n    return varying;\n}\n","export default function Subroutine() {\n\n    var subroutine = (this instanceof Subroutine) ? this : {};\n\n    subroutine.create = function(name, type, fn) {\n        subroutine[name] = {\n            name: name,\n            type: type || 'float',\n            fn: fn,\n            resourceType: \"subroutine\"\n        };\n\n        subroutine[name].link = function(program) {\n            return this;\n        }\n\n        subroutine[name].load = function(fn) {\n            subroutine[name].fn = fn;\n            return this;\n        }\n\n        subroutine[name].header = function() {\n            return this.fn.toString();\n        }\n\n        return subroutine[name];\n    };\n\n    return subroutine;\n}\n","import Shader from './shader';\n\nexport default function Program(glContext, resources) {\n\n    var program = {},\n        ctx = glContext,\n        kernels = {},\n        shader = new Shader(glContext);\n\n    program.create = function(name, vs, fs) {\n        var name = name || \"default\",\n            vs = vs || \"default\",\n            fs = fs || \"default\",\n            deps = [];\n\n        if (kernels.hasOwnProperty(name)) {\n            this.delete(name);\n        }\n\n        kernels[name] = ctx.createProgram();\n        kernels[name].vs = vs;\n        kernels[name].fs = fs;\n\n        ctx.attachShader(kernels[name], kernels[name].vs);\n        ctx.attachShader(kernels[name], kernels[name].fs);\n        ctx.linkProgram(kernels[name]);\n        var linked = ctx.getProgramParameter(kernels[name], ctx.LINK_STATUS);\n        if (!linked) {\n            var lastError = ctx.getProgramInfoLog(kernels[name]);\n            throw (\"Error in program linking:\" + lastError);\n            ctx.deleteProgram(kernels[name]);\n        }\n\n        deps = deps.concat(kernels[name].vs.deps);\n        deps = deps.concat(kernels[name].fs.deps);\n        kernels[name].deps = deps;\n\n    }\n\n    program.use = function(name, vertex_shader_source, fragment_shader_source) {\n        if (kernels.hasOwnProperty(name)) {\n            ctx.useProgram(kernels[name]);\n            resources.link(kernels[name], kernels[name].deps);\n            return kernels[name];\n        } \n        else {\n            shader.create(name, vertex_shader_source, fragment_shader_source);\n            this.create(name, shader[name].vs, shader[name].fs);\n            ctx.useProgram(kernels[name]);\n            resources.link(kernels[name], kernels[name].deps);\n            return kernels[name];\n        }\n    }\n\n    program.delete = function(name) {\n        if (kernels.hasOwnProperty(name)) {\n            ctx.detachShader(kernels[name], kernels[name].vs);\n            ctx.detachShader(kernels[name], kernels[name].fs);\n            ctx.deleteProgram(kernels[name]);\n            delete kernels[name];\n        }\n    }\n\n    return program;\n}\n","export default function Shader(glContext) {\n    \n    var shader = (this instanceof Shader) ? this : {},\n        ctx = glContext;\n        // resource = glResource,\n        // parameters = ctx._dict || {};\n\n    shader.create = function(name, vertex_shader_source, fragment_shader_source){\n        shader[name] = {\n            name: name,\n            vertex_shader_source: vertex_shader_source, \n            fragment_shader_source: fragment_shader_source, \n            vs: createShader(ctx, ctx.VERTEX_SHADER, vertex_shader_source),\n            fs: createShader(ctx, ctx.FRAGMENT_SHADER, fragment_shader_source)\n        }\n\n        // shader[name].vs = createShader(ctx, ctx.VERTEX_SHADER, vertex_shader_source);\n        // shader[name].fs = createShader(ctx, ctx.FRAGMENT_SHADER, fragment_shader_source);\n        // console.log(shader[name].vs);\n        // console.log(shader[name].fs);\n\n        shader[name].vs.deps = [];\n        addDeps(vertex_shader_source, shader[name].vs.deps);\n        shader[name].fs.deps = [];\n        addDeps(fragment_shader_source, shader[name].vs.deps);\n\n        function createShader(gl, type, source) {\n            var shader = gl.createShader(type);\n            gl.shaderSource(shader, source);\n            gl.compileShader(shader);\n            var success = gl.getShaderParameter(shader, gl.COMPILE_STATUS);\n            if (success) {\n                return shader;\n            }\n            // console.log('NO!');\n            console.log(gl.getShaderInfoLog(shader));\n            gl.deleteShader(shader);\n        }\n\n        function addDeps(source, deps){\n            var re = /\\s*(attribute|uniform)\\s+\\w+\\s+(\\w+)/;        \n            source.split('\\n').forEach(function(v){\n                var result = re.exec(v);\n                if(result){\n                    deps.push(result[2]);\n                }\n            });\n        }\n    }\n\n    return shader;\n}\n","import Resource from './resource';\nimport Program from './program';\nimport Framebuffer from './framebuffer';\n// import Reactive from './reactive';\n\nexport default function FlexGL(arg) {\n\n    var flexgl = (this instanceof FlexGL) ? this : {};\n\n\n    var options = arg || {},\n        container = options.container || null,\n        canvas = options.canvas || document.createElement(\"canvas\"),\n        width = options.width || null,\n        height = options.height || null,\n        padding = options.padding || {\n            left: 0,\n            right: 0,\n            top: 0,\n            bottom: 0\n        },\n        ctx = options.context || options.ctx || null,\n        kernels = {},\n        sharedFunction = options.sharedFunction || {};\n\n\n\n    if (typeof(canvas) == \"string\") {\n        if (canvas[0] == \"#\") canvas = document.getElementById(cavnas.substring(1));\n        else canvas = document.getElementById(cavnas);\n    }\n    if (container) {\n        container = (typeof(container) == \"string\") ? document.getElementById(container) : container;\n        if (width === null) width = container.clientWidth;\n        if (height === null) height = container.clientHeight;\n    }\n    // width -= padding.left + padding.right;\n    // height -= padding.top + padding.bottom;\n    canvas.width = width;\n    canvas.height = height;\n    canvas.style.position = \"absolute\";\n    canvas.style.marginLeft = padding.left + \"px\";\n    canvas.style.marginTop = padding.top + \"px\";\n\n\n    if (ctx === null)\n        ctx = setupWebGL(canvas);\n    flexgl.ctx = ctx;\n    flexgl.canvas = canvas;\n    flexgl.resources = resources;\n    ctx._dict = options.env || options.dict || options.dictionary || {};\n\n    var resources = new Resource(ctx),\n        framebuffers = new Framebuffer(ctx),\n        program = new Program(ctx, resources),\n        realProgram = null;\n\n\n    var blendExt = ctx.getExtension(\"EXT_blend_minmax\");\n    if (blendExt) {\n        ctx.MAX_EXT = blendExt.MAX_EXT;\n        ctx.MIN_EXT = blendExt.MIN_EXT;\n    }\n    ctx.ext = ctx.getExtension(\"ANGLE_instanced_arrays\");\n    enableExtension([\n        \"OES_texture_float\",\n        \"OES_texture_float_linear\",\n        // \"OES_texture_half_float\",\n        // \"OES_texture_half_float_linear\"\n    ]);\n    if (container)\n        container.appendChild(canvas);\n\n\n    function setupWebGL(canvas) {\n        var names = [\"webgl\", \"experimental-webgl\"];\n        var gl = null;\n        for (var i = 0; i < names.length; ++i) {\n            try {\n                gl = canvas.getContext(names[i]);\n            } catch (e) {}\n            if (gl) break;\n        }\n        return gl;\n    }\n\n    function enableExtension(extensions) {\n        if (!Array.isArray(extensions)) extensions = [extensions];\n        extensions.forEach(function(extension) {\n            var extProps = ctx.getExtension(extension);\n            if (extProps !== null) {\n                Object.keys(extProps).forEach(function(ep) {\n                    if (!ext.hasOwnProperty(ep)) {\n                        ctx.ext[ep] = extProps[ep];\n                    }\n                })\n            }\n        });\n    };\n    flexgl.enableExtension = enableExtension;\n\n    /**\n     * Allocate Attributes in vertex buffer array stored in GPU memory\n     * @param  {String} name attribute name\n     * @param  {String} type attribute type: float, vec2, ...\n     * @param  {Array} data data values\n     * @return {Object}      FLexGL object\n     */\n    flexgl.attribute = function(name, type, data) {\n        resources.allocate(\"attribute\", name, type, data);\n        if (!flexgl.attribute.hasOwnProperty(name)) {\n            Object.defineProperty(flexgl.attribute, name, { //after allocating, flexgl gets new key attribute, helping easily change attribute data.\n                get(){\n                    return resources.attribute[name];\n                },\n                set(data){\n                    resources.attribute[name].load(data);\n                }\n            });\n        }\n        return flexgl;\n    };\n    // flexgl.buffer = flexgl.attribute; //alias\n\n    /**\n     * Create a Uniform variable for WebGL shader programs\n     * @param  {String} name attribute name\n     * @param  {String} type uniform variable type: float, vec2, ...\n     * @param  {Array} data data values\n     * @return {Object}      FLexGL object\n     */\n    flexgl.uniform = function(name, type, data) {\n        resources.allocate(\"uniform\", name, type, data);\n        if (!flexgl.uniform.hasOwnProperty(name)) {\n            Object.defineProperty(flexgl.uniform, name, {\n                get: function() {\n                    return resources.uniform[name];\n                },\n                set: function(data) {\n                    resources.uniform[name].load(data);\n                    if (ctx.isProgram(realProgram))                    ///////////////////\n                        resources.uniform[name].link(realProgram);        /////////////////\n                }\n            });\n        }\n        return flexgl;\n    };\n\n    flexgl.uniform.serialize = function(aoa) {\n        var sa = [];\n        aoa.forEach(function(a) {\n            sa = sa.concat(a);\n        })\n        return sa;\n    }\n\n    /**\n     * Create a Uniform variable for WebGL shader programs\n     * @param  {String} name attribute name\n     * @param  {String} type texture type: unsigned_byte or float, ...\n     * @param  {Array} data data values\n     * @param  {Array} dim [width, height]\n     * @param  {String} [channel='alpha'] WebGL formats (rgba, alpha)\n     * @param  {Object} [sampler=null] FLexGL Uniform Object\n     * @return {Object}      FLexGL object\n     */\n    flexgl.texture = function(name, type, data, dim, channel, sampler) {\n        resources.allocate(\"texture\", name, type, dim, channel, data, sampler);\n        Object.defineProperty(flexgl.texture, name, {\n            get: function() {\n                return resources.texture[name];\n            },\n            set: function(data) {\n                resources.texture[name].load(data);\n            }\n        });\n        return flexgl;\n    }\n\n    flexgl.texture.update = function(name, data, offset, dim) {\n        resources.texture[name].update(data, offset, dim);\n    }\n\n    /**\n     * Create a Uniform variable for WebGL shader programs\n     * @param  {String} name attribute name\n     * @param  {String} [type] Varying variable type: float, vec2, ...\n     * @param  {Number} [size=1] data array\n     * @return {Object}      FLexGL object\n     */\n    flexgl.varying = function(name, type, size) {\n        resources.allocate(\"varying\", name, type, size);\n        return flexgl;\n    };\n\n    /**\n     * Create a Uniform variable for WebGL shader programs\n     * @param  {String} name attribute name\n     * @param  {String} type attribute type: float, vec2, ...\n     * @param  {Array} dim [width, height]\n     * @param  {Object} [texture=null] FLexGL Texture Object\n     * @return {Object}      FLexGL object\n     */\n    flexgl.framebuffer = function(name, type, dim, texture) {\n        var texture = texture || resources.allocate('texture', name, type, dim, 'rgba', null);\n        framebuffers.create(name, type, dim, texture);\n\n        if (!flexgl.framebuffer.hasOwnProperty(name)) {\n            Object.defineProperty(flexgl.framebuffer, name, {\n                get: function() {\n                    return framebuffers[name];\n                }\n            });\n        }\n        return flexgl;\n    }\n\n    flexgl.framebuffer.enableRead = function(name, program) {\n        framebuffers[name].enableRead(program);\n    }\n\n    flexgl.bindFramebuffer = function(fbName) {\n        if (fbName === null)\n            ctx.bindFramebuffer(ctx.FRAMEBUFFER, null);\n        else\n            ctx.bindFramebuffer(ctx.FRAMEBUFFER, framebuffers[fbName].ptr);\n    }\n\n    flexgl.subroutine = function(name, type, fn) {\n        resources.allocate(\"subroutine\", name, type, fn);\n        return flexgl;\n    }\n\n    flexgl.parameter = function(keyValuePairs) {\n        Object.keys(keyValuePairs).forEach(function(key) {\n            ctx._dict[key] = keyValuePairs[key];\n            if (Array.isArray(ctx._dict[key])) {\n                var i = 0;\n                Object.defineProperty(ctx._dict, key, {\n                    get: function() {\n                        return keyValuePairs[key][i++];\n                    },\n                    set: function(newArray) {\n                        i = 0;\n                        ctx._dict[key] = newArray;\n                    }\n                });\n            } else if(typeof(ctx._dict[key]) == 'object') {\n                var dictKeys = Object.keys(ctx._dict[key]);\n                fxgl.uniform('dict'+key, 'float', dictKeys.map(d=>ctx._dict[key][d]));\n            }\n        })\n        return flexgl;\n    }\n\n    flexgl.dictionary = flexgl.parameter;\n    flexgl.shader = program.shader;\n\n\n    flexgl.app = function(name, options) {\n        return function(args){\n            realProgram = program.use(name, options.vertex_shader_source, options.fragment_shader_source);\n            return options.render.call(flexgl, args);\n        }\n    }\n\n    flexgl.dimension = function() {\n        return [canvas.width, canvas.height];\n    }\n\n    return flexgl;\n}\n\n\n\n        // if(num === 0){\n        //     ctx.viewport(0, 0, 1024, 1);\n        //     realProgram = program.use(name, options.vertex_shader_source, options.fragment_shader_source);\n        //     this.bindFramebuffer('f_sum_texture');\n\n        //     // this.attribute['a_position'].link(realProgram);\n        //     // this.attribute['a_texcoord'].link(realProgram);\n        //     // this.texture['u_texture'].link(realProgram);\n        // }\n\n        // else if(num === 1){\n        //     ctx.viewport(0, 0, ctx.canvas.width, ctx.canvas.height);\n        //     realProgram = program.use(name, options.vertex_shader_source, options.fragment_shader_source);\n            \n        //     this.bindFramebuffer(null);\n        //     // this.attribute['a_position'].link(realProgram);\n        //     // this.attribute['a_texcoord'].link(realProgram);\n        //     // this.framebuffer.enableRead('f_sum_texture', realProgram);\n        // }\n\n        // else if(num === 2){\n        //     this.bindFramebuffer(null);\n\n        //     ctx.viewport(0, 0, 1024, 1);\n        //     this.bindFramebuffer('f_mem_texture_1');\n        //     realProgram = program.use(name, options.vertex_shader_source, options.fragment_shader_source);\n\n        //     // this.attribute['a_position'].link(realProgram);\n        //     // this.attribute['a_texcoord'].link(realProgram);\n        //     // this.framebuffer.enableRead('f_mem_texture_0', realProgram);\n        //     // this.framebuffer.enableRead('f_sum_texture', realProgram);\n        // }\n\n        // else if(num === 3){\n        //     this.bindFramebuffer(null);\n\n        //     ctx.viewport(0, 0, 1024, 1);\n        //     this.bindFramebuffer('f_mem_texture_0');\n        //     realProgram = program.use(name, options.vertex_shader_source, options.fragment_shader_source);\n\n        //     // this.attribute['a_position'].link(realProgram);\n        //     // this.attribute['a_texcoord'].link(realProgram);\n        //     // this.framebuffer.enableRead('f_mem_texture_1', realProgram);\n        //     // this.framebuffer.enableRead('f_sum_texture', realProgram);\n        // }\n\n        // else if(num === 4){\n        //     this.bindFramebuffer(null); \n        //     ctx.viewport(0, 0, ctx.canvas.width, ctx.canvas.height);\n        //     realProgram = program.use(name, options.vertex_shader_source, options.fragment_shader_source);\n\n        //     // this.attribute['a_position'].link(realProgram);\n        //     // this.attribute['a_texcoord'].link(realProgram);\n        //     // this.framebuffer.enableRead('f_mem_texture_1', realProgram);\n        // }\n\n        // else if(num === 5){\n        //     this.bindFramebuffer(null);\n        //     ctx.viewport(0, 0, ctx.canvas.width, ctx.canvas.height);\n        //     realProgram = program.use(name, options.vertex_shader_source, options.fragment_shader_source);\n\n        //     // this.attribute['a_position'].link(realProgram);\n        //     // this.attribute['a_texcoord'].link(realProgram);\n        //     // this.framebuffer.enableRead('f_mem_texture_0', realProgram);\n        // }\n\n        // this.uniform['u_texture'].link(realProgram);    \n        // var draw = options.render || options.draw;\n\n        // return function(args) {\n        //     // var gl = flexgl.program(name);\n        //     return draw.call(ctx, args);\n        // }\n\n        // ctx.drawArrays(ctx.LINES, 0, 2);","import Texture from './texture';\n\nexport default function Framebuffer(glContext) {\n\n    var framebuffer = (this instanceof Framebuffer) ? this : {},\n        ctx = glContext;\n\n    framebuffer.create = function(name, type, dim, texture) {\n\n        framebuffer[name] = {\n            ptr: ctx.createFramebuffer(),\n            name: name,\n            type: type || \"float\",\n            width: dim[0] || 1024,\n            height: dim[1] || 1024,\n            texture: texture || null,\n            // renderbuffer: ctx.createRenderbuffer(),\n        }\n\n        // if (framebuffer[name].texture === null) {\n        //     var buf = (type == 'float') ?\n        //         new Float32Array(dim[0] * dim[1] * 4) :\n        //         new Uint8Array(dim[0] * dim[1] * 4);\n        //     framebuffer[name].texture = Texture(ctx).create(name, type, dim, \"rgba\", buf);\n        // }\n\n        // var renderbuffer = framebuffer[name].renderbuffer;\n        ctx.bindFramebuffer(ctx.FRAMEBUFFER, framebuffer[name].ptr);\n        // ctx.bindRenderbuffer(ctx.RENDERBUFFER, renderbuffer);\n        // ctx.renderbufferStorage(\n        //     ctx.RENDERBUFFER,\n        //     ctx.DEPTH_COMPONENT16,\n        //     framebuffer[name].width,\n        //     framebuffer[name].height\n        // );\n        ctx.framebufferTexture2D(\n            ctx.FRAMEBUFFER,\n            ctx.COLOR_ATTACHMENT0,\n            ctx.TEXTURE_2D,\n            framebuffer[name].texture.ptr,\n            0\n        );\n        // ctx.framebufferRenderbuffer(\n        //     ctx.FRAMEBUFFER,\n        //     ctx.DEPTH_ATTACHMENT,\n        //     ctx.RENDERBUFFER,\n        //     renderbuffer\n        // );\n        // ctx.bindRenderbuffer(ctx.RENDERBUFFER, null);\n        ctx.bindFramebuffer(ctx.FRAMEBUFFER, null);\n\n        framebuffer[name].enableRead = function(program) {\n            ctx.activeTexture(ctx.TEXTURE0 + this.texture.index);\n            ctx.bindTexture(ctx.TEXTURE_2D, this.texture.ptr);\n            this.texture.location = ctx.getUniformLocation(program, this.texture.name);\n            ctx.uniform1i(this.texture.location, this.texture.index);\n        };\n\n        // framebuffer[name].enableRead = function(program) {\n        //     this.texture.link(program);\n        // };\n\n        framebuffer[name].delete = function() {\n            ctx.bindRenderbuffer(gl.RENDERBUFFER, null);\n            ctx.bindFramebuffer(gl.FRAMEBUFFER, null);\n            ctx.deleteRenderbuffer(this.renderbuffer);\n            ctx.deleteTexture(this.texture.ptr)\n            ctx.deleteFramebuffer(this.ptr);\n        };\n\n        return framebuffer[name];\n    }\n\n    return framebuffer;\n}\n","import FlexGL from './flexgl';\n\n(function makeGlobal() {\n  if (typeof window !== 'undefined') {\n    window.FlexGL = FlexGL;\n  }\n})();\n"],"sourceRoot":""} \ No newline at end of file diff --git a/dist/flexgl.min.js b/dist/flexgl.min.js new file mode 100644 index 0000000..281e95f --- /dev/null +++ b/dist/flexgl.min.js @@ -0,0 +1,2 @@ +!function(e){var t={};function r(n){if(t[n])return t[n].exports;var i=t[n]={i:n,l:!1,exports:{}};return e[n].call(i.exports,i,i.exports,r),i.l=!0,i.exports}r.m=e,r.c=t,r.d=function(e,t,n){r.o(e,t)||Object.defineProperty(e,t,{enumerable:!0,get:n})},r.r=function(e){"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},r.t=function(e,t){if(1&t&&(e=r(e)),8&t)return e;if(4&t&&"object"==typeof e&&e&&e.__esModule)return e;var n=Object.create(null);if(r.r(n),Object.defineProperty(n,"default",{enumerable:!0,value:e}),2&t&&"string"!=typeof e)for(var i in e)r.d(n,i,function(t){return e[t]}.bind(null,i));return n},r.n=function(e){var t=e&&e.__esModule?function(){return e.default}:function(){return e};return r.d(t,"a",t),t},r.o=function(e,t){return Object.prototype.hasOwnProperty.call(e,t)},r.p="",r(r.s=0)}([function(e,t,r){"use strict";function n(e,t,r,i){var a=this instanceof n?this:{},u=e;function o(e){var t=[];return e.forEach(function(e){t=t.concat(e)}),t}return a.create=function(e,t,r){return Array.isArray(r)&&r.filter(function(e){return Array.isArray(e)})&&(r=o(r)),a[e]={type:t,name:e,data:r,location:null,size:parseInt(t.slice(3,4))||parseInt(t.slice(4,5))||1},a[e].link=function(e){return void 0!==this.data&&null!==this.data&&(this.location=u.getUniformLocation(e,this.name),function(){var e,t=this.type,r=this.location,n=this.size,i=this.data;Array.isArray(i)&&i.filter(function(e){return Array.isArray(e)})&&(i=o(i)),"float"!=t&&"int"!=t||Array.isArray(i)||ArrayBuffer.isView(i)||(i=[i]),"vec"==t.slice(0,3)||"float"==t?(e=new Float32Array(i),u["uniform"+n+"fv"](r,e)):"ivec"==t.slice(0,4)||"int"==t?(e=new Int32Array(i),u["uniform"+n+"iv"](r,e)):"mat"==t.slice(0,3)?(e=new Float32Array(i),u["uniformMatrix"+n+"fv"](r,!1,e)):"sampler2D"==t&&i.hasOwnProperty("resourceType")&&"texture"==i.resourceType&&(u.activeTexture(u.TEXTURE0+i.index),u.bindTexture(u.TEXTURE_2D,i.ptr),u.uniform1i(r,i.index))}.call(this)),this},a[e].load=function(e){return this.data=e,this},a[e].header=function(){var e="uniform "+this.type+" "+this.name,r=0;return"sampler2D"!=this.type&&(r=this.data.length/this.size),r>1&&"mat4"!=t&&(e+="["+r+"]"),e+";\n"},a[e]},a}function i(e){var t=this instanceof i?this:{},r={};t.uniform=new n(e),t.attribute=new function e(t){var r=this instanceof e?this:{},n=t,i=0;function a(e,t){(Array.isArray(t)||ArrayBuffer.isView(t))&&(ArrayBuffer.isView(t)||(t=new Float32Array(t)),r[e].data=t,n.bindBuffer(n.ARRAY_BUFFER,r[e].ptr),n.bufferData(n.ARRAY_BUFFER,t,n.STATIC_DRAW))}return r.create=function(e,t,u){return r[e]={name:e,type:t||"float",data:null,location:i++,ptr:n.createBuffer(),size:parseInt(t.slice(3,4))||1},null!==u&&u.length&&a(e,u),r[e].link=function(e){return n.bindBuffer(n.ARRAY_BUFFER,this.ptr),this.location=n.getAttribLocation(e,this.name),n.vertexAttribPointer(this.location,this.size,n.FLOAT,!1,0,0),n.enableVertexAttribArray(this.location),this},r[e].load=function(e){return a(this.name,e),this},r[e].header=function(){return"attribute "+this.type+" "+this.name+";\n"},r[e].delete=function(){n.deleteBuffer(this.ptr)},r[e]},r}(e),t.texture=new function e(t){var r=this instanceof e?this:{},i=t,a=0;function u(e,t){var n=i[r[e].type.toUpperCase()],a=i[r[e].channel.toUpperCase()],u=r[e].dim[0],o=r[e].dim[1];r[e].data=t,i.bindTexture(i.TEXTURE_2D,r[e].ptr),i.texImage2D(i.TEXTURE_2D,0,a,u,o,0,a,n,t),i.texParameteri(i.TEXTURE_2D,i.TEXTURE_MIN_FILTER,i.NEAREST),i.texParameteri(i.TEXTURE_2D,i.TEXTURE_MAG_FILTER,i.NEAREST),i.texParameteri(i.TEXTURE_2D,i.TEXTURE_WRAP_S,i.CLAMP_TO_EDGE),i.texParameteri(i.TEXTURE_2D,i.TEXTURE_WRAP_T,i.CLAMP_TO_EDGE),i.bindTexture(i.TEXTURE_2D,null)}return r.create=function(e,o,s,f,c,l){var d=r.hasOwnProperty(e)?r[e].index:a++;return r[e]={name:e,index:d,type:o||"float",dim:s||[512,512],channel:f||"alpha",data:null,location:null,sampler:l||null,ptr:i.createTexture()},u(e,c),null===r[e].sampler?r[e].sampler=n(i).create(e,"sampler2D",r[e]):r[e].sampler.data=r[e],r[e].link=function(t){return void 0!==this.sampler.data&&null!==this.sampler.data||(this.sampler.data=r[e]),this.sampler.link(t),this},r[e].load=function(e){return u(this.name,e),this},r[e].copyFromFBO=function(){i.bindTexture(i.TEXTURE_2D,this.ptr),i.copyTexImage2D(i.TEXTURE_2D,0,i.RGBA,0,0,this.dim[0],this.dim[1],0),i.bindTexture(i.TEXTURE_2D,null)},r[e].update=function(e,t,n){return function(e,t,n,a){var u=i[r[e].type.toUpperCase()],o=i[r[e].channel.toUpperCase()],s=a[0]||r[e].dim[0],f=a[1]||r[e].dim[1];i.bindTexture(i.TEXTURE_2D,r[e].ptr),i.texSubImage2D(i.TEXTURE_2D,0,n[0],n[1],s,f,o,u,t),i.bindTexture(i.TEXTURE_2D,null)}(this.name,e,t,n),this},r[e].resize=function(e,t){this.dim=e,u(this.name,t)},r[e].delete=function(){t.deleteTexture(this.ptr)},r[e].header=function(){return this.name==this.sampler.name?"uniform sampler2D "+this.sampler.name+";\n":""},r[e]},r}(e),t.varying=new function e(t){var r=this instanceof e?this:{};return r.create=function(e,t,n){return r[e]={name:e,type:t||"float",size:n||1},r[e].link=function(){},r[e].header=function(){var e="varying "+this.type+" "+this.name;return this.size>1&&(e+="["+this.size+"]"),e+";\n"},r[e]},r}(e),t.subroutine=new function e(){var t=this instanceof e?this:{};return t.create=function(e,r,n){return t[e]={name:e,type:r||"float",fn:n,resourceType:"subroutine"},t[e].link=function(e){return this},t[e].load=function(r){return t[e].fn=r,this},t[e].header=function(){return this.fn.toString()},t[e]},t};var a=["uniform","attribute","texture","varying","subroutine"];return t.allocate=function(e){if(-1===a.indexOf(e))throw Error("Error: Invalid resource type: "+e);var n=t[e].create.apply(null,Array.prototype.slice.call(arguments,1));return n.resourceType=e,r[n.name]=n,r.hasOwnProperty(n.name)||Object.defineProperty(r,n.name,{get:function(){return r[n.name]},set:function(e){r[n.name].load(e)}}),n},t.link=function(e,t){(Array.isArray(t)?t:Object.keys(r)).forEach(function(t){r.hasOwnProperty(t)&&r[t].link(e)})},t.get=function(e){return r[e]},t.create=t.allocate,t}r.r(t),"undefined"!=typeof window&&(window.FlexGL=function e(t){var r=this instanceof e?this:{},n=t||{},a=n.container||null,u=n.canvas||document.createElement("canvas"),o=n.width||null,s=n.height||null,f=n.padding||{left:0,right:0,top:0,bottom:0},c=n.context||n.ctx||null;n.sharedFunction,"string"==typeof u&&(u="#"==u[0]?document.getElementById(cavnas.substring(1)):document.getElementById(cavnas)),a&&(a="string"==typeof a?document.getElementById(a):a,null===o&&(o=a.clientWidth),null===s&&(s=a.clientHeight)),u.width=o,u.height=s,u.style.position="absolute",u.style.marginLeft=f.left+"px",u.style.marginTop=f.top+"px",null===c&&(c=function(e){for(var t=["webgl","experimental-webgl"],r=null,n=0;nl.attribute[e],set(t){l.attribute[e].load(t)}}),r},r.uniform=function(e,t,n){return l.allocate("uniform",e,t,n),r.uniform.hasOwnProperty(e)||Object.defineProperty(r.uniform,e,{get:function(){return l.uniform[e]},set:function(t){l.uniform[e].load(t),c.isProgram(p)&&l.uniform[e].link(p)}}),r},r.uniform.serialize=function(e){var t=[];return e.forEach(function(e){t=t.concat(e)}),t},r.texture=function(e,t,n,i,a,u){return l.allocate("texture",e,t,i,a,n,u),Object.defineProperty(r.texture,e,{get:function(){return l.texture[e]},set:function(t){l.texture[e].load(t)}}),r},r.texture.update=function(e,t,r,n){l.texture[e].update(t,r,n)},r.varying=function(e,t,n){return l.allocate("varying",e,t,n),r},r.framebuffer=function(e,t,n,i){return i=i||l.allocate("texture",e,t,n,"rgba",null),d.create(e,t,n,i),r.framebuffer.hasOwnProperty(e)||Object.defineProperty(r.framebuffer,e,{get:function(){return d[e]}}),r},r.framebuffer.enableRead=function(e,t){d[e].enableRead(t)},r.bindFramebuffer=function(e){null===e?c.bindFramebuffer(c.FRAMEBUFFER,null):c.bindFramebuffer(c.FRAMEBUFFER,d[e].ptr)},r.subroutine=function(e,t,n){return l.allocate("subroutine",e,t,n),r},r.parameter=function(e){return Object.keys(e).forEach(function(t){if(c._dict[t]=e[t],Array.isArray(c._dict[t])){var r=0;Object.defineProperty(c._dict,t,{get:function(){return e[t][r++]},set:function(e){r=0,c._dict[t]=e}})}else if("object"==typeof c._dict[t]){var n=Object.keys(c._dict[t]);fxgl.uniform("dict"+t,"float",n.map(e=>c._dict[t][e]))}}),r},r.dictionary=r.parameter,r.shader=h.shader,r.app=function(e,t){return function(n){return p=h.use(e,t.vertex_shader_source,t.fragment_shader_source),t.render.call(r,n)}},r.dimension=function(){return[u.width,u.height]},r})}]); +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///webpack/bootstrap","webpack:///./src/uniform.js","webpack:///./src/resource.js","webpack:///./src/attribute.js","webpack:///./src/texture.js","webpack:///./src/varying.js","webpack:///./src/subroutine.js","webpack:///./src/flexgl.js","webpack:///./src/bundle.js","webpack:///./src/framebuffer.js","webpack:///./src/program.js","webpack:///./src/shader.js"],"names":["installedModules","__webpack_require__","moduleId","exports","module","i","l","modules","call","m","c","d","name","getter","o","Object","defineProperty","enumerable","get","r","Symbol","toStringTag","value","t","mode","__esModule","ns","create","key","bind","n","default","object","property","prototype","hasOwnProperty","p","s","e","Uniform","glContext","type","data","uniform","this","ctx","serializeArray","arrayOfArray","sa","forEach","a","concat","Array","isArray","filter","location","size","parseInt","slice","link","program","getUniformLocation","buf","ArrayBuffer","isView","Float32Array","Int32Array","resourceType","activeTexture","TEXTURE0","index","bindTexture","TEXTURE_2D","ptr","uniform1i","load","header","len","length","Resource","resource","gpuResources","attribute","Attribute","attributeID","setAttribute","bindBuffer","ARRAY_BUFFER","bufferData","STATIC_DRAW","createBuffer","getAttribLocation","vertexAttribPointer","FLOAT","enableVertexAttribArray","arrayBuffer","delete","deleteBuffer","texture","Texture","textureID","setTexture","texData","toUpperCase","format","channel","width","dim","height","texImage2D","texParameteri","TEXTURE_MIN_FILTER","NEAREST","TEXTURE_MAG_FILTER","TEXTURE_WRAP_S","CLAMP_TO_EDGE","TEXTURE_WRAP_T","sampler","texIndex","createTexture","copyFromFBO","copyTexImage2D","RGBA","update","offset","texSubImage2D","resize","deleteTexture","varying","Varying","subroutine","Subroutine","fn","toString","resourceTypes","allocate","indexOf","Error","res","apply","arguments","set","resource_names","keys","resourceName","flexgl","window","FlexGL","arg","options","container","canvas","document","createElement","padding","left","right","top","bottom","context","sharedFunction","getElementById","cavnas","substring","clientWidth","clientHeight","style","position","marginLeft","marginTop","names","gl","getContext","resources","_dict","env","dict","dictionary","framebuffers","Framebuffer","framebuffer","createFramebuffer","bindFramebuffer","FRAMEBUFFER","framebufferTexture2D","COLOR_ATTACHMENT0","enableRead","bindRenderbuffer","RENDERBUFFER","deleteRenderbuffer","renderbuffer","deleteFramebuffer","kernels","shader","Shader","vertex_shader_source","fragment_shader_source","createShader","source","shaderSource","compileShader","getShaderParameter","COMPILE_STATUS","console","log","getShaderInfoLog","deleteShader","addDeps","deps","re","split","v","result","exec","push","vs","VERTEX_SHADER","fs","FRAGMENT_SHADER","createProgram","attachShader","linkProgram","getProgramParameter","LINK_STATUS","getProgramInfoLog","use","useProgram","detachShader","deleteProgram","realProgram","blendExt","getExtension","enableExtension","extensions","extension","extProps","ep","ext","MAX_EXT","MIN_EXT","appendChild","[object Object]","isProgram","serialize","aoa","fbName","parameter","keyValuePairs","newArray","dictKeys","fxgl","map","app","args","render","dimension"],"mappings":"aACA,IAAAA,KAGA,SAAAC,EAAAC,GAGA,GAAAF,EAAAE,GACA,OAAAF,EAAAE,GAAAC,QAGA,IAAAC,EAAAJ,EAAAE,IACAG,EAAAH,EACAI,KACAH,YAUA,OANAI,EAAAL,GAAAM,KAAAJ,EAAAD,QAAAC,IAAAD,QAAAF,GAGAG,EAAAE,KAGAF,EAAAD,QAKAF,EAAAQ,EAAAF,EAGAN,EAAAS,EAAAV,EAGAC,EAAAU,EAAA,SAAAR,EAAAS,EAAAC,GACAZ,EAAAa,EAAAX,EAAAS,IACAG,OAAAC,eAAAb,EAAAS,GAA0CK,cAAAC,IAAAL,KAK1CZ,EAAAkB,EAAA,SAAAhB,GACA,oBAAAiB,eAAAC,aACAN,OAAAC,eAAAb,EAAAiB,OAAAC,aAAwDC,MAAA,WAExDP,OAAAC,eAAAb,EAAA,cAAiDmB,YAQjDrB,EAAAsB,EAAA,SAAAD,EAAAE,GAEA,GADA,EAAAA,IAAAF,EAAArB,EAAAqB,IACA,EAAAE,EAAA,OAAAF,EACA,KAAAE,GAAA,iBAAAF,QAAAG,WAAA,OAAAH,EACA,IAAAI,EAAAX,OAAAY,OAAA,MAGA,GAFA1B,EAAAkB,EAAAO,GACAX,OAAAC,eAAAU,EAAA,WAAyCT,cAAAK,UACzC,EAAAE,GAAA,iBAAAF,EAAA,QAAAM,KAAAN,EAAArB,EAAAU,EAAAe,EAAAE,EAAA,SAAAA,GAAgH,OAAAN,EAAAM,IAAqBC,KAAA,KAAAD,IACrI,OAAAF,GAIAzB,EAAA6B,EAAA,SAAA1B,GACA,IAAAS,EAAAT,KAAAqB,WACA,WAA2B,OAAArB,EAAA2B,SAC3B,WAAiC,OAAA3B,GAEjC,OADAH,EAAAU,EAAAE,EAAA,IAAAA,GACAA,GAIAZ,EAAAa,EAAA,SAAAkB,EAAAC,GAAsD,OAAAlB,OAAAmB,UAAAC,eAAA3B,KAAAwB,EAAAC,IAGtDhC,EAAAmC,EAAA,GAIAnC,IAAAoC,EAAA,cAAAC,EAAAf,EAAAJ,GAAA,aClFA,SAAAoB,EAAAC,EAAA5B,EAAA6B,EAAAC,GAEA,IAAAC,EAAAC,gBAAAL,EAAAK,QACAC,EAAAL,EAEA,SAAAM,EAAAC,GACA,IAAAC,KAIA,OAHAD,EAAAE,QAAA,SAAAC,GACAF,IAAAG,OAAAD,KAEAF,EAsFA,OAhDAL,EAAAhB,OAAA,SAAAf,EAAA6B,EAAAC,GA4CA,OA1CAU,MAAAC,QAAAX,IACAA,EAAAY,OAAA,SAAA3C,GAAmD,OAAAyC,MAAAC,QAAA1C,OAEnD+B,EAAAI,EAAAJ,IAGAC,EAAA/B,IACA6B,OACA7B,OACA8B,OACAa,SAAA,KACAC,KAAAC,SAAAhB,EAAAiB,MAAA,OAAAD,SAAAhB,EAAAiB,MAAA,UAGAf,EAAA/B,GAAA+C,KAAA,SAAAC,GAKA,gBAJAhB,KAAAF,MAAA,OAAAE,KAAAF,OACAE,KAAAW,SAAAV,EAAAgB,mBAAAD,EAAAhB,KAAAhC,MArDA,WACA,IAcAkD,EAdArB,EAAAG,KAAAH,KACAc,EAAAX,KAAAW,SACAC,EAAAZ,KAAAY,KACAd,EAAAE,KAAAF,KAEAU,MAAAC,QAAAX,IACAA,EAAAY,OAAA,SAAA3C,GAAmD,OAAAyC,MAAAC,QAAA1C,OAEnD+B,EAAAI,EAAAJ,IAGA,SAAAD,GAAA,OAAAA,GAAAW,MAAAC,QAAAX,IAAAqB,YAAAC,OAAAtB,KACAA,OAGA,OAAAD,EAAAiB,MAAA,eAAAjB,GACAqB,EAAA,IAAAG,aAAAvB,GACAG,EAAA,UAAAW,EAAA,MAAAD,EAAAO,IACS,QAAArB,EAAAiB,MAAA,aAAAjB,GACTqB,EAAA,IAAAI,WAAAxB,GACAG,EAAA,UAAAW,EAAA,MAAAD,EAAAO,IACS,OAAArB,EAAAiB,MAAA,MACTI,EAAA,IAAAG,aAAAvB,GACAG,EAAA,gBAAAW,EAAA,MAAAD,KAAAO,IACS,aAAArB,GACTC,EAAAP,eAAA,4BAAAO,EAAAyB,eAEAtB,EAAAuB,cAAAvB,EAAAwB,SAAA3B,EAAA4B,OACAzB,EAAA0B,YAAA1B,EAAA2B,WAAA9B,EAAA+B,KACA5B,EAAA6B,UAAAnB,EAAAb,EAAA4B,SAwBA9D,KAAAoC,OAEAA,MAGAD,EAAA/B,GAAA+D,KAAA,SAAAjC,GAEA,OADAE,KAAAF,OACAE,MAGAD,EAAA/B,GAAAgE,OAAA,WACA,IAAAA,EAAA,WAAAhC,KAAAH,KAAA,IAAAG,KAAAhC,KACAiE,EAAA,EAUA,MARA,aAAAjC,KAAAH,OACAoC,EAAAjC,KAAAF,KAAAoC,OAAAlC,KAAAY,MAIAqB,EAAA,WAAApC,IACAmC,GAAA,IAAAC,EAAA,KAEAD,EAAA,OAGAjC,EAAA/B,IAIA+B,EC1FA,SAAAoC,EAAAvC,GAEA,IAAAwC,EAAApC,gBAAAmC,EAAAnC,QACAqC,KAEAD,EAAArC,QAAA,IAAAJ,EAAAC,GACAwC,EAAAE,UAAA,ICZA,SAAAC,EAAA3C,GAEA,IAAA0C,EAAAtC,gBAAAuC,EAAAvC,QACAC,EAAAL,EACA4C,EAAA,EAEA,SAAAC,EAAAzE,EAAA8B,IAEAU,MAAAC,QAAAX,IAAAqB,YAAAC,OAAAtB,MAEAqB,YAAAC,OAAAtB,KAEAA,EAAA,IAAAuB,aAAAvB,IAEAwC,EAAAtE,GAAA8B,OACAG,EAAAyC,WAAAzC,EAAA0C,aAAAL,EAAAtE,GAAA6D,KACA5B,EAAA2C,WAAA3C,EAAA0C,aAAA7C,EAAAG,EAAA4C,cA8CA,OAxCAP,EAAAvD,OAAA,SAAAf,EAAA6B,EAAAC,GAqCA,OAnCAwC,EAAAtE,IACAA,OACA6B,QAAA,QACAC,KAAA,KACAa,SAAA6B,IACAX,IAAA5B,EAAA6C,eACAlC,KAAAC,SAAAhB,EAAAiB,MAAA,UAGA,OAAAhB,KAAAoC,QACaO,EAAAzE,EAAA8B,GAEbwC,EAAAtE,GAAA+C,KAAA,SAAAC,GAMA,OAJAf,EAAAyC,WAAAzC,EAAA0C,aAAA3C,KAAA6B,KACA7B,KAAAW,SAAAV,EAAA8C,kBAAA/B,EAAAhB,KAAAhC,MACAiC,EAAA+C,oBAAAhD,KAAAW,SAAAX,KAAAY,KAAAX,EAAAgD,SAAA,KACAhD,EAAAiD,wBAAAlD,KAAAW,UACAX,MAGAsC,EAAAtE,GAAA+D,KAAA,SAAAoB,GAGA,OADAV,EAAAzC,KAAAhC,KAAAmF,GACAnD,MAGAsC,EAAAtE,GAAAgE,OAAA,WACa,mBAAAhC,KAAAH,KAAA,IAAAG,KAAAhC,KAAA,OAGbsE,EAAAtE,GAAAoF,OAAA,WACAnD,EAAAoD,aAAArD,KAAA6B,MAGAS,EAAAtE,IAGAsE,EDlDA,CAAA1C,GACAwC,EAAAkB,QAAA,IEXA,SAAAC,EAAA3D,GAEA,IAAA0D,EAAAtD,gBAAAuD,EAAAvD,QACAC,EAAAL,EACA4D,EAAA,EAEA,SAAAC,EAAAzF,EAAA0F,GAEA,IAAA7D,EAAAI,EAAAqD,EAAAtF,GAAA6B,KAAA8D,eACAC,EAAA3D,EAAAqD,EAAAtF,GAAA6F,QAAAF,eACAG,EAAAR,EAAAtF,GAAA+F,IAAA,GACAC,EAAAV,EAAAtF,GAAA+F,IAAA,GAEAT,EAAAtF,GAAA8B,KAAA4D,EAEAzD,EAAA0B,YAAA1B,EAAA2B,WAAA0B,EAAAtF,GAAA6D,KACA5B,EAAAgE,WAAAhE,EAAA2B,WAAA,EAAAgC,EAAAE,EAAAE,EAAA,EAAAJ,EAAA/D,EAAA6D,GACAzD,EAAAiE,cAAAjE,EAAA2B,WAAA3B,EAAAkE,mBAAAlE,EAAAmE,SACAnE,EAAAiE,cAAAjE,EAAA2B,WAAA3B,EAAAoE,mBAAApE,EAAAmE,SACAnE,EAAAiE,cAAAjE,EAAA2B,WAAA3B,EAAAqE,eAAArE,EAAAsE,eACAtE,EAAAiE,cAAAjE,EAAA2B,WAAA3B,EAAAuE,eAAAvE,EAAAsE,eACAtE,EAAA0B,YAAA1B,EAAA2B,WAAA,MA+GA,OAnFA0B,EAAAvE,OAAA,SAAAf,EAAA6B,EAAAkE,EAAAF,EAAA/D,EAAA2E,GACA,IAAAC,EAAApB,EAAA/D,eAAAvB,GAAAsF,EAAAtF,GAAA0D,MAAA8B,IA+EA,OA9EAF,EAAAtF,IACAA,OACA0D,MAAAgD,EACA7E,QAAA,QACAkE,QAAA,SACAF,WAAA,QACA/D,KAAA,KACAa,SAAA,KACA8D,WAAA,KACA5C,IAAA5B,EAAA0E,iBAIAlB,EAAAzF,EAAA8B,GAEA,OAAAwD,EAAAtF,GAAAyG,QACAnB,EAAAtF,GAAAyG,QAAA9E,EAAAM,GAAAlB,OAAAf,EAAA,YAAAsF,EAAAtF,IAEAsF,EAAAtF,GAAAyG,QAAA3E,KAAAwD,EAAAtF,GAGAsF,EAAAtF,GAAA+C,KAAA,SAAAC,GAaA,gBAPAhB,KAAAyE,QAAA3E,MAAA,OAAAE,KAAAyE,QAAA3E,OACAE,KAAAyE,QAAA3E,KAAAwD,EAAAtF,IAEAgC,KAAAyE,QAAA1D,KAAAC,GAIAhB,MAGAsD,EAAAtF,GAAA+D,KAAA,SAAA2B,GAEA,OADAD,EAAAzD,KAAAhC,KAAA0F,GACA1D,MAGAsD,EAAAtF,GAAA4G,YAAA,WACA3E,EAAA0B,YAAA1B,EAAA2B,WAAA5B,KAAA6B,KACA5B,EAAA4E,eACA5E,EAAA2B,WACA,EACA3B,EAAA6E,KACA,EACA,EACA9E,KAAA+D,IAAA,GACA/D,KAAA+D,IAAA,GACA,GAEA9D,EAAA0B,YAAA1B,EAAA2B,WAAA,OAGA0B,EAAAtF,GAAA+G,OAAA,SAAArB,EAAAsB,EAAAjB,GAEA,OAtFA,SAAA/F,EAAA0F,EAAAsB,EAAAjB,GACA,IAAAlE,EAAAI,EAAAqD,EAAAtF,GAAA6B,KAAA8D,eACAC,EAAA3D,EAAAqD,EAAAtF,GAAA6F,QAAAF,eACAG,EAAAC,EAAA,IAAAT,EAAAtF,GAAA+F,IAAA,GACAC,EAAAD,EAAA,IAAAT,EAAAtF,GAAA+F,IAAA,GAEA9D,EAAA0B,YAAA1B,EAAA2B,WAAA0B,EAAAtF,GAAA6D,KACA5B,EAAAgF,cAAAhF,EAAA2B,WAAA,EAAAoD,EAAA,GAAAA,EAAA,GAAAlB,EAAAE,EAAAJ,EAAA/D,EAAA6D,GACAzD,EAAA0B,YAAA1B,EAAA2B,WAAA,MARA,CAqFA5B,KAAAhC,KAAA0F,EAAAsB,EAAAjB,GACA/D,MAGAsD,EAAAtF,GAAAkH,OAAA,SAAAnB,EAAAjE,GACAE,KAAA+D,MACAN,EAAAzD,KAAAhC,KAAA8B,IAGAwD,EAAAtF,GAAAoF,OAAA,WACAxD,EAAAuF,cAAAnF,KAAA6B,MAGAyB,EAAAtF,GAAAgE,OAAA,WACA,OAAAhC,KAAAhC,MAAAgC,KAAAyE,QAAAzG,KACA,qBAAAgC,KAAAyE,QAAAzG,KAAA,MAEA,IAGAsF,EAAAtF,IAGAsF,EFzHA,CAAA1D,GACAwC,EAAAgD,QAAA,IGdA,SAAAC,EAAAzF,GAEA,IAAAwF,EAAApF,gBAAAqF,EAAArF,QAsBA,OAnBAoF,EAAArG,OAAA,SAAAf,EAAA6B,EAAAe,GAgBA,OAfAwE,EAAApH,IACAA,OACA6B,QAAA,QACAe,QAAA,GAGAwE,EAAApH,GAAA+C,KAAA,aAEAqE,EAAApH,GAAAgE,OAAA,WACA,IAAAA,EAAA,WAAAhC,KAAAH,KAAA,IAAAG,KAAAhC,KAGA,OAFAgC,KAAAY,KAAA,IACAoB,GAAA,IAAAhC,KAAAY,KAAA,KACAoB,EAAA,OAGAoD,EAAApH,IAGAoH,EHVA,CAAAxF,GACAwC,EAAAkD,WAAA,IIfA,SAAAC,IAEA,IAAAD,EAAAtF,gBAAAuF,EAAAvF,QA0BA,OAxBAsF,EAAAvG,OAAA,SAAAf,EAAA6B,EAAA2F,GAqBA,OApBAF,EAAAtH,IACAA,OACA6B,QAAA,QACA2F,KACAjE,aAAA,cAGA+D,EAAAtH,GAAA+C,KAAA,SAAAC,GACA,OAAAhB,MAGAsF,EAAAtH,GAAA+D,KAAA,SAAAyD,GAEA,OADAF,EAAAtH,GAAAwH,KACAxF,MAGAsF,EAAAtH,GAAAgE,OAAA,WACA,OAAAhC,KAAAwF,GAAAC,YAGAH,EAAAtH,IAGAsH,GJXA,IAAAI,GAAA,wDAkCA,OAhCAtD,EAAAuD,SAAA,SAAA9F,GAEA,QAAA6F,EAAAE,QAAA/F,GAAkD,MAAAgG,MAAA,iCAAAhG,GAElD,IAAAiG,EAAA1D,EAAAvC,GAAAd,OAAAgH,MAAA,KAAAvF,MAAAlB,UAAAwB,MAAAlD,KAAAoI,UAAA,IAWA,OAVAF,EAAAvE,aAAA1B,EACAwC,EAAAyD,EAAA9H,MAAA8H,EAEAzD,EAAA9C,eAAAuG,EAAA9H,OAEAG,OAAAC,eAAAiE,EAAAyD,EAAA9H,MACAM,IAAA,WAAiC,OAAA+D,EAAAyD,EAAA9H,OACjCiI,IAAA,SAAAnG,GAAqCuC,EAAAyD,EAAA9H,MAAA+D,KAAAjC,MAGrCgG,GAGA1D,EAAArB,KAAA,SAAAC,EAAAkF,IACA1F,MAAAC,QAAAyF,KAAA/H,OAAAgI,KAAA9D,IACAhC,QAAA,SAAA+F,GACA/D,EAAA9C,eAAA6G,IACA/D,EAAA+D,GAAArF,KAAAC,MAKAoB,EAAA9D,IAAA,SAAAN,GACA,OAAAqE,EAAArE,IAEAoE,EAAArD,OAAAqD,EAAAuD,SAEAvD,EK2NAiE,OC3QA,oBAAAC,SACAA,OAAAC,ODCA,SAAAA,EAAAC,GAEA,IAAAH,EAAArG,gBAAAuG,EAAAvG,QAGAyG,EAAAD,MACAE,EAAAD,EAAAC,WAAA,KACAC,EAAAF,EAAAE,QAAAC,SAAAC,cAAA,UACA/C,EAAA2C,EAAA3C,OAAA,KACAE,EAAAyC,EAAAzC,QAAA,KACA8C,EAAAL,EAAAK,UACAC,KAAA,EACAC,MAAA,EACAC,IAAA,EACAC,OAAA,GAEAjH,EAAAwG,EAAAU,SAAAV,EAAAxG,KAAA,KAEAwG,EAAAW,eAIA,iBAAA3J,IACAkJ,EAAA,KAAAA,EAAA,GAAAC,SAAAS,eAAAC,OAAAC,UAAA,IACAX,SAAAS,eAAAC,SAEAZ,IACAA,EAAA,iBAAAxH,EAAA0H,SAAAS,eAAAX,KACA,OAAA5C,MAAA4C,EAAAc,aACA,OAAAxD,MAAA0C,EAAAe,eAIAd,EAAA7C,QACA6C,EAAA3C,SACA2C,EAAAe,MAAAC,SAAA,WACAhB,EAAAe,MAAAE,WAAAd,EAAAC,KAAA,KACAJ,EAAAe,MAAAG,UAAAf,EAAAG,IAAA,KAGA,OAAAhH,IACAA,EA4BA,SAAA0G,GAGA,IAFA,IAAAmB,GAAA,8BACAC,EAAA,KACAtK,EAAA,EAAuBA,EAAAqK,EAAA5F,SAAkBzE,EAAA,CACzC,IACAsK,EAAApB,EAAAqB,WAAAF,EAAArK,IACa,MAAAiC,IACb,GAAAqI,EAAA,MAEA,OAAAA,EATA,CA5BApB,IACAN,EAAApG,MACAoG,EAAAM,SACAN,EAAA4B,YACAhI,EAAAiI,MAAAzB,EAAA0B,KAAA1B,EAAA2B,MAAA3B,EAAA4B,eAEA,IAAAJ,EAAA,IAAA9F,EAAAlC,GACAqI,EAAA,IEnDA,SAAAC,EAAA3I,GAEA,IAAA4I,EAAAxI,gBAAAuI,EAAAvI,QACAC,EAAAL,EAoEA,OAlEA4I,EAAAzJ,OAAA,SAAAf,EAAA6B,EAAAkE,EAAAT,GA+DA,OA7DAkF,EAAAxK,IACA6D,IAAA5B,EAAAwI,oBACAzK,OACA6B,QAAA,QACAiE,MAAAC,EAAA,SACAC,OAAAD,EAAA,SACAT,WAAA,MAYArD,EAAAyI,gBAAAzI,EAAA0I,YAAAH,EAAAxK,GAAA6D,KAQA5B,EAAA2I,qBACA3I,EAAA0I,YACA1I,EAAA4I,kBACA5I,EAAA2B,WACA4G,EAAAxK,GAAAsF,QAAAzB,IACA,GASA5B,EAAAyI,gBAAAzI,EAAA0I,YAAA,MAEAH,EAAAxK,GAAA8K,WAAA,SAAA9H,GACAf,EAAAuB,cAAAvB,EAAAwB,SAAAzB,KAAAsD,QAAA5B,OACAzB,EAAA0B,YAAA1B,EAAA2B,WAAA5B,KAAAsD,QAAAzB,KACA7B,KAAAsD,QAAA3C,SAAAV,EAAAgB,mBAAAD,EAAAhB,KAAAsD,QAAAtF,MACAiC,EAAA6B,UAAA9B,KAAAsD,QAAA3C,SAAAX,KAAAsD,QAAA5B,QAOA8G,EAAAxK,GAAAoF,OAAA,WACAnD,EAAA8I,iBAAAhB,GAAAiB,aAAA,MACA/I,EAAAyI,gBAAAX,GAAAY,YAAA,MACA1I,EAAAgJ,mBAAAjJ,KAAAkJ,cACAjJ,EAAAkF,cAAAnF,KAAAsD,QAAAzB,KACA5B,EAAAkJ,kBAAAnJ,KAAA6B,MAGA2G,EAAAxK,IAGAwK,EFpBA,CAAAvI,GACAe,EAAA,IGpDA,SAAApB,EAAAqI,GAEA,IAAAjH,KACAf,EAAAL,EACAwJ,KACAC,EAAA,ICPA,SAAAC,EAAA1J,GAEA,IAAAyJ,EAAArJ,gBAAAsJ,EAAAtJ,QACAC,EAAAL,EA+CA,OA3CAyJ,EAAAtK,OAAA,SAAAf,EAAAuL,EAAAC,GAmBA,SAAAC,EAAA1B,EAAAlI,EAAA6J,GACA,IAAAL,EAAAtB,EAAA0B,aAAA5J,GAIA,GAHAkI,EAAA4B,aAAAN,EAAAK,GACA3B,EAAA6B,cAAAP,GACAtB,EAAA8B,mBAAAR,EAAAtB,EAAA+B,gBAEA,OAAAT,EAGAU,QAAAC,IAAAjC,EAAAkC,iBAAAZ,IACAtB,EAAAmC,aAAAb,GAGA,SAAAc,EAAAT,EAAAU,GACA,IAAAC,EAAA,uCACAX,EAAAY,MAAA,MAAAjK,QAAA,SAAAkK,GACA,IAAAC,EAAAH,EAAAI,KAAAF,GACAC,GACAJ,EAAAM,KAAAF,EAAA,MApCAnB,EAAArL,IACAA,OACAuL,uBACAC,yBACAmB,GAAAlB,EAAAxJ,IAAA2K,cAAArB,GACAsB,GAAApB,EAAAxJ,IAAA6K,gBAAAtB,IAQAH,EAAArL,GAAA2M,GAAAP,QACAD,EAAAZ,EAAAF,EAAArL,GAAA2M,GAAAP,MACAf,EAAArL,GAAA6M,GAAAT,QACAD,EAAAX,EAAAH,EAAArL,GAAA2M,GAAAP,OA0BAf,ED3CA,CAAAzJ,GAwDA,OAtDAoB,EAAAjC,OAAA,SAAAf,EAAA2M,EAAAE,GACA7M,KAAA,UACA2M,KAAA,UACAE,KAAA,UAFA,IAGAT,KAcA,GAZAhB,EAAA7J,eAAAvB,IACAgC,KAAAoD,OAAApF,GAGAoL,EAAApL,GAAAiC,EAAA8K,gBACA3B,EAAApL,GAAA2M,KACAvB,EAAApL,GAAA6M,KAEA5K,EAAA+K,aAAA5B,EAAApL,GAAAoL,EAAApL,GAAA2M,IACA1K,EAAA+K,aAAA5B,EAAApL,GAAAoL,EAAApL,GAAA6M,IACA5K,EAAAgL,YAAA7B,EAAApL,KACAiC,EAAAiL,oBAAA9B,EAAApL,GAAAiC,EAAAkL,aAGA,iCADAlL,EAAAmL,kBAAAhC,EAAApL,IAMAoM,GADAA,IAAA7J,OAAA6I,EAAApL,GAAA2M,GAAAP,OACA7J,OAAA6I,EAAApL,GAAA6M,GAAAT,MACAhB,EAAApL,GAAAoM,QAIApJ,EAAAqK,IAAA,SAAArN,EAAAuL,EAAAC,GACA,OAAAJ,EAAA7J,eAAAvB,IACAiC,EAAAqL,WAAAlC,EAAApL,IACAiK,EAAAlH,KAAAqI,EAAApL,GAAAoL,EAAApL,GAAAoM,MACAhB,EAAApL,KAGAqL,EAAAtK,OAAAf,EAAAuL,EAAAC,GACAxJ,KAAAjB,OAAAf,EAAAqL,EAAArL,GAAA2M,GAAAtB,EAAArL,GAAA6M,IACA5K,EAAAqL,WAAAlC,EAAApL,IACAiK,EAAAlH,KAAAqI,EAAApL,GAAAoL,EAAApL,GAAAoM,MACAhB,EAAApL,KAIAgD,EAAAoC,OAAA,SAAApF,GACAoL,EAAA7J,eAAAvB,KACAiC,EAAAsL,aAAAnC,EAAApL,GAAAoL,EAAApL,GAAA2M,IACA1K,EAAAsL,aAAAnC,EAAApL,GAAAoL,EAAApL,GAAA6M,IACA5K,EAAAuL,cAAApC,EAAApL,WACAoL,EAAApL,KAIAgD,EHTA,CAAAf,EAAAgI,GACAwD,EAAA,KAGAC,EAAAzL,EAAA0L,aAAA,oBA4BA,SAAAC,EAAAC,GACArL,MAAAC,QAAAoL,YACAA,EAAAxL,QAAA,SAAAyL,GACA,IAAAC,EAAA9L,EAAA0L,aAAAG,GACA,OAAAC,GACA5N,OAAAgI,KAAA4F,GAAA1L,QAAA,SAAA2L,GACAC,IAAA1M,eAAAyM,KACA/L,EAAAgM,IAAAD,GAAAD,EAAAC,QAiLA,OAnNAN,IACAzL,EAAAiM,QAAAR,EAAAQ,QACAjM,EAAAkM,QAAAT,EAAAS,SAEAlM,EAAAgM,IAAAhM,EAAA0L,aAAA,0BACAC,GACA,oBACA,6BAIAlF,GACAA,EAAA0F,YAAAzF,GA4BAN,EAAAuF,kBASAvF,EAAA/D,UAAA,SAAAtE,EAAA6B,EAAAC,GAYA,OAXAmI,EAAAtC,SAAA,YAAA3H,EAAA6B,EAAAC,GACAuG,EAAA/D,UAAA/C,eAAAvB,IACAG,OAAAC,eAAAiI,EAAA/D,UAAAtE,GACAM,IAAA,IACA2J,EAAA3F,UAAAtE,GAEAqO,IAAAvM,GACAmI,EAAA3F,UAAAtE,GAAA+D,KAAAjC,MAIAuG,GAWAA,EAAAtG,QAAA,SAAA/B,EAAA6B,EAAAC,GAcA,OAbAmI,EAAAtC,SAAA,UAAA3H,EAAA6B,EAAAC,GACAuG,EAAAtG,QAAAR,eAAAvB,IACAG,OAAAC,eAAAiI,EAAAtG,QAAA/B,GACAM,IAAA,WACA,OAAA2J,EAAAlI,QAAA/B,IAEAiI,IAAA,SAAAnG,GACAmI,EAAAlI,QAAA/B,GAAA+D,KAAAjC,GACAG,EAAAqM,UAAAb,IACAxD,EAAAlI,QAAA/B,GAAA+C,KAAA0K,MAIApF,GAGAA,EAAAtG,QAAAwM,UAAA,SAAAC,GACA,IAAApM,KAIA,OAHAoM,EAAAnM,QAAA,SAAAC,GACAF,IAAAG,OAAAD,KAEAF,GAaAiG,EAAA/C,QAAA,SAAAtF,EAAA6B,EAAAC,EAAAiE,EAAAF,EAAAY,GAUA,OATAwD,EAAAtC,SAAA,UAAA3H,EAAA6B,EAAAkE,EAAAF,EAAA/D,EAAA2E,GACAtG,OAAAC,eAAAiI,EAAA/C,QAAAtF,GACAM,IAAA,WACA,OAAA2J,EAAA3E,QAAAtF,IAEAiI,IAAA,SAAAnG,GACAmI,EAAA3E,QAAAtF,GAAA+D,KAAAjC,MAGAuG,GAGAA,EAAA/C,QAAAyB,OAAA,SAAA/G,EAAA8B,EAAAkF,EAAAjB,GACAkE,EAAA3E,QAAAtF,GAAA+G,OAAAjF,EAAAkF,EAAAjB,IAUAsC,EAAAjB,QAAA,SAAApH,EAAA6B,EAAAe,GAEA,OADAqH,EAAAtC,SAAA,UAAA3H,EAAA6B,EAAAe,GACAyF,GAWAA,EAAAmC,YAAA,SAAAxK,EAAA6B,EAAAkE,EAAAT,GAWA,OAVAA,KAAA2E,EAAAtC,SAAA,UAAA3H,EAAA6B,EAAAkE,EAAA,aACAuE,EAAAvJ,OAAAf,EAAA6B,EAAAkE,EAAAT,GAEA+C,EAAAmC,YAAAjJ,eAAAvB,IACAG,OAAAC,eAAAiI,EAAAmC,YAAAxK,GACAM,IAAA,WACA,OAAAgK,EAAAtK,MAIAqI,GAGAA,EAAAmC,YAAAM,WAAA,SAAA9K,EAAAgD,GACAsH,EAAAtK,GAAA8K,WAAA9H,IAGAqF,EAAAqC,gBAAA,SAAA+D,GACA,OAAAA,EACAxM,EAAAyI,gBAAAzI,EAAA0I,YAAA,MAEA1I,EAAAyI,gBAAAzI,EAAA0I,YAAAL,EAAAmE,GAAA5K,MAGAwE,EAAAf,WAAA,SAAAtH,EAAA6B,EAAA2F,GAEA,OADAyC,EAAAtC,SAAA,aAAA3H,EAAA6B,EAAA2F,GACAa,GAGAA,EAAAqG,UAAA,SAAAC,GAmBA,OAlBAxO,OAAAgI,KAAAwG,GAAAtM,QAAA,SAAArB,GAEA,GADAiB,EAAAiI,MAAAlJ,GAAA2N,EAAA3N,GACAwB,MAAAC,QAAAR,EAAAiI,MAAAlJ,IAAA,CACA,IAAAvB,EAAA,EACAU,OAAAC,eAAA6B,EAAAiI,MAAAlJ,GACAV,IAAA,WACA,OAAAqO,EAAA3N,GAAAvB,MAEAwI,IAAA,SAAA2G,GACAnP,EAAA,EACAwC,EAAAiI,MAAAlJ,GAAA4N,UAGa,oBAAA3M,EAAAiI,MAAAlJ,GAAA,CACb,IAAA6N,EAAA1O,OAAAgI,KAAAlG,EAAAiI,MAAAlJ,IACA8N,KAAA/M,QAAA,OAAAf,EAAA,QAAA6N,EAAAE,IAAAhP,GAAAkC,EAAAiI,MAAAlJ,GAAAjB,QAGAsI,GAGAA,EAAAgC,WAAAhC,EAAAqG,UACArG,EAAAgD,OAAArI,EAAAqI,OAGAhD,EAAA2G,IAAA,SAAAhP,EAAAyI,GACA,gBAAAwG,GAEA,OADAxB,EAAAzK,EAAAqK,IAAArN,EAAAyI,EAAA8C,qBAAA9C,EAAA+C,wBACA/C,EAAAyG,OAAAtP,KAAAyI,EAAA4G,KAIA5G,EAAA8G,UAAA,WACA,OAAAxG,EAAA7C,MAAA6C,EAAA3C,SAGAqC","file":"flexgl.min.js","sourcesContent":[" \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, { enumerable: true, get: getter });\n \t\t}\n \t};\n\n \t// define __esModule on exports\n \t__webpack_require__.r = function(exports) {\n \t\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n \t\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n \t\t}\n \t\tObject.defineProperty(exports, '__esModule', { value: true });\n \t};\n\n \t// create a fake namespace object\n \t// mode & 1: value is a module id, require it\n \t// mode & 2: merge all properties of value into the ns\n \t// mode & 4: return value when already ns object\n \t// mode & 8|1: behave like require\n \t__webpack_require__.t = function(value, mode) {\n \t\tif(mode & 1) value = __webpack_require__(value);\n \t\tif(mode & 8) return value;\n \t\tif((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;\n \t\tvar ns = Object.create(null);\n \t\t__webpack_require__.r(ns);\n \t\tObject.defineProperty(ns, 'default', { enumerable: true, value: value });\n \t\tif(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));\n \t\treturn ns;\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = 0);\n","export default function Uniform(glContext, name, type, data) {\n\n    var uniform = (this instanceof Uniform) ? this : {},\n        ctx = glContext;\n\n    function serializeArray(arrayOfArray) {\n        var sa = [];\n        arrayOfArray.forEach(function(a){\n            sa = sa.concat(a);\n        })\n        return sa;\n    }\n\n    function setUniform() {\n        var type = this.type,\n            location = this.location,\n            size = this.size,\n            data = this.data;\n\n        if(Array.isArray(data)) {\n            var hasArray = data.filter(function(d){return Array.isArray(d);});\n            if(hasArray)\n                data = serializeArray(data);\n        }\n\n        if((type == 'float' || type == 'int') && !Array.isArray(data) && !ArrayBuffer.isView(data))\n            data = [data];\n\n        var buf;\n        if (type.slice(0,3) == 'vec' || type == 'float') {\n            buf = new Float32Array(data);\n            ctx['uniform' + size + 'fv'](location, buf);\n        } else if(type.slice(0,4) == 'ivec' || type == 'int'){\n            buf = new Int32Array(data);\n            ctx['uniform' + size + 'iv'](location, buf);\n        } else if(type.slice(0,3) == 'mat') {\n            buf = new Float32Array(data);\n            ctx['uniformMatrix' + size + 'fv'](location, false, buf);\n        } else if(type == 'sampler2D') {\n            if(data.hasOwnProperty('resourceType') && data.resourceType == 'texture') {\n                // console.log('bind ' + data.index);\n                ctx.activeTexture(ctx.TEXTURE0 + data.index);\n                ctx.bindTexture(ctx.TEXTURE_2D, data.ptr);\n                ctx.uniform1i(location, data.index);\n            }\n        }\n    }\n\n    uniform.create = function(name, type, data) {\n\n        if(Array.isArray(data)) {\n            var hasArray = data.filter(function(d){return Array.isArray(d);});\n            if(hasArray)\n                data = serializeArray(data);\n        }\n\n        uniform[name] = {\n            type: type,\n            name: name,\n            data: data,\n            location: null,\n            size: parseInt(type.slice(3,4)) || parseInt(type.slice(4,5)) || 1\n        };\n\n        uniform[name].link = function(program) {\n            if(typeof this.data !== 'undefined' && this.data !== null) {\n                this.location = ctx.getUniformLocation(program, this.name);\n                setUniform.call(this);\n            }\n            return this;\n        };\n\n        uniform[name].load = function(data) {\n            this.data = data;\n            return this;\n        };\n\n        uniform[name].header = function() {\n            var header = 'uniform ' + this.type + ' ' + this.name,\n                len = 0;\n\n            if(this.type != 'sampler2D') {\n                len = this.data.length / this.size;\n            }\n\n            //TODO: fix declaration for matrix\n            if(len > 1 && type != 'mat4') {\n                header += '[' + len + ']';\n            }\n            return header + ';\\n';\n        };\n\n        return uniform[name];\n    }\n\n\n    return uniform;\n}\n","import Uniform from './uniform';\nimport Attribute from './attribute';\nimport Texture from './texture';\nimport Varying from './varying';\nimport Subroutine from './subroutine';\n\nexport default function Resource(glContext) \n{\n    var resource = (this instanceof Resource) ? this : {},\n        gpuResources = {};\n\n    resource.uniform = new Uniform(glContext);\n    resource.attribute = new Attribute(glContext);\n    resource.texture = new Texture(glContext);\n    resource.varying = new Varying(glContext);\n    resource.subroutine = new Subroutine();\n\n    var resourceTypes = ['uniform', 'attribute', 'texture', 'varying', 'subroutine'];\n\n    resource.allocate = function(type) \n    {\n        if (resourceTypes.indexOf(type) === -1)  { throw Error(\"Error: Invalid resource type: \" + type); }\n        \n        var res = resource[type].create.apply(null, Array.prototype.slice.call(arguments, 1));\n        res.resourceType = type;\n        gpuResources[res.name] = res;\n\n        if (!gpuResources.hasOwnProperty(res.name)) \n        {\n            Object.defineProperty(gpuResources, res.name, {\n                get: function() { return gpuResources[res.name];},\n                set: function(data) { gpuResources[res.name].load(data);}\n            });\n        }\n        return res;\n    };\n\n    resource.link = function(program, resource_names) {\n        var requiredResourceNames = (Array.isArray(resource_names)) ? resource_names : Object.keys(gpuResources);\n        requiredResourceNames.forEach(function(resourceName) {\n            if (gpuResources.hasOwnProperty(resourceName)) {\n                gpuResources[resourceName].link(program);\n            }\n        })\n    };\n\n    resource.get = function(name) {\n        return gpuResources[name];\n    }\n    resource.create = resource.allocate;\n\n    return resource;\n};\n","export default function Attribute(glContext) \n{\n    var attribute = (this instanceof Attribute) ? this : {},\n        ctx = glContext,\n        attributeID = 0;\n\n    function setAttribute(name, data) \n    {\n        if( Array.isArray(data) || ArrayBuffer.isView(data) )\n        {\n            if(!ArrayBuffer.isView(data)) \n            {\n                data = new Float32Array(data);\n            }\n            attribute[name].data = data;\n            ctx.bindBuffer(ctx.ARRAY_BUFFER, attribute[name].ptr);\n            ctx.bufferData(ctx.ARRAY_BUFFER, data, ctx.STATIC_DRAW);\n            // console.log(attribute[name].ptr === attribute[name].old_ptr);\n            // attribute[name].old_ptr = attribute[name].ptr;\n        }\n    }\n\n    attribute.create = function(name, type, data) \n    {\n        attribute[name] = {\n            name: name,\n            type: type || 'float',\n            data: null,\n            location: attributeID++,\n            ptr: ctx.createBuffer(),\n            size: parseInt(type.slice(3,4)) || 1\n        };\n\n        if(data !== null && data.length) \n            {setAttribute(name, data);}\n\n        attribute[name].link = function(program) \n        {\n            ctx.bindBuffer(ctx.ARRAY_BUFFER, this.ptr);\n            this.location = ctx.getAttribLocation(program, this.name);\n            ctx.vertexAttribPointer(this.location, this.size, ctx.FLOAT, false, 0, 0);\n            ctx.enableVertexAttribArray(this.location);\n            return this;\n        }\n\n        attribute[name].load = function(arrayBuffer) \n        {\n            setAttribute(this.name, arrayBuffer);\n            return this;\n        }\n\n        attribute[name].header = function() \n            {return 'attribute ' + this.type + ' ' + this.name + ';\\n';}\n        \n\n        attribute[name].delete = function() {\n            ctx.deleteBuffer(this.ptr);\n        }\n\n        return attribute[name];\n    };\n\n    return attribute;\n}\n","import Uniform from \"./uniform\";\n\nexport default function Texture(glContext) {\n\n    var texture = (this instanceof Texture) ? this : {},\n        ctx = glContext,\n        textureID = 0;\n\n    function setTexture(name, texData) \n    {\n        var type = ctx[texture[name].type.toUpperCase()],\n            format = ctx[texture[name].channel.toUpperCase()],\n            width = texture[name].dim[0],\n            height = texture[name].dim[1];\n\n        texture[name].data = texData;\n\n        ctx.bindTexture(ctx.TEXTURE_2D, texture[name].ptr);\n        ctx.texImage2D(ctx.TEXTURE_2D, 0, format, width, height, 0, format, type, texData);\n        ctx.texParameteri(ctx.TEXTURE_2D, ctx.TEXTURE_MIN_FILTER, ctx.NEAREST);\n        ctx.texParameteri(ctx.TEXTURE_2D, ctx.TEXTURE_MAG_FILTER, ctx.NEAREST);\n        ctx.texParameteri(ctx.TEXTURE_2D, ctx.TEXTURE_WRAP_S, ctx.CLAMP_TO_EDGE);\n        ctx.texParameteri(ctx.TEXTURE_2D, ctx.TEXTURE_WRAP_T, ctx.CLAMP_TO_EDGE);\n        ctx.bindTexture(ctx.TEXTURE_2D, null);\n    }\n\n    function updateTexture(name, texData, offset, dim) {\n        var type = ctx[texture[name].type.toUpperCase()],\n            format = ctx[texture[name].channel.toUpperCase()],\n            width = dim[0] || texture[name].dim[0],\n            height = dim[1] || texture[name].dim[1];\n\n        ctx.bindTexture(ctx.TEXTURE_2D, texture[name].ptr);\n        ctx.texSubImage2D(ctx.TEXTURE_2D, 0, offset[0], offset[1], width, height, format, type, texData);\n        ctx.bindTexture(ctx.TEXTURE_2D, null);\n    }\n\n    // TODO: Add support for texture compression\n    // function compressTexture(texData) {\n    //\n    //     var ext = (\n    //       ctx.getExtension(\"WEBGL_compressed_texture_s3tc\") ||\n    //       ctx.getExtension(\"MOZ_WEBGL_compressed_texture_s3tc\") ||\n    //       ctx.getExtension(\"WEBKIT_WEBGL_compressed_texture_s3tc\")\n    //     );\n    //\n    //     ctx.compressedTexImage2D(ctx.TEXTURE_2D, 0, ext.COMPRESSED_RGBA_S3TC_DXT3_EXT, texture[name].dim[0], texture[name].dim[1], 0, texData);\n    //     ctx.texParameteri(ctx.TEXTURE_2D, ctx.TEXTURE_MAG_FILTER, ctx.LINEAR);\n    //     ctx.texParameteri(ctx.TEXTURE_2D, ctx.TEXTURE_MIN_FILTER, ctx.LINEAR);\n    // }\n\n    texture.create = function(name, type, dim, channel, data, sampler) {\n        var texIndex = (texture.hasOwnProperty(name)) ? texture[name].index : textureID++;\n        texture[name] = {\n            name: name,\n            index: texIndex,\n            type: type || \"float\",\n            dim: dim || [512, 512],\n            channel: channel || \"alpha\",\n            data: null,\n            location: null,\n            sampler: sampler || null,\n            ptr: ctx.createTexture()\n        };\n\n        // if(data !== null && data.length)\n        setTexture(name, data);\n\n        if (texture[name].sampler === null) {\n            texture[name].sampler = Uniform(ctx).create(name, 'sampler2D', texture[name]);\n        } else {\n            texture[name].sampler.data = texture[name];\n        }\n\n        texture[name].link = function(program) {\n            // if (this.data !== null) {\n                // ctx.activeTexture(ctx.TEXTURE0 + this.index);\n                // ctx.bindTexture(ctx.TEXTURE_2D, this.ptr);\n                // this.location = ctx.getUniformLocation(program, this.name);\n                // ctx.uniform1i(this.location, this.index);\n                if (typeof(this.sampler.data) == 'undefined' || this.sampler.data === null)\n                    this.sampler.data = texture[name];\n\n                this.sampler.link(program);\n            // }else{\n            //     console.log('texture data is null!!!')\n            // }\n            return this;\n        }\n\n        texture[name].load = function(texData) {\n            setTexture(this.name, texData);\n            return this;\n        }\n\n        texture[name].copyFromFBO = function() {\n            ctx.bindTexture(ctx.TEXTURE_2D, this.ptr);\n            ctx.copyTexImage2D(\n                ctx.TEXTURE_2D,\n                0,\n                ctx.RGBA,\n                0,\n                0,\n                this.dim[0],\n                this.dim[1],\n                0\n            );\n            ctx.bindTexture(ctx.TEXTURE_2D, null);\n        }\n\n        texture[name].update = function(texData, offset, dim) {\n            updateTexture(this.name, texData, offset, dim);\n            return this;\n        }\n        \n        texture[name].resize = function(dim, data) {\n            this.dim = dim;\n            setTexture(this.name, data);\n        }\n\n        texture[name].delete = function() {\n            glContext.deleteTexture(this.ptr);\n        }\n\n        texture[name].header = function() {\n            if (this.name == this.sampler.name)\n                return 'uniform sampler2D ' + this.sampler.name + ';\\n';\n            else\n                return '';\n        }\n\n        return texture[name];\n    }\n\n    return texture;\n}\n","export default function Varying(glContext) {\n\n    var varying = (this instanceof Varying) ? this : {},\n        ctx = glContext;\n\n    varying.create = function(name, type, size) {\n        varying[name] = {\n            name: name,\n            type: type || 'float',\n            size: size || 1,\n        };\n\n        varying[name].link = function() {};\n\n        varying[name].header = function() {\n            var header = 'varying ' + this.type + ' ' + this.name;\n            if(this.size > 1)\n                header += '[' + this.size + ']';\n            return header + ';\\n';\n        }\n\n        return varying[name];\n    }\n\n    return varying;\n}\n","export default function Subroutine() {\n\n    var subroutine = (this instanceof Subroutine) ? this : {};\n\n    subroutine.create = function(name, type, fn) {\n        subroutine[name] = {\n            name: name,\n            type: type || 'float',\n            fn: fn,\n            resourceType: \"subroutine\"\n        };\n\n        subroutine[name].link = function(program) {\n            return this;\n        }\n\n        subroutine[name].load = function(fn) {\n            subroutine[name].fn = fn;\n            return this;\n        }\n\n        subroutine[name].header = function() {\n            return this.fn.toString();\n        }\n\n        return subroutine[name];\n    };\n\n    return subroutine;\n}\n","import Resource from './resource';\nimport Program from './program';\nimport Framebuffer from './framebuffer';\n// import Reactive from './reactive';\n\nexport default function FlexGL(arg) {\n\n    var flexgl = (this instanceof FlexGL) ? this : {};\n\n\n    var options = arg || {},\n        container = options.container || null,\n        canvas = options.canvas || document.createElement(\"canvas\"),\n        width = options.width || null,\n        height = options.height || null,\n        padding = options.padding || {\n            left: 0,\n            right: 0,\n            top: 0,\n            bottom: 0\n        },\n        ctx = options.context || options.ctx || null,\n        kernels = {},\n        sharedFunction = options.sharedFunction || {};\n\n\n\n    if (typeof(canvas) == \"string\") {\n        if (canvas[0] == \"#\") canvas = document.getElementById(cavnas.substring(1));\n        else canvas = document.getElementById(cavnas);\n    }\n    if (container) {\n        container = (typeof(container) == \"string\") ? document.getElementById(container) : container;\n        if (width === null) width = container.clientWidth;\n        if (height === null) height = container.clientHeight;\n    }\n    // width -= padding.left + padding.right;\n    // height -= padding.top + padding.bottom;\n    canvas.width = width;\n    canvas.height = height;\n    canvas.style.position = \"absolute\";\n    canvas.style.marginLeft = padding.left + \"px\";\n    canvas.style.marginTop = padding.top + \"px\";\n\n\n    if (ctx === null)\n        ctx = setupWebGL(canvas);\n    flexgl.ctx = ctx;\n    flexgl.canvas = canvas;\n    flexgl.resources = resources;\n    ctx._dict = options.env || options.dict || options.dictionary || {};\n\n    var resources = new Resource(ctx),\n        framebuffers = new Framebuffer(ctx),\n        program = new Program(ctx, resources),\n        realProgram = null;\n\n\n    var blendExt = ctx.getExtension(\"EXT_blend_minmax\");\n    if (blendExt) {\n        ctx.MAX_EXT = blendExt.MAX_EXT;\n        ctx.MIN_EXT = blendExt.MIN_EXT;\n    }\n    ctx.ext = ctx.getExtension(\"ANGLE_instanced_arrays\");\n    enableExtension([\n        \"OES_texture_float\",\n        \"OES_texture_float_linear\",\n        // \"OES_texture_half_float\",\n        // \"OES_texture_half_float_linear\"\n    ]);\n    if (container)\n        container.appendChild(canvas);\n\n\n    function setupWebGL(canvas) {\n        var names = [\"webgl\", \"experimental-webgl\"];\n        var gl = null;\n        for (var i = 0; i < names.length; ++i) {\n            try {\n                gl = canvas.getContext(names[i]);\n            } catch (e) {}\n            if (gl) break;\n        }\n        return gl;\n    }\n\n    function enableExtension(extensions) {\n        if (!Array.isArray(extensions)) extensions = [extensions];\n        extensions.forEach(function(extension) {\n            var extProps = ctx.getExtension(extension);\n            if (extProps !== null) {\n                Object.keys(extProps).forEach(function(ep) {\n                    if (!ext.hasOwnProperty(ep)) {\n                        ctx.ext[ep] = extProps[ep];\n                    }\n                })\n            }\n        });\n    };\n    flexgl.enableExtension = enableExtension;\n\n    /**\n     * Allocate Attributes in vertex buffer array stored in GPU memory\n     * @param  {String} name attribute name\n     * @param  {String} type attribute type: float, vec2, ...\n     * @param  {Array} data data values\n     * @return {Object}      FLexGL object\n     */\n    flexgl.attribute = function(name, type, data) {\n        resources.allocate(\"attribute\", name, type, data);\n        if (!flexgl.attribute.hasOwnProperty(name)) {\n            Object.defineProperty(flexgl.attribute, name, { //after allocating, flexgl gets new key attribute, helping easily change attribute data.\n                get(){\n                    return resources.attribute[name];\n                },\n                set(data){\n                    resources.attribute[name].load(data);\n                }\n            });\n        }\n        return flexgl;\n    };\n    // flexgl.buffer = flexgl.attribute; //alias\n\n    /**\n     * Create a Uniform variable for WebGL shader programs\n     * @param  {String} name attribute name\n     * @param  {String} type uniform variable type: float, vec2, ...\n     * @param  {Array} data data values\n     * @return {Object}      FLexGL object\n     */\n    flexgl.uniform = function(name, type, data) {\n        resources.allocate(\"uniform\", name, type, data);\n        if (!flexgl.uniform.hasOwnProperty(name)) {\n            Object.defineProperty(flexgl.uniform, name, {\n                get: function() {\n                    return resources.uniform[name];\n                },\n                set: function(data) {\n                    resources.uniform[name].load(data);\n                    if (ctx.isProgram(realProgram))                    ///////////////////\n                        resources.uniform[name].link(realProgram);        /////////////////\n                }\n            });\n        }\n        return flexgl;\n    };\n\n    flexgl.uniform.serialize = function(aoa) {\n        var sa = [];\n        aoa.forEach(function(a) {\n            sa = sa.concat(a);\n        })\n        return sa;\n    }\n\n    /**\n     * Create a Uniform variable for WebGL shader programs\n     * @param  {String} name attribute name\n     * @param  {String} type texture type: unsigned_byte or float, ...\n     * @param  {Array} data data values\n     * @param  {Array} dim [width, height]\n     * @param  {String} [channel='alpha'] WebGL formats (rgba, alpha)\n     * @param  {Object} [sampler=null] FLexGL Uniform Object\n     * @return {Object}      FLexGL object\n     */\n    flexgl.texture = function(name, type, data, dim, channel, sampler) {\n        resources.allocate(\"texture\", name, type, dim, channel, data, sampler);\n        Object.defineProperty(flexgl.texture, name, {\n            get: function() {\n                return resources.texture[name];\n            },\n            set: function(data) {\n                resources.texture[name].load(data);\n            }\n        });\n        return flexgl;\n    }\n\n    flexgl.texture.update = function(name, data, offset, dim) {\n        resources.texture[name].update(data, offset, dim);\n    }\n\n    /**\n     * Create a Uniform variable for WebGL shader programs\n     * @param  {String} name attribute name\n     * @param  {String} [type] Varying variable type: float, vec2, ...\n     * @param  {Number} [size=1] data array\n     * @return {Object}      FLexGL object\n     */\n    flexgl.varying = function(name, type, size) {\n        resources.allocate(\"varying\", name, type, size);\n        return flexgl;\n    };\n\n    /**\n     * Create a Uniform variable for WebGL shader programs\n     * @param  {String} name attribute name\n     * @param  {String} type attribute type: float, vec2, ...\n     * @param  {Array} dim [width, height]\n     * @param  {Object} [texture=null] FLexGL Texture Object\n     * @return {Object}      FLexGL object\n     */\n    flexgl.framebuffer = function(name, type, dim, texture) {\n        var texture = texture || resources.allocate('texture', name, type, dim, 'rgba', null);\n        framebuffers.create(name, type, dim, texture);\n\n        if (!flexgl.framebuffer.hasOwnProperty(name)) {\n            Object.defineProperty(flexgl.framebuffer, name, {\n                get: function() {\n                    return framebuffers[name];\n                }\n            });\n        }\n        return flexgl;\n    }\n\n    flexgl.framebuffer.enableRead = function(name, program) {\n        framebuffers[name].enableRead(program);\n    }\n\n    flexgl.bindFramebuffer = function(fbName) {\n        if (fbName === null)\n            ctx.bindFramebuffer(ctx.FRAMEBUFFER, null);\n        else\n            ctx.bindFramebuffer(ctx.FRAMEBUFFER, framebuffers[fbName].ptr);\n    }\n\n    flexgl.subroutine = function(name, type, fn) {\n        resources.allocate(\"subroutine\", name, type, fn);\n        return flexgl;\n    }\n\n    flexgl.parameter = function(keyValuePairs) {\n        Object.keys(keyValuePairs).forEach(function(key) {\n            ctx._dict[key] = keyValuePairs[key];\n            if (Array.isArray(ctx._dict[key])) {\n                var i = 0;\n                Object.defineProperty(ctx._dict, key, {\n                    get: function() {\n                        return keyValuePairs[key][i++];\n                    },\n                    set: function(newArray) {\n                        i = 0;\n                        ctx._dict[key] = newArray;\n                    }\n                });\n            } else if(typeof(ctx._dict[key]) == 'object') {\n                var dictKeys = Object.keys(ctx._dict[key]);\n                fxgl.uniform('dict'+key, 'float', dictKeys.map(d=>ctx._dict[key][d]));\n            }\n        })\n        return flexgl;\n    }\n\n    flexgl.dictionary = flexgl.parameter;\n    flexgl.shader = program.shader;\n\n\n    flexgl.app = function(name, options) {\n        return function(args){\n            realProgram = program.use(name, options.vertex_shader_source, options.fragment_shader_source);\n            return options.render.call(flexgl, args);\n        }\n    }\n\n    flexgl.dimension = function() {\n        return [canvas.width, canvas.height];\n    }\n\n    return flexgl;\n}\n\n\n\n        // if(num === 0){\n        //     ctx.viewport(0, 0, 1024, 1);\n        //     realProgram = program.use(name, options.vertex_shader_source, options.fragment_shader_source);\n        //     this.bindFramebuffer('f_sum_texture');\n\n        //     // this.attribute['a_position'].link(realProgram);\n        //     // this.attribute['a_texcoord'].link(realProgram);\n        //     // this.texture['u_texture'].link(realProgram);\n        // }\n\n        // else if(num === 1){\n        //     ctx.viewport(0, 0, ctx.canvas.width, ctx.canvas.height);\n        //     realProgram = program.use(name, options.vertex_shader_source, options.fragment_shader_source);\n            \n        //     this.bindFramebuffer(null);\n        //     // this.attribute['a_position'].link(realProgram);\n        //     // this.attribute['a_texcoord'].link(realProgram);\n        //     // this.framebuffer.enableRead('f_sum_texture', realProgram);\n        // }\n\n        // else if(num === 2){\n        //     this.bindFramebuffer(null);\n\n        //     ctx.viewport(0, 0, 1024, 1);\n        //     this.bindFramebuffer('f_mem_texture_1');\n        //     realProgram = program.use(name, options.vertex_shader_source, options.fragment_shader_source);\n\n        //     // this.attribute['a_position'].link(realProgram);\n        //     // this.attribute['a_texcoord'].link(realProgram);\n        //     // this.framebuffer.enableRead('f_mem_texture_0', realProgram);\n        //     // this.framebuffer.enableRead('f_sum_texture', realProgram);\n        // }\n\n        // else if(num === 3){\n        //     this.bindFramebuffer(null);\n\n        //     ctx.viewport(0, 0, 1024, 1);\n        //     this.bindFramebuffer('f_mem_texture_0');\n        //     realProgram = program.use(name, options.vertex_shader_source, options.fragment_shader_source);\n\n        //     // this.attribute['a_position'].link(realProgram);\n        //     // this.attribute['a_texcoord'].link(realProgram);\n        //     // this.framebuffer.enableRead('f_mem_texture_1', realProgram);\n        //     // this.framebuffer.enableRead('f_sum_texture', realProgram);\n        // }\n\n        // else if(num === 4){\n        //     this.bindFramebuffer(null); \n        //     ctx.viewport(0, 0, ctx.canvas.width, ctx.canvas.height);\n        //     realProgram = program.use(name, options.vertex_shader_source, options.fragment_shader_source);\n\n        //     // this.attribute['a_position'].link(realProgram);\n        //     // this.attribute['a_texcoord'].link(realProgram);\n        //     // this.framebuffer.enableRead('f_mem_texture_1', realProgram);\n        // }\n\n        // else if(num === 5){\n        //     this.bindFramebuffer(null);\n        //     ctx.viewport(0, 0, ctx.canvas.width, ctx.canvas.height);\n        //     realProgram = program.use(name, options.vertex_shader_source, options.fragment_shader_source);\n\n        //     // this.attribute['a_position'].link(realProgram);\n        //     // this.attribute['a_texcoord'].link(realProgram);\n        //     // this.framebuffer.enableRead('f_mem_texture_0', realProgram);\n        // }\n\n        // this.uniform['u_texture'].link(realProgram);    \n        // var draw = options.render || options.draw;\n\n        // return function(args) {\n        //     // var gl = flexgl.program(name);\n        //     return draw.call(ctx, args);\n        // }\n\n        // ctx.drawArrays(ctx.LINES, 0, 2);","import FlexGL from './flexgl';\n\n(function makeGlobal() {\n  if (typeof window !== 'undefined') {\n    window.FlexGL = FlexGL;\n  }\n})();\n","import Texture from './texture';\n\nexport default function Framebuffer(glContext) {\n\n    var framebuffer = (this instanceof Framebuffer) ? this : {},\n        ctx = glContext;\n\n    framebuffer.create = function(name, type, dim, texture) {\n\n        framebuffer[name] = {\n            ptr: ctx.createFramebuffer(),\n            name: name,\n            type: type || \"float\",\n            width: dim[0] || 1024,\n            height: dim[1] || 1024,\n            texture: texture || null,\n            // renderbuffer: ctx.createRenderbuffer(),\n        }\n\n        // if (framebuffer[name].texture === null) {\n        //     var buf = (type == 'float') ?\n        //         new Float32Array(dim[0] * dim[1] * 4) :\n        //         new Uint8Array(dim[0] * dim[1] * 4);\n        //     framebuffer[name].texture = Texture(ctx).create(name, type, dim, \"rgba\", buf);\n        // }\n\n        // var renderbuffer = framebuffer[name].renderbuffer;\n        ctx.bindFramebuffer(ctx.FRAMEBUFFER, framebuffer[name].ptr);\n        // ctx.bindRenderbuffer(ctx.RENDERBUFFER, renderbuffer);\n        // ctx.renderbufferStorage(\n        //     ctx.RENDERBUFFER,\n        //     ctx.DEPTH_COMPONENT16,\n        //     framebuffer[name].width,\n        //     framebuffer[name].height\n        // );\n        ctx.framebufferTexture2D(\n            ctx.FRAMEBUFFER,\n            ctx.COLOR_ATTACHMENT0,\n            ctx.TEXTURE_2D,\n            framebuffer[name].texture.ptr,\n            0\n        );\n        // ctx.framebufferRenderbuffer(\n        //     ctx.FRAMEBUFFER,\n        //     ctx.DEPTH_ATTACHMENT,\n        //     ctx.RENDERBUFFER,\n        //     renderbuffer\n        // );\n        // ctx.bindRenderbuffer(ctx.RENDERBUFFER, null);\n        ctx.bindFramebuffer(ctx.FRAMEBUFFER, null);\n\n        framebuffer[name].enableRead = function(program) {\n            ctx.activeTexture(ctx.TEXTURE0 + this.texture.index);\n            ctx.bindTexture(ctx.TEXTURE_2D, this.texture.ptr);\n            this.texture.location = ctx.getUniformLocation(program, this.texture.name);\n            ctx.uniform1i(this.texture.location, this.texture.index);\n        };\n\n        // framebuffer[name].enableRead = function(program) {\n        //     this.texture.link(program);\n        // };\n\n        framebuffer[name].delete = function() {\n            ctx.bindRenderbuffer(gl.RENDERBUFFER, null);\n            ctx.bindFramebuffer(gl.FRAMEBUFFER, null);\n            ctx.deleteRenderbuffer(this.renderbuffer);\n            ctx.deleteTexture(this.texture.ptr)\n            ctx.deleteFramebuffer(this.ptr);\n        };\n\n        return framebuffer[name];\n    }\n\n    return framebuffer;\n}\n","import Shader from './shader';\n\nexport default function Program(glContext, resources) {\n\n    var program = {},\n        ctx = glContext,\n        kernels = {},\n        shader = new Shader(glContext);\n\n    program.create = function(name, vs, fs) {\n        var name = name || \"default\",\n            vs = vs || \"default\",\n            fs = fs || \"default\",\n            deps = [];\n\n        if (kernels.hasOwnProperty(name)) {\n            this.delete(name);\n        }\n\n        kernels[name] = ctx.createProgram();\n        kernels[name].vs = vs;\n        kernels[name].fs = fs;\n\n        ctx.attachShader(kernels[name], kernels[name].vs);\n        ctx.attachShader(kernels[name], kernels[name].fs);\n        ctx.linkProgram(kernels[name]);\n        var linked = ctx.getProgramParameter(kernels[name], ctx.LINK_STATUS);\n        if (!linked) {\n            var lastError = ctx.getProgramInfoLog(kernels[name]);\n            throw (\"Error in program linking:\" + lastError);\n            ctx.deleteProgram(kernels[name]);\n        }\n\n        deps = deps.concat(kernels[name].vs.deps);\n        deps = deps.concat(kernels[name].fs.deps);\n        kernels[name].deps = deps;\n\n    }\n\n    program.use = function(name, vertex_shader_source, fragment_shader_source) {\n        if (kernels.hasOwnProperty(name)) {\n            ctx.useProgram(kernels[name]);\n            resources.link(kernels[name], kernels[name].deps);\n            return kernels[name];\n        } \n        else {\n            shader.create(name, vertex_shader_source, fragment_shader_source);\n            this.create(name, shader[name].vs, shader[name].fs);\n            ctx.useProgram(kernels[name]);\n            resources.link(kernels[name], kernels[name].deps);\n            return kernels[name];\n        }\n    }\n\n    program.delete = function(name) {\n        if (kernels.hasOwnProperty(name)) {\n            ctx.detachShader(kernels[name], kernels[name].vs);\n            ctx.detachShader(kernels[name], kernels[name].fs);\n            ctx.deleteProgram(kernels[name]);\n            delete kernels[name];\n        }\n    }\n\n    return program;\n}\n","export default function Shader(glContext) {\n    \n    var shader = (this instanceof Shader) ? this : {},\n        ctx = glContext;\n        // resource = glResource,\n        // parameters = ctx._dict || {};\n\n    shader.create = function(name, vertex_shader_source, fragment_shader_source){\n        shader[name] = {\n            name: name,\n            vertex_shader_source: vertex_shader_source, \n            fragment_shader_source: fragment_shader_source, \n            vs: createShader(ctx, ctx.VERTEX_SHADER, vertex_shader_source),\n            fs: createShader(ctx, ctx.FRAGMENT_SHADER, fragment_shader_source)\n        }\n\n        // shader[name].vs = createShader(ctx, ctx.VERTEX_SHADER, vertex_shader_source);\n        // shader[name].fs = createShader(ctx, ctx.FRAGMENT_SHADER, fragment_shader_source);\n        // console.log(shader[name].vs);\n        // console.log(shader[name].fs);\n\n        shader[name].vs.deps = [];\n        addDeps(vertex_shader_source, shader[name].vs.deps);\n        shader[name].fs.deps = [];\n        addDeps(fragment_shader_source, shader[name].vs.deps);\n\n        function createShader(gl, type, source) {\n            var shader = gl.createShader(type);\n            gl.shaderSource(shader, source);\n            gl.compileShader(shader);\n            var success = gl.getShaderParameter(shader, gl.COMPILE_STATUS);\n            if (success) {\n                return shader;\n            }\n            // console.log('NO!');\n            console.log(gl.getShaderInfoLog(shader));\n            gl.deleteShader(shader);\n        }\n\n        function addDeps(source, deps){\n            var re = /\\s*(attribute|uniform)\\s+\\w+\\s+(\\w+)/;        \n            source.split('\\n').forEach(function(v){\n                var result = re.exec(v);\n                if(result){\n                    deps.push(result[2]);\n                }\n            });\n        }\n    }\n\n    return shader;\n}\n"],"sourceRoot":""} \ No newline at end of file diff --git a/example/example.html b/example/example.html index 881c506..bf1d233 100644 --- a/example/example.html +++ b/example/example.html @@ -1,6 +1,16 @@ - + Refresh + + + + + + + + + + diff --git a/example/simpledraw.js b/example/simpledraw.js index 0b34573..74b6151 100644 --- a/example/simpledraw.js +++ b/example/simpledraw.js @@ -1,33 +1,307 @@ var fxgl = new FlexGL({ container: document.body, - width: 800, - height: 600 }); -//Allocate and manage resources -fxgl.attribute('aVertexPos', 'vec2', [-1, 0, 1, 1, 0, -1.0]) - .uniform('uColor', 'vec4', [0.5, 0.0, 0.5, 1.0]); +var zeroArray = new Float32Array(1024*4); +for(var i = 0; i < zeroArray.length; i++){ + zeroArray[i] = 0.0; +} +fxgl.attribute('a_position', 'vec2', [-1.0, -1.0, 1.0, 1.0]) + .attribute('a_texcoord', 'float', [0.0, 1.0]) + .framebuffer('f_sum_texture', 'float', [1024, 1]) + .texture('f_mem_texture_0', 'float', zeroArray, [1024, 1], 'rgba') + .texture('f_mem_texture_1', 'float', zeroArray, [1024, 1], 'rgba') + .framebuffer('f_mem_texture_0', 'float', [1024, 1], fxgl.texture['f_mem_texture_0']) + .framebuffer('f_mem_texture_1', 'float', [1024, 1], fxgl.texture['f_mem_texture_1']); -//Create Program -var simpleDraw = fxgl.app('drawTriangle', { - vs: function() { - gl_Position = vec4(this.aVertexPos, 0, 1.0); - }, - fs: function() { - gl_FragColor = this.uColor; - }, - render: function(len) { - this.drawArrays(this.TRIANGLES, 0, len); +var sumData = fxgl.app('sumData', { + vertex_shader_source: ` + attribute vec2 a_position; + attribute float a_texcoord; + varying float v_texcoord; + + void main(){ + gl_Position = vec4(a_position, 0, 1.0); + v_texcoord = a_texcoord; + } + `, + fragment_shader_source: ` + precision highp float; + uniform sampler2D u_texture; + varying float v_texcoord; + + void main(){ + float sum = 0.0; + for(float i = 0.0; i < 1024.0; i++){ + sum += texture2D(u_texture, vec2(v_texcoord, (i+0.5)/1024.0)).a; + } + sum /= 1024.0; + gl_FragColor = vec4(sum, 0.0, 0.0, 1.0); + + } + `, + render: function(args){ + this.ctx.viewport(0, 0, 1024, 1); + this.bindFramebuffer('f_sum_texture'); + this.ctx.drawArrays(this.ctx.LINES, 0, 2); + } +}); + +var swingDataA = fxgl.app('swingDataA', { + vertex_shader_source: ` + attribute vec2 a_position; + attribute float a_texcoord; + varying float v_texcoord; + + void main(){ + gl_Position = vec4(a_position, 0, 1.0); + v_texcoord = a_texcoord; + } + `, + fragment_shader_source: ` + precision highp float; + uniform sampler2D f_sum_texture; + uniform sampler2D f_mem_texture_0; + varying float v_texcoord; + + void main(){ + float v0 = texture2D(f_sum_texture, vec2(v_texcoord, 0.5)).x; + float v1 = texture2D(f_mem_texture_0, vec2(v_texcoord, 0.5)).x; + gl_FragColor = vec4(v0+v1, 0.0, 0.0, 1.0); + } + `, + render: function(args){ + this.ctx.viewport(0, 0, 1024, 1); + this.bindFramebuffer('f_mem_texture_1'); + this.ctx.drawArrays(this.ctx.LINES, 0, 2); + } +}); + +var drawDataA = fxgl.app('drawDataA', { + vertex_shader_source: ` + attribute vec2 a_position; + attribute float a_texcoord; + varying float v_texcoord; + + void main(){ + gl_Position = vec4(a_position, 0, 1.0); + v_texcoord = a_texcoord; + } + `, + fragment_shader_source: ` + precision highp float; + uniform sampler2D f_mem_texture_1; + varying float v_texcoord; + + void main(){ + gl_FragColor = texture2D(f_mem_texture_1, vec2(v_texcoord, 0.5)); + } + `, + render: function(args){ + this.ctx.viewport(0, 0, this.ctx.canvas.width, this.ctx.canvas.height); + this.bindFramebuffer(null); + this.ctx.drawArrays(this.ctx.LINES, 0, 2); + } +}); + +var swingDataB = fxgl.app('swingDataB', { + vertex_shader_source: ` + attribute vec2 a_position; + attribute float a_texcoord; + varying float v_texcoord; + + void main(){ + gl_Position = vec4(a_position, 0, 1.0); + v_texcoord = a_texcoord; + } + `, + fragment_shader_source: ` + precision highp float; + uniform sampler2D f_sum_texture; + uniform sampler2D f_mem_texture_1; + varying float v_texcoord; + + void main(){ + float v0 = texture2D(f_sum_texture, vec2(v_texcoord, 0.5)).x; + float v1 = texture2D(f_mem_texture_1, vec2(v_texcoord, 0.5)).x; + gl_FragColor = vec4(v0+v1, 0.0, 0.0, 1.0); + } + `, + render: function(args){ + this.ctx.viewport(0, 0, 1024, 1); + this.bindFramebuffer('f_mem_texture_0'); + this.ctx.drawArrays(this.ctx.LINES, 0, 2); + } +}); + +var drawDataB = fxgl.app('drawDataB', { + vertex_shader_source: ` + attribute vec2 a_position; + attribute float a_texcoord; + varying float v_texcoord; + + void main(){ + gl_Position = vec4(a_position, 0, 1.0); + v_texcoord = a_texcoord; + } + `, + fragment_shader_source: ` + precision highp float; + uniform sampler2D f_mem_texture_0; + varying float v_texcoord; + + void main(){ + gl_FragColor = texture2D(f_mem_texture_0, vec2(v_texcoord, 0.5)); + } + `, + render: function(args){ + this.ctx.viewport(0, 0, this.ctx.canvas.width, this.ctx.canvas.height); + this.bindFramebuffer(null); + this.ctx.drawArrays(this.ctx.LINES, 0, 2); + } +}); + +var total_data = []; +for(var j = 0; j < 100; j++){ + var typedArray = new Float32Array(1024*1024); + for (var i = 0; i < typedArray.length; i++){ + typedArray[i] = Math.random()/100; + } + total_data.push(typedArray); +} + + +var pointer = -1; + +var updateButton = document.querySelector('.update'); +var updateClickStream = Rx.Observable.fromEvent(updateButton, 'click'); + +var requestStream = updateClickStream.startWith('startup click') + .map(function(){ + pointer++; + if(pointer > 99){ + alert('Out of Stack!'); + pointer--; + } + return pointer; + }); + +var responseStream = requestStream + .map(function(p){ + return total_data[pointer]; + }); + + + +var i = 0; +responseStream.subscribe(function(data_chunk){ + if(pointer === 0){ + fxgl.texture('u_texture', 'float', data_chunk, [1024, 1024]); + } + else{ + fxgl.texture['u_texture'] = data_chunk; + } + + sumData(); + if(i%2 === 0){ + swingDataA(); + drawDataA(); } + else{ + swingDataB(); + drawDataB(); + } + i++; }); -simpleDraw(3); // draw triangle -//change vertices and color -fxgl.attribute.aVertexPos = [ - -0.5, -0.5, 0.5, -0.5, -0.5, 0.5, - -0.5, 0.5, 0.5, -0.5, 0.5, 0.5 -]; -fxgl.uniform.uColor = [0.0, 0.5, 0.5, 1.0] +// fxgl.attribute('aVertexPos', 'vec2', [-1, 0, 1, 1, 0, -1.0]) +// .uniform('uColor', 'vec4', [0.1, 1.0, 0.75, 1.0]); + +// //Create Program +// var simpleDraw = fxgl.app('drawTriangle', { +// vs: function() { +// gl_Position = vec4(this.aVertexPos, 0, 1.0); +// }, +// fs: function() { +// gl_FragColor = this.uColor; +// }, +// render: function(len) { +// this.drawArrays(this.TRIANGLES, 0, len); +// } +// }); + +// simpleDraw(3); // draw triangle + +// //change vertices and color +// fxgl.attribute.aVertexPos = [ +// -0.5, -0.5, 0.5, -0.5, -0.5, 0.5, +// -0.5, 0.5, 0.5, -0.5, 0.5, 0.5 +// ]; +// fxgl.uniform.uColor = [1.0, 1.0, 0.5, 1.0] + +// simpleDraw(6); // draw rectangle + + + +//Create Program + + //gl_FragColor = vec4(0.0, texture2D(u_texture, vec2(v_texcoord, (88.0+0.5)/1024.0)).a, 0.0, 1.0); + // gl_FragColor = vec4(sum, 0.0, 0.0, 1.0); + + + + +// fxgl.app('drawLine1', { +// vsource: ` +// attribute vec2 a_position; + +// void main(){ +// gl_Position = vec4(a_position, 0, 1.0); +// } +// `, +// fsource: ` +// precision highp float; + +// void main(){ +// gl_FragColor = vec4(0.5, 0.5, 0.0, 1.0); +// } +// `, +// }); + +// simpleDraw(); // draw triangle +// gl_FragColor = vec4(texture2D(u_texture, vec2(217, 217))[3], texture2D(u_texture, vec2(117, 117))[3], texture2D(u_texture, vec2(0, 0))[3], 1.0); + //uniform float u_color; \ + // gl_FragColor = vec4(u_color, u_color, u_color, 1.0); \ + // gl_FragColor = vec4(texture2D(u_texture, vec2(0.5, 0.5))[0], texture2D(u_texture, vec2(0.2, 0.2))[0], texture2D(u_texture, vec2(0.3, 0.3))[0], 1.0); + + + + // fxgl.app('drawLine1', { + // vsource: ` + // attribute vec2 a_position; + // attribute float a_texcoord; + // varying float v_texcoord; + + // void main(){ + // gl_Position = vec4(a_position, 0, 1.0); + // v_texcoord = a_texcoord; + // } + // `, + // fsource: ` + // precision highp float; + // uniform sampler2D f_sum_texture; + // varying float v_texcoord; + + // void main(){ + // gl_FragColor = texture2D(f_sum_texture, vec2(v_texcoord, 0.5)); + // } + // `, + // }, 1); + + + + + + -simpleDraw(6); // draw rectangle diff --git a/image.png b/image.png new file mode 100644 index 0000000..6774ae1 Binary files /dev/null and b/image.png differ diff --git a/package.json b/package.json index 504108e..e8b48ff 100644 --- a/package.json +++ b/package.json @@ -27,7 +27,12 @@ }, "homepage": "https://github.com/jpkli/flexgl#readme", "devDependencies": { - "uglifyjs-webpack-plugin": "^1.1.8", - "webpack": "^3.10.0" + "uglifyjs-webpack-plugin": "^1.1.6", + "webpack-cli": "^3.1.0", + "webpack-command": "^0.4.1" + }, + "dependencies": { + "npm": "^6.3.0", + "rxjs": "^6.2.2" } } diff --git a/play_framebuffer_1/main.css b/play_framebuffer_1/main.css new file mode 100644 index 0000000..a7f28b9 --- /dev/null +++ b/play_framebuffer_1/main.css @@ -0,0 +1,9 @@ +@import url("https://webglfundamentals.org/webgl/resources/webgl-tutorials.css"); +body { + margin: 0; +} +canvas { + width: 100vw; + height: 100vh; + display: block; +} diff --git a/play_framebuffer_1/main.html b/play_framebuffer_1/main.html new file mode 100644 index 0000000..ef7b1e4 --- /dev/null +++ b/play_framebuffer_1/main.html @@ -0,0 +1,64 @@ + + + + + + + + + + + + + + + + + diff --git a/play_framebuffer_1/main.js b/play_framebuffer_1/main.js new file mode 100644 index 0000000..7a1a1a2 --- /dev/null +++ b/play_framebuffer_1/main.js @@ -0,0 +1,219 @@ +"use strict"; + +function main() { + // Get A WebGL context + /** @type {HTMLCanvasElement} */ + var canvas = document.getElementById("canvas"); + var gl = canvas.getContext("webgl"); + if (!gl) { + return; + } + + // setup GLSL program + var program = webglUtils.createProgramFromScripts(gl, ["3d-vertex-shader", "3d-fragment-shader"]); + + // look up where the vertex data needs to go. + var positionLocation = gl.getAttribLocation(program, "a_position"); + var texcoordLocation = gl.getAttribLocation(program, "a_texcoord"); + + // lookup uniforms + var textureLocation = gl.getUniformLocation(program, "u_texture"); + var flagLocation = gl.getUniformLocation(program, "flag"); + + // Create a buffer for positions + var positionBuffer = gl.createBuffer(); + // Bind it to ARRAY_BUFFER (think of it as ARRAY_BUFFER = positionBuffer) + gl.bindBuffer(gl.ARRAY_BUFFER, positionBuffer); + // Put the positions in the buffer + setGeometry(gl); + + // provide texture coordinates for the rectangle. + var texcoordBuffer = gl.createBuffer(); + gl.bindBuffer(gl.ARRAY_BUFFER, texcoordBuffer); + // Set Texcoords. + setTexcoords(gl); + + // Create a texture. + var texture = gl.createTexture(); + gl.bindTexture(gl.TEXTURE_2D, texture); + + { + // fill texture with 3x2 pixels + const level = 0; + const internalFormat = gl.LUMINANCE; + const width = 4; + const height = 4; + const border = 0; + const format = gl.LUMINANCE; + const type = gl.UNSIGNED_BYTE; + const data = new Uint8Array([ + 0, 255, 0, 255, + 255, 0, 255, 0, + 0, 255, 0, 255, + 255, 0, 255, 0 + ]); + const alignment = 1; + gl.pixelStorei(gl.UNPACK_ALIGNMENT, alignment); + gl.texImage2D(gl.TEXTURE_2D, level, internalFormat, width, height, border, + format, type, data); + + // set the filtering so we don't need mips and it's not filtered + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); + } + + // Create a texture to render to + const targetTextureWidth = 2; + const targetTextureHeight = 2; + const targetTexture = gl.createTexture(); + gl.bindTexture(gl.TEXTURE_2D, targetTexture); + + { + // define size and format of level 0 + const level = 0; + const internalFormat = gl.RGBA; + const border = 0; + const format = gl.RGBA; + const type = gl.UNSIGNED_BYTE; + const data = null; + gl.texImage2D(gl.TEXTURE_2D, level, internalFormat, + targetTextureWidth, targetTextureHeight, border, + format, type, data); + + // set the filtering so we don't need mips + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); + } + + // Create and bind the framebuffer + const fb = gl.createFramebuffer(); + gl.bindFramebuffer(gl.FRAMEBUFFER, fb); + + // attach the texture as the first color attachment + const attachmentPoint = gl.COLOR_ATTACHMENT0; + const level = 0; + gl.framebufferTexture2D(gl.FRAMEBUFFER, attachmentPoint, gl.TEXTURE_2D, targetTexture, level); + + + requestAnimationFrame(drawScene); + + + function drawCube(flag) { + // Tell it to use our program (pair of shaders) + gl.useProgram(program); + + // Turn on the position attribute + gl.enableVertexAttribArray(positionLocation); + + // Bind the position buffer. + gl.bindBuffer(gl.ARRAY_BUFFER, positionBuffer); + + // Tell the position attribute how to get data out of positionBuffer (ARRAY_BUFFER) + var size = 3; // 3 components per iteration + var type = gl.FLOAT; // the data is 32bit floats + var normalize = false; // don't normalize the data + var stride = 0; // 0 = move forward size * sizeof(type) each iteration to get the next position + var offset = 0; // start at the beginning of the buffer + gl.vertexAttribPointer( + positionLocation, size, type, normalize, stride, offset); + + // Turn on the teccord attribute + gl.enableVertexAttribArray(texcoordLocation); + + // Bind the position buffer. + gl.bindBuffer(gl.ARRAY_BUFFER, texcoordBuffer); + + // Tell the position attribute how to get data out of positionBuffer (ARRAY_BUFFER) + var size = 2; // 2 components per iteration + var type = gl.FLOAT; // the data is 32bit floats + var normalize = false; // don't normalize the data + var stride = 0; // 0 = move forward size * sizeof(type) each iteration to get the next position + var offset = 0; // start at the beginning of the buffer + gl.vertexAttribPointer( + texcoordLocation, size, type, normalize, stride, offset); + + + // Tell the shader to use texture unit 0 for u_texture + gl.uniform1i(textureLocation, 0); + gl.uniform1i(flagLocation, flag); + + // Draw the geometry. + gl.drawArrays(gl.TRIANGLES, 0, 6); + } + + // Draw the scene. + function drawScene() { + + { + // render to our targetTexture by binding the framebuffer + gl.bindFramebuffer(gl.FRAMEBUFFER, fb); + + // render cube with our 3x2 texture + gl.bindTexture(gl.TEXTURE_2D, texture); + + // Tell WebGL how to convert from clip space to pixels + gl.viewport(0, 0, targetTextureWidth, targetTextureHeight); + + // Clear the canvas AND the depth buffer. + gl.clearColor(0, 0, 0, 0); // clear to blue + gl.clear(gl.COLOR_BUFFER_BIT); + + drawCube(0); + } + + { + // render to the canvas + gl.bindFramebuffer(gl.FRAMEBUFFER, null); + + // render the cube with the texture we just rendered to + gl.bindTexture(gl.TEXTURE_2D, targetTexture); + // gl.bindTexture(gl.TEXTURE_2D, texture); + + // Tell WebGL how to convert from clip space to pixels + gl.viewport(0, 0, gl.canvas.width, gl.canvas.height); + + // Clear the canvas AND the depth buffer. + gl.clearColor(0, 0, 0, 0); // clear to white + gl.clear(gl.COLOR_BUFFER_BIT); + + drawCube(1); + } + + requestAnimationFrame(drawScene); + } +} + +// Fill the buffer with the values that define a cube. +function setGeometry(gl) { + var positions = new Float32Array( + [ + -1.0, -1.0, 0.0, + -1.0, 1.0, 0.0, + 1.0, -1.0, 0.0, + -1.0, 1.0, 0.0, + 1.0, 1.0, 0.0, + 1.0, -1.0, 0.0, + ]); + gl.bufferData(gl.ARRAY_BUFFER, positions, gl.STATIC_DRAW); +} + +// Fill the buffer with texture coordinates the cube. +function setTexcoords(gl) { + gl.bufferData( + gl.ARRAY_BUFFER, + new Float32Array( + [ + 0, 0, + 0, 1, + 1, 0, + 0, 1, + 1, 1, + 1, 0, + ]), + gl.STATIC_DRAW); +} + +main(); diff --git a/play_framebuffer_2/main.css b/play_framebuffer_2/main.css new file mode 100644 index 0000000..a7f28b9 --- /dev/null +++ b/play_framebuffer_2/main.css @@ -0,0 +1,9 @@ +@import url("https://webglfundamentals.org/webgl/resources/webgl-tutorials.css"); +body { + margin: 0; +} +canvas { + width: 100vw; + height: 100vh; + display: block; +} diff --git a/play_framebuffer_2/main.html b/play_framebuffer_2/main.html new file mode 100644 index 0000000..92adfe7 --- /dev/null +++ b/play_framebuffer_2/main.html @@ -0,0 +1,66 @@ + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/play_framebuffer_2/main.js b/play_framebuffer_2/main.js new file mode 100644 index 0000000..bf7c58d --- /dev/null +++ b/play_framebuffer_2/main.js @@ -0,0 +1,278 @@ +"use strict"; + +function main() { + // Get A WebGL context + /** @type {HTMLCanvasElement} */ + var canvas = document.getElementById("canvas"); + var gl = canvas.getContext("webgl"); + if (!gl) { + return; + } + + // var blendExt = gl.getExtension("EXT_blend_minmax"); + // if (blendExt) { + // gl.MAX_EXT = blendExt.MAX_EXT; + // gl.MIN_EXT = blendExt.MIN_EXT; + // } + // gl.ext = gl.getExtension("ANGLE_instanced_arrays"); + gl.getExtension("OES_texture_float"); + gl.getExtension("OES_texture_float_linear"); + + // setup GLSL program + var program = webglUtils.createProgramFromScripts(gl, ["3d-vertex-shader", "3d-fragment-shader"]); + + // look up where the vertex data needs to go. + var positionLocation = gl.getAttribLocation(program, "a_position"); + var texcoordLocation = gl.getAttribLocation(program, "a_texcoord"); + + // lookup uniforms + var textureLocation = gl.getUniformLocation(program, "u_texture"); + var flagLocation = gl.getUniformLocation(program, "flag"); + + // Create a buffer for positions + var positionBuffer = gl.createBuffer(); + // Bind it to ARRAY_BUFFER (think of it as ARRAY_BUFFER = positionBuffer) + gl.bindBuffer(gl.ARRAY_BUFFER, positionBuffer); + // Put the positions in the buffer + setGeometry(gl); + + // provide texture coordinates for the rectangle. + var texcoordBuffer = gl.createBuffer(); + gl.bindBuffer(gl.ARRAY_BUFFER, texcoordBuffer); + // Set Texcoords. + setTexcoords(gl); + + // Create a texture. + var texture = gl.createTexture(); + gl.bindTexture(gl.TEXTURE_2D, texture); + + { + // fill texture with 3x2 pixels + const level = 0; + const internalFormat = gl.ALPHA; + const width = 1024; + const height = 1024; + const border = 0; + const format = gl.ALPHA; + const type = gl.FLOAT; + // const type = gl.UNSIGNED_BYTE; + + // var data = []; + // for(var j = 0; j < 100; j++){ + // var typedArray = new Float32Array(1024*1024); + // for (var i = 0; i < typedArray.length; i++){ + // typedArray[i] = Math.random(); + // } + // data.push(typedArray); + // } + + var data = new Float32Array(1024*1024); + for(var i = 0; i < data.length; i++){ + data[i] = Math.random(); + } + + // var typedArray = new Float32Array(1024*1024); + // var typedArray = new Uint8Array(1024*1024); + + const alignment = 1; + gl.pixelStorei(gl.UNPACK_ALIGNMENT, alignment); + gl.texImage2D(gl.TEXTURE_2D, level, internalFormat, width, height, border, + format, type, data); + + // set the filtering so we don't need mips and it's not filtered + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); + } + + // Create a texture to render to + const targetTextureWidth = 1024; + const targetTextureHeight = 1; + const targetTexture = gl.createTexture(); + gl.bindTexture(gl.TEXTURE_2D, targetTexture); + + { + // define size and format of level 0 + const level = 0; + const internalFormat = gl.RGBA; + const border = 0; + const format = gl.RGBA; + const type = gl.FLOAT; + // const type = gl.UNSIGNED_BYTE; + gl.texImage2D(gl.TEXTURE_2D, level, internalFormat, + targetTextureWidth, targetTextureHeight, border, + format, type, null); + + // set the filtering so we don't need mips + // gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR); + // gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); + // gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); + // + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); + } + + // Create and bind the framebuffer + const fb = gl.createFramebuffer(); + gl.bindFramebuffer(gl.FRAMEBUFFER, fb); + + // attach the texture as the first color attachment + const attachmentPoint = gl.COLOR_ATTACHMENT0; + const level = 0; + gl.framebufferTexture2D(gl.FRAMEBUFFER, attachmentPoint, gl.TEXTURE_2D, targetTexture, level); + + + requestAnimationFrame(drawScene); + + + function drawCube(flag) { + // Tell it to use our program (pair of shaders) + gl.useProgram(program); + + // Turn on the position attribute + gl.enableVertexAttribArray(positionLocation); + + // Bind the position buffer. + gl.bindBuffer(gl.ARRAY_BUFFER, positionBuffer); + + // Tell the position attribute how to get data out of positionBuffer (ARRAY_BUFFER) + var size = 2; // 3 components per iteration + var type = gl.FLOAT; // the data is 32bit floats + var normalize = false; // don't normalize the data + var stride = 0; // 0 = move forward size * sizeof(type) each iteration to get the next position + var offset = 0; // start at the beginning of the buffer + gl.vertexAttribPointer( + positionLocation, size, type, normalize, stride, offset); + + // Turn on the teccord attribute + gl.enableVertexAttribArray(texcoordLocation); + + // Bind the position buffer. + gl.bindBuffer(gl.ARRAY_BUFFER, texcoordBuffer); + + // Tell the position attribute how to get data out of positionBuffer (ARRAY_BUFFER) + var size = 1; // 2 components per iteration + var type = gl.FLOAT; // the data is 32bit floats + var normalize = false; // don't normalize the data + var stride = 0; // 0 = move forward size * sizeof(type) each iteration to get the next position + var offset = 0; // start at the beginning of the buffer + gl.vertexAttribPointer( + texcoordLocation, size, type, normalize, stride, offset); + + + // Tell the shader to use texture unit 0 for u_texture + gl.uniform1i(textureLocation, 0); + gl.uniform1i(flagLocation, flag); + + // Draw the geometry. + gl.drawArrays(gl.LINES, 0, 2); + } + + // Draw the scene. + function drawScene() { + + { + // render to our targetTexture by binding the framebuffer + gl.bindFramebuffer(gl.FRAMEBUFFER, fb); + + // render cube with our 3x2 texture + gl.bindTexture(gl.TEXTURE_2D, texture); + + // Tell WebGL how to convert from clip space to pixels + gl.viewport(0, 0, targetTextureWidth, targetTextureHeight); + + // Clear the canvas AND the depth buffer. + gl.clearColor(0, 0, 0, 0); // clear to blue + gl.clear(gl.COLOR_BUFFER_BIT); + + drawCube(0); + } + + { + // render to the canvas + gl.bindFramebuffer(gl.FRAMEBUFFER, null); + + // render the cube with the texture we just rendered to + gl.bindTexture(gl.TEXTURE_2D, targetTexture); + // gl.bindTexture(gl.TEXTURE_2D, texture); + + // Tell WebGL how to convert from clip space to pixels + gl.viewport(0, 0, gl.canvas.width, gl.canvas.height); + + // Clear the canvas AND the depth buffer. + gl.clearColor(0, 0, 0, 0); // clear to white + gl.clear(gl.COLOR_BUFFER_BIT); + + drawCube(1); + } + + requestAnimationFrame(drawScene); + } +} + +// Fill the buffer with the values that define a cube. +// function setGeometry(gl) { +// var positions = new Float32Array( +// [ +// -1.0, -1.0, 0.0, +// -1.0, 1.0, 0.0, +// 1.0, -1.0, 0.0, +// -1.0, 1.0, 0.0, +// 1.0, 1.0, 0.0, +// 1.0, -1.0, 0.0, +// ]); +// gl.bufferData(gl.ARRAY_BUFFER, positions, gl.STATIC_DRAW); +// } +function setGeometry(gl) { + var positions = new Float32Array( + [ + -1.0, -1.0, + 1.0, 1.0, + ]); + gl.bufferData(gl.ARRAY_BUFFER, positions, gl.STATIC_DRAW); +} +// function setTexcoords(gl) { +// gl.bufferData( +// gl.ARRAY_BUFFER, +// new Float32Array( +// [ +// 0, 0, +// 0, 1, +// 1, 0, +// 0, 1, +// 1, 1, +// 1, 0, +// ]), +// gl.STATIC_DRAW); +// } +// Fill the buffer with texture coordinates the cube. +// function setTexcoords(gl) { +// gl.bufferData( +// gl.ARRAY_BUFFER, +// new Float32Array( +// [ +// 0.0, 0.0, +// 1.0, 0.0, +// 0.0, 0.0, +// 1.0, 0.0, +// 1.0, 0.0, +// 0.0, 0.0, +// ]), +// gl.STATIC_DRAW); +// } +function setTexcoords(gl) { + gl.bufferData( + gl.ARRAY_BUFFER, + new Float32Array( + [ + 0.0, + 1.0, + ]), + gl.STATIC_DRAW); +} + + +main(); diff --git a/src/attribute.js b/src/attribute.js index 6f404d3..51fb398 100644 --- a/src/attribute.js +++ b/src/attribute.js @@ -1,18 +1,27 @@ -export default function Attribute(glContext) { - +export default function Attribute(glContext) +{ var attribute = (this instanceof Attribute) ? this : {}, ctx = glContext, attributeID = 0; - function setAttribute(name, data) { - if(Array.isArray(data) || ArrayBuffer.isView(data)){ - if(!ArrayBuffer.isView(data)) data = new Float32Array(data); + function setAttribute(name, data) + { + if( Array.isArray(data) || ArrayBuffer.isView(data) ) + { + if(!ArrayBuffer.isView(data)) + { + data = new Float32Array(data); + } attribute[name].data = data; ctx.bindBuffer(ctx.ARRAY_BUFFER, attribute[name].ptr); ctx.bufferData(ctx.ARRAY_BUFFER, data, ctx.STATIC_DRAW); + // console.log(attribute[name].ptr === attribute[name].old_ptr); + // attribute[name].old_ptr = attribute[name].ptr; } } - attribute.create = function(name, type, data) { + + attribute.create = function(name, type, data) + { attribute[name] = { name: name, type: type || 'float', @@ -22,9 +31,11 @@ export default function Attribute(glContext) { size: parseInt(type.slice(3,4)) || 1 }; - if(data !== null && data.length) setAttribute(name, data); + if(data !== null && data.length) + {setAttribute(name, data);} - attribute[name].link = function(program) { + attribute[name].link = function(program) + { ctx.bindBuffer(ctx.ARRAY_BUFFER, this.ptr); this.location = ctx.getAttribLocation(program, this.name); ctx.vertexAttribPointer(this.location, this.size, ctx.FLOAT, false, 0, 0); @@ -32,14 +43,15 @@ export default function Attribute(glContext) { return this; } - attribute[name].load = function(arrayBuffer) { + attribute[name].load = function(arrayBuffer) + { setAttribute(this.name, arrayBuffer); return this; } - attribute[name].header = function() { - return 'attribute ' + this.type + ' ' + this.name + ';\n'; - } + attribute[name].header = function() + {return 'attribute ' + this.type + ' ' + this.name + ';\n';} + attribute[name].delete = function() { ctx.deleteBuffer(this.ptr); diff --git a/src/flexgl.js b/src/flexgl.js index ff3db89..913f59f 100644 --- a/src/flexgl.js +++ b/src/flexgl.js @@ -1,12 +1,14 @@ import Resource from './resource'; -import ProgramManager from './program'; -import Shader from './shader'; +import Program from './program'; import Framebuffer from './framebuffer'; +// import Reactive from './reactive'; export default function FlexGL(arg) { - var flexgl = (this instanceof FlexGL) ? this : {}, - options = arg || {}, + var flexgl = (this instanceof FlexGL) ? this : {}; + + + var options = arg || {}, container = options.container || null, canvas = options.canvas || document.createElement("canvas"), width = options.width || null, @@ -19,10 +21,10 @@ export default function FlexGL(arg) { }, ctx = options.context || options.ctx || null, kernels = {}, - program = null, sharedFunction = options.sharedFunction || {}; + if (typeof(canvas) == "string") { if (canvas[0] == "#") canvas = document.getElementById(cavnas.substring(1)); else canvas = document.getElementById(cavnas); @@ -40,25 +42,25 @@ export default function FlexGL(arg) { canvas.style.marginLeft = padding.left + "px"; canvas.style.marginTop = padding.top + "px"; + if (ctx === null) ctx = setupWebGL(canvas); flexgl.ctx = ctx; flexgl.canvas = canvas; - + flexgl.resources = resources; ctx._dict = options.env || options.dict || options.dictionary || {}; - var resources = new Resource(ctx), framebuffers = new Framebuffer(ctx), - programManager = new ProgramManager(ctx, resources), - shaders = new Shader(ctx, resources); + program = new Program(ctx, resources), + realProgram = null; + var blendExt = ctx.getExtension("EXT_blend_minmax"); if (blendExt) { ctx.MAX_EXT = blendExt.MAX_EXT; ctx.MIN_EXT = blendExt.MIN_EXT; } - ctx.ext = ctx.getExtension("ANGLE_instanced_arrays"); enableExtension([ "OES_texture_float", @@ -66,10 +68,10 @@ export default function FlexGL(arg) { // "OES_texture_half_float", // "OES_texture_half_float_linear" ]); - if (container) container.appendChild(canvas); + function setupWebGL(canvas) { var names = ["webgl", "experimental-webgl"]; var gl = null; @@ -95,7 +97,6 @@ export default function FlexGL(arg) { } }); }; - flexgl.enableExtension = enableExtension; /** @@ -107,17 +108,19 @@ export default function FlexGL(arg) { */ flexgl.attribute = function(name, type, data) { resources.allocate("attribute", name, type, data); - Object.defineProperty(flexgl.attribute, name, { - get: function() { - return resources.attribute[name]; - }, - set: function(data) { - resources.attribute[name].load(data); - } - }); + if (!flexgl.attribute.hasOwnProperty(name)) { + Object.defineProperty(flexgl.attribute, name, { //after allocating, flexgl gets new key attribute, helping easily change attribute data. + get(){ + return resources.attribute[name]; + }, + set(data){ + resources.attribute[name].load(data); + } + }); + } return flexgl; }; - flexgl.buffer = flexgl.attribute; //alias + // flexgl.buffer = flexgl.attribute; //alias /** * Create a Uniform variable for WebGL shader programs @@ -135,8 +138,8 @@ export default function FlexGL(arg) { }, set: function(data) { resources.uniform[name].load(data); - if (ctx.isProgram(program)) - resources.uniform[name].link(program); + if (ctx.isProgram(realProgram)) /////////////////// + resources.uniform[name].link(realProgram); ///////////////// } }); } @@ -200,8 +203,8 @@ export default function FlexGL(arg) { */ flexgl.framebuffer = function(name, type, dim, texture) { var texture = texture || resources.allocate('texture', name, type, dim, 'rgba', null); - framebuffers.create(name, type, dim, texture); + if (!flexgl.framebuffer.hasOwnProperty(name)) { Object.defineProperty(flexgl.framebuffer, name, { get: function() { @@ -212,7 +215,7 @@ export default function FlexGL(arg) { return flexgl; } - flexgl.framebuffer.enableRead = function(name) { + flexgl.framebuffer.enableRead = function(name, program) { framebuffers[name].enableRead(program); } @@ -251,31 +254,13 @@ export default function FlexGL(arg) { } flexgl.dictionary = flexgl.parameter; + flexgl.shader = program.shader; - flexgl.shader = programManager.shader; - - flexgl.program = function(name, vs, fs) { - program = programManager.program(name, vs, fs); - return ctx; - } - - flexgl.createProgram = function(name, vs, fs) { - program = programManager.create(name, vs, fs); - return ctx; - } - - flexgl.app = function(name, props) { - var vs = flexgl.shader.vertex(props.vs), - fs = flexgl.shader.fragment(props.fs), - fb = props.framebuffer || null; - - flexgl.program(name, vs, fs); - var draw = props.render || props.draw; - - return function(args) { - var gl = flexgl.program(name); - return draw.call(gl, args); + flexgl.app = function(name, options) { + return function(args){ + realProgram = program.use(name, options.vertex_shader_source, options.fragment_shader_source); + return options.render.call(flexgl, args); } } @@ -283,7 +268,6 @@ export default function FlexGL(arg) { return [canvas.width, canvas.height]; } - flexgl.resources = resources; - return flexgl; } + diff --git a/src/framebuffer.js b/src/framebuffer.js index 2a50cb7..2d2bc97 100644 --- a/src/framebuffer.js +++ b/src/framebuffer.js @@ -14,25 +14,25 @@ export default function Framebuffer(glContext) { width: dim[0] || 1024, height: dim[1] || 1024, texture: texture || null, - renderbuffer: ctx.createRenderbuffer(), + // renderbuffer: ctx.createRenderbuffer(), } - if (framebuffer[name].texture === null) { - var buf = (type == 'float') ? - new Float32Array(dim[0] * dim[1] * 4) : - new Uint8Array(dim[0] * dim[1] * 4); - framebuffer[name].texture = Texture(ctx).create(name, type, dim, "rgba", buf); - } + // if (framebuffer[name].texture === null) { + // var buf = (type == 'float') ? + // new Float32Array(dim[0] * dim[1] * 4) : + // new Uint8Array(dim[0] * dim[1] * 4); + // framebuffer[name].texture = Texture(ctx).create(name, type, dim, "rgba", buf); + // } - var renderbuffer = framebuffer[name].renderbuffer; + // var renderbuffer = framebuffer[name].renderbuffer; ctx.bindFramebuffer(ctx.FRAMEBUFFER, framebuffer[name].ptr); - ctx.bindRenderbuffer(ctx.RENDERBUFFER, renderbuffer); - ctx.renderbufferStorage( - ctx.RENDERBUFFER, - ctx.DEPTH_COMPONENT16, - framebuffer[name].width, - framebuffer[name].height - ); + // ctx.bindRenderbuffer(ctx.RENDERBUFFER, renderbuffer); + // ctx.renderbufferStorage( + // ctx.RENDERBUFFER, + // ctx.DEPTH_COMPONENT16, + // framebuffer[name].width, + // framebuffer[name].height + // ); ctx.framebufferTexture2D( ctx.FRAMEBUFFER, ctx.COLOR_ATTACHMENT0, @@ -40,13 +40,13 @@ export default function Framebuffer(glContext) { framebuffer[name].texture.ptr, 0 ); - ctx.framebufferRenderbuffer( - ctx.FRAMEBUFFER, - ctx.DEPTH_ATTACHMENT, - ctx.RENDERBUFFER, - renderbuffer - ); - ctx.bindRenderbuffer(ctx.RENDERBUFFER, null); + // ctx.framebufferRenderbuffer( + // ctx.FRAMEBUFFER, + // ctx.DEPTH_ATTACHMENT, + // ctx.RENDERBUFFER, + // renderbuffer + // ); + // ctx.bindRenderbuffer(ctx.RENDERBUFFER, null); ctx.bindFramebuffer(ctx.FRAMEBUFFER, null); framebuffer[name].enableRead = function(program) { @@ -56,6 +56,10 @@ export default function Framebuffer(glContext) { ctx.uniform1i(this.texture.location, this.texture.index); }; + // framebuffer[name].enableRead = function(program) { + // this.texture.link(program); + // }; + framebuffer[name].delete = function() { ctx.bindRenderbuffer(gl.RENDERBUFFER, null); ctx.bindFramebuffer(gl.FRAMEBUFFER, null); diff --git a/src/program.js b/src/program.js index 8bb88a9..67271f3 100644 --- a/src/program.js +++ b/src/program.js @@ -2,26 +2,24 @@ import Shader from './shader'; export default function Program(glContext, resources) { - var program, + var program = {}, ctx = glContext, - pm = {}, kernels = {}, - shaders = new Shader(glContext, resources); + shader = new Shader(glContext); - pm.create = function(name, vs, fs) { + program.create = function(name, vs, fs) { var name = name || "default", vs = vs || "default", fs = fs || "default", deps = []; if (kernels.hasOwnProperty(name)) { - pm.delete(name); + this.delete(name); } kernels[name] = ctx.createProgram(); - - kernels[name].vs = (typeof vs == "object") ? vs : shaders.vertex[vs]; - kernels[name].fs = (typeof fs == "object") ? fs : shaders.fragment[fs]; + kernels[name].vs = vs; + kernels[name].fs = fs; ctx.attachShader(kernels[name], kernels[name].vs); ctx.attachShader(kernels[name], kernels[name].fs); @@ -31,28 +29,30 @@ export default function Program(glContext, resources) { var lastError = ctx.getProgramInfoLog(kernels[name]); throw ("Error in program linking:" + lastError); ctx.deleteProgram(kernels[name]); - return null; } deps = deps.concat(kernels[name].vs.deps); deps = deps.concat(kernels[name].fs.deps); kernels[name].deps = deps; - return kernels[name]; } - pm.use = pm.program = function(name, vs, fs) { + program.use = function(name, vertex_shader_source, fragment_shader_source) { if (kernels.hasOwnProperty(name)) { - program = kernels[name]; - ctx.useProgram(program); - resources.link(program, program.deps); - return program; - } else { - return pm.create(name, vs, fs); + ctx.useProgram(kernels[name]); + resources.link(kernels[name], kernels[name].deps); + return kernels[name]; + } + else { + shader.create(name, vertex_shader_source, fragment_shader_source); + this.create(name, shader[name].vs, shader[name].fs); + ctx.useProgram(kernels[name]); + resources.link(kernels[name], kernels[name].deps); + return kernels[name]; } } - pm.delete = function(name) { + program.delete = function(name) { if (kernels.hasOwnProperty(name)) { ctx.detachShader(kernels[name], kernels[name].vs); ctx.detachShader(kernels[name], kernels[name].fs); @@ -61,27 +61,5 @@ export default function Program(glContext, resources) { } } - pm.shader = function(arg, fn) { - var options = arg; - shaders.create(options, fn); - return pm; - } - - pm.shader.vertex = function(fn) { - var options = { - type: "vertex" - }; - if (fn.name) options.name = fn.name; - return shaders.create(options, fn); - } - - pm.shader.fragment = function(fn) { - var options = { - type: "fragment" - }; - if (fn.name) options.name = fn.name; - return shaders.create(options, fn); - } - - return pm; + return program; } diff --git a/src/reactive.js b/src/reactive.js new file mode 100644 index 0000000..c03f55f --- /dev/null +++ b/src/reactive.js @@ -0,0 +1,31 @@ +function Reactive(data, chunk) +{ + var reactive = (this instanceof Reactive) ? this : { + pointer : -1, + size: chunk, + data_all: data + }; + + reactive.updateButton = document.querySelector('.update'); + reactive.updateClickStream = Rx.Observable.fromEvent(reactive.updateButton, 'click'); + + reactive.requestStream = reactive.updateClickStream.startWith('startup click') + .flatMap(function(){ + reactive.pointer++; + return reactive.pointer; + }); + + reactive.responseStream = reactive.requestStream + .flatMap(function(p){ + return reactive.data_all.slice(p*reactive.size, (p+1)*reactive.size); + }); + + reactive.responseStream.subscribe(function(d){ + reactive.data_chunk = d; + reactive.sum += reactive.data_chunk.reduce(function(x,y){ + return x+y; + }) + }); + + return reactive; +} diff --git a/src/resource.js b/src/resource.js index 284219a..02c629f 100644 --- a/src/resource.js +++ b/src/resource.js @@ -4,7 +4,8 @@ import Texture from './texture'; import Varying from './varying'; import Subroutine from './subroutine'; -export default function Resource(glContext) { +export default function Resource(glContext) +{ var resource = (this instanceof Resource) ? this : {}, gpuResources = {}; @@ -16,38 +17,36 @@ export default function Resource(glContext) { var resourceTypes = ['uniform', 'attribute', 'texture', 'varying', 'subroutine']; - resource.allocate = function(type, props) { - if (resourceTypes.indexOf(type) === -1) { - throw Error("Error: Invalid resource type: " + type); - } + resource.allocate = function(type) + { + if (resourceTypes.indexOf(type) === -1) { throw Error("Error: Invalid resource type: " + type); } + var res = resource[type].create.apply(null, Array.prototype.slice.call(arguments, 1)); res.resourceType = type; gpuResources[res.name] = res; - if (!gpuResources.hasOwnProperty(res.name)) { + + if (!gpuResources.hasOwnProperty(res.name)) + { Object.defineProperty(gpuResources, res.name, { - get: function() { - return gpuResources[res.name]; - }, - set: function(data) { - gpuResources[res.name].load(data); - } + get: function() { return gpuResources[res.name];}, + set: function(data) { gpuResources[res.name].load(data);} }); } return res; }; - resource.link = function(program, resources) { - var requiredResources = (Array.isArray(resources)) ? resources : Object.keys(gpuResources); - requiredResources.forEach(function(resourceName) { - if (gpuResources.hasOwnProperty(resourceName)) + resource.link = function(program, resource_names) { + var requiredResourceNames = (Array.isArray(resource_names)) ? resource_names : Object.keys(gpuResources); + requiredResourceNames.forEach(function(resourceName) { + if (gpuResources.hasOwnProperty(resourceName)) { gpuResources[resourceName].link(program); + } }) }; resource.get = function(name) { return gpuResources[name]; } - resource.create = resource.allocate; return resource; diff --git a/src/shader.js b/src/shader.js index 6003c99..9001ae7 100644 --- a/src/shader.js +++ b/src/shader.js @@ -1,227 +1,46 @@ -export default function Shader(glContext, glResource) { +export default function Shader(glContext) { var shader = (this instanceof Shader) ? this : {}, - ctx = glContext, - resource = glResource, - parameters = ctx._dict || {}; - - shader.vertex = {}; - shader.fragment = {}; - - var shaderType = { - vertex: ctx.VERTEX_SHADER, - fragment: ctx.FRAGMENT_SHADER - }; - - // Convert JS functions to GLSL codes - function toGLSL(returnType, name, fn){ - - var glsl = returnType + ' ' + - name + '(' + applyEnvParameters(fn.toString()) - .replace( - /var\s+([\w|\d]+)\s*=\s*new\s+([\w|\d]+)\((.*)\)/g, - function(expr, name, dtype, value){ - var parts; - if(value) - parts = [dtype.toLowerCase(), name, '=', value]; - else - parts = [dtype.toLowerCase(), name]; - - return parts.join(' ') - } - ) - .replace(/for\s*\(\s*var\s+/g, 'for(int ') - .replace(/var\s/g, 'float ') - .replace(/this./g, '') - .replace(/\$(.*)\((.*)\)\s*(=|;)/g, "$1 $2 $3"); - // .replace(/\$(.*?)\./g, "$1 ") - - if(name == "main") { - glsl = glsl.replace(/function.*\(\s*([\s\S]*?)\s*{/, '){') + "\n"; - } else { - var args = glsl.match(/function.*\(\s*([\s\S]*?)\s*\)/)[1]; - - if(args != "") { - args = args.replace(/\$([\w|\d]+)_/g, "$1 "); + ctx = glContext; + // resource = glResource, + // parameters = ctx._dict || {}; + + shader.create = function(name, vertex_shader_source, fragment_shader_source){ + shader[name] = { + name: name, + vertex_shader_source: vertex_shader_source, + fragment_shader_source: fragment_shader_source, + vs: createShader(ctx, ctx.VERTEX_SHADER, vertex_shader_source), + fs: createShader(ctx, ctx.FRAGMENT_SHADER, fragment_shader_source) + } + + shader[name].vs.deps = []; + addDeps(vertex_shader_source, shader[name].vs.deps); + shader[name].fs.deps = []; + addDeps(fragment_shader_source, shader[name].vs.deps); + + function createShader(gl, type, source) { + var shader = gl.createShader(type); + gl.shaderSource(shader, source); + gl.compileShader(shader); + var success = gl.getShaderParameter(shader, gl.COMPILE_STATUS); + if (success) { + return shader; } - glsl = glsl - .replace(/function.*\(\s*([\s\S]*?)\s*\)/, args+')') + "\n"; - } - return glsl; - } - - //set parameters in JS functions before converting to GLSL codes - function applyEnvParameters(str){ - //find all $(...) and replace them with parameters - var envParam = Object.keys(parameters); - if(envParam.length > 0){ - var re = new RegExp("\\$\\(("+envParam.join("|")+")\\)","g"); - str = str.replace(re, function(matched){ - return parameters[matched.slice(2,matched.length-1)]; - }); - } - - // Make uniforms to be used as parameters in shaders, like $(uniformName) - // var envUniforms = Object.keys(resource.uniform); - // re = new RegExp("\\$\\(("+envUniforms.join("|")+")\\)","g"); - // str = str.replace(re, function(matched){ - // return resource.uniform[matched.slice(2,matched.length-1)].data; - // }); - - return str; - } - - function compile(shaderType, shaderSource) { - if (shaderType !== ctx.VERTEX_SHADER && shaderType !== ctx.FRAGMENT_SHADER) { - throw ("Error: unknown shader type"); - } - var _shader = ctx.createShader(shaderType); - ctx.shaderSource(_shader, shaderSource); - ctx.compileShader(_shader); - - // Check the compile status, get compile error if any - var compiled = ctx.getShaderParameter(_shader, ctx.COMPILE_STATUS); - if (!compiled) { - var lastError = ctx.getShaderInfoLog(_shader); - console.log(shaderSource + '\n ===================================================='); - throw new Error("Error compiling shader '" + _shader + "':" + lastError); - - ctx.deleteShader(_shader); - return null; + // console.log('NO!'); + console.log(gl.getShaderInfoLog(shader)); + gl.deleteShader(shader); } - return _shader; - } - - function getDeps(fn) { - var deps = [], - sourceCode = fn.toString(), - shaderArgs = sourceCode.match(/function\s.*?\(([^)]*)\)/), - args = (shaderArgs !== null && shaderArgs.length) ? shaderArgs[1] : []; - // args = args.replace(/(?:\r\n|\r|\n|\s)/g, ''); - // - if(args.length) { - deps = args.split(',').map(function(arg) { - return arg.replace(/\/\*.*\*\//, '').trim(); - }).filter(function(arg) { - return arg; - }); - } - - var extraDeps = getExtraDeps(sourceCode); - if(extraDeps.length) { - deps = deps.concat(extraDeps - .filter(function(d){ - return deps.indexOf(d) === -1; - })) - } - - return deps; - } - - function getExtraDeps(fnString) { - var extraDeps = fnString.match(/this\.(\w+)/g); - if(extraDeps !== null) { - extraDeps = extraDeps.map(function(d){ - return d.slice(5); - }); - } - return extraDeps || []; - } - - function declareDep(dep) { - var res = resource.get(dep); - if(typeof res === 'undefined') - throw new Error('Resource/dependence "' + dep + '" is not found.'); - if(res.resourceType == 'subroutine') - return toGLSL(res.type, res.name, res.fn); - else - return res.header(); - } - - function uniqueDeps(deps) { - var names = {}; - deps.forEach(function(d, i){ - names[d] = i; - }); - - return Object.keys(names); - } - - shader.create = function(arg, fn){ - var option = arg || {}, - name = option.name || "default", - type = option.type || "vertex", - deps = option.require || option.deps || [], - precision = option.precision || "high", - debug = option.debug || false, - main = option.main || fn || function() {}; - - var shaderSource = 'precision ' + precision + 'p float;\n'; - - if(deps.length === 0) deps = uniqueDeps(getDeps(main)); - - //get dependence from subroutines if any - var extraDeps = [], - subRoutines = []; - - deps.forEach(function(dep){ - var res = resource.get(dep); - if(typeof res == 'undefined') { - console.log(dep); - throw Error ('Error! Undefined variable in shader: '+ dep.name); - } - if(res.resourceType == 'subroutine') { - subRoutines.push(res.name); - var subDeps = getExtraDeps(res.fn.toString()); - if(subDeps.length) { - //TODO: make this recursive to check all subroutine deps - subDeps.forEach(function(sdep){ - var sres = resource.get(sdep); - if(sres.resourceType == 'subroutine') - extraDeps = extraDeps.concat(getExtraDeps(sres.fn.toString())); - }) - - extraDeps = extraDeps.concat(subDeps); + function addDeps(source, deps){ + var re = /\s*(attribute|uniform)\s+\w+\s+(\w+)/; + source.split('\n').forEach(function(v){ + var result = re.exec(v); + if(result){ + deps.push(result[2]); } - } - }) - - if(extraDeps.length) { - var allDeps = extraDeps - // .filter(function(d){ - // return deps.indexOf(d) === -1; - // }) - .concat(deps.filter(function(d){ - return subRoutines.indexOf(d) === -1; - })) - .concat(subRoutines); - - deps = uniqueDeps(allDeps); - } - - - if(Array.isArray(deps)){ - deps.forEach(function(dep){ - shaderSource += declareDep(dep); }); - } else if(typeof(deps) == 'object') { - Object.keys(deps).forEach(function(resourceType){ - deps[resourceType].forEach(function(dep){ - shaderSource += declareDep(dep); - }); - }) } - - shaderSource += toGLSL('void', 'main', main); - if(debug) - console.log(shaderSource); - var _shader = compile(shaderType[type], shaderSource); - _shader._shaderType = shaderType[type]; - _shader.deps = deps; - _shader.source = shaderSource; - shader[type][name] = _shader; - return _shader; } return shader; diff --git a/src/shader_expire.js b/src/shader_expire.js new file mode 100644 index 0000000..03a1a6d --- /dev/null +++ b/src/shader_expire.js @@ -0,0 +1,230 @@ +export default function Shader(glContext, glResource) { + + var shader = (this instanceof Shader) ? this : {}, + ctx = glContext, + resource = glResource, + parameters = ctx._dict || {}; + + shader.vertex = {}; + shader.fragment = {}; + + var shaderType = { + vertex: ctx.VERTEX_SHADER, + fragment: ctx.FRAGMENT_SHADER + }; + + // Convert JS functions to GLSL codes + function toGLSL(returnType, name, fn){ + + var glsl = returnType + ' ' + + name + '(' + applyEnvParameters(fn.toString()) + .replace( + /var\s+([\w|\d]+)\s*=\s*new\s+([\w|\d]+)\((.*)\)/g, + function(expr, name, dtype, value){ + var parts; + if(value) + parts = [dtype.toLowerCase(), name, '=', value]; + else + parts = [dtype.toLowerCase(), name]; + + return parts.join(' ') + } + ) + .replace(/for\s*\(\s*var\s+/g, 'for(int ') + .replace(/var\s/g, 'float ') + .replace(/this./g, '') + .replace(/\$(.*)\((.*)\)\s*(=|;)/g, "$1 $2 $3"); + // .replace(/\$(.*?)\./g, "$1 ") + + if(name == "main") { + glsl = glsl.replace(/function.*\(\s*([\s\S]*?)\s*{/, '){') + "\n"; + } else { + var args = glsl.match(/function.*\(\s*([\s\S]*?)\s*\)/)[1]; + + if(args != "") { + args = args.replace(/\$([\w|\d]+)_/g, "$1 "); + } + glsl = glsl + .replace(/function.*\(\s*([\s\S]*?)\s*\)/, args+')') + "\n"; + } + return glsl; + } + + //set parameters in JS functions before converting to GLSL codes + function applyEnvParameters(str){ + //find all $(...) and replace them with parameters + var envParam = Object.keys(parameters); + if(envParam.length > 0){ + var re = new RegExp("\\$\\(("+envParam.join("|")+")\\)","g"); + str = str.replace(re, function(matched){ + return parameters[matched.slice(2,matched.length-1)]; + }); + } + + // Make uniforms to be used as parameters in shaders, like $(uniformName) + // var envUniforms = Object.keys(resource.uniform); + // re = new RegExp("\\$\\(("+envUniforms.join("|")+")\\)","g"); + // str = str.replace(re, function(matched){ + // return resource.uniform[matched.slice(2,matched.length-1)].data; + // }); + + return str; + } + + function compile(shaderType, shaderSource) { + if (shaderType !== ctx.VERTEX_SHADER && shaderType !== ctx.FRAGMENT_SHADER) { + throw ("Error: unknown shader type"); + } + var _shader = ctx.createShader(shaderType); + ctx.shaderSource(_shader, shaderSource); + ctx.compileShader(_shader); + + // Check the compile status, get compile error if any + var compiled = ctx.getShaderParameter(_shader, ctx.COMPILE_STATUS); + if (!compiled) { + var lastError = ctx.getShaderInfoLog(_shader); + console.log(shaderSource + '\n ===================================================='); + throw new Error("Error compiling shader '" + _shader + "':" + lastError); + + ctx.deleteShader(_shader); + return null; + } + + return _shader; + } + + function getDeps(fn) { + var deps = [], + sourceCode = fn.toString(), + shaderArgs = sourceCode.match(/function\s.*?\(([^)]*)\)/), + args = (shaderArgs !== null && shaderArgs.length) ? shaderArgs[1] : []; + // args = args.replace(/(?:\r\n|\r|\n|\s)/g, ''); + // + if(args.length) { + deps = args.split(',').map(function(arg) { + return arg.replace(/\/\*.*\*\//, '').trim(); + }).filter(function(arg) { + return arg; + }); + } + + var extraDeps = getExtraDeps(sourceCode); + if(extraDeps.length) { + deps = deps.concat(extraDeps + .filter(function(d){ + return deps.indexOf(d) === -1; + })) + } + + return deps; + } + + function getExtraDeps(fnString) { + var extraDeps = fnString.match(/this\.(\w+)/g); + if(extraDeps !== null) { + extraDeps = extraDeps.map(function(d){ + return d.slice(5); + }); + } + return extraDeps || []; + } + + function declareDep(dep) { + var res = resource.get(dep); + if(typeof res === 'undefined') + throw new Error('Resource/dependence "' + dep + '" is not found.'); + if(res.resourceType == 'subroutine') + return toGLSL(res.type, res.name, res.fn); + else + return res.header(); + } + + function uniqueDeps(deps) { + var names = {}; + deps.forEach(function(d, i){ + names[d] = i; + }); + + return Object.keys(names); + } + + shader.create = function(arg, fn){ + var option = arg || {}, + name = option.name || "default", + type = option.type || "vertex", + deps = option.require || option.deps || [], + precision = option.precision || "high", + debug = option.debug || false, + main = option.main || fn || function() {}; + + var shaderSource = 'precision ' + precision + 'p float;\n'; + + if(deps.length === 0) deps = uniqueDeps(getDeps(main)); + + //get dependence from subroutines if any + var extraDeps = [], + subRoutines = []; + + deps.forEach(function(dep){ + var res = resource.get(dep); + if(typeof res == 'undefined') { + console.log(dep); + throw Error ('Error! Undefined variable in shader: '+ dep.name); + } + if(res.resourceType == 'subroutine') { + subRoutines.push(res.name); + var subDeps = getExtraDeps(res.fn.toString()); + if(subDeps.length) { + //TODO: make this recursive to check all subroutine deps + subDeps.forEach(function(sdep){ + var sres = resource.get(sdep); + if(sres.resourceType == 'subroutine') + extraDeps = extraDeps.concat(getExtraDeps(sres.fn.toString())); + }) + + extraDeps = extraDeps.concat(subDeps); + } + } + }) + + if(extraDeps.length) { + var allDeps = extraDeps + // .filter(function(d){ + // return deps.indexOf(d) === -1; + // }) + .concat(deps.filter(function(d){ + return subRoutines.indexOf(d) === -1; + })) + .concat(subRoutines); + + deps = uniqueDeps(allDeps); + } + + + if(Array.isArray(deps)){ + deps.forEach(function(dep){ + shaderSource += declareDep(dep); + }); + } else if(typeof(deps) == 'object') { + Object.keys(deps).forEach(function(resourceType){ + deps[resourceType].forEach(function(dep){ + shaderSource += declareDep(dep); + }); + }) + } + + shaderSource += toGLSL('void', 'main', main); + + if(debug) + console.log(shaderSource); + + var _shader = compile(shaderType[type], shaderSource); + _shader._shaderType = shaderType[type]; + _shader.deps = deps; + _shader.source = shaderSource; + shader[type][name] = _shader; + return _shader; + } + + return shader; +} diff --git a/src/texture.js b/src/texture.js index dc9fc92..5c3f245 100644 --- a/src/texture.js +++ b/src/texture.js @@ -6,7 +6,8 @@ export default function Texture(glContext) { ctx = glContext, textureID = 0; - function setTexture(name, texData) { + function setTexture(name, texData) + { var type = ctx[texture[name].type.toUpperCase()], format = ctx[texture[name].channel.toUpperCase()], width = texture[name].dim[0], @@ -72,7 +73,7 @@ export default function Texture(glContext) { } texture[name].link = function(program) { - if (this.data !== null) { + // if (this.data !== null) { // ctx.activeTexture(ctx.TEXTURE0 + this.index); // ctx.bindTexture(ctx.TEXTURE_2D, this.ptr); // this.location = ctx.getUniformLocation(program, this.name); @@ -81,7 +82,9 @@ export default function Texture(glContext) { this.sampler.data = texture[name]; this.sampler.link(program); - } + // }else{ + // console.log('texture data is null!!!') + // } return this; } @@ -109,7 +112,7 @@ export default function Texture(glContext) { updateTexture(this.name, texData, offset, dim); return this; } - + texture[name].resize = function(dim, data) { this.dim = dim; setTexture(this.name, data); diff --git a/src/uniform.js b/src/uniform.js index c2803b2..9afa972 100644 --- a/src/uniform.js +++ b/src/uniform.js @@ -38,6 +38,7 @@ export default function Uniform(glContext, name, type, data) { ctx['uniformMatrix' + size + 'fv'](location, false, buf); } else if(type == 'sampler2D') { if(data.hasOwnProperty('resourceType') && data.resourceType == 'texture') { + // console.log('bind ' + data.index); ctx.activeTexture(ctx.TEXTURE0 + data.index); ctx.bindTexture(ctx.TEXTURE_2D, data.ptr); ctx.uniform1i(location, data.index);