Skip to content

Instantly share code, notes, and snippets.

@guybrush
Last active May 14, 2016 01:50
Show Gist options
  • Save guybrush/b24b0e57095992181f8fbda24ea72514 to your computer and use it in GitHub Desktop.
Save guybrush/b24b0e57095992181f8fbda24ea72514 to your computer and use it in GitHub Desktop.
requirebin sketch
const mat4 = require('gl-mat4')
const bunny = require('bunny')
const fit = require('canvas-fit')
const normals = require('angle-normals')
const canvas = document.body.appendChild(document.createElement('canvas'))
const regl = require('regl')(canvas)
const camera = require('canvas-orbit-camera')(canvas)
window.addEventListener('resize', fit(canvas), false)
var bunnyOffsets = []
var bunnyColors = []
var i = 0, n = 10
var n2 = n/2
camera.distance = n*10
for (var x=0;x<n;x++)
for (var y=0;y<n;y++)
for (var z=0;z<n;z++) {
bunnyColors [i ] = x / n
bunnyColors [i+1] = y / n
bunnyColors [i+2] = z / n
bunnyOffsets[i ] = (x-n2) * 10
bunnyOffsets[i+1] = (y-n2) * 10
bunnyOffsets[i+2] = (z-n2) * 10
i += 3
}
const bunniesInstanced = regl({
frag: `
precision mediump float;
varying vec3 vnormal;
varying vec4 vcolor;
void main () {
gl_FragColor = vec4(vcolor.xyz*abs(vnormal), 1.0);
}`,
vert: `
precision mediump float;
uniform mat4 proj;
uniform mat4 model;
uniform mat4 view;
attribute vec3 position;
attribute vec3 normal;
attribute vec3 offset;
attribute vec3 color;
varying vec4 vcolor;
varying vec3 vnormal;
void main () {
vnormal = normal;
vcolor = vec4(color,1.0);
vec4 pos = model * vec4(position, 1.0);
pos += vec4(offset, 0.0);
gl_Position = proj * view * pos;
}`,
attributes: {
position: regl.buffer(bunny.positions),
normal: regl.buffer(normals(bunny.cells, bunny.positions)),
offset: {
buffer:regl.buffer(bunnyOffsets),
size: 3,
stride: 12,
divisor:1
},
color: {
buffer:regl.buffer(bunnyColors),
size: 3,
stride: 12,
divisor:1
}
},
elements: regl.elements(bunny.cells),
uniforms: {
proj: mat4.perspective([], Math.PI / 2, window.innerWidth / window.innerHeight, 0.01, 1000),
model: mat4.identity([]),
view: regl.prop('view')
},
instances: n*n*n
})
regl.frame(function (props, count) {
regl.clear({
color: [0, 0, 0, 1]
})
camera.tick()
bunniesInstanced({ view: camera.view() })
})
This file has been truncated, but you can view the full file.
setTimeout(function(){
;require=(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
module.exports = adjoint;
/**
* Calculates the adjugate of a mat4
*
* @param {mat4} out the receiving matrix
* @param {mat4} a the source matrix
* @returns {mat4} out
*/
function adjoint(out, a) {
var a00 = a[0], a01 = a[1], a02 = a[2], a03 = a[3],
a10 = a[4], a11 = a[5], a12 = a[6], a13 = a[7],
a20 = a[8], a21 = a[9], a22 = a[10], a23 = a[11],
a30 = a[12], a31 = a[13], a32 = a[14], a33 = a[15];
out[0] = (a11 * (a22 * a33 - a23 * a32) - a21 * (a12 * a33 - a13 * a32) + a31 * (a12 * a23 - a13 * a22));
out[1] = -(a01 * (a22 * a33 - a23 * a32) - a21 * (a02 * a33 - a03 * a32) + a31 * (a02 * a23 - a03 * a22));
out[2] = (a01 * (a12 * a33 - a13 * a32) - a11 * (a02 * a33 - a03 * a32) + a31 * (a02 * a13 - a03 * a12));
out[3] = -(a01 * (a12 * a23 - a13 * a22) - a11 * (a02 * a23 - a03 * a22) + a21 * (a02 * a13 - a03 * a12));
out[4] = -(a10 * (a22 * a33 - a23 * a32) - a20 * (a12 * a33 - a13 * a32) + a30 * (a12 * a23 - a13 * a22));
out[5] = (a00 * (a22 * a33 - a23 * a32) - a20 * (a02 * a33 - a03 * a32) + a30 * (a02 * a23 - a03 * a22));
out[6] = -(a00 * (a12 * a33 - a13 * a32) - a10 * (a02 * a33 - a03 * a32) + a30 * (a02 * a13 - a03 * a12));
out[7] = (a00 * (a12 * a23 - a13 * a22) - a10 * (a02 * a23 - a03 * a22) + a20 * (a02 * a13 - a03 * a12));
out[8] = (a10 * (a21 * a33 - a23 * a31) - a20 * (a11 * a33 - a13 * a31) + a30 * (a11 * a23 - a13 * a21));
out[9] = -(a00 * (a21 * a33 - a23 * a31) - a20 * (a01 * a33 - a03 * a31) + a30 * (a01 * a23 - a03 * a21));
out[10] = (a00 * (a11 * a33 - a13 * a31) - a10 * (a01 * a33 - a03 * a31) + a30 * (a01 * a13 - a03 * a11));
out[11] = -(a00 * (a11 * a23 - a13 * a21) - a10 * (a01 * a23 - a03 * a21) + a20 * (a01 * a13 - a03 * a11));
out[12] = -(a10 * (a21 * a32 - a22 * a31) - a20 * (a11 * a32 - a12 * a31) + a30 * (a11 * a22 - a12 * a21));
out[13] = (a00 * (a21 * a32 - a22 * a31) - a20 * (a01 * a32 - a02 * a31) + a30 * (a01 * a22 - a02 * a21));
out[14] = -(a00 * (a11 * a32 - a12 * a31) - a10 * (a01 * a32 - a02 * a31) + a30 * (a01 * a12 - a02 * a11));
out[15] = (a00 * (a11 * a22 - a12 * a21) - a10 * (a01 * a22 - a02 * a21) + a20 * (a01 * a12 - a02 * a11));
return out;
};
},{}],2:[function(require,module,exports){
module.exports = clone;
/**
* Creates a new mat4 initialized with values from an existing matrix
*
* @param {mat4} a matrix to clone
* @returns {mat4} a new 4x4 matrix
*/
function clone(a) {
var out = new Float32Array(16);
out[0] = a[0];
out[1] = a[1];
out[2] = a[2];
out[3] = a[3];
out[4] = a[4];
out[5] = a[5];
out[6] = a[6];
out[7] = a[7];
out[8] = a[8];
out[9] = a[9];
out[10] = a[10];
out[11] = a[11];
out[12] = a[12];
out[13] = a[13];
out[14] = a[14];
out[15] = a[15];
return out;
};
},{}],3:[function(require,module,exports){
module.exports = copy;
/**
* Copy the values from one mat4 to another
*
* @param {mat4} out the receiving matrix
* @param {mat4} a the source matrix
* @returns {mat4} out
*/
function copy(out, a) {
out[0] = a[0];
out[1] = a[1];
out[2] = a[2];
out[3] = a[3];
out[4] = a[4];
out[5] = a[5];
out[6] = a[6];
out[7] = a[7];
out[8] = a[8];
out[9] = a[9];
out[10] = a[10];
out[11] = a[11];
out[12] = a[12];
out[13] = a[13];
out[14] = a[14];
out[15] = a[15];
return out;
};
},{}],4:[function(require,module,exports){
module.exports = create;
/**
* Creates a new identity mat4
*
* @returns {mat4} a new 4x4 matrix
*/
function create() {
var out = new Float32Array(16);
out[0] = 1;
out[1] = 0;
out[2] = 0;
out[3] = 0;
out[4] = 0;
out[5] = 1;
out[6] = 0;
out[7] = 0;
out[8] = 0;
out[9] = 0;
out[10] = 1;
out[11] = 0;
out[12] = 0;
out[13] = 0;
out[14] = 0;
out[15] = 1;
return out;
};
},{}],5:[function(require,module,exports){
module.exports = determinant;
/**
* Calculates the determinant of a mat4
*
* @param {mat4} a the source matrix
* @returns {Number} determinant of a
*/
function determinant(a) {
var a00 = a[0], a01 = a[1], a02 = a[2], a03 = a[3],
a10 = a[4], a11 = a[5], a12 = a[6], a13 = a[7],
a20 = a[8], a21 = a[9], a22 = a[10], a23 = a[11],
a30 = a[12], a31 = a[13], a32 = a[14], a33 = a[15],
b00 = a00 * a11 - a01 * a10,
b01 = a00 * a12 - a02 * a10,
b02 = a00 * a13 - a03 * a10,
b03 = a01 * a12 - a02 * a11,
b04 = a01 * a13 - a03 * a11,
b05 = a02 * a13 - a03 * a12,
b06 = a20 * a31 - a21 * a30,
b07 = a20 * a32 - a22 * a30,
b08 = a20 * a33 - a23 * a30,
b09 = a21 * a32 - a22 * a31,
b10 = a21 * a33 - a23 * a31,
b11 = a22 * a33 - a23 * a32;
// Calculate the determinant
return b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;
};
},{}],6:[function(require,module,exports){
module.exports = fromQuat;
/**
* Creates a matrix from a quaternion rotation.
*
* @param {mat4} out mat4 receiving operation result
* @param {quat4} q Rotation quaternion
* @returns {mat4} out
*/
function fromQuat(out, q) {
var x = q[0], y = q[1], z = q[2], w = q[3],
x2 = x + x,
y2 = y + y,
z2 = z + z,
xx = x * x2,
yx = y * x2,
yy = y * y2,
zx = z * x2,
zy = z * y2,
zz = z * z2,
wx = w * x2,
wy = w * y2,
wz = w * z2;
out[0] = 1 - yy - zz;
out[1] = yx + wz;
out[2] = zx - wy;
out[3] = 0;
out[4] = yx - wz;
out[5] = 1 - xx - zz;
out[6] = zy + wx;
out[7] = 0;
out[8] = zx + wy;
out[9] = zy - wx;
out[10] = 1 - xx - yy;
out[11] = 0;
out[12] = 0;
out[13] = 0;
out[14] = 0;
out[15] = 1;
return out;
};
},{}],7:[function(require,module,exports){
module.exports = fromRotationTranslation;
/**
* Creates a matrix from a quaternion rotation and vector translation
* This is equivalent to (but much faster than):
*
* mat4.identity(dest);
* mat4.translate(dest, vec);
* var quatMat = mat4.create();
* quat4.toMat4(quat, quatMat);
* mat4.multiply(dest, quatMat);
*
* @param {mat4} out mat4 receiving operation result
* @param {quat4} q Rotation quaternion
* @param {vec3} v Translation vector
* @returns {mat4} out
*/
function fromRotationTranslation(out, q, v) {
// Quaternion math
var x = q[0], y = q[1], z = q[2], w = q[3],
x2 = x + x,
y2 = y + y,
z2 = z + z,
xx = x * x2,
xy = x * y2,
xz = x * z2,
yy = y * y2,
yz = y * z2,
zz = z * z2,
wx = w * x2,
wy = w * y2,
wz = w * z2;
out[0] = 1 - (yy + zz);
out[1] = xy + wz;
out[2] = xz - wy;
out[3] = 0;
out[4] = xy - wz;
out[5] = 1 - (xx + zz);
out[6] = yz + wx;
out[7] = 0;
out[8] = xz + wy;
out[9] = yz - wx;
out[10] = 1 - (xx + yy);
out[11] = 0;
out[12] = v[0];
out[13] = v[1];
out[14] = v[2];
out[15] = 1;
return out;
};
},{}],8:[function(require,module,exports){
module.exports = frustum;
/**
* Generates a frustum matrix with the given bounds
*
* @param {mat4} out mat4 frustum matrix will be written into
* @param {Number} left Left bound of the frustum
* @param {Number} right Right bound of the frustum
* @param {Number} bottom Bottom bound of the frustum
* @param {Number} top Top bound of the frustum
* @param {Number} near Near bound of the frustum
* @param {Number} far Far bound of the frustum
* @returns {mat4} out
*/
function frustum(out, left, right, bottom, top, near, far) {
var rl = 1 / (right - left),
tb = 1 / (top - bottom),
nf = 1 / (near - far);
out[0] = (near * 2) * rl;
out[1] = 0;
out[2] = 0;
out[3] = 0;
out[4] = 0;
out[5] = (near * 2) * tb;
out[6] = 0;
out[7] = 0;
out[8] = (right + left) * rl;
out[9] = (top + bottom) * tb;
out[10] = (far + near) * nf;
out[11] = -1;
out[12] = 0;
out[13] = 0;
out[14] = (far * near * 2) * nf;
out[15] = 0;
return out;
};
},{}],9:[function(require,module,exports){
module.exports = identity;
/**
* Set a mat4 to the identity matrix
*
* @param {mat4} out the receiving matrix
* @returns {mat4} out
*/
function identity(out) {
out[0] = 1;
out[1] = 0;
out[2] = 0;
out[3] = 0;
out[4] = 0;
out[5] = 1;
out[6] = 0;
out[7] = 0;
out[8] = 0;
out[9] = 0;
out[10] = 1;
out[11] = 0;
out[12] = 0;
out[13] = 0;
out[14] = 0;
out[15] = 1;
return out;
};
},{}],10:[function(require,module,exports){
module.exports = invert;
/**
* Inverts a mat4
*
* @param {mat4} out the receiving matrix
* @param {mat4} a the source matrix
* @returns {mat4} out
*/
function invert(out, a) {
var a00 = a[0], a01 = a[1], a02 = a[2], a03 = a[3],
a10 = a[4], a11 = a[5], a12 = a[6], a13 = a[7],
a20 = a[8], a21 = a[9], a22 = a[10], a23 = a[11],
a30 = a[12], a31 = a[13], a32 = a[14], a33 = a[15],
b00 = a00 * a11 - a01 * a10,
b01 = a00 * a12 - a02 * a10,
b02 = a00 * a13 - a03 * a10,
b03 = a01 * a12 - a02 * a11,
b04 = a01 * a13 - a03 * a11,
b05 = a02 * a13 - a03 * a12,
b06 = a20 * a31 - a21 * a30,
b07 = a20 * a32 - a22 * a30,
b08 = a20 * a33 - a23 * a30,
b09 = a21 * a32 - a22 * a31,
b10 = a21 * a33 - a23 * a31,
b11 = a22 * a33 - a23 * a32,
// Calculate the determinant
det = b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;
if (!det) {
return null;
}
det = 1.0 / det;
out[0] = (a11 * b11 - a12 * b10 + a13 * b09) * det;
out[1] = (a02 * b10 - a01 * b11 - a03 * b09) * det;
out[2] = (a31 * b05 - a32 * b04 + a33 * b03) * det;
out[3] = (a22 * b04 - a21 * b05 - a23 * b03) * det;
out[4] = (a12 * b08 - a10 * b11 - a13 * b07) * det;
out[5] = (a00 * b11 - a02 * b08 + a03 * b07) * det;
out[6] = (a32 * b02 - a30 * b05 - a33 * b01) * det;
out[7] = (a20 * b05 - a22 * b02 + a23 * b01) * det;
out[8] = (a10 * b10 - a11 * b08 + a13 * b06) * det;
out[9] = (a01 * b08 - a00 * b10 - a03 * b06) * det;
out[10] = (a30 * b04 - a31 * b02 + a33 * b00) * det;
out[11] = (a21 * b02 - a20 * b04 - a23 * b00) * det;
out[12] = (a11 * b07 - a10 * b09 - a12 * b06) * det;
out[13] = (a00 * b09 - a01 * b07 + a02 * b06) * det;
out[14] = (a31 * b01 - a30 * b03 - a32 * b00) * det;
out[15] = (a20 * b03 - a21 * b01 + a22 * b00) * det;
return out;
};
},{}],11:[function(require,module,exports){
var identity = require('./identity');
module.exports = lookAt;
/**
* Generates a look-at matrix with the given eye position, focal point, and up axis
*
* @param {mat4} out mat4 frustum matrix will be written into
* @param {vec3} eye Position of the viewer
* @param {vec3} center Point the viewer is looking at
* @param {vec3} up vec3 pointing up
* @returns {mat4} out
*/
function lookAt(out, eye, center, up) {
var x0, x1, x2, y0, y1, y2, z0, z1, z2, len,
eyex = eye[0],
eyey = eye[1],
eyez = eye[2],
upx = up[0],
upy = up[1],
upz = up[2],
centerx = center[0],
centery = center[1],
centerz = center[2];
if (Math.abs(eyex - centerx) < 0.000001 &&
Math.abs(eyey - centery) < 0.000001 &&
Math.abs(eyez - centerz) < 0.000001) {
return identity(out);
}
z0 = eyex - centerx;
z1 = eyey - centery;
z2 = eyez - centerz;
len = 1 / Math.sqrt(z0 * z0 + z1 * z1 + z2 * z2);
z0 *= len;
z1 *= len;
z2 *= len;
x0 = upy * z2 - upz * z1;
x1 = upz * z0 - upx * z2;
x2 = upx * z1 - upy * z0;
len = Math.sqrt(x0 * x0 + x1 * x1 + x2 * x2);
if (!len) {
x0 = 0;
x1 = 0;
x2 = 0;
} else {
len = 1 / len;
x0 *= len;
x1 *= len;
x2 *= len;
}
y0 = z1 * x2 - z2 * x1;
y1 = z2 * x0 - z0 * x2;
y2 = z0 * x1 - z1 * x0;
len = Math.sqrt(y0 * y0 + y1 * y1 + y2 * y2);
if (!len) {
y0 = 0;
y1 = 0;
y2 = 0;
} else {
len = 1 / len;
y0 *= len;
y1 *= len;
y2 *= len;
}
out[0] = x0;
out[1] = y0;
out[2] = z0;
out[3] = 0;
out[4] = x1;
out[5] = y1;
out[6] = z1;
out[7] = 0;
out[8] = x2;
out[9] = y2;
out[10] = z2;
out[11] = 0;
out[12] = -(x0 * eyex + x1 * eyey + x2 * eyez);
out[13] = -(y0 * eyex + y1 * eyey + y2 * eyez);
out[14] = -(z0 * eyex + z1 * eyey + z2 * eyez);
out[15] = 1;
return out;
};
},{"./identity":9}],12:[function(require,module,exports){
module.exports = multiply;
/**
* Multiplies two mat4's
*
* @param {mat4} out the receiving matrix
* @param {mat4} a the first operand
* @param {mat4} b the second operand
* @returns {mat4} out
*/
function multiply(out, a, b) {
var a00 = a[0], a01 = a[1], a02 = a[2], a03 = a[3],
a10 = a[4], a11 = a[5], a12 = a[6], a13 = a[7],
a20 = a[8], a21 = a[9], a22 = a[10], a23 = a[11],
a30 = a[12], a31 = a[13], a32 = a[14], a33 = a[15];
// Cache only the current line of the second matrix
var b0 = b[0], b1 = b[1], b2 = b[2], b3 = b[3];
out[0] = b0*a00 + b1*a10 + b2*a20 + b3*a30;
out[1] = b0*a01 + b1*a11 + b2*a21 + b3*a31;
out[2] = b0*a02 + b1*a12 + b2*a22 + b3*a32;
out[3] = b0*a03 + b1*a13 + b2*a23 + b3*a33;
b0 = b[4]; b1 = b[5]; b2 = b[6]; b3 = b[7];
out[4] = b0*a00 + b1*a10 + b2*a20 + b3*a30;
out[5] = b0*a01 + b1*a11 + b2*a21 + b3*a31;
out[6] = b0*a02 + b1*a12 + b2*a22 + b3*a32;
out[7] = b0*a03 + b1*a13 + b2*a23 + b3*a33;
b0 = b[8]; b1 = b[9]; b2 = b[10]; b3 = b[11];
out[8] = b0*a00 + b1*a10 + b2*a20 + b3*a30;
out[9] = b0*a01 + b1*a11 + b2*a21 + b3*a31;
out[10] = b0*a02 + b1*a12 + b2*a22 + b3*a32;
out[11] = b0*a03 + b1*a13 + b2*a23 + b3*a33;
b0 = b[12]; b1 = b[13]; b2 = b[14]; b3 = b[15];
out[12] = b0*a00 + b1*a10 + b2*a20 + b3*a30;
out[13] = b0*a01 + b1*a11 + b2*a21 + b3*a31;
out[14] = b0*a02 + b1*a12 + b2*a22 + b3*a32;
out[15] = b0*a03 + b1*a13 + b2*a23 + b3*a33;
return out;
};
},{}],13:[function(require,module,exports){
module.exports = ortho;
/**
* Generates a orthogonal projection matrix with the given bounds
*
* @param {mat4} out mat4 frustum matrix will be written into
* @param {number} left Left bound of the frustum
* @param {number} right Right bound of the frustum
* @param {number} bottom Bottom bound of the frustum
* @param {number} top Top bound of the frustum
* @param {number} near Near bound of the frustum
* @param {number} far Far bound of the frustum
* @returns {mat4} out
*/
function ortho(out, left, right, bottom, top, near, far) {
var lr = 1 / (left - right),
bt = 1 / (bottom - top),
nf = 1 / (near - far);
out[0] = -2 * lr;
out[1] = 0;
out[2] = 0;
out[3] = 0;
out[4] = 0;
out[5] = -2 * bt;
out[6] = 0;
out[7] = 0;
out[8] = 0;
out[9] = 0;
out[10] = 2 * nf;
out[11] = 0;
out[12] = (left + right) * lr;
out[13] = (top + bottom) * bt;
out[14] = (far + near) * nf;
out[15] = 1;
return out;
};
},{}],14:[function(require,module,exports){
module.exports = perspective;
/**
* Generates a perspective projection matrix with the given bounds
*
* @param {mat4} out mat4 frustum matrix will be written into
* @param {number} fovy Vertical field of view in radians
* @param {number} aspect Aspect ratio. typically viewport width/height
* @param {number} near Near bound of the frustum
* @param {number} far Far bound of the frustum
* @returns {mat4} out
*/
function perspective(out, fovy, aspect, near, far) {
var f = 1.0 / Math.tan(fovy / 2),
nf = 1 / (near - far);
out[0] = f / aspect;
out[1] = 0;
out[2] = 0;
out[3] = 0;
out[4] = 0;
out[5] = f;
out[6] = 0;
out[7] = 0;
out[8] = 0;
out[9] = 0;
out[10] = (far + near) * nf;
out[11] = -1;
out[12] = 0;
out[13] = 0;
out[14] = (2 * far * near) * nf;
out[15] = 0;
return out;
};
},{}],15:[function(require,module,exports){
module.exports = perspectiveFromFieldOfView;
/**
* Generates a perspective projection matrix with the given field of view.
* This is primarily useful for generating projection matrices to be used
* with the still experiemental WebVR API.
*
* @param {mat4} out mat4 frustum matrix will be written into
* @param {number} fov Object containing the following values: upDegrees, downDegrees, leftDegrees, rightDegrees
* @param {number} near Near bound of the frustum
* @param {number} far Far bound of the frustum
* @returns {mat4} out
*/
function perspectiveFromFieldOfView(out, fov, near, far) {
var upTan = Math.tan(fov.upDegrees * Math.PI/180.0),
downTan = Math.tan(fov.downDegrees * Math.PI/180.0),
leftTan = Math.tan(fov.leftDegrees * Math.PI/180.0),
rightTan = Math.tan(fov.rightDegrees * Math.PI/180.0),
xScale = 2.0 / (leftTan + rightTan),
yScale = 2.0 / (upTan + downTan);
out[0] = xScale;
out[1] = 0.0;
out[2] = 0.0;
out[3] = 0.0;
out[4] = 0.0;
out[5] = yScale;
out[6] = 0.0;
out[7] = 0.0;
out[8] = -((leftTan - rightTan) * xScale * 0.5);
out[9] = ((upTan - downTan) * yScale * 0.5);
out[10] = far / (near - far);
out[11] = -1.0;
out[12] = 0.0;
out[13] = 0.0;
out[14] = (far * near) / (near - far);
out[15] = 0.0;
return out;
}
},{}],16:[function(require,module,exports){
module.exports = rotate;
/**
* Rotates a mat4 by the given angle
*
* @param {mat4} out the receiving matrix
* @param {mat4} a the matrix to rotate
* @param {Number} rad the angle to rotate the matrix by
* @param {vec3} axis the axis to rotate around
* @returns {mat4} out
*/
function rotate(out, a, rad, axis) {
var x = axis[0], y = axis[1], z = axis[2],
len = Math.sqrt(x * x + y * y + z * z),
s, c, t,
a00, a01, a02, a03,
a10, a11, a12, a13,
a20, a21, a22, a23,
b00, b01, b02,
b10, b11, b12,
b20, b21, b22;
if (Math.abs(len) < 0.000001) { return null; }
len = 1 / len;
x *= len;
y *= len;
z *= len;
s = Math.sin(rad);
c = Math.cos(rad);
t = 1 - c;
a00 = a[0]; a01 = a[1]; a02 = a[2]; a03 = a[3];
a10 = a[4]; a11 = a[5]; a12 = a[6]; a13 = a[7];
a20 = a[8]; a21 = a[9]; a22 = a[10]; a23 = a[11];
// Construct the elements of the rotation matrix
b00 = x * x * t + c; b01 = y * x * t + z * s; b02 = z * x * t - y * s;
b10 = x * y * t - z * s; b11 = y * y * t + c; b12 = z * y * t + x * s;
b20 = x * z * t + y * s; b21 = y * z * t - x * s; b22 = z * z * t + c;
// Perform rotation-specific matrix multiplication
out[0] = a00 * b00 + a10 * b01 + a20 * b02;
out[1] = a01 * b00 + a11 * b01 + a21 * b02;
out[2] = a02 * b00 + a12 * b01 + a22 * b02;
out[3] = a03 * b00 + a13 * b01 + a23 * b02;
out[4] = a00 * b10 + a10 * b11 + a20 * b12;
out[5] = a01 * b10 + a11 * b11 + a21 * b12;
out[6] = a02 * b10 + a12 * b11 + a22 * b12;
out[7] = a03 * b10 + a13 * b11 + a23 * b12;
out[8] = a00 * b20 + a10 * b21 + a20 * b22;
out[9] = a01 * b20 + a11 * b21 + a21 * b22;
out[10] = a02 * b20 + a12 * b21 + a22 * b22;
out[11] = a03 * b20 + a13 * b21 + a23 * b22;
if (a !== out) { // If the source and destination differ, copy the unchanged last row
out[12] = a[12];
out[13] = a[13];
out[14] = a[14];
out[15] = a[15];
}
return out;
};
},{}],17:[function(require,module,exports){
module.exports = rotateX;
/**
* Rotates a matrix by the given angle around the X axis
*
* @param {mat4} out the receiving matrix
* @param {mat4} a the matrix to rotate
* @param {Number} rad the angle to rotate the matrix by
* @returns {mat4} out
*/
function rotateX(out, a, rad) {
var s = Math.sin(rad),
c = Math.cos(rad),
a10 = a[4],
a11 = a[5],
a12 = a[6],
a13 = a[7],
a20 = a[8],
a21 = a[9],
a22 = a[10],
a23 = a[11];
if (a !== out) { // If the source and destination differ, copy the unchanged rows
out[0] = a[0];
out[1] = a[1];
out[2] = a[2];
out[3] = a[3];
out[12] = a[12];
out[13] = a[13];
out[14] = a[14];
out[15] = a[15];
}
// Perform axis-specific matrix multiplication
out[4] = a10 * c + a20 * s;
out[5] = a11 * c + a21 * s;
out[6] = a12 * c + a22 * s;
out[7] = a13 * c + a23 * s;
out[8] = a20 * c - a10 * s;
out[9] = a21 * c - a11 * s;
out[10] = a22 * c - a12 * s;
out[11] = a23 * c - a13 * s;
return out;
};
},{}],18:[function(require,module,exports){
module.exports = rotateY;
/**
* Rotates a matrix by the given angle around the Y axis
*
* @param {mat4} out the receiving matrix
* @param {mat4} a the matrix to rotate
* @param {Number} rad the angle to rotate the matrix by
* @returns {mat4} out
*/
function rotateY(out, a, rad) {
var s = Math.sin(rad),
c = Math.cos(rad),
a00 = a[0],
a01 = a[1],
a02 = a[2],
a03 = a[3],
a20 = a[8],
a21 = a[9],
a22 = a[10],
a23 = a[11];
if (a !== out) { // If the source and destination differ, copy the unchanged rows
out[4] = a[4];
out[5] = a[5];
out[6] = a[6];
out[7] = a[7];
out[12] = a[12];
out[13] = a[13];
out[14] = a[14];
out[15] = a[15];
}
// Perform axis-specific matrix multiplication
out[0] = a00 * c - a20 * s;
out[1] = a01 * c - a21 * s;
out[2] = a02 * c - a22 * s;
out[3] = a03 * c - a23 * s;
out[8] = a00 * s + a20 * c;
out[9] = a01 * s + a21 * c;
out[10] = a02 * s + a22 * c;
out[11] = a03 * s + a23 * c;
return out;
};
},{}],19:[function(require,module,exports){
module.exports = rotateZ;
/**
* Rotates a matrix by the given angle around the Z axis
*
* @param {mat4} out the receiving matrix
* @param {mat4} a the matrix to rotate
* @param {Number} rad the angle to rotate the matrix by
* @returns {mat4} out
*/
function rotateZ(out, a, rad) {
var s = Math.sin(rad),
c = Math.cos(rad),
a00 = a[0],
a01 = a[1],
a02 = a[2],
a03 = a[3],
a10 = a[4],
a11 = a[5],
a12 = a[6],
a13 = a[7];
if (a !== out) { // If the source and destination differ, copy the unchanged last row
out[8] = a[8];
out[9] = a[9];
out[10] = a[10];
out[11] = a[11];
out[12] = a[12];
out[13] = a[13];
out[14] = a[14];
out[15] = a[15];
}
// Perform axis-specific matrix multiplication
out[0] = a00 * c + a10 * s;
out[1] = a01 * c + a11 * s;
out[2] = a02 * c + a12 * s;
out[3] = a03 * c + a13 * s;
out[4] = a10 * c - a00 * s;
out[5] = a11 * c - a01 * s;
out[6] = a12 * c - a02 * s;
out[7] = a13 * c - a03 * s;
return out;
};
},{}],20:[function(require,module,exports){
module.exports = scale;
/**
* Scales the mat4 by the dimensions in the given vec3
*
* @param {mat4} out the receiving matrix
* @param {mat4} a the matrix to scale
* @param {vec3} v the vec3 to scale the matrix by
* @returns {mat4} out
**/
function scale(out, a, v) {
var x = v[0], y = v[1], z = v[2];
out[0] = a[0] * x;
out[1] = a[1] * x;
out[2] = a[2] * x;
out[3] = a[3] * x;
out[4] = a[4] * y;
out[5] = a[5] * y;
out[6] = a[6] * y;
out[7] = a[7] * y;
out[8] = a[8] * z;
out[9] = a[9] * z;
out[10] = a[10] * z;
out[11] = a[11] * z;
out[12] = a[12];
out[13] = a[13];
out[14] = a[14];
out[15] = a[15];
return out;
};
},{}],21:[function(require,module,exports){
module.exports = str;
/**
* Returns a string representation of a mat4
*
* @param {mat4} mat matrix to represent as a string
* @returns {String} string representation of the matrix
*/
function str(a) {
return 'mat4(' + a[0] + ', ' + a[1] + ', ' + a[2] + ', ' + a[3] + ', ' +
a[4] + ', ' + a[5] + ', ' + a[6] + ', ' + a[7] + ', ' +
a[8] + ', ' + a[9] + ', ' + a[10] + ', ' + a[11] + ', ' +
a[12] + ', ' + a[13] + ', ' + a[14] + ', ' + a[15] + ')';
};
},{}],22:[function(require,module,exports){
module.exports = translate;
/**
* Translate a mat4 by the given vector
*
* @param {mat4} out the receiving matrix
* @param {mat4} a the matrix to translate
* @param {vec3} v vector to translate by
* @returns {mat4} out
*/
function translate(out, a, v) {
var x = v[0], y = v[1], z = v[2],
a00, a01, a02, a03,
a10, a11, a12, a13,
a20, a21, a22, a23;
if (a === out) {
out[12] = a[0] * x + a[4] * y + a[8] * z + a[12];
out[13] = a[1] * x + a[5] * y + a[9] * z + a[13];
out[14] = a[2] * x + a[6] * y + a[10] * z + a[14];
out[15] = a[3] * x + a[7] * y + a[11] * z + a[15];
} else {
a00 = a[0]; a01 = a[1]; a02 = a[2]; a03 = a[3];
a10 = a[4]; a11 = a[5]; a12 = a[6]; a13 = a[7];
a20 = a[8]; a21 = a[9]; a22 = a[10]; a23 = a[11];
out[0] = a00; out[1] = a01; out[2] = a02; out[3] = a03;
out[4] = a10; out[5] = a11; out[6] = a12; out[7] = a13;
out[8] = a20; out[9] = a21; out[10] = a22; out[11] = a23;
out[12] = a00 * x + a10 * y + a20 * z + a[12];
out[13] = a01 * x + a11 * y + a21 * z + a[13];
out[14] = a02 * x + a12 * y + a22 * z + a[14];
out[15] = a03 * x + a13 * y + a23 * z + a[15];
}
return out;
};
},{}],23:[function(require,module,exports){
module.exports = transpose;
/**
* Transpose the values of a mat4
*
* @param {mat4} out the receiving matrix
* @param {mat4} a the source matrix
* @returns {mat4} out
*/
function transpose(out, a) {
// If we are transposing ourselves we can skip a few steps but have to cache some values
if (out === a) {
var a01 = a[1], a02 = a[2], a03 = a[3],
a12 = a[6], a13 = a[7],
a23 = a[11];
out[1] = a[4];
out[2] = a[8];
out[3] = a[12];
out[4] = a01;
out[6] = a[9];
out[7] = a[13];
out[8] = a02;
out[9] = a12;
out[11] = a[14];
out[12] = a03;
out[13] = a13;
out[14] = a23;
} else {
out[0] = a[0];
out[1] = a[4];
out[2] = a[8];
out[3] = a[12];
out[4] = a[1];
out[5] = a[5];
out[6] = a[9];
out[7] = a[13];
out[8] = a[2];
out[9] = a[6];
out[10] = a[10];
out[11] = a[14];
out[12] = a[3];
out[13] = a[7];
out[14] = a[11];
out[15] = a[15];
}
return out;
};
},{}],"gl-mat4":[function(require,module,exports){
module.exports = {
create: require('./create')
, clone: require('./clone')
, copy: require('./copy')
, identity: require('./identity')
, transpose: require('./transpose')
, invert: require('./invert')
, adjoint: require('./adjoint')
, determinant: require('./determinant')
, multiply: require('./multiply')
, translate: require('./translate')
, scale: require('./scale')
, rotate: require('./rotate')
, rotateX: require('./rotateX')
, rotateY: require('./rotateY')
, rotateZ: require('./rotateZ')
, fromRotationTranslation: require('./fromRotationTranslation')
, fromQuat: require('./fromQuat')
, frustum: require('./frustum')
, perspective: require('./perspective')
, perspectiveFromFieldOfView: require('./perspectiveFromFieldOfView')
, ortho: require('./ortho')
, lookAt: require('./lookAt')
, str: require('./str')
}
},{"./adjoint":1,"./clone":2,"./copy":3,"./create":4,"./determinant":5,"./fromQuat":6,"./fromRotationTranslation":7,"./frustum":8,"./identity":9,"./invert":10,"./lookAt":11,"./multiply":12,"./ortho":13,"./perspective":14,"./perspectiveFromFieldOfView":15,"./rotate":16,"./rotateX":17,"./rotateY":18,"./rotateZ":19,"./scale":20,"./str":21,"./translate":22,"./transpose":23}]},{},[])
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../../home/admin/browserify-cdn/node_modules/browserify/node_modules/browser-pack/_prelude.js","adjoint.js","clone.js","copy.js","create.js","determinant.js","fromQuat.js","fromRotationTranslation.js","frustum.js","identity.js","invert.js","lookAt.js","multiply.js","ortho.js","perspective.js","perspectiveFromFieldOfView.js","rotate.js","rotateX.js","rotateY.js","rotateZ.js","scale.js","str.js","translate.js","transpose.js","index.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","module.exports = adjoint;\n\n/**\n * Calculates the adjugate of a mat4\n *\n * @param {mat4} out the receiving matrix\n * @param {mat4} a the source matrix\n * @returns {mat4} out\n */\nfunction adjoint(out, a) {\n    var a00 = a[0], a01 = a[1], a02 = a[2], a03 = a[3],\n        a10 = a[4], a11 = a[5], a12 = a[6], a13 = a[7],\n        a20 = a[8], a21 = a[9], a22 = a[10], a23 = a[11],\n        a30 = a[12], a31 = a[13], a32 = a[14], a33 = a[15];\n\n    out[0]  =  (a11 * (a22 * a33 - a23 * a32) - a21 * (a12 * a33 - a13 * a32) + a31 * (a12 * a23 - a13 * a22));\n    out[1]  = -(a01 * (a22 * a33 - a23 * a32) - a21 * (a02 * a33 - a03 * a32) + a31 * (a02 * a23 - a03 * a22));\n    out[2]  =  (a01 * (a12 * a33 - a13 * a32) - a11 * (a02 * a33 - a03 * a32) + a31 * (a02 * a13 - a03 * a12));\n    out[3]  = -(a01 * (a12 * a23 - a13 * a22) - a11 * (a02 * a23 - a03 * a22) + a21 * (a02 * a13 - a03 * a12));\n    out[4]  = -(a10 * (a22 * a33 - a23 * a32) - a20 * (a12 * a33 - a13 * a32) + a30 * (a12 * a23 - a13 * a22));\n    out[5]  =  (a00 * (a22 * a33 - a23 * a32) - a20 * (a02 * a33 - a03 * a32) + a30 * (a02 * a23 - a03 * a22));\n    out[6]  = -(a00 * (a12 * a33 - a13 * a32) - a10 * (a02 * a33 - a03 * a32) + a30 * (a02 * a13 - a03 * a12));\n    out[7]  =  (a00 * (a12 * a23 - a13 * a22) - a10 * (a02 * a23 - a03 * a22) + a20 * (a02 * a13 - a03 * a12));\n    out[8]  =  (a10 * (a21 * a33 - a23 * a31) - a20 * (a11 * a33 - a13 * a31) + a30 * (a11 * a23 - a13 * a21));\n    out[9]  = -(a00 * (a21 * a33 - a23 * a31) - a20 * (a01 * a33 - a03 * a31) + a30 * (a01 * a23 - a03 * a21));\n    out[10] =  (a00 * (a11 * a33 - a13 * a31) - a10 * (a01 * a33 - a03 * a31) + a30 * (a01 * a13 - a03 * a11));\n    out[11] = -(a00 * (a11 * a23 - a13 * a21) - a10 * (a01 * a23 - a03 * a21) + a20 * (a01 * a13 - a03 * a11));\n    out[12] = -(a10 * (a21 * a32 - a22 * a31) - a20 * (a11 * a32 - a12 * a31) + a30 * (a11 * a22 - a12 * a21));\n    out[13] =  (a00 * (a21 * a32 - a22 * a31) - a20 * (a01 * a32 - a02 * a31) + a30 * (a01 * a22 - a02 * a21));\n    out[14] = -(a00 * (a11 * a32 - a12 * a31) - a10 * (a01 * a32 - a02 * a31) + a30 * (a01 * a12 - a02 * a11));\n    out[15] =  (a00 * (a11 * a22 - a12 * a21) - a10 * (a01 * a22 - a02 * a21) + a20 * (a01 * a12 - a02 * a11));\n    return out;\n};","module.exports = clone;\n\n/**\n * Creates a new mat4 initialized with values from an existing matrix\n *\n * @param {mat4} a matrix to clone\n * @returns {mat4} a new 4x4 matrix\n */\nfunction clone(a) {\n    var out = new Float32Array(16);\n    out[0] = a[0];\n    out[1] = a[1];\n    out[2] = a[2];\n    out[3] = a[3];\n    out[4] = a[4];\n    out[5] = a[5];\n    out[6] = a[6];\n    out[7] = a[7];\n    out[8] = a[8];\n    out[9] = a[9];\n    out[10] = a[10];\n    out[11] = a[11];\n    out[12] = a[12];\n    out[13] = a[13];\n    out[14] = a[14];\n    out[15] = a[15];\n    return out;\n};","module.exports = copy;\n\n/**\n * Copy the values from one mat4 to another\n *\n * @param {mat4} out the receiving matrix\n * @param {mat4} a the source matrix\n * @returns {mat4} out\n */\nfunction copy(out, a) {\n    out[0] = a[0];\n    out[1] = a[1];\n    out[2] = a[2];\n    out[3] = a[3];\n    out[4] = a[4];\n    out[5] = a[5];\n    out[6] = a[6];\n    out[7] = a[7];\n    out[8] = a[8];\n    out[9] = a[9];\n    out[10] = a[10];\n    out[11] = a[11];\n    out[12] = a[12];\n    out[13] = a[13];\n    out[14] = a[14];\n    out[15] = a[15];\n    return out;\n};","module.exports = create;\n\n/**\n * Creates a new identity mat4\n *\n * @returns {mat4} a new 4x4 matrix\n */\nfunction create() {\n    var out = new Float32Array(16);\n    out[0] = 1;\n    out[1] = 0;\n    out[2] = 0;\n    out[3] = 0;\n    out[4] = 0;\n    out[5] = 1;\n    out[6] = 0;\n    out[7] = 0;\n    out[8] = 0;\n    out[9] = 0;\n    out[10] = 1;\n    out[11] = 0;\n    out[12] = 0;\n    out[13] = 0;\n    out[14] = 0;\n    out[15] = 1;\n    return out;\n};","module.exports = determinant;\n\n/**\n * Calculates the determinant of a mat4\n *\n * @param {mat4} a the source matrix\n * @returns {Number} determinant of a\n */\nfunction determinant(a) {\n    var a00 = a[0], a01 = a[1], a02 = a[2], a03 = a[3],\n        a10 = a[4], a11 = a[5], a12 = a[6], a13 = a[7],\n        a20 = a[8], a21 = a[9], a22 = a[10], a23 = a[11],\n        a30 = a[12], a31 = a[13], a32 = a[14], a33 = a[15],\n\n        b00 = a00 * a11 - a01 * a10,\n        b01 = a00 * a12 - a02 * a10,\n        b02 = a00 * a13 - a03 * a10,\n        b03 = a01 * a12 - a02 * a11,\n        b04 = a01 * a13 - a03 * a11,\n        b05 = a02 * a13 - a03 * a12,\n        b06 = a20 * a31 - a21 * a30,\n        b07 = a20 * a32 - a22 * a30,\n        b08 = a20 * a33 - a23 * a30,\n        b09 = a21 * a32 - a22 * a31,\n        b10 = a21 * a33 - a23 * a31,\n        b11 = a22 * a33 - a23 * a32;\n\n    // Calculate the determinant\n    return b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;\n};","module.exports = fromQuat;\n\n/**\n * Creates a matrix from a quaternion rotation.\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {quat4} q Rotation quaternion\n * @returns {mat4} out\n */\nfunction fromQuat(out, q) {\n    var x = q[0], y = q[1], z = q[2], w = q[3],\n        x2 = x + x,\n        y2 = y + y,\n        z2 = z + z,\n\n        xx = x * x2,\n        yx = y * x2,\n        yy = y * y2,\n        zx = z * x2,\n        zy = z * y2,\n        zz = z * z2,\n        wx = w * x2,\n        wy = w * y2,\n        wz = w * z2;\n\n    out[0] = 1 - yy - zz;\n    out[1] = yx + wz;\n    out[2] = zx - wy;\n    out[3] = 0;\n\n    out[4] = yx - wz;\n    out[5] = 1 - xx - zz;\n    out[6] = zy + wx;\n    out[7] = 0;\n\n    out[8] = zx + wy;\n    out[9] = zy - wx;\n    out[10] = 1 - xx - yy;\n    out[11] = 0;\n\n    out[12] = 0;\n    out[13] = 0;\n    out[14] = 0;\n    out[15] = 1;\n\n    return out;\n};","module.exports = fromRotationTranslation;\n\n/**\n * Creates a matrix from a quaternion rotation and vector translation\n * This is equivalent to (but much faster than):\n *\n *     mat4.identity(dest);\n *     mat4.translate(dest, vec);\n *     var quatMat = mat4.create();\n *     quat4.toMat4(quat, quatMat);\n *     mat4.multiply(dest, quatMat);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {quat4} q Rotation quaternion\n * @param {vec3} v Translation vector\n * @returns {mat4} out\n */\nfunction fromRotationTranslation(out, q, v) {\n    // Quaternion math\n    var x = q[0], y = q[1], z = q[2], w = q[3],\n        x2 = x + x,\n        y2 = y + y,\n        z2 = z + z,\n\n        xx = x * x2,\n        xy = x * y2,\n        xz = x * z2,\n        yy = y * y2,\n        yz = y * z2,\n        zz = z * z2,\n        wx = w * x2,\n        wy = w * y2,\n        wz = w * z2;\n\n    out[0] = 1 - (yy + zz);\n    out[1] = xy + wz;\n    out[2] = xz - wy;\n    out[3] = 0;\n    out[4] = xy - wz;\n    out[5] = 1 - (xx + zz);\n    out[6] = yz + wx;\n    out[7] = 0;\n    out[8] = xz + wy;\n    out[9] = yz - wx;\n    out[10] = 1 - (xx + yy);\n    out[11] = 0;\n    out[12] = v[0];\n    out[13] = v[1];\n    out[14] = v[2];\n    out[15] = 1;\n    \n    return out;\n};","module.exports = frustum;\n\n/**\n * Generates a frustum matrix with the given bounds\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {Number} left Left bound of the frustum\n * @param {Number} right Right bound of the frustum\n * @param {Number} bottom Bottom bound of the frustum\n * @param {Number} top Top bound of the frustum\n * @param {Number} near Near bound of the frustum\n * @param {Number} far Far bound of the frustum\n * @returns {mat4} out\n */\nfunction frustum(out, left, right, bottom, top, near, far) {\n    var rl = 1 / (right - left),\n        tb = 1 / (top - bottom),\n        nf = 1 / (near - far);\n    out[0] = (near * 2) * rl;\n    out[1] = 0;\n    out[2] = 0;\n    out[3] = 0;\n    out[4] = 0;\n    out[5] = (near * 2) * tb;\n    out[6] = 0;\n    out[7] = 0;\n    out[8] = (right + left) * rl;\n    out[9] = (top + bottom) * tb;\n    out[10] = (far + near) * nf;\n    out[11] = -1;\n    out[12] = 0;\n    out[13] = 0;\n    out[14] = (far * near * 2) * nf;\n    out[15] = 0;\n    return out;\n};","module.exports = identity;\n\n/**\n * Set a mat4 to the identity matrix\n *\n * @param {mat4} out the receiving matrix\n * @returns {mat4} out\n */\nfunction identity(out) {\n    out[0] = 1;\n    out[1] = 0;\n    out[2] = 0;\n    out[3] = 0;\n    out[4] = 0;\n    out[5] = 1;\n    out[6] = 0;\n    out[7] = 0;\n    out[8] = 0;\n    out[9] = 0;\n    out[10] = 1;\n    out[11] = 0;\n    out[12] = 0;\n    out[13] = 0;\n    out[14] = 0;\n    out[15] = 1;\n    return out;\n};","module.exports = invert;\n\n/**\n * Inverts a mat4\n *\n * @param {mat4} out the receiving matrix\n * @param {mat4} a the source matrix\n * @returns {mat4} out\n */\nfunction invert(out, a) {\n    var a00 = a[0], a01 = a[1], a02 = a[2], a03 = a[3],\n        a10 = a[4], a11 = a[5], a12 = a[6], a13 = a[7],\n        a20 = a[8], a21 = a[9], a22 = a[10], a23 = a[11],\n        a30 = a[12], a31 = a[13], a32 = a[14], a33 = a[15],\n\n        b00 = a00 * a11 - a01 * a10,\n        b01 = a00 * a12 - a02 * a10,\n        b02 = a00 * a13 - a03 * a10,\n        b03 = a01 * a12 - a02 * a11,\n        b04 = a01 * a13 - a03 * a11,\n        b05 = a02 * a13 - a03 * a12,\n        b06 = a20 * a31 - a21 * a30,\n        b07 = a20 * a32 - a22 * a30,\n        b08 = a20 * a33 - a23 * a30,\n        b09 = a21 * a32 - a22 * a31,\n        b10 = a21 * a33 - a23 * a31,\n        b11 = a22 * a33 - a23 * a32,\n\n        // Calculate the determinant\n        det = b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;\n\n    if (!det) { \n        return null; \n    }\n    det = 1.0 / det;\n\n    out[0] = (a11 * b11 - a12 * b10 + a13 * b09) * det;\n    out[1] = (a02 * b10 - a01 * b11 - a03 * b09) * det;\n    out[2] = (a31 * b05 - a32 * b04 + a33 * b03) * det;\n    out[3] = (a22 * b04 - a21 * b05 - a23 * b03) * det;\n    out[4] = (a12 * b08 - a10 * b11 - a13 * b07) * det;\n    out[5] = (a00 * b11 - a02 * b08 + a03 * b07) * det;\n    out[6] = (a32 * b02 - a30 * b05 - a33 * b01) * det;\n    out[7] = (a20 * b05 - a22 * b02 + a23 * b01) * det;\n    out[8] = (a10 * b10 - a11 * b08 + a13 * b06) * det;\n    out[9] = (a01 * b08 - a00 * b10 - a03 * b06) * det;\n    out[10] = (a30 * b04 - a31 * b02 + a33 * b00) * det;\n    out[11] = (a21 * b02 - a20 * b04 - a23 * b00) * det;\n    out[12] = (a11 * b07 - a10 * b09 - a12 * b06) * det;\n    out[13] = (a00 * b09 - a01 * b07 + a02 * b06) * det;\n    out[14] = (a31 * b01 - a30 * b03 - a32 * b00) * det;\n    out[15] = (a20 * b03 - a21 * b01 + a22 * b00) * det;\n\n    return out;\n};","var identity = require('./identity');\n\nmodule.exports = lookAt;\n\n/**\n * Generates a look-at matrix with the given eye position, focal point, and up axis\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {vec3} eye Position of the viewer\n * @param {vec3} center Point the viewer is looking at\n * @param {vec3} up vec3 pointing up\n * @returns {mat4} out\n */\nfunction lookAt(out, eye, center, up) {\n    var x0, x1, x2, y0, y1, y2, z0, z1, z2, len,\n        eyex = eye[0],\n        eyey = eye[1],\n        eyez = eye[2],\n        upx = up[0],\n        upy = up[1],\n        upz = up[2],\n        centerx = center[0],\n        centery = center[1],\n        centerz = center[2];\n\n    if (Math.abs(eyex - centerx) < 0.000001 &&\n        Math.abs(eyey - centery) < 0.000001 &&\n        Math.abs(eyez - centerz) < 0.000001) {\n        return identity(out);\n    }\n\n    z0 = eyex - centerx;\n    z1 = eyey - centery;\n    z2 = eyez - centerz;\n\n    len = 1 / Math.sqrt(z0 * z0 + z1 * z1 + z2 * z2);\n    z0 *= len;\n    z1 *= len;\n    z2 *= len;\n\n    x0 = upy * z2 - upz * z1;\n    x1 = upz * z0 - upx * z2;\n    x2 = upx * z1 - upy * z0;\n    len = Math.sqrt(x0 * x0 + x1 * x1 + x2 * x2);\n    if (!len) {\n        x0 = 0;\n        x1 = 0;\n        x2 = 0;\n    } else {\n        len = 1 / len;\n        x0 *= len;\n        x1 *= len;\n        x2 *= len;\n    }\n\n    y0 = z1 * x2 - z2 * x1;\n    y1 = z2 * x0 - z0 * x2;\n    y2 = z0 * x1 - z1 * x0;\n\n    len = Math.sqrt(y0 * y0 + y1 * y1 + y2 * y2);\n    if (!len) {\n        y0 = 0;\n        y1 = 0;\n        y2 = 0;\n    } else {\n        len = 1 / len;\n        y0 *= len;\n        y1 *= len;\n        y2 *= len;\n    }\n\n    out[0] = x0;\n    out[1] = y0;\n    out[2] = z0;\n    out[3] = 0;\n    out[4] = x1;\n    out[5] = y1;\n    out[6] = z1;\n    out[7] = 0;\n    out[8] = x2;\n    out[9] = y2;\n    out[10] = z2;\n    out[11] = 0;\n    out[12] = -(x0 * eyex + x1 * eyey + x2 * eyez);\n    out[13] = -(y0 * eyex + y1 * eyey + y2 * eyez);\n    out[14] = -(z0 * eyex + z1 * eyey + z2 * eyez);\n    out[15] = 1;\n\n    return out;\n};","module.exports = multiply;\n\n/**\n * Multiplies two mat4's\n *\n * @param {mat4} out the receiving matrix\n * @param {mat4} a the first operand\n * @param {mat4} b the second operand\n * @returns {mat4} out\n */\nfunction multiply(out, a, b) {\n    var a00 = a[0], a01 = a[1], a02 = a[2], a03 = a[3],\n        a10 = a[4], a11 = a[5], a12 = a[6], a13 = a[7],\n        a20 = a[8], a21 = a[9], a22 = a[10], a23 = a[11],\n        a30 = a[12], a31 = a[13], a32 = a[14], a33 = a[15];\n\n    // Cache only the current line of the second matrix\n    var b0  = b[0], b1 = b[1], b2 = b[2], b3 = b[3];  \n    out[0] = b0*a00 + b1*a10 + b2*a20 + b3*a30;\n    out[1] = b0*a01 + b1*a11 + b2*a21 + b3*a31;\n    out[2] = b0*a02 + b1*a12 + b2*a22 + b3*a32;\n    out[3] = b0*a03 + b1*a13 + b2*a23 + b3*a33;\n\n    b0 = b[4]; b1 = b[5]; b2 = b[6]; b3 = b[7];\n    out[4] = b0*a00 + b1*a10 + b2*a20 + b3*a30;\n    out[5] = b0*a01 + b1*a11 + b2*a21 + b3*a31;\n    out[6] = b0*a02 + b1*a12 + b2*a22 + b3*a32;\n    out[7] = b0*a03 + b1*a13 + b2*a23 + b3*a33;\n\n    b0 = b[8]; b1 = b[9]; b2 = b[10]; b3 = b[11];\n    out[8] = b0*a00 + b1*a10 + b2*a20 + b3*a30;\n    out[9] = b0*a01 + b1*a11 + b2*a21 + b3*a31;\n    out[10] = b0*a02 + b1*a12 + b2*a22 + b3*a32;\n    out[11] = b0*a03 + b1*a13 + b2*a23 + b3*a33;\n\n    b0 = b[12]; b1 = b[13]; b2 = b[14]; b3 = b[15];\n    out[12] = b0*a00 + b1*a10 + b2*a20 + b3*a30;\n    out[13] = b0*a01 + b1*a11 + b2*a21 + b3*a31;\n    out[14] = b0*a02 + b1*a12 + b2*a22 + b3*a32;\n    out[15] = b0*a03 + b1*a13 + b2*a23 + b3*a33;\n    return out;\n};","module.exports = ortho;\n\n/**\n * Generates a orthogonal projection matrix with the given bounds\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {number} left Left bound of the frustum\n * @param {number} right Right bound of the frustum\n * @param {number} bottom Bottom bound of the frustum\n * @param {number} top Top bound of the frustum\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum\n * @returns {mat4} out\n */\nfunction ortho(out, left, right, bottom, top, near, far) {\n    var lr = 1 / (left - right),\n        bt = 1 / (bottom - top),\n        nf = 1 / (near - far);\n    out[0] = -2 * lr;\n    out[1] = 0;\n    out[2] = 0;\n    out[3] = 0;\n    out[4] = 0;\n    out[5] = -2 * bt;\n    out[6] = 0;\n    out[7] = 0;\n    out[8] = 0;\n    out[9] = 0;\n    out[10] = 2 * nf;\n    out[11] = 0;\n    out[12] = (left + right) * lr;\n    out[13] = (top + bottom) * bt;\n    out[14] = (far + near) * nf;\n    out[15] = 1;\n    return out;\n};","module.exports = perspective;\n\n/**\n * Generates a perspective projection matrix with the given bounds\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {number} fovy Vertical field of view in radians\n * @param {number} aspect Aspect ratio. typically viewport width/height\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum\n * @returns {mat4} out\n */\nfunction perspective(out, fovy, aspect, near, far) {\n    var f = 1.0 / Math.tan(fovy / 2),\n        nf = 1 / (near - far);\n    out[0] = f / aspect;\n    out[1] = 0;\n    out[2] = 0;\n    out[3] = 0;\n    out[4] = 0;\n    out[5] = f;\n    out[6] = 0;\n    out[7] = 0;\n    out[8] = 0;\n    out[9] = 0;\n    out[10] = (far + near) * nf;\n    out[11] = -1;\n    out[12] = 0;\n    out[13] = 0;\n    out[14] = (2 * far * near) * nf;\n    out[15] = 0;\n    return out;\n};","module.exports = perspectiveFromFieldOfView;\n\n/**\n * Generates a perspective projection matrix with the given field of view.\n * This is primarily useful for generating projection matrices to be used\n * with the still experiemental WebVR API.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {number} fov Object containing the following values: upDegrees, downDegrees, leftDegrees, rightDegrees\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum\n * @returns {mat4} out\n */\nfunction perspectiveFromFieldOfView(out, fov, near, far) {\n    var upTan = Math.tan(fov.upDegrees * Math.PI/180.0),\n        downTan = Math.tan(fov.downDegrees * Math.PI/180.0),\n        leftTan = Math.tan(fov.leftDegrees * Math.PI/180.0),\n        rightTan = Math.tan(fov.rightDegrees * Math.PI/180.0),\n        xScale = 2.0 / (leftTan + rightTan),\n        yScale = 2.0 / (upTan + downTan);\n\n    out[0] = xScale;\n    out[1] = 0.0;\n    out[2] = 0.0;\n    out[3] = 0.0;\n    out[4] = 0.0;\n    out[5] = yScale;\n    out[6] = 0.0;\n    out[7] = 0.0;\n    out[8] = -((leftTan - rightTan) * xScale * 0.5);\n    out[9] = ((upTan - downTan) * yScale * 0.5);\n    out[10] = far / (near - far);\n    out[11] = -1.0;\n    out[12] = 0.0;\n    out[13] = 0.0;\n    out[14] = (far * near) / (near - far);\n    out[15] = 0.0;\n    return out;\n}\n\n","module.exports = rotate;\n\n/**\n * Rotates a mat4 by the given angle\n *\n * @param {mat4} out the receiving matrix\n * @param {mat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @param {vec3} axis the axis to rotate around\n * @returns {mat4} out\n */\nfunction rotate(out, a, rad, axis) {\n    var x = axis[0], y = axis[1], z = axis[2],\n        len = Math.sqrt(x * x + y * y + z * z),\n        s, c, t,\n        a00, a01, a02, a03,\n        a10, a11, a12, a13,\n        a20, a21, a22, a23,\n        b00, b01, b02,\n        b10, b11, b12,\n        b20, b21, b22;\n\n    if (Math.abs(len) < 0.000001) { return null; }\n    \n    len = 1 / len;\n    x *= len;\n    y *= len;\n    z *= len;\n\n    s = Math.sin(rad);\n    c = Math.cos(rad);\n    t = 1 - c;\n\n    a00 = a[0]; a01 = a[1]; a02 = a[2]; a03 = a[3];\n    a10 = a[4]; a11 = a[5]; a12 = a[6]; a13 = a[7];\n    a20 = a[8]; a21 = a[9]; a22 = a[10]; a23 = a[11];\n\n    // Construct the elements of the rotation matrix\n    b00 = x * x * t + c; b01 = y * x * t + z * s; b02 = z * x * t - y * s;\n    b10 = x * y * t - z * s; b11 = y * y * t + c; b12 = z * y * t + x * s;\n    b20 = x * z * t + y * s; b21 = y * z * t - x * s; b22 = z * z * t + c;\n\n    // Perform rotation-specific matrix multiplication\n    out[0] = a00 * b00 + a10 * b01 + a20 * b02;\n    out[1] = a01 * b00 + a11 * b01 + a21 * b02;\n    out[2] = a02 * b00 + a12 * b01 + a22 * b02;\n    out[3] = a03 * b00 + a13 * b01 + a23 * b02;\n    out[4] = a00 * b10 + a10 * b11 + a20 * b12;\n    out[5] = a01 * b10 + a11 * b11 + a21 * b12;\n    out[6] = a02 * b10 + a12 * b11 + a22 * b12;\n    out[7] = a03 * b10 + a13 * b11 + a23 * b12;\n    out[8] = a00 * b20 + a10 * b21 + a20 * b22;\n    out[9] = a01 * b20 + a11 * b21 + a21 * b22;\n    out[10] = a02 * b20 + a12 * b21 + a22 * b22;\n    out[11] = a03 * b20 + a13 * b21 + a23 * b22;\n\n    if (a !== out) { // If the source and destination differ, copy the unchanged last row\n        out[12] = a[12];\n        out[13] = a[13];\n        out[14] = a[14];\n        out[15] = a[15];\n    }\n    return out;\n};","module.exports = rotateX;\n\n/**\n * Rotates a matrix by the given angle around the X axis\n *\n * @param {mat4} out the receiving matrix\n * @param {mat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\nfunction rotateX(out, a, rad) {\n    var s = Math.sin(rad),\n        c = Math.cos(rad),\n        a10 = a[4],\n        a11 = a[5],\n        a12 = a[6],\n        a13 = a[7],\n        a20 = a[8],\n        a21 = a[9],\n        a22 = a[10],\n        a23 = a[11];\n\n    if (a !== out) { // If the source and destination differ, copy the unchanged rows\n        out[0]  = a[0];\n        out[1]  = a[1];\n        out[2]  = a[2];\n        out[3]  = a[3];\n        out[12] = a[12];\n        out[13] = a[13];\n        out[14] = a[14];\n        out[15] = a[15];\n    }\n\n    // Perform axis-specific matrix multiplication\n    out[4] = a10 * c + a20 * s;\n    out[5] = a11 * c + a21 * s;\n    out[6] = a12 * c + a22 * s;\n    out[7] = a13 * c + a23 * s;\n    out[8] = a20 * c - a10 * s;\n    out[9] = a21 * c - a11 * s;\n    out[10] = a22 * c - a12 * s;\n    out[11] = a23 * c - a13 * s;\n    return out;\n};","module.exports = rotateY;\n\n/**\n * Rotates a matrix by the given angle around the Y axis\n *\n * @param {mat4} out the receiving matrix\n * @param {mat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\nfunction rotateY(out, a, rad) {\n    var s = Math.sin(rad),\n        c = Math.cos(rad),\n        a00 = a[0],\n        a01 = a[1],\n        a02 = a[2],\n        a03 = a[3],\n        a20 = a[8],\n        a21 = a[9],\n        a22 = a[10],\n        a23 = a[11];\n\n    if (a !== out) { // If the source and destination differ, copy the unchanged rows\n        out[4]  = a[4];\n        out[5]  = a[5];\n        out[6]  = a[6];\n        out[7]  = a[7];\n        out[12] = a[12];\n        out[13] = a[13];\n        out[14] = a[14];\n        out[15] = a[15];\n    }\n\n    // Perform axis-specific matrix multiplication\n    out[0] = a00 * c - a20 * s;\n    out[1] = a01 * c - a21 * s;\n    out[2] = a02 * c - a22 * s;\n    out[3] = a03 * c - a23 * s;\n    out[8] = a00 * s + a20 * c;\n    out[9] = a01 * s + a21 * c;\n    out[10] = a02 * s + a22 * c;\n    out[11] = a03 * s + a23 * c;\n    return out;\n};","module.exports = rotateZ;\n\n/**\n * Rotates a matrix by the given angle around the Z axis\n *\n * @param {mat4} out the receiving matrix\n * @param {mat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\nfunction rotateZ(out, a, rad) {\n    var s = Math.sin(rad),\n        c = Math.cos(rad),\n        a00 = a[0],\n        a01 = a[1],\n        a02 = a[2],\n        a03 = a[3],\n        a10 = a[4],\n        a11 = a[5],\n        a12 = a[6],\n        a13 = a[7];\n\n    if (a !== out) { // If the source and destination differ, copy the unchanged last row\n        out[8]  = a[8];\n        out[9]  = a[9];\n        out[10] = a[10];\n        out[11] = a[11];\n        out[12] = a[12];\n        out[13] = a[13];\n        out[14] = a[14];\n        out[15] = a[15];\n    }\n\n    // Perform axis-specific matrix multiplication\n    out[0] = a00 * c + a10 * s;\n    out[1] = a01 * c + a11 * s;\n    out[2] = a02 * c + a12 * s;\n    out[3] = a03 * c + a13 * s;\n    out[4] = a10 * c - a00 * s;\n    out[5] = a11 * c - a01 * s;\n    out[6] = a12 * c - a02 * s;\n    out[7] = a13 * c - a03 * s;\n    return out;\n};","module.exports = scale;\n\n/**\n * Scales the mat4 by the dimensions in the given vec3\n *\n * @param {mat4} out the receiving matrix\n * @param {mat4} a the matrix to scale\n * @param {vec3} v the vec3 to scale the matrix by\n * @returns {mat4} out\n **/\nfunction scale(out, a, v) {\n    var x = v[0], y = v[1], z = v[2];\n\n    out[0] = a[0] * x;\n    out[1] = a[1] * x;\n    out[2] = a[2] * x;\n    out[3] = a[3] * x;\n    out[4] = a[4] * y;\n    out[5] = a[5] * y;\n    out[6] = a[6] * y;\n    out[7] = a[7] * y;\n    out[8] = a[8] * z;\n    out[9] = a[9] * z;\n    out[10] = a[10] * z;\n    out[11] = a[11] * z;\n    out[12] = a[12];\n    out[13] = a[13];\n    out[14] = a[14];\n    out[15] = a[15];\n    return out;\n};","module.exports = str;\n\n/**\n * Returns a string representation of a mat4\n *\n * @param {mat4} mat matrix to represent as a string\n * @returns {String} string representation of the matrix\n */\nfunction str(a) {\n    return 'mat4(' + a[0] + ', ' + a[1] + ', ' + a[2] + ', ' + a[3] + ', ' +\n                    a[4] + ', ' + a[5] + ', ' + a[6] + ', ' + a[7] + ', ' +\n                    a[8] + ', ' + a[9] + ', ' + a[10] + ', ' + a[11] + ', ' + \n                    a[12] + ', ' + a[13] + ', ' + a[14] + ', ' + a[15] + ')';\n};","module.exports = translate;\n\n/**\n * Translate a mat4 by the given vector\n *\n * @param {mat4} out the receiving matrix\n * @param {mat4} a the matrix to translate\n * @param {vec3} v vector to translate by\n * @returns {mat4} out\n */\nfunction translate(out, a, v) {\n    var x = v[0], y = v[1], z = v[2],\n        a00, a01, a02, a03,\n        a10, a11, a12, a13,\n        a20, a21, a22, a23;\n\n    if (a === out) {\n        out[12] = a[0] * x + a[4] * y + a[8] * z + a[12];\n        out[13] = a[1] * x + a[5] * y + a[9] * z + a[13];\n        out[14] = a[2] * x + a[6] * y + a[10] * z + a[14];\n        out[15] = a[3] * x + a[7] * y + a[11] * z + a[15];\n    } else {\n        a00 = a[0]; a01 = a[1]; a02 = a[2]; a03 = a[3];\n        a10 = a[4]; a11 = a[5]; a12 = a[6]; a13 = a[7];\n        a20 = a[8]; a21 = a[9]; a22 = a[10]; a23 = a[11];\n\n        out[0] = a00; out[1] = a01; out[2] = a02; out[3] = a03;\n        out[4] = a10; out[5] = a11; out[6] = a12; out[7] = a13;\n        out[8] = a20; out[9] = a21; out[10] = a22; out[11] = a23;\n\n        out[12] = a00 * x + a10 * y + a20 * z + a[12];\n        out[13] = a01 * x + a11 * y + a21 * z + a[13];\n        out[14] = a02 * x + a12 * y + a22 * z + a[14];\n        out[15] = a03 * x + a13 * y + a23 * z + a[15];\n    }\n\n    return out;\n};","module.exports = transpose;\n\n/**\n * Transpose the values of a mat4\n *\n * @param {mat4} out the receiving matrix\n * @param {mat4} a the source matrix\n * @returns {mat4} out\n */\nfunction transpose(out, a) {\n    // If we are transposing ourselves we can skip a few steps but have to cache some values\n    if (out === a) {\n        var a01 = a[1], a02 = a[2], a03 = a[3],\n            a12 = a[6], a13 = a[7],\n            a23 = a[11];\n\n        out[1] = a[4];\n        out[2] = a[8];\n        out[3] = a[12];\n        out[4] = a01;\n        out[6] = a[9];\n        out[7] = a[13];\n        out[8] = a02;\n        out[9] = a12;\n        out[11] = a[14];\n        out[12] = a03;\n        out[13] = a13;\n        out[14] = a23;\n    } else {\n        out[0] = a[0];\n        out[1] = a[4];\n        out[2] = a[8];\n        out[3] = a[12];\n        out[4] = a[1];\n        out[5] = a[5];\n        out[6] = a[9];\n        out[7] = a[13];\n        out[8] = a[2];\n        out[9] = a[6];\n        out[10] = a[10];\n        out[11] = a[14];\n        out[12] = a[3];\n        out[13] = a[7];\n        out[14] = a[11];\n        out[15] = a[15];\n    }\n    \n    return out;\n};","module.exports = {\n  create: require('./create')\n  , clone: require('./clone')\n  , copy: require('./copy')\n  , identity: require('./identity')\n  , transpose: require('./transpose')\n  , invert: require('./invert')\n  , adjoint: require('./adjoint')\n  , determinant: require('./determinant')\n  , multiply: require('./multiply')\n  , translate: require('./translate')\n  , scale: require('./scale')\n  , rotate: require('./rotate')\n  , rotateX: require('./rotateX')\n  , rotateY: require('./rotateY')\n  , rotateZ: require('./rotateZ')\n  , fromRotationTranslation: require('./fromRotationTranslation')\n  , fromQuat: require('./fromQuat')\n  , frustum: require('./frustum')\n  , perspective: require('./perspective')\n  , perspectiveFromFieldOfView: require('./perspectiveFromFieldOfView')\n  , ortho: require('./ortho')\n  , lookAt: require('./lookAt')\n  , str: require('./str')\n}"]}
require=(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({"bunny":[function(require,module,exports){
exports.positions=[[1.301895,0.122622,2.550061],[1.045326,0.139058,2.835156],[0.569251,0.155925,2.805125],[0.251886,0.144145,2.82928],[0.063033,0.131726,3.01408],[-0.277753,0.135892,3.10716],[-0.441048,0.277064,2.594331],[-1.010956,0.095285,2.668983],[-1.317639,0.069897,2.325448],[-0.751691,0.264681,2.381496],[0.684137,0.31134,2.364574],[1.347931,0.302882,2.201434],[-1.736903,0.029894,1.724111],[-1.319986,0.11998,0.912925],[1.538077,0.157372,0.481711],[1.951975,0.081742,1.1641],[1.834768,0.095832,1.602682],[2.446122,0.091817,1.37558],[2.617615,0.078644,0.742801],[-1.609748,0.04973,-0.238721],[-1.281973,0.230984,-0.180916],[-1.074501,0.248204,0.034007],[-1.201734,0.058499,0.402234],[-1.444454,0.054783,0.149579],[-4.694605,5.075882,1.043427],[-3.95963,7.767394,0.758447],[-4.753339,5.339817,0.665061],[-1.150325,9.133327,-0.368552],[-4.316107,2.893611,0.44399],[-0.809202,9.312575,-0.466061],[0.085626,5.963693,1.685666],[-1.314853,9.00142,-0.1339],[-4.364182,3.072556,1.436712],[-2.022074,7.323396,0.678657],[1.990887,6.13023,0.479643],[-3.295525,7.878917,1.409353],[0.571308,6.197569,0.670657],[0.89661,6.20018,0.337056],[0.331851,6.162372,1.186371],[-4.840066,5.599874,2.296069],[2.138989,6.031291,0.228335],[0.678923,6.026173,1.894052],[-0.781682,5.601573,1.836738],[1.181315,6.239007,0.393293],[-3.606308,7.376476,2.661452],[-0.579059,4.042511,-1.540883],[-3.064069,8.630253,-2.597539],[-2.157271,6.837012,0.300191],[-2.966013,7.821581,-1.13697],[-2.34426,8.122965,0.409043],[-0.951684,5.874251,1.415119],[-2.834853,7.748319,0.182406],[-3.242493,7.820096,0.373674],[-0.208532,5.992846,1.252084],[-3.048085,8.431527,-2.129795],[1.413245,5.806324,2.243906],[-0.051222,6.064901,0.696093],[-4.204306,2.700062,0.713875],[-4.610997,6.343405,0.344272],[-3.291336,9.30531,-3.340445],[-3.27211,7.559239,-2.324016],[-4.23882,6.498344,3.18452],[-3.945317,6.377804,3.38625],[-4.906378,5.472265,1.315193],[-3.580131,7.846717,0.709666],[-1.995504,6.645459,0.688487],[-2.595651,7.86054,0.793351],[-0.008849,0.305871,0.184484],[-0.029011,0.314116,-0.257312],[-2.522424,7.565392,1.804212],[-1.022993,8.650826,-0.855609],[-3.831265,6.595426,3.266783],[-4.042525,6.855724,3.060663],[-4.17126,7.404742,2.391387],[3.904526,3.767693,0.092179],[0.268076,6.086802,1.469223],[-3.320456,8.753222,-2.08969],[1.203048,6.26925,0.612407],[-4.406479,2.985974,0.853691],[-3.226889,6.615215,-0.404243],[0.346326,1.60211,3.509858],[-3.955476,7.253323,2.722392],[-1.23204,0.068935,1.68794],[0.625436,6.196455,1.333156],[4.469132,2.165298,1.70525],[0.950053,6.262899,0.922441],[-2.980404,5.25474,-0.663155],[-4.859043,6.28741,1.537081],[-3.077453,4.641475,-0.892167],[-0.44002,8.222503,-0.771454],[-4.034112,7.639786,0.389935],[-3.696045,6.242042,3.394679],[-1.221806,7.783617,0.196451],[0.71461,6.149895,1.656636],[-4.713539,6.163154,0.495369],[-1.509869,0.913044,-0.832413],[-1.547249,2.066753,-0.852669],[-3.757734,5.793742,3.455794],[-0.831911,0.199296,1.718536],[-3.062763,7.52718,-1.550559],[0.938688,6.103354,1.820958],[-4.037033,2.412311,0.988026],[-4.130746,2.571806,1.101689],[-0.693664,9.174283,-0.952323],[-1.286742,1.079679,-0.751219],[1.543185,1.408925,3.483132],[1.535973,2.047979,3.655029],[0.93844,5.84101,2.195219],[-0.684401,5.918492,1.20109],[1.28844,2.008676,3.710781],[-3.586722,7.435506,-1.454737],[-0.129975,4.384192,2.930593],[-1.030531,0.281374,3.214273],[-3.058751,8.137238,-3.227714],[3.649524,4.592226,1.340021],[-3.354828,7.322425,-1.412086],[0.936449,6.209237,1.512693],[-1.001832,3.590411,-1.545892],[-3.770486,4.593242,2.477056],[-0.971925,0.067797,0.921384],[-4.639832,6.865407,2.311791],[-0.441014,8.093595,-0.595999],[-2.004852,6.37142,1.635383],[4.759591,1.92818,0.328328],[3.748064,1.224074,2.140484],[-0.703601,5.285476,2.251988],[0.59532,6.21893,0.981004],[0.980799,6.257026,1.24223],[1.574697,6.204981,0.381628],[1.149594,6.173608,1.660763],[-3.501963,5.895989,3.456576],[1.071122,5.424198,2.588717],[-0.774693,8.473335,-0.276957],[3.849959,4.15542,0.396742],[-0.801715,4.973149,-1.068582],[-2.927676,0.625112,2.326393],[2.669682,4.045542,2.971184],[-4.391324,4.74086,0.343463],[1.520129,6.270031,0.775471],[1.837586,6.084731,0.109188],[1.271475,5.975024,2.032355],[-3.487968,4.513249,2.605871],[-1.32234,1.517264,-0.691879],[-1.080301,1.648226,-0.805526],[-3.365703,6.910166,-0.454902],[1.36034,0.432238,3.075004],[-3.305013,5.774685,3.39142],[3.88432,0.654141,0.12574],[3.57254,0.377934,0.302501],[4.196136,0.807999,0.212229],[3.932997,0.543123,0.380579],[4.023704,3.286125,0.537597],[1.864455,4.916544,2.691677],[-4.775427,6.499498,1.440153],[-3.464928,3.68234,2.766356],[3.648972,1.751262,2.157485],[1.179111,3.238846,3.774796],[-0.171164,0.299126,-0.592669],[-4.502912,3.316656,0.875188],[-0.948454,9.214025,-0.679508],[1.237665,6.288593,1.046],[1.523423,6.268963,1.139544],[1.436519,6.140608,1.739316],[3.723607,1.504355,2.136762],[2.009495,4.045514,3.22053],[-1.921944,7.249905,0.213973],[1.254068,1.205518,3.474709],[-0.317087,5.996269,0.525872],[-2.996914,3.934607,2.900178],[-3.316873,4.028154,2.785696],[-3.400267,4.280157,2.689268],[-3.134842,4.564875,2.697192],[1.480563,4.692567,2.834068],[0.873682,1.315452,3.541585],[1.599355,0.91622,3.246769],[-3.292102,7.125914,2.768515],[3.74296,4.511299,0.616539],[4.698935,1.55336,0.26921],[-3.274387,3.299421,2.823946],[-2.88809,3.410699,2.955248],[1.171407,1.76905,3.688472],[1.430276,3.92483,3.473666],[3.916941,2.553308,0.018941],[0.701632,2.442372,3.778639],[1.562657,2.302778,3.660957],[4.476622,1.152407,0.182131],[-0.61136,5.761367,1.598838],[-3.102154,3.691687,2.903738],[1.816012,5.546167,2.380308],[3.853928,4.25066,0.750017],[1.234681,3.581665,3.673723],[1.862271,1.361863,3.355209],[1.346844,4.146995,3.327877],[1.70672,4.080043,3.274307],[0.897242,1.908983,3.6969],[-0.587022,9.191132,-0.565301],[-0.217426,5.674606,2.019968],[0.278925,6.120777,0.485403],[1.463328,3.578742,-2.001464],[-3.072985,4.264581,2.789502],[3.62353,4.673843,0.383452],[-3.053491,8.752377,-2.908434],[-2.628687,4.505072,2.755601],[0.891047,5.113781,2.748272],[-2.923732,3.06515,2.866368],[0.848008,4.754252,2.896972],[-3.319184,8.811641,-2.327412],[0.12864,8.814781,-1.334456],[1.549501,4.549331,-1.28243],[1.647161,3.738973,3.507719],[1.250888,0.945599,3.348739],[3.809662,4.038822,0.053142],[1.483166,0.673327,3.09156],[0.829726,3.635921,3.713103],[1.352914,5.226651,2.668113],[2.237352,4.37414,3.016386],[4.507929,0.889447,0.744249],[4.57304,1.010981,0.496588],[3.931422,1.720989,2.088175],[-0.463177,5.989835,0.834346],[-2.811236,3.745023,2.969587],[-2.805135,4.219721,2.841108],[-2.836842,4.802543,2.60826],[1.776716,2.084611,3.568638],[4.046881,1.463478,2.106273],[0.316265,5.944313,1.892785],[-2.86347,2.776049,2.77242],[-2.673644,3.116508,2.907104],[-2.621149,4.018502,2.903409],[-2.573447,5.198013,2.477481],[1.104039,2.278985,3.722469],[-4.602743,4.306413,0.902296],[-2.684878,1.510731,0.535039],[0.092036,8.473269,-0.99413],[-1.280472,5.602393,1.928105],[-1.0279,4.121582,-1.403103],[-2.461081,3.304477,2.957317],[-2.375929,3.659383,2.953233],[1.417579,2.715389,3.718767],[0.819727,2.948823,3.810639],[1.329962,0.761779,3.203724],[1.73952,5.295229,2.537725],[0.952523,3.945016,3.548229],[-2.569498,0.633669,2.84818],[-2.276676,0.757013,2.780717],[-2.013147,7.354429,-0.003202],[0.93143,1.565913,3.600325],[1.249014,1.550556,3.585842],[2.287252,4.072353,3.124544],[-4.7349,7.006244,1.690653],[-3.500602,8.80386,-2.009196],[-0.582629,5.549138,2.000923],[-1.865297,6.356066,1.313593],[-3.212154,2.376143,-0.565593],[2.092889,3.493536,-1.727931],[-2.528501,2.784531,2.833758],[-2.565697,4.893154,2.559605],[-2.153366,5.04584,2.465215],[1.631311,2.568241,3.681445],[2.150193,4.699227,2.807505],[0.507599,5.01813,2.775892],[4.129862,1.863698,2.015101],[3.578279,4.50766,-0.009598],[3.491023,4.806749,1.549265],[0.619485,1.625336,3.605125],[1.107499,2.932557,3.790061],[-2.082292,6.99321,0.742601],[4.839909,1.379279,0.945274],[3.591328,4.322645,-0.259497],[1.055245,0.710686,3.16553],[-3.026494,7.842227,1.624553],[0.146569,6.119214,0.981673],[-2.043687,2.614509,2.785526],[-2.302242,3.047775,2.936355],[-2.245686,4.100424,2.87794],[2.116148,5.063507,2.572204],[-1.448406,7.64559,0.251692],[2.550717,4.9268,2.517526],[-2.955456,7.80293,-1.782407],[1.882995,4.637167,2.895436],[-2.014924,3.398262,2.954896],[-2.273654,4.771227,2.611418],[-2.162723,7.876761,0.702473],[-0.198659,5.823062,1.739272],[-1.280908,2.133189,-0.921241],[2.039932,4.251568,3.136579],[1.477815,4.354333,3.108325],[0.560504,3.744128,3.6913],[-2.234018,1.054373,2.352782],[-3.189156,7.686661,-2.514955],[-3.744736,7.69963,2.116973],[-2.283366,2.878365,2.87882],[-2.153786,4.457481,2.743529],[4.933978,1.677287,0.713773],[3.502146,0.535336,1.752511],[1.825169,4.419253,3.081198],[3.072331,0.280979,0.106534],[-0.508381,1.220392,2.878049],[-3.138824,8.445394,-1.659711],[-2.056425,2.954815,2.897241],[-2.035343,5.398477,2.215842],[-3.239915,7.126798,-0.712547],[-1.867923,7.989805,0.526518],[1.23405,6.248973,1.387189],[-0.216492,8.320933,-0.862495],[-2.079659,3.755709,2.928563],[-1.78595,4.300374,2.805295],[-1.856589,5.10678,2.386572],[-1.714362,5.544778,2.004623],[1.722403,4.200291,-1.408161],[0.195386,0.086928,-1.318006],[1.393693,3.013404,3.710686],[-0.415307,8.508471,-0.996883],[-1.853777,0.755635,2.757275],[-1.724057,3.64533,2.884251],[-1.884511,4.927802,2.530885],[-1.017174,7.783908,-0.227078],[-1.7798,2.342513,2.741749],[-1.841329,3.943996,2.88436],[1.430388,5.468067,2.503467],[-2.030296,0.940028,2.611088],[-1.677028,1.215666,2.607771],[-1.74092,2.832564,2.827295],[4.144673,0.631374,0.503358],[4.238811,0.653992,0.762436],[-1.847016,2.082815,2.642674],[4.045764,3.194073,0.852117],[-1.563989,8.112739,0.303102],[-1.781627,1.794836,2.602338],[-1.493749,2.533799,2.797251],[-1.934496,4.690689,2.658999],[-1.499174,5.777946,1.747498],[-2.387409,0.851291,1.500524],[-1.872211,8.269987,0.392533],[-4.647726,6.765771,0.833653],[-3.157482,0.341958,-0.20671],[-1.725766,3.24703,2.883579],[-1.458199,4.079031,2.836325],[-1.621548,4.515869,2.719266],[-1.607292,4.918914,2.505881],[-1.494661,5.556239,1.991599],[-1.727269,7.423769,0.012337],[-1.382497,1.161322,2.640222],[-1.52129,4.681714,2.615467],[-4.247127,2.792812,1.250843],[-1.576338,0.742947,2.769799],[-1.499257,2.172763,2.743142],[-1.480392,3.103261,2.862262],[1.049137,2.625836,3.775384],[-1.368063,1.791587,2.695516],[-1.307839,2.344534,2.767575],[-1.336758,5.092221,2.355225],[-1.5617,5.301749,2.21625],[-1.483362,8.537704,0.196752],[-1.517348,8.773614,0.074053],[-1.474302,1.492731,2.641433],[2.48718,0.644247,-0.920226],[0.818091,0.422682,3.171218],[-3.623398,6.930094,3.033045],[1.676333,3.531039,3.591591],[1.199939,5.683873,2.365623],[-1.223851,8.841201,0.025414],[-1.286307,3.847643,2.918044],[-1.25857,4.810831,2.543605],[2.603662,5.572146,1.991854],[0.138984,5.779724,2.077834],[-1.267039,3.175169,2.890889],[-1.293616,3.454612,2.911774],[-2.60112,1.277184,0.07724],[2.552779,3.649877,3.163643],[-1.038983,1.248011,2.605933],[-1.288709,4.390967,2.761214],[-1.034218,5.485963,2.011467],[-1.185576,1.464842,2.624335],[-1.045682,2.54896,2.761102],[4.259176,1.660627,2.018096],[-0.961707,1.717183,2.598342],[-1.044603,3.147464,2.855335],[-0.891998,4.685429,2.669696],[-1.027561,5.081672,2.377939],[4.386506,0.832434,0.510074],[-1.014225,9.064991,-0.175352],[-1.218752,2.895443,2.823785],[-0.972075,4.432669,2.788005],[-2.714986,0.52425,1.509798],[-0.699248,1.517219,2.645738],[-1.161581,2.078852,2.722795],[-0.845249,3.286247,2.996471],[1.068329,4.443444,2.993863],[3.98132,3.715557,1.027775],[1.658097,3.982428,-1.651688],[-4.053701,2.449888,0.734746],[-0.910935,2.214149,2.702393],[0.087824,3.96165,3.439344],[-0.779714,3.724134,2.993429],[-1.051093,3.810797,2.941957],[-0.644941,4.3859,2.870863],[-2.98403,8.666895,-3.691888],[-0.754304,2.508325,2.812999],[-4.635524,3.662891,0.913005],[-0.983299,4.125978,2.915378],[4.916497,1.905209,0.621315],[4.874983,1.728429,0.468521],[2.33127,5.181957,2.441697],[-0.653711,2.253387,2.7949],[-3.623744,8.978795,-2.46192],[-4.555927,6.160279,0.215755],[-4.940628,5.806712,1.18383],[3.308506,2.40326,-0.910776],[0.58835,5.251928,-0.992886],[2.152215,5.449733,2.331679],[-0.712755,0.766765,3.280375],[-0.741771,1.9716,2.657235],[-4.828957,5.566946,2.635623],[-3.474788,8.696771,-1.776121],[1.770417,6.205561,1.331627],[-0.620626,4.064721,2.968972],[-1.499187,2.307735,-0.978901],[4.098793,2.330245,1.667951],[1.940444,6.167057,0.935904],[-2.314436,1.104995,1.681277],[-2.733629,7.742793,1.7705],[-0.452248,4.719868,2.740834],[-0.649143,4.951713,2.541296],[-0.479417,9.43959,-0.676324],[-2.251853,6.559275,0.046819],[0.033531,8.316907,-0.789939],[-0.513125,0.995673,3.125462],[-2.637602,1.039747,0.602434],[1.527513,6.230089,1.430903],[4.036124,2.609846,1.506498],[-3.559828,7.877892,1.228076],[-4.570736,4.960193,0.838201],[-0.432121,5.157731,2.467518],[-1.206735,4.562511,-1.237054],[-0.823768,3.788746,-1.567481],[-3.095544,7.353613,-1.024577],[-4.056088,7.631119,2.062001],[-0.289385,5.382261,2.329421],[1.69752,6.136483,1.667037],[-0.168758,5.061138,2.617453],[2.853576,1.605528,-1.229958],[-4.514319,6.586675,0.352756],[-2.558081,7.741151,1.29295],[1.61116,5.92358,2.071534],[3.936921,3.354857,0.091755],[-0.1633,1.119272,3.147975],[0.067551,1.593475,3.38212],[-1.303239,2.328184,-1.011672],[-0.438093,0.73423,3.398384],[-4.62767,3.898187,0.849573],[0.286853,4.165281,3.284834],[-2.968052,8.492812,-3.493693],[-0.111896,3.696111,3.53791],[-3.808245,8.451731,-1.574742],[0.053416,5.558764,2.31107],[3.956269,3.012071,0.11121],[-0.710956,8.106561,-0.665154],[0.234725,2.717326,3.722379],[-0.031594,2.76411,3.657347],[-0.017371,4.700633,2.81911],[0.215064,5.034859,2.721426],[-0.111151,8.480333,-0.649399],[3.97942,3.575478,0.362219],[0.392962,4.735392,2.874321],[4.17015,2.085087,1.865999],[0.169054,1.244786,3.337709],[0.020049,3.165818,3.721736],[0.248212,3.595518,3.698376],[0.130706,5.295541,2.540034],[-4.541357,4.798332,1.026866],[-1.277485,1.289518,-0.667272],[3.892133,3.54263,-0.078056],[4.057379,3.03669,0.997913],[0.287719,0.884758,3.251787],[0.535771,1.144701,3.400096],[0.585303,1.399362,3.505353],[0.191551,2.076246,3.549355],[0.328656,2.394576,3.649623],[0.413124,3.240728,3.771515],[0.630361,4.501549,2.963623],[0.529441,5.854392,2.120225],[3.805796,3.769958,-0.162079],[3.447279,4.344846,-0.467276],[0.377618,5.551116,2.426017],[0.409355,1.821269,3.606333],[0.719959,2.194726,3.703851],[0.495922,3.501519,3.755661],[0.603408,5.354097,2.603088],[-4.605056,7.531978,1.19579],[0.907972,0.973128,3.356513],[0.750134,3.356137,3.765847],[0.4496,3.993244,3.504544],[-3.030738,7.48947,-1.259169],[0.707505,5.602005,2.43476],[0.668944,0.654891,3.213797],[0.593244,2.700978,3.791427],[1.467759,3.30327,3.71035],[3.316249,2.436388,2.581175],[3.26138,1.724425,2.539028],[-1.231292,7.968263,0.281414],[-0.108773,8.712307,-0.790607],[4.445684,1.819442,1.896988],[1.998959,2.281499,3.49447],[2.162269,2.113817,3.365449],[4.363397,1.406731,1.922714],[4.808,2.225842,0.611127],[2.735919,0.771812,-0.701142],[1.897735,2.878428,3.583482],[-3.31616,5.331985,3.212394],[-3.3314,6.018137,3.313018],[-3.503183,6.480103,3.222216],[-1.904453,5.750392,1.913324],[-1.339735,3.559592,-1.421817],[-1.044242,8.22539,0.037414],[1.643492,3.110676,3.647424],[3.992832,3.686244,0.710946],[1.774207,1.71842,3.475768],[-3.438842,5.5713,3.427818],[4.602447,1.2583,1.619528],[-0.925516,7.930042,0.072336],[-1.252093,3.846565,-1.420761],[-3.426857,5.072419,2.97806],[-3.160408,6.152629,3.061869],[3.739931,3.367082,2.041273],[1.027419,4.235891,3.251253],[4.777703,1.887452,1.560409],[-3.318528,6.733796,2.982968],[2.929265,4.962579,2.271079],[3.449761,2.838629,2.474576],[-3.280159,5.029875,2.787514],[4.068939,2.993629,0.741567],[0.303312,8.70927,-1.121972],[0.229852,8.981322,-1.186075],[-0.011045,9.148156,-1.047057],[-2.942683,5.579613,2.929297],[-3.145409,5.698727,3.205778],[-3.019089,6.30887,2.794323],[-3.217135,6.468191,2.970032],[-3.048298,6.993641,2.623378],[-3.07429,6.660982,2.702434],[3.612011,2.5574,2.25349],[2.54516,4.553967,2.75884],[-1.683759,7.400787,0.250868],[-1.756066,7.463557,0.448031],[-3.023761,5.149697,2.673539],[3.112376,2.677218,2.782378],[2.835327,4.581196,2.567146],[-2.973799,7.225458,2.506988],[-0.591645,8.740662,-0.505845],[3.782861,2.04337,2.03066],[3.331604,3.36343,2.605047],[2.966866,1.205497,2.537432],[0.002669,9.654748,-1.355559],[2.632801,0.58497,2.540311],[-2.819398,5.087372,2.521098],[2.616193,5.332961,2.194288],[-3.193973,4.925634,2.607924],[-3.12618,5.27524,2.944544],[-0.426003,8.516354,-0.501528],[2.802717,1.387643,2.751649],[-3.120597,7.889111,-2.75431],[2.636648,1.71702,2.991302],[-2.853151,6.711792,2.430276],[-2.843836,6.962865,2.400842],[1.9696,3.199023,3.504514],[-2.461751,0.386352,3.008994],[1.64127,0.495758,3.02958],[-4.330472,5.409831,0.025287],[-2.912387,5.980416,2.844261],[-2.490069,0.211078,2.985391],[3.581816,4.809118,0.733728],[2.693199,2.647213,3.126709],[-0.182964,8.184108,-0.638459],[-2.226855,0.444711,2.946552],[-0.720175,8.115055,0.017689],[2.645302,4.316212,2.850139],[-0.232764,9.329503,-0.918639],[4.852365,1.471901,0.65275],[2.76229,2.014994,2.957755],[-2.808374,5.354301,2.644695],[-2.790967,6.406963,2.547985],[-1.342684,0.418488,-1.669183],[2.690675,5.593587,-0.041236],[4.660146,1.6318,1.713314],[2.775667,3.007229,3.111332],[-0.396696,8.963432,-0.706202],[2.446707,2.740617,3.321433],[-4.803209,5.884634,2.603672],[-2.652003,1.6541,1.5078],[3.932327,3.972874,0.831924],[2.135906,0.955587,2.986608],[2.486131,2.053802,3.124115],[-0.386706,8.115753,-0.37565],[-2.720727,7.325044,2.224878],[-1.396946,7.638016,-0.16486],[-0.62083,7.989771,-0.144413],[-2.653272,5.729684,2.667679],[3.038188,4.65835,2.364142],[2.381721,0.739472,2.788992],[-2.345829,5.474929,2.380633],[-2.518983,6.080562,2.479383],[-2.615793,6.839622,2.186116],[-2.286566,0.143752,2.766848],[-4.771219,6.508766,1.070797],[3.717308,2.905019,2.097994],[2.50521,3.016743,3.295898],[2.208448,1.56029,3.216806],[3.346783,1.01254,2.119951],[2.653503,3.26122,3.175738],[-2.359636,5.827519,2.402297],[-1.952693,0.558102,2.853307],[-0.321562,9.414885,-1.187501],[3.138923,1.405072,2.520765],[1.493728,1.780051,3.621969],[3.01817,0.907291,2.336909],[3.183548,1.185297,2.352175],[1.608619,5.006753,2.695131],[-4.723919,6.836107,1.095288],[-1.017586,8.865429,-0.149328],[4.730762,1.214014,0.64008],[-2.135182,6.647907,1.495471],[-2.420382,6.546114,2.108209],[-2.458053,7.186346,1.896623],[3.437124,0.275798,1.138203],[0.095925,8.725832,-0.926481],[2.417376,2.429869,3.287659],[2.279951,1.200317,3.049994],[2.674753,2.326926,3.044059],[-2.328123,6.849164,1.75751],[-3.418616,7.853407,0.126248],[-3.151587,7.77543,-0.110889],[2.349144,5.653242,2.05869],[-2.273236,6.085631,2.242888],[-4.560601,4.525342,1.261241],[2.866334,3.796067,2.934717],[-2.17493,6.505518,1.791367],[3.12059,3.283157,2.818869],[3.037703,3.562356,2.866653],[0.066233,9.488418,-1.248237],[2.749941,0.975018,2.573371],[-2.155749,5.801033,2.204009],[-2.162778,6.261889,2.028596],[1.936874,0.459142,2.956718],[3.176249,4.335541,2.440447],[4.356599,1.029423,1.700589],[3.873502,3.082678,1.80431],[2.895489,4.243034,2.735259],[-0.095774,9.468195,-1.07451],[-1.124982,7.886808,-0.480851],[3.032304,3.065454,2.897927],[3.692687,4.5961,0.957858],[-3.013045,3.807235,-1.098381],[-0.790012,8.92912,-0.367572],[1.905793,0.73179,2.996728],[3.530396,3.426233,2.356583],[2.12299,0.624933,2.929167],[-2.069196,6.039284,2.01251],[-3.565623,7.182525,2.850039],[2.959264,2.376337,2.829242],[2.949071,1.822483,2.793933],[4.036142,0.763803,1.703744],[-1.993527,6.180318,1.804936],[-0.030987,0.766389,3.344766],[-0.549683,8.225193,-0.189341],[-0.765469,8.272246,-0.127174],[-2.947047,7.541648,-0.414113],[-3.050327,9.10114,-3.435619],[3.488566,2.231807,2.399836],[3.352283,4.727851,1.946438],[4.741011,2.162773,1.499574],[-1.815093,6.072079,1.580722],[-3.720969,8.267927,-0.984713],[1.932826,3.714052,3.427488],[3.323617,4.438961,2.20732],[0.254111,9.26364,-1.373244],[-1.493384,7.868585,-0.450051],[-0.841901,0.776135,-1.619467],[0.243537,6.027668,0.091687],[0.303057,0.313022,-0.531105],[-0.435273,0.474098,3.481552],[2.121507,2.622389,3.486293],[1.96194,1.101753,3.159584],[3.937991,3.407551,1.551392],[0.070906,0.295753,1.377185],[-1.93588,7.631764,0.651674],[-2.523531,0.744818,-0.30985],[2.891496,3.319875,2.983079],[4.781765,1.547061,1.523129],[-2.256064,7.571251,0.973716],[3.244861,3.058249,2.724392],[-0.145855,0.437775,3.433662],[1.586296,5.658538,2.358487],[3.658336,3.774921,2.071837],[2.840463,4.817098,2.46376],[-1.219464,8.122542,-0.672808],[-2.520906,2.664486,-1.034346],[-1.315417,8.471365,-0.709557],[3.429165,3.74686,2.446169],[3.074579,3.840758,2.767409],[3.569443,3.166337,2.333647],[2.294337,3.280051,3.359346],[2.21816,3.66578,3.269222],[2.158662,4.151444,-1.357919],[1.13862,4.380986,-1.404565],[3.388382,2.749931,-0.840949],[3.059892,5.084848,2.026066],[3.204739,2.075145,2.640706],[3.387065,1.42617,2.305275],[3.910398,2.670742,1.750179],[3.471512,1.945821,2.395881],[4.08082,1.070654,1.960171],[-1.057861,0.133036,2.146707],[-0.151749,5.53551,-0.624323],[3.233099,4.003778,2.571172],[2.611726,5.319199,-0.499388],[2.682909,1.094499,-1.206247],[-1.22823,7.656887,0.041409],[-2.293247,7.259189,0.013844],[0.081315,0.202174,3.286381],[-1.002038,5.794454,-0.187194],[3.448856,4.08091,2.258325],[0.287883,9.006888,-1.550641],[-3.851019,4.059839,-0.646922],[3.610966,4.205438,1.913129],[2.239042,2.950872,3.449959],[0.216305,0.442843,3.328052],[1.87141,2.470745,3.574559],[3.811378,2.768718,-0.228364],[2.511081,1.362724,2.969349],[-1.59813,7.866506,0.440184],[-3.307975,2.851072,-0.894978],[-0.107011,8.90573,-0.884399],[-3.855315,2.842597,-0.434541],[2.517853,1.090768,2.799687],[3.791709,2.36685,2.002703],[4.06294,2.773922,0.452723],[-2.973289,7.61703,-0.623653],[-2.95509,8.924462,-3.446319],[2.861402,0.562592,2.184397],[-1.109725,8.594206,-0.076812],[-0.725722,7.924485,-0.381133],[-1.485587,1.329994,-0.654405],[-4.342113,3.233735,1.752922],[-2.968049,7.955519,-2.09405],[-3.130948,0.446196,0.85287],[-4.958475,5.757329,1.447055],[-3.086547,7.615193,-1.953168],[-3.751923,5.412821,3.373373],[-4.599645,7.480953,1.677134],[1.133992,0.274871,0.032249],[-2.956512,8.126905,-1.785461],[-0.960645,4.73065,-1.191786],[-2.871064,0.875559,0.424881],[-4.932114,5.99614,1.483845],[-2.981761,8.124612,-1.387276],[0.362298,8.978545,-1.368024],[-4.408375,3.046271,0.602373],[2.865841,2.322263,-1.344625],[-4.7848,5.620895,0.594432],[-2.88322,0.338931,1.67231],[-4.688101,6.772931,1.872318],[-4.903948,6.164698,1.27135],[2.85663,1.005647,-0.906843],[2.691286,0.209811,0.050512],[-4.693636,6.477556,0.665796],[-4.472331,6.861067,0.477318],[0.883065,0.204907,3.073933],[-0.995867,8.048729,-0.653897],[-0.794663,5.670397,-0.390119],[3.313153,1.638006,-0.722289],[-4.856459,5.394758,1.032591],[-3.005448,7.783023,-0.819641],[3.11891,2.036974,-1.08689],[-2.364319,2.408419,2.63419],[-2.927132,8.75435,-3.537159],[-3.296222,7.964629,-3.134625],[-1.642041,4.13417,-1.301665],[2.030759,0.176372,-1.030923],[-4.559069,3.751053,0.548453],[3.438385,4.59454,-0.243215],[-2.561769,7.93935,0.177696],[2.990593,1.335314,-0.943177],[1.2808,0.276396,-0.49072],[-0.318889,0.290684,0.211143],[3.54614,3.342635,-0.767878],[-3.073372,7.780018,-2.357807],[-4.455388,4.387245,0.361038],[-4.659393,6.276064,2.767014],[0.636799,4.482223,-1.426284],[-2.987681,8.072969,-2.45245],[-2.610445,0.763554,1.792054],[3.358241,2.006707,-0.802973],[-0.498347,0.251594,0.962885],[3.1322,0.683312,2.038777],[-4.389801,7.493776,0.690247],[0.431467,4.22119,-1.614215],[-4.376181,3.213141,0.273255],[-4.872319,5.715645,0.829714],[-4.826893,6.195334,0.849912],[3.516562,2.23732,-0.677597],[3.131656,1.698841,-0.975761],[-4.754925,5.411666,1.989303],[-2.987299,7.320765,-0.629479],[-3.757635,3.274862,-0.744022],[3.487044,2.541999,-0.699933],[-4.53274,4.649505,0.77093],[-1.424192,0.099423,2.633327],[3.090867,2.476975,-1.146957],[-2.713256,0.815622,2.17311],[3.348121,3.254167,-0.984896],[-3.031379,0.16453,-0.309937],[-0.949757,4.518137,-1.309172],[-0.889509,0.095256,1.288803],[3.539594,1.966105,-0.553965],[-4.60612,7.127749,0.811958],[-2.332953,1.444713,1.624548],[3.136293,2.95805,-1.138272],[3.540808,3.069058,-0.735285],[3.678852,2.362375,-0.452543],[-4.648898,7.37438,0.954791],[-0.646871,0.19037,3.344746],[2.2825,0.29343,-0.826273],[-4.422291,7.183959,0.557517],[-4.694668,5.246103,2.541768],[-4.583691,4.145486,0.600207],[-2.934854,7.912513,-1.539269],[-3.067861,7.817472,-0.546501],[3.825095,3.229512,-0.237547],[2.532494,0.323059,2.387105],[-2.514583,0.692857,1.23597],[-4.736805,7.214384,1.259421],[-2.98071,8.409903,-2.468199],[2.621468,1.385844,-1.406355],[3.811447,3.560855,1.847828],[3.432925,1.497205,-0.489784],[3.746609,3.631538,-0.39067],[3.594909,2.832257,-0.576012],[-0.404192,5.300188,-0.856561],[-4.762996,6.483774,1.702648],[-4.756612,6.786223,1.43682],[-2.965309,8.437217,-2.785495],[2.863867,0.74087,-0.429684],[4.02503,2.968753,1.392419],[3.669036,1.833858,-0.304971],[-2.888864,0.720537,0.778057],[-2.36982,0.979443,1.054447],[-2.959259,8.222303,-2.659724],[-3.467825,7.545739,-2.333445],[2.153426,0.446256,-1.20523],[-3.229807,9.189699,-3.596609],[-3.72486,8.773707,-2.046671],[3.687218,3.297751,-0.523746],[1.381025,0.08815,-1.185668],[-2.796828,7.205622,-0.208783],[3.647194,4.066232,-0.291507],[-4.578376,3.885556,1.52546],[-2.840262,0.63094,1.89499],[-2.429514,0.922118,1.820781],[-4.675079,6.573925,2.423363],[2.806207,4.320188,-1.027372],[-1.289608,0.097241,1.321661],[-3.010731,8.141334,-2.866148],[3.202291,1.235617,-0.549025],[4.094792,2.477519,0.304581],[2.948403,0.966873,-0.664857],[-4.83297,5.920587,2.095461],[-2.169693,7.257277,0.946184],[-1.335807,3.057597,-1.303166],[-1.037877,0.64151,-1.685271],[2.627919,0.089814,0.439074],[3.815794,3.808102,1.730493],[-2.973455,8.433141,-3.08872],[-2.391558,7.331428,1.658264],[-4.333107,4.529978,1.850516],[-4.640293,3.767107,1.168841],[3.600716,4.46931,1.734024],[3.880803,1.730158,-0.172736],[3.814183,4.262372,1.167042],[4.37325,0.829542,1.413729],[2.490447,5.75111,0.011492],[3.460003,4.962436,1.188971],[3.918419,3.814234,1.358271],[-0.807595,8.840504,-0.953711],[3.752855,4.20577,1.57177],[-2.991085,8.816501,-3.244595],[-2.333196,7.128889,1.551985],[3.977718,3.570941,1.25937],[4.360071,0.755579,1.079916],[4.637579,1.027973,1.032567],[-2.317,7.421066,1.329589],[-1.013404,8.293662,-0.7823],[4.548023,1.020644,1.420462],[4.763258,1.266798,1.296203],[4.896,2.073084,1.255213],[4.015005,3.325226,1.093879],[4.94885,1.860936,0.894463],[-2.189645,6.954634,1.270077],[4.887442,1.720992,1.288526],[-3.184068,7.871802,0.956189],[-1.274318,0.839887,-1.224389],[-2.919521,7.84432,0.541629],[-2.994586,7.766102,1.96867],[-3.417504,9.241714,-3.093201],[-3.174563,7.466456,2.473617],[-3.263067,9.069412,-3.003459],[-2.841592,0.529833,2.693434],[-3.611069,9.158804,-2.829871],[-4.642828,5.927526,0.320549],[-3.809308,9.051035,-2.692749],[-2.837582,7.487987,-0.106206],[4.773025,2.330442,1.213899],[4.897435,2.209906,0.966657],[-3.067637,8.164062,-1.12661],[-3.122129,8.08074,-0.899194],[4.571019,2.358113,1.462054],[4.584884,2.454418,0.709466],[-3.661093,7.146581,-0.475948],[4.735131,2.415859,0.933939],[4.207556,2.540018,1.218293],[-3.607595,7.89161,-0.121172],[-1.527952,0.775564,-1.061903],[4.53874,2.503273,1.099583],[-3.938837,7.587988,0.082449],[-4.853582,6.152409,1.787943],[-4.752214,6.247234,2.296873],[4.602935,2.363955,0.488901],[-1.81638,6.365879,0.868272],[0.595467,4.744074,-1.32483],[1.87635,3.511986,-1.842924],[4.330947,2.534326,0.720503],[4.108736,2.750805,0.904552],[-1.890939,8.492628,-0.290768],[-3.504309,6.173058,-0.422804],[-1.611992,6.196732,0.648736],[-3.899149,7.826123,1.088845],[-3.078303,3.008813,-1.035784],[-2.798999,7.844899,1.340061],[-1.248839,5.959105,0.041761],[0.767779,4.337318,3.090817],[-3.831177,7.515605,2.432261],[-1.667528,6.156208,0.365267],[-1.726078,6.237384,1.100059],[-3.972037,4.520832,-0.370756],[-4.40449,7.636357,1.520425],[-1.34506,6.004054,1.293159],[-1.233556,6.049933,0.500651],[-3.696869,7.79732,0.37979],[-3.307798,8.949964,-2.698113],[-1.997295,6.615056,1.103691],[-3.219222,8.336394,-1.150614],[-3.452623,8.31866,-0.9417],[-3.94641,2.990494,2.212592],[-3.250025,8.030414,-0.596097],[-2.02375,1.571333,2.397939],[-3.190358,7.665013,2.268183],[-2.811918,7.618526,2.145587],[-1.005265,5.892303,0.072158],[-0.93721,5.974148,0.906669],[-4.646072,7.492193,1.45312],[-0.252931,1.797654,3.140638],[-1.076064,5.738433,1.695953],[-3.980534,7.744391,1.735791],[-0.721187,5.939396,0.526032],[-0.42818,5.919755,0.229001],[-1.43429,6.11622,0.93863],[-0.985638,5.939683,0.290636],[-4.433836,7.461372,1.966437],[-3.696398,7.844859,1.547325],[-3.390772,7.820186,1.812204],[-2.916787,7.864019,0.804341],[-3.715952,8.037269,-0.591341],[-4.204634,7.72919,1.119866],[-4.592233,5.592883,0.246264],[3.307299,5.061701,1.622917],[-3.515159,7.601467,2.368914],[-3.435742,8.533457,-1.37916],[-0.269421,4.545635,-1.366445],[-2.542124,3.768736,-1.258512],[-3.034003,7.873773,1.256854],[-2.801399,7.856028,1.080137],[3.29354,5.220894,1.081767],[-2.35109,1.299486,1.01206],[-3.232213,7.768136,2.047563],[3.290415,5.217525,0.68019],[-3.415109,7.731034,2.144326],[3.440357,4.962463,0.373387],[3.147346,5.352121,1.386923],[2.847252,5.469051,1.831981],[3.137682,5.410222,1.050188],[3.102694,5.310456,1.676434],[-3.044601,0.39515,1.994084],[2.903647,5.561338,1.518598],[-3.810148,8.093598,-0.889131],[4.234835,0.803054,1.593271],[3.240165,5.228747,0.325955],[3.037452,5.509825,0.817137],[2.635031,5.795187,1.439724],[3.071607,5.318303,0.080142],[2.909167,5.611751,1.155874],[3.044889,5.465928,0.486566],[2.502256,5.770673,1.740054],[-0.067497,0.086416,-1.190239],[2.33326,5.906051,0.138295],[0.65096,4.205423,3.308767],[-2.671137,7.936535,0.432731],[2.14463,5.879214,1.866047],[-4.776469,5.890689,0.561986],[2.72432,5.655145,0.211951],[2.730488,5.751455,0.695894],[2.572682,5.869295,1.152663],[1.906776,5.739123,2.196551],[2.344414,5.999961,0.772922],[-3.377905,7.448708,-1.863251],[2.285149,5.968156,1.459258],[2.385989,5.928974,0.3689],[2.192111,6.087516,0.959901],[2.36372,6.001101,1.074346],[1.972022,6.079603,1.591175],[1.87615,5.976698,1.91554],[-3.824761,9.05372,-2.928615],[2.044704,6.129704,1.263111],[-2.583046,0.849537,2.497344],[-0.078825,2.342205,3.520322],[-0.704686,0.537165,3.397194],[-0.257449,3.235334,3.647545],[-0.332064,1.448284,3.022583],[-2.200146,0.898284,-0.447212],[-2.497508,1.745446,1.829167],[0.30702,4.416315,2.978956],[-3.205197,3.479307,-1.040582],[0.110069,9.347725,-1.563686],[-0.82754,0.883886,3.065838],[-2.017103,1.244785,2.42512],[-0.421091,2.309929,3.153898],[-0.491604,3.796072,3.16245],[2.786955,3.501241,-1.340214],[-3.229055,4.380713,-0.899241],[3.730768,0.76845,1.90312],[-0.561079,2.652382,3.152463],[-3.461471,3.086496,2.662505],[-0.661405,3.446009,3.179939],[-0.915351,0.636755,3.243708],[-2.992964,8.915628,-3.729833],[-0.439627,3.502104,3.42665],[-1.154217,0.883181,2.800835],[-1.736193,1.465474,2.595489],[-0.423928,3.24435,3.548277],[-0.511153,2.871046,3.379749],[-0.675722,2.991756,3.143262],[-1.092602,0.599103,3.090639],[-0.89821,2.836952,2.840023],[-2.658412,0.781376,0.960575],[-2.271455,1.222857,1.330478],[-0.877861,1.111222,2.72263],[-0.306959,2.876987,3.556044],[-3.839274,7.84138,-0.918404],[-0.172094,4.083799,3.141708],[-1.548332,0.2529,2.864655],[-0.217353,4.873911,-1.223104],[-3.384242,3.181056,-0.95579],[-2.731704,0.382421,2.895502],[-1.285037,0.551267,2.947675],[0.077224,4.246579,3.066738],[-0.479979,1.77955,2.860011],[-0.716375,1.224694,2.666751],[-0.54622,3.138255,3.393457],[-2.33413,1.821222,2.124883],[-0.50653,2.037147,2.897465],[2.451291,1.211389,-1.466589],[-3.160047,2.894081,2.724286],[-4.137258,5.433431,3.21201],[0.462896,0.320456,-0.174837],[-0.37458,2.609447,3.379253],[-3.095244,0.256205,2.196446],[-4.197985,5.732991,3.262924],[-0.729747,0.246036,0.497036],[-2.356189,5.062,-0.965619],[-1.609036,0.25962,-1.487367],[-4.074381,6.074061,3.409459],[-3.619304,4.0022,2.65705],[-0.543393,8.742896,-1.056622],[-4.30356,6.858934,2.879642],[-0.716688,2.901831,-2.11202],[1.547362,0.083189,1.138764],[-0.250916,0.275268,1.201344],[-3.778035,3.13624,2.466177],[-4.594316,5.771342,3.01694],[-3.717706,3.442887,2.603344],[-4.311163,5.224669,3.019373],[-0.610389,2.095161,-1.923515],[-3.040086,6.196918,-0.429149],[-3.802695,3.768247,2.545523],[-0.159541,2.043362,3.328549],[-3.744329,4.31785,2.491889],[-3.047939,0.214155,1.873639],[-4.41685,6.113058,3.166774],[-1.165133,0.460692,-1.742134],[-1.371289,4.249996,-1.317935],[-3.447883,0.3521,0.466205],[-4.495555,6.465548,2.944147],[-3.455335,0.171653,0.390816],[-3.964028,4.017196,2.376009],[-1.323595,1.763126,-0.750772],[-3.971142,5.277524,-0.19496],[-3.222052,0.237723,0.872229],[-4.403784,3.89107,1.872077],[-3.333311,0.342997,0.661016],[-4.495871,4.29606,1.63608],[-3.636081,2.760711,2.361949],[-4.487235,3.559608,1.66737],[-4.719787,7.26888,1.658722],[-1.086143,9.035741,-0.707144],[-2.339693,1.600485,-0.404817],[-4.642011,7.123829,1.990987],[-1.498077,3.854035,-1.369787],[-4.188372,4.729363,2.02983],[-3.116344,5.882284,-0.468884],[-4.305236,4.246417,1.976991],[-3.022509,0.22819,1.065688],[-2.799916,0.52022,1.128319],[-4.262823,3.534409,2.020383],[-4.221533,3.947676,2.11735],[-3.744353,4.391712,-0.6193],[-1.272905,0.156694,-1.741753],[-3.62491,2.669825,-0.549664],[-4.180756,3.096179,1.987215],[-4.059276,4.305313,2.232924],[-2.812753,0.183226,1.370267],[-4.032437,3.512234,2.309985],[-0.03787,0.28188,0.530391],[-4.711562,5.468653,2.822838],[-4.500636,6.953314,2.564445],[-4.479433,7.216991,2.270682],[3.990562,0.50522,0.716309],[-2.512229,6.863447,-0.100658],[-2.968058,6.956639,-0.37061],[2.550375,3.142683,-1.54068],[-2.320059,3.521605,-1.279397],[-4.556319,6.64662,2.745363],[-4.281091,7.108116,2.667598],[-2.050095,8.411689,0.121353],[-2.44854,1.135487,0.851875],[3.121815,0.699943,-0.277167],[-4.69877,6.00376,2.843035],[-1.360599,8.824742,-0.595597],[1.128437,0.171611,0.301691],[-4.360146,6.289423,0.042233],[1.400795,4.088829,-1.620409],[-3.193462,8.460137,-3.559446],[-3.168771,8.878431,-3.635795],[-3.434275,9.304302,-3.460878],[-3.349993,8.808093,-3.38179],[-3.304823,8.323865,-3.325905],[-3.572607,9.308843,-3.207672],[-3.166393,8.201215,-3.43014],[-3.451638,9.05331,-3.351345],[-3.309591,8.549758,-3.375055],[-3.527992,8.793926,-3.100376],[-3.6287,8.981677,-3.076319],[-3.445505,8.001887,-2.8273],[-3.408011,8.221014,-3.039237],[-3.65928,8.740382,-2.808856],[-3.878019,8.797295,-2.462866],[-3.515132,8.232341,-2.747739],[-3.460331,8.51524,-3.06818],[-3.403703,7.658628,-2.648789],[-3.507113,8.00159,-2.582275],[-3.607373,8.174737,-2.401723],[-3.749043,8.378084,-2.226959],[-3.648514,8.502213,-2.6138],[-2.534199,0.904753,2.021148],[1.4083,5.744252,-0.571402],[-3.852536,8.571009,-2.352358],[2.868255,5.373126,-0.163705],[2.224363,4.669891,-1.061586],[-4.528281,4.885838,1.340274],[1.30817,4.609629,-1.28762],[-4.519698,3.422501,1.354826],[-3.549955,7.783228,-2.332859],[1.12313,6.120856,0.045115],[-3.620324,7.57716,-2.033423],[-0.798833,2.624133,-1.992682],[-3.617587,7.783148,-2.051383],[-3.669293,8.103776,-2.10227],[-3.892417,8.667436,-2.167288],[-0.537435,0.285345,-0.176267],[-0.841522,3.299866,-1.887861],[-0.761547,3.647082,-1.798953],[-3.661544,7.85708,-1.867924],[-3.886763,8.551783,-1.889171],[-0.591244,1.549749,-1.714784],[-0.775276,1.908218,-1.597609],[-0.961458,2.573273,-1.695549],[-2.215672,1.335009,2.143031],[-4.622674,4.130242,1.220683],[1.07344,0.290099,1.584734],[-0.976906,2.92171,-1.76667],[-1.13696,3.194401,-1.513455],[-3.743262,7.99949,-1.629286],[-2.876359,4.900986,-0.879556],[0.550835,3.905557,-2.031372],[0.777647,4.992314,-1.215703],[1.445881,4.266201,-1.414663],[1.274222,5.510543,-0.824495],[-0.864685,2.318581,-1.702389],[-0.627458,3.820722,-1.743153],[-3.867699,8.30866,-1.850066],[1.635287,5.45587,-0.83844],[-1.037876,2.538589,-1.513504],[-4.38993,4.73926,1.699639],[0.048709,4.765232,-1.279506],[-0.626548,1.339887,-1.595114],[-3.682827,7.643453,-1.723398],[-3.868783,8.180191,-1.511743],[-0.76988,1.508373,-1.419599],[-1.138374,2.766765,-1.448163],[1.699883,5.780752,-0.475361],[1.214305,0.308517,1.866405],[-1.713642,0.373461,-1.265204],[-1.582388,0.58294,-1.267977],[-0.879549,1.821581,-1.313787],[0.519057,5.858757,-0.381397],[-3.770989,2.449208,-0.132655],[0.087576,0.156713,-1.53616],[-0.942622,2.146534,-1.421494],[-1.026192,1.022164,-1.145423],[-0.964079,1.645473,-1.067631],[-1.109128,2.458789,-1.29106],[-1.037478,0.209489,-1.805424],[-3.724391,7.599686,-1.273458],[-3.787898,7.951792,-1.304794],[3.821677,2.165581,-0.181535],[-2.39467,0.304606,-0.570375],[-2.352928,1.0439,2.079369],[-0.288899,9.640684,-1.006079],[-3.472118,7.263001,-1.080326],[-1.240769,0.972352,-0.976446],[-1.845253,0.356801,-0.995574],[-2.32279,7.915361,-0.057477],[-1.08092,2.179315,-1.168821],[4.598833,2.156768,0.280264],[-4.725417,6.442373,2.056809],[-0.490347,9.46429,-0.981092],[-1.99652,0.09737,-0.765828],[-1.137793,1.888846,-0.894165],[-0.37247,4.29661,-1.465199],[-0.184631,5.692946,-0.421398],[-3.751694,7.742231,-1.086908],[-1.001416,1.298225,-0.904674],[-3.536884,7.190777,-0.788609],[-3.737597,7.511281,-0.940052],[-1.766651,0.669388,-0.873054],[3.112245,3.474345,-1.129672],[-0.175504,3.81298,-2.0479],[-3.766762,7.412514,-0.681569],[-0.63375,9.439424,-0.785128],[-0.518199,4.768982,-1.258625],[0.790619,4.212759,-1.610218],[-3.761951,3.742528,-0.756283],[0.897483,5.679808,-0.612423],[2.221126,4.427468,-1.252155],[-0.728577,5.846457,0.062702],[0.194451,9.503908,-1.482461],[-0.099243,9.385459,-1.39564],[0.643185,3.636855,-2.180247],[0.894522,5.900601,-0.356935],[2.595516,4.75731,-0.893245],[1.108497,3.936893,-1.905098],[1.989894,5.789726,-0.343268],[-3.802345,7.655508,-0.613817],[2.339353,4.96257,-0.90308],[0.12564,4.013324,-1.879236],[-4.078965,3.683254,-0.445439],[2.092899,5.256128,-0.831607],[0.427571,0.291769,1.272964],[2.335549,3.480056,-1.581949],[-0.15687,0.324827,-1.648922],[-0.536522,5.760786,-0.203535],[1.507082,0.078251,-0.923109],[-1.854742,0.134826,2.698774],[-3.939827,3.168498,-0.526144],[-3.98461,3.39869,-0.533212],[-3.961738,4.217132,-0.489147],[4.273789,2.181164,0.153786],[-0.470498,5.645664,-0.439079],[-0.414539,5.488017,-0.673379],[-0.097462,5.062739,-1.114863],[1.198092,5.882232,-0.391699],[2.855834,5.085022,-0.498678],[1.037998,4.129757,-1.701811],[1.728091,5.068444,-1.063761],[-3.832258,2.625141,-0.311384],[-4.078526,3.070256,-0.284362],[-4.080365,3.954243,-0.440471],[-0.152578,5.276267,-0.929815],[-1.489635,8.928082,-0.295891],[0.759294,5.15585,-1.087374],[-4.000338,2.801647,-0.235135],[-4.290801,3.823209,-0.19374],[-4.221493,4.25618,-0.189894],[-4.066195,4.71916,-0.201724],[-0.155386,4.076396,-1.662865],[3.054571,4.414305,-0.825985],[-1.652919,8.726499,-0.388504],[-3.042753,0.560068,-0.126425],[-2.434456,1.118088,-0.213563],[-2.623502,1.845062,-0.283697],[-4.233371,3.43941,-0.202918],[2.726702,3.82071,-1.280097],[0.184199,4.14639,-1.673653],[-1.289203,0.624562,-1.560929],[-3.823676,7.382458,-0.407223],[0.476667,5.064419,-1.143742],[-3.873651,4.955112,-0.269389],[1.349666,5.312227,-1.000274],[-2.043776,8.434488,-0.108891],[-2.763964,0.733395,-0.129294],[-4.380505,3.664409,-0.024546],[-0.71211,5.341811,-0.803281],[-3.960858,7.183112,-0.118407],[-3.822277,7.712853,-0.263221],[-2.346808,8.108588,0.063244],[-1.841731,8.642999,-0.142496],[-2.600055,0.985604,-0.043595],[-3.513057,2.213243,-0.044151],[-3.963492,2.603055,-0.080898],[-4.258066,3.14537,-0.027046],[-4.261572,5.00334,0.13004],[0.795464,3.99873,-1.905688],[-3.300873,0.384761,0.013271],[-2.770244,0.881942,0.077313],[-3.456227,1.993871,0.301054],[-4.441987,3.914144,0.177867],[-4.367075,6.611414,0.165312],[-3.201767,0.576292,0.105769],[-3.174354,0.645009,0.440373],[-2.996576,0.74262,0.161325],[-2.724979,1.656497,0.092983],[-3.261757,2.017742,-0.070763],[-4.280173,4.518235,-0.002999],[-4.471073,5.945358,0.05202],[-3.877137,2.40743,0.274928],[-4.371219,4.252758,0.078039],[-3.400914,0.40983,0.238599],[-4.44293,3.523242,0.146339],[-4.574528,5.279761,0.353923],[-4.226643,7.191282,0.269256],[-4.16361,2.843204,0.097727],[-4.528506,5.011661,0.536625],[0.35514,5.664802,-0.572814],[2.508711,5.580976,-0.266636],[2.556226,3.633779,-1.426362],[1.878456,4.533714,-1.223744],[2.460709,4.440241,-1.1395],[2.218589,5.514603,-0.560066],[2.263712,5.737023,-0.250694],[2.964981,3.814858,-1.139927],[0.991384,5.304131,-0.999867],[2.81187,4.547292,-0.916025],[2.918089,4.768382,-0.702808],[3.262403,4.414286,-0.657935],[0.652136,6.089113,0.069089],[3.361389,3.5052,-0.946123],[2.613042,5.037192,-0.697153],[0.094339,4.36858,-1.451238],[3.290862,4.155716,-0.732318],[2.658063,4.073614,-1.217455],[3.260349,3.753257,-0.946819],[1.124268,4.862463,-1.207855],[3.35158,4.899247,-0.027586],[3.194057,4.691257,-0.524566],[3.090119,5.116085,-0.23255],[2.418965,3.811753,-1.419399],[2.191789,3.877038,-1.47023],[4.043166,2.034188,0.015477],[-1.026966,0.86766,-1.410912],[1.937563,3.860005,-1.617465],[2.98904,4.101806,-0.998132],[-0.142611,5.865305,-0.100872],[3.972673,2.292069,0.089463],[3.23349,3.959925,-0.849829],[0.16304,5.857276,-0.216704],[4.122964,1.770061,-0.114906],[2.099057,4.978374,-0.98449],[3.502411,3.76181,-0.667502],[2.079484,5.939614,-0.036205],[-0.084568,3.525193,-2.253506],[0.423859,4.06095,-1.845327],[1.6013,6.006466,-0.153429],[0.271701,3.844964,-2.078748],[0.273577,5.218904,-0.994711],[-0.410578,3.92165,-1.773635],[1.941954,5.60041,-0.621569],[0.100825,5.462131,-0.774256],[-0.53016,3.619892,-2.027451],[-0.822371,5.517453,-0.605747],[-2.474925,7.670892,-0.020174],[4.01571,0.830194,-0.013793],[-0.400092,5.094112,-1.041992],[-2.887284,5.581246,-0.525324],[-1.559841,6.050972,0.079301],[-0.469317,3.291673,-2.235211],[0.337397,3.467926,-2.295458],[-2.632074,5.573701,-0.582717],[-0.030318,6.011395,0.276616],[-0.934373,0.388987,-1.780523],[-2.661263,5.844838,-0.425966],[0.549353,5.489646,-0.807268],[-2.194355,6.197491,-0.109322],[-2.289618,5.664813,-0.581098],[1.583583,3.796366,-1.844498],[0.855295,0.215979,-1.425557],[-2.627569,5.300236,-0.767174],[4.333347,2.384332,0.399129],[-1.880401,5.583843,-0.696561],[-2.172346,5.324859,-0.846246],[-2.27058,5.906265,-0.388373],[-1.960049,5.889346,-0.397593],[0.965756,3.67547,-2.105671],[-2.014066,6.431125,0.287254],[-1.776173,5.287097,-0.89091],[-2.025852,5.089562,-0.980218],[-1.886418,6.108358,-0.000667],[-1.600803,5.785347,-0.491069],[-1.66188,4.968053,-1.042535],[-1.600621,5.962818,-0.188044],[-1.588831,5.615418,-0.665456],[4.46901,1.880138,0.057248],[-1.978845,0.927399,-0.554856],[-1.408074,5.325266,-0.83967],[1.923123,4.843955,-1.101389],[-2.87378,0.117106,-0.412735],[-1.222193,5.62638,-0.539981],[-2.632537,0.166349,-0.489218],[-1.370865,5.838832,-0.341026],[-1.067742,5.448874,-0.692701],[-1.073798,5.220878,-0.908779],[-1.147562,4.950417,-1.079727],[-2.789115,4.531047,-1.042713],[-3.550826,4.170487,-0.806058],[-3.331694,4.798177,-0.69568],[-3.689404,4.688543,-0.534317],[-3.511509,5.106246,-0.483632],[1.796344,0.076137,0.080455],[-3.306354,5.473605,-0.478764],[-2.692503,3.346604,-1.20959],[-3.963056,5.187462,3.113156],[-3.901231,6.391477,-0.246984],[4.484234,1.518638,-0.001617],[4.308829,1.657716,-0.119275],[4.290045,1.339528,-0.110626],[-3.514938,3.524974,-0.909109],[-2.1943,2.12163,-0.71966],[4.108206,1.091087,-0.11416],[3.785312,1.392435,-0.28588],[4.092886,1.480476,-0.210655],[-2.965937,6.469006,-0.379085],[-3.708581,2.962974,-0.63979],[-3.297971,2.218917,-0.299872],[3.806949,0.804703,-0.11438],[3.747957,1.059258,-0.273069],[-3.101827,4.111444,-1.006255],[-1.536445,4.658913,-1.195049],[-3.549826,2.450555,-0.375694],[-3.676495,2.108366,0.534323],[-3.674738,5.925075,-0.400011],[-2.250115,2.848335,-1.121174],[-3.698062,5.667567,-0.381396],[3.468966,0.734643,-0.190624],[-3.97972,5.670078,-0.26874],[-3.002087,4.337837,-1.033421],[-3.356392,2.608308,-0.713323],[-1.833016,3.359983,-1.28775],[-1.989069,3.632416,-1.305607],[3.591254,0.542371,0.026146],[3.364927,1.082572,-0.342613],[-3.393759,3.866801,-0.937266],[-4.124865,5.549529,-0.161729],[-4.423423,5.687223,0.000103],[-1.496881,2.601785,-1.114328],[-2.642297,6.496932,-0.264175],[-3.684236,6.819423,-0.320233],[-2.286996,3.167067,-1.246651],[-1.624896,8.44848,-0.530014],[-3.666787,2.159266,0.268149],[-2.402625,2.011243,-0.56446],[-2.736166,2.259839,-0.6943],[-2.168611,3.89078,-1.292206],[-2.065956,3.345708,-1.281346],[-2.778147,2.675605,-0.995706],[-3.507431,4.513272,-0.71829],[-2.301184,4.293911,-1.238182],[3.205808,0.211078,0.394349],[-2.129936,4.870577,-1.080781],[-2.287977,2.496593,-0.934069],[-2.701833,2.931814,-1.114509],[3.294795,0.50631,-0.081062],[-2.552829,7.468771,-0.021541],[3.06721,0.944066,-0.43074],[-2.86086,1.973622,-0.303132],[-3.598818,5.419613,-0.401645],[-1.524381,0.080156,-1.61662],[-1.907291,2.646274,-1.039438],[2.950783,0.407562,-0.105407],[-1.663048,1.655038,-0.689787],[-1.728102,1.110064,-0.635963],[-2.085823,7.686296,-0.159745],[2.883518,3.157009,-1.30858],[-2.724116,0.417169,-0.389719],[-1.788636,7.862672,-0.346413],[-2.186418,1.249609,-0.434583],[-3.092434,2.606657,-0.860002],[-1.737314,3.874201,-1.330986],[2.564522,0.422967,-0.390903],[1.670782,3.538432,-1.924753],[-2.338131,4.02578,-1.286673],[-1.916516,4.054121,-1.301788],[2.87159,2.034949,-1.267139],[-1.931518,3.062883,-1.197227],[-0.816602,0.135682,3.104104],[0.469392,0.213916,-1.489608],[2.574055,1.950091,-1.514427],[2.733595,2.682546,-1.461213],[-1.915407,4.693647,-1.151721],[-3.412883,5.867094,-0.450528],[2.28822,0.120432,-0.04102],[2.244477,0.14424,-0.376933],[-1.676198,3.570698,-1.328031],[-1.821193,4.366982,-1.266271],[-1.552208,8.099221,-0.53262],[-1.727419,2.39097,-0.989456],[-2.468226,4.711663,-1.069766],[-2.451669,6.113319,-0.273788],[2.635447,2.295842,-1.518361],[-2.020809,8.150253,-0.246714],[2.292455,0.805596,-1.3042],[2.641556,1.65665,-1.466962],[2.409062,2.842538,-1.635025],[2.456682,1.459484,-1.57543],[-1.691047,3.173582,-1.247082],[-1.865642,1.957608,-0.768683],[-3.401579,0.20407,0.100932],[2.301981,1.7102,-1.650461],[2.342929,2.611944,-1.690713],[-1.676111,2.923894,-1.17835],[-2.992039,3.547631,-1.118945],[-3.571677,6.504634,-0.375455],[2.141764,1.460869,-1.702464],[-3.221958,5.146049,-0.615632],[2.19238,2.949367,-1.747242],[2.320791,2.232971,-1.706842],[2.088678,2.585235,-1.813159],[-2.196404,0.592218,-0.569709],[-2.120811,1.836483,-0.62338],[-1.949935,2.271249,-0.874128],[2.235901,1.110183,-1.510719],[2.020157,3.241128,-1.803917],[2.054336,1.949394,-1.792332],[-3.094117,4.996595,-0.740238],[2.038063,0.635949,-1.402041],[1.980644,1.684408,-1.76778],[1.587432,3.306542,-1.991131],[1.935322,0.976267,-1.602208],[1.922621,1.235522,-1.698813],[1.712495,1.911874,-1.903234],[1.912802,2.259273,-1.888698],[1.884367,0.355453,-1.312633],[1.676427,0.76283,-1.539455],[1.78453,2.83662,-1.943035],[1.697312,0.120281,-1.150324],[1.648318,2.484973,-1.999505],[-4.051804,5.958472,-0.231731],[-1.964823,1.464607,-0.58115],[1.55996,2.183486,-1.971378],[1.628125,1.045912,-1.707832],[1.701684,1.540428,-1.827156],[1.567475,4.869481,-1.184665],[1.432492,0.843779,-1.648083],[1.173837,2.978983,-2.156687],[1.235287,3.37975,-2.09515],[1.252589,1.525293,-1.949205],[1.159334,2.336379,-2.105361],[1.49061,2.695263,-2.083216],[-4.122486,6.782604,-0.02545],[1.173388,0.279193,-1.423418],[1.505684,0.380815,-1.414395],[1.391423,1.343031,-1.843557],[1.263449,2.73225,-2.144961],[1.295858,0.597122,-1.515628],[1.245851,3.729126,-1.993015],[-2.761439,6.23717,-0.365856],[0.978887,1.664888,-2.046633],[1.219542,0.982729,-1.785486],[1.315915,1.91748,-2.02788],[-3.052746,2.127222,-0.369082],[0.977656,1.36223,-1.944119],[0.936122,3.39447,-2.203007],[-2.740036,4.184702,-1.122849],[0.853581,2.864694,-2.260847],[0.719569,0.818762,-1.763618],[0.839115,1.159359,-1.907943],[0.932069,1.94559,-2.117962],[0.579321,3.326747,-2.299369],[0.86324,0.597822,-1.565106],[0.574567,1.158452,-1.943123],[0.525138,2.137252,-2.213867],[0.779941,2.342019,-2.206157],[0.915255,2.618102,-2.209041],[0.526426,3.02241,-2.321826],[0.495431,2.521396,-2.295905],[0.80799,3.156817,-2.286432],[0.273556,1.304936,-2.012509],[0.664326,1.530024,-2.048722],[0.219173,2.32907,-2.323212],[0.405324,0.695359,-1.704884],[0.398827,0.946649,-1.843899],[0.345109,1.608829,-2.100174],[-2.356743,0.062032,-0.4947],[-3.001084,0.27146,2.560034],[-2.064663,0.303055,-0.697324],[0.221271,3.174023,-2.374399],[0.195842,0.437865,-1.621473],[-0.385613,0.297763,1.960096],[1.999609,0.108928,-0.79125],[0.351698,9.227494,-1.57565],[0.021477,2.191913,-2.309353],[0.246381,2.836575,-2.356365],[1.543281,0.237539,1.901906],[0.031881,9.147022,-1.454203],[-0.001881,1.648503,-2.108044],[0.333423,1.907088,-2.204533],[0.044063,2.634032,-2.368412],[-0.028148,3.053684,-2.390082],[0.02413,3.34297,-2.36544],[-0.272645,9.02879,-1.238685],[-0.006348,0.832044,-1.758222],[-0.321105,1.458754,-1.886313],[-0.153948,8.618809,-1.105353],[-0.409303,1.137783,-1.720556],[-0.410054,1.742789,-1.957989],[-0.287905,2.380404,-2.294509],[-0.261375,2.646629,-2.356322],[-0.221986,3.215303,-2.345844],[-0.31608,0.687581,-1.71901],[-0.537705,0.855802,-1.648585],[-0.142834,1.193053,-1.87371],[-0.24371,2.044435,-2.176958],[-0.437999,2.959748,-2.299698],[-0.78895,0.176226,-1.729046],[-0.608509,0.546932,-1.734032],[-0.693698,4.478782,-1.369372],[-0.669153,8.469645,-0.911149],[-0.741857,1.082705,-1.458474],[-0.554059,2.440325,-2.141785],[2.09261,0.153182,2.57581],[1.792547,0.111794,2.563777],[1.855787,0.189541,2.835089],[1.492601,0.232246,2.987681],[-0.284918,0.236687,3.429738],[2.604841,0.11997,1.01506],[0.331271,0.168113,3.124031],[0.280606,0.308368,2.495937],[0.544591,0.325711,2.081274],[0.193145,0.19154,-0.977556],[3.810099,0.42324,1.032202],[3.54622,0.379245,1.392814],[0.61402,0.276328,0.849356],[-1.198628,0.144953,2.911457],[4.17199,0.68037,1.391526],[0.88279,0.321339,2.059129],[1.93035,0.109992,2.054154],[1.620331,0.121986,2.37203],[2.374812,0.10921,1.734876],[-0.031227,0.294412,2.593687],[4.075018,0.561914,1.038065],[-0.570366,0.126583,2.975558],[0.950052,0.318463,1.804012],[1.130034,0.117125,0.98385],[2.123049,0.08946,1.665911],[2.087572,0.068621,0.335013],[2.927337,0.167117,0.289611],[0.528876,0.313434,3.205969],[1.174911,0.162744,1.328262],[-4.88844,5.59535,1.661134],[-4.709607,5.165338,1.324082],[0.871199,0.277021,1.263831],[-3.910877,2.349318,1.272269],[1.56824,0.118605,2.768112],[1.179176,0.152617,-0.858003],[1.634629,0.247872,2.128625],[-4.627425,5.126935,1.617836],[3.845542,0.54907,1.45601],[2.654006,0.165508,1.637169],[-0.678324,0.26488,1.974741],[2.451139,0.100377,0.213768],[0.633199,0.286719,0.403357],[-0.533042,0.2524,1.373267],[0.99317,0.171106,0.624966],[-0.100063,0.306466,2.170225],[1.245943,0.092351,0.661031],[1.390414,0.198996,-0.0864],[-4.457265,5.030531,2.138242],[2.89776,0.146575,1.297468],[1.802703,0.088824,-0.490405],[1.055447,0.309261,2.392437],[2.300436,0.142429,2.104254],[2.33399,0.187756,2.416935],[2.325183,0.134349,0.574063],[2.410924,0.370971,2.637115],[1.132924,0.290511,3.061],[1.764028,0.070212,-0.80535],[2.156994,0.397657,2.844061],[0.920711,0.225527,-0.882456],[-4.552135,5.24096,2.85514],[0.210016,0.309396,2.064296],[0.612067,0.136815,-1.086002],[3.150236,0.426757,1.802703],[-0.24824,0.282258,1.470997],[0.974269,0.301311,-0.640898],[-4.401413,5.03966,2.535553],[0.644319,0.274006,-0.817806],[0.332922,0.309077,0.108474],[3.610001,0.317447,0.689353],[3.335681,0.358195,0.118477],[0.623544,0.318983,-0.4193],[-0.11012,0.307747,1.831331],[-0.407528,0.291044,2.282935],[0.069783,0.285095,0.950289],[0.970135,0.310392,-0.283742],[0.840564,0.306898,0.098854],[-0.541827,0.267753,1.683795],[-3.956082,4.55713,2.297164],[-4.161036,2.834481,1.64183],[-4.093952,4.977551,2.747747],[2.661819,0.261867,1.926145],[-3.749926,2.161875,0.895238],[-2.497776,1.3629,0.791855],[0.691482,0.304968,1.582939],[-4.013193,4.830963,2.4769],[-3.639585,2.091265,1.304415],[-3.9767,2.563053,1.6284],[-3.979915,2.788616,1.977977],[0.388782,0.312656,1.709168],[-3.40873,1.877324,0.851652],[-3.671637,5.136974,3.170734],[-3.12964,1.852012,0.157682],[-3.629687,4.852698,2.686837],[-3.196164,1.793459,0.452804],[-3.746338,2.31357,1.648551],[2.992192,0.125251,0.575976],[-3.254051,0.054431,0.314152],[-3.474644,1.925288,1.134116],[-3.418372,2.022882,1.578901],[-2.920955,1.705403,0.29842],[-3.57229,2.152022,1.607572],[-3.251259,0.09013,-0.106174],[-3.299952,1.877781,1.348623],[-3.666819,2.441459,2.004838],[-2.912646,1.824748,-0.045348],[-3.399511,2.479484,2.340393],[-3.009754,0.015286,0.075567],[-3.381443,2.316937,2.156923],[-3.352801,2.133341,1.857366],[-3.01788,1.687685,0.645867],[-2.931857,1.678712,1.158472],[-3.301008,0.08836,0.591001],[1.358025,0.19795,1.599144],[-2.999565,1.845016,1.618396],[-2.767957,0.028397,-0.196436],[-2.93962,2.078779,2.140593],[-3.346648,2.674056,2.518097],[3.324322,0.20822,0.628605],[3.091677,0.137202,0.9345],[-2.881807,0.009952,0.318439],[-2.764946,1.786619,1.693439],[-2.905542,1.932343,1.900002],[-3.140854,2.271384,2.274946],[-2.88995,2.487856,2.574759],[-2.367194,-0.000943,-0.15576],[-3.050738,0.068703,0.742988],[-2.759525,1.55679,0.877782],[-3.151775,2.48054,2.482749],[-2.578618,-0.002885,0.165716],[-2.651618,1.877246,1.981189],[-2.933973,0.133731,1.631023],[1.047628,0.100284,-1.085248],[-1.585123,0.062083,-1.394896],[-2.287917,-0.002671,0.214434],[-2.524899,0.007481,0.471788],[-2.815492,2.188198,2.343294],[-2.095142,-0.003149,-0.094574],[-2.172686,-0.000133,0.47963],[-2.732704,0.074306,1.742079],[-2.49653,2.145668,2.42691],[-1.343683,0.047721,-1.506391],[-2.581185,0.048703,0.975528],[-2.905101,0.083158,2.010052],[-2.601514,2.007801,2.223089],[-2.339464,0.02634,1.484304],[-2.907873,0.10367,2.378149],[-1.368796,0.062516,-1.049125],[-1.93244,0.02443,-0.427603],[-2.705081,0.060513,2.303802],[3.372155,0.206274,0.892293],[-1.761827,0.093202,-1.037404],[-1.700667,0.0397,-0.614221],[-1.872291,0.011979,-0.135753],[-1.929257,0.074005,0.728999],[-2.520128,0.049665,1.99054],[-2.699411,0.10092,2.603116],[3.211701,0.27302,1.423357],[-1.445362,0.1371,-0.626491],[2.921332,0.259112,1.645525],[-0.993242,0.058686,-1.408916],[-0.944986,0.157541,-1.097665],[-2.154301,0.032749,1.882001],[-2.108789,1.988557,2.442673],[-1.015659,0.25497,-0.416665],[-1.898411,0.015872,0.16715],[-1.585517,0.027121,0.453445],[-2.311105,0.061264,2.327061],[-2.637042,0.152224,2.832201],[-2.087515,2.292972,2.617585],[-0.750611,0.056697,-1.504516],[-0.472029,0.075654,-1.360203],[-0.710798,0.139244,-1.183863],[-0.97755,0.26052,-0.831167],[-0.655814,0.260843,-0.880068],[-0.897513,0.275537,-0.133042],[-2.049194,0.084947,2.455422],[-0.177837,0.076362,-1.449009],[-0.553393,0.279083,-0.59573],[-1.788636,0.06163,2.231198],[-0.34761,0.255578,-0.999614],[-1.398589,0.036482,0.65871],[-1.133918,0.05617,0.69473],[-1.43369,0.058226,1.977865],[-2.505459,1.492266,1.19295]]
exports.cells=[[2,1661,3],[1676,7,6],[712,1694,9],[3,1674,1662],[11,1672,0],[1705,0,1],[5,6,1674],[4,5,1674],[7,8,712],[2,1662,10],[1,10,1705],[11,1690,1672],[1705,11,0],[5,1676,6],[7,9,6],[7,712,9],[2,3,1662],[3,4,1674],[1,2,10],[12,82,1837],[1808,12,1799],[1808,1799,1796],[12,861,82],[861,1808,13],[1808,861,12],[1799,12,1816],[1680,14,1444],[15,17,16],[14,1678,1700],[16,17,1679],[15,1660,17],[14,1084,1678],[15,1708,18],[15,18,1660],[1680,1084,14],[1680,15,1084],[15,1680,1708],[793,813,119],[1076,793,119],[1076,1836,22],[23,19,20],[21,1076,22],[21,22,23],[23,20,21],[1076,119,1836],[806,634,470],[432,1349,806],[251,42,125],[809,1171,791],[953,631,827],[634,1210,1176],[157,1832,1834],[56,219,53],[126,38,83],[37,85,43],[59,1151,1154],[83,75,41],[77,85,138],[201,948,46],[1362,36,37],[452,775,885],[1237,95,104],[966,963,1262],[85,77,43],[36,85,37],[1018,439,1019],[41,225,481],[85,83,127],[93,83,41],[935,972,962],[116,93,100],[98,82,813],[41,75,225],[298,751,54],[1021,415,1018],[77,138,128],[766,823,1347],[593,121,573],[905,885,667],[786,744,747],[100,41,107],[604,334,765],[779,450,825],[968,962,969],[225,365,481],[365,283,196],[161,160,303],[875,399,158],[328,1817,954],[62,61,1079],[358,81,72],[74,211,133],[160,161,138],[91,62,1079],[167,56,1405],[56,167,219],[913,914,48],[344,57,102],[43,77,128],[1075,97,1079],[389,882,887],[219,108,53],[1242,859,120],[604,840,618],[754,87,762],[197,36,1362],[1439,88,1200],[1652,304,89],[81,44,940],[445,463,151],[717,520,92],[129,116,100],[1666,1811,624],[1079,97,91],[62,91,71],[688,898,526],[463,74,133],[278,826,99],[961,372,42],[799,94,1007],[100,93,41],[1314,943,1301],[184,230,109],[875,1195,231],[133,176,189],[751,755,826],[101,102,57],[1198,513,117],[748,518,97],[1145,1484,1304],[358,658,81],[971,672,993],[445,151,456],[252,621,122],[36,271,126],[85,36,126],[116,83,93],[141,171,1747],[1081,883,103],[1398,1454,149],[457,121,593],[127,116,303],[697,70,891],[457,891,1652],[1058,1668,112],[518,130,97],[214,319,131],[185,1451,1449],[463,133,516],[1428,123,177],[113,862,561],[215,248,136],[186,42,251],[127,83,116],[160,85,127],[162,129,140],[154,169,1080],[169,170,1080],[210,174,166],[1529,1492,1524],[450,875,231],[399,875,450],[171,141,170],[113,1155,452],[131,319,360],[44,175,904],[452,872,113],[746,754,407],[147,149,150],[309,390,1148],[53,186,283],[757,158,797],[303,129,162],[429,303,162],[154,168,169],[673,164,193],[38,271,75],[320,288,1022],[246,476,173],[175,548,904],[182,728,456],[199,170,169],[168,199,169],[199,171,170],[184,238,230],[246,247,180],[1496,1483,1467],[147,150,148],[828,472,445],[53,108,186],[56,53,271],[186,961,42],[1342,391,57],[1664,157,1834],[1070,204,178],[178,204,179],[285,215,295],[692,55,360],[192,193,286],[359,673,209],[586,195,653],[121,89,573],[202,171,199],[238,515,311],[174,210,240],[174,105,166],[717,276,595],[1155,1149,452],[1405,56,197],[53,283,30],[75,53,30],[45,235,1651],[210,166,490],[181,193,192],[185,620,217],[26,798,759],[1070,226,204],[220,187,179],[220,168,187],[202,222,171],[359,209,181],[182,456,736],[964,167,1405],[76,250,414],[807,1280,1833],[70,883,1652],[227,179,204],[221,199,168],[221,202,199],[360,494,131],[214,241,319],[105,247,166],[205,203,260],[388,480,939],[482,855,211],[8,807,1833],[226,255,204],[228,221,168],[166,173,490],[701,369,702],[211,855,262],[631,920,630],[1448,1147,1584],[255,227,204],[237,220,179],[228,168,220],[222,256,555],[215,259,279],[126,271,38],[108,50,186],[227,236,179],[236,237,179],[220,237,228],[228,202,221],[256,222,202],[555,256,229],[259,152,279],[27,1296,31],[186,50,961],[961,234,372],[1651,235,812],[1572,1147,1448],[255,226,1778],[255,236,227],[256,257,229],[106,184,109],[241,410,188],[177,578,620],[209,673,181],[1136,1457,79],[1507,245,718],[255,273,236],[275,410,241],[206,851,250],[1459,253,1595],[1406,677,1650],[228,274,202],[202,281,256],[348,239,496],[205,172,203],[369,248,702],[261,550,218],[261,465,550],[574,243,566],[921,900,1220],[291,273,255],[348,238,265],[109,230,194],[149,380,323],[443,270,421],[272,291,255],[274,228,237],[274,292,202],[281,257,256],[276,543,341],[152,259,275],[1111,831,249],[632,556,364],[299,273,291],[299,236,273],[280,237,236],[202,292,281],[247,246,173],[282,49,66],[1620,1233,1553],[299,280,236],[280,305,237],[237,305,274],[306,292,274],[330,257,281],[246,194,264],[166,247,173],[912,894,896],[611,320,244],[1154,1020,907],[969,962,290],[272,299,291],[305,318,274],[145,212,240],[164,248,285],[259,277,275],[193,164,295],[269,240,210],[1033,288,320],[46,948,206],[336,280,299],[330,281,292],[257,307,300],[369,136,248],[145,240,269],[502,84,465],[193,295,286],[164,285,295],[282,302,49],[161,303,429],[318,306,274],[306,330,292],[315,257,330],[315,307,257],[307,352,300],[300,352,308],[275,277,403],[353,1141,333],[1420,425,47],[611,313,320],[85,126,83],[128,1180,43],[303,116,129],[280,314,305],[314,318,305],[190,181,242],[203,214,131],[820,795,815],[322,299,272],[322,336,299],[315,339,307],[172,152,617],[172,214,203],[321,1033,320],[1401,941,946],[85,160,138],[976,454,951],[747,60,786],[317,322,272],[339,352,307],[266,33,867],[163,224,218],[247,614,180],[648,639,553],[388,172,205],[611,345,313],[313,345,320],[160,127,303],[454,672,951],[317,329,322],[314,280,336],[306,338,330],[330,339,315],[1236,115,436],[342,321,320],[1046,355,328],[328,346,325],[325,346,317],[367,314,336],[314,337,318],[337,306,318],[338,343,330],[342,320,345],[355,349,328],[346,329,317],[347,336,322],[314,362,337],[330,343,339],[340,308,352],[135,906,1022],[239,156,491],[194,230,486],[40,1015,1003],[321,355,1046],[329,382,322],[382,347,322],[347,367,336],[337,371,306],[306,371,338],[1681,296,1493],[286,172,388],[230,348,486],[348,183,486],[384,332,830],[328,349,346],[367,362,314],[371,343,338],[339,351,352],[57,344,78],[342,355,321],[386,346,349],[386,350,346],[346,350,329],[347,366,367],[343,363,339],[323,380,324],[152,275,241],[345,1045,342],[350,374,329],[339,363,351],[234,340,352],[353,361,354],[40,34,1015],[373,355,342],[373,349,355],[374,382,329],[366,347,382],[371,363,343],[351,379,352],[379,372,352],[372,234,352],[156,190,491],[319,241,692],[354,361,31],[366,377,367],[363,379,351],[133,590,516],[197,56,271],[1045,370,342],[370,373,342],[374,350,386],[377,366,382],[367,395,362],[400,337,362],[400,371,337],[378,363,371],[106,109,614],[181,673,193],[953,920,631],[376,349,373],[376,386,349],[378,379,363],[224,375,218],[279,152,172],[361,619,381],[1347,823,795],[760,857,384],[392,374,386],[394,395,367],[383,371,400],[383,378,371],[218,375,261],[197,271,36],[414,454,976],[385,376,373],[1051,382,374],[387,394,367],[377,387,367],[395,400,362],[279,172,295],[30,365,225],[450,231,825],[385,373,370],[398,374,392],[1051,377,382],[396,378,383],[348,496,183],[295,172,286],[357,269,495],[1148,390,1411],[75,30,225],[206,76,54],[412,386,376],[412,392,386],[396,383,400],[651,114,878],[123,1241,506],[238,311,265],[381,653,29],[618,815,334],[427,1032,411],[298,414,976],[791,332,384],[129,100,140],[412,404,392],[392,404,398],[140,107,360],[395,394,400],[423,379,378],[385,412,376],[406,94,58],[419,415,1021],[422,423,378],[423,125,379],[258,508,238],[311,156,265],[213,287,491],[449,411,1024],[412,1068,404],[55,140,360],[76,414,54],[394,416,400],[400,416,396],[422,378,396],[1258,796,789],[427,411,449],[427,297,1032],[1385,1366,483],[417,448,284],[1507,341,245],[162,140,444],[658,44,81],[433,125,423],[438,251,125],[429,162,439],[1342,57,1348],[765,766,442],[697,891,695],[1057,396,416],[440,423,422],[440,433,423],[433,438,125],[438,196,251],[74,482,211],[1136,79,144],[29,195,424],[242,1004,492],[57,757,28],[414,298,54],[238,348,230],[224,163,124],[295,215,279],[495,269,490],[449,446,427],[446,297,427],[1020,1163,909],[128,138,419],[66,980,443],[415,439,1018],[111,396,1057],[111,422,396],[840,249,831],[593,664,596],[218,550,155],[109,194,180],[483,268,855],[161,415,419],[1737,232,428],[360,107,494],[1006,1011,410],[444,140,55],[919,843,430],[190,242,213],[275,403,410],[131,494,488],[449,663,446],[138,161,419],[128,419,34],[439,162,444],[460,440,422],[440,438,433],[472,74,445],[491,190,213],[238,508,515],[46,206,54],[972,944,962],[1241,1428,1284],[111,460,422],[470,432,806],[248,164,702],[1025,467,453],[553,1235,648],[263,114,881],[267,293,896],[469,438,440],[455,196,438],[287,242,492],[239,265,156],[213,242,287],[1684,746,63],[663,474,446],[415,161,429],[140,100,107],[1055,459,467],[469,455,438],[259,542,277],[446,474,466],[446,466,447],[439,444,1019],[614,109,180],[190,359,181],[156,497,190],[726,474,663],[1023,458,459],[461,440,460],[269,210,490],[246,180,194],[590,133,189],[163,218,155],[467,468,453],[1063,1029,111],[111,1029,460],[1029,464,460],[461,469,440],[150,149,323],[828,445,456],[375,502,261],[474,475,466],[573,426,462],[478,1023,477],[478,458,1023],[458,479,467],[459,458,467],[468,393,453],[464,461,460],[484,365,455],[1232,182,1380],[172,617,214],[547,694,277],[542,547,277],[184,258,238],[261,502,465],[467,479,468],[484,455,469],[1380,182,864],[475,476,466],[80,447,476],[466,476,447],[415,429,439],[479,487,468],[487,287,468],[492,393,468],[260,469,461],[481,365,484],[531,473,931],[692,360,319],[726,495,474],[468,287,492],[480,464,1029],[260,461,464],[494,481,484],[74,472,482],[174,240,212],[223,106,614],[486,477,485],[478,496,458],[491,487,479],[123,402,177],[488,469,260],[488,484,469],[265,239,348],[248,215,285],[474,490,475],[477,486,478],[458,496,479],[239,491,479],[1584,1147,1334],[488,494,484],[401,123,506],[495,490,474],[490,173,475],[80,476,264],[491,287,487],[480,1029,1004],[480,205,464],[173,476,475],[485,194,486],[486,183,478],[478,183,496],[496,239,479],[848,1166,60],[268,262,855],[205,260,464],[260,203,488],[203,131,488],[246,264,476],[194,485,264],[1002,310,1664],[311,515,497],[515,359,497],[565,359,515],[1250,1236,301],[736,456,151],[654,174,567],[577,534,648],[519,505,645],[725,565,508],[150,1723,148],[584,502,505],[584,526,502],[502,526,84],[607,191,682],[560,499,660],[607,517,191],[1038,711,124],[951,672,971],[716,507,356],[868,513,1198],[615,794,608],[682,191,174],[1313,928,1211],[617,241,214],[511,71,91],[408,800,792],[192,286,525],[80,485,447],[91,97,130],[1675,324,888],[207,756,532],[582,1097,1124],[311,497,156],[510,130,146],[523,511,510],[608,708,616],[546,690,650],[511,527,358],[536,146,518],[465,418,550],[418,709,735],[520,514,500],[584,505,519],[536,518,509],[146,536,510],[538,527,511],[876,263,669],[646,524,605],[510,536,523],[527,175,358],[724,876,669],[721,724,674],[524,683,834],[558,509,522],[558,536,509],[523,538,511],[611,243,574],[528,706,556],[668,541,498],[523,537,538],[527,540,175],[532,756,533],[1013,60,747],[551,698,699],[92,520,500],[535,536,558],[536,569,523],[538,540,527],[539,548,175],[567,212,145],[401,896,293],[534,675,639],[1510,595,1507],[557,545,530],[569,536,535],[537,540,538],[540,539,175],[569,537,523],[1135,718,47],[587,681,626],[580,535,558],[99,747,278],[701,565,725],[665,132,514],[665,514,575],[132,549,653],[176,651,189],[65,47,266],[597,569,535],[569,581,537],[537,581,540],[563,539,540],[539,564,548],[1509,1233,1434],[132,653,740],[550,710,155],[714,721,644],[410,1011,188],[732,534,586],[560,562,729],[555,557,222],[580,558,545],[597,535,580],[581,563,540],[5,821,1676],[576,215,136],[649,457,741],[564,539,563],[124,711,224],[550,668,710],[550,541,668],[565,701,673],[560,613,499],[233,532,625],[545,555,580],[601,581,569],[594,904,548],[1463,1425,434],[185,149,1454],[721,674,644],[185,380,149],[577,424,586],[462,586,559],[597,601,569],[594,548,564],[566,603,574],[165,543,544],[457,89,121],[586,424,195],[725,587,606],[1078,582,1124],[588,925,866],[462,559,593],[189,878,590],[555,229,580],[602,563,581],[904,594,956],[434,1425,1438],[1024,112,821],[572,587,626],[600,597,580],[599,591,656],[600,580,229],[601,622,581],[581,622,602],[602,564,563],[602,594,564],[603,611,574],[498,529,546],[697,1145,70],[592,628,626],[610,597,600],[597,610,601],[222,557,171],[604,765,799],[573,462,593],[133,200,176],[729,607,627],[1011,692,188],[518,146,130],[585,687,609],[682,627,607],[1712,599,656],[562,592,607],[643,656,654],[257,600,229],[601,633,622],[623,594,602],[174,212,567],[725,606,701],[609,701,606],[610,633,601],[633,642,622],[380,216,324],[142,143,1249],[501,732,586],[534,577,586],[648,1235,577],[610,641,633],[310,1002,1831],[618,334,604],[1710,145,269],[707,498,659],[501,586,462],[625,501,462],[726,663,691],[300,600,257],[641,610,600],[622,629,602],[602,629,623],[55,692,444],[518,748,509],[929,1515,1411],[620,578,267],[71,511,358],[707,668,498],[650,687,585],[600,300,641],[641,657,633],[1675,888,1669],[622,636,629],[505,502,375],[541,529,498],[332,420,1053],[637,551,638],[534,639,648],[69,623,873],[300,512,641],[633,657,642],[562,660,579],[687,637,638],[709,646,605],[775,738,885],[559,549,132],[646,683,524],[641,512,657],[266,897,949],[1712,643,1657],[184,727,258],[674,724,669],[699,714,647],[628,659,572],[657,662,642],[571,881,651],[517,607,504],[598,706,528],[598,694,547],[640,552,560],[655,693,698],[698,693,721],[91,510,511],[144,301,1136],[324,216,888],[870,764,1681],[575,514,520],[276,544,543],[658,175,44],[645,505,711],[659,546,572],[700,524,655],[605,700,529],[266,867,897],[1695,1526,764],[579,659,628],[654,591,682],[586,549,559],[698,721,714],[896,401,506],[640,734,599],[664,665,575],[621,629,636],[1712,656,643],[547,644,598],[710,668,707],[640,560,734],[655,698,551],[694,528,277],[512,662,657],[504,592,626],[688,584,519],[152,241,617],[587,725,681],[598,669,706],[526,670,84],[598,528,694],[710,707,499],[579,592,562],[660,659,579],[323,324,1134],[326,895,473],[195,29,653],[84,670,915],[560,660,562],[504,626,681],[711,505,224],[651,881,114],[216,620,889],[1362,678,197],[493,99,48],[1659,691,680],[529,690,546],[430,843,709],[655,524,693],[174,191,105],[674,669,598],[98,712,82],[572,546,585],[72,61,71],[912,911,894],[106,223,184],[664,132,665],[843,646,709],[635,699,136],[699,698,714],[593,132,664],[688,526,584],[185,177,620],[533,675,534],[687,638,635],[1652,89,457],[896,506,912],[132,740,514],[689,685,282],[691,449,680],[48,436,493],[136,699,647],[739,640,554],[549,586,653],[532,533,625],[1530,695,649],[653,381,619],[736,151,531],[188,692,241],[177,402,578],[33,689,867],[689,33,685],[593,559,132],[949,65,266],[711,1038,661],[939,480,1004],[609,369,701],[616,552,615],[619,361,740],[151,463,516],[513,521,117],[691,663,449],[186,251,196],[333,302,327],[613,560,552],[616,613,552],[690,551,637],[660,707,659],[704,208,1203],[418,735,550],[163,708,124],[524,834,693],[554,640,599],[245,341,165],[565,673,359],[155,710,708],[105,191,517],[1515,198,1411],[1709,554,599],[60,289,786],[838,1295,1399],[533,534,625],[710,499,708],[556,632,410],[217,620,216],[591,627,682],[504,503,223],[643,654,567],[690,637,650],[545,557,555],[174,654,682],[719,691,1659],[727,681,508],[645,711,661],[794,615,739],[565,515,508],[282,685,302],[1150,397,1149],[638,699,635],[544,685,33],[719,726,691],[1742,1126,1733],[1724,1475,148],[556,410,403],[185,217,380],[503,504,681],[277,556,403],[32,1178,158],[1712,1709,599],[605,529,541],[635,136,369],[687,635,369],[529,700,690],[700,551,690],[89,304,573],[625,534,732],[730,302,685],[503,681,727],[702,673,701],[730,327,302],[327,353,333],[596,664,575],[660,499,707],[585,546,650],[560,729,734],[700,655,551],[176,571,651],[517,504,223],[730,685,544],[1661,1682,726],[1682,495,726],[1250,301,917],[605,524,700],[609,687,369],[516,389,895],[1553,686,1027],[673,702,164],[656,591,654],[520,596,575],[402,123,401],[828,456,728],[1645,677,1653],[528,556,277],[638,551,699],[190,497,359],[276,730,544],[1117,1525,933],[1027,686,1306],[155,708,163],[709,605,541],[647,644,547],[650,637,687],[599,734,591],[578,293,267],[1682,357,495],[510,91,130],[734,729,627],[576,542,215],[709,541,735],[735,541,550],[276,500,730],[500,327,730],[653,619,740],[414,851,454],[734,627,591],[729,562,607],[615,552,640],[525,181,192],[308,512,300],[223,503,727],[266,165,33],[92,500,276],[321,1046,1033],[585,609,606],[1200,1559,86],[628,572,626],[301,436,803],[714,644,647],[708,499,613],[721,693,724],[514,353,327],[353,740,361],[344,158,78],[708,613,616],[615,640,739],[500,514,327],[514,740,353],[1449,177,185],[462,233,625],[851,405,1163],[608,616,615],[647,542,576],[625,732,501],[1097,582,1311],[1235,424,577],[579,628,592],[607,592,504],[24,432,470],[105,614,247],[104,742,471],[542,259,215],[365,196,455],[1420,47,65],[223,727,184],[547,542,647],[572,585,606],[587,572,606],[262,780,1370],[647,576,136],[644,674,598],[271,53,75],[727,508,258],[471,742,142],[505,375,224],[357,1710,269],[725,508,681],[659,498,546],[743,1178,32],[1195,634,231],[1176,24,470],[743,1110,1178],[135,809,857],[63,746,407],[634,1176,470],[159,1112,27],[1176,1685,24],[399,450,779],[1178,856,875],[751,744,54],[436,48,772],[634,1108,1210],[769,1285,1286],[751,298,755],[746,1684,754],[754,924,87],[722,1625,756],[87,839,153],[489,795,820],[758,808,1518],[839,840,153],[831,1111,959],[1111,749,959],[810,1253,1363],[1247,1394,713],[1388,1329,1201],[1242,120,761],[857,791,384],[758,1523,808],[296,764,1504],[70,1652,891],[207,233,1638],[1348,57,28],[858,420,332],[964,1379,1278],[420,1194,816],[784,1076,1186],[1076,21,1186],[1710,767,1],[849,822,778],[806,137,787],[786,790,744],[790,54,744],[771,63,407],[785,852,818],[774,1823,272],[895,151,516],[135,1022,809],[99,826,48],[48,826,755],[808,705,408],[833,441,716],[1733,743,32],[1385,836,852],[772,827,737],[1005,49,781],[793,1697,813],[1518,441,1537],[1139,1132,859],[782,801,770],[1510,1530,676],[770,814,835],[231,787,825],[207,722,756],[26,771,798],[782,863,865],[832,54,790],[865,842,507],[799,765,94],[1175,1261,1353],[800,408,805],[262,986,200],[792,800,814],[801,792,770],[704,1203,1148],[356,1514,822],[165,544,33],[561,776,113],[1043,738,775],[815,831,820],[773,792,801],[772,48,914],[772,737,803],[436,772,803],[808,817,705],[1624,822,1527],[588,1144,788],[799,762,604],[821,1520,1676],[854,803,666],[828,482,472],[445,74,463],[831,489,820],[828,836,482],[716,782,763],[334,815,766],[815,823,766],[334,766,765],[819,805,837],[1716,1521,1412],[1684,924,754],[800,805,819],[1709,829,554],[806,1349,137],[99,1013,747],[341,595,276],[817,810,818],[1176,1691,1685],[763,782,865],[830,846,1052],[865,1499,842],[982,846,1053],[847,832,790],[1178,875,158],[817,818,705],[1302,1392,45],[96,417,284],[223,614,517],[356,507,1514],[1166,848,1179],[1349,432,26],[717,92,276],[770,835,863],[522,509,1745],[847,841,832],[832,841,46],[829,739,554],[802,824,39],[397,1043,775],[1567,849,778],[1385,483,855],[1349,26,1346],[441,801,782],[402,401,293],[1043,667,738],[759,798,1007],[819,837,728],[728,837,828],[837,852,828],[1537,441,833],[148,1475,147],[805,705,837],[716,441,782],[483,1371,780],[814,819,844],[845,753,1336],[1661,719,4],[862,847,790],[737,827,666],[201,46,841],[810,785,818],[408,705,805],[1560,1536,849],[1585,853,1786],[7,1668,807],[7,807,8],[822,1514,1527],[800,819,814],[847,862,841],[991,857,760],[705,818,837],[808,408,773],[402,293,578],[791,858,332],[1480,1228,1240],[814,844,835],[785,1385,852],[1132,120,859],[1743,1726,684],[1704,783,1279],[1623,1694,1731],[959,489,831],[1518,808,773],[862,872,841],[441,773,801],[331,512,308],[380,217,216],[841,872,201],[818,852,837],[448,1480,1240],[856,1108,1195],[1527,1514,1526],[819,182,1232],[871,724,693],[852,836,828],[770,792,814],[803,737,666],[751,826,278],[1674,1727,1699],[849,356,822],[871,693,834],[507,842,1514],[1406,1097,869],[1328,1349,1346],[823,815,795],[744,751,278],[1110,856,1178],[520,717,316],[871,834,683],[884,876,724],[165,266,47],[716,763,507],[216,889,888],[853,1585,1570],[1536,716,356],[886,873,623],[782,770,863],[432,24,26],[683,882,871],[884,724,871],[114,876,884],[516,590,389],[11,1218,1628],[862,113,872],[886,623,629],[830,1052,1120],[762,153,604],[773,408,792],[763,865,507],[153,840,604],[882,884,871],[531,151,326],[886,890,873],[133,262,200],[819,1232,844],[621,636,122],[645,892,519],[1130,1076,784],[114,263,876],[1670,10,1663],[911,670,894],[452,885,872],[872,885,201],[887,882,683],[878,884,882],[590,878,882],[890,867,689],[897,629,621],[897,886,629],[819,728,182],[519,893,688],[894,670,526],[898,894,526],[1536,356,849],[810,1363,785],[878,114,884],[879,888,892],[892,889,893],[893,898,688],[895,683,843],[895,887,683],[889,620,267],[590,882,389],[418,465,84],[949,897,621],[897,890,886],[889,267,893],[898,267,896],[531,326,473],[189,651,878],[843,683,646],[897,867,890],[888,889,892],[893,267,898],[896,894,898],[473,895,843],[895,389,887],[974,706,669],[513,1115,521],[326,151,895],[809,791,857],[211,262,133],[920,923,947],[923,90,947],[90,25,947],[25,972,935],[64,431,899],[52,899,901],[903,905,59],[437,967,73],[839,1242,761],[904,975,44],[917,301,144],[915,670,911],[905,201,885],[1684,63,1685],[1033,1194,288],[950,913,755],[912,918,911],[950,914,913],[506,918,912],[922,919,915],[911,922,915],[1004,451,492],[1263,553,639],[922,911,918],[630,920,947],[916,506,926],[916,918,506],[521,1115,1098],[916,922,918],[919,418,915],[83,38,75],[24,1685,771],[110,1230,1213],[712,8,1837],[922,930,919],[919,430,418],[1395,1402,1187],[930,922,916],[594,623,69],[35,431,968],[35,968,969],[866,924,1684],[1625,1263,675],[631,630,52],[930,931,919],[430,709,418],[302,333,49],[1446,978,1138],[799,1007,798],[931,843,919],[947,25,64],[885,738,667],[1262,963,964],[899,970,901],[1401,946,938],[1117,933,1091],[1685,63,771],[905,948,201],[979,937,980],[951,953,950],[937,270,443],[1154,903,59],[1194,954,1067],[909,405,907],[850,1151,59],[1769,811,1432],[76,206,250],[938,946,966],[965,927,942],[938,966,957],[955,975,904],[927,965,934],[52,51,631],[59,905,667],[431,935,968],[786,289,561],[252,122,671],[481,494,107],[954,1817,1067],[795,25,90],[958,965,945],[795,972,25],[902,983,955],[972,489,944],[1256,29,424],[671,331,945],[946,958,963],[956,955,904],[902,955,956],[671,512,331],[945,331,961],[662,671,122],[671,662,512],[934,65,927],[630,947,52],[666,631,910],[850,59,667],[961,331,234],[1024,411,1042],[890,69,873],[252,671,945],[975,290,940],[283,186,196],[30,283,365],[950,755,298],[946,965,958],[985,290,975],[969,290,985],[405,851,206],[935,431,64],[941,1423,1420],[964,963,167],[942,252,945],[78,757,57],[49,1005,66],[937,979,270],[631,666,827],[980,937,443],[66,689,282],[421,902,956],[947,64,52],[35,979,899],[951,971,953],[762,87,153],[27,31,381],[924,839,87],[946,963,966],[331,308,340],[957,966,1262],[473,843,931],[953,971,920],[270,969,902],[935,962,968],[51,1005,781],[969,983,902],[437,73,940],[69,421,956],[761,249,840],[263,974,669],[962,944,967],[962,437,290],[985,975,955],[907,405,948],[720,957,1262],[25,935,64],[176,200,571],[108,945,50],[250,851,414],[200,986,571],[881,974,263],[827,772,953],[970,899,980],[29,159,27],[234,331,340],[948,405,206],[980,899,979],[986,984,571],[571,984,881],[990,706,974],[946,934,965],[970,980,66],[1113,1486,1554],[984,981,881],[881,987,974],[689,66,443],[1005,901,66],[983,985,955],[165,47,718],[987,990,974],[1370,986,262],[901,970,66],[51,901,1005],[981,987,881],[988,706,990],[942,945,965],[290,437,940],[64,899,52],[988,556,706],[941,934,946],[431,35,899],[996,989,984],[984,989,981],[981,989,987],[35,969,270],[1370,995,986],[986,995,984],[989,999,987],[987,992,990],[992,988,990],[962,967,437],[951,950,976],[979,35,270],[421,270,902],[998,995,1370],[987,999,992],[988,364,556],[969,985,983],[689,443,890],[995,1000,984],[219,958,108],[998,1000,995],[999,997,992],[914,953,772],[845,1336,745],[806,787,231],[1000,996,984],[989,996,999],[50,945,961],[443,421,69],[797,158,779],[1098,1463,434],[996,1009,999],[1001,988,992],[1001,364,988],[903,907,905],[26,759,973],[997,1001,992],[632,364,1001],[1346,26,973],[998,1008,1000],[1000,1009,996],[531,931,736],[252,949,621],[286,388,525],[1174,1008,998],[1009,1010,999],[999,1010,997],[1014,1001,997],[614,105,517],[958,945,108],[525,1004,242],[963,958,219],[233,426,304],[1000,1008,1009],[1010,1014,997],[1001,1006,632],[824,413,39],[642,636,622],[480,388,205],[28,757,797],[1014,1006,1001],[1006,410,632],[975,940,44],[1234,420,858],[54,832,46],[1009,1012,1010],[167,963,219],[41,481,107],[1017,1010,1012],[122,636,662],[939,525,388],[525,939,1004],[950,953,914],[829,1735,739],[1008,880,1015],[1008,1015,1009],[1263,639,675],[956,594,69],[795,90,1347],[1179,848,1013],[759,1007,973],[1009,1015,1012],[1012,1016,1017],[1017,1014,1010],[1019,1011,1006],[927,65,949],[649,316,595],[913,48,755],[976,950,298],[1003,1015,880],[1018,1006,1014],[1021,1018,1014],[444,692,1011],[451,1029,1063],[1185,851,1163],[29,27,381],[181,525,242],[1021,1014,1017],[1016,1021,1017],[1018,1019,1006],[1019,444,1011],[927,949,942],[451,393,492],[903,1154,907],[391,101,57],[94,765,58],[419,1016,1012],[949,252,942],[907,1020,909],[765,442,58],[94,406,908],[1007,94,908],[34,1012,1015],[34,419,1012],[419,1021,1016],[451,1057,393],[907,948,905],[1034,1073,1039],[1061,906,1619],[1068,960,1034],[471,1249,104],[112,1024,1042],[372,379,125],[341,543,165],[141,1094,170],[566,243,1061],[398,1034,1039],[325,317,1823],[1493,296,1724],[850,667,1043],[1054,297,1065],[1619,135,1074],[1061,243,906],[680,1024,821],[1103,96,1245],[1440,1123,1491],[1047,1025,1044],[672,454,1231],[1484,697,1530],[993,672,1231],[178,154,1088],[1044,1041,1066],[112,1062,1058],[1530,649,676],[178,1088,1040],[1046,328,954],[243,244,1022],[954,1194,1033],[1042,411,1032],[971,993,1056],[960,1093,1034],[1754,1338,232],[385,1064,412],[1057,1063,111],[748,1071,1447],[1530,697,695],[971,1056,1270],[977,1059,1211],[649,741,316],[1060,1452,1030],[353,354,1323],[695,768,649],[398,404,1034],[596,316,741],[1836,119,13],[1513,1115,1528],[883,1081,1652],[1039,1073,1048],[462,426,233],[31,1296,354],[1055,1047,1066],[1032,1054,1045],[1521,310,1224],[119,861,13],[1194,1234,288],[1109,1771,1070],[1166,1160,776],[1044,1035,1041],[1026,960,1064],[1050,1032,1045],[1049,1041,387],[115,1013,99],[1046,954,1033],[1321,920,971],[611,1058,345],[1048,1066,1049],[1023,1055,1073],[1029,451,1004],[118,1094,141],[1094,1080,170],[1042,1032,1050],[1026,1064,385],[15,16,1084],[1096,1079,61],[1075,1071,748],[325,1817,328],[909,1163,405],[1022,1234,809],[374,398,1051],[1082,72,81],[1023,1034,1093],[1817,1794,1067],[86,1445,1400],[1507,1535,1510],[1079,1096,1075],[568,1478,1104],[1070,178,1040],[1034,1023,1073],[776,1155,113],[1103,143,142],[1140,81,73],[1082,81,1140],[1060,1030,936],[1040,1086,1109],[370,1065,385],[61,72,1082],[1087,1096,1144],[1040,1088,1086],[1651,812,752],[1062,1050,1045],[187,154,178],[179,187,178],[1099,1344,1101],[1668,1058,807],[1073,1055,1048],[1099,1336,1344],[1283,943,1123],[1049,387,1051],[1024,680,449],[61,1082,1100],[967,749,1111],[1439,1037,88],[742,1505,142],[398,1039,1051],[1107,1336,1099],[1344,1542,1101],[142,1505,1103],[477,1093,447],[477,1023,1093],[471,142,1249],[1041,1035,394],[1328,568,1104],[61,1100,1096],[154,1092,1088],[112,1042,1050],[154,187,168],[435,235,45],[1075,1096,1087],[97,1075,748],[1049,1066,1041],[816,1067,1028],[846,982,1142],[1245,96,284],[1092,154,1080],[1057,451,1063],[387,377,1051],[1055,1025,1047],[1075,1087,1089],[1106,1108,856],[1068,1034,404],[1480,1545,868],[906,135,1619],[1074,991,1095],[570,566,1061],[1025,453,1044],[745,1336,1107],[1035,1057,416],[1092,1102,1129],[1074,135,991],[1105,745,1107],[447,1026,446],[394,387,1041],[73,81,940],[1118,1108,1106],[1210,1108,874],[243,1022,906],[412,1064,1068],[1280,611,603],[960,447,1093],[1051,1039,1049],[1040,1109,1070],[1471,1037,1439],[69,890,443],[1377,703,1374],[1092,1080,1102],[1096,1100,788],[1096,788,1144],[1114,967,1111],[446,1026,297],[70,1112,883],[453,393,1057],[1118,874,1108],[1054,370,1045],[1080,1094,1102],[1039,1048,1049],[428,753,845],[1047,1044,1066],[1044,453,1035],[1472,731,1512],[1126,1121,743],[743,1121,1110],[1032,297,1054],[1480,868,1216],[71,358,72],[1133,967,1114],[1105,1119,745],[1035,453,1057],[1026,447,960],[454,851,1190],[1030,1477,652],[589,816,1028],[1110,1121,1106],[1122,1118,1106],[1116,874,1118],[1048,1055,1066],[1194,1067,816],[744,278,747],[745,1120,845],[845,1052,428],[1105,1780,1119],[1065,297,385],[1098,1529,1463],[731,1060,936],[235,434,812],[1445,1525,1117],[1106,1121,1122],[1122,1127,1118],[1127,1116,1118],[1094,118,1732],[1119,1120,745],[1406,1124,1097],[435,117,235],[1462,1440,1037],[1126,1129,1121],[1088,1092,1129],[1133,73,967],[1120,1052,845],[812,434,752],[1441,1559,1200],[1131,588,413],[1054,1065,370],[235,1098,434],[1052,1142,428],[1737,428,1142],[1496,1446,1483],[1182,1083,1654],[1121,1129,1122],[1732,1116,1127],[768,457,649],[761,1114,249],[1064,960,1068],[1135,1481,1136],[1126,952,1129],[1087,588,1131],[1087,1144,588],[859,788,1139],[1140,1133,1132],[1133,1140,73],[1822,570,1061],[394,1035,416],[1055,1023,459],[80,264,485],[1119,1128,1120],[145,1658,567],[695,891,768],[1129,1102,1122],[1122,1102,1127],[1416,1077,1413],[297,1026,385],[1052,846,1142],[1445,1117,1400],[952,1086,1129],[1714,1089,1131],[1131,1089,1087],[1100,1139,788],[112,1050,1062],[1323,354,1296],[49,333,1141],[1142,982,1737],[79,1457,1091],[1088,1129,1086],[1102,1094,1127],[1127,1094,1732],[1100,1082,1139],[1082,1132,1139],[1082,1140,1132],[1150,1043,397],[60,1166,289],[1696,1146,1698],[1297,1202,1313],[409,1297,1313],[1234,1194,420],[1408,1391,1394],[424,1235,1243],[1203,309,1148],[485,477,447],[1152,1156,850],[1153,1149,1155],[1153,1157,1149],[1149,1152,1150],[1156,1154,1151],[776,1153,1155],[1157,1152,1149],[1217,1393,1208],[1156,1159,1154],[1153,1165,1157],[1165,1152,1157],[1159,1020,1154],[1161,1153,776],[1161,1165,1153],[1165,1158,1152],[1152,1158,1156],[1158,1159,1156],[1166,776,561],[1160,1161,776],[1161,1164,1165],[1161,1160,1164],[1158,1162,1159],[1159,1162,1020],[1270,1321,971],[1164,1170,1165],[1165,1162,1158],[1162,1163,1020],[588,788,925],[1166,1167,1160],[1165,1170,1162],[1160,1167,1164],[1162,1170,1163],[1179,1167,1166],[1167,1168,1164],[1164,1168,1170],[1168,1169,1170],[1234,1022,288],[802,39,866],[1179,1168,1167],[1169,1173,1170],[1170,1173,1163],[1173,1185,1163],[1360,1267,1364],[1169,1185,1173],[611,244,243],[900,1226,1376],[1260,1408,1350],[618,840,831],[1181,1183,1179],[1179,1184,1168],[1208,1274,1291],[1183,1184,1179],[1168,1184,1169],[1387,1395,1254],[1208,1204,1172],[1182,1197,1083],[1187,1083,1197],[1213,1183,1181],[1169,1207,1185],[135,857,991],[1013,1213,1181],[1189,1183,1213],[1183,1189,1184],[1169,1184,1207],[1207,1190,1185],[1180,1389,1288],[1191,1192,1640],[1640,1192,1090],[1090,1205,1654],[1654,1205,1182],[1188,1395,1187],[1126,743,1733],[788,859,925],[809,1234,1171],[1193,1197,1182],[1189,1199,1184],[1639,1191,1637],[1639,1212,1191],[1205,1193,1182],[1198,1187,1197],[1199,1207,1184],[332,1053,846],[1090,1192,1205],[117,1188,1187],[435,1188,117],[435,1206,1188],[1199,1189,1213],[420,816,1053],[1212,1215,1191],[117,1187,1198],[45,1206,435],[120,1132,1133],[874,1116,1210],[1191,1215,1192],[1193,1216,1197],[1216,1198,1197],[1199,1214,1207],[117,521,235],[1220,1311,1078],[1220,900,1311],[1653,1215,1212],[1192,1225,1205],[1205,1209,1193],[1209,1216,1193],[1389,1217,1172],[1207,1214,454],[171,557,1747],[1805,1078,1787],[1805,1219,1078],[1198,1216,868],[666,910,854],[1230,1231,1213],[1213,1231,1199],[1199,1231,1214],[1219,1220,1078],[1215,1221,1192],[1192,1221,1225],[1225,1228,1205],[1205,1228,1209],[1209,1228,1216],[1464,1325,1223],[1215,1227,1221],[1228,1480,1216],[1226,1653,1376],[1653,1249,1215],[1221,1240,1225],[1225,1240,1228],[839,761,840],[1238,1219,1805],[1238,1220,1219],[1232,1380,1375],[1226,1249,1653],[1221,1227,1240],[233,207,532],[110,1236,1230],[1248,1231,1230],[1231,454,1214],[1249,1227,1215],[1248,1056,1231],[489,959,944],[448,1240,284],[925,859,1242],[1805,1244,1238],[1252,1220,1238],[1252,921,1220],[1236,1251,1230],[1230,1251,1248],[1056,993,1231],[1031,1264,1263],[68,1186,157],[1227,1245,1240],[1103,1245,143],[1243,1235,612],[1252,95,921],[1249,1226,1237],[1390,1387,1254],[1120,384,830],[830,332,846],[1227,143,1245],[1315,1369,1358],[1356,1269,1386],[972,795,489],[1831,1224,310],[1250,1255,1251],[1251,1056,1248],[1256,1243,103],[658,358,175],[1620,1238,1244],[1620,1252,1238],[1506,95,1252],[104,1249,1237],[1249,143,1227],[1268,1419,1329],[634,806,231],[618,831,815],[924,1242,839],[1255,1270,1251],[1251,1270,1056],[866,925,1242],[103,29,1256],[424,1243,1256],[134,1651,752],[1250,917,1255],[1172,1204,1260],[1352,1036,1276],[1265,1201,1329],[804,1282,1259],[1259,1294,723],[335,1330,1305],[407,762,799],[875,856,1195],[32,158,344],[967,944,749],[372,125,42],[1175,1354,1261],[553,612,1235],[1259,1273,1294],[1294,1283,723],[757,78,158],[407,799,798],[901,51,52],[139,1386,1389],[1386,1269,1389],[1389,1269,1217],[1148,1590,1268],[1428,1449,1450],[804,1281,1282],[1273,1259,1282],[158,399,779],[771,407,798],[521,1098,235],[917,1312,1255],[1312,1270,1255],[1217,1269,1393],[1195,1108,634],[1110,1106,856],[1210,1691,1176],[27,1112,1145],[1296,27,1145],[1171,858,791],[704,1148,1290],[1430,1436,1437],[1282,1308,1273],[1300,943,1283],[1393,1355,1274],[720,1278,769],[1287,1059,1399],[1310,1388,1272],[1312,1321,1270],[851,1185,1190],[1296,1145,1304],[26,24,771],[51,910,631],[1329,1290,1268],[1290,1148,1268],[1298,1293,733],[1281,1293,1282],[1282,1293,1308],[1308,1299,1273],[1300,1283,1294],[1340,943,1300],[1340,1301,943],[407,754,762],[1287,1399,1295],[34,139,128],[1288,1172,1260],[120,1133,1114],[1306,1113,1511],[1464,1223,1292],[1299,1294,1273],[1299,1300,1294],[1286,1295,838],[1285,1247,1286],[1247,713,1286],[1201,1265,1390],[1378,1368,1357],[1482,1320,917],[917,1320,1312],[850,1156,1151],[588,39,413],[1324,1306,686],[789,1365,928],[1223,1326,1292],[1292,1326,1298],[869,1097,1311],[790,786,561],[1323,1304,932],[1323,1296,1304],[1317,1324,686],[1306,368,1113],[1325,1342,1223],[1326,1348,1298],[1293,1327,1308],[1308,1318,1299],[704,1290,1258],[1320,1321,1312],[761,120,1114],[1684,802,866],[1674,6,1727],[1316,1323,932],[1335,1337,1305],[1348,1327,1293],[1298,1348,1293],[1333,1300,1299],[1333,1343,1300],[1328,1301,1340],[1328,1314,1301],[838,1399,1319],[921,1237,900],[409,1391,1408],[1376,1653,677],[1281,804,1458],[1331,1324,1317],[1324,368,1306],[368,1338,1307],[1327,797,1308],[797,1345,1308],[1308,1345,1318],[1318,1333,1299],[1341,1147,1572],[923,1321,1320],[923,920,1321],[39,588,866],[1141,1323,1316],[1330,1335,1305],[1337,1335,1336],[1339,1332,1325],[1223,1342,1326],[1342,1348,1326],[1348,797,1327],[1345,1333,1318],[1343,1340,1300],[1419,1265,1329],[1347,1320,1584],[1535,1141,1316],[1078,1311,582],[1344,1335,1330],[753,1331,1337],[368,1324,1331],[753,368,1331],[1332,1485,1325],[1325,1485,1342],[787,1343,1333],[137,1328,1340],[973,1341,1479],[406,1147,1341],[1171,1234,858],[1141,1535,1322],[49,1141,1322],[1344,1336,1335],[973,908,1341],[766,1347,1584],[1347,923,1320],[781,49,1322],[368,232,1338],[787,1340,1343],[787,137,1340],[568,1346,973],[58,1147,406],[442,1334,1147],[58,442,1147],[442,766,1334],[90,923,1347],[428,368,753],[779,1333,1345],[825,787,1333],[137,1349,1328],[1328,1346,568],[908,406,1341],[924,866,1242],[1336,753,1337],[428,232,368],[1115,777,1098],[1348,28,797],[797,779,1345],[779,825,1333],[1007,908,973],[583,1351,880],[1365,1246,977],[1658,145,1710],[1310,796,1388],[718,245,165],[1302,1272,1254],[1174,1351,583],[1174,715,1351],[1358,1260,1204],[1374,1373,1276],[1377,1374,1276],[678,1362,1382],[1377,1276,254],[139,34,40],[1008,1174,583],[1396,1286,1319],[768,891,457],[1316,932,1535],[1289,1371,1360],[182,736,864],[1355,1364,1274],[860,1367,1354],[1362,1222,1382],[1376,869,1311],[1590,1411,198],[1232,1375,877],[1394,1295,1286],[880,1356,1386],[880,1351,1356],[1211,1059,1287],[197,678,1405],[880,1386,1003],[1368,1253,1357],[1357,1253,1036],[715,1289,1364],[1354,1367,703],[1383,877,1375],[1266,1288,1260],[1373,1374,703],[1372,1289,1174],[1303,1366,1378],[1351,715,1355],[1665,1666,624],[1309,1357,1036],[900,1237,1226],[1174,1289,715],[1337,1331,1317],[1360,1303,1359],[1267,1354,1175],[1241,1284,1414],[1377,254,929],[1385,855,836],[1396,1319,1436],[1361,1366,1303],[1381,1368,1378],[1313,1211,1391],[1368,1385,1363],[813,82,861],[1058,1280,807],[893,519,892],[1359,1303,860],[1382,1350,1247],[1371,1303,1360],[1267,1175,1271],[769,1286,1396],[712,1837,82],[1366,1385,1381],[1365,796,1310],[1003,1386,40],[780,1371,1370],[561,862,790],[1284,1380,864],[1449,1428,177],[611,1280,1058],[1284,1375,1380],[926,506,1241],[1305,1337,1317],[309,1203,208],[1388,1201,1390],[1309,1036,1352],[1377,929,1411],[1399,1059,1257],[1112,70,1145],[289,1166,561],[1288,1389,1172],[1362,37,1180],[713,1394,1286],[1355,1393,1269],[1401,1423,941],[1274,1271,1384],[860,1378,1367],[715,1364,1355],[677,1406,869],[1297,1358,1202],[1388,1258,1329],[1180,1288,1266],[1008,583,880],[1524,1425,1463],[1390,1403,1387],[1278,1379,1247],[1278,1247,1285],[964,1278,1262],[1358,1369,1202],[1715,1699,1726],[926,1241,1414],[1341,1572,1479],[926,930,916],[1397,51,781],[409,1358,1297],[1236,436,301],[1376,677,869],[1351,1355,1356],[758,1534,1523],[1378,1357,1367],[977,1211,1365],[1135,1136,854],[1394,1391,1295],[1266,1260,1222],[1365,1302,1246],[1232,877,844],[736,930,864],[1408,1358,409],[1508,817,1523],[1381,1385,1368],[718,854,910],[854,718,1135],[1382,1222,1350],[1391,1211,1287],[1391,1287,1295],[1257,1651,134],[1414,1284,864],[1291,1369,1315],[1202,928,1313],[86,1400,1413],[1413,1200,86],[1263,1625,1031],[1413,1400,1404],[1002,1664,1834],[930,926,1414],[1399,1257,134],[520,316,596],[1393,1274,1208],[1657,1655,1712],[1407,1404,1400],[1404,1410,1413],[1649,1229,1406],[1362,1266,1222],[1384,1271,1175],[900,1376,1311],[1274,1384,1291],[1291,1384,1431],[1433,1396,1436],[1267,1359,1354],[309,1353,703],[838,1319,1286],[1407,1410,1404],[441,1518,773],[1241,123,1428],[1622,1521,1224],[1217,1208,1172],[1130,793,1076],[425,1409,1481],[1481,1409,1533],[1303,1378,860],[1350,1408,1394],[1246,1651,977],[1289,1360,1364],[1727,1694,1623],[1417,1407,1533],[1417,1410,1407],[1406,1650,1649],[1319,134,1437],[1414,864,930],[1406,1229,1124],[1354,1359,860],[1433,769,1396],[1417,1533,1409],[1416,1413,1410],[1415,1416,1410],[95,1237,921],[1392,1254,1395],[1360,1359,1267],[1258,1290,1329],[1180,128,1389],[1420,1409,425],[1417,1418,1410],[1418,1415,1410],[1422,1077,1416],[1247,1350,1394],[37,43,1180],[1204,1315,1358],[1428,1383,1375],[1356,1355,1269],[1409,1418,1417],[1302,45,1246],[1421,1416,1415],[1421,1422,1416],[1422,1494,1077],[957,720,938],[1423,1409,1420],[1423,1418,1409],[752,434,1438],[1260,1358,1408],[1363,1385,785],[1423,1426,1418],[1426,1424,1418],[1229,1649,1124],[1222,1260,1350],[1508,1523,1137],[1278,1285,769],[1482,917,144],[1418,1424,1415],[1425,1422,1421],[1425,1524,1422],[1272,1388,1390],[1391,409,1313],[1378,1366,1381],[1371,483,1361],[720,1262,1278],[29,103,159],[1271,1364,1267],[1424,1427,1415],[1537,1522,1518],[134,752,1438],[1420,934,941],[1428,1375,1284],[1277,1224,1831],[1362,1180,1266],[1401,1426,1423],[1577,1369,1291],[268,483,262],[1383,1450,1456],[1384,1175,1431],[1430,1415,1427],[1430,1421,1415],[1430,1425,1421],[1379,1382,1247],[1252,1553,1429],[1206,1392,1395],[1433,1430,1427],[309,208,1353],[1272,1390,1254],[1361,483,1366],[1523,817,808],[1302,1254,1392],[1371,1361,1303],[1426,1435,1424],[1435,1433,1424],[1433,1427,1424],[720,769,1433],[796,1258,1388],[1590,1419,1268],[1289,1372,1371],[1305,1317,1509],[998,1372,1174],[40,1386,139],[1261,1354,703],[1364,1271,1274],[134,1438,1437],[1436,1319,1437],[1317,686,1509],[1484,932,1304],[1434,1432,1509],[1420,65,934],[931,930,736],[1367,1357,1309],[1372,1370,1371],[1204,1208,1315],[1426,938,1435],[1368,1363,1253],[1207,454,1190],[1302,1310,1272],[309,1377,390],[390,1377,1411],[1370,1372,998],[1411,1590,1148],[720,1433,1435],[1450,1383,1428],[1379,678,1382],[1405,678,1379],[1208,1291,1315],[1399,134,1319],[1367,1309,1373],[1373,1352,1276],[596,741,593],[553,1264,612],[1433,1436,1430],[1437,1438,1430],[964,1405,1379],[1373,1309,1352],[1265,1403,1390],[1233,1618,1434],[1365,1310,1302],[789,796,1365],[720,1435,938],[128,139,1389],[1466,933,1525],[1191,1640,1637],[1314,1442,943],[1141,353,1323],[1489,1138,1474],[1462,1477,1440],[1474,1138,1488],[1442,1314,1443],[1446,1030,1546],[1484,1145,697],[1549,1443,1445],[1470,1572,1468],[1397,1239,1507],[1649,1825,1824],[1259,1440,1477],[1451,1450,1449],[978,1446,652],[1454,1456,1451],[1451,1456,1450],[341,1507,595],[933,1547,79],[804,1452,1060],[1454,1455,1456],[1398,1460,1454],[1455,877,1456],[1277,1831,1825],[804,1060,1458],[1339,1459,1595],[1314,1104,1443],[933,1448,1547],[147,1460,1398],[1460,1461,1454],[1454,1461,1455],[1292,1125,1464],[417,1531,1480],[1459,1339,1325],[811,1756,335],[1512,936,1490],[777,1529,1098],[147,1475,1460],[1464,253,1459],[836,855,482],[1487,1486,1307],[1104,1501,1443],[1439,1200,1532],[1475,1469,1460],[1460,1469,1461],[1325,1464,1459],[1277,1825,1649],[1532,1200,1077],[844,877,1455],[1572,933,1466],[1479,568,973],[1509,335,1305],[1339,1595,1759],[1469,1476,1461],[1461,1476,1455],[1104,1470,1468],[1464,1472,253],[1117,1091,1407],[1756,1542,335],[1206,1395,1188],[335,1542,1330],[835,844,1455],[1471,1598,1462],[1491,1442,1441],[835,1455,1476],[1441,1442,1443],[1489,1474,1473],[1251,1236,1250],[1030,1452,1477],[1598,1439,1532],[978,1598,1492],[1426,1401,938],[1448,1584,1482],[1724,1497,1475],[1475,1497,1469],[1484,1535,932],[1307,1486,1113],[1487,696,1495],[1037,1491,1441],[1030,1446,936],[1453,1487,1495],[696,1467,1495],[1138,1489,1483],[1497,1143,1469],[1469,1143,1476],[652,1598,978],[850,1043,1150],[1482,1584,1320],[1731,98,1697],[1113,1554,1573],[1524,1532,1494],[1496,1467,696],[1452,1259,1477],[296,1504,1497],[1504,1143,1497],[1143,1499,1476],[718,910,1498],[868,1540,1528],[817,1253,810],[1490,696,1487],[1440,1491,1037],[1510,676,595],[1488,1492,1517],[781,1239,1397],[1467,1519,1503],[1500,1307,1759],[1149,397,452],[1504,1514,1143],[1514,842,1143],[1125,733,1458],[1503,1531,1555],[1276,1036,1137],[1440,723,1123],[1036,1508,1137],[817,1508,1253],[103,883,1112],[1458,731,1472],[1512,1490,1487],[1487,1453,1486],[1138,978,1488],[1036,1253,1508],[1398,149,147],[1474,1517,1513],[1125,1458,1472],[1486,1453,1554],[1518,1534,758],[345,1058,1062],[928,1202,1369],[1554,1541,1505],[1464,1125,1472],[1504,764,1514],[304,426,573],[1505,742,1506],[1479,1572,1478],[1519,1483,1489],[833,716,1069],[1522,1534,1518],[1115,1513,777],[811,335,1432],[1591,1533,1407],[777,1517,1529],[1513,1517,777],[1498,910,1397],[1069,1539,833],[833,1539,1537],[1522,1551,1534],[1534,1551,1523],[1538,1137,1523],[910,51,1397],[1367,1373,703],[1466,1525,1468],[157,1186,1832],[1429,1511,1506],[1573,1505,1506],[1259,1452,804],[1503,1495,1467],[262,483,780],[1572,1466,1468],[1536,1556,716],[716,1556,1069],[1544,1523,1551],[1544,1538,1523],[1511,1573,1506],[933,1572,1448],[1543,1537,1539],[1537,1543,1522],[1091,933,79],[1519,1540,1545],[1549,1445,86],[1069,1548,1539],[1548,1543,1539],[1543,1551,1522],[1500,1487,1307],[68,784,1186],[1552,1544,1551],[1550,1538,1544],[1538,1550,1137],[1519,1473,1540],[1547,1448,1482],[1560,1563,1536],[1536,1563,1556],[1556,1548,1069],[1543,1558,1551],[1137,1550,1276],[1453,1495,1555],[1561,1543,1548],[1543,1561,1558],[1558,1566,1551],[1552,1550,1544],[1569,1557,1550],[1557,1276,1550],[1276,1557,254],[1531,1503,1480],[1535,1530,1510],[1545,1503,1519],[1547,1482,79],[1566,1552,1551],[1552,1569,1550],[1503,1545,1480],[703,1377,309],[1625,675,756],[1037,1441,88],[929,254,1557],[849,1567,1560],[1556,1564,1548],[1492,1529,1517],[1252,1429,1506],[1553,1027,1429],[1453,1555,1541],[1554,1453,1541],[1233,686,1553],[1328,1104,1314],[1564,1576,1548],[1548,1576,1561],[1557,1562,929],[1520,112,1668],[1483,1446,1138],[778,1570,1567],[1563,1564,1556],[1561,1565,1558],[1565,1566,1558],[1569,1552,1566],[1562,1557,1569],[1530,1535,1484],[1387,1402,1395],[1621,1634,1387],[1567,1568,1560],[1560,1568,1563],[1571,1569,1566],[1344,1330,1542],[1577,1431,1353],[1638,233,304],[1524,1463,1529],[1353,1431,1175],[1077,1200,1413],[1478,1470,1104],[1568,1575,1563],[1563,1575,1564],[1575,1576,1564],[1561,1576,1565],[1565,1574,1566],[1562,1515,929],[1555,96,1541],[1531,417,96],[1555,1531,96],[1246,45,1651],[208,1577,1353],[1586,1568,1567],[1574,1571,1566],[1571,1583,1569],[1474,1513,1528],[1239,1322,1535],[1478,1572,1470],[1570,1586,1567],[1488,1517,1474],[8,1833,1837],[1123,1442,1491],[1589,1568,1586],[1576,1594,1565],[1565,1594,1574],[1562,198,1515],[1559,1441,1549],[1441,1443,1549],[1135,425,1481],[1239,1535,1507],[1595,1487,1500],[1570,1585,1586],[1589,1578,1568],[1568,1578,1575],[1579,1569,1583],[1177,1577,208],[115,1236,110],[1578,1593,1575],[1587,1576,1575],[1576,1581,1594],[1571,1582,1583],[1588,1579,1583],[1579,1580,1562],[1569,1579,1562],[1562,1580,198],[1027,1511,1429],[1589,1593,1578],[1587,1581,1576],[1582,1574,1594],[1574,1582,1571],[1575,1593,1587],[1583,1582,1588],[1580,1590,198],[1587,1593,1581],[1505,1541,96],[1369,1577,1177],[1573,1554,1505],[1479,1478,568],[1585,1589,1586],[1369,1177,704],[766,1584,1334],[977,1257,1059],[1091,1591,1407],[1591,1091,1457],[1585,1604,1589],[1581,1592,1594],[1602,1582,1594],[1582,1608,1588],[1608,1579,1588],[1579,1597,1580],[1419,1590,1580],[1597,1419,1580],[1431,1577,1291],[1589,1604,1593],[1601,1596,1593],[1593,1596,1581],[1306,1511,1027],[1511,1113,1573],[1786,1412,1585],[1412,1604,1585],[1581,1596,1592],[1592,1602,1594],[1608,1599,1579],[1599,1611,1579],[1579,1611,1597],[1512,1487,253],[1519,1489,1473],[1545,1540,868],[1083,1187,1402],[1117,1407,1400],[1292,733,1125],[284,1240,1245],[1604,1600,1593],[1600,1601,1593],[1582,1607,1608],[789,1369,704],[1467,1483,1519],[1601,1613,1596],[1596,1613,1592],[1602,1607,1582],[1620,1553,1252],[1601,1605,1613],[1592,1613,1602],[1602,1606,1607],[1608,1609,1599],[1599,1609,1611],[1603,1597,1611],[1265,1419,1597],[1603,1265,1597],[1392,1206,45],[928,1369,789],[1474,1528,1473],[1104,1468,1501],[1412,1521,1604],[1613,1631,1602],[1607,1610,1608],[1608,1610,1609],[1476,863,835],[1495,1503,1555],[1498,1397,718],[1520,1668,7],[1604,1615,1600],[1605,1601,1600],[1602,1631,1606],[1606,1610,1607],[1759,1595,1500],[1292,1298,733],[1615,1604,1521],[1609,1603,1611],[652,1462,1598],[1468,1525,1445],[1443,1501,1445],[1134,1723,150],[1521,1622,1615],[1615,1616,1600],[1616,1605,1600],[1605,1616,1612],[1605,1612,1613],[1612,1617,1613],[1613,1617,1631],[1606,1614,1610],[1265,1603,1403],[448,417,1480],[1595,253,1487],[1501,1468,1445],[1383,1456,877],[1490,1496,696],[1610,1627,1609],[1627,1621,1609],[1591,1481,1533],[1598,1471,1439],[1353,1261,703],[1606,1631,1614],[1609,1621,1403],[1532,1077,1494],[1528,1115,513],[1546,652,1446],[1211,928,1365],[1540,1473,1528],[1078,1502,1787],[1425,1430,1438],[1617,1630,1631],[959,749,944],[566,570,603],[1716,310,1521],[775,452,397],[1615,1636,1616],[1616,1636,1612],[1610,1632,1627],[789,704,1258],[1457,1481,1591],[1769,1756,811],[207,1629,722],[1629,1625,722],[1224,1277,1622],[1622,1636,1615],[1636,1646,1612],[1612,1630,1617],[1631,1626,1614],[1614,1632,1610],[1506,104,95],[1481,1457,1136],[1123,943,1442],[936,1446,1496],[1499,863,1476],[1629,1031,1625],[1233,1509,686],[1633,1634,1621],[1621,1387,1403],[1472,1512,253],[1177,208,704],[1277,1636,1622],[1626,1632,1614],[1627,1633,1621],[936,1496,1490],[185,1454,1451],[731,936,1512],[1638,1635,207],[553,1263,1264],[1653,1212,1639],[1633,1627,1632],[1633,1387,1634],[1458,1060,731],[368,1307,1113],[1264,1031,1629],[1152,850,1150],[1277,1644,1636],[1646,1637,1612],[1637,1630,1612],[1647,1631,1630],[1647,1626,1631],[1422,1524,1494],[1030,652,1546],[1635,1629,207],[1635,1264,1629],[1639,1646,1636],[1637,1640,1630],[1641,1632,1626],[1632,1642,1633],[1633,1643,1387],[842,1499,1143],[865,863,1499],[1516,978,1492],[67,1130,784],[1103,1505,96],[88,1441,1200],[1644,1639,1636],[1640,1647,1630],[1647,1641,1626],[1633,1648,1643],[1492,1532,1524],[1488,1516,1492],[1037,1471,1462],[612,1264,1635],[1502,1078,1124],[1641,1642,1632],[1648,1633,1642],[1528,513,868],[1492,1598,1532],[1095,991,760],[679,157,1664],[760,1128,1785],[1277,1650,1644],[320,1022,244],[1559,1549,86],[1676,1520,7],[1488,978,1516],[1095,760,1785],[1128,384,1120],[304,312,1638],[1081,1638,312],[1081,1635,1638],[103,612,1635],[652,1477,1462],[1650,1645,1644],[1645,1639,1644],[1639,1637,1646],[1640,1090,1647],[1654,1641,1647],[1654,1642,1641],[1654,1648,1642],[1643,1402,1387],[1432,335,1509],[384,1128,760],[1652,312,304],[103,1243,612],[1277,1649,1650],[1090,1654,1647],[1643,1648,1402],[1134,324,1675],[679,68,157],[1652,1081,312],[1136,301,803],[1653,1639,1645],[723,1440,1259],[803,854,1136],[104,1506,742],[1112,159,103],[1654,1083,1648],[977,1651,1257],[1397,1507,718],[1081,103,1635],[1650,677,1645],[1083,1402,1648],[1706,1655,1671],[1624,1704,1711],[767,2,1],[608,794,294],[1678,1683,1686],[767,1682,2],[1669,1692,1675],[296,1681,764],[1671,1656,1672],[17,1673,1679],[1706,1671,1673],[1662,1674,1699],[1655,1657,1656],[418,84,915],[1526,1514,764],[1658,1657,567],[870,1695,764],[813,1697,98],[1659,821,5],[60,1013,848],[1013,110,1213],[661,1038,1692],[1660,1703,17],[1693,1673,17],[1663,1715,1743],[1013,115,110],[344,1733,32],[1670,1663,1743],[1670,1743,1738],[1677,1670,1738],[1661,4,3],[1084,1683,1678],[1728,793,1130],[1683,1767,1196],[1677,1738,1196],[1279,1786,853],[294,1038,608],[1279,1689,1786],[870,18,1708],[870,1680,1695],[1705,10,1670],[1084,1767,1683],[1196,1738,1686],[1750,870,1681],[1750,18,870],[1773,1703,1660],[1135,47,425],[150,323,1134],[1707,1655,1706],[1741,344,1687],[1685,1691,1684],[1684,1691,802],[1672,1656,0],[1038,124,608],[1671,1672,1690],[1628,1218,1767],[1686,1275,1667],[1493,1750,1681],[1773,18,1750],[1773,1660,18],[1679,1671,16],[1735,1706,1673],[1667,1678,1686],[1688,1658,1],[1656,1688,0],[1293,1281,1458],[1698,1678,1667],[1696,1130,1722],[1698,1667,1696],[1715,1662,1699],[1692,1038,294],[1682,767,357],[1669,661,1692],[802,1702,824],[1028,1067,1784],[822,1624,778],[119,813,861],[1218,1670,1677],[1703,1693,17],[1658,1710,1],[750,1730,1729],[1701,750,1729],[1693,1735,1673],[1731,1694,98],[1691,1702,802],[783,1729,1719],[1680,870,1708],[1707,1709,1655],[533,756,675],[1691,1210,1702],[11,1705,1670],[1767,1218,1196],[1218,1677,1196],[1664,1716,1721],[1729,1725,1719],[1729,1072,1725],[1210,1116,1702],[1702,1720,824],[1682,1661,2],[1713,1719,1721],[1716,1786,1713],[1730,1722,1072],[294,1717,1811],[1692,294,1666],[1659,680,821],[824,1720,1714],[1726,1731,1718],[345,1062,1045],[1738,1743,1275],[1075,1089,1071],[783,1719,1689],[1275,684,1728],[1692,1666,1665],[1675,1692,1665],[294,1811,1666],[1716,1664,310],[1678,1698,1700],[6,9,1727],[676,649,595],[381,31,361],[1723,1804,1772],[1727,9,1694],[1720,1089,1714],[1786,1716,1412],[1683,1196,1686],[1718,1697,1085],[1116,1739,1702],[1739,1734,1720],[1702,1739,1720],[1089,1720,1734],[509,748,1745],[1743,1715,1726],[1717,294,794],[1116,1732,1739],[1718,1731,1697],[1696,1667,1130],[1134,1665,1723],[1694,712,98],[101,1687,102],[391,1736,101],[662,636,642],[1734,1447,1089],[1089,1447,1071],[436,99,493],[1689,1279,783],[1485,1465,1342],[1736,1687,101],[344,1741,1733],[1741,1742,1733],[1735,829,1706],[829,1707,1706],[1485,1332,1465],[952,1126,1742],[1747,1447,1734],[879,892,645],[1730,1146,1696],[829,1709,1707],[1709,1712,1655],[118,1739,1732],[1332,1744,1465],[1687,1749,1741],[1741,1758,1742],[679,1072,68],[1072,1722,68],[118,1747,1739],[1747,1734,1739],[1465,1744,1736],[1736,1740,1687],[1704,1701,783],[1665,624,1723],[1722,1130,67],[1025,1055,467],[1444,14,1701],[558,522,530],[1657,1658,1688],[1339,1746,1332],[1332,1748,1744],[1687,1740,1749],[1741,1749,1758],[1109,952,1742],[1747,118,141],[1671,1690,1628],[1671,1628,16],[1657,1688,1656],[1745,748,1447],[357,767,1710],[1746,1748,1332],[1146,1700,1698],[1759,1307,1338],[1239,781,1322],[1745,1447,1747],[522,1745,1747],[316,717,595],[148,1493,1724],[1758,1109,1742],[1725,1072,679],[726,719,1661],[1695,1680,1526],[1772,1750,1493],[148,1772,1493],[1542,1751,1101],[952,1109,1086],[1744,1752,1736],[1736,1752,1740],[1753,1755,1740],[391,1342,1736],[821,112,1520],[557,530,1747],[530,522,1747],[994,879,645],[1542,1756,1751],[1813,1693,1703],[1746,1754,1748],[1748,1764,1744],[1752,1757,1740],[1740,1757,1753],[1749,1740,1755],[1755,1763,1749],[1763,1758,1749],[1275,1743,684],[1813,1735,1693],[1107,1099,1101],[1723,624,1804],[1403,1603,1609],[1748,1754,1764],[1744,1757,1752],[1760,1109,1758],[1465,1736,1342],[436,115,99],[1686,1738,1275],[1751,1766,1101],[1759,1754,1746],[1755,1753,1763],[1570,1279,853],[1701,1146,750],[1655,1656,1671],[11,1670,1218],[1761,1751,1756],[1766,1107,1101],[1726,1623,1731],[1711,1704,1279],[67,784,68],[558,530,545],[1620,1618,1233],[1769,1761,1756],[102,1687,344],[1338,1754,1759],[1754,232,1764],[1744,1765,1757],[1757,1763,1753],[1762,1760,1758],[1760,1771,1109],[1339,1759,1746],[1675,1665,1134],[1730,1696,1722],[1774,1751,1761],[1766,1780,1107],[1780,1105,1107],[1764,1765,1744],[1763,1762,1758],[1772,1773,1750],[1811,1813,1703],[1434,1769,1432],[1780,1766,1751],[232,1781,1764],[1711,1279,1570],[1688,1,0],[1774,1780,1751],[1764,1781,1765],[1765,1768,1757],[1757,1768,1763],[1777,1782,1760],[1762,1777,1760],[1769,1774,1761],[1763,1777,1762],[1760,1782,1771],[232,1737,1781],[1768,1776,1763],[272,255,774],[1669,994,661],[1618,1769,1434],[1765,589,1768],[1770,1777,1763],[1701,1729,783],[1783,1774,1769],[1789,1780,1774],[589,1775,1768],[1776,1770,1763],[1782,1778,1771],[1771,1778,1070],[624,1703,1773],[624,1811,1703],[1620,1244,1618],[1779,1769,1618],[1779,1783,1769],[739,1735,1813],[1775,1776,1768],[1790,1777,1770],[1777,1778,1782],[1725,679,1721],[733,1293,1458],[1802,1618,1244],[1802,1779,1618],[1788,1783,1779],[1789,1774,1783],[1796,1780,1789],[1796,1119,1780],[1823,1817,325],[1699,1727,1623],[750,1146,1730],[1497,1724,296],[1128,1119,1796],[61,62,71],[1131,413,824],[1114,1111,249],[1784,1776,1775],[1123,723,1283],[1791,1788,1779],[1788,1789,1783],[1095,1797,1074],[1028,1784,1775],[1784,1770,1776],[1777,1790,1778],[1793,1797,1095],[1797,1800,1074],[1798,1790,1770],[1805,1802,1244],[1802,1791,1779],[1792,1789,1788],[1793,1785,1128],[1793,1095,1785],[1074,1800,1619],[741,457,593],[1798,1770,1784],[1798,1794,1790],[1786,1689,1713],[684,1726,1718],[1728,1085,793],[1795,1787,1502],[1806,1802,1805],[1819,1788,1791],[1067,1798,1784],[1790,1794,1778],[1795,1502,1124],[1801,1805,1787],[1807,1791,1802],[1807,1819,1791],[1819,1792,1788],[1799,1128,1796],[994,645,661],[684,1085,1728],[684,1718,1085],[1699,1623,1726],[1801,1787,1795],[1808,1789,1792],[1808,1796,1789],[1799,1793,1128],[1809,1797,1793],[1809,1803,1797],[1803,1800,1797],[1067,1794,1798],[774,255,1778],[1673,1671,1679],[879,1669,888],[19,1807,1802],[1810,1619,1800],[879,994,1669],[1794,774,1778],[1723,1772,148],[1804,1773,1772],[1814,1795,1124],[1649,1814,1124],[1814,1801,1795],[1812,1806,1805],[19,1802,1806],[19,1819,1807],[1810,1800,1803],[1804,624,1773],[1714,1131,824],[1801,1812,1805],[1812,19,1806],[1808,1792,1819],[1799,1809,1793],[1821,1810,1803],[1717,739,1813],[1061,1619,1822],[1794,1817,774],[79,1482,144],[1815,1801,1814],[23,1819,19],[589,1028,1775],[1817,1823,774],[1689,1719,1713],[1824,1814,1649],[1827,1818,1801],[1818,1812,1801],[1818,19,1812],[1818,20,19],[1816,1809,1799],[1821,1803,1809],[1822,1619,1810],[124,708,608],[1663,10,1715],[1815,1827,1801],[1820,1808,1819],[23,1820,1819],[603,1810,1821],[603,1822,1810],[1085,1697,793],[1628,1690,11],[1527,1704,1624],[1730,1072,1729],[1526,1444,1704],[1526,1680,1444],[1704,1444,1701],[1816,1821,1809],[1722,67,68],[317,272,1823],[1716,1713,1721],[16,1628,1767],[1527,1526,1704],[1824,1826,1814],[1814,1826,1815],[1818,21,20],[1835,1808,1820],[603,570,1822],[226,1070,1778],[1013,1181,1179],[1721,679,1664],[1717,1813,1811],[1828,1827,1815],[22,1820,23],[22,1835,1820],[1830,603,1821],[719,1659,5],[643,567,1657],[1717,794,739],[1825,1826,1824],[1828,1815,1826],[1829,21,1818],[1808,1835,13],[4,719,5],[10,1662,1715],[1828,1832,1827],[1832,1818,1827],[12,1833,1816],[1833,1821,1816],[1833,1830,1821],[14,1146,1701],[1186,1829,1818],[1280,603,1830],[14,1700,1146],[1667,1728,1130],[1825,1834,1826],[1834,1828,1826],[1832,1186,1818],[1836,13,1835],[1624,1711,1570],[778,1624,1570],[1719,1725,1721],[1002,1825,1831],[1002,1834,1825],[1834,1832,1828],[1186,21,1829],[1836,1835,22],[1837,1833,12],[1280,1830,1833],[1667,1275,1728],[16,1767,1084],[589,1765,1838],[1765,1781,1838],[1781,1737,1838],[1737,982,1838],[982,1053,1838],[1053,816,1838],[816,589,1838]]
},{}]},{},[])
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../../home/admin/browserify-cdn/node_modules/browserify/node_modules/browser-pack/_prelude.js","index.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","exports.positions=[[1.301895,0.122622,2.550061],[1.045326,0.139058,2.835156],[0.569251,0.155925,2.805125],[0.251886,0.144145,2.82928],[0.063033,0.131726,3.01408],[-0.277753,0.135892,3.10716],[-0.441048,0.277064,2.594331],[-1.010956,0.095285,2.668983],[-1.317639,0.069897,2.325448],[-0.751691,0.264681,2.381496],[0.684137,0.31134,2.364574],[1.347931,0.302882,2.201434],[-1.736903,0.029894,1.724111],[-1.319986,0.11998,0.912925],[1.538077,0.157372,0.481711],[1.951975,0.081742,1.1641],[1.834768,0.095832,1.602682],[2.446122,0.091817,1.37558],[2.617615,0.078644,0.742801],[-1.609748,0.04973,-0.238721],[-1.281973,0.230984,-0.180916],[-1.074501,0.248204,0.034007],[-1.201734,0.058499,0.402234],[-1.444454,0.054783,0.149579],[-4.694605,5.075882,1.043427],[-3.95963,7.767394,0.758447],[-4.753339,5.339817,0.665061],[-1.150325,9.133327,-0.368552],[-4.316107,2.893611,0.44399],[-0.809202,9.312575,-0.466061],[0.085626,5.963693,1.685666],[-1.314853,9.00142,-0.1339],[-4.364182,3.072556,1.436712],[-2.022074,7.323396,0.678657],[1.990887,6.13023,0.479643],[-3.295525,7.878917,1.409353],[0.571308,6.197569,0.670657],[0.89661,6.20018,0.337056],[0.331851,6.162372,1.186371],[-4.840066,5.599874,2.296069],[2.138989,6.031291,0.228335],[0.678923,6.026173,1.894052],[-0.781682,5.601573,1.836738],[1.181315,6.239007,0.393293],[-3.606308,7.376476,2.661452],[-0.579059,4.042511,-1.540883],[-3.064069,8.630253,-2.597539],[-2.157271,6.837012,0.300191],[-2.966013,7.821581,-1.13697],[-2.34426,8.122965,0.409043],[-0.951684,5.874251,1.415119],[-2.834853,7.748319,0.182406],[-3.242493,7.820096,0.373674],[-0.208532,5.992846,1.252084],[-3.048085,8.431527,-2.129795],[1.413245,5.806324,2.243906],[-0.051222,6.064901,0.696093],[-4.204306,2.700062,0.713875],[-4.610997,6.343405,0.344272],[-3.291336,9.30531,-3.340445],[-3.27211,7.559239,-2.324016],[-4.23882,6.498344,3.18452],[-3.945317,6.377804,3.38625],[-4.906378,5.472265,1.315193],[-3.580131,7.846717,0.709666],[-1.995504,6.645459,0.688487],[-2.595651,7.86054,0.793351],[-0.008849,0.305871,0.184484],[-0.029011,0.314116,-0.257312],[-2.522424,7.565392,1.804212],[-1.022993,8.650826,-0.855609],[-3.831265,6.595426,3.266783],[-4.042525,6.855724,3.060663],[-4.17126,7.404742,2.391387],[3.904526,3.767693,0.092179],[0.268076,6.086802,1.469223],[-3.320456,8.753222,-2.08969],[1.203048,6.26925,0.612407],[-4.406479,2.985974,0.853691],[-3.226889,6.615215,-0.404243],[0.346326,1.60211,3.509858],[-3.955476,7.253323,2.722392],[-1.23204,0.068935,1.68794],[0.625436,6.196455,1.333156],[4.469132,2.165298,1.70525],[0.950053,6.262899,0.922441],[-2.980404,5.25474,-0.663155],[-4.859043,6.28741,1.537081],[-3.077453,4.641475,-0.892167],[-0.44002,8.222503,-0.771454],[-4.034112,7.639786,0.389935],[-3.696045,6.242042,3.394679],[-1.221806,7.783617,0.196451],[0.71461,6.149895,1.656636],[-4.713539,6.163154,0.495369],[-1.509869,0.913044,-0.832413],[-1.547249,2.066753,-0.852669],[-3.757734,5.793742,3.455794],[-0.831911,0.199296,1.718536],[-3.062763,7.52718,-1.550559],[0.938688,6.103354,1.820958],[-4.037033,2.412311,0.988026],[-4.130746,2.571806,1.101689],[-0.693664,9.174283,-0.952323],[-1.286742,1.079679,-0.751219],[1.543185,1.408925,3.483132],[1.535973,2.047979,3.655029],[0.93844,5.84101,2.195219],[-0.684401,5.918492,1.20109],[1.28844,2.008676,3.710781],[-3.586722,7.435506,-1.454737],[-0.129975,4.384192,2.930593],[-1.030531,0.281374,3.214273],[-3.058751,8.137238,-3.227714],[3.649524,4.592226,1.340021],[-3.354828,7.322425,-1.412086],[0.936449,6.209237,1.512693],[-1.001832,3.590411,-1.545892],[-3.770486,4.593242,2.477056],[-0.971925,0.067797,0.921384],[-4.639832,6.865407,2.311791],[-0.441014,8.093595,-0.595999],[-2.004852,6.37142,1.635383],[4.759591,1.92818,0.328328],[3.748064,1.224074,2.140484],[-0.703601,5.285476,2.251988],[0.59532,6.21893,0.981004],[0.980799,6.257026,1.24223],[1.574697,6.204981,0.381628],[1.149594,6.173608,1.660763],[-3.501963,5.895989,3.456576],[1.071122,5.424198,2.588717],[-0.774693,8.473335,-0.276957],[3.849959,4.15542,0.396742],[-0.801715,4.973149,-1.068582],[-2.927676,0.625112,2.326393],[2.669682,4.045542,2.971184],[-4.391324,4.74086,0.343463],[1.520129,6.270031,0.775471],[1.837586,6.084731,0.109188],[1.271475,5.975024,2.032355],[-3.487968,4.513249,2.605871],[-1.32234,1.517264,-0.691879],[-1.080301,1.648226,-0.805526],[-3.365703,6.910166,-0.454902],[1.36034,0.432238,3.075004],[-3.305013,5.774685,3.39142],[3.88432,0.654141,0.12574],[3.57254,0.377934,0.302501],[4.196136,0.807999,0.212229],[3.932997,0.543123,0.380579],[4.023704,3.286125,0.537597],[1.864455,4.916544,2.691677],[-4.775427,6.499498,1.440153],[-3.464928,3.68234,2.766356],[3.648972,1.751262,2.157485],[1.179111,3.238846,3.774796],[-0.171164,0.299126,-0.592669],[-4.502912,3.316656,0.875188],[-0.948454,9.214025,-0.679508],[1.237665,6.288593,1.046],[1.523423,6.268963,1.139544],[1.436519,6.140608,1.739316],[3.723607,1.504355,2.136762],[2.009495,4.045514,3.22053],[-1.921944,7.249905,0.213973],[1.254068,1.205518,3.474709],[-0.317087,5.996269,0.525872],[-2.996914,3.934607,2.900178],[-3.316873,4.028154,2.785696],[-3.400267,4.280157,2.689268],[-3.134842,4.564875,2.697192],[1.480563,4.692567,2.834068],[0.873682,1.315452,3.541585],[1.599355,0.91622,3.246769],[-3.292102,7.125914,2.768515],[3.74296,4.511299,0.616539],[4.698935,1.55336,0.26921],[-3.274387,3.299421,2.823946],[-2.88809,3.410699,2.955248],[1.171407,1.76905,3.688472],[1.430276,3.92483,3.473666],[3.916941,2.553308,0.018941],[0.701632,2.442372,3.778639],[1.562657,2.302778,3.660957],[4.476622,1.152407,0.182131],[-0.61136,5.761367,1.598838],[-3.102154,3.691687,2.903738],[1.816012,5.546167,2.380308],[3.853928,4.25066,0.750017],[1.234681,3.581665,3.673723],[1.862271,1.361863,3.355209],[1.346844,4.146995,3.327877],[1.70672,4.080043,3.274307],[0.897242,1.908983,3.6969],[-0.587022,9.191132,-0.565301],[-0.217426,5.674606,2.019968],[0.278925,6.120777,0.485403],[1.463328,3.578742,-2.001464],[-3.072985,4.264581,2.789502],[3.62353,4.673843,0.383452],[-3.053491,8.752377,-2.908434],[-2.628687,4.505072,2.755601],[0.891047,5.113781,2.748272],[-2.923732,3.06515,2.866368],[0.848008,4.754252,2.896972],[-3.319184,8.811641,-2.327412],[0.12864,8.814781,-1.334456],[1.549501,4.549331,-1.28243],[1.647161,3.738973,3.507719],[1.250888,0.945599,3.348739],[3.809662,4.038822,0.053142],[1.483166,0.673327,3.09156],[0.829726,3.635921,3.713103],[1.352914,5.226651,2.668113],[2.237352,4.37414,3.016386],[4.507929,0.889447,0.744249],[4.57304,1.010981,0.496588],[3.931422,1.720989,2.088175],[-0.463177,5.989835,0.834346],[-2.811236,3.745023,2.969587],[-2.805135,4.219721,2.841108],[-2.836842,4.802543,2.60826],[1.776716,2.084611,3.568638],[4.046881,1.463478,2.106273],[0.316265,5.944313,1.892785],[-2.86347,2.776049,2.77242],[-2.673644,3.116508,2.907104],[-2.621149,4.018502,2.903409],[-2.573447,5.198013,2.477481],[1.104039,2.278985,3.722469],[-4.602743,4.306413,0.902296],[-2.684878,1.510731,0.535039],[0.092036,8.473269,-0.99413],[-1.280472,5.602393,1.928105],[-1.0279,4.121582,-1.403103],[-2.461081,3.304477,2.957317],[-2.375929,3.659383,2.953233],[1.417579,2.715389,3.718767],[0.819727,2.948823,3.810639],[1.329962,0.761779,3.203724],[1.73952,5.295229,2.537725],[0.952523,3.945016,3.548229],[-2.569498,0.633669,2.84818],[-2.276676,0.757013,2.780717],[-2.013147,7.354429,-0.003202],[0.93143,1.565913,3.600325],[1.249014,1.550556,3.585842],[2.287252,4.072353,3.124544],[-4.7349,7.006244,1.690653],[-3.500602,8.80386,-2.009196],[-0.582629,5.549138,2.000923],[-1.865297,6.356066,1.313593],[-3.212154,2.376143,-0.565593],[2.092889,3.493536,-1.727931],[-2.528501,2.784531,2.833758],[-2.565697,4.893154,2.559605],[-2.153366,5.04584,2.465215],[1.631311,2.568241,3.681445],[2.150193,4.699227,2.807505],[0.507599,5.01813,2.775892],[4.129862,1.863698,2.015101],[3.578279,4.50766,-0.009598],[3.491023,4.806749,1.549265],[0.619485,1.625336,3.605125],[1.107499,2.932557,3.790061],[-2.082292,6.99321,0.742601],[4.839909,1.379279,0.945274],[3.591328,4.322645,-0.259497],[1.055245,0.710686,3.16553],[-3.026494,7.842227,1.624553],[0.146569,6.119214,0.981673],[-2.043687,2.614509,2.785526],[-2.302242,3.047775,2.936355],[-2.245686,4.100424,2.87794],[2.116148,5.063507,2.572204],[-1.448406,7.64559,0.251692],[2.550717,4.9268,2.517526],[-2.955456,7.80293,-1.782407],[1.882995,4.637167,2.895436],[-2.014924,3.398262,2.954896],[-2.273654,4.771227,2.611418],[-2.162723,7.876761,0.702473],[-0.198659,5.823062,1.739272],[-1.280908,2.133189,-0.921241],[2.039932,4.251568,3.136579],[1.477815,4.354333,3.108325],[0.560504,3.744128,3.6913],[-2.234018,1.054373,2.352782],[-3.189156,7.686661,-2.514955],[-3.744736,7.69963,2.116973],[-2.283366,2.878365,2.87882],[-2.153786,4.457481,2.743529],[4.933978,1.677287,0.713773],[3.502146,0.535336,1.752511],[1.825169,4.419253,3.081198],[3.072331,0.280979,0.106534],[-0.508381,1.220392,2.878049],[-3.138824,8.445394,-1.659711],[-2.056425,2.954815,2.897241],[-2.035343,5.398477,2.215842],[-3.239915,7.126798,-0.712547],[-1.867923,7.989805,0.526518],[1.23405,6.248973,1.387189],[-0.216492,8.320933,-0.862495],[-2.079659,3.755709,2.928563],[-1.78595,4.300374,2.805295],[-1.856589,5.10678,2.386572],[-1.714362,5.544778,2.004623],[1.722403,4.200291,-1.408161],[0.195386,0.086928,-1.318006],[1.393693,3.013404,3.710686],[-0.415307,8.508471,-0.996883],[-1.853777,0.755635,2.757275],[-1.724057,3.64533,2.884251],[-1.884511,4.927802,2.530885],[-1.017174,7.783908,-0.227078],[-1.7798,2.342513,2.741749],[-1.841329,3.943996,2.88436],[1.430388,5.468067,2.503467],[-2.030296,0.940028,2.611088],[-1.677028,1.215666,2.607771],[-1.74092,2.832564,2.827295],[4.144673,0.631374,0.503358],[4.238811,0.653992,0.762436],[-1.847016,2.082815,2.642674],[4.045764,3.194073,0.852117],[-1.563989,8.112739,0.303102],[-1.781627,1.794836,2.602338],[-1.493749,2.533799,2.797251],[-1.934496,4.690689,2.658999],[-1.499174,5.777946,1.747498],[-2.387409,0.851291,1.500524],[-1.872211,8.269987,0.392533],[-4.647726,6.765771,0.833653],[-3.157482,0.341958,-0.20671],[-1.725766,3.24703,2.883579],[-1.458199,4.079031,2.836325],[-1.621548,4.515869,2.719266],[-1.607292,4.918914,2.505881],[-1.494661,5.556239,1.991599],[-1.727269,7.423769,0.012337],[-1.382497,1.161322,2.640222],[-1.52129,4.681714,2.615467],[-4.247127,2.792812,1.250843],[-1.576338,0.742947,2.769799],[-1.499257,2.172763,2.743142],[-1.480392,3.103261,2.862262],[1.049137,2.625836,3.775384],[-1.368063,1.791587,2.695516],[-1.307839,2.344534,2.767575],[-1.336758,5.092221,2.355225],[-1.5617,5.301749,2.21625],[-1.483362,8.537704,0.196752],[-1.517348,8.773614,0.074053],[-1.474302,1.492731,2.641433],[2.48718,0.644247,-0.920226],[0.818091,0.422682,3.171218],[-3.623398,6.930094,3.033045],[1.676333,3.531039,3.591591],[1.199939,5.683873,2.365623],[-1.223851,8.841201,0.025414],[-1.286307,3.847643,2.918044],[-1.25857,4.810831,2.543605],[2.603662,5.572146,1.991854],[0.138984,5.779724,2.077834],[-1.267039,3.175169,2.890889],[-1.293616,3.454612,2.911774],[-2.60112,1.277184,0.07724],[2.552779,3.649877,3.163643],[-1.038983,1.248011,2.605933],[-1.288709,4.390967,2.761214],[-1.034218,5.485963,2.011467],[-1.185576,1.464842,2.624335],[-1.045682,2.54896,2.761102],[4.259176,1.660627,2.018096],[-0.961707,1.717183,2.598342],[-1.044603,3.147464,2.855335],[-0.891998,4.685429,2.669696],[-1.027561,5.081672,2.377939],[4.386506,0.832434,0.510074],[-1.014225,9.064991,-0.175352],[-1.218752,2.895443,2.823785],[-0.972075,4.432669,2.788005],[-2.714986,0.52425,1.509798],[-0.699248,1.517219,2.645738],[-1.161581,2.078852,2.722795],[-0.845249,3.286247,2.996471],[1.068329,4.443444,2.993863],[3.98132,3.715557,1.027775],[1.658097,3.982428,-1.651688],[-4.053701,2.449888,0.734746],[-0.910935,2.214149,2.702393],[0.087824,3.96165,3.439344],[-0.779714,3.724134,2.993429],[-1.051093,3.810797,2.941957],[-0.644941,4.3859,2.870863],[-2.98403,8.666895,-3.691888],[-0.754304,2.508325,2.812999],[-4.635524,3.662891,0.913005],[-0.983299,4.125978,2.915378],[4.916497,1.905209,0.621315],[4.874983,1.728429,0.468521],[2.33127,5.181957,2.441697],[-0.653711,2.253387,2.7949],[-3.623744,8.978795,-2.46192],[-4.555927,6.160279,0.215755],[-4.940628,5.806712,1.18383],[3.308506,2.40326,-0.910776],[0.58835,5.251928,-0.992886],[2.152215,5.449733,2.331679],[-0.712755,0.766765,3.280375],[-0.741771,1.9716,2.657235],[-4.828957,5.566946,2.635623],[-3.474788,8.696771,-1.776121],[1.770417,6.205561,1.331627],[-0.620626,4.064721,2.968972],[-1.499187,2.307735,-0.978901],[4.098793,2.330245,1.667951],[1.940444,6.167057,0.935904],[-2.314436,1.104995,1.681277],[-2.733629,7.742793,1.7705],[-0.452248,4.719868,2.740834],[-0.649143,4.951713,2.541296],[-0.479417,9.43959,-0.676324],[-2.251853,6.559275,0.046819],[0.033531,8.316907,-0.789939],[-0.513125,0.995673,3.125462],[-2.637602,1.039747,0.602434],[1.527513,6.230089,1.430903],[4.036124,2.609846,1.506498],[-3.559828,7.877892,1.228076],[-4.570736,4.960193,0.838201],[-0.432121,5.157731,2.467518],[-1.206735,4.562511,-1.237054],[-0.823768,3.788746,-1.567481],[-3.095544,7.353613,-1.024577],[-4.056088,7.631119,2.062001],[-0.289385,5.382261,2.329421],[1.69752,6.136483,1.667037],[-0.168758,5.061138,2.617453],[2.853576,1.605528,-1.229958],[-4.514319,6.586675,0.352756],[-2.558081,7.741151,1.29295],[1.61116,5.92358,2.071534],[3.936921,3.354857,0.091755],[-0.1633,1.119272,3.147975],[0.067551,1.593475,3.38212],[-1.303239,2.328184,-1.011672],[-0.438093,0.73423,3.398384],[-4.62767,3.898187,0.849573],[0.286853,4.165281,3.284834],[-2.968052,8.492812,-3.493693],[-0.111896,3.696111,3.53791],[-3.808245,8.451731,-1.574742],[0.053416,5.558764,2.31107],[3.956269,3.012071,0.11121],[-0.710956,8.106561,-0.665154],[0.234725,2.717326,3.722379],[-0.031594,2.76411,3.657347],[-0.017371,4.700633,2.81911],[0.215064,5.034859,2.721426],[-0.111151,8.480333,-0.649399],[3.97942,3.575478,0.362219],[0.392962,4.735392,2.874321],[4.17015,2.085087,1.865999],[0.169054,1.244786,3.337709],[0.020049,3.165818,3.721736],[0.248212,3.595518,3.698376],[0.130706,5.295541,2.540034],[-4.541357,4.798332,1.026866],[-1.277485,1.289518,-0.667272],[3.892133,3.54263,-0.078056],[4.057379,3.03669,0.997913],[0.287719,0.884758,3.251787],[0.535771,1.144701,3.400096],[0.585303,1.399362,3.505353],[0.191551,2.076246,3.549355],[0.328656,2.394576,3.649623],[0.413124,3.240728,3.771515],[0.630361,4.501549,2.963623],[0.529441,5.854392,2.120225],[3.805796,3.769958,-0.162079],[3.447279,4.344846,-0.467276],[0.377618,5.551116,2.426017],[0.409355,1.821269,3.606333],[0.719959,2.194726,3.703851],[0.495922,3.501519,3.755661],[0.603408,5.354097,2.603088],[-4.605056,7.531978,1.19579],[0.907972,0.973128,3.356513],[0.750134,3.356137,3.765847],[0.4496,3.993244,3.504544],[-3.030738,7.48947,-1.259169],[0.707505,5.602005,2.43476],[0.668944,0.654891,3.213797],[0.593244,2.700978,3.791427],[1.467759,3.30327,3.71035],[3.316249,2.436388,2.581175],[3.26138,1.724425,2.539028],[-1.231292,7.968263,0.281414],[-0.108773,8.712307,-0.790607],[4.445684,1.819442,1.896988],[1.998959,2.281499,3.49447],[2.162269,2.113817,3.365449],[4.363397,1.406731,1.922714],[4.808,2.225842,0.611127],[2.735919,0.771812,-0.701142],[1.897735,2.878428,3.583482],[-3.31616,5.331985,3.212394],[-3.3314,6.018137,3.313018],[-3.503183,6.480103,3.222216],[-1.904453,5.750392,1.913324],[-1.339735,3.559592,-1.421817],[-1.044242,8.22539,0.037414],[1.643492,3.110676,3.647424],[3.992832,3.686244,0.710946],[1.774207,1.71842,3.475768],[-3.438842,5.5713,3.427818],[4.602447,1.2583,1.619528],[-0.925516,7.930042,0.072336],[-1.252093,3.846565,-1.420761],[-3.426857,5.072419,2.97806],[-3.160408,6.152629,3.061869],[3.739931,3.367082,2.041273],[1.027419,4.235891,3.251253],[4.777703,1.887452,1.560409],[-3.318528,6.733796,2.982968],[2.929265,4.962579,2.271079],[3.449761,2.838629,2.474576],[-3.280159,5.029875,2.787514],[4.068939,2.993629,0.741567],[0.303312,8.70927,-1.121972],[0.229852,8.981322,-1.186075],[-0.011045,9.148156,-1.047057],[-2.942683,5.579613,2.929297],[-3.145409,5.698727,3.205778],[-3.019089,6.30887,2.794323],[-3.217135,6.468191,2.970032],[-3.048298,6.993641,2.623378],[-3.07429,6.660982,2.702434],[3.612011,2.5574,2.25349],[2.54516,4.553967,2.75884],[-1.683759,7.400787,0.250868],[-1.756066,7.463557,0.448031],[-3.023761,5.149697,2.673539],[3.112376,2.677218,2.782378],[2.835327,4.581196,2.567146],[-2.973799,7.225458,2.506988],[-0.591645,8.740662,-0.505845],[3.782861,2.04337,2.03066],[3.331604,3.36343,2.605047],[2.966866,1.205497,2.537432],[0.002669,9.654748,-1.355559],[2.632801,0.58497,2.540311],[-2.819398,5.087372,2.521098],[2.616193,5.332961,2.194288],[-3.193973,4.925634,2.607924],[-3.12618,5.27524,2.944544],[-0.426003,8.516354,-0.501528],[2.802717,1.387643,2.751649],[-3.120597,7.889111,-2.75431],[2.636648,1.71702,2.991302],[-2.853151,6.711792,2.430276],[-2.843836,6.962865,2.400842],[1.9696,3.199023,3.504514],[-2.461751,0.386352,3.008994],[1.64127,0.495758,3.02958],[-4.330472,5.409831,0.025287],[-2.912387,5.980416,2.844261],[-2.490069,0.211078,2.985391],[3.581816,4.809118,0.733728],[2.693199,2.647213,3.126709],[-0.182964,8.184108,-0.638459],[-2.226855,0.444711,2.946552],[-0.720175,8.115055,0.017689],[2.645302,4.316212,2.850139],[-0.232764,9.329503,-0.918639],[4.852365,1.471901,0.65275],[2.76229,2.014994,2.957755],[-2.808374,5.354301,2.644695],[-2.790967,6.406963,2.547985],[-1.342684,0.418488,-1.669183],[2.690675,5.593587,-0.041236],[4.660146,1.6318,1.713314],[2.775667,3.007229,3.111332],[-0.396696,8.963432,-0.706202],[2.446707,2.740617,3.321433],[-4.803209,5.884634,2.603672],[-2.652003,1.6541,1.5078],[3.932327,3.972874,0.831924],[2.135906,0.955587,2.986608],[2.486131,2.053802,3.124115],[-0.386706,8.115753,-0.37565],[-2.720727,7.325044,2.224878],[-1.396946,7.638016,-0.16486],[-0.62083,7.989771,-0.144413],[-2.653272,5.729684,2.667679],[3.038188,4.65835,2.364142],[2.381721,0.739472,2.788992],[-2.345829,5.474929,2.380633],[-2.518983,6.080562,2.479383],[-2.615793,6.839622,2.186116],[-2.286566,0.143752,2.766848],[-4.771219,6.508766,1.070797],[3.717308,2.905019,2.097994],[2.50521,3.016743,3.295898],[2.208448,1.56029,3.216806],[3.346783,1.01254,2.119951],[2.653503,3.26122,3.175738],[-2.359636,5.827519,2.402297],[-1.952693,0.558102,2.853307],[-0.321562,9.414885,-1.187501],[3.138923,1.405072,2.520765],[1.493728,1.780051,3.621969],[3.01817,0.907291,2.336909],[3.183548,1.185297,2.352175],[1.608619,5.006753,2.695131],[-4.723919,6.836107,1.095288],[-1.017586,8.865429,-0.149328],[4.730762,1.214014,0.64008],[-2.135182,6.647907,1.495471],[-2.420382,6.546114,2.108209],[-2.458053,7.186346,1.896623],[3.437124,0.275798,1.138203],[0.095925,8.725832,-0.926481],[2.417376,2.429869,3.287659],[2.279951,1.200317,3.049994],[2.674753,2.326926,3.044059],[-2.328123,6.849164,1.75751],[-3.418616,7.853407,0.126248],[-3.151587,7.77543,-0.110889],[2.349144,5.653242,2.05869],[-2.273236,6.085631,2.242888],[-4.560601,4.525342,1.261241],[2.866334,3.796067,2.934717],[-2.17493,6.505518,1.791367],[3.12059,3.283157,2.818869],[3.037703,3.562356,2.866653],[0.066233,9.488418,-1.248237],[2.749941,0.975018,2.573371],[-2.155749,5.801033,2.204009],[-2.162778,6.261889,2.028596],[1.936874,0.459142,2.956718],[3.176249,4.335541,2.440447],[4.356599,1.029423,1.700589],[3.873502,3.082678,1.80431],[2.895489,4.243034,2.735259],[-0.095774,9.468195,-1.07451],[-1.124982,7.886808,-0.480851],[3.032304,3.065454,2.897927],[3.692687,4.5961,0.957858],[-3.013045,3.807235,-1.098381],[-0.790012,8.92912,-0.367572],[1.905793,0.73179,2.996728],[3.530396,3.426233,2.356583],[2.12299,0.624933,2.929167],[-2.069196,6.039284,2.01251],[-3.565623,7.182525,2.850039],[2.959264,2.376337,2.829242],[2.949071,1.822483,2.793933],[4.036142,0.763803,1.703744],[-1.993527,6.180318,1.804936],[-0.030987,0.766389,3.344766],[-0.549683,8.225193,-0.189341],[-0.765469,8.272246,-0.127174],[-2.947047,7.541648,-0.414113],[-3.050327,9.10114,-3.435619],[3.488566,2.231807,2.399836],[3.352283,4.727851,1.946438],[4.741011,2.162773,1.499574],[-1.815093,6.072079,1.580722],[-3.720969,8.267927,-0.984713],[1.932826,3.714052,3.427488],[3.323617,4.438961,2.20732],[0.254111,9.26364,-1.373244],[-1.493384,7.868585,-0.450051],[-0.841901,0.776135,-1.619467],[0.243537,6.027668,0.091687],[0.303057,0.313022,-0.531105],[-0.435273,0.474098,3.481552],[2.121507,2.622389,3.486293],[1.96194,1.101753,3.159584],[3.937991,3.407551,1.551392],[0.070906,0.295753,1.377185],[-1.93588,7.631764,0.651674],[-2.523531,0.744818,-0.30985],[2.891496,3.319875,2.983079],[4.781765,1.547061,1.523129],[-2.256064,7.571251,0.973716],[3.244861,3.058249,2.724392],[-0.145855,0.437775,3.433662],[1.586296,5.658538,2.358487],[3.658336,3.774921,2.071837],[2.840463,4.817098,2.46376],[-1.219464,8.122542,-0.672808],[-2.520906,2.664486,-1.034346],[-1.315417,8.471365,-0.709557],[3.429165,3.74686,2.446169],[3.074579,3.840758,2.767409],[3.569443,3.166337,2.333647],[2.294337,3.280051,3.359346],[2.21816,3.66578,3.269222],[2.158662,4.151444,-1.357919],[1.13862,4.380986,-1.404565],[3.388382,2.749931,-0.840949],[3.059892,5.084848,2.026066],[3.204739,2.075145,2.640706],[3.387065,1.42617,2.305275],[3.910398,2.670742,1.750179],[3.471512,1.945821,2.395881],[4.08082,1.070654,1.960171],[-1.057861,0.133036,2.146707],[-0.151749,5.53551,-0.624323],[3.233099,4.003778,2.571172],[2.611726,5.319199,-0.499388],[2.682909,1.094499,-1.206247],[-1.22823,7.656887,0.041409],[-2.293247,7.259189,0.013844],[0.081315,0.202174,3.286381],[-1.002038,5.794454,-0.187194],[3.448856,4.08091,2.258325],[0.287883,9.006888,-1.550641],[-3.851019,4.059839,-0.646922],[3.610966,4.205438,1.913129],[2.239042,2.950872,3.449959],[0.216305,0.442843,3.328052],[1.87141,2.470745,3.574559],[3.811378,2.768718,-0.228364],[2.511081,1.362724,2.969349],[-1.59813,7.866506,0.440184],[-3.307975,2.851072,-0.894978],[-0.107011,8.90573,-0.884399],[-3.855315,2.842597,-0.434541],[2.517853,1.090768,2.799687],[3.791709,2.36685,2.002703],[4.06294,2.773922,0.452723],[-2.973289,7.61703,-0.623653],[-2.95509,8.924462,-3.446319],[2.861402,0.562592,2.184397],[-1.109725,8.594206,-0.076812],[-0.725722,7.924485,-0.381133],[-1.485587,1.329994,-0.654405],[-4.342113,3.233735,1.752922],[-2.968049,7.955519,-2.09405],[-3.130948,0.446196,0.85287],[-4.958475,5.757329,1.447055],[-3.086547,7.615193,-1.953168],[-3.751923,5.412821,3.373373],[-4.599645,7.480953,1.677134],[1.133992,0.274871,0.032249],[-2.956512,8.126905,-1.785461],[-0.960645,4.73065,-1.191786],[-2.871064,0.875559,0.424881],[-4.932114,5.99614,1.483845],[-2.981761,8.124612,-1.387276],[0.362298,8.978545,-1.368024],[-4.408375,3.046271,0.602373],[2.865841,2.322263,-1.344625],[-4.7848,5.620895,0.594432],[-2.88322,0.338931,1.67231],[-4.688101,6.772931,1.872318],[-4.903948,6.164698,1.27135],[2.85663,1.005647,-0.906843],[2.691286,0.209811,0.050512],[-4.693636,6.477556,0.665796],[-4.472331,6.861067,0.477318],[0.883065,0.204907,3.073933],[-0.995867,8.048729,-0.653897],[-0.794663,5.670397,-0.390119],[3.313153,1.638006,-0.722289],[-4.856459,5.394758,1.032591],[-3.005448,7.783023,-0.819641],[3.11891,2.036974,-1.08689],[-2.364319,2.408419,2.63419],[-2.927132,8.75435,-3.537159],[-3.296222,7.964629,-3.134625],[-1.642041,4.13417,-1.301665],[2.030759,0.176372,-1.030923],[-4.559069,3.751053,0.548453],[3.438385,4.59454,-0.243215],[-2.561769,7.93935,0.177696],[2.990593,1.335314,-0.943177],[1.2808,0.276396,-0.49072],[-0.318889,0.290684,0.211143],[3.54614,3.342635,-0.767878],[-3.073372,7.780018,-2.357807],[-4.455388,4.387245,0.361038],[-4.659393,6.276064,2.767014],[0.636799,4.482223,-1.426284],[-2.987681,8.072969,-2.45245],[-2.610445,0.763554,1.792054],[3.358241,2.006707,-0.802973],[-0.498347,0.251594,0.962885],[3.1322,0.683312,2.038777],[-4.389801,7.493776,0.690247],[0.431467,4.22119,-1.614215],[-4.376181,3.213141,0.273255],[-4.872319,5.715645,0.829714],[-4.826893,6.195334,0.849912],[3.516562,2.23732,-0.677597],[3.131656,1.698841,-0.975761],[-4.754925,5.411666,1.989303],[-2.987299,7.320765,-0.629479],[-3.757635,3.274862,-0.744022],[3.487044,2.541999,-0.699933],[-4.53274,4.649505,0.77093],[-1.424192,0.099423,2.633327],[3.090867,2.476975,-1.146957],[-2.713256,0.815622,2.17311],[3.348121,3.254167,-0.984896],[-3.031379,0.16453,-0.309937],[-0.949757,4.518137,-1.309172],[-0.889509,0.095256,1.288803],[3.539594,1.966105,-0.553965],[-4.60612,7.127749,0.811958],[-2.332953,1.444713,1.624548],[3.136293,2.95805,-1.138272],[3.540808,3.069058,-0.735285],[3.678852,2.362375,-0.452543],[-4.648898,7.37438,0.954791],[-0.646871,0.19037,3.344746],[2.2825,0.29343,-0.826273],[-4.422291,7.183959,0.557517],[-4.694668,5.246103,2.541768],[-4.583691,4.145486,0.600207],[-2.934854,7.912513,-1.539269],[-3.067861,7.817472,-0.546501],[3.825095,3.229512,-0.237547],[2.532494,0.323059,2.387105],[-2.514583,0.692857,1.23597],[-4.736805,7.214384,1.259421],[-2.98071,8.409903,-2.468199],[2.621468,1.385844,-1.406355],[3.811447,3.560855,1.847828],[3.432925,1.497205,-0.489784],[3.746609,3.631538,-0.39067],[3.594909,2.832257,-0.576012],[-0.404192,5.300188,-0.856561],[-4.762996,6.483774,1.702648],[-4.756612,6.786223,1.43682],[-2.965309,8.437217,-2.785495],[2.863867,0.74087,-0.429684],[4.02503,2.968753,1.392419],[3.669036,1.833858,-0.304971],[-2.888864,0.720537,0.778057],[-2.36982,0.979443,1.054447],[-2.959259,8.222303,-2.659724],[-3.467825,7.545739,-2.333445],[2.153426,0.446256,-1.20523],[-3.229807,9.189699,-3.596609],[-3.72486,8.773707,-2.046671],[3.687218,3.297751,-0.523746],[1.381025,0.08815,-1.185668],[-2.796828,7.205622,-0.208783],[3.647194,4.066232,-0.291507],[-4.578376,3.885556,1.52546],[-2.840262,0.63094,1.89499],[-2.429514,0.922118,1.820781],[-4.675079,6.573925,2.423363],[2.806207,4.320188,-1.027372],[-1.289608,0.097241,1.321661],[-3.010731,8.141334,-2.866148],[3.202291,1.235617,-0.549025],[4.094792,2.477519,0.304581],[2.948403,0.966873,-0.664857],[-4.83297,5.920587,2.095461],[-2.169693,7.257277,0.946184],[-1.335807,3.057597,-1.303166],[-1.037877,0.64151,-1.685271],[2.627919,0.089814,0.439074],[3.815794,3.808102,1.730493],[-2.973455,8.433141,-3.08872],[-2.391558,7.331428,1.658264],[-4.333107,4.529978,1.850516],[-4.640293,3.767107,1.168841],[3.600716,4.46931,1.734024],[3.880803,1.730158,-0.172736],[3.814183,4.262372,1.167042],[4.37325,0.829542,1.413729],[2.490447,5.75111,0.011492],[3.460003,4.962436,1.188971],[3.918419,3.814234,1.358271],[-0.807595,8.840504,-0.953711],[3.752855,4.20577,1.57177],[-2.991085,8.816501,-3.244595],[-2.333196,7.128889,1.551985],[3.977718,3.570941,1.25937],[4.360071,0.755579,1.079916],[4.637579,1.027973,1.032567],[-2.317,7.421066,1.329589],[-1.013404,8.293662,-0.7823],[4.548023,1.020644,1.420462],[4.763258,1.266798,1.296203],[4.896,2.073084,1.255213],[4.015005,3.325226,1.093879],[4.94885,1.860936,0.894463],[-2.189645,6.954634,1.270077],[4.887442,1.720992,1.288526],[-3.184068,7.871802,0.956189],[-1.274318,0.839887,-1.224389],[-2.919521,7.84432,0.541629],[-2.994586,7.766102,1.96867],[-3.417504,9.241714,-3.093201],[-3.174563,7.466456,2.473617],[-3.263067,9.069412,-3.003459],[-2.841592,0.529833,2.693434],[-3.611069,9.158804,-2.829871],[-4.642828,5.927526,0.320549],[-3.809308,9.051035,-2.692749],[-2.837582,7.487987,-0.106206],[4.773025,2.330442,1.213899],[4.897435,2.209906,0.966657],[-3.067637,8.164062,-1.12661],[-3.122129,8.08074,-0.899194],[4.571019,2.358113,1.462054],[4.584884,2.454418,0.709466],[-3.661093,7.146581,-0.475948],[4.735131,2.415859,0.933939],[4.207556,2.540018,1.218293],[-3.607595,7.89161,-0.121172],[-1.527952,0.775564,-1.061903],[4.53874,2.503273,1.099583],[-3.938837,7.587988,0.082449],[-4.853582,6.152409,1.787943],[-4.752214,6.247234,2.296873],[4.602935,2.363955,0.488901],[-1.81638,6.365879,0.868272],[0.595467,4.744074,-1.32483],[1.87635,3.511986,-1.842924],[4.330947,2.534326,0.720503],[4.108736,2.750805,0.904552],[-1.890939,8.492628,-0.290768],[-3.504309,6.173058,-0.422804],[-1.611992,6.196732,0.648736],[-3.899149,7.826123,1.088845],[-3.078303,3.008813,-1.035784],[-2.798999,7.844899,1.340061],[-1.248839,5.959105,0.041761],[0.767779,4.337318,3.090817],[-3.831177,7.515605,2.432261],[-1.667528,6.156208,0.365267],[-1.726078,6.237384,1.100059],[-3.972037,4.520832,-0.370756],[-4.40449,7.636357,1.520425],[-1.34506,6.004054,1.293159],[-1.233556,6.049933,0.500651],[-3.696869,7.79732,0.37979],[-3.307798,8.949964,-2.698113],[-1.997295,6.615056,1.103691],[-3.219222,8.336394,-1.150614],[-3.452623,8.31866,-0.9417],[-3.94641,2.990494,2.212592],[-3.250025,8.030414,-0.596097],[-2.02375,1.571333,2.397939],[-3.190358,7.665013,2.268183],[-2.811918,7.618526,2.145587],[-1.005265,5.892303,0.072158],[-0.93721,5.974148,0.906669],[-4.646072,7.492193,1.45312],[-0.252931,1.797654,3.140638],[-1.076064,5.738433,1.695953],[-3.980534,7.744391,1.735791],[-0.721187,5.939396,0.526032],[-0.42818,5.919755,0.229001],[-1.43429,6.11622,0.93863],[-0.985638,5.939683,0.290636],[-4.433836,7.461372,1.966437],[-3.696398,7.844859,1.547325],[-3.390772,7.820186,1.812204],[-2.916787,7.864019,0.804341],[-3.715952,8.037269,-0.591341],[-4.204634,7.72919,1.119866],[-4.592233,5.592883,0.246264],[3.307299,5.061701,1.622917],[-3.515159,7.601467,2.368914],[-3.435742,8.533457,-1.37916],[-0.269421,4.545635,-1.366445],[-2.542124,3.768736,-1.258512],[-3.034003,7.873773,1.256854],[-2.801399,7.856028,1.080137],[3.29354,5.220894,1.081767],[-2.35109,1.299486,1.01206],[-3.232213,7.768136,2.047563],[3.290415,5.217525,0.68019],[-3.415109,7.731034,2.144326],[3.440357,4.962463,0.373387],[3.147346,5.352121,1.386923],[2.847252,5.469051,1.831981],[3.137682,5.410222,1.050188],[3.102694,5.310456,1.676434],[-3.044601,0.39515,1.994084],[2.903647,5.561338,1.518598],[-3.810148,8.093598,-0.889131],[4.234835,0.803054,1.593271],[3.240165,5.228747,0.325955],[3.037452,5.509825,0.817137],[2.635031,5.795187,1.439724],[3.071607,5.318303,0.080142],[2.909167,5.611751,1.155874],[3.044889,5.465928,0.486566],[2.502256,5.770673,1.740054],[-0.067497,0.086416,-1.190239],[2.33326,5.906051,0.138295],[0.65096,4.205423,3.308767],[-2.671137,7.936535,0.432731],[2.14463,5.879214,1.866047],[-4.776469,5.890689,0.561986],[2.72432,5.655145,0.211951],[2.730488,5.751455,0.695894],[2.572682,5.869295,1.152663],[1.906776,5.739123,2.196551],[2.344414,5.999961,0.772922],[-3.377905,7.448708,-1.863251],[2.285149,5.968156,1.459258],[2.385989,5.928974,0.3689],[2.192111,6.087516,0.959901],[2.36372,6.001101,1.074346],[1.972022,6.079603,1.591175],[1.87615,5.976698,1.91554],[-3.824761,9.05372,-2.928615],[2.044704,6.129704,1.263111],[-2.583046,0.849537,2.497344],[-0.078825,2.342205,3.520322],[-0.704686,0.537165,3.397194],[-0.257449,3.235334,3.647545],[-0.332064,1.448284,3.022583],[-2.200146,0.898284,-0.447212],[-2.497508,1.745446,1.829167],[0.30702,4.416315,2.978956],[-3.205197,3.479307,-1.040582],[0.110069,9.347725,-1.563686],[-0.82754,0.883886,3.065838],[-2.017103,1.244785,2.42512],[-0.421091,2.309929,3.153898],[-0.491604,3.796072,3.16245],[2.786955,3.501241,-1.340214],[-3.229055,4.380713,-0.899241],[3.730768,0.76845,1.90312],[-0.561079,2.652382,3.152463],[-3.461471,3.086496,2.662505],[-0.661405,3.446009,3.179939],[-0.915351,0.636755,3.243708],[-2.992964,8.915628,-3.729833],[-0.439627,3.502104,3.42665],[-1.154217,0.883181,2.800835],[-1.736193,1.465474,2.595489],[-0.423928,3.24435,3.548277],[-0.511153,2.871046,3.379749],[-0.675722,2.991756,3.143262],[-1.092602,0.599103,3.090639],[-0.89821,2.836952,2.840023],[-2.658412,0.781376,0.960575],[-2.271455,1.222857,1.330478],[-0.877861,1.111222,2.72263],[-0.306959,2.876987,3.556044],[-3.839274,7.84138,-0.918404],[-0.172094,4.083799,3.141708],[-1.548332,0.2529,2.864655],[-0.217353,4.873911,-1.223104],[-3.384242,3.181056,-0.95579],[-2.731704,0.382421,2.895502],[-1.285037,0.551267,2.947675],[0.077224,4.246579,3.066738],[-0.479979,1.77955,2.860011],[-0.716375,1.224694,2.666751],[-0.54622,3.138255,3.393457],[-2.33413,1.821222,2.124883],[-0.50653,2.037147,2.897465],[2.451291,1.211389,-1.466589],[-3.160047,2.894081,2.724286],[-4.137258,5.433431,3.21201],[0.462896,0.320456,-0.174837],[-0.37458,2.609447,3.379253],[-3.095244,0.256205,2.196446],[-4.197985,5.732991,3.262924],[-0.729747,0.246036,0.497036],[-2.356189,5.062,-0.965619],[-1.609036,0.25962,-1.487367],[-4.074381,6.074061,3.409459],[-3.619304,4.0022,2.65705],[-0.543393,8.742896,-1.056622],[-4.30356,6.858934,2.879642],[-0.716688,2.901831,-2.11202],[1.547362,0.083189,1.138764],[-0.250916,0.275268,1.201344],[-3.778035,3.13624,2.466177],[-4.594316,5.771342,3.01694],[-3.717706,3.442887,2.603344],[-4.311163,5.224669,3.019373],[-0.610389,2.095161,-1.923515],[-3.040086,6.196918,-0.429149],[-3.802695,3.768247,2.545523],[-0.159541,2.043362,3.328549],[-3.744329,4.31785,2.491889],[-3.047939,0.214155,1.873639],[-4.41685,6.113058,3.166774],[-1.165133,0.460692,-1.742134],[-1.371289,4.249996,-1.317935],[-3.447883,0.3521,0.466205],[-4.495555,6.465548,2.944147],[-3.455335,0.171653,0.390816],[-3.964028,4.017196,2.376009],[-1.323595,1.763126,-0.750772],[-3.971142,5.277524,-0.19496],[-3.222052,0.237723,0.872229],[-4.403784,3.89107,1.872077],[-3.333311,0.342997,0.661016],[-4.495871,4.29606,1.63608],[-3.636081,2.760711,2.361949],[-4.487235,3.559608,1.66737],[-4.719787,7.26888,1.658722],[-1.086143,9.035741,-0.707144],[-2.339693,1.600485,-0.404817],[-4.642011,7.123829,1.990987],[-1.498077,3.854035,-1.369787],[-4.188372,4.729363,2.02983],[-3.116344,5.882284,-0.468884],[-4.305236,4.246417,1.976991],[-3.022509,0.22819,1.065688],[-2.799916,0.52022,1.128319],[-4.262823,3.534409,2.020383],[-4.221533,3.947676,2.11735],[-3.744353,4.391712,-0.6193],[-1.272905,0.156694,-1.741753],[-3.62491,2.669825,-0.549664],[-4.180756,3.096179,1.987215],[-4.059276,4.305313,2.232924],[-2.812753,0.183226,1.370267],[-4.032437,3.512234,2.309985],[-0.03787,0.28188,0.530391],[-4.711562,5.468653,2.822838],[-4.500636,6.953314,2.564445],[-4.479433,7.216991,2.270682],[3.990562,0.50522,0.716309],[-2.512229,6.863447,-0.100658],[-2.968058,6.956639,-0.37061],[2.550375,3.142683,-1.54068],[-2.320059,3.521605,-1.279397],[-4.556319,6.64662,2.745363],[-4.281091,7.108116,2.667598],[-2.050095,8.411689,0.121353],[-2.44854,1.135487,0.851875],[3.121815,0.699943,-0.277167],[-4.69877,6.00376,2.843035],[-1.360599,8.824742,-0.595597],[1.128437,0.171611,0.301691],[-4.360146,6.289423,0.042233],[1.400795,4.088829,-1.620409],[-3.193462,8.460137,-3.559446],[-3.168771,8.878431,-3.635795],[-3.434275,9.304302,-3.460878],[-3.349993,8.808093,-3.38179],[-3.304823,8.323865,-3.325905],[-3.572607,9.308843,-3.207672],[-3.166393,8.201215,-3.43014],[-3.451638,9.05331,-3.351345],[-3.309591,8.549758,-3.375055],[-3.527992,8.793926,-3.100376],[-3.6287,8.981677,-3.076319],[-3.445505,8.001887,-2.8273],[-3.408011,8.221014,-3.039237],[-3.65928,8.740382,-2.808856],[-3.878019,8.797295,-2.462866],[-3.515132,8.232341,-2.747739],[-3.460331,8.51524,-3.06818],[-3.403703,7.658628,-2.648789],[-3.507113,8.00159,-2.582275],[-3.607373,8.174737,-2.401723],[-3.749043,8.378084,-2.226959],[-3.648514,8.502213,-2.6138],[-2.534199,0.904753,2.021148],[1.4083,5.744252,-0.571402],[-3.852536,8.571009,-2.352358],[2.868255,5.373126,-0.163705],[2.224363,4.669891,-1.061586],[-4.528281,4.885838,1.340274],[1.30817,4.609629,-1.28762],[-4.519698,3.422501,1.354826],[-3.549955,7.783228,-2.332859],[1.12313,6.120856,0.045115],[-3.620324,7.57716,-2.033423],[-0.798833,2.624133,-1.992682],[-3.617587,7.783148,-2.051383],[-3.669293,8.103776,-2.10227],[-3.892417,8.667436,-2.167288],[-0.537435,0.285345,-0.176267],[-0.841522,3.299866,-1.887861],[-0.761547,3.647082,-1.798953],[-3.661544,7.85708,-1.867924],[-3.886763,8.551783,-1.889171],[-0.591244,1.549749,-1.714784],[-0.775276,1.908218,-1.597609],[-0.961458,2.573273,-1.695549],[-2.215672,1.335009,2.143031],[-4.622674,4.130242,1.220683],[1.07344,0.290099,1.584734],[-0.976906,2.92171,-1.76667],[-1.13696,3.194401,-1.513455],[-3.743262,7.99949,-1.629286],[-2.876359,4.900986,-0.879556],[0.550835,3.905557,-2.031372],[0.777647,4.992314,-1.215703],[1.445881,4.266201,-1.414663],[1.274222,5.510543,-0.824495],[-0.864685,2.318581,-1.702389],[-0.627458,3.820722,-1.743153],[-3.867699,8.30866,-1.850066],[1.635287,5.45587,-0.83844],[-1.037876,2.538589,-1.513504],[-4.38993,4.73926,1.699639],[0.048709,4.765232,-1.279506],[-0.626548,1.339887,-1.595114],[-3.682827,7.643453,-1.723398],[-3.868783,8.180191,-1.511743],[-0.76988,1.508373,-1.419599],[-1.138374,2.766765,-1.448163],[1.699883,5.780752,-0.475361],[1.214305,0.308517,1.866405],[-1.713642,0.373461,-1.265204],[-1.582388,0.58294,-1.267977],[-0.879549,1.821581,-1.313787],[0.519057,5.858757,-0.381397],[-3.770989,2.449208,-0.132655],[0.087576,0.156713,-1.53616],[-0.942622,2.146534,-1.421494],[-1.026192,1.022164,-1.145423],[-0.964079,1.645473,-1.067631],[-1.109128,2.458789,-1.29106],[-1.037478,0.209489,-1.805424],[-3.724391,7.599686,-1.273458],[-3.787898,7.951792,-1.304794],[3.821677,2.165581,-0.181535],[-2.39467,0.304606,-0.570375],[-2.352928,1.0439,2.079369],[-0.288899,9.640684,-1.006079],[-3.472118,7.263001,-1.080326],[-1.240769,0.972352,-0.976446],[-1.845253,0.356801,-0.995574],[-2.32279,7.915361,-0.057477],[-1.08092,2.179315,-1.168821],[4.598833,2.156768,0.280264],[-4.725417,6.442373,2.056809],[-0.490347,9.46429,-0.981092],[-1.99652,0.09737,-0.765828],[-1.137793,1.888846,-0.894165],[-0.37247,4.29661,-1.465199],[-0.184631,5.692946,-0.421398],[-3.751694,7.742231,-1.086908],[-1.001416,1.298225,-0.904674],[-3.536884,7.190777,-0.788609],[-3.737597,7.511281,-0.940052],[-1.766651,0.669388,-0.873054],[3.112245,3.474345,-1.129672],[-0.175504,3.81298,-2.0479],[-3.766762,7.412514,-0.681569],[-0.63375,9.439424,-0.785128],[-0.518199,4.768982,-1.258625],[0.790619,4.212759,-1.610218],[-3.761951,3.742528,-0.756283],[0.897483,5.679808,-0.612423],[2.221126,4.427468,-1.252155],[-0.728577,5.846457,0.062702],[0.194451,9.503908,-1.482461],[-0.099243,9.385459,-1.39564],[0.643185,3.636855,-2.180247],[0.894522,5.900601,-0.356935],[2.595516,4.75731,-0.893245],[1.108497,3.936893,-1.905098],[1.989894,5.789726,-0.343268],[-3.802345,7.655508,-0.613817],[2.339353,4.96257,-0.90308],[0.12564,4.013324,-1.879236],[-4.078965,3.683254,-0.445439],[2.092899,5.256128,-0.831607],[0.427571,0.291769,1.272964],[2.335549,3.480056,-1.581949],[-0.15687,0.324827,-1.648922],[-0.536522,5.760786,-0.203535],[1.507082,0.078251,-0.923109],[-1.854742,0.134826,2.698774],[-3.939827,3.168498,-0.526144],[-3.98461,3.39869,-0.533212],[-3.961738,4.217132,-0.489147],[4.273789,2.181164,0.153786],[-0.470498,5.645664,-0.439079],[-0.414539,5.488017,-0.673379],[-0.097462,5.062739,-1.114863],[1.198092,5.882232,-0.391699],[2.855834,5.085022,-0.498678],[1.037998,4.129757,-1.701811],[1.728091,5.068444,-1.063761],[-3.832258,2.625141,-0.311384],[-4.078526,3.070256,-0.284362],[-4.080365,3.954243,-0.440471],[-0.152578,5.276267,-0.929815],[-1.489635,8.928082,-0.295891],[0.759294,5.15585,-1.087374],[-4.000338,2.801647,-0.235135],[-4.290801,3.823209,-0.19374],[-4.221493,4.25618,-0.189894],[-4.066195,4.71916,-0.201724],[-0.155386,4.076396,-1.662865],[3.054571,4.414305,-0.825985],[-1.652919,8.726499,-0.388504],[-3.042753,0.560068,-0.126425],[-2.434456,1.118088,-0.213563],[-2.623502,1.845062,-0.283697],[-4.233371,3.43941,-0.202918],[2.726702,3.82071,-1.280097],[0.184199,4.14639,-1.673653],[-1.289203,0.624562,-1.560929],[-3.823676,7.382458,-0.407223],[0.476667,5.064419,-1.143742],[-3.873651,4.955112,-0.269389],[1.349666,5.312227,-1.000274],[-2.043776,8.434488,-0.108891],[-2.763964,0.733395,-0.129294],[-4.380505,3.664409,-0.024546],[-0.71211,5.341811,-0.803281],[-3.960858,7.183112,-0.118407],[-3.822277,7.712853,-0.263221],[-2.346808,8.108588,0.063244],[-1.841731,8.642999,-0.142496],[-2.600055,0.985604,-0.043595],[-3.513057,2.213243,-0.044151],[-3.963492,2.603055,-0.080898],[-4.258066,3.14537,-0.027046],[-4.261572,5.00334,0.13004],[0.795464,3.99873,-1.905688],[-3.300873,0.384761,0.013271],[-2.770244,0.881942,0.077313],[-3.456227,1.993871,0.301054],[-4.441987,3.914144,0.177867],[-4.367075,6.611414,0.165312],[-3.201767,0.576292,0.105769],[-3.174354,0.645009,0.440373],[-2.996576,0.74262,0.161325],[-2.724979,1.656497,0.092983],[-3.261757,2.017742,-0.070763],[-4.280173,4.518235,-0.002999],[-4.471073,5.945358,0.05202],[-3.877137,2.40743,0.274928],[-4.371219,4.252758,0.078039],[-3.400914,0.40983,0.238599],[-4.44293,3.523242,0.146339],[-4.574528,5.279761,0.353923],[-4.226643,7.191282,0.269256],[-4.16361,2.843204,0.097727],[-4.528506,5.011661,0.536625],[0.35514,5.664802,-0.572814],[2.508711,5.580976,-0.266636],[2.556226,3.633779,-1.426362],[1.878456,4.533714,-1.223744],[2.460709,4.440241,-1.1395],[2.218589,5.514603,-0.560066],[2.263712,5.737023,-0.250694],[2.964981,3.814858,-1.139927],[0.991384,5.304131,-0.999867],[2.81187,4.547292,-0.916025],[2.918089,4.768382,-0.702808],[3.262403,4.414286,-0.657935],[0.652136,6.089113,0.069089],[3.361389,3.5052,-0.946123],[2.613042,5.037192,-0.697153],[0.094339,4.36858,-1.451238],[3.290862,4.155716,-0.732318],[2.658063,4.073614,-1.217455],[3.260349,3.753257,-0.946819],[1.124268,4.862463,-1.207855],[3.35158,4.899247,-0.027586],[3.194057,4.691257,-0.524566],[3.090119,5.116085,-0.23255],[2.418965,3.811753,-1.419399],[2.191789,3.877038,-1.47023],[4.043166,2.034188,0.015477],[-1.026966,0.86766,-1.410912],[1.937563,3.860005,-1.617465],[2.98904,4.101806,-0.998132],[-0.142611,5.865305,-0.100872],[3.972673,2.292069,0.089463],[3.23349,3.959925,-0.849829],[0.16304,5.857276,-0.216704],[4.122964,1.770061,-0.114906],[2.099057,4.978374,-0.98449],[3.502411,3.76181,-0.667502],[2.079484,5.939614,-0.036205],[-0.084568,3.525193,-2.253506],[0.423859,4.06095,-1.845327],[1.6013,6.006466,-0.153429],[0.271701,3.844964,-2.078748],[0.273577,5.218904,-0.994711],[-0.410578,3.92165,-1.773635],[1.941954,5.60041,-0.621569],[0.100825,5.462131,-0.774256],[-0.53016,3.619892,-2.027451],[-0.822371,5.517453,-0.605747],[-2.474925,7.670892,-0.020174],[4.01571,0.830194,-0.013793],[-0.400092,5.094112,-1.041992],[-2.887284,5.581246,-0.525324],[-1.559841,6.050972,0.079301],[-0.469317,3.291673,-2.235211],[0.337397,3.467926,-2.295458],[-2.632074,5.573701,-0.582717],[-0.030318,6.011395,0.276616],[-0.934373,0.388987,-1.780523],[-2.661263,5.844838,-0.425966],[0.549353,5.489646,-0.807268],[-2.194355,6.197491,-0.109322],[-2.289618,5.664813,-0.581098],[1.583583,3.796366,-1.844498],[0.855295,0.215979,-1.425557],[-2.627569,5.300236,-0.767174],[4.333347,2.384332,0.399129],[-1.880401,5.583843,-0.696561],[-2.172346,5.324859,-0.846246],[-2.27058,5.906265,-0.388373],[-1.960049,5.889346,-0.397593],[0.965756,3.67547,-2.105671],[-2.014066,6.431125,0.287254],[-1.776173,5.287097,-0.89091],[-2.025852,5.089562,-0.980218],[-1.886418,6.108358,-0.000667],[-1.600803,5.785347,-0.491069],[-1.66188,4.968053,-1.042535],[-1.600621,5.962818,-0.188044],[-1.588831,5.615418,-0.665456],[4.46901,1.880138,0.057248],[-1.978845,0.927399,-0.554856],[-1.408074,5.325266,-0.83967],[1.923123,4.843955,-1.101389],[-2.87378,0.117106,-0.412735],[-1.222193,5.62638,-0.539981],[-2.632537,0.166349,-0.489218],[-1.370865,5.838832,-0.341026],[-1.067742,5.448874,-0.692701],[-1.073798,5.220878,-0.908779],[-1.147562,4.950417,-1.079727],[-2.789115,4.531047,-1.042713],[-3.550826,4.170487,-0.806058],[-3.331694,4.798177,-0.69568],[-3.689404,4.688543,-0.534317],[-3.511509,5.106246,-0.483632],[1.796344,0.076137,0.080455],[-3.306354,5.473605,-0.478764],[-2.692503,3.346604,-1.20959],[-3.963056,5.187462,3.113156],[-3.901231,6.391477,-0.246984],[4.484234,1.518638,-0.001617],[4.308829,1.657716,-0.119275],[4.290045,1.339528,-0.110626],[-3.514938,3.524974,-0.909109],[-2.1943,2.12163,-0.71966],[4.108206,1.091087,-0.11416],[3.785312,1.392435,-0.28588],[4.092886,1.480476,-0.210655],[-2.965937,6.469006,-0.379085],[-3.708581,2.962974,-0.63979],[-3.297971,2.218917,-0.299872],[3.806949,0.804703,-0.11438],[3.747957,1.059258,-0.273069],[-3.101827,4.111444,-1.006255],[-1.536445,4.658913,-1.195049],[-3.549826,2.450555,-0.375694],[-3.676495,2.108366,0.534323],[-3.674738,5.925075,-0.400011],[-2.250115,2.848335,-1.121174],[-3.698062,5.667567,-0.381396],[3.468966,0.734643,-0.190624],[-3.97972,5.670078,-0.26874],[-3.002087,4.337837,-1.033421],[-3.356392,2.608308,-0.713323],[-1.833016,3.359983,-1.28775],[-1.989069,3.632416,-1.305607],[3.591254,0.542371,0.026146],[3.364927,1.082572,-0.342613],[-3.393759,3.866801,-0.937266],[-4.124865,5.549529,-0.161729],[-4.423423,5.687223,0.000103],[-1.496881,2.601785,-1.114328],[-2.642297,6.496932,-0.264175],[-3.684236,6.819423,-0.320233],[-2.286996,3.167067,-1.246651],[-1.624896,8.44848,-0.530014],[-3.666787,2.159266,0.268149],[-2.402625,2.011243,-0.56446],[-2.736166,2.259839,-0.6943],[-2.168611,3.89078,-1.292206],[-2.065956,3.345708,-1.281346],[-2.778147,2.675605,-0.995706],[-3.507431,4.513272,-0.71829],[-2.301184,4.293911,-1.238182],[3.205808,0.211078,0.394349],[-2.129936,4.870577,-1.080781],[-2.287977,2.496593,-0.934069],[-2.701833,2.931814,-1.114509],[3.294795,0.50631,-0.081062],[-2.552829,7.468771,-0.021541],[3.06721,0.944066,-0.43074],[-2.86086,1.973622,-0.303132],[-3.598818,5.419613,-0.401645],[-1.524381,0.080156,-1.61662],[-1.907291,2.646274,-1.039438],[2.950783,0.407562,-0.105407],[-1.663048,1.655038,-0.689787],[-1.728102,1.110064,-0.635963],[-2.085823,7.686296,-0.159745],[2.883518,3.157009,-1.30858],[-2.724116,0.417169,-0.389719],[-1.788636,7.862672,-0.346413],[-2.186418,1.249609,-0.434583],[-3.092434,2.606657,-0.860002],[-1.737314,3.874201,-1.330986],[2.564522,0.422967,-0.390903],[1.670782,3.538432,-1.924753],[-2.338131,4.02578,-1.286673],[-1.916516,4.054121,-1.301788],[2.87159,2.034949,-1.267139],[-1.931518,3.062883,-1.197227],[-0.816602,0.135682,3.104104],[0.469392,0.213916,-1.489608],[2.574055,1.950091,-1.514427],[2.733595,2.682546,-1.461213],[-1.915407,4.693647,-1.151721],[-3.412883,5.867094,-0.450528],[2.28822,0.120432,-0.04102],[2.244477,0.14424,-0.376933],[-1.676198,3.570698,-1.328031],[-1.821193,4.366982,-1.266271],[-1.552208,8.099221,-0.53262],[-1.727419,2.39097,-0.989456],[-2.468226,4.711663,-1.069766],[-2.451669,6.113319,-0.273788],[2.635447,2.295842,-1.518361],[-2.020809,8.150253,-0.246714],[2.292455,0.805596,-1.3042],[2.641556,1.65665,-1.466962],[2.409062,2.842538,-1.635025],[2.456682,1.459484,-1.57543],[-1.691047,3.173582,-1.247082],[-1.865642,1.957608,-0.768683],[-3.401579,0.20407,0.100932],[2.301981,1.7102,-1.650461],[2.342929,2.611944,-1.690713],[-1.676111,2.923894,-1.17835],[-2.992039,3.547631,-1.118945],[-3.571677,6.504634,-0.375455],[2.141764,1.460869,-1.702464],[-3.221958,5.146049,-0.615632],[2.19238,2.949367,-1.747242],[2.320791,2.232971,-1.706842],[2.088678,2.585235,-1.813159],[-2.196404,0.592218,-0.569709],[-2.120811,1.836483,-0.62338],[-1.949935,2.271249,-0.874128],[2.235901,1.110183,-1.510719],[2.020157,3.241128,-1.803917],[2.054336,1.949394,-1.792332],[-3.094117,4.996595,-0.740238],[2.038063,0.635949,-1.402041],[1.980644,1.684408,-1.76778],[1.587432,3.306542,-1.991131],[1.935322,0.976267,-1.602208],[1.922621,1.235522,-1.698813],[1.712495,1.911874,-1.903234],[1.912802,2.259273,-1.888698],[1.884367,0.355453,-1.312633],[1.676427,0.76283,-1.539455],[1.78453,2.83662,-1.943035],[1.697312,0.120281,-1.150324],[1.648318,2.484973,-1.999505],[-4.051804,5.958472,-0.231731],[-1.964823,1.464607,-0.58115],[1.55996,2.183486,-1.971378],[1.628125,1.045912,-1.707832],[1.701684,1.540428,-1.827156],[1.567475,4.869481,-1.184665],[1.432492,0.843779,-1.648083],[1.173837,2.978983,-2.156687],[1.235287,3.37975,-2.09515],[1.252589,1.525293,-1.949205],[1.159334,2.336379,-2.105361],[1.49061,2.695263,-2.083216],[-4.122486,6.782604,-0.02545],[1.173388,0.279193,-1.423418],[1.505684,0.380815,-1.414395],[1.391423,1.343031,-1.843557],[1.263449,2.73225,-2.144961],[1.295858,0.597122,-1.515628],[1.245851,3.729126,-1.993015],[-2.761439,6.23717,-0.365856],[0.978887,1.664888,-2.046633],[1.219542,0.982729,-1.785486],[1.315915,1.91748,-2.02788],[-3.052746,2.127222,-0.369082],[0.977656,1.36223,-1.944119],[0.936122,3.39447,-2.203007],[-2.740036,4.184702,-1.122849],[0.853581,2.864694,-2.260847],[0.719569,0.818762,-1.763618],[0.839115,1.159359,-1.907943],[0.932069,1.94559,-2.117962],[0.579321,3.326747,-2.299369],[0.86324,0.597822,-1.565106],[0.574567,1.158452,-1.943123],[0.525138,2.137252,-2.213867],[0.779941,2.342019,-2.206157],[0.915255,2.618102,-2.209041],[0.526426,3.02241,-2.321826],[0.495431,2.521396,-2.295905],[0.80799,3.156817,-2.286432],[0.273556,1.304936,-2.012509],[0.664326,1.530024,-2.048722],[0.219173,2.32907,-2.323212],[0.405324,0.695359,-1.704884],[0.398827,0.946649,-1.843899],[0.345109,1.608829,-2.100174],[-2.356743,0.062032,-0.4947],[-3.001084,0.27146,2.560034],[-2.064663,0.303055,-0.697324],[0.221271,3.174023,-2.374399],[0.195842,0.437865,-1.621473],[-0.385613,0.297763,1.960096],[1.999609,0.108928,-0.79125],[0.351698,9.227494,-1.57565],[0.021477,2.191913,-2.309353],[0.246381,2.836575,-2.356365],[1.543281,0.237539,1.901906],[0.031881,9.147022,-1.454203],[-0.001881,1.648503,-2.108044],[0.333423,1.907088,-2.204533],[0.044063,2.634032,-2.368412],[-0.028148,3.053684,-2.390082],[0.02413,3.34297,-2.36544],[-0.272645,9.02879,-1.238685],[-0.006348,0.832044,-1.758222],[-0.321105,1.458754,-1.886313],[-0.153948,8.618809,-1.105353],[-0.409303,1.137783,-1.720556],[-0.410054,1.742789,-1.957989],[-0.287905,2.380404,-2.294509],[-0.261375,2.646629,-2.356322],[-0.221986,3.215303,-2.345844],[-0.31608,0.687581,-1.71901],[-0.537705,0.855802,-1.648585],[-0.142834,1.193053,-1.87371],[-0.24371,2.044435,-2.176958],[-0.437999,2.959748,-2.299698],[-0.78895,0.176226,-1.729046],[-0.608509,0.546932,-1.734032],[-0.693698,4.478782,-1.369372],[-0.669153,8.469645,-0.911149],[-0.741857,1.082705,-1.458474],[-0.554059,2.440325,-2.141785],[2.09261,0.153182,2.57581],[1.792547,0.111794,2.563777],[1.855787,0.189541,2.835089],[1.492601,0.232246,2.987681],[-0.284918,0.236687,3.429738],[2.604841,0.11997,1.01506],[0.331271,0.168113,3.124031],[0.280606,0.308368,2.495937],[0.544591,0.325711,2.081274],[0.193145,0.19154,-0.977556],[3.810099,0.42324,1.032202],[3.54622,0.379245,1.392814],[0.61402,0.276328,0.849356],[-1.198628,0.144953,2.911457],[4.17199,0.68037,1.391526],[0.88279,0.321339,2.059129],[1.93035,0.109992,2.054154],[1.620331,0.121986,2.37203],[2.374812,0.10921,1.734876],[-0.031227,0.294412,2.593687],[4.075018,0.561914,1.038065],[-0.570366,0.126583,2.975558],[0.950052,0.318463,1.804012],[1.130034,0.117125,0.98385],[2.123049,0.08946,1.665911],[2.087572,0.068621,0.335013],[2.927337,0.167117,0.289611],[0.528876,0.313434,3.205969],[1.174911,0.162744,1.328262],[-4.88844,5.59535,1.661134],[-4.709607,5.165338,1.324082],[0.871199,0.277021,1.263831],[-3.910877,2.349318,1.272269],[1.56824,0.118605,2.768112],[1.179176,0.152617,-0.858003],[1.634629,0.247872,2.128625],[-4.627425,5.126935,1.617836],[3.845542,0.54907,1.45601],[2.654006,0.165508,1.637169],[-0.678324,0.26488,1.974741],[2.451139,0.100377,0.213768],[0.633199,0.286719,0.403357],[-0.533042,0.2524,1.373267],[0.99317,0.171106,0.624966],[-0.100063,0.306466,2.170225],[1.245943,0.092351,0.661031],[1.390414,0.198996,-0.0864],[-4.457265,5.030531,2.138242],[2.89776,0.146575,1.297468],[1.802703,0.088824,-0.490405],[1.055447,0.309261,2.392437],[2.300436,0.142429,2.104254],[2.33399,0.187756,2.416935],[2.325183,0.134349,0.574063],[2.410924,0.370971,2.637115],[1.132924,0.290511,3.061],[1.764028,0.070212,-0.80535],[2.156994,0.397657,2.844061],[0.920711,0.225527,-0.882456],[-4.552135,5.24096,2.85514],[0.210016,0.309396,2.064296],[0.612067,0.136815,-1.086002],[3.150236,0.426757,1.802703],[-0.24824,0.282258,1.470997],[0.974269,0.301311,-0.640898],[-4.401413,5.03966,2.535553],[0.644319,0.274006,-0.817806],[0.332922,0.309077,0.108474],[3.610001,0.317447,0.689353],[3.335681,0.358195,0.118477],[0.623544,0.318983,-0.4193],[-0.11012,0.307747,1.831331],[-0.407528,0.291044,2.282935],[0.069783,0.285095,0.950289],[0.970135,0.310392,-0.283742],[0.840564,0.306898,0.098854],[-0.541827,0.267753,1.683795],[-3.956082,4.55713,2.297164],[-4.161036,2.834481,1.64183],[-4.093952,4.977551,2.747747],[2.661819,0.261867,1.926145],[-3.749926,2.161875,0.895238],[-2.497776,1.3629,0.791855],[0.691482,0.304968,1.582939],[-4.013193,4.830963,2.4769],[-3.639585,2.091265,1.304415],[-3.9767,2.563053,1.6284],[-3.979915,2.788616,1.977977],[0.388782,0.312656,1.709168],[-3.40873,1.877324,0.851652],[-3.671637,5.136974,3.170734],[-3.12964,1.852012,0.157682],[-3.629687,4.852698,2.686837],[-3.196164,1.793459,0.452804],[-3.746338,2.31357,1.648551],[2.992192,0.125251,0.575976],[-3.254051,0.054431,0.314152],[-3.474644,1.925288,1.134116],[-3.418372,2.022882,1.578901],[-2.920955,1.705403,0.29842],[-3.57229,2.152022,1.607572],[-3.251259,0.09013,-0.106174],[-3.299952,1.877781,1.348623],[-3.666819,2.441459,2.004838],[-2.912646,1.824748,-0.045348],[-3.399511,2.479484,2.340393],[-3.009754,0.015286,0.075567],[-3.381443,2.316937,2.156923],[-3.352801,2.133341,1.857366],[-3.01788,1.687685,0.645867],[-2.931857,1.678712,1.158472],[-3.301008,0.08836,0.591001],[1.358025,0.19795,1.599144],[-2.999565,1.845016,1.618396],[-2.767957,0.028397,-0.196436],[-2.93962,2.078779,2.140593],[-3.346648,2.674056,2.518097],[3.324322,0.20822,0.628605],[3.091677,0.137202,0.9345],[-2.881807,0.009952,0.318439],[-2.764946,1.786619,1.693439],[-2.905542,1.932343,1.900002],[-3.140854,2.271384,2.274946],[-2.88995,2.487856,2.574759],[-2.367194,-0.000943,-0.15576],[-3.050738,0.068703,0.742988],[-2.759525,1.55679,0.877782],[-3.151775,2.48054,2.482749],[-2.578618,-0.002885,0.165716],[-2.651618,1.877246,1.981189],[-2.933973,0.133731,1.631023],[1.047628,0.100284,-1.085248],[-1.585123,0.062083,-1.394896],[-2.287917,-0.002671,0.214434],[-2.524899,0.007481,0.471788],[-2.815492,2.188198,2.343294],[-2.095142,-0.003149,-0.094574],[-2.172686,-0.000133,0.47963],[-2.732704,0.074306,1.742079],[-2.49653,2.145668,2.42691],[-1.343683,0.047721,-1.506391],[-2.581185,0.048703,0.975528],[-2.905101,0.083158,2.010052],[-2.601514,2.007801,2.223089],[-2.339464,0.02634,1.484304],[-2.907873,0.10367,2.378149],[-1.368796,0.062516,-1.049125],[-1.93244,0.02443,-0.427603],[-2.705081,0.060513,2.303802],[3.372155,0.206274,0.892293],[-1.761827,0.093202,-1.037404],[-1.700667,0.0397,-0.614221],[-1.872291,0.011979,-0.135753],[-1.929257,0.074005,0.728999],[-2.520128,0.049665,1.99054],[-2.699411,0.10092,2.603116],[3.211701,0.27302,1.423357],[-1.445362,0.1371,-0.626491],[2.921332,0.259112,1.645525],[-0.993242,0.058686,-1.408916],[-0.944986,0.157541,-1.097665],[-2.154301,0.032749,1.882001],[-2.108789,1.988557,2.442673],[-1.015659,0.25497,-0.416665],[-1.898411,0.015872,0.16715],[-1.585517,0.027121,0.453445],[-2.311105,0.061264,2.327061],[-2.637042,0.152224,2.832201],[-2.087515,2.292972,2.617585],[-0.750611,0.056697,-1.504516],[-0.472029,0.075654,-1.360203],[-0.710798,0.139244,-1.183863],[-0.97755,0.26052,-0.831167],[-0.655814,0.260843,-0.880068],[-0.897513,0.275537,-0.133042],[-2.049194,0.084947,2.455422],[-0.177837,0.076362,-1.449009],[-0.553393,0.279083,-0.59573],[-1.788636,0.06163,2.231198],[-0.34761,0.255578,-0.999614],[-1.398589,0.036482,0.65871],[-1.133918,0.05617,0.69473],[-1.43369,0.058226,1.977865],[-2.505459,1.492266,1.19295]]\nexports.cells=[[2,1661,3],[1676,7,6],[712,1694,9],[3,1674,1662],[11,1672,0],[1705,0,1],[5,6,1674],[4,5,1674],[7,8,712],[2,1662,10],[1,10,1705],[11,1690,1672],[1705,11,0],[5,1676,6],[7,9,6],[7,712,9],[2,3,1662],[3,4,1674],[1,2,10],[12,82,1837],[1808,12,1799],[1808,1799,1796],[12,861,82],[861,1808,13],[1808,861,12],[1799,12,1816],[1680,14,1444],[15,17,16],[14,1678,1700],[16,17,1679],[15,1660,17],[14,1084,1678],[15,1708,18],[15,18,1660],[1680,1084,14],[1680,15,1084],[15,1680,1708],[793,813,119],[1076,793,119],[1076,1836,22],[23,19,20],[21,1076,22],[21,22,23],[23,20,21],[1076,119,1836],[806,634,470],[432,1349,806],[251,42,125],[809,1171,791],[953,631,827],[634,1210,1176],[157,1832,1834],[56,219,53],[126,38,83],[37,85,43],[59,1151,1154],[83,75,41],[77,85,138],[201,948,46],[1362,36,37],[452,775,885],[1237,95,104],[966,963,1262],[85,77,43],[36,85,37],[1018,439,1019],[41,225,481],[85,83,127],[93,83,41],[935,972,962],[116,93,100],[98,82,813],[41,75,225],[298,751,54],[1021,415,1018],[77,138,128],[766,823,1347],[593,121,573],[905,885,667],[786,744,747],[100,41,107],[604,334,765],[779,450,825],[968,962,969],[225,365,481],[365,283,196],[161,160,303],[875,399,158],[328,1817,954],[62,61,1079],[358,81,72],[74,211,133],[160,161,138],[91,62,1079],[167,56,1405],[56,167,219],[913,914,48],[344,57,102],[43,77,128],[1075,97,1079],[389,882,887],[219,108,53],[1242,859,120],[604,840,618],[754,87,762],[197,36,1362],[1439,88,1200],[1652,304,89],[81,44,940],[445,463,151],[717,520,92],[129,116,100],[1666,1811,624],[1079,97,91],[62,91,71],[688,898,526],[463,74,133],[278,826,99],[961,372,42],[799,94,1007],[100,93,41],[1314,943,1301],[184,230,109],[875,1195,231],[133,176,189],[751,755,826],[101,102,57],[1198,513,117],[748,518,97],[1145,1484,1304],[358,658,81],[971,672,993],[445,151,456],[252,621,122],[36,271,126],[85,36,126],[116,83,93],[141,171,1747],[1081,883,103],[1398,1454,149],[457,121,593],[127,116,303],[697,70,891],[457,891,1652],[1058,1668,112],[518,130,97],[214,319,131],[185,1451,1449],[463,133,516],[1428,123,177],[113,862,561],[215,248,136],[186,42,251],[127,83,116],[160,85,127],[162,129,140],[154,169,1080],[169,170,1080],[210,174,166],[1529,1492,1524],[450,875,231],[399,875,450],[171,141,170],[113,1155,452],[131,319,360],[44,175,904],[452,872,113],[746,754,407],[147,149,150],[309,390,1148],[53,186,283],[757,158,797],[303,129,162],[429,303,162],[154,168,169],[673,164,193],[38,271,75],[320,288,1022],[246,476,173],[175,548,904],[182,728,456],[199,170,169],[168,199,169],[199,171,170],[184,238,230],[246,247,180],[1496,1483,1467],[147,150,148],[828,472,445],[53,108,186],[56,53,271],[186,961,42],[1342,391,57],[1664,157,1834],[1070,204,178],[178,204,179],[285,215,295],[692,55,360],[192,193,286],[359,673,209],[586,195,653],[121,89,573],[202,171,199],[238,515,311],[174,210,240],[174,105,166],[717,276,595],[1155,1149,452],[1405,56,197],[53,283,30],[75,53,30],[45,235,1651],[210,166,490],[181,193,192],[185,620,217],[26,798,759],[1070,226,204],[220,187,179],[220,168,187],[202,222,171],[359,209,181],[182,456,736],[964,167,1405],[76,250,414],[807,1280,1833],[70,883,1652],[227,179,204],[221,199,168],[221,202,199],[360,494,131],[214,241,319],[105,247,166],[205,203,260],[388,480,939],[482,855,211],[8,807,1833],[226,255,204],[228,221,168],[166,173,490],[701,369,702],[211,855,262],[631,920,630],[1448,1147,1584],[255,227,204],[237,220,179],[228,168,220],[222,256,555],[215,259,279],[126,271,38],[108,50,186],[227,236,179],[236,237,179],[220,237,228],[228,202,221],[256,222,202],[555,256,229],[259,152,279],[27,1296,31],[186,50,961],[961,234,372],[1651,235,812],[1572,1147,1448],[255,226,1778],[255,236,227],[256,257,229],[106,184,109],[241,410,188],[177,578,620],[209,673,181],[1136,1457,79],[1507,245,718],[255,273,236],[275,410,241],[206,851,250],[1459,253,1595],[1406,677,1650],[228,274,202],[202,281,256],[348,239,496],[205,172,203],[369,248,702],[261,550,218],[261,465,550],[574,243,566],[921,900,1220],[291,273,255],[348,238,265],[109,230,194],[149,380,323],[443,270,421],[272,291,255],[274,228,237],[274,292,202],[281,257,256],[276,543,341],[152,259,275],[1111,831,249],[632,556,364],[299,273,291],[299,236,273],[280,237,236],[202,292,281],[247,246,173],[282,49,66],[1620,1233,1553],[299,280,236],[280,305,237],[237,305,274],[306,292,274],[330,257,281],[246,194,264],[166,247,173],[912,894,896],[611,320,244],[1154,1020,907],[969,962,290],[272,299,291],[305,318,274],[145,212,240],[164,248,285],[259,277,275],[193,164,295],[269,240,210],[1033,288,320],[46,948,206],[336,280,299],[330,281,292],[257,307,300],[369,136,248],[145,240,269],[502,84,465],[193,295,286],[164,285,295],[282,302,49],[161,303,429],[318,306,274],[306,330,292],[315,257,330],[315,307,257],[307,352,300],[300,352,308],[275,277,403],[353,1141,333],[1420,425,47],[611,313,320],[85,126,83],[128,1180,43],[303,116,129],[280,314,305],[314,318,305],[190,181,242],[203,214,131],[820,795,815],[322,299,272],[322,336,299],[315,339,307],[172,152,617],[172,214,203],[321,1033,320],[1401,941,946],[85,160,138],[976,454,951],[747,60,786],[317,322,272],[339,352,307],[266,33,867],[163,224,218],[247,614,180],[648,639,553],[388,172,205],[611,345,313],[313,345,320],[160,127,303],[454,672,951],[317,329,322],[314,280,336],[306,338,330],[330,339,315],[1236,115,436],[342,321,320],[1046,355,328],[328,346,325],[325,346,317],[367,314,336],[314,337,318],[337,306,318],[338,343,330],[342,320,345],[355,349,328],[346,329,317],[347,336,322],[314,362,337],[330,343,339],[340,308,352],[135,906,1022],[239,156,491],[194,230,486],[40,1015,1003],[321,355,1046],[329,382,322],[382,347,322],[347,367,336],[337,371,306],[306,371,338],[1681,296,1493],[286,172,388],[230,348,486],[348,183,486],[384,332,830],[328,349,346],[367,362,314],[371,343,338],[339,351,352],[57,344,78],[342,355,321],[386,346,349],[386,350,346],[346,350,329],[347,366,367],[343,363,339],[323,380,324],[152,275,241],[345,1045,342],[350,374,329],[339,363,351],[234,340,352],[353,361,354],[40,34,1015],[373,355,342],[373,349,355],[374,382,329],[366,347,382],[371,363,343],[351,379,352],[379,372,352],[372,234,352],[156,190,491],[319,241,692],[354,361,31],[366,377,367],[363,379,351],[133,590,516],[197,56,271],[1045,370,342],[370,373,342],[374,350,386],[377,366,382],[367,395,362],[400,337,362],[400,371,337],[378,363,371],[106,109,614],[181,673,193],[953,920,631],[376,349,373],[376,386,349],[378,379,363],[224,375,218],[279,152,172],[361,619,381],[1347,823,795],[760,857,384],[392,374,386],[394,395,367],[383,371,400],[383,378,371],[218,375,261],[197,271,36],[414,454,976],[385,376,373],[1051,382,374],[387,394,367],[377,387,367],[395,400,362],[279,172,295],[30,365,225],[450,231,825],[385,373,370],[398,374,392],[1051,377,382],[396,378,383],[348,496,183],[295,172,286],[357,269,495],[1148,390,1411],[75,30,225],[206,76,54],[412,386,376],[412,392,386],[396,383,400],[651,114,878],[123,1241,506],[238,311,265],[381,653,29],[618,815,334],[427,1032,411],[298,414,976],[791,332,384],[129,100,140],[412,404,392],[392,404,398],[140,107,360],[395,394,400],[423,379,378],[385,412,376],[406,94,58],[419,415,1021],[422,423,378],[423,125,379],[258,508,238],[311,156,265],[213,287,491],[449,411,1024],[412,1068,404],[55,140,360],[76,414,54],[394,416,400],[400,416,396],[422,378,396],[1258,796,789],[427,411,449],[427,297,1032],[1385,1366,483],[417,448,284],[1507,341,245],[162,140,444],[658,44,81],[433,125,423],[438,251,125],[429,162,439],[1342,57,1348],[765,766,442],[697,891,695],[1057,396,416],[440,423,422],[440,433,423],[433,438,125],[438,196,251],[74,482,211],[1136,79,144],[29,195,424],[242,1004,492],[57,757,28],[414,298,54],[238,348,230],[224,163,124],[295,215,279],[495,269,490],[449,446,427],[446,297,427],[1020,1163,909],[128,138,419],[66,980,443],[415,439,1018],[111,396,1057],[111,422,396],[840,249,831],[593,664,596],[218,550,155],[109,194,180],[483,268,855],[161,415,419],[1737,232,428],[360,107,494],[1006,1011,410],[444,140,55],[919,843,430],[190,242,213],[275,403,410],[131,494,488],[449,663,446],[138,161,419],[128,419,34],[439,162,444],[460,440,422],[440,438,433],[472,74,445],[491,190,213],[238,508,515],[46,206,54],[972,944,962],[1241,1428,1284],[111,460,422],[470,432,806],[248,164,702],[1025,467,453],[553,1235,648],[263,114,881],[267,293,896],[469,438,440],[455,196,438],[287,242,492],[239,265,156],[213,242,287],[1684,746,63],[663,474,446],[415,161,429],[140,100,107],[1055,459,467],[469,455,438],[259,542,277],[446,474,466],[446,466,447],[439,444,1019],[614,109,180],[190,359,181],[156,497,190],[726,474,663],[1023,458,459],[461,440,460],[269,210,490],[246,180,194],[590,133,189],[163,218,155],[467,468,453],[1063,1029,111],[111,1029,460],[1029,464,460],[461,469,440],[150,149,323],[828,445,456],[375,502,261],[474,475,466],[573,426,462],[478,1023,477],[478,458,1023],[458,479,467],[459,458,467],[468,393,453],[464,461,460],[484,365,455],[1232,182,1380],[172,617,214],[547,694,277],[542,547,277],[184,258,238],[261,502,465],[467,479,468],[484,455,469],[1380,182,864],[475,476,466],[80,447,476],[466,476,447],[415,429,439],[479,487,468],[487,287,468],[492,393,468],[260,469,461],[481,365,484],[531,473,931],[692,360,319],[726,495,474],[468,287,492],[480,464,1029],[260,461,464],[494,481,484],[74,472,482],[174,240,212],[223,106,614],[486,477,485],[478,496,458],[491,487,479],[123,402,177],[488,469,260],[488,484,469],[265,239,348],[248,215,285],[474,490,475],[477,486,478],[458,496,479],[239,491,479],[1584,1147,1334],[488,494,484],[401,123,506],[495,490,474],[490,173,475],[80,476,264],[491,287,487],[480,1029,1004],[480,205,464],[173,476,475],[485,194,486],[486,183,478],[478,183,496],[496,239,479],[848,1166,60],[268,262,855],[205,260,464],[260,203,488],[203,131,488],[246,264,476],[194,485,264],[1002,310,1664],[311,515,497],[515,359,497],[565,359,515],[1250,1236,301],[736,456,151],[654,174,567],[577,534,648],[519,505,645],[725,565,508],[150,1723,148],[584,502,505],[584,526,502],[502,526,84],[607,191,682],[560,499,660],[607,517,191],[1038,711,124],[951,672,971],[716,507,356],[868,513,1198],[615,794,608],[682,191,174],[1313,928,1211],[617,241,214],[511,71,91],[408,800,792],[192,286,525],[80,485,447],[91,97,130],[1675,324,888],[207,756,532],[582,1097,1124],[311,497,156],[510,130,146],[523,511,510],[608,708,616],[546,690,650],[511,527,358],[536,146,518],[465,418,550],[418,709,735],[520,514,500],[584,505,519],[536,518,509],[146,536,510],[538,527,511],[876,263,669],[646,524,605],[510,536,523],[527,175,358],[724,876,669],[721,724,674],[524,683,834],[558,509,522],[558,536,509],[523,538,511],[611,243,574],[528,706,556],[668,541,498],[523,537,538],[527,540,175],[532,756,533],[1013,60,747],[551,698,699],[92,520,500],[535,536,558],[536,569,523],[538,540,527],[539,548,175],[567,212,145],[401,896,293],[534,675,639],[1510,595,1507],[557,545,530],[569,536,535],[537,540,538],[540,539,175],[569,537,523],[1135,718,47],[587,681,626],[580,535,558],[99,747,278],[701,565,725],[665,132,514],[665,514,575],[132,549,653],[176,651,189],[65,47,266],[597,569,535],[569,581,537],[537,581,540],[563,539,540],[539,564,548],[1509,1233,1434],[132,653,740],[550,710,155],[714,721,644],[410,1011,188],[732,534,586],[560,562,729],[555,557,222],[580,558,545],[597,535,580],[581,563,540],[5,821,1676],[576,215,136],[649,457,741],[564,539,563],[124,711,224],[550,668,710],[550,541,668],[565,701,673],[560,613,499],[233,532,625],[545,555,580],[601,581,569],[594,904,548],[1463,1425,434],[185,149,1454],[721,674,644],[185,380,149],[577,424,586],[462,586,559],[597,601,569],[594,548,564],[566,603,574],[165,543,544],[457,89,121],[586,424,195],[725,587,606],[1078,582,1124],[588,925,866],[462,559,593],[189,878,590],[555,229,580],[602,563,581],[904,594,956],[434,1425,1438],[1024,112,821],[572,587,626],[600,597,580],[599,591,656],[600,580,229],[601,622,581],[581,622,602],[602,564,563],[602,594,564],[603,611,574],[498,529,546],[697,1145,70],[592,628,626],[610,597,600],[597,610,601],[222,557,171],[604,765,799],[573,462,593],[133,200,176],[729,607,627],[1011,692,188],[518,146,130],[585,687,609],[682,627,607],[1712,599,656],[562,592,607],[643,656,654],[257,600,229],[601,633,622],[623,594,602],[174,212,567],[725,606,701],[609,701,606],[610,633,601],[633,642,622],[380,216,324],[142,143,1249],[501,732,586],[534,577,586],[648,1235,577],[610,641,633],[310,1002,1831],[618,334,604],[1710,145,269],[707,498,659],[501,586,462],[625,501,462],[726,663,691],[300,600,257],[641,610,600],[622,629,602],[602,629,623],[55,692,444],[518,748,509],[929,1515,1411],[620,578,267],[71,511,358],[707,668,498],[650,687,585],[600,300,641],[641,657,633],[1675,888,1669],[622,636,629],[505,502,375],[541,529,498],[332,420,1053],[637,551,638],[534,639,648],[69,623,873],[300,512,641],[633,657,642],[562,660,579],[687,637,638],[709,646,605],[775,738,885],[559,549,132],[646,683,524],[641,512,657],[266,897,949],[1712,643,1657],[184,727,258],[674,724,669],[699,714,647],[628,659,572],[657,662,642],[571,881,651],[517,607,504],[598,706,528],[598,694,547],[640,552,560],[655,693,698],[698,693,721],[91,510,511],[144,301,1136],[324,216,888],[870,764,1681],[575,514,520],[276,544,543],[658,175,44],[645,505,711],[659,546,572],[700,524,655],[605,700,529],[266,867,897],[1695,1526,764],[579,659,628],[654,591,682],[586,549,559],[698,721,714],[896,401,506],[640,734,599],[664,665,575],[621,629,636],[1712,656,643],[547,644,598],[710,668,707],[640,560,734],[655,698,551],[694,528,277],[512,662,657],[504,592,626],[688,584,519],[152,241,617],[587,725,681],[598,669,706],[526,670,84],[598,528,694],[710,707,499],[579,592,562],[660,659,579],[323,324,1134],[326,895,473],[195,29,653],[84,670,915],[560,660,562],[504,626,681],[711,505,224],[651,881,114],[216,620,889],[1362,678,197],[493,99,48],[1659,691,680],[529,690,546],[430,843,709],[655,524,693],[174,191,105],[674,669,598],[98,712,82],[572,546,585],[72,61,71],[912,911,894],[106,223,184],[664,132,665],[843,646,709],[635,699,136],[699,698,714],[593,132,664],[688,526,584],[185,177,620],[533,675,534],[687,638,635],[1652,89,457],[896,506,912],[132,740,514],[689,685,282],[691,449,680],[48,436,493],[136,699,647],[739,640,554],[549,586,653],[532,533,625],[1530,695,649],[653,381,619],[736,151,531],[188,692,241],[177,402,578],[33,689,867],[689,33,685],[593,559,132],[949,65,266],[711,1038,661],[939,480,1004],[609,369,701],[616,552,615],[619,361,740],[151,463,516],[513,521,117],[691,663,449],[186,251,196],[333,302,327],[613,560,552],[616,613,552],[690,551,637],[660,707,659],[704,208,1203],[418,735,550],[163,708,124],[524,834,693],[554,640,599],[245,341,165],[565,673,359],[155,710,708],[105,191,517],[1515,198,1411],[1709,554,599],[60,289,786],[838,1295,1399],[533,534,625],[710,499,708],[556,632,410],[217,620,216],[591,627,682],[504,503,223],[643,654,567],[690,637,650],[545,557,555],[174,654,682],[719,691,1659],[727,681,508],[645,711,661],[794,615,739],[565,515,508],[282,685,302],[1150,397,1149],[638,699,635],[544,685,33],[719,726,691],[1742,1126,1733],[1724,1475,148],[556,410,403],[185,217,380],[503,504,681],[277,556,403],[32,1178,158],[1712,1709,599],[605,529,541],[635,136,369],[687,635,369],[529,700,690],[700,551,690],[89,304,573],[625,534,732],[730,302,685],[503,681,727],[702,673,701],[730,327,302],[327,353,333],[596,664,575],[660,499,707],[585,546,650],[560,729,734],[700,655,551],[176,571,651],[517,504,223],[730,685,544],[1661,1682,726],[1682,495,726],[1250,301,917],[605,524,700],[609,687,369],[516,389,895],[1553,686,1027],[673,702,164],[656,591,654],[520,596,575],[402,123,401],[828,456,728],[1645,677,1653],[528,556,277],[638,551,699],[190,497,359],[276,730,544],[1117,1525,933],[1027,686,1306],[155,708,163],[709,605,541],[647,644,547],[650,637,687],[599,734,591],[578,293,267],[1682,357,495],[510,91,130],[734,729,627],[576,542,215],[709,541,735],[735,541,550],[276,500,730],[500,327,730],[653,619,740],[414,851,454],[734,627,591],[729,562,607],[615,552,640],[525,181,192],[308,512,300],[223,503,727],[266,165,33],[92,500,276],[321,1046,1033],[585,609,606],[1200,1559,86],[628,572,626],[301,436,803],[714,644,647],[708,499,613],[721,693,724],[514,353,327],[353,740,361],[344,158,78],[708,613,616],[615,640,739],[500,514,327],[514,740,353],[1449,177,185],[462,233,625],[851,405,1163],[608,616,615],[647,542,576],[625,732,501],[1097,582,1311],[1235,424,577],[579,628,592],[607,592,504],[24,432,470],[105,614,247],[104,742,471],[542,259,215],[365,196,455],[1420,47,65],[223,727,184],[547,542,647],[572,585,606],[587,572,606],[262,780,1370],[647,576,136],[644,674,598],[271,53,75],[727,508,258],[471,742,142],[505,375,224],[357,1710,269],[725,508,681],[659,498,546],[743,1178,32],[1195,634,231],[1176,24,470],[743,1110,1178],[135,809,857],[63,746,407],[634,1176,470],[159,1112,27],[1176,1685,24],[399,450,779],[1178,856,875],[751,744,54],[436,48,772],[634,1108,1210],[769,1285,1286],[751,298,755],[746,1684,754],[754,924,87],[722,1625,756],[87,839,153],[489,795,820],[758,808,1518],[839,840,153],[831,1111,959],[1111,749,959],[810,1253,1363],[1247,1394,713],[1388,1329,1201],[1242,120,761],[857,791,384],[758,1523,808],[296,764,1504],[70,1652,891],[207,233,1638],[1348,57,28],[858,420,332],[964,1379,1278],[420,1194,816],[784,1076,1186],[1076,21,1186],[1710,767,1],[849,822,778],[806,137,787],[786,790,744],[790,54,744],[771,63,407],[785,852,818],[774,1823,272],[895,151,516],[135,1022,809],[99,826,48],[48,826,755],[808,705,408],[833,441,716],[1733,743,32],[1385,836,852],[772,827,737],[1005,49,781],[793,1697,813],[1518,441,1537],[1139,1132,859],[782,801,770],[1510,1530,676],[770,814,835],[231,787,825],[207,722,756],[26,771,798],[782,863,865],[832,54,790],[865,842,507],[799,765,94],[1175,1261,1353],[800,408,805],[262,986,200],[792,800,814],[801,792,770],[704,1203,1148],[356,1514,822],[165,544,33],[561,776,113],[1043,738,775],[815,831,820],[773,792,801],[772,48,914],[772,737,803],[436,772,803],[808,817,705],[1624,822,1527],[588,1144,788],[799,762,604],[821,1520,1676],[854,803,666],[828,482,472],[445,74,463],[831,489,820],[828,836,482],[716,782,763],[334,815,766],[815,823,766],[334,766,765],[819,805,837],[1716,1521,1412],[1684,924,754],[800,805,819],[1709,829,554],[806,1349,137],[99,1013,747],[341,595,276],[817,810,818],[1176,1691,1685],[763,782,865],[830,846,1052],[865,1499,842],[982,846,1053],[847,832,790],[1178,875,158],[817,818,705],[1302,1392,45],[96,417,284],[223,614,517],[356,507,1514],[1166,848,1179],[1349,432,26],[717,92,276],[770,835,863],[522,509,1745],[847,841,832],[832,841,46],[829,739,554],[802,824,39],[397,1043,775],[1567,849,778],[1385,483,855],[1349,26,1346],[441,801,782],[402,401,293],[1043,667,738],[759,798,1007],[819,837,728],[728,837,828],[837,852,828],[1537,441,833],[148,1475,147],[805,705,837],[716,441,782],[483,1371,780],[814,819,844],[845,753,1336],[1661,719,4],[862,847,790],[737,827,666],[201,46,841],[810,785,818],[408,705,805],[1560,1536,849],[1585,853,1786],[7,1668,807],[7,807,8],[822,1514,1527],[800,819,814],[847,862,841],[991,857,760],[705,818,837],[808,408,773],[402,293,578],[791,858,332],[1480,1228,1240],[814,844,835],[785,1385,852],[1132,120,859],[1743,1726,684],[1704,783,1279],[1623,1694,1731],[959,489,831],[1518,808,773],[862,872,841],[441,773,801],[331,512,308],[380,217,216],[841,872,201],[818,852,837],[448,1480,1240],[856,1108,1195],[1527,1514,1526],[819,182,1232],[871,724,693],[852,836,828],[770,792,814],[803,737,666],[751,826,278],[1674,1727,1699],[849,356,822],[871,693,834],[507,842,1514],[1406,1097,869],[1328,1349,1346],[823,815,795],[744,751,278],[1110,856,1178],[520,717,316],[871,834,683],[884,876,724],[165,266,47],[716,763,507],[216,889,888],[853,1585,1570],[1536,716,356],[886,873,623],[782,770,863],[432,24,26],[683,882,871],[884,724,871],[114,876,884],[516,590,389],[11,1218,1628],[862,113,872],[886,623,629],[830,1052,1120],[762,153,604],[773,408,792],[763,865,507],[153,840,604],[882,884,871],[531,151,326],[886,890,873],[133,262,200],[819,1232,844],[621,636,122],[645,892,519],[1130,1076,784],[114,263,876],[1670,10,1663],[911,670,894],[452,885,872],[872,885,201],[887,882,683],[878,884,882],[590,878,882],[890,867,689],[897,629,621],[897,886,629],[819,728,182],[519,893,688],[894,670,526],[898,894,526],[1536,356,849],[810,1363,785],[878,114,884],[879,888,892],[892,889,893],[893,898,688],[895,683,843],[895,887,683],[889,620,267],[590,882,389],[418,465,84],[949,897,621],[897,890,886],[889,267,893],[898,267,896],[531,326,473],[189,651,878],[843,683,646],[897,867,890],[888,889,892],[893,267,898],[896,894,898],[473,895,843],[895,389,887],[974,706,669],[513,1115,521],[326,151,895],[809,791,857],[211,262,133],[920,923,947],[923,90,947],[90,25,947],[25,972,935],[64,431,899],[52,899,901],[903,905,59],[437,967,73],[839,1242,761],[904,975,44],[917,301,144],[915,670,911],[905,201,885],[1684,63,1685],[1033,1194,288],[950,913,755],[912,918,911],[950,914,913],[506,918,912],[922,919,915],[911,922,915],[1004,451,492],[1263,553,639],[922,911,918],[630,920,947],[916,506,926],[916,918,506],[521,1115,1098],[916,922,918],[919,418,915],[83,38,75],[24,1685,771],[110,1230,1213],[712,8,1837],[922,930,919],[919,430,418],[1395,1402,1187],[930,922,916],[594,623,69],[35,431,968],[35,968,969],[866,924,1684],[1625,1263,675],[631,630,52],[930,931,919],[430,709,418],[302,333,49],[1446,978,1138],[799,1007,798],[931,843,919],[947,25,64],[885,738,667],[1262,963,964],[899,970,901],[1401,946,938],[1117,933,1091],[1685,63,771],[905,948,201],[979,937,980],[951,953,950],[937,270,443],[1154,903,59],[1194,954,1067],[909,405,907],[850,1151,59],[1769,811,1432],[76,206,250],[938,946,966],[965,927,942],[938,966,957],[955,975,904],[927,965,934],[52,51,631],[59,905,667],[431,935,968],[786,289,561],[252,122,671],[481,494,107],[954,1817,1067],[795,25,90],[958,965,945],[795,972,25],[902,983,955],[972,489,944],[1256,29,424],[671,331,945],[946,958,963],[956,955,904],[902,955,956],[671,512,331],[945,331,961],[662,671,122],[671,662,512],[934,65,927],[630,947,52],[666,631,910],[850,59,667],[961,331,234],[1024,411,1042],[890,69,873],[252,671,945],[975,290,940],[283,186,196],[30,283,365],[950,755,298],[946,965,958],[985,290,975],[969,290,985],[405,851,206],[935,431,64],[941,1423,1420],[964,963,167],[942,252,945],[78,757,57],[49,1005,66],[937,979,270],[631,666,827],[980,937,443],[66,689,282],[421,902,956],[947,64,52],[35,979,899],[951,971,953],[762,87,153],[27,31,381],[924,839,87],[946,963,966],[331,308,340],[957,966,1262],[473,843,931],[953,971,920],[270,969,902],[935,962,968],[51,1005,781],[969,983,902],[437,73,940],[69,421,956],[761,249,840],[263,974,669],[962,944,967],[962,437,290],[985,975,955],[907,405,948],[720,957,1262],[25,935,64],[176,200,571],[108,945,50],[250,851,414],[200,986,571],[881,974,263],[827,772,953],[970,899,980],[29,159,27],[234,331,340],[948,405,206],[980,899,979],[986,984,571],[571,984,881],[990,706,974],[946,934,965],[970,980,66],[1113,1486,1554],[984,981,881],[881,987,974],[689,66,443],[1005,901,66],[983,985,955],[165,47,718],[987,990,974],[1370,986,262],[901,970,66],[51,901,1005],[981,987,881],[988,706,990],[942,945,965],[290,437,940],[64,899,52],[988,556,706],[941,934,946],[431,35,899],[996,989,984],[984,989,981],[981,989,987],[35,969,270],[1370,995,986],[986,995,984],[989,999,987],[987,992,990],[992,988,990],[962,967,437],[951,950,976],[979,35,270],[421,270,902],[998,995,1370],[987,999,992],[988,364,556],[969,985,983],[689,443,890],[995,1000,984],[219,958,108],[998,1000,995],[999,997,992],[914,953,772],[845,1336,745],[806,787,231],[1000,996,984],[989,996,999],[50,945,961],[443,421,69],[797,158,779],[1098,1463,434],[996,1009,999],[1001,988,992],[1001,364,988],[903,907,905],[26,759,973],[997,1001,992],[632,364,1001],[1346,26,973],[998,1008,1000],[1000,1009,996],[531,931,736],[252,949,621],[286,388,525],[1174,1008,998],[1009,1010,999],[999,1010,997],[1014,1001,997],[614,105,517],[958,945,108],[525,1004,242],[963,958,219],[233,426,304],[1000,1008,1009],[1010,1014,997],[1001,1006,632],[824,413,39],[642,636,622],[480,388,205],[28,757,797],[1014,1006,1001],[1006,410,632],[975,940,44],[1234,420,858],[54,832,46],[1009,1012,1010],[167,963,219],[41,481,107],[1017,1010,1012],[122,636,662],[939,525,388],[525,939,1004],[950,953,914],[829,1735,739],[1008,880,1015],[1008,1015,1009],[1263,639,675],[956,594,69],[795,90,1347],[1179,848,1013],[759,1007,973],[1009,1015,1012],[1012,1016,1017],[1017,1014,1010],[1019,1011,1006],[927,65,949],[649,316,595],[913,48,755],[976,950,298],[1003,1015,880],[1018,1006,1014],[1021,1018,1014],[444,692,1011],[451,1029,1063],[1185,851,1163],[29,27,381],[181,525,242],[1021,1014,1017],[1016,1021,1017],[1018,1019,1006],[1019,444,1011],[927,949,942],[451,393,492],[903,1154,907],[391,101,57],[94,765,58],[419,1016,1012],[949,252,942],[907,1020,909],[765,442,58],[94,406,908],[1007,94,908],[34,1012,1015],[34,419,1012],[419,1021,1016],[451,1057,393],[907,948,905],[1034,1073,1039],[1061,906,1619],[1068,960,1034],[471,1249,104],[112,1024,1042],[372,379,125],[341,543,165],[141,1094,170],[566,243,1061],[398,1034,1039],[325,317,1823],[1493,296,1724],[850,667,1043],[1054,297,1065],[1619,135,1074],[1061,243,906],[680,1024,821],[1103,96,1245],[1440,1123,1491],[1047,1025,1044],[672,454,1231],[1484,697,1530],[993,672,1231],[178,154,1088],[1044,1041,1066],[112,1062,1058],[1530,649,676],[178,1088,1040],[1046,328,954],[243,244,1022],[954,1194,1033],[1042,411,1032],[971,993,1056],[960,1093,1034],[1754,1338,232],[385,1064,412],[1057,1063,111],[748,1071,1447],[1530,697,695],[971,1056,1270],[977,1059,1211],[649,741,316],[1060,1452,1030],[353,354,1323],[695,768,649],[398,404,1034],[596,316,741],[1836,119,13],[1513,1115,1528],[883,1081,1652],[1039,1073,1048],[462,426,233],[31,1296,354],[1055,1047,1066],[1032,1054,1045],[1521,310,1224],[119,861,13],[1194,1234,288],[1109,1771,1070],[1166,1160,776],[1044,1035,1041],[1026,960,1064],[1050,1032,1045],[1049,1041,387],[115,1013,99],[1046,954,1033],[1321,920,971],[611,1058,345],[1048,1066,1049],[1023,1055,1073],[1029,451,1004],[118,1094,141],[1094,1080,170],[1042,1032,1050],[1026,1064,385],[15,16,1084],[1096,1079,61],[1075,1071,748],[325,1817,328],[909,1163,405],[1022,1234,809],[374,398,1051],[1082,72,81],[1023,1034,1093],[1817,1794,1067],[86,1445,1400],[1507,1535,1510],[1079,1096,1075],[568,1478,1104],[1070,178,1040],[1034,1023,1073],[776,1155,113],[1103,143,142],[1140,81,73],[1082,81,1140],[1060,1030,936],[1040,1086,1109],[370,1065,385],[61,72,1082],[1087,1096,1144],[1040,1088,1086],[1651,812,752],[1062,1050,1045],[187,154,178],[179,187,178],[1099,1344,1101],[1668,1058,807],[1073,1055,1048],[1099,1336,1344],[1283,943,1123],[1049,387,1051],[1024,680,449],[61,1082,1100],[967,749,1111],[1439,1037,88],[742,1505,142],[398,1039,1051],[1107,1336,1099],[1344,1542,1101],[142,1505,1103],[477,1093,447],[477,1023,1093],[471,142,1249],[1041,1035,394],[1328,568,1104],[61,1100,1096],[154,1092,1088],[112,1042,1050],[154,187,168],[435,235,45],[1075,1096,1087],[97,1075,748],[1049,1066,1041],[816,1067,1028],[846,982,1142],[1245,96,284],[1092,154,1080],[1057,451,1063],[387,377,1051],[1055,1025,1047],[1075,1087,1089],[1106,1108,856],[1068,1034,404],[1480,1545,868],[906,135,1619],[1074,991,1095],[570,566,1061],[1025,453,1044],[745,1336,1107],[1035,1057,416],[1092,1102,1129],[1074,135,991],[1105,745,1107],[447,1026,446],[394,387,1041],[73,81,940],[1118,1108,1106],[1210,1108,874],[243,1022,906],[412,1064,1068],[1280,611,603],[960,447,1093],[1051,1039,1049],[1040,1109,1070],[1471,1037,1439],[69,890,443],[1377,703,1374],[1092,1080,1102],[1096,1100,788],[1096,788,1144],[1114,967,1111],[446,1026,297],[70,1112,883],[453,393,1057],[1118,874,1108],[1054,370,1045],[1080,1094,1102],[1039,1048,1049],[428,753,845],[1047,1044,1066],[1044,453,1035],[1472,731,1512],[1126,1121,743],[743,1121,1110],[1032,297,1054],[1480,868,1216],[71,358,72],[1133,967,1114],[1105,1119,745],[1035,453,1057],[1026,447,960],[454,851,1190],[1030,1477,652],[589,816,1028],[1110,1121,1106],[1122,1118,1106],[1116,874,1118],[1048,1055,1066],[1194,1067,816],[744,278,747],[745,1120,845],[845,1052,428],[1105,1780,1119],[1065,297,385],[1098,1529,1463],[731,1060,936],[235,434,812],[1445,1525,1117],[1106,1121,1122],[1122,1127,1118],[1127,1116,1118],[1094,118,1732],[1119,1120,745],[1406,1124,1097],[435,117,235],[1462,1440,1037],[1126,1129,1121],[1088,1092,1129],[1133,73,967],[1120,1052,845],[812,434,752],[1441,1559,1200],[1131,588,413],[1054,1065,370],[235,1098,434],[1052,1142,428],[1737,428,1142],[1496,1446,1483],[1182,1083,1654],[1121,1129,1122],[1732,1116,1127],[768,457,649],[761,1114,249],[1064,960,1068],[1135,1481,1136],[1126,952,1129],[1087,588,1131],[1087,1144,588],[859,788,1139],[1140,1133,1132],[1133,1140,73],[1822,570,1061],[394,1035,416],[1055,1023,459],[80,264,485],[1119,1128,1120],[145,1658,567],[695,891,768],[1129,1102,1122],[1122,1102,1127],[1416,1077,1413],[297,1026,385],[1052,846,1142],[1445,1117,1400],[952,1086,1129],[1714,1089,1131],[1131,1089,1087],[1100,1139,788],[112,1050,1062],[1323,354,1296],[49,333,1141],[1142,982,1737],[79,1457,1091],[1088,1129,1086],[1102,1094,1127],[1127,1094,1732],[1100,1082,1139],[1082,1132,1139],[1082,1140,1132],[1150,1043,397],[60,1166,289],[1696,1146,1698],[1297,1202,1313],[409,1297,1313],[1234,1194,420],[1408,1391,1394],[424,1235,1243],[1203,309,1148],[485,477,447],[1152,1156,850],[1153,1149,1155],[1153,1157,1149],[1149,1152,1150],[1156,1154,1151],[776,1153,1155],[1157,1152,1149],[1217,1393,1208],[1156,1159,1154],[1153,1165,1157],[1165,1152,1157],[1159,1020,1154],[1161,1153,776],[1161,1165,1153],[1165,1158,1152],[1152,1158,1156],[1158,1159,1156],[1166,776,561],[1160,1161,776],[1161,1164,1165],[1161,1160,1164],[1158,1162,1159],[1159,1162,1020],[1270,1321,971],[1164,1170,1165],[1165,1162,1158],[1162,1163,1020],[588,788,925],[1166,1167,1160],[1165,1170,1162],[1160,1167,1164],[1162,1170,1163],[1179,1167,1166],[1167,1168,1164],[1164,1168,1170],[1168,1169,1170],[1234,1022,288],[802,39,866],[1179,1168,1167],[1169,1173,1170],[1170,1173,1163],[1173,1185,1163],[1360,1267,1364],[1169,1185,1173],[611,244,243],[900,1226,1376],[1260,1408,1350],[618,840,831],[1181,1183,1179],[1179,1184,1168],[1208,1274,1291],[1183,1184,1179],[1168,1184,1169],[1387,1395,1254],[1208,1204,1172],[1182,1197,1083],[1187,1083,1197],[1213,1183,1181],[1169,1207,1185],[135,857,991],[1013,1213,1181],[1189,1183,1213],[1183,1189,1184],[1169,1184,1207],[1207,1190,1185],[1180,1389,1288],[1191,1192,1640],[1640,1192,1090],[1090,1205,1654],[1654,1205,1182],[1188,1395,1187],[1126,743,1733],[788,859,925],[809,1234,1171],[1193,1197,1182],[1189,1199,1184],[1639,1191,1637],[1639,1212,1191],[1205,1193,1182],[1198,1187,1197],[1199,1207,1184],[332,1053,846],[1090,1192,1205],[117,1188,1187],[435,1188,117],[435,1206,1188],[1199,1189,1213],[420,816,1053],[1212,1215,1191],[117,1187,1198],[45,1206,435],[120,1132,1133],[874,1116,1210],[1191,1215,1192],[1193,1216,1197],[1216,1198,1197],[1199,1214,1207],[117,521,235],[1220,1311,1078],[1220,900,1311],[1653,1215,1212],[1192,1225,1205],[1205,1209,1193],[1209,1216,1193],[1389,1217,1172],[1207,1214,454],[171,557,1747],[1805,1078,1787],[1805,1219,1078],[1198,1216,868],[666,910,854],[1230,1231,1213],[1213,1231,1199],[1199,1231,1214],[1219,1220,1078],[1215,1221,1192],[1192,1221,1225],[1225,1228,1205],[1205,1228,1209],[1209,1228,1216],[1464,1325,1223],[1215,1227,1221],[1228,1480,1216],[1226,1653,1376],[1653,1249,1215],[1221,1240,1225],[1225,1240,1228],[839,761,840],[1238,1219,1805],[1238,1220,1219],[1232,1380,1375],[1226,1249,1653],[1221,1227,1240],[233,207,532],[110,1236,1230],[1248,1231,1230],[1231,454,1214],[1249,1227,1215],[1248,1056,1231],[489,959,944],[448,1240,284],[925,859,1242],[1805,1244,1238],[1252,1220,1238],[1252,921,1220],[1236,1251,1230],[1230,1251,1248],[1056,993,1231],[1031,1264,1263],[68,1186,157],[1227,1245,1240],[1103,1245,143],[1243,1235,612],[1252,95,921],[1249,1226,1237],[1390,1387,1254],[1120,384,830],[830,332,846],[1227,143,1245],[1315,1369,1358],[1356,1269,1386],[972,795,489],[1831,1224,310],[1250,1255,1251],[1251,1056,1248],[1256,1243,103],[658,358,175],[1620,1238,1244],[1620,1252,1238],[1506,95,1252],[104,1249,1237],[1249,143,1227],[1268,1419,1329],[634,806,231],[618,831,815],[924,1242,839],[1255,1270,1251],[1251,1270,1056],[866,925,1242],[103,29,1256],[424,1243,1256],[134,1651,752],[1250,917,1255],[1172,1204,1260],[1352,1036,1276],[1265,1201,1329],[804,1282,1259],[1259,1294,723],[335,1330,1305],[407,762,799],[875,856,1195],[32,158,344],[967,944,749],[372,125,42],[1175,1354,1261],[553,612,1235],[1259,1273,1294],[1294,1283,723],[757,78,158],[407,799,798],[901,51,52],[139,1386,1389],[1386,1269,1389],[1389,1269,1217],[1148,1590,1268],[1428,1449,1450],[804,1281,1282],[1273,1259,1282],[158,399,779],[771,407,798],[521,1098,235],[917,1312,1255],[1312,1270,1255],[1217,1269,1393],[1195,1108,634],[1110,1106,856],[1210,1691,1176],[27,1112,1145],[1296,27,1145],[1171,858,791],[704,1148,1290],[1430,1436,1437],[1282,1308,1273],[1300,943,1283],[1393,1355,1274],[720,1278,769],[1287,1059,1399],[1310,1388,1272],[1312,1321,1270],[851,1185,1190],[1296,1145,1304],[26,24,771],[51,910,631],[1329,1290,1268],[1290,1148,1268],[1298,1293,733],[1281,1293,1282],[1282,1293,1308],[1308,1299,1273],[1300,1283,1294],[1340,943,1300],[1340,1301,943],[407,754,762],[1287,1399,1295],[34,139,128],[1288,1172,1260],[120,1133,1114],[1306,1113,1511],[1464,1223,1292],[1299,1294,1273],[1299,1300,1294],[1286,1295,838],[1285,1247,1286],[1247,713,1286],[1201,1265,1390],[1378,1368,1357],[1482,1320,917],[917,1320,1312],[850,1156,1151],[588,39,413],[1324,1306,686],[789,1365,928],[1223,1326,1292],[1292,1326,1298],[869,1097,1311],[790,786,561],[1323,1304,932],[1323,1296,1304],[1317,1324,686],[1306,368,1113],[1325,1342,1223],[1326,1348,1298],[1293,1327,1308],[1308,1318,1299],[704,1290,1258],[1320,1321,1312],[761,120,1114],[1684,802,866],[1674,6,1727],[1316,1323,932],[1335,1337,1305],[1348,1327,1293],[1298,1348,1293],[1333,1300,1299],[1333,1343,1300],[1328,1301,1340],[1328,1314,1301],[838,1399,1319],[921,1237,900],[409,1391,1408],[1376,1653,677],[1281,804,1458],[1331,1324,1317],[1324,368,1306],[368,1338,1307],[1327,797,1308],[797,1345,1308],[1308,1345,1318],[1318,1333,1299],[1341,1147,1572],[923,1321,1320],[923,920,1321],[39,588,866],[1141,1323,1316],[1330,1335,1305],[1337,1335,1336],[1339,1332,1325],[1223,1342,1326],[1342,1348,1326],[1348,797,1327],[1345,1333,1318],[1343,1340,1300],[1419,1265,1329],[1347,1320,1584],[1535,1141,1316],[1078,1311,582],[1344,1335,1330],[753,1331,1337],[368,1324,1331],[753,368,1331],[1332,1485,1325],[1325,1485,1342],[787,1343,1333],[137,1328,1340],[973,1341,1479],[406,1147,1341],[1171,1234,858],[1141,1535,1322],[49,1141,1322],[1344,1336,1335],[973,908,1341],[766,1347,1584],[1347,923,1320],[781,49,1322],[368,232,1338],[787,1340,1343],[787,137,1340],[568,1346,973],[58,1147,406],[442,1334,1147],[58,442,1147],[442,766,1334],[90,923,1347],[428,368,753],[779,1333,1345],[825,787,1333],[137,1349,1328],[1328,1346,568],[908,406,1341],[924,866,1242],[1336,753,1337],[428,232,368],[1115,777,1098],[1348,28,797],[797,779,1345],[779,825,1333],[1007,908,973],[583,1351,880],[1365,1246,977],[1658,145,1710],[1310,796,1388],[718,245,165],[1302,1272,1254],[1174,1351,583],[1174,715,1351],[1358,1260,1204],[1374,1373,1276],[1377,1374,1276],[678,1362,1382],[1377,1276,254],[139,34,40],[1008,1174,583],[1396,1286,1319],[768,891,457],[1316,932,1535],[1289,1371,1360],[182,736,864],[1355,1364,1274],[860,1367,1354],[1362,1222,1382],[1376,869,1311],[1590,1411,198],[1232,1375,877],[1394,1295,1286],[880,1356,1386],[880,1351,1356],[1211,1059,1287],[197,678,1405],[880,1386,1003],[1368,1253,1357],[1357,1253,1036],[715,1289,1364],[1354,1367,703],[1383,877,1375],[1266,1288,1260],[1373,1374,703],[1372,1289,1174],[1303,1366,1378],[1351,715,1355],[1665,1666,624],[1309,1357,1036],[900,1237,1226],[1174,1289,715],[1337,1331,1317],[1360,1303,1359],[1267,1354,1175],[1241,1284,1414],[1377,254,929],[1385,855,836],[1396,1319,1436],[1361,1366,1303],[1381,1368,1378],[1313,1211,1391],[1368,1385,1363],[813,82,861],[1058,1280,807],[893,519,892],[1359,1303,860],[1382,1350,1247],[1371,1303,1360],[1267,1175,1271],[769,1286,1396],[712,1837,82],[1366,1385,1381],[1365,796,1310],[1003,1386,40],[780,1371,1370],[561,862,790],[1284,1380,864],[1449,1428,177],[611,1280,1058],[1284,1375,1380],[926,506,1241],[1305,1337,1317],[309,1203,208],[1388,1201,1390],[1309,1036,1352],[1377,929,1411],[1399,1059,1257],[1112,70,1145],[289,1166,561],[1288,1389,1172],[1362,37,1180],[713,1394,1286],[1355,1393,1269],[1401,1423,941],[1274,1271,1384],[860,1378,1367],[715,1364,1355],[677,1406,869],[1297,1358,1202],[1388,1258,1329],[1180,1288,1266],[1008,583,880],[1524,1425,1463],[1390,1403,1387],[1278,1379,1247],[1278,1247,1285],[964,1278,1262],[1358,1369,1202],[1715,1699,1726],[926,1241,1414],[1341,1572,1479],[926,930,916],[1397,51,781],[409,1358,1297],[1236,436,301],[1376,677,869],[1351,1355,1356],[758,1534,1523],[1378,1357,1367],[977,1211,1365],[1135,1136,854],[1394,1391,1295],[1266,1260,1222],[1365,1302,1246],[1232,877,844],[736,930,864],[1408,1358,409],[1508,817,1523],[1381,1385,1368],[718,854,910],[854,718,1135],[1382,1222,1350],[1391,1211,1287],[1391,1287,1295],[1257,1651,134],[1414,1284,864],[1291,1369,1315],[1202,928,1313],[86,1400,1413],[1413,1200,86],[1263,1625,1031],[1413,1400,1404],[1002,1664,1834],[930,926,1414],[1399,1257,134],[520,316,596],[1393,1274,1208],[1657,1655,1712],[1407,1404,1400],[1404,1410,1413],[1649,1229,1406],[1362,1266,1222],[1384,1271,1175],[900,1376,1311],[1274,1384,1291],[1291,1384,1431],[1433,1396,1436],[1267,1359,1354],[309,1353,703],[838,1319,1286],[1407,1410,1404],[441,1518,773],[1241,123,1428],[1622,1521,1224],[1217,1208,1172],[1130,793,1076],[425,1409,1481],[1481,1409,1533],[1303,1378,860],[1350,1408,1394],[1246,1651,977],[1289,1360,1364],[1727,1694,1623],[1417,1407,1533],[1417,1410,1407],[1406,1650,1649],[1319,134,1437],[1414,864,930],[1406,1229,1124],[1354,1359,860],[1433,769,1396],[1417,1533,1409],[1416,1413,1410],[1415,1416,1410],[95,1237,921],[1392,1254,1395],[1360,1359,1267],[1258,1290,1329],[1180,128,1389],[1420,1409,425],[1417,1418,1410],[1418,1415,1410],[1422,1077,1416],[1247,1350,1394],[37,43,1180],[1204,1315,1358],[1428,1383,1375],[1356,1355,1269],[1409,1418,1417],[1302,45,1246],[1421,1416,1415],[1421,1422,1416],[1422,1494,1077],[957,720,938],[1423,1409,1420],[1423,1418,1409],[752,434,1438],[1260,1358,1408],[1363,1385,785],[1423,1426,1418],[1426,1424,1418],[1229,1649,1124],[1222,1260,1350],[1508,1523,1137],[1278,1285,769],[1482,917,144],[1418,1424,1415],[1425,1422,1421],[1425,1524,1422],[1272,1388,1390],[1391,409,1313],[1378,1366,1381],[1371,483,1361],[720,1262,1278],[29,103,159],[1271,1364,1267],[1424,1427,1415],[1537,1522,1518],[134,752,1438],[1420,934,941],[1428,1375,1284],[1277,1224,1831],[1362,1180,1266],[1401,1426,1423],[1577,1369,1291],[268,483,262],[1383,1450,1456],[1384,1175,1431],[1430,1415,1427],[1430,1421,1415],[1430,1425,1421],[1379,1382,1247],[1252,1553,1429],[1206,1392,1395],[1433,1430,1427],[309,208,1353],[1272,1390,1254],[1361,483,1366],[1523,817,808],[1302,1254,1392],[1371,1361,1303],[1426,1435,1424],[1435,1433,1424],[1433,1427,1424],[720,769,1433],[796,1258,1388],[1590,1419,1268],[1289,1372,1371],[1305,1317,1509],[998,1372,1174],[40,1386,139],[1261,1354,703],[1364,1271,1274],[134,1438,1437],[1436,1319,1437],[1317,686,1509],[1484,932,1304],[1434,1432,1509],[1420,65,934],[931,930,736],[1367,1357,1309],[1372,1370,1371],[1204,1208,1315],[1426,938,1435],[1368,1363,1253],[1207,454,1190],[1302,1310,1272],[309,1377,390],[390,1377,1411],[1370,1372,998],[1411,1590,1148],[720,1433,1435],[1450,1383,1428],[1379,678,1382],[1405,678,1379],[1208,1291,1315],[1399,134,1319],[1367,1309,1373],[1373,1352,1276],[596,741,593],[553,1264,612],[1433,1436,1430],[1437,1438,1430],[964,1405,1379],[1373,1309,1352],[1265,1403,1390],[1233,1618,1434],[1365,1310,1302],[789,796,1365],[720,1435,938],[128,139,1389],[1466,933,1525],[1191,1640,1637],[1314,1442,943],[1141,353,1323],[1489,1138,1474],[1462,1477,1440],[1474,1138,1488],[1442,1314,1443],[1446,1030,1546],[1484,1145,697],[1549,1443,1445],[1470,1572,1468],[1397,1239,1507],[1649,1825,1824],[1259,1440,1477],[1451,1450,1449],[978,1446,652],[1454,1456,1451],[1451,1456,1450],[341,1507,595],[933,1547,79],[804,1452,1060],[1454,1455,1456],[1398,1460,1454],[1455,877,1456],[1277,1831,1825],[804,1060,1458],[1339,1459,1595],[1314,1104,1443],[933,1448,1547],[147,1460,1398],[1460,1461,1454],[1454,1461,1455],[1292,1125,1464],[417,1531,1480],[1459,1339,1325],[811,1756,335],[1512,936,1490],[777,1529,1098],[147,1475,1460],[1464,253,1459],[836,855,482],[1487,1486,1307],[1104,1501,1443],[1439,1200,1532],[1475,1469,1460],[1460,1469,1461],[1325,1464,1459],[1277,1825,1649],[1532,1200,1077],[844,877,1455],[1572,933,1466],[1479,568,973],[1509,335,1305],[1339,1595,1759],[1469,1476,1461],[1461,1476,1455],[1104,1470,1468],[1464,1472,253],[1117,1091,1407],[1756,1542,335],[1206,1395,1188],[335,1542,1330],[835,844,1455],[1471,1598,1462],[1491,1442,1441],[835,1455,1476],[1441,1442,1443],[1489,1474,1473],[1251,1236,1250],[1030,1452,1477],[1598,1439,1532],[978,1598,1492],[1426,1401,938],[1448,1584,1482],[1724,1497,1475],[1475,1497,1469],[1484,1535,932],[1307,1486,1113],[1487,696,1495],[1037,1491,1441],[1030,1446,936],[1453,1487,1495],[696,1467,1495],[1138,1489,1483],[1497,1143,1469],[1469,1143,1476],[652,1598,978],[850,1043,1150],[1482,1584,1320],[1731,98,1697],[1113,1554,1573],[1524,1532,1494],[1496,1467,696],[1452,1259,1477],[296,1504,1497],[1504,1143,1497],[1143,1499,1476],[718,910,1498],[868,1540,1528],[817,1253,810],[1490,696,1487],[1440,1491,1037],[1510,676,595],[1488,1492,1517],[781,1239,1397],[1467,1519,1503],[1500,1307,1759],[1149,397,452],[1504,1514,1143],[1514,842,1143],[1125,733,1458],[1503,1531,1555],[1276,1036,1137],[1440,723,1123],[1036,1508,1137],[817,1508,1253],[103,883,1112],[1458,731,1472],[1512,1490,1487],[1487,1453,1486],[1138,978,1488],[1036,1253,1508],[1398,149,147],[1474,1517,1513],[1125,1458,1472],[1486,1453,1554],[1518,1534,758],[345,1058,1062],[928,1202,1369],[1554,1541,1505],[1464,1125,1472],[1504,764,1514],[304,426,573],[1505,742,1506],[1479,1572,1478],[1519,1483,1489],[833,716,1069],[1522,1534,1518],[1115,1513,777],[811,335,1432],[1591,1533,1407],[777,1517,1529],[1513,1517,777],[1498,910,1397],[1069,1539,833],[833,1539,1537],[1522,1551,1534],[1534,1551,1523],[1538,1137,1523],[910,51,1397],[1367,1373,703],[1466,1525,1468],[157,1186,1832],[1429,1511,1506],[1573,1505,1506],[1259,1452,804],[1503,1495,1467],[262,483,780],[1572,1466,1468],[1536,1556,716],[716,1556,1069],[1544,1523,1551],[1544,1538,1523],[1511,1573,1506],[933,1572,1448],[1543,1537,1539],[1537,1543,1522],[1091,933,79],[1519,1540,1545],[1549,1445,86],[1069,1548,1539],[1548,1543,1539],[1543,1551,1522],[1500,1487,1307],[68,784,1186],[1552,1544,1551],[1550,1538,1544],[1538,1550,1137],[1519,1473,1540],[1547,1448,1482],[1560,1563,1536],[1536,1563,1556],[1556,1548,1069],[1543,1558,1551],[1137,1550,1276],[1453,1495,1555],[1561,1543,1548],[1543,1561,1558],[1558,1566,1551],[1552,1550,1544],[1569,1557,1550],[1557,1276,1550],[1276,1557,254],[1531,1503,1480],[1535,1530,1510],[1545,1503,1519],[1547,1482,79],[1566,1552,1551],[1552,1569,1550],[1503,1545,1480],[703,1377,309],[1625,675,756],[1037,1441,88],[929,254,1557],[849,1567,1560],[1556,1564,1548],[1492,1529,1517],[1252,1429,1506],[1553,1027,1429],[1453,1555,1541],[1554,1453,1541],[1233,686,1553],[1328,1104,1314],[1564,1576,1548],[1548,1576,1561],[1557,1562,929],[1520,112,1668],[1483,1446,1138],[778,1570,1567],[1563,1564,1556],[1561,1565,1558],[1565,1566,1558],[1569,1552,1566],[1562,1557,1569],[1530,1535,1484],[1387,1402,1395],[1621,1634,1387],[1567,1568,1560],[1560,1568,1563],[1571,1569,1566],[1344,1330,1542],[1577,1431,1353],[1638,233,304],[1524,1463,1529],[1353,1431,1175],[1077,1200,1413],[1478,1470,1104],[1568,1575,1563],[1563,1575,1564],[1575,1576,1564],[1561,1576,1565],[1565,1574,1566],[1562,1515,929],[1555,96,1541],[1531,417,96],[1555,1531,96],[1246,45,1651],[208,1577,1353],[1586,1568,1567],[1574,1571,1566],[1571,1583,1569],[1474,1513,1528],[1239,1322,1535],[1478,1572,1470],[1570,1586,1567],[1488,1517,1474],[8,1833,1837],[1123,1442,1491],[1589,1568,1586],[1576,1594,1565],[1565,1594,1574],[1562,198,1515],[1559,1441,1549],[1441,1443,1549],[1135,425,1481],[1239,1535,1507],[1595,1487,1500],[1570,1585,1586],[1589,1578,1568],[1568,1578,1575],[1579,1569,1583],[1177,1577,208],[115,1236,110],[1578,1593,1575],[1587,1576,1575],[1576,1581,1594],[1571,1582,1583],[1588,1579,1583],[1579,1580,1562],[1569,1579,1562],[1562,1580,198],[1027,1511,1429],[1589,1593,1578],[1587,1581,1576],[1582,1574,1594],[1574,1582,1571],[1575,1593,1587],[1583,1582,1588],[1580,1590,198],[1587,1593,1581],[1505,1541,96],[1369,1577,1177],[1573,1554,1505],[1479,1478,568],[1585,1589,1586],[1369,1177,704],[766,1584,1334],[977,1257,1059],[1091,1591,1407],[1591,1091,1457],[1585,1604,1589],[1581,1592,1594],[1602,1582,1594],[1582,1608,1588],[1608,1579,1588],[1579,1597,1580],[1419,1590,1580],[1597,1419,1580],[1431,1577,1291],[1589,1604,1593],[1601,1596,1593],[1593,1596,1581],[1306,1511,1027],[1511,1113,1573],[1786,1412,1585],[1412,1604,1585],[1581,1596,1592],[1592,1602,1594],[1608,1599,1579],[1599,1611,1579],[1579,1611,1597],[1512,1487,253],[1519,1489,1473],[1545,1540,868],[1083,1187,1402],[1117,1407,1400],[1292,733,1125],[284,1240,1245],[1604,1600,1593],[1600,1601,1593],[1582,1607,1608],[789,1369,704],[1467,1483,1519],[1601,1613,1596],[1596,1613,1592],[1602,1607,1582],[1620,1553,1252],[1601,1605,1613],[1592,1613,1602],[1602,1606,1607],[1608,1609,1599],[1599,1609,1611],[1603,1597,1611],[1265,1419,1597],[1603,1265,1597],[1392,1206,45],[928,1369,789],[1474,1528,1473],[1104,1468,1501],[1412,1521,1604],[1613,1631,1602],[1607,1610,1608],[1608,1610,1609],[1476,863,835],[1495,1503,1555],[1498,1397,718],[1520,1668,7],[1604,1615,1600],[1605,1601,1600],[1602,1631,1606],[1606,1610,1607],[1759,1595,1500],[1292,1298,733],[1615,1604,1521],[1609,1603,1611],[652,1462,1598],[1468,1525,1445],[1443,1501,1445],[1134,1723,150],[1521,1622,1615],[1615,1616,1600],[1616,1605,1600],[1605,1616,1612],[1605,1612,1613],[1612,1617,1613],[1613,1617,1631],[1606,1614,1610],[1265,1603,1403],[448,417,1480],[1595,253,1487],[1501,1468,1445],[1383,1456,877],[1490,1496,696],[1610,1627,1609],[1627,1621,1609],[1591,1481,1533],[1598,1471,1439],[1353,1261,703],[1606,1631,1614],[1609,1621,1403],[1532,1077,1494],[1528,1115,513],[1546,652,1446],[1211,928,1365],[1540,1473,1528],[1078,1502,1787],[1425,1430,1438],[1617,1630,1631],[959,749,944],[566,570,603],[1716,310,1521],[775,452,397],[1615,1636,1616],[1616,1636,1612],[1610,1632,1627],[789,704,1258],[1457,1481,1591],[1769,1756,811],[207,1629,722],[1629,1625,722],[1224,1277,1622],[1622,1636,1615],[1636,1646,1612],[1612,1630,1617],[1631,1626,1614],[1614,1632,1610],[1506,104,95],[1481,1457,1136],[1123,943,1442],[936,1446,1496],[1499,863,1476],[1629,1031,1625],[1233,1509,686],[1633,1634,1621],[1621,1387,1403],[1472,1512,253],[1177,208,704],[1277,1636,1622],[1626,1632,1614],[1627,1633,1621],[936,1496,1490],[185,1454,1451],[731,936,1512],[1638,1635,207],[553,1263,1264],[1653,1212,1639],[1633,1627,1632],[1633,1387,1634],[1458,1060,731],[368,1307,1113],[1264,1031,1629],[1152,850,1150],[1277,1644,1636],[1646,1637,1612],[1637,1630,1612],[1647,1631,1630],[1647,1626,1631],[1422,1524,1494],[1030,652,1546],[1635,1629,207],[1635,1264,1629],[1639,1646,1636],[1637,1640,1630],[1641,1632,1626],[1632,1642,1633],[1633,1643,1387],[842,1499,1143],[865,863,1499],[1516,978,1492],[67,1130,784],[1103,1505,96],[88,1441,1200],[1644,1639,1636],[1640,1647,1630],[1647,1641,1626],[1633,1648,1643],[1492,1532,1524],[1488,1516,1492],[1037,1471,1462],[612,1264,1635],[1502,1078,1124],[1641,1642,1632],[1648,1633,1642],[1528,513,868],[1492,1598,1532],[1095,991,760],[679,157,1664],[760,1128,1785],[1277,1650,1644],[320,1022,244],[1559,1549,86],[1676,1520,7],[1488,978,1516],[1095,760,1785],[1128,384,1120],[304,312,1638],[1081,1638,312],[1081,1635,1638],[103,612,1635],[652,1477,1462],[1650,1645,1644],[1645,1639,1644],[1639,1637,1646],[1640,1090,1647],[1654,1641,1647],[1654,1642,1641],[1654,1648,1642],[1643,1402,1387],[1432,335,1509],[384,1128,760],[1652,312,304],[103,1243,612],[1277,1649,1650],[1090,1654,1647],[1643,1648,1402],[1134,324,1675],[679,68,157],[1652,1081,312],[1136,301,803],[1653,1639,1645],[723,1440,1259],[803,854,1136],[104,1506,742],[1112,159,103],[1654,1083,1648],[977,1651,1257],[1397,1507,718],[1081,103,1635],[1650,677,1645],[1083,1402,1648],[1706,1655,1671],[1624,1704,1711],[767,2,1],[608,794,294],[1678,1683,1686],[767,1682,2],[1669,1692,1675],[296,1681,764],[1671,1656,1672],[17,1673,1679],[1706,1671,1673],[1662,1674,1699],[1655,1657,1656],[418,84,915],[1526,1514,764],[1658,1657,567],[870,1695,764],[813,1697,98],[1659,821,5],[60,1013,848],[1013,110,1213],[661,1038,1692],[1660,1703,17],[1693,1673,17],[1663,1715,1743],[1013,115,110],[344,1733,32],[1670,1663,1743],[1670,1743,1738],[1677,1670,1738],[1661,4,3],[1084,1683,1678],[1728,793,1130],[1683,1767,1196],[1677,1738,1196],[1279,1786,853],[294,1038,608],[1279,1689,1786],[870,18,1708],[870,1680,1695],[1705,10,1670],[1084,1767,1683],[1196,1738,1686],[1750,870,1681],[1750,18,870],[1773,1703,1660],[1135,47,425],[150,323,1134],[1707,1655,1706],[1741,344,1687],[1685,1691,1684],[1684,1691,802],[1672,1656,0],[1038,124,608],[1671,1672,1690],[1628,1218,1767],[1686,1275,1667],[1493,1750,1681],[1773,18,1750],[1773,1660,18],[1679,1671,16],[1735,1706,1673],[1667,1678,1686],[1688,1658,1],[1656,1688,0],[1293,1281,1458],[1698,1678,1667],[1696,1130,1722],[1698,1667,1696],[1715,1662,1699],[1692,1038,294],[1682,767,357],[1669,661,1692],[802,1702,824],[1028,1067,1784],[822,1624,778],[119,813,861],[1218,1670,1677],[1703,1693,17],[1658,1710,1],[750,1730,1729],[1701,750,1729],[1693,1735,1673],[1731,1694,98],[1691,1702,802],[783,1729,1719],[1680,870,1708],[1707,1709,1655],[533,756,675],[1691,1210,1702],[11,1705,1670],[1767,1218,1196],[1218,1677,1196],[1664,1716,1721],[1729,1725,1719],[1729,1072,1725],[1210,1116,1702],[1702,1720,824],[1682,1661,2],[1713,1719,1721],[1716,1786,1713],[1730,1722,1072],[294,1717,1811],[1692,294,1666],[1659,680,821],[824,1720,1714],[1726,1731,1718],[345,1062,1045],[1738,1743,1275],[1075,1089,1071],[783,1719,1689],[1275,684,1728],[1692,1666,1665],[1675,1692,1665],[294,1811,1666],[1716,1664,310],[1678,1698,1700],[6,9,1727],[676,649,595],[381,31,361],[1723,1804,1772],[1727,9,1694],[1720,1089,1714],[1786,1716,1412],[1683,1196,1686],[1718,1697,1085],[1116,1739,1702],[1739,1734,1720],[1702,1739,1720],[1089,1720,1734],[509,748,1745],[1743,1715,1726],[1717,294,794],[1116,1732,1739],[1718,1731,1697],[1696,1667,1130],[1134,1665,1723],[1694,712,98],[101,1687,102],[391,1736,101],[662,636,642],[1734,1447,1089],[1089,1447,1071],[436,99,493],[1689,1279,783],[1485,1465,1342],[1736,1687,101],[344,1741,1733],[1741,1742,1733],[1735,829,1706],[829,1707,1706],[1485,1332,1465],[952,1126,1742],[1747,1447,1734],[879,892,645],[1730,1146,1696],[829,1709,1707],[1709,1712,1655],[118,1739,1732],[1332,1744,1465],[1687,1749,1741],[1741,1758,1742],[679,1072,68],[1072,1722,68],[118,1747,1739],[1747,1734,1739],[1465,1744,1736],[1736,1740,1687],[1704,1701,783],[1665,624,1723],[1722,1130,67],[1025,1055,467],[1444,14,1701],[558,522,530],[1657,1658,1688],[1339,1746,1332],[1332,1748,1744],[1687,1740,1749],[1741,1749,1758],[1109,952,1742],[1747,118,141],[1671,1690,1628],[1671,1628,16],[1657,1688,1656],[1745,748,1447],[357,767,1710],[1746,1748,1332],[1146,1700,1698],[1759,1307,1338],[1239,781,1322],[1745,1447,1747],[522,1745,1747],[316,717,595],[148,1493,1724],[1758,1109,1742],[1725,1072,679],[726,719,1661],[1695,1680,1526],[1772,1750,1493],[148,1772,1493],[1542,1751,1101],[952,1109,1086],[1744,1752,1736],[1736,1752,1740],[1753,1755,1740],[391,1342,1736],[821,112,1520],[557,530,1747],[530,522,1747],[994,879,645],[1542,1756,1751],[1813,1693,1703],[1746,1754,1748],[1748,1764,1744],[1752,1757,1740],[1740,1757,1753],[1749,1740,1755],[1755,1763,1749],[1763,1758,1749],[1275,1743,684],[1813,1735,1693],[1107,1099,1101],[1723,624,1804],[1403,1603,1609],[1748,1754,1764],[1744,1757,1752],[1760,1109,1758],[1465,1736,1342],[436,115,99],[1686,1738,1275],[1751,1766,1101],[1759,1754,1746],[1755,1753,1763],[1570,1279,853],[1701,1146,750],[1655,1656,1671],[11,1670,1218],[1761,1751,1756],[1766,1107,1101],[1726,1623,1731],[1711,1704,1279],[67,784,68],[558,530,545],[1620,1618,1233],[1769,1761,1756],[102,1687,344],[1338,1754,1759],[1754,232,1764],[1744,1765,1757],[1757,1763,1753],[1762,1760,1758],[1760,1771,1109],[1339,1759,1746],[1675,1665,1134],[1730,1696,1722],[1774,1751,1761],[1766,1780,1107],[1780,1105,1107],[1764,1765,1744],[1763,1762,1758],[1772,1773,1750],[1811,1813,1703],[1434,1769,1432],[1780,1766,1751],[232,1781,1764],[1711,1279,1570],[1688,1,0],[1774,1780,1751],[1764,1781,1765],[1765,1768,1757],[1757,1768,1763],[1777,1782,1760],[1762,1777,1760],[1769,1774,1761],[1763,1777,1762],[1760,1782,1771],[232,1737,1781],[1768,1776,1763],[272,255,774],[1669,994,661],[1618,1769,1434],[1765,589,1768],[1770,1777,1763],[1701,1729,783],[1783,1774,1769],[1789,1780,1774],[589,1775,1768],[1776,1770,1763],[1782,1778,1771],[1771,1778,1070],[624,1703,1773],[624,1811,1703],[1620,1244,1618],[1779,1769,1618],[1779,1783,1769],[739,1735,1813],[1775,1776,1768],[1790,1777,1770],[1777,1778,1782],[1725,679,1721],[733,1293,1458],[1802,1618,1244],[1802,1779,1618],[1788,1783,1779],[1789,1774,1783],[1796,1780,1789],[1796,1119,1780],[1823,1817,325],[1699,1727,1623],[750,1146,1730],[1497,1724,296],[1128,1119,1796],[61,62,71],[1131,413,824],[1114,1111,249],[1784,1776,1775],[1123,723,1283],[1791,1788,1779],[1788,1789,1783],[1095,1797,1074],[1028,1784,1775],[1784,1770,1776],[1777,1790,1778],[1793,1797,1095],[1797,1800,1074],[1798,1790,1770],[1805,1802,1244],[1802,1791,1779],[1792,1789,1788],[1793,1785,1128],[1793,1095,1785],[1074,1800,1619],[741,457,593],[1798,1770,1784],[1798,1794,1790],[1786,1689,1713],[684,1726,1718],[1728,1085,793],[1795,1787,1502],[1806,1802,1805],[1819,1788,1791],[1067,1798,1784],[1790,1794,1778],[1795,1502,1124],[1801,1805,1787],[1807,1791,1802],[1807,1819,1791],[1819,1792,1788],[1799,1128,1796],[994,645,661],[684,1085,1728],[684,1718,1085],[1699,1623,1726],[1801,1787,1795],[1808,1789,1792],[1808,1796,1789],[1799,1793,1128],[1809,1797,1793],[1809,1803,1797],[1803,1800,1797],[1067,1794,1798],[774,255,1778],[1673,1671,1679],[879,1669,888],[19,1807,1802],[1810,1619,1800],[879,994,1669],[1794,774,1778],[1723,1772,148],[1804,1773,1772],[1814,1795,1124],[1649,1814,1124],[1814,1801,1795],[1812,1806,1805],[19,1802,1806],[19,1819,1807],[1810,1800,1803],[1804,624,1773],[1714,1131,824],[1801,1812,1805],[1812,19,1806],[1808,1792,1819],[1799,1809,1793],[1821,1810,1803],[1717,739,1813],[1061,1619,1822],[1794,1817,774],[79,1482,144],[1815,1801,1814],[23,1819,19],[589,1028,1775],[1817,1823,774],[1689,1719,1713],[1824,1814,1649],[1827,1818,1801],[1818,1812,1801],[1818,19,1812],[1818,20,19],[1816,1809,1799],[1821,1803,1809],[1822,1619,1810],[124,708,608],[1663,10,1715],[1815,1827,1801],[1820,1808,1819],[23,1820,1819],[603,1810,1821],[603,1822,1810],[1085,1697,793],[1628,1690,11],[1527,1704,1624],[1730,1072,1729],[1526,1444,1704],[1526,1680,1444],[1704,1444,1701],[1816,1821,1809],[1722,67,68],[317,272,1823],[1716,1713,1721],[16,1628,1767],[1527,1526,1704],[1824,1826,1814],[1814,1826,1815],[1818,21,20],[1835,1808,1820],[603,570,1822],[226,1070,1778],[1013,1181,1179],[1721,679,1664],[1717,1813,1811],[1828,1827,1815],[22,1820,23],[22,1835,1820],[1830,603,1821],[719,1659,5],[643,567,1657],[1717,794,739],[1825,1826,1824],[1828,1815,1826],[1829,21,1818],[1808,1835,13],[4,719,5],[10,1662,1715],[1828,1832,1827],[1832,1818,1827],[12,1833,1816],[1833,1821,1816],[1833,1830,1821],[14,1146,1701],[1186,1829,1818],[1280,603,1830],[14,1700,1146],[1667,1728,1130],[1825,1834,1826],[1834,1828,1826],[1832,1186,1818],[1836,13,1835],[1624,1711,1570],[778,1624,1570],[1719,1725,1721],[1002,1825,1831],[1002,1834,1825],[1834,1832,1828],[1186,21,1829],[1836,1835,22],[1837,1833,12],[1280,1830,1833],[1667,1275,1728],[16,1767,1084],[589,1765,1838],[1765,1781,1838],[1781,1737,1838],[1737,982,1838],[982,1053,1838],[1053,816,1838],[816,589,1838]]\n"]}
require=(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
module.exports = getSize
function getSize(element) {
// Handle cases where the element is not already
// attached to the DOM by briefly appending it
// to document.body, and removing it again later.
if (element === window || element === document.body) {
return [window.innerWidth, window.innerHeight]
}
if (!element.parentNode) {
var temporary = true
document.body.appendChild(element)
}
var bounds = element.getBoundingClientRect()
var styles = getComputedStyle(element)
var height = (bounds.height|0)
+ parse(styles.getPropertyValue('margin-top'))
+ parse(styles.getPropertyValue('margin-bottom'))
var width = (bounds.width|0)
+ parse(styles.getPropertyValue('margin-left'))
+ parse(styles.getPropertyValue('margin-right'))
if (temporary) {
document.body.removeChild(element)
}
return [width, height]
}
function parse(prop) {
return parseFloat(prop) || 0
}
},{}],"canvas-fit":[function(require,module,exports){
var size = require('element-size')
module.exports = fit
var scratch = new Float32Array(2)
function fit(canvas, parent, scale) {
var isSVG = canvas.nodeName.toUpperCase() === 'SVG'
canvas.style.position = canvas.style.position || 'absolute'
canvas.style.top = 0
canvas.style.left = 0
resize.scale = parseFloat(scale || 1)
resize.parent = parent
return resize()
function resize() {
var p = resize.parent || canvas.parentNode
if (typeof p === 'function') {
var dims = p(scratch) || scratch
var width = dims[0]
var height = dims[1]
} else
if (p && p !== document.body) {
var psize = size(p)
var width = psize[0]|0
var height = psize[1]|0
} else {
var width = window.innerWidth
var height = window.innerHeight
}
if (isSVG) {
canvas.setAttribute('width', width * resize.scale + 'px')
canvas.setAttribute('height', height * resize.scale + 'px')
} else {
canvas.width = width * resize.scale
canvas.height = height * resize.scale
}
canvas.style.width = width + 'px'
canvas.style.height = height + 'px'
return resize
}
}
},{"element-size":1}]},{},[])
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL2hvbWUvYWRtaW4vYnJvd3NlcmlmeS1jZG4vbm9kZV9tb2R1bGVzL2Jyb3dzZXJpZnkvbm9kZV9tb2R1bGVzL2Jyb3dzZXItcGFjay9fcHJlbHVkZS5qcyIsIm5vZGVfbW9kdWxlcy9lbGVtZW50LXNpemUvaW5kZXguanMiLCJpbmRleC5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTtBQ0FBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FDbENBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBIiwiZmlsZSI6ImdlbmVyYXRlZC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzQ29udGVudCI6WyIoZnVuY3Rpb24gZSh0LG4scil7ZnVuY3Rpb24gcyhvLHUpe2lmKCFuW29dKXtpZighdFtvXSl7dmFyIGE9dHlwZW9mIHJlcXVpcmU9PVwiZnVuY3Rpb25cIiYmcmVxdWlyZTtpZighdSYmYSlyZXR1cm4gYShvLCEwKTtpZihpKXJldHVybiBpKG8sITApO3ZhciBmPW5ldyBFcnJvcihcIkNhbm5vdCBmaW5kIG1vZHVsZSAnXCIrbytcIidcIik7dGhyb3cgZi5jb2RlPVwiTU9EVUxFX05PVF9GT1VORFwiLGZ9dmFyIGw9bltvXT17ZXhwb3J0czp7fX07dFtvXVswXS5jYWxsKGwuZXhwb3J0cyxmdW5jdGlvbihlKXt2YXIgbj10W29dWzFdW2VdO3JldHVybiBzKG4/bjplKX0sbCxsLmV4cG9ydHMsZSx0LG4scil9cmV0dXJuIG5bb10uZXhwb3J0c312YXIgaT10eXBlb2YgcmVxdWlyZT09XCJmdW5jdGlvblwiJiZyZXF1aXJlO2Zvcih2YXIgbz0wO288ci5sZW5ndGg7bysrKXMocltvXSk7cmV0dXJuIHN9KSIsIm1vZHVsZS5leHBvcnRzID0gZ2V0U2l6ZVxuXG5mdW5jdGlvbiBnZXRTaXplKGVsZW1lbnQpIHtcbiAgLy8gSGFuZGxlIGNhc2VzIHdoZXJlIHRoZSBlbGVtZW50IGlzIG5vdCBhbHJlYWR5XG4gIC8vIGF0dGFjaGVkIHRvIHRoZSBET00gYnkgYnJpZWZseSBhcHBlbmRpbmcgaXRcbiAgLy8gdG8gZG9jdW1lbnQuYm9keSwgYW5kIHJlbW92aW5nIGl0IGFnYWluIGxhdGVyLlxuICBpZiAoZWxlbWVudCA9PT0gd2luZG93IHx8IGVsZW1lbnQgPT09IGRvY3VtZW50LmJvZHkpIHtcbiAgICByZXR1cm4gW3dpbmRvdy5pbm5lcldpZHRoLCB3aW5kb3cuaW5uZXJIZWlnaHRdXG4gIH1cblxuICBpZiAoIWVsZW1lbnQucGFyZW50Tm9kZSkge1xuICAgIHZhciB0ZW1wb3JhcnkgPSB0cnVlXG4gICAgZG9jdW1lbnQuYm9keS5hcHBlbmRDaGlsZChlbGVtZW50KVxuICB9XG5cbiAgdmFyIGJvdW5kcyA9IGVsZW1lbnQuZ2V0Qm91bmRpbmdDbGllbnRSZWN0KClcbiAgdmFyIHN0eWxlcyA9IGdldENvbXB1dGVkU3R5bGUoZWxlbWVudClcbiAgdmFyIGhlaWdodCA9IChib3VuZHMuaGVpZ2h0fDApXG4gICAgKyBwYXJzZShzdHlsZXMuZ2V0UHJvcGVydHlWYWx1ZSgnbWFyZ2luLXRvcCcpKVxuICAgICsgcGFyc2Uoc3R5bGVzLmdldFByb3BlcnR5VmFsdWUoJ21hcmdpbi1ib3R0b20nKSlcbiAgdmFyIHdpZHRoICA9IChib3VuZHMud2lkdGh8MClcbiAgICArIHBhcnNlKHN0eWxlcy5nZXRQcm9wZXJ0eVZhbHVlKCdtYXJnaW4tbGVmdCcpKVxuICAgICsgcGFyc2Uoc3R5bGVzLmdldFByb3BlcnR5VmFsdWUoJ21hcmdpbi1yaWdodCcpKVxuXG4gIGlmICh0ZW1wb3JhcnkpIHtcbiAgICBkb2N1bWVudC5ib2R5LnJlbW92ZUNoaWxkKGVsZW1lbnQpXG4gIH1cblxuICByZXR1cm4gW3dpZHRoLCBoZWlnaHRdXG59XG5cbmZ1bmN0aW9uIHBhcnNlKHByb3ApIHtcbiAgcmV0dXJuIHBhcnNlRmxvYXQocHJvcCkgfHwgMFxufVxuIiwidmFyIHNpemUgPSByZXF1aXJlKCdlbGVtZW50LXNpemUnKVxuXG5tb2R1bGUuZXhwb3J0cyA9IGZpdFxuXG52YXIgc2NyYXRjaCA9IG5ldyBGbG9hdDMyQXJyYXkoMilcblxuZnVuY3Rpb24gZml0KGNhbnZhcywgcGFyZW50LCBzY2FsZSkge1xuICB2YXIgaXNTVkcgPSBjYW52YXMubm9kZU5hbWUudG9VcHBlckNhc2UoKSA9PT0gJ1NWRydcblxuICBjYW52YXMuc3R5bGUucG9zaXRpb24gPSBjYW52YXMuc3R5bGUucG9zaXRpb24gfHwgJ2Fic29sdXRlJ1xuICBjYW52YXMuc3R5bGUudG9wID0gMFxuICBjYW52YXMuc3R5bGUubGVmdCA9IDBcblxuICByZXNpemUuc2NhbGUgID0gcGFyc2VGbG9hdChzY2FsZSB8fCAxKVxuICByZXNpemUucGFyZW50ID0gcGFyZW50XG5cbiAgcmV0dXJuIHJlc2l6ZSgpXG5cbiAgZnVuY3Rpb24gcmVzaXplKCkge1xuICAgIHZhciBwID0gcmVzaXplLnBhcmVudCB8fCBjYW52YXMucGFyZW50Tm9kZVxuICAgIGlmICh0eXBlb2YgcCA9PT0gJ2Z1bmN0aW9uJykge1xuICAgICAgdmFyIGRpbXMgICA9IHAoc2NyYXRjaCkgfHwgc2NyYXRjaFxuICAgICAgdmFyIHdpZHRoICA9IGRpbXNbMF1cbiAgICAgIHZhciBoZWlnaHQgPSBkaW1zWzFdXG4gICAgfSBlbHNlXG4gICAgaWYgKHAgJiYgcCAhPT0gZG9jdW1lbnQuYm9keSkge1xuICAgICAgdmFyIHBzaXplICA9IHNpemUocClcbiAgICAgIHZhciB3aWR0aCAgPSBwc2l6ZVswXXwwXG4gICAgICB2YXIgaGVpZ2h0ID0gcHNpemVbMV18MFxuICAgIH0gZWxzZSB7XG4gICAgICB2YXIgd2lkdGggID0gd2luZG93LmlubmVyV2lkdGhcbiAgICAgIHZhciBoZWlnaHQgPSB3aW5kb3cuaW5uZXJIZWlnaHRcbiAgICB9XG5cbiAgICBpZiAoaXNTVkcpIHtcbiAgICAgIGNhbnZhcy5zZXRBdHRyaWJ1dGUoJ3dpZHRoJywgd2lkdGggKiByZXNpemUuc2NhbGUgKyAncHgnKVxuICAgICAgY2FudmFzLnNldEF0dHJpYnV0ZSgnaGVpZ2h0JywgaGVpZ2h0ICogcmVzaXplLnNjYWxlICsgJ3B4JylcbiAgICB9IGVsc2Uge1xuICAgICAgY2FudmFzLndpZHRoID0gd2lkdGggKiByZXNpemUuc2NhbGVcbiAgICAgIGNhbnZhcy5oZWlnaHQgPSBoZWlnaHQgKiByZXNpemUuc2NhbGVcbiAgICB9XG5cbiAgICBjYW52YXMuc3R5bGUud2lkdGggPSB3aWR0aCArICdweCdcbiAgICBjYW52YXMuc3R5bGUuaGVpZ2h0ID0gaGVpZ2h0ICsgJ3B4J1xuXG4gICAgcmV0dXJuIHJlc2l6ZVxuICB9XG59XG4iXX0=
require=(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({"angle-normals":[function(require,module,exports){
'use strict'
module.exports = angleNormals
function hypot(x, y, z) {
return Math.sqrt(Math.pow(x,2) + Math.pow(y,2) + Math.pow(z,2))
}
function weight(s, r, a) {
return Math.atan2(r, (s - a))
}
function mulAdd(dest, s, x, y, z) {
dest[0] += s * x
dest[1] += s * y
dest[2] += s * z
}
function angleNormals(cells, positions) {
var numVerts = positions.length
var numCells = cells.length
//Allocate normal array
var normals = new Array(numVerts)
for(var i=0; i<numVerts; ++i) {
normals[i] = [0,0,0]
}
//Scan cells, and
for(var i=0; i<numCells; ++i) {
var cell = cells[i]
var a = positions[cell[0]]
var b = positions[cell[1]]
var c = positions[cell[2]]
var abx = a[0] - b[0]
var aby = a[1] - b[1]
var abz = a[2] - b[2]
var ab = hypot(abx, aby, abz)
var bcx = b[0] - c[0]
var bcy = b[1] - c[1]
var bcz = b[2] - c[2]
var bc = hypot(bcx, bcy, bcz)
var cax = c[0] - a[0]
var cay = c[1] - a[1]
var caz = c[2] - a[2]
var ca = hypot(cax, cay, caz)
if(Math.min(ab, bc, ca) < 1e-6) {
continue
}
var s = 0.5 * (ab + bc + ca)
var r = Math.sqrt((s - ab)*(s - bc)*(s - ca)/s)
var nx = aby * bcz - abz * bcy
var ny = abz * bcx - abx * bcz
var nz = abx * bcy - aby * bcx
var nl = hypot(nx, ny, nz)
nx /= nl
ny /= nl
nz /= nl
mulAdd(normals[cell[0]], weight(s, r, bc), nx, ny, nz)
mulAdd(normals[cell[1]], weight(s, r, ca), nx, ny, nz)
mulAdd(normals[cell[2]], weight(s, r, ab), nx, ny, nz)
}
//Normalize all the normals
for(var i=0; i<numVerts; ++i) {
var n = normals[i]
var l = Math.sqrt(
Math.pow(n[0], 2) +
Math.pow(n[1], 2) +
Math.pow(n[2], 2))
if(l < 1e-8) {
n[0] = 1
n[1] = 0
n[2] = 0
continue
}
n[0] /= l
n[1] /= l
n[2] /= l
}
return normals
}
},{}]},{},[])
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL2hvbWUvYWRtaW4vYnJvd3NlcmlmeS1jZG4vbm9kZV9tb2R1bGVzL2Jyb3dzZXJpZnkvbm9kZV9tb2R1bGVzL2Jyb3dzZXItcGFjay9fcHJlbHVkZS5qcyIsImFuZ2xlLW5vcm1hbHMuanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7QUNBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSIsImZpbGUiOiJnZW5lcmF0ZWQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlc0NvbnRlbnQiOlsiKGZ1bmN0aW9uIGUodCxuLHIpe2Z1bmN0aW9uIHMobyx1KXtpZighbltvXSl7aWYoIXRbb10pe3ZhciBhPXR5cGVvZiByZXF1aXJlPT1cImZ1bmN0aW9uXCImJnJlcXVpcmU7aWYoIXUmJmEpcmV0dXJuIGEobywhMCk7aWYoaSlyZXR1cm4gaShvLCEwKTt2YXIgZj1uZXcgRXJyb3IoXCJDYW5ub3QgZmluZCBtb2R1bGUgJ1wiK28rXCInXCIpO3Rocm93IGYuY29kZT1cIk1PRFVMRV9OT1RfRk9VTkRcIixmfXZhciBsPW5bb109e2V4cG9ydHM6e319O3Rbb11bMF0uY2FsbChsLmV4cG9ydHMsZnVuY3Rpb24oZSl7dmFyIG49dFtvXVsxXVtlXTtyZXR1cm4gcyhuP246ZSl9LGwsbC5leHBvcnRzLGUsdCxuLHIpfXJldHVybiBuW29dLmV4cG9ydHN9dmFyIGk9dHlwZW9mIHJlcXVpcmU9PVwiZnVuY3Rpb25cIiYmcmVxdWlyZTtmb3IodmFyIG89MDtvPHIubGVuZ3RoO28rKylzKHJbb10pO3JldHVybiBzfSkiLCIndXNlIHN0cmljdCdcblxubW9kdWxlLmV4cG9ydHMgPSBhbmdsZU5vcm1hbHNcblxuZnVuY3Rpb24gaHlwb3QoeCwgeSwgeikge1xuICByZXR1cm4gTWF0aC5zcXJ0KE1hdGgucG93KHgsMikgKyBNYXRoLnBvdyh5LDIpICsgTWF0aC5wb3coeiwyKSlcbn1cblxuZnVuY3Rpb24gd2VpZ2h0KHMsIHIsIGEpIHtcbiAgcmV0dXJuIE1hdGguYXRhbjIociwgKHMgLSBhKSlcbn1cblxuZnVuY3Rpb24gbXVsQWRkKGRlc3QsIHMsIHgsIHksIHopIHtcbiAgZGVzdFswXSArPSBzICogeFxuICBkZXN0WzFdICs9IHMgKiB5XG4gIGRlc3RbMl0gKz0gcyAqIHpcbn1cblxuZnVuY3Rpb24gYW5nbGVOb3JtYWxzKGNlbGxzLCBwb3NpdGlvbnMpIHtcbiAgdmFyIG51bVZlcnRzID0gcG9zaXRpb25zLmxlbmd0aFxuICB2YXIgbnVtQ2VsbHMgPSBjZWxscy5sZW5ndGhcblxuICAvL0FsbG9jYXRlIG5vcm1hbCBhcnJheVxuICB2YXIgbm9ybWFscyA9IG5ldyBBcnJheShudW1WZXJ0cylcbiAgZm9yKHZhciBpPTA7IGk8bnVtVmVydHM7ICsraSkge1xuICAgIG5vcm1hbHNbaV0gPSBbMCwwLDBdXG4gIH1cblxuICAvL1NjYW4gY2VsbHMsIGFuZFxuICBmb3IodmFyIGk9MDsgaTxudW1DZWxsczsgKytpKSB7XG4gICAgdmFyIGNlbGwgPSBjZWxsc1tpXVxuICAgIHZhciBhID0gcG9zaXRpb25zW2NlbGxbMF1dXG4gICAgdmFyIGIgPSBwb3NpdGlvbnNbY2VsbFsxXV1cbiAgICB2YXIgYyA9IHBvc2l0aW9uc1tjZWxsWzJdXVxuXG4gICAgdmFyIGFieCA9IGFbMF0gLSBiWzBdXG4gICAgdmFyIGFieSA9IGFbMV0gLSBiWzFdXG4gICAgdmFyIGFieiA9IGFbMl0gLSBiWzJdXG4gICAgdmFyIGFiID0gaHlwb3QoYWJ4LCBhYnksIGFieilcblxuICAgIHZhciBiY3ggPSBiWzBdIC0gY1swXVxuICAgIHZhciBiY3kgPSBiWzFdIC0gY1sxXVxuICAgIHZhciBiY3ogPSBiWzJdIC0gY1syXVxuICAgIHZhciBiYyA9IGh5cG90KGJjeCwgYmN5LCBiY3opXG5cbiAgICB2YXIgY2F4ID0gY1swXSAtIGFbMF1cbiAgICB2YXIgY2F5ID0gY1sxXSAtIGFbMV1cbiAgICB2YXIgY2F6ID0gY1syXSAtIGFbMl1cbiAgICB2YXIgY2EgPSBoeXBvdChjYXgsIGNheSwgY2F6KVxuXG4gICAgaWYoTWF0aC5taW4oYWIsIGJjLCBjYSkgPCAxZS02KSB7XG4gICAgICBjb250aW51ZVxuICAgIH1cblxuICAgIHZhciBzID0gMC41ICogKGFiICsgYmMgKyBjYSlcbiAgICB2YXIgciA9IE1hdGguc3FydCgocyAtIGFiKSoocyAtIGJjKSoocyAtIGNhKS9zKVxuXG4gICAgdmFyIG54ID0gYWJ5ICogYmN6IC0gYWJ6ICogYmN5XG4gICAgdmFyIG55ID0gYWJ6ICogYmN4IC0gYWJ4ICogYmN6XG4gICAgdmFyIG56ID0gYWJ4ICogYmN5IC0gYWJ5ICogYmN4XG4gICAgdmFyIG5sID0gaHlwb3QobngsIG55LCBueilcbiAgICBueCAvPSBubFxuICAgIG55IC89IG5sXG4gICAgbnogLz0gbmxcblxuICAgIG11bEFkZChub3JtYWxzW2NlbGxbMF1dLCB3ZWlnaHQocywgciwgYmMpLCBueCwgbnksIG56KVxuICAgIG11bEFkZChub3JtYWxzW2NlbGxbMV1dLCB3ZWlnaHQocywgciwgY2EpLCBueCwgbnksIG56KVxuICAgIG11bEFkZChub3JtYWxzW2NlbGxbMl1dLCB3ZWlnaHQocywgciwgYWIpLCBueCwgbnksIG56KVxuICB9XG5cbiAgLy9Ob3JtYWxpemUgYWxsIHRoZSBub3JtYWxzXG4gIGZvcih2YXIgaT0wOyBpPG51bVZlcnRzOyArK2kpIHtcbiAgICB2YXIgbiA9IG5vcm1hbHNbaV1cbiAgICB2YXIgbCA9IE1hdGguc3FydChcbiAgICAgIE1hdGgucG93KG5bMF0sIDIpICtcbiAgICAgIE1hdGgucG93KG5bMV0sIDIpICtcbiAgICAgIE1hdGgucG93KG5bMl0sIDIpKVxuICAgIGlmKGwgPCAxZS04KSB7XG4gICAgICBuWzBdID0gMVxuICAgICAgblsxXSA9IDBcbiAgICAgIG5bMl0gPSAwXG4gICAgICBjb250aW51ZVxuICAgIH1cbiAgICBuWzBdIC89IGxcbiAgICBuWzFdIC89IGxcbiAgICBuWzJdIC89IGxcbiAgfVxuXG4gIHJldHVybiBub3JtYWxzXG59XG4iXX0=
require=(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
var glTypes = require('./constants/dtypes.json')
var check = require('./util/check')
var GL_FLOAT = 5126
module.exports = function wrapAttributeState (
gl,
extensions,
limits,
bufferState,
stringStore) {
var attributeState = {}
function AttributeRecord () {
this.pointer = false
this.x = 0.0
this.y = 0.0
this.z = 0.0
this.w = 0.0
this.buffer = null
this.size = 0
this.normalized = false
this.type = GL_FLOAT
this.offset = 0
this.stride = 0
this.divisor = 0
}
function attributeRecordsEqual (left, right, size) {
if (!left.pointer) {
return !right.pointer &&
left.x === right.x &&
left.y === right.y &&
left.z === right.z &&
left.w === right.w
} else {
return right.pointer &&
left.buffer === right.buffer &&
left.size === size &&
left.normalized === right.normalized &&
left.type === right.type &&
left.offset === right.offset &&
left.stride === right.stride &&
left.divisor === right.divisor
}
}
function setAttributeRecord (left, right, size) {
var pointer = left.pointer = right.pointer
if (pointer) {
left.buffer = right.buffer
left.size = size
left.normalized = right.normalized
left.type = right.type
left.offset = right.offset
left.stride = right.stride
left.divisor = right.divisor
} else {
left.x = right.x
left.y = right.y
left.z = right.z
left.w = right.w
}
}
var NUM_ATTRIBUTES = limits.maxAttributes
var attributeBindings = new Array(NUM_ATTRIBUTES)
for (var i = 0; i < NUM_ATTRIBUTES; ++i) {
attributeBindings[i] = new AttributeRecord()
}
function AttributeStack (name) {
this.records = []
this.name = name
}
function stackTop (stack) {
var records = stack.records
return records[records.length - 1]
}
// ===================================================
// BIND AN ATTRIBUTE
// ===================================================
function bindAttributeRecord (index, current, next, insize) {
var size = next.size || insize
if (attributeRecordsEqual(current, next, size)) {
return
}
if (!next.pointer) {
if (current.pointer) {
gl.disableVertexAttribArray(index)
}
gl.vertexAttrib4f(index, next.x, next.y, next.z, next.w)
} else {
if (!current.pointer) {
gl.enableVertexAttribArray(index)
}
if (current.buffer !== next.buffer) {
next.buffer.bind()
}
gl.vertexAttribPointer(
index,
size,
next.type,
next.normalized,
next.stride,
next.offset)
var extInstancing = extensions.angle_instanced_arrays
if (extInstancing) {
extInstancing.vertexAttribDivisorANGLE(index, next.divisor)
}
}
setAttributeRecord(current, next, size)
}
function bindAttribute (index, current, attribStack, size) {
bindAttributeRecord(index, current, stackTop(attribStack), size)
}
// ===================================================
// DEFINE A NEW ATTRIBUTE
// ===================================================
function defAttribute (name) {
var id = stringStore.id(name)
var result = attributeState[id]
if (!result) {
result = attributeState[id] = new AttributeStack(name)
}
return result
}
function createAttributeBox (name) {
var stack = [new AttributeRecord()]
check.saveCommandRef(stack)
function alloc (data) {
var box
if (stack.length <= 0) {
box = new AttributeRecord()
} else {
box = stack.pop()
}
if (typeof data === 'number') {
box.pointer = false
box.x = data
box.y = 0
box.z = 0
box.w = 0
} else if (Array.isArray(data)) {
box.pointer = false
box.x = data[0]
box.y = data[1]
box.z = data[2]
box.w = data[3]
} else {
var buffer = bufferState.getBuffer(data)
var size = 0
var stride = 0
var offset = 0
var divisor = 0
var normalized = false
var type = GL_FLOAT
if (!buffer) {
buffer = bufferState.getBuffer(data.buffer)
check(buffer, 'missing or invalid buffer for attribute "' +
name + '" called from command ' + box._commandRef)
size = data.size || 0
stride = data.stride || 0
offset = data.offset || 0
divisor = data.divisor || 0
normalized = data.normalized || false
type = buffer.dtype
if ('type' in data) {
type = glTypes[data.type]
}
} else {
type = buffer.dtype
}
box.pointer = true
box.buffer = buffer
box.size = size
box.offset = offset
box.stride = stride
box.divisor = divisor
box.normalized = normalized
box.type = type
}
return box
}
function free (box) {
stack.push(box)
}
return {
alloc: alloc,
free: free
}
}
return {
bindings: attributeBindings,
bind: bindAttribute,
bindRecord: bindAttributeRecord,
def: defAttribute,
box: createAttributeBox,
state: attributeState
}
}
},{"./constants/dtypes.json":5,"./util/check":21}],2:[function(require,module,exports){
// Array and element buffer creation
var check = require('./util/check')
var isTypedArray = require('./util/is-typed-array')
var isNDArrayLike = require('./util/is-ndarray')
var arrayTypes = require('./constants/arraytypes.json')
var bufferTypes = require('./constants/dtypes.json')
var values = require('./util/values')
var GL_STATIC_DRAW = 35044
var GL_BYTE = 5120
var GL_UNSIGNED_BYTE = 5121
var GL_SHORT = 5122
var GL_UNSIGNED_SHORT = 5123
var GL_INT = 5124
var GL_UNSIGNED_INT = 5125
var GL_FLOAT = 5126
var usageTypes = {
'static': 35044,
'dynamic': 35048,
'stream': 35040
}
function typedArrayCode (data) {
return arrayTypes[Object.prototype.toString.call(data)] | 0
}
function makeTypedArray (dtype, args) {
switch (dtype) {
case GL_UNSIGNED_BYTE:
return new Uint8Array(args)
case GL_UNSIGNED_SHORT:
return new Uint16Array(args)
case GL_UNSIGNED_INT:
return new Uint32Array(args)
case GL_BYTE:
return new Int8Array(args)
case GL_SHORT:
return new Int16Array(args)
case GL_INT:
return new Int32Array(args)
case GL_FLOAT:
return new Float32Array(args)
default:
return null
}
}
function flatten (result, data, dimension) {
var ptr = 0
for (var i = 0; i < data.length; ++i) {
var v = data[i]
for (var j = 0; j < dimension; ++j) {
result[ptr++] = v[j]
}
}
}
function transpose (result, data, shapeX, shapeY, strideX, strideY, offset) {
var ptr = 0
for (var i = 0; i < shapeX; ++i) {
for (var j = 0; j < shapeY; ++j) {
result[ptr++] = data[strideX * i + strideY * j + offset]
}
}
return result
}
module.exports = function wrapBufferState (gl) {
var bufferCount = 0
var bufferSet = {}
function REGLBuffer (buffer, type) {
this.id = bufferCount++
this.buffer = buffer
this.type = type
this.usage = GL_STATIC_DRAW
this.byteLength = 0
this.dimension = 1
this.data = null
this.dtype = GL_UNSIGNED_BYTE
}
REGLBuffer.prototype.bind = function () {
gl.bindBuffer(this.type, this.buffer)
}
function refresh (buffer) {
if (!gl.isBuffer(buffer.buffer)) {
buffer.buffer = gl.createBuffer()
}
buffer.bind()
gl.bufferData(buffer.type, buffer.data || buffer.byteLength, buffer.usage)
}
function destroy (buffer) {
var handle = buffer.buffer
check(handle, 'buffer must not be deleted already')
if (gl.isBuffer(handle)) {
gl.deleteBuffer(handle)
}
buffer.buffer = null
delete bufferSet[buffer.id]
}
function createBuffer (options, type, deferInit) {
var handle = gl.createBuffer()
var buffer = new REGLBuffer(handle, type)
bufferSet[buffer.id] = buffer
function reglBuffer (input) {
var options = input || {}
if (Array.isArray(options) ||
isTypedArray(options) ||
isNDArrayLike(options)) {
options = {
data: options
}
} else if (typeof options === 'number') {
options = {
length: options | 0
}
} else if (options === null || options === void 0) {
options = {}
}
check.type(
options, 'object',
'buffer arguments must be an object, a number or an array')
if ('usage' in options) {
var usage = options.usage
check.parameter(usage, usageTypes, 'invalid buffer usage')
buffer.usage = usageTypes[options.usage]
} else {
buffer.usage = GL_STATIC_DRAW
}
var dtype = 0
if ('type' in options) {
check.parameter(options.type, bufferTypes, 'invalid buffer type')
dtype = bufferTypes[options.type]
}
var dimension = (options.dimension | 0) || 1
var byteLength = 0
var data = null
if ('data' in options) {
data = options.data
if (data === null) {
byteLength = options.length | 0
} else {
if (isNDArrayLike(data)) {
var shape = data.shape
var stride = data.stride
var offset = data.offset
var shapeX = 0
var shapeY = 0
var strideX = 0
var strideY = 0
if (shape.length === 1) {
shapeX = shape[0]
shapeY = 1
strideX = stride[0]
strideY = 0
} else if (shape.length === 2) {
shapeX = shape[0]
shapeY = shape[1]
strideX = stride[0]
strideY = stride[1]
} else {
check.raise('invalid shape')
}
dtype = dtype || typedArrayCode(data) || GL_FLOAT
dimension = shapeY
data = transpose(
makeTypedArray(dtype, shapeX * shapeY),
data.data,
shapeX, shapeY,
strideX, strideY,
offset)
} else if (Array.isArray(data)) {
if (data.length > 0 && Array.isArray(data[0])) {
dimension = data[0].length
dtype = dtype || GL_FLOAT
var result = makeTypedArray(dtype, data.length * dimension)
data = flatten(result, data, dimension)
data = result
} else {
dtype = dtype || GL_FLOAT
data = makeTypedArray(dtype, data)
}
} else {
check.isTypedArray(data, 'invalid data type buffer data')
dtype = dtype || typedArrayCode(data)
}
byteLength = data.byteLength
}
} else if ('length' in options) {
byteLength = options.length | 0
check.nni(byteLength, 'buffer length must be a nonnegative integer')
}
buffer.data = data
buffer.dtype = dtype || GL_UNSIGNED_BYTE
buffer.byteLength = byteLength
buffer.dimension = dimension
refresh(buffer)
return reglBuffer
}
if (!deferInit) {
reglBuffer(options)
}
reglBuffer._reglType = 'buffer'
reglBuffer._buffer = buffer
reglBuffer.destroy = function () { destroy(buffer) }
return reglBuffer
}
return {
create: createBuffer,
clear: function () {
values(bufferSet).forEach(destroy)
},
refresh: function () {
values(bufferSet).forEach(refresh)
},
getBuffer: function (wrapper) {
if (wrapper && wrapper._buffer instanceof REGLBuffer) {
return wrapper._buffer
}
return null
}
}
}
},{"./constants/arraytypes.json":4,"./constants/dtypes.json":5,"./util/check":21,"./util/is-ndarray":25,"./util/is-typed-array":26,"./util/values":32}],3:[function(require,module,exports){
var check = require('./util/check')
var createEnvironment = require('./util/codegen')
var primTypes = require('./constants/primitives.json')
var GL_ELEMENT_ARRAY_BUFFER = 34963
var GL_FRAGMENT_SHADER = 35632
var GL_VERTEX_SHADER = 35633
var GL_FLOAT = 5126
var GL_FLOAT_VEC2 = 35664
var GL_FLOAT_VEC3 = 35665
var GL_FLOAT_VEC4 = 35666
var GL_INT = 5124
var GL_INT_VEC2 = 35667
var GL_INT_VEC3 = 35668
var GL_INT_VEC4 = 35669
var GL_BOOL = 35670
var GL_BOOL_VEC2 = 35671
var GL_BOOL_VEC3 = 35672
var GL_BOOL_VEC4 = 35673
var GL_FLOAT_MAT2 = 35674
var GL_FLOAT_MAT3 = 35675
var GL_FLOAT_MAT4 = 35676
var GL_SAMPLER_2D = 35678
var GL_SAMPLER_CUBE = 35680
var GL_TRIANGLES = 4
var GL_CULL_FACE = 0x0B44
var GL_BLEND = 0x0BE2
var GL_DITHER = 0x0BD0
var GL_STENCIL_TEST = 0x0B90
var GL_DEPTH_TEST = 0x0B71
var GL_SCISSOR_TEST = 0x0C11
var GL_POLYGON_OFFSET_FILL = 0x8037
var GL_SAMPLE_ALPHA_TO_COVERAGE = 0x809E
var GL_SAMPLE_COVERAGE = 0x80A0
var GL_FRONT = 1028
var GL_BACK = 1029
var GL_CW = 0x0900
var GL_CCW = 0x0901
var GL_MIN_EXT = 0x8007
var GL_MAX_EXT = 0x8008
var blendFuncs = {
'0': 0,
'1': 1,
'zero': 0,
'one': 1,
'src color': 768,
'one minus src color': 769,
'src alpha': 770,
'one minus src alpha': 771,
'dst color': 774,
'one minus dst color': 775,
'dst alpha': 772,
'one minus dst alpha': 773,
'constant color': 32769,
'one minus constant color': 32770,
'constant alpha': 32771,
'one minus constant alpha': 32772,
'src alpha saturate': 776
}
var compareFuncs = {
'never': 512,
'less': 513,
'<': 513,
'equal': 514,
'=': 514,
'==': 514,
'===': 514,
'lequal': 515,
'<=': 515,
'greater': 516,
'>': 516,
'notequal': 517,
'!=': 517,
'!==': 517,
'gequal': 518,
'>=': 518,
'always': 519
}
var stencilOps = {
'0': 0,
'zero': 0,
'keep': 7680,
'replace': 7681,
'increment': 7682,
'decrement': 7683,
'increment wrap': 34055,
'decrement wrap': 34056,
'invert': 5386
}
function typeLength (x) {
switch (x) {
case GL_FLOAT_VEC2:
case GL_INT_VEC2:
case GL_BOOL_VEC2:
return 2
case GL_FLOAT_VEC3:
case GL_INT_VEC3:
case GL_BOOL_VEC3:
return 3
case GL_FLOAT_VEC4:
case GL_INT_VEC4:
case GL_BOOL_VEC4:
return 4
default:
return 1
}
}
function setUniformString (gl, type, location, value) {
var infix
var separator = ','
switch (type) {
case GL_FLOAT:
infix = '1f'
break
case GL_FLOAT_VEC2:
infix = '2fv'
break
case GL_FLOAT_VEC3:
infix = '3fv'
break
case GL_FLOAT_VEC4:
infix = '4fv'
break
case GL_BOOL:
case GL_INT:
infix = '1i'
break
case GL_BOOL_VEC2:
case GL_INT_VEC2:
infix = '2iv'
break
case GL_BOOL_VEC3:
case GL_INT_VEC3:
infix = '3iv'
break
case GL_BOOL_VEC4:
case GL_INT_VEC4:
infix = '4iv'
break
case GL_FLOAT_MAT2:
infix = 'Matrix2fv'
separator = ',false,'
break
case GL_FLOAT_MAT3:
infix = 'Matrix3fv'
separator = ',false,'
break
case GL_FLOAT_MAT4:
infix = 'Matrix4fv'
separator = ',false,'
break
default:
check.raise('unsupported uniform type')
}
return gl + '.uniform' + infix + '(' + location + separator + value + ');'
}
function stackTop (x) {
return x + '[' + x + '.length-1]'
}
// Need to process framebuffer first in options list
function optionPriority (a, b) {
if (a === 'framebuffer') {
return -1
}
if (a < b) {
return -1
} else if (a > b) {
return 1
}
return 0
}
module.exports = function reglCompiler (
gl,
stringStore,
extensions,
limits,
bufferState,
elementState,
textureState,
framebufferState,
glState,
uniformState,
attributeState,
shaderState,
drawState,
frameState,
reglPoll) {
var contextState = glState.contextState
var blendEquations = {
'add': 32774,
'subtract': 32778,
'reverse subtract': 32779
}
if (extensions.ext_blend_minmax) {
blendEquations.min = GL_MIN_EXT
blendEquations.max = GL_MAX_EXT
}
var drawCallCounter = 0
// ===================================================
// ===================================================
// SHADER SINGLE DRAW OPERATION
// ===================================================
// ===================================================
function compileShaderDraw (program) {
var env = createEnvironment()
var link = env.link
var draw = env.proc('draw')
var def = draw.def
var GL = link(gl)
var PROGRAM = link(program.program)
var BIND_ATTRIBUTE = link(attributeState.bind)
var DRAW_STATE = {
count: link(drawState.count),
offset: link(drawState.offset),
instances: link(drawState.instances),
primitive: link(drawState.primitive)
}
var ELEMENT_STATE = link(elementState.elements)
var TEXTURE_UNIFORMS = []
// bind the program
draw(GL, '.useProgram(', PROGRAM, ');')
// set up attribute state
program.attributes.forEach(function (attribute) {
var STACK = link(attributeState.def(attribute.name))
draw(BIND_ATTRIBUTE, '(',
attribute.location, ',',
link(attributeState.bindings[attribute.location]), ',',
STACK, ',',
typeLength(attribute.info.type), ');')
})
// set up uniforms
program.uniforms.forEach(function (uniform) {
var LOCATION = link(uniform.location)
var STACK = link(uniformState.def(uniform.name))
var TOP = STACK + '[' + STACK + '.length-1]'
var type = uniform.info.type
if (type === GL_SAMPLER_2D || type === GL_SAMPLER_CUBE) {
var TEX_VALUE = def(TOP + '._texture')
TEXTURE_UNIFORMS.push(TEX_VALUE)
draw(setUniformString(GL, GL_INT, LOCATION, TEX_VALUE + '.bind()'))
} else {
draw(setUniformString(GL, type, LOCATION, TOP))
}
})
// unbind textures immediately
TEXTURE_UNIFORMS.forEach(function (TEX_VALUE) {
draw(TEX_VALUE, '.unbind();')
})
// Execute draw command
var CUR_PRIMITIVE = def(stackTop(DRAW_STATE.primitive))
var CUR_COUNT = def(stackTop(DRAW_STATE.count))
var CUR_OFFSET = def(stackTop(DRAW_STATE.offset))
var CUR_ELEMENTS = def(stackTop(ELEMENT_STATE))
// Only execute draw command if number elements is > 0
draw('if(', CUR_COUNT, '){')
var instancing = extensions.angle_instanced_arrays
if (instancing) {
var CUR_INSTANCES = def(stackTop(DRAW_STATE.instances))
var INSTANCE_EXT = link(instancing)
draw(
'if(', CUR_ELEMENTS, '){',
CUR_ELEMENTS, '.bind();',
'if(', CUR_INSTANCES, '>0){',
INSTANCE_EXT, '.drawElementsInstancedANGLE(',
CUR_PRIMITIVE, ',',
CUR_COUNT, ',',
CUR_ELEMENTS, '.type,',
CUR_OFFSET, ',',
CUR_INSTANCES, ');}else{',
GL, '.drawElements(',
CUR_PRIMITIVE, ',',
CUR_COUNT, ',',
CUR_ELEMENTS, '.type,',
CUR_OFFSET, ');}',
'}else if(', CUR_INSTANCES, '>0){',
INSTANCE_EXT, '.drawArraysInstancedANGLE(',
CUR_PRIMITIVE, ',',
CUR_OFFSET, ',',
CUR_COUNT, ',',
CUR_INSTANCES, ');}else{',
GL, '.drawArrays(',
CUR_PRIMITIVE, ',',
CUR_OFFSET, ',',
CUR_COUNT, ');}}')
} else {
draw(
'if(', CUR_ELEMENTS, '){',
CUR_ELEMENTS, '.bind();',
GL, '.drawElements(',
CUR_PRIMITIVE, ',',
CUR_COUNT, ',',
CUR_ELEMENTS, '.type,',
CUR_OFFSET, ');',
'}else{',
GL, '.drawArrays(',
CUR_PRIMITIVE, ',',
CUR_OFFSET, ',',
CUR_COUNT, ');}}')
}
return env.compile().draw
}
// ===================================================
// ===================================================
// BATCH DRAW OPERATION
// ===================================================
// ===================================================
function compileBatch (
program, options, uniforms, attributes, staticOptions) {
// -------------------------------
// code generation helpers
// -------------------------------
var env = createEnvironment()
var link = env.link
var batch = env.proc('batch')
var exit = env.block()
var def = batch.def
var arg = batch.arg
// -------------------------------
// regl state
// -------------------------------
var GL = link(gl)
var PROGRAM = link(program.program)
var BIND_ATTRIBUTE = link(attributeState.bind)
var BIND_ATTRIBUTE_RECORD = link(attributeState.bindRecord)
var FRAME_STATE = link(frameState)
var FRAMEBUFFER_STATE = link(framebufferState)
var DRAW_STATE = {
count: link(drawState.count),
offset: link(drawState.offset),
instances: link(drawState.instances),
primitive: link(drawState.primitive)
}
var CONTEXT_STATE = {}
var ELEMENTS = link(elementState.elements)
var CUR_COUNT = def(stackTop(DRAW_STATE.count))
var CUR_OFFSET = def(stackTop(DRAW_STATE.offset))
var CUR_PRIMITIVE = def(stackTop(DRAW_STATE.primitive))
var CUR_ELEMENTS = def(stackTop(ELEMENTS))
var CUR_INSTANCES
var INSTANCE_EXT
var instancing = extensions.angle_instanced_arrays
if (instancing) {
CUR_INSTANCES = def(stackTop(DRAW_STATE.instances))
INSTANCE_EXT = link(instancing)
}
var hasDynamicElements = 'elements' in options
function linkContext (x) {
var result = CONTEXT_STATE[x]
if (result) {
return result
}
result = CONTEXT_STATE[x] = link(contextState[x])
return result
}
// -------------------------------
// batch/argument vars
// -------------------------------
var NUM_ARGS = arg()
var ARGS = arg()
var ARG = def()
var BATCH_ID = def()
// -------------------------------
// load a dynamic variable
// -------------------------------
var dynamicVars = {}
function dyn (x) {
var id = x.id
var result = dynamicVars[id]
if (result) {
return result
}
if (x.func) {
result = batch.def(
link(x.data), '(', ARG, ',', BATCH_ID, ',', FRAME_STATE, ')')
} else {
result = batch.def(ARG, '.', x.data)
}
dynamicVars[id] = result
return result
}
// -------------------------------
// retrieves the first name-matching record from an ActiveInfo list
// -------------------------------
function findInfo (list, name) {
for (var i = 0; i < list.length; ++i) {
if (list[i].name === name) {
return list[i]
}
}
return null
}
// -------------------------------
// bind shader
// -------------------------------
batch(GL, '.useProgram(', PROGRAM, ');')
// -------------------------------
// set static uniforms
// -------------------------------
program.uniforms.forEach(function (uniform) {
if (uniform.name in uniforms) {
return
}
var LOCATION = link(uniform.location)
var STACK = link(uniformState.def(uniform.name))
var TOP = STACK + '[' + STACK + '.length-1]'
var type = uniform.info.type
if (type === GL_SAMPLER_2D || type === GL_SAMPLER_CUBE) {
var TEX_VALUE = def(TOP + '._texture')
batch(setUniformString(GL, GL_INT, LOCATION, TEX_VALUE + '.bind()'))
exit(TEX_VALUE, '.unbind();')
} else {
batch(setUniformString(GL, type, LOCATION, TOP))
}
})
// -------------------------------
// set static attributes
// -------------------------------
program.attributes.forEach(function (attribute) {
if (attribute.name in attributes) {
return
}
var STACK = link(attributeState.def(attribute.name))
batch(BIND_ATTRIBUTE, '(',
attribute.location, ',',
link(attributeState.bindings[attribute.location]), ',',
STACK, ',',
typeLength(attribute.info.type), ');')
})
// -------------------------------
// set static element buffer
// -------------------------------
if (!hasDynamicElements) {
batch(
'if(', CUR_ELEMENTS, ')',
GL, '.bindBuffer(', GL_ELEMENT_ARRAY_BUFFER, ',', CUR_ELEMENTS, '.buffer.buffer);')
}
// -------------------------------
// loop over all arguments
// -------------------------------
batch(
'for(', BATCH_ID, '=0;', BATCH_ID, '<', NUM_ARGS, ';++', BATCH_ID, '){',
ARG, '=', ARGS, '[', BATCH_ID, '];')
// -------------------------------
// set dynamic flags
// -------------------------------
Object.keys(options).sort(optionPriority).forEach(function (option) {
var VALUE = dyn(options[option])
function setCap (flag) {
batch(
'if(', VALUE, '){',
GL, '.enable(', flag, ');}else{',
GL, '.disable(', flag, ');}')
}
switch (option) {
case 'framebuffer':
var VIEWPORT_STATE = linkContext('viewport')
var SCISSOR_STATE = linkContext('scissor.box')
batch(
'if(', FRAMEBUFFER_STATE, '.push(',
VALUE, '&&', VALUE, '._framebuffer)){',
FRAMEBUFFER_STATE, '.poll();',
VIEWPORT_STATE, '.setDirty();',
SCISSOR_STATE, '.setDirty();',
'}')
break
// Caps
case 'cull.enable':
setCap(GL_CULL_FACE)
break
case 'blend.enable':
setCap(GL_BLEND)
break
case 'dither':
setCap(GL_DITHER)
break
case 'stencil.enable':
setCap(GL_STENCIL_TEST)
break
case 'depth.enable':
setCap(GL_DEPTH_TEST)
break
case 'scissor.enable':
setCap(GL_SCISSOR_TEST)
break
case 'polygonOffset.enable':
setCap(GL_POLYGON_OFFSET_FILL)
break
case 'sample.alpha':
setCap(GL_SAMPLE_ALPHA_TO_COVERAGE)
break
case 'sample.enable':
setCap(GL_SAMPLE_COVERAGE)
break
case 'depth.mask':
batch(GL, '.depthMask(', VALUE, ');')
break
case 'depth.func':
var DEPTH_FUNCS = link(compareFuncs)
batch(GL, '.depthFunc(', DEPTH_FUNCS, '[', VALUE, ']);')
break
case 'depth.range':
batch(GL, '.depthRange(', VALUE, '[0],', VALUE, '[1]);')
break
case 'blend.color':
batch(GL, '.blendColor(',
VALUE, '[0],',
VALUE, '[1],',
VALUE, '[2],',
VALUE, '[3]);')
break
case 'blend.equation':
var BLEND_EQUATIONS = link(blendEquations)
batch(
'if(typeof ', VALUE, '==="string"){',
GL, '.blendEquation(', BLEND_EQUATIONS, '[', VALUE, ']);',
'}else{',
GL, '.blendEquationSeparate(',
BLEND_EQUATIONS, '[', VALUE, '.rgb],',
BLEND_EQUATIONS, '[', VALUE, '.alpha]);',
'}')
break
case 'blend.func':
var BLEND_FUNCS = link(blendFuncs)
batch(
GL, '.blendFuncSeparate(',
BLEND_FUNCS,
'["srcRGB" in ', VALUE, '?', VALUE, '.srcRGB:', VALUE, '.src],',
BLEND_FUNCS,
'["dstRGB" in ', VALUE, '?', VALUE, '.dstRGB:', VALUE, '.dst],',
BLEND_FUNCS,
'["srcAlpha" in ', VALUE, '?', VALUE, '.srcAlpha:', VALUE, '.src],',
BLEND_FUNCS,
'["dstAlpha" in ', VALUE, '?', VALUE, '.dstAlpha:', VALUE, '.dst]);')
break
case 'stencil.mask':
batch(GL, '.stencilMask(', VALUE, ');')
break
case 'stencil.func':
var STENCIL_FUNCS = link(compareFuncs)
batch(GL, '.stencilFunc(',
STENCIL_FUNCS, '[', VALUE, '.cmp||"always"],',
VALUE, '.ref|0,',
'"mask" in ', VALUE, '?', VALUE, '.mask:-1);')
break
case 'stencil.opFront':
case 'stencil.opBack':
var STENCIL_OPS = link(stencilOps)
batch(GL, '.stencilOpSeparate(',
option === 'stencil.opFront' ? GL_FRONT : GL_BACK, ',',
STENCIL_OPS, '[', VALUE, '.fail||"keep"],',
STENCIL_OPS, '[', VALUE, '.zfail||"keep"],',
STENCIL_OPS, '[', VALUE, '.pass||"keep"]);')
break
case 'polygonOffset.offset':
batch(GL, '.polygonOffset(',
VALUE, '.factor||0,',
VALUE, '.units||0);')
break
case 'cull.face':
batch(GL, '.cullFace(',
VALUE, '==="front"?', GL_FRONT, ':', GL_BACK, ');')
break
case 'lineWidth':
batch(GL, '.lineWidth(', VALUE, ');')
break
case 'frontFace':
batch(GL, '.frontFace(',
VALUE, '==="cw"?', GL_CW, ':', GL_CCW, ');')
break
case 'colorMask':
batch(GL, '.colorMask(',
VALUE, '[0],',
VALUE, '[1],',
VALUE, '[2],',
VALUE, '[3]);')
break
case 'sample.coverage':
batch(GL, '.sampleCoverage(',
VALUE, '.value,',
VALUE, '.invert);')
break
case 'scissor.box':
case 'viewport':
var BOX_STATE = linkContext(option)
batch(BOX_STATE, '.push(',
VALUE, '.x||0,',
VALUE, '.y||0,',
VALUE, '.w||-1,',
VALUE, '.h||-1);')
break
case 'primitive':
case 'offset':
case 'count':
case 'elements':
case 'instances':
break
default:
check.raise('unsupported option for batch', option)
}
})
// update viewport/scissor box state and restore framebuffer
if ('viewport' in options || 'framebuffer' in options) {
batch(linkContext('viewport'), '.poll();')
}
if ('scissor.box' in options || 'framebuffer' in options) {
batch(linkContext('scissor.box'), '.poll();')
}
if ('framebuffer' in options) {
batch(FRAMEBUFFER_STATE, '.pop();')
}
// -------------------------------
// set dynamic uniforms
// -------------------------------
var programUniforms = program.uniforms
var DYNAMIC_TEXTURES = []
Object.keys(uniforms).forEach(function (uniform) {
var data = findInfo(programUniforms, uniform)
if (!data) {
return
}
var TYPE = data.info.type
var LOCATION = link(data.location)
var VALUE = dyn(uniforms[uniform])
if (data.info.type === GL_SAMPLER_2D ||
data.info.type === GL_SAMPLER_CUBE) {
var TEX_VALUE = def(VALUE + '._texture')
DYNAMIC_TEXTURES.push(TEX_VALUE)
batch(setUniformString(GL, GL_INT, LOCATION, TEX_VALUE + '.bind()'))
} else {
batch(setUniformString(GL, TYPE, LOCATION, VALUE))
}
})
DYNAMIC_TEXTURES.forEach(function (VALUE) {
batch(VALUE, '.unbind();')
})
// -------------------------------
// set dynamic attributes
// -------------------------------
var programAttributes = program.attributes
Object.keys(attributes).forEach(function (attribute) {
var data = findInfo(programAttributes, attribute)
if (!data) {
return
}
var BOX = link(attributeState.box(attribute))
var ATTRIB_VALUE = dyn(attributes[attribute])
var RECORD = def(BOX + '.alloc(' + ATTRIB_VALUE + ')')
batch(BIND_ATTRIBUTE_RECORD, '(',
data.location, ',',
link(attributeState.bindings[data.location]), ',',
RECORD, ',',
typeLength(data.info.type), ');')
exit(BOX, '.free(', RECORD, ');')
})
// -------------------------------
// set dynamic attributes
// -------------------------------
if (options.count) {
batch(CUR_COUNT, '=', dyn(options.count), ';')
}
if (options.offset) {
batch(CUR_OFFSET, '=', dyn(options.offset), ';')
}
if (options.primitive) {
batch(
CUR_PRIMITIVE, '=', link(primTypes), '[', dyn(options.primitive), '];')
}
if (instancing && options.instances) {
batch(CUR_INSTANCES, '=', dyn(options.instances), ';')
}
function useElementOption (x) {
return hasDynamicElements && !(x in options || x in staticOptions)
}
if (hasDynamicElements) {
var dynElements = dyn(options.elements)
batch(CUR_ELEMENTS, '=',
dynElements, '?', dynElements, '._elements:null;')
}
if (useElementOption('offset')) {
batch(CUR_OFFSET, '=0;')
}
// Emit draw command
batch('if(', CUR_ELEMENTS, '){')
if (useElementOption('count')) {
batch(CUR_COUNT, '=', CUR_ELEMENTS, '.vertCount;')
}
batch('if(', CUR_COUNT, '>0){')
if (useElementOption('primitive')) {
batch(CUR_PRIMITIVE, '=', CUR_ELEMENTS, '.primType;')
}
if (hasDynamicElements) {
batch(
GL,
'.bindBuffer(',
GL_ELEMENT_ARRAY_BUFFER, ',',
CUR_ELEMENTS, '.buffer.buffer);')
}
if (instancing) {
batch(
'if(', CUR_INSTANCES, '>0){',
INSTANCE_EXT, '.drawElementsInstancedANGLE(',
CUR_PRIMITIVE, ',',
CUR_COUNT, ',',
CUR_ELEMENTS, '.type,',
CUR_OFFSET, ',',
CUR_INSTANCES, ');}else ')
}
batch(
GL, '.drawElements(',
CUR_PRIMITIVE, ',',
CUR_COUNT, ',',
CUR_ELEMENTS, '.type,',
CUR_OFFSET, ');')
batch('}}else if(', CUR_COUNT, '>0){')
if (!useElementOption('count')) {
if (useElementOption('primitive')) {
batch(CUR_PRIMITIVE, '=', GL_TRIANGLES, ';')
}
if (instancing) {
batch(
'if(', CUR_INSTANCES, '>0){',
INSTANCE_EXT, '.drawArraysInstancedANGLE(',
CUR_PRIMITIVE, ',',
CUR_OFFSET, ',',
CUR_COUNT, ',',
CUR_INSTANCES, ');}else{')
}
batch(
GL, '.drawArrays(',
CUR_PRIMITIVE, ',',
CUR_OFFSET, ',',
CUR_COUNT, ');')
if (instancing) {
batch('}')
}
}
batch('}}', exit)
// -------------------------------
// compile and return
// -------------------------------
return env.compile().batch
}
// ===================================================
// ===================================================
// MAIN DRAW COMMAND
// ===================================================
// ===================================================
function compileCommand (
staticOptions, staticUniforms, staticAttributes,
dynamicOptions, dynamicUniforms, dynamicAttributes,
hasDynamic) {
// Create code generation environment
var env = createEnvironment()
var link = env.link
var block = env.block
var proc = env.proc
var callId = drawCallCounter++
// -------------------------------
// Common state variables
// -------------------------------
var GL_POLL = link(reglPoll)
var SHADER_STATE = link(shaderState)
var FRAMEBUFFER_STATE = link(framebufferState)
var DRAW_STATE = {
count: link(drawState.count),
offset: link(drawState.offset),
instances: link(drawState.instances),
primitive: link(drawState.primitive)
}
var ELEMENT_STATE = link(elementState.elements)
var PRIM_TYPES = link(primTypes)
var COMPARE_FUNCS = link(compareFuncs)
var STENCIL_OPS = link(stencilOps)
var CONTEXT_STATE = {}
function linkContext (x) {
var result = CONTEXT_STATE[x]
if (result) {
return result
}
result = CONTEXT_STATE[x] = link(contextState[x])
return result
}
// ==========================================================
// STATIC STATE
// ==========================================================
// Code blocks for the static sections
var entry = block()
var exit = block()
// -------------------------------
// update default context state variables
// -------------------------------
function handleStaticOption (param, value) {
var STATE_STACK = linkContext(param)
entry(STATE_STACK, '.push(', value, ');')
exit(STATE_STACK, '.pop();')
}
Object.keys(staticOptions).sort(optionPriority).forEach(function (param) {
var value = staticOptions[param]
switch (param) {
case 'frag':
case 'vert':
var shaderId = stringStore.id(value)
shaderState.shader(
param === 'frag' ? GL_FRAGMENT_SHADER : GL_VERTEX_SHADER,
shaderId)
entry(SHADER_STATE, '.', param, '.push(', shaderId, ');')
exit(SHADER_STATE, '.', param, '.pop();')
break
case 'framebuffer':
var fbo = framebufferState.getFramebuffer(value)
check(value === null || fbo, 'invalid framebuffer object')
var VIEWPORT_STATE = linkContext('viewport')
var SCISSOR_STATE = linkContext('scissor.box')
entry('if(', FRAMEBUFFER_STATE, '.push(', link(
value && value._framebuffer), ')){',
VIEWPORT_STATE, '.setDirty();',
SCISSOR_STATE, '.setDirty();',
'}')
exit('if(', FRAMEBUFFER_STATE, '.pop()){',
VIEWPORT_STATE, '.setDirty();',
SCISSOR_STATE, '.setDirty();',
'}')
break
// Update draw state
case 'count':
case 'offset':
case 'instances':
check.nni(value, param)
entry(DRAW_STATE[param], '.push(', value, ');')
exit(DRAW_STATE[param], '.pop();')
break
// Update primitive type
case 'primitive':
check.parameter(value, primTypes, 'not a valid drawing primitive')
var primType = primTypes[value]
entry(DRAW_STATE.primitive, '.push(', primType, ');')
exit(DRAW_STATE.primitive, '.pop();')
break
// Update element buffer
case 'elements':
var elements = elementState.getElements(value)
var hasPrimitive = !('primitive' in staticOptions)
var hasCount = !('count' in staticOptions)
if (elements) {
var ELEMENTS = link(elements)
entry(ELEMENT_STATE, '.push(', ELEMENTS, ');')
if (hasPrimitive) {
entry(DRAW_STATE.primitive, '.push(', ELEMENTS, '.primType);')
}
if (hasCount) {
entry(DRAW_STATE.count, '.push(', ELEMENTS, '.vertCount);')
}
} else {
entry(ELEMENT_STATE, '.push(null);')
if (hasPrimitive) {
entry(DRAW_STATE.primitive, '.push(', GL_TRIANGLES, ');')
}
if (hasCount) {
entry(DRAW_STATE.count, '.push(0);')
}
}
if (hasPrimitive) {
exit(DRAW_STATE.primitive, '.pop();')
}
if (hasCount) {
exit(DRAW_STATE.count, '.pop();')
}
if (!('offset' in staticOptions)) {
entry(DRAW_STATE.offset, '.push(0);')
exit(DRAW_STATE.offset, '.pop();')
}
exit(ELEMENT_STATE, '.pop();')
break
case 'cull.enable':
case 'blend.enable':
case 'dither':
case 'stencil.enable':
case 'depth.enable':
case 'scissor.enable':
case 'polygonOffset.enable':
case 'sample.alpha':
case 'sample.enable':
case 'depth.mask':
check.type(value, 'boolean', param)
handleStaticOption(param, value)
break
case 'depth.func':
check.parameter(value, compareFuncs, param)
handleStaticOption(param, compareFuncs[value])
break
case 'depth.range':
check(
Array.isArray(value) &&
value.length === 2 &&
value[0] <= value[1],
'depth range is 2d array')
var DEPTH_RANGE_STACK = linkContext(param)
entry(DEPTH_RANGE_STACK, '.push(', value[0], ',', value[1], ');')
exit(DEPTH_RANGE_STACK, '.pop();')
break
case 'blend.func':
var BLEND_FUNC_STACK = linkContext(param)
check.type(value, 'object', 'blend func must be an object')
var srcRGB = ('srcRGB' in value ? value.srcRGB : value.src)
var srcAlpha = ('srcAlpha' in value ? value.srcAlpha : value.src)
var dstRGB = ('dstRGB' in value ? value.dstRGB : value.dst)
var dstAlpha = ('dstAlpha' in value ? value.dstAlpha : value.dst)
check.parameter(srcRGB, blendFuncs)
check.parameter(srcAlpha, blendFuncs)
check.parameter(dstRGB, blendFuncs)
check.parameter(dstAlpha, blendFuncs)
entry(BLEND_FUNC_STACK, '.push(',
blendFuncs[srcRGB], ',',
blendFuncs[dstRGB], ',',
blendFuncs[srcAlpha], ',',
blendFuncs[dstAlpha], ');')
exit(BLEND_FUNC_STACK, '.pop();')
break
case 'blend.equation':
var BLEND_EQUATION_STACK = linkContext(param)
if (typeof value === 'string') {
check.parameter(value, blendEquations, 'invalid blend equation')
entry(BLEND_EQUATION_STACK,
'.push(',
blendEquations[value], ',',
blendEquations[value], ');')
} else if (typeof value === 'object') {
check.parameter(
value.rgb, blendEquations, 'invalid blend equation rgb')
check.parameter(
value.alpha, blendEquations, 'invalid blend equation alpha')
entry(BLEND_EQUATION_STACK,
'.push(',
blendEquations[value.rgb], ',',
blendEquations[value.alpha], ');')
} else {
check.raise('invalid blend equation')
}
exit(BLEND_EQUATION_STACK, '.pop();')
break
case 'blend.color':
check(
Array.isArray(value) &&
value.length === 4,
'blend color is a 4d array')
var BLEND_COLOR_STACK = linkContext(param)
entry(BLEND_COLOR_STACK,
'.push(',
value[0], ',',
value[1], ',',
value[2], ',',
value[3], ');')
exit(BLEND_COLOR_STACK, '.pop();')
break
case 'stencil.mask':
check.type(value, 'number', 'stencil mask must be an integer')
var STENCIL_MASK_STACK = linkContext(param)
entry(STENCIL_MASK_STACK, '.push(', value, ');')
exit(STENCIL_MASK_STACK, '.pop();')
break
case 'stencil.func':
check.type(value, 'object', 'stencil func must be an object')
var cmp = value.cmp || 'keep'
var ref = value.ref || 0
var mask = 'mask' in value ? value.mask : -1
check.parameter(cmp, compareFuncs, 'invalid stencil func cmp')
check.type(ref, 'number', 'stencil func ref')
check.type(mask, 'number', 'stencil func mask')
var STENCIL_FUNC_STACK = linkContext(param)
entry(STENCIL_FUNC_STACK, '.push(',
compareFuncs[cmp], ',',
ref, ',',
mask, ');')
exit(STENCIL_FUNC_STACK, '.pop();')
break
case 'stencil.opFront':
case 'stencil.opBack':
check.type(value, 'object', param)
var fail = value.fail || 'keep'
var zfail = value.zfail || 'keep'
var pass = value.pass || 'keep'
check.parameter(fail, stencilOps, param)
check.parameter(zfail, stencilOps, param)
check.parameter(pass, stencilOps, param)
var STENCIL_OP_STACK = linkContext(param)
entry(STENCIL_OP_STACK, '.push(',
stencilOps[fail], ',',
stencilOps[zfail], ',',
stencilOps[pass], ');')
exit(STENCIL_OP_STACK, '.pop();')
break
case 'polygonOffset.offset':
check.type(value, 'object', param)
var factor = value.factor || 0
var units = value.units || 0
check.type(factor, 'number', 'offset.factor')
check.type(units, 'number', 'offset.units')
var POLYGON_OFFSET_STACK = linkContext(param)
entry(POLYGON_OFFSET_STACK, '.push(',
factor, ',', units, ');')
exit(POLYGON_OFFSET_STACK, '.pop();')
break
case 'cull.face':
var face = 0
if (value === 'front') {
face = GL_FRONT
} else if (value === 'back') {
face = GL_BACK
}
check(!!face, 'cull.face')
var CULL_FACE_STACK = linkContext(param)
entry(CULL_FACE_STACK, '.push(', face, ');')
exit(CULL_FACE_STACK, '.pop();')
break
case 'lineWidth':
var lineWidthDims = limits.lineWidthDims
check(
typeof value === 'number' &&
value >= lineWidthDims[0] &&
value <= lineWidthDims[1],
'invalid line width, must positive number between ' +
lineWidthDims[0] + ' and ' + lineWidthDims[1])
handleStaticOption(param, value)
break
case 'frontFace':
var orientation = 0
if (value === 'cw') {
orientation = GL_CW
} else if (value === 'ccw') {
orientation = GL_CCW
}
check(!!orientation, 'frontFace')
var FRONT_FACE_STACK = linkContext(param)
entry(FRONT_FACE_STACK, '.push(', orientation, ');')
exit(FRONT_FACE_STACK, '.pop();')
break
case 'colorMask':
check(Array.isArray(value) && value.length === 4, 'color mask must be length 4 array')
var COLOR_MASK_STACK = linkContext(param)
entry(COLOR_MASK_STACK, '.push(',
value.map(function (v) { return !!v }).join(),
');')
exit(COLOR_MASK_STACK, '.pop();')
break
case 'sample.coverage':
check.type(value, 'object', param)
var sampleValue = 'value' in value ? value.value : 1
var sampleInvert = !!value.invert
check(
typeof sampleValue === 'number' &&
sampleValue >= 0 && sampleValue <= 1,
'sample value')
var SAMPLE_COVERAGE_STACK = linkContext(param)
entry(SAMPLE_COVERAGE_STACK, '.push(',
sampleValue, ',', sampleInvert, ');')
exit(SAMPLE_COVERAGE_STACK, '.pop();')
break
case 'viewport':
case 'scissor.box':
check(typeof value === 'object' && value, param + ' is an object')
var X = value.x || 0
var Y = value.y || 0
var W = -1
var H = -1
check(typeof X === 'number' && X >= 0, param + '.x must be a positive int')
check(typeof Y === 'number' && Y >= 0, param + '.y must be a positive int')
if ('w' in value) {
W = value.w
check(typeof W === 'number' && W >= 0, param + '.w must be a positive int')
}
if ('h' in value) {
H = value.h
check(typeof H === 'number' && H >= 0, param + '.h must be a positive int')
}
var BOX_STACK = linkContext(param)
entry(BOX_STACK, '.push(', X, ',', Y, ',', W, ',', H, ');')
exit(BOX_STACK, '.pop();')
break
default:
// TODO Should this just be a warning instead?
check.raise('unsupported parameter ' + param)
break
}
})
// -------------------------------
// update static uniforms
// -------------------------------
Object.keys(staticUniforms).forEach(function (uniform) {
var STACK = link(uniformState.def(uniform))
var VALUE
var value = staticUniforms[uniform]
if (typeof value === 'function' && value._reglType) {
VALUE = link(value)
} else if (Array.isArray(value)) {
VALUE = link(value.slice())
} else {
VALUE = +value
}
entry(STACK, '.push(', VALUE, ');')
exit(STACK, '.pop();')
})
// -------------------------------
// update default attributes
// -------------------------------
Object.keys(staticAttributes).forEach(function (attribute) {
var data = staticAttributes[attribute]
var ATTRIBUTE = link(attributeState.def(attribute))
var BOX = link(attributeState.box(attribute).alloc(data))
entry(ATTRIBUTE, '.records.push(', BOX, ');')
exit(ATTRIBUTE, '.records.pop();')
})
// ==========================================================
// DYNAMIC STATE (for scope and draw)
// ==========================================================
// Generated code blocks for dynamic state flags
var dynamicEntry = env.block()
var dynamicExit = env.block()
var FRAMESTATE
var DYNARGS
if (hasDynamic) {
FRAMESTATE = link(frameState)
DYNARGS = entry.def()
}
var dynamicVars = {}
function dyn (x) {
var id = x.id
var result = dynamicVars[id]
if (result) {
return result
}
if (x.func) {
result = dynamicEntry.def(
link(x.data), '(', DYNARGS, ',0,', FRAMESTATE, ')')
} else {
result = dynamicEntry.def(DYNARGS, '.', x.data)
}
dynamicVars[id] = result
return result
}
// -------------------------------
// dynamic context state variables
// -------------------------------
Object.keys(dynamicOptions).sort(optionPriority).forEach(function (param) {
// Link in dynamic variable
var variable = dyn(dynamicOptions[param])
switch (param) {
case 'framebuffer':
var VIEWPORT_STATE = linkContext('viewport')
var SCISSOR_STATE = linkContext('scissor.box')
dynamicEntry('if(',
FRAMEBUFFER_STATE, '.push(',
variable, '&&', variable, '._framebuffer)){',
VIEWPORT_STATE, '.setDirty();',
SCISSOR_STATE, '.setDirty();',
'}')
dynamicExit('if(',
FRAMEBUFFER_STATE, '.pop()){',
VIEWPORT_STATE, '.setDirty();',
SCISSOR_STATE, '.setDirty();',
'}')
break
case 'cull.enable':
case 'blend.enable':
case 'dither':
case 'stencil.enable':
case 'depth.enable':
case 'scissor.enable':
case 'polygonOffset.enable':
case 'sample.alpha':
case 'sample.enable':
case 'lineWidth':
case 'depth.mask':
var STATE_STACK = linkContext(param)
dynamicEntry(STATE_STACK, '.push(', variable, ');')
dynamicExit(STATE_STACK, '.pop();')
break
// Draw calls
case 'count':
case 'offset':
case 'instances':
var DRAW_STACK = DRAW_STATE[param]
dynamicEntry(DRAW_STACK, '.push(', variable, ');')
dynamicExit(DRAW_STACK, '.pop();')
break
case 'primitive':
var PRIM_STACK = DRAW_STATE.primitive
dynamicEntry(PRIM_STACK, '.push(', PRIM_TYPES, '[', variable, ']);')
dynamicExit(PRIM_STACK, '.pop();')
break
case 'depth.func':
var DEPTH_FUNC_STACK = linkContext(param)
dynamicEntry(DEPTH_FUNC_STACK, '.push(', COMPARE_FUNCS, '[', variable, ']);')
dynamicExit(DEPTH_FUNC_STACK, '.pop();')
break
case 'blend.func':
var BLEND_FUNC_STACK = linkContext(param)
var BLEND_FUNCS = link(blendFuncs)
dynamicEntry(
BLEND_FUNC_STACK, '.push(',
BLEND_FUNCS,
'["srcRGB" in ', variable, '?', variable, '.srcRGB:', variable, '.src],',
BLEND_FUNCS,
'["dstRGB" in ', variable, '?', variable, '.dstRGB:', variable, '.dst],',
BLEND_FUNCS,
'["srcAlpha" in ', variable, '?', variable, '.srcAlpha:', variable, '.src],',
BLEND_FUNCS,
'["dstAlpha" in ', variable, '?', variable, '.dstAlpha:', variable, '.dst]);')
dynamicExit(BLEND_FUNC_STACK, '.pop();')
break
case 'blend.equation':
var BLEND_EQUATION_STACK = linkContext(param)
var BLEND_EQUATIONS = link(blendEquations)
dynamicEntry(
'if(typeof ', variable, '==="string"){',
BLEND_EQUATION_STACK, '.push(',
BLEND_EQUATIONS, '[', variable, '],',
BLEND_EQUATIONS, '[', variable, ']);',
'}else{',
BLEND_EQUATION_STACK, '.push(',
BLEND_EQUATIONS, '[', variable, '.rgb],',
BLEND_EQUATIONS, '[', variable, '.alpha]);',
'}')
dynamicExit(BLEND_EQUATION_STACK, '.pop();')
break
case 'blend.color':
var BLEND_COLOR_STACK = linkContext(param)
dynamicEntry(BLEND_COLOR_STACK, '.push(',
variable, '[0],',
variable, '[1],',
variable, '[2],',
variable, '[3]);')
dynamicExit(BLEND_COLOR_STACK, '.pop();')
break
case 'stencil.mask':
var STENCIL_MASK_STACK = linkContext(param)
dynamicEntry(STENCIL_MASK_STACK, '.push(', variable, ');')
dynamicExit(STENCIL_MASK_STACK, '.pop();')
break
case 'stencil.func':
var STENCIL_FUNC_STACK = linkContext(param)
dynamicEntry(STENCIL_FUNC_STACK, '.push(',
COMPARE_FUNCS, '[', variable, '.cmp],',
variable, '.ref|0,',
'"mask" in ', variable, '?', variable, '.mask:-1);')
dynamicExit(STENCIL_FUNC_STACK, '.pop();')
break
case 'stencil.opFront':
case 'stencil.opBack':
var STENCIL_OP_STACK = linkContext(param)
dynamicEntry(STENCIL_OP_STACK, '.push(',
STENCIL_OPS, '[', variable, '.fail||"keep"],',
STENCIL_OPS, '[', variable, '.zfail||"keep"],',
STENCIL_OPS, '[', variable, '.pass||"keep"]);')
dynamicExit(STENCIL_OP_STACK, '.pop();')
break
case 'polygonOffset.offset':
var POLYGON_OFFSET_STACK = linkContext(param)
dynamicEntry(POLYGON_OFFSET_STACK, '.push(',
variable, '.factor||0,',
variable, '.units||0);')
dynamicExit(POLYGON_OFFSET_STACK, '.pop();')
break
case 'cull.face':
var CULL_FACE_STACK = linkContext(param)
dynamicEntry(CULL_FACE_STACK, '.push(',
variable, '==="front"?', GL_FRONT, ':', GL_BACK, ');')
dynamicExit(CULL_FACE_STACK, '.pop();')
break
case 'frontFace':
var FRONT_FACE_STACK = linkContext(param)
dynamicEntry(FRONT_FACE_STACK, '.push(',
variable, '==="cw"?', GL_CW, ':', GL_CCW, ');')
dynamicExit(FRONT_FACE_STACK, '.pop();')
break
case 'colorMask':
var COLOR_MASK_STACK = linkContext(param)
dynamicEntry(COLOR_MASK_STACK, '.push(',
variable, '[0],',
variable, '[1],',
variable, '[2],',
variable, '[3]);')
dynamicExit(COLOR_MASK_STACK, '.pop();')
break
case 'sample.coverage':
var SAMPLE_COVERAGE_STACK = linkContext(param)
dynamicEntry(SAMPLE_COVERAGE_STACK, '.push(',
variable, '.value,',
variable, '.invert);')
dynamicExit(SAMPLE_COVERAGE_STACK, '.pop();')
break
case 'scissor.box':
case 'viewport':
var BOX_STACK = linkContext(param)
dynamicEntry(BOX_STACK, '.push(',
variable, '.x||0,',
variable, '.y||0,',
'"w" in ', variable, '?', variable, '.w:-1,',
'"h" in ', variable, '?', variable, '.h:-1);')
dynamicExit(BOX_STACK, '.pop();')
break
case 'elements':
var hasPrimitive =
!('primitive' in dynamicOptions) &&
!('primitive' in staticOptions)
var hasCount =
!('count' in dynamicOptions) &&
!('count' in staticOptions)
var hasOffset =
!('offset' in dynamicOptions) &&
!('offset' in staticOptions)
var ELEMENTS = dynamicEntry.def()
dynamicEntry(
'if(', variable, '){',
ELEMENTS, '=', variable, '._elements;',
ELEMENT_STATE, '.push(', ELEMENTS, ');',
!hasPrimitive ? ''
: DRAW_STATE.primitive + '.push(' + ELEMENTS + '.primType);',
!hasCount ? ''
: DRAW_STATE.count + '.push(' + ELEMENTS + '.vertCount);',
!hasOffset ? ''
: DRAW_STATE.offset + '.push(' + ELEMENTS + '.offset);',
'}else{',
ELEMENT_STATE, '.push(null);',
'}')
dynamicExit(
ELEMENT_STATE, '.pop();',
'if(', variable, '){',
hasPrimitive ? DRAW_STATE.primitive + '.pop();' : '',
hasCount ? DRAW_STATE.count + '.pop();' : '',
hasOffset ? DRAW_STATE.offset + '.pop();' : '',
'}')
break
default:
check.raise('unsupported dynamic option: ' + param)
}
})
// -------------------------------
// dynamic uniforms
// -------------------------------
Object.keys(dynamicUniforms).forEach(function (uniform) {
var STACK = link(uniformState.def(uniform))
var VALUE = dyn(dynamicUniforms[uniform])
dynamicEntry(STACK, '.push(', VALUE, ');')
dynamicExit(STACK, '.pop();')
})
// -------------------------------
// dynamic attributes
// -------------------------------
Object.keys(dynamicAttributes).forEach(function (attribute) {
var ATTRIBUTE = link(attributeState.def(attribute))
var VALUE = dyn(dynamicAttributes[attribute])
var BOX = link(attributeState.box(attribute))
dynamicEntry(ATTRIBUTE, '.records.push(',
BOX, '.alloc(', VALUE, '));')
dynamicExit(BOX, '.free(', ATTRIBUTE, '.records.pop());')
})
// ==========================================================
// SCOPE PROCEDURE
// ==========================================================
var scope = proc('scope')
var SCOPE_ARGS = scope.arg()
var SCOPE_BODY = scope.arg()
scope(entry)
if (hasDynamic) {
scope(
DYNARGS, '=', SCOPE_ARGS, ';',
dynamicEntry)
}
scope(
SCOPE_BODY, '();',
hasDynamic ? dynamicExit : '',
exit)
// -------------------------------
// update shader program only for DRAW and batch
// -------------------------------
var commonDraw = block()
var CURRENT_PROGRAM = commonDraw.def()
if (staticOptions.frag && staticOptions.vert) {
var fragSrc = staticOptions.frag
var vertSrc = staticOptions.vert
commonDraw(CURRENT_PROGRAM, '=', link(
shaderState.program(
stringStore.id(vertSrc),
stringStore.id(fragSrc))), ';')
} else {
commonDraw(CURRENT_PROGRAM, '=',
SHADER_STATE, '.program', '(',
SHADER_STATE, '.vert[', SHADER_STATE, '.vert.length-1]', ',',
SHADER_STATE, '.frag[', SHADER_STATE, '.frag.length-1]', ');')
}
// ==========================================================
// DRAW PROCEDURE
// ==========================================================
var draw = proc('draw')
draw(entry, commonDraw)
if (hasDynamic) {
draw(
DYNARGS, '=', draw.arg(), ';',
dynamicEntry)
}
draw(
GL_POLL, '();',
'if(', CURRENT_PROGRAM, ')',
CURRENT_PROGRAM, '.draw(', hasDynamic ? DYNARGS : '', ');',
hasDynamic ? dynamicExit : '',
exit)
// ==========================================================
// BATCH DRAW
// ==========================================================
var batch = proc('batch')
batch(entry, commonDraw)
var EXEC_BATCH = link(function (program, count, args) {
var proc = program.batchCache[callId]
if (!proc) {
proc = program.batchCache[callId] = compileBatch(
program, dynamicOptions, dynamicUniforms, dynamicAttributes,
staticOptions)
}
return proc(count, args)
})
batch(
'if(', CURRENT_PROGRAM, '){',
GL_POLL, '();',
EXEC_BATCH, '(',
CURRENT_PROGRAM, ',',
batch.arg(), ',',
batch.arg(), ');')
// Set dirty on all dynamic flags
Object.keys(dynamicOptions).forEach(function (option) {
var STATE = CONTEXT_STATE[option]
if (STATE) {
batch(STATE, '.setDirty();')
}
})
batch('}', exit)
// -------------------------------
// eval and bind
// -------------------------------
return env.compile()
}
return {
draw: compileShaderDraw,
command: compileCommand
}
}
},{"./constants/primitives.json":6,"./util/check":21,"./util/codegen":23}],4:[function(require,module,exports){
module.exports={
"[object Int8Array]": 5120
, "[object Int16Array]": 5122
, "[object Int32Array]": 5124
, "[object Uint8Array]": 5121
, "[object Uint8ClampedArray]": 5121
, "[object Uint16Array]": 5123
, "[object Uint32Array]": 5125
, "[object Float32Array]": 5126
, "[object Float64Array]": 5121
, "[object ArrayBuffer]": 5121
}
},{}],5:[function(require,module,exports){
module.exports={
"int8": 5120
, "int16": 5122
, "int32": 5124
, "uint8": 5121
, "uint16": 5123
, "uint32": 5125
, "float": 5126
}
},{}],6:[function(require,module,exports){
module.exports={
"points": 0,
"lines": 1,
"line loop": 2,
"line strip": 3,
"triangles": 4,
"triangle strip": 5,
"triangle fan": 6
}
},{}],7:[function(require,module,exports){
// Context and canvas creation helper functions
/*globals HTMLElement,WebGLRenderingContext*/
var check = require('./util/check')
var extend = require('./util/extend')
function createCanvas (element, options) {
var canvas = document.createElement('canvas')
var args = getContext(canvas, options)
extend(canvas.style, {
border: 0,
margin: 0,
padding: 0,
top: 0,
left: 0
})
element.appendChild(canvas)
if (element === document.body) {
canvas.style.position = 'absolute'
extend(element.style, {
margin: 0,
padding: 0
})
}
var scale = +args.options.pixelRatio
function resize () {
var w = window.innerWidth
var h = window.innerHeight
if (element !== document.body) {
var bounds = element.getBoundingClientRect()
w = bounds.right - bounds.left
h = bounds.top - bounds.bottom
}
canvas.width = scale * w
canvas.height = scale * h
extend(canvas.style, {
width: w + 'px',
height: h + 'px'
})
}
window.addEventListener('resize', resize, false)
var prevDestroy = args.options.onDestroy
args.options = extend(extend({}, args.options), {
onDestroy: function () {
window.removeEventListener('resize', resize)
element.removeChild(canvas)
prevDestroy && prevDestroy()
}
})
resize()
return args
}
function getContext (canvas, options) {
var glOptions = options.glOptions || {}
function get (name) {
try {
return canvas.getContext(name, glOptions)
} catch (e) {
return null
}
}
var gl = get('webgl') ||
get('experimental-webgl') ||
get('webgl-experimental')
check(gl, 'webgl not supported')
return {
gl: gl,
options: extend({
pixelRatio: window.devicePixelRatio
}, options)
}
}
module.exports = function parseArgs (args) {
if (typeof document === 'undefined' ||
typeof HTMLElement === 'undefined') {
return {
gl: args[0],
options: args[1] || {}
}
}
var element = document.body
var options = args[1] || {}
if (typeof args[0] === 'string') {
element = document.querySelector(args[0]) || document.body
} else if (typeof args[0] === 'object') {
if (args[0] instanceof HTMLElement) {
element = args[0]
} else if (args[0] instanceof WebGLRenderingContext) {
return {
gl: args[0],
options: extend({
pixelRatio: 1
}, options)
}
} else {
options = args[0]
}
}
if (element.nodeName && element.nodeName.toUpperCase() === 'CANVAS') {
return getContext(element, options)
} else {
return createCanvas(element, options)
}
}
},{"./util/check":21,"./util/extend":24}],8:[function(require,module,exports){
var GL_TRIANGLES = 4
module.exports = function wrapDrawState (gl) {
var primitive = [ GL_TRIANGLES ]
var count = [ -1 ]
var offset = [ 0 ]
var instances = [ 0 ]
return {
primitive: primitive,
count: count,
offset: offset,
instances: instances
}
}
},{}],9:[function(require,module,exports){
var VARIABLE_COUNTER = 0
function DynamicVariable (isFunc, data) {
this.id = (VARIABLE_COUNTER++)
this.func = isFunc
this.data = data
}
function defineDynamic (data, path) {
switch (typeof data) {
case 'boolean':
case 'number':
case 'string':
return new DynamicVariable(false, data)
case 'function':
return new DynamicVariable(true, data)
default:
return defineDynamic
}
}
function isDynamic (x) {
return (typeof x === 'function' && !x._reglType) ||
x instanceof DynamicVariable
}
function unbox (x, path) {
if (x instanceof DynamicVariable) {
return x
} else if (typeof x === 'function' &&
x !== defineDynamic) {
return new DynamicVariable(true, x)
}
return new DynamicVariable(false, path)
}
module.exports = {
define: defineDynamic,
isDynamic: isDynamic,
unbox: unbox
}
},{}],10:[function(require,module,exports){
var check = require('./util/check')
var isTypedArray = require('./util/is-typed-array')
var isNDArrayLike = require('./util/is-ndarray')
var primTypes = require('./constants/primitives.json')
var GL_POINTS = 0
var GL_LINES = 1
var GL_TRIANGLES = 4
var GL_BYTE = 5120
var GL_UNSIGNED_BYTE = 5121
var GL_SHORT = 5122
var GL_UNSIGNED_SHORT = 5123
var GL_INT = 5124
var GL_UNSIGNED_INT = 5125
var GL_ELEMENT_ARRAY_BUFFER = 34963
module.exports = function wrapElementsState (gl, extensions, bufferState) {
var elements = [ null ]
function REGLElementBuffer () {
this.buffer = null
this.primType = GL_TRIANGLES
this.vertCount = 0
this.type = 0
}
REGLElementBuffer.prototype.bind = function () {
this.buffer.bind()
}
function createElements (options) {
var elements = new REGLElementBuffer()
var buffer = bufferState.create(null, GL_ELEMENT_ARRAY_BUFFER, true)
elements.buffer = buffer._buffer
function reglElements (input) {
var options = input
var ext32bit = extensions.oes_element_index_uint
// Upload data to vertex buffer
if (!options) {
buffer()
} else if (typeof options === 'number') {
buffer(options)
} else {
var data = null
var usage = 'static'
var byteLength = 0
if (
Array.isArray(options) ||
isTypedArray(options) ||
isNDArrayLike(options)) {
data = options
} else {
check.type(options, 'object', 'invalid arguments for elements')
if ('data' in options) {
data = options.data
}
if ('usage' in options) {
usage = options.usage
}
if ('length' in options) {
byteLength = options.length
}
}
if (Array.isArray(data) ||
(isNDArrayLike(data) && data.dtype === 'array') ||
'type' in options) {
buffer({
type: options.type ||
(ext32bit
? 'uint32'
: 'uint16'),
usage: usage,
data: data,
length: byteLength
})
} else {
buffer({
usage: usage,
data: data,
length: byteLength
})
}
if (Array.isArray(data) || isTypedArray(data)) {
buffer.dimension = 3
}
}
// try to guess default primitive type and arguments
var vertCount = elements.buffer.byteLength
var type = 0
switch (elements.buffer.dtype) {
case GL_UNSIGNED_BYTE:
case GL_BYTE:
type = GL_UNSIGNED_BYTE
break
case GL_UNSIGNED_SHORT:
case GL_SHORT:
type = GL_UNSIGNED_SHORT
vertCount >>= 1
break
case GL_UNSIGNED_INT:
case GL_INT:
check(ext32bit, '32 bit element buffers not supported')
type = GL_UNSIGNED_INT
vertCount >>= 2
break
default:
check.raise('invalid element buffer type')
}
// try to guess primitive type from cell dimension
var primType = GL_TRIANGLES
var dimension = elements.buffer.dimension
if (dimension === 1) primType = GL_POINTS
if (dimension === 2) primType = GL_LINES
if (dimension === 3) primType = GL_TRIANGLES
// if manual override present, use that
if (typeof options === 'object') {
if ('primitive' in options) {
var primitive = options.primitive
check.parameter(primitive, primTypes)
primType = primTypes[primitive]
}
if ('count' in options) {
vertCount = options.vertCount | 0
}
}
// update properties for element buffer
elements.primType = primType
elements.vertCount = vertCount
elements.type = type
return reglElements
}
reglElements(options)
reglElements._reglType = 'elements'
reglElements._elements = elements
reglElements.destroy = function () {
check(elements.buffer !== null, 'must not double destroy elements')
buffer.destroy()
elements.buffer = null
}
return reglElements
}
return {
create: createElements,
elements: elements,
getElements: function (elements) {
if (elements && elements._elements instanceof REGLElementBuffer) {
return elements._elements
}
return null
}
}
}
},{"./constants/primitives.json":6,"./util/check":21,"./util/is-ndarray":25,"./util/is-typed-array":26}],11:[function(require,module,exports){
module.exports = function createExtensionCache (gl) {
var extensions = {}
function refreshExtensions () {
[
'oes_texture_float',
'oes_texture_float_linear',
'oes_texture_half_float',
'oes_texture_half_float_linear',
'oes_standard_derivatives',
'oes_element_index_uint',
'oes_fbo_render_mipmap',
'webgl_depth_texture',
'webgl_draw_buffers',
'webgl_color_buffer_float',
'ext_texture_filter_anisotropic',
'ext_frag_depth',
'ext_blend_minmax',
'ext_shader_texture_lod',
'ext_color_buffer_half_float',
'ext_srgb',
'angle_instanced_arrays',
'webgl_compressed_texture_s3tc',
'webgl_compressed_texture_atc',
'webgl_compressed_texture_pvrtc',
'webgl_compressed_texture_etc1'
].forEach(function (ext) {
try {
extensions[ext] = gl.getExtension(ext)
} catch (e) {}
})
}
refreshExtensions()
return {
extensions: extensions,
refresh: refreshExtensions
}
}
},{}],12:[function(require,module,exports){
var check = require('./util/check')
var values = require('./util/values')
var extend = require('./util/extend')
// We store these constants so that the minifier can inline them
var GL_FRAMEBUFFER = 0x8D40
var GL_RENDERBUFFER = 0x8D41
var GL_TEXTURE_2D = 0x0DE1
var GL_TEXTURE_CUBE_MAP = 0x8513
var GL_TEXTURE_CUBE_MAP_POSITIVE_X = 0x8515
var GL_COLOR_ATTACHMENT0 = 0x8CE0
var GL_DEPTH_ATTACHMENT = 0x8D00
var GL_STENCIL_ATTACHMENT = 0x8D20
var GL_DEPTH_STENCIL_ATTACHMENT = 0x821A
var GL_UNSIGNED_BYTE = 0x1401
var GL_FLOAT = 0x1406
var GL_HALF_FLOAT_OES = 0x8D61
var GL_RGBA = 0x1908
var GL_RGBA4 = 0x8056
var GL_RGB5_A1 = 0x8057
var GL_RGB565 = 0x8D62
var GL_DEPTH_COMPONENT16 = 0x81A5
var GL_STENCIL_INDEX8 = 0x8D48
var GL_DEPTH_COMPONENT = 0x1902
var GL_DEPTH_STENCIL = 0x84F9
var GL_SRGB8_ALPHA8_EXT = 0x8C43
var GL_RGBA32F_EXT = 0x8814
var GL_RGBA16F_EXT = 0x881A
var GL_RGB16F_EXT = 0x881B
var GL_FRAMEBUFFER_COMPLETE = 0x8CD5
var GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT = 0x8CD6
var GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT = 0x8CD7
var GL_FRAMEBUFFER_INCOMPLETE_DIMENSIONS = 0x8CD9
var GL_FRAMEBUFFER_UNSUPPORTED = 0x8CDD
var GL_BACK = 1029
var BACK_BUFFER = [GL_BACK]
module.exports = function wrapFBOState (
gl,
extensions,
limits,
textureState,
renderbufferState) {
var statusCode = {}
statusCode[GL_FRAMEBUFFER_COMPLETE] = 'complete'
statusCode[GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT] = 'incomplete attachment'
statusCode[GL_FRAMEBUFFER_INCOMPLETE_DIMENSIONS] = 'incomplete dimensions'
statusCode[GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT] = 'incomplete, missing attachment'
statusCode[GL_FRAMEBUFFER_UNSUPPORTED] = 'unsupported'
var colorTextureFormats = {
'rgba': GL_RGBA
}
var colorRenderbufferFormats = {
'rgba4': GL_RGBA4,
'rgb565': GL_RGB565,
'rgb5 a1': GL_RGB5_A1
}
if (extensions.ext_srgb) {
colorRenderbufferFormats['srgba'] = GL_SRGB8_ALPHA8_EXT
}
if (extensions.ext_color_buffer_half_float) {
colorRenderbufferFormats['rgba16f'] = GL_RGBA16F_EXT
colorRenderbufferFormats['rgb16f'] = GL_RGB16F_EXT
}
if (extensions.webgl_color_buffer_float) {
colorRenderbufferFormats['rgba32f'] = GL_RGBA32F_EXT
}
var depthRenderbufferFormatEnums = [GL_DEPTH_COMPONENT16]
var stencilRenderbufferFormatEnums = [GL_STENCIL_INDEX8]
var depthStencilRenderbufferFormatEnums = [GL_DEPTH_STENCIL]
var depthTextureFormatEnums = []
var stencilTextureFormatEnums = []
var depthStencilTextureFormatEnums = []
if (extensions.webgl_depth_texture) {
depthTextureFormatEnums.push(GL_DEPTH_COMPONENT)
depthStencilTextureFormatEnums.push(GL_DEPTH_STENCIL)
}
var colorFormats = extend(extend({},
colorTextureFormats),
colorRenderbufferFormats)
var colorTextureFormatEnums = values(colorTextureFormats)
var colorRenderbufferFormatEnums = values(colorRenderbufferFormats)
var highestPrecision = GL_UNSIGNED_BYTE
var colorTypes = {
'uint8': GL_UNSIGNED_BYTE
}
if (extensions.oes_texture_half_float) {
highestPrecision = colorTypes['half float'] = GL_HALF_FLOAT_OES
}
if (extensions.oes_texture_float) {
highestPrecision = colorTypes.float = GL_FLOAT
}
colorTypes.best = highestPrecision
var DRAW_BUFFERS = (function () {
var result = new Array(limits.maxDrawbuffers)
for (var i = 0; i <= limits.maxDrawbuffers; ++i) {
var row = result[i] = new Array(i)
for (var j = 0; j < i; ++j) {
row[j] = GL_COLOR_ATTACHMENT0 + j
}
}
return result
})()
function FramebufferAttachment (target, level, texture, renderbuffer) {
this.target = target
this.level = level
this.texture = texture
this.renderbuffer = renderbuffer
}
function decRef (attachment) {
if (attachment) {
if (attachment.texture) {
attachment.texture._texture.decRef()
}
if (attachment.renderbuffer) {
attachment.renderbuffer._renderbuffer.decRef()
}
}
}
function checkFormat (attachment, texFormats, rbFormats) {
if (attachment.texture) {
check.oneOf(attachment.texture._texture.params.internalformat, texFormats,
'unsupported texture format for attachment')
} else {
check.oneOf(attachment.renderbuffer._renderbuffer.format, rbFormats,
'unsupported renderbuffer format for attachment')
}
}
function incRefAndCheckShape (attachment, framebuffer) {
var width = framebuffer.width
var height = framebuffer.height
if (attachment.texture) {
var texture = attachment.texture._texture
var tw = Math.max(1, texture.params.width >> attachment.level)
var th = Math.max(1, texture.params.height >> attachment.level)
width = width || tw
height = height || th
check(tw === width && th === height,
'inconsistent width/height for supplied texture')
check(texture.pollId < 0,
'polling fbo textures not supported')
texture.refCount += 1
} else {
var renderbuffer = attachment.renderbuffer._renderbuffer
width = width || renderbuffer.width
height = height || renderbuffer.height
check(
renderbuffer.width === width && renderbuffer.height === height,
'inconsistent width/height for renderbuffer')
check(
colorRenderbufferFormatEnums.indexOf(renderbuffer.format) >= 0,
'renderbuffer format not compatible with color channels')
renderbuffer.refCount += 1
}
framebuffer.width = width
framebuffer.height = height
}
function attach (location, attachment) {
if (attachment) {
if (attachment.texture) {
gl.framebufferTexture2D(
GL_FRAMEBUFFER,
location,
attachment.target,
attachment.texture._texture.texture,
attachment.level)
} else {
gl.framebufferRenderbuffer(
GL_FRAMEBUFFER,
location,
GL_RENDERBUFFER,
attachment.renderbuffer._renderbuffer.renderbuffer)
}
} else {
gl.framebufferTexture2D(
GL_FRAMEBUFFER,
location,
GL_TEXTURE_2D,
null,
0)
}
}
function tryUpdateAttachment (
attachment,
isTexture,
format,
type,
width,
height) {
if (attachment.texture) {
var texture = attachment.texture
if (isTexture) {
texture({
format: format,
type: type,
width: width,
height: height
})
texture._texture.refCount += 1
return true
}
} else {
var renderbuffer = attachment.renderbuffer
if (!isTexture) {
renderbuffer({
format: format,
width: width,
height: height
})
renderbuffer._renderbuffer.refCount += 1
return true
}
}
decRef(attachment)
return false
}
function parseAttachment (attachment) {
var target = GL_TEXTURE_2D
var level = 0
var texture = null
var renderbuffer = null
var data = attachment
if (typeof attachment === 'object') {
data = attachment.data
if ('level' in attachment) {
level = attachment.level | 0
}
if ('target' in attachment) {
target = attachment.target | 0
}
}
check.type(data, 'function', 'invalid attachment data')
var type = attachment._reglType
if (type === 'texture') {
texture = attachment
if (texture._texture.target === GL_TEXTURE_CUBE_MAP) {
check(
target >= GL_TEXTURE_CUBE_MAP_POSITIVE_X &&
target < GL_TEXTURE_CUBE_MAP_POSITIVE_X + 6,
'invalid cube map target')
} else {
check(target === GL_TEXTURE_2D)
}
// TODO check miplevel is consistent
} else if (type === 'renderbuffer') {
renderbuffer = attachment
target = GL_RENDERBUFFER
level = 0
} else {
check.raise('invalid regl object for attachment')
}
return new FramebufferAttachment(target, level, texture, renderbuffer)
}
function unwrapAttachment (attachment) {
return attachment && (attachment.texture || attachment.renderbuffer)
}
var framebufferCount = 0
var framebufferSet = {}
var framebufferStack = [null]
var framebufferDirty = true
function REGLFramebuffer () {
this.id = framebufferCount++
framebufferSet[this.id] = this
this.framebuffer = null
this.width = 0
this.height = 0
this.colorAttachments = []
this.depthAttachment = null
this.stencilAttachment = null
this.depthStencilAttachment = null
this.ownsColor = false
this.ownsDepthStencil = false
}
function refresh (framebuffer) {
if (!gl.isFramebuffer(framebuffer.framebuffer)) {
framebuffer.framebuffer = gl.createFramebuffer()
}
framebufferDirty = true
gl.bindFramebuffer(GL_FRAMEBUFFER, framebuffer.framebuffer)
var colorAttachments = framebuffer.colorAttachments
for (var i = 0; i < colorAttachments.length; ++i) {
attach(GL_COLOR_ATTACHMENT0 + i, colorAttachments[i])
}
for (i = colorAttachments.length; i < limits.maxColorAttachments; ++i) {
attach(GL_COLOR_ATTACHMENT0 + i, null)
}
attach(GL_DEPTH_ATTACHMENT, framebuffer.depthAttachment)
attach(GL_STENCIL_ATTACHMENT, framebuffer.stencilAttachment)
attach(GL_DEPTH_STENCIL_ATTACHMENT, framebuffer.depthStencilAttachment)
if (extensions.webgl_draw_buffers) {
extensions.webgl_draw_buffers.drawBuffersWEBGL(
DRAW_BUFFERS[colorAttachments.length])
}
// Check status code
var status = gl.checkFramebufferStatus(GL_FRAMEBUFFER)
if (status !== GL_FRAMEBUFFER_COMPLETE) {
check.raise('framebuffer configuration not supported, status = ' +
statusCode[status])
}
}
function decFBORefs (framebuffer) {
framebuffer.colorAttachments.forEach(decRef)
decRef(framebuffer.depthAttachment)
decRef(framebuffer.stencilAttachment)
decRef(framebuffer.depthStencilAttachment)
}
function destroy (framebuffer) {
var handle = framebuffer.framebuffer
check(handle, 'must not double destroy framebuffer')
if (gl.isFramebuffer(handle)) {
gl.deleteFramebuffer(handle)
}
}
function createFBO (options) {
var framebuffer = new REGLFramebuffer()
function reglFramebuffer (input) {
var i
var options = input || {}
var extDrawBuffers = extensions.webgl_draw_buffers
var width = 0
var height = 0
if ('shape' in options) {
var shape = options.shape
check(Array.isArray(shape) && shape.length >= 2,
'invalid shape for framebuffer')
width = shape[0]
height = shape[1]
} else {
if ('radius' in options) {
width = height = options.radius
}
if ('width' in options) {
width = options.width
}
if ('height' in options) {
height = options.height
}
}
// colorType, numColors
var colorBuffers = null
var ownsColor = false
if ('colorBuffers' in options || 'colorBuffer' in options) {
var colorInputs = options.colorBuffers || options.colorBuffer
if (!Array.isArray(colorInputs)) {
colorInputs = [colorInputs]
}
framebuffer.width = width
framebuffer.height = height
if (colorInputs.length > 1) {
check(extDrawBuffers, 'multiple render targets not supported')
}
check(colorInputs.length >= 0,
'must specify at least one color attachment')
// Wrap color attachments
colorBuffers = colorInputs.map(parseAttachment)
// Check head node
for (i = 0; i < colorBuffers.length; ++i) {
var colorAttachment = colorBuffers[i]
checkFormat(
colorAttachment,
colorTextureFormatEnums,
colorRenderbufferFormatEnums)
incRefAndCheckShape(
colorAttachment,
framebuffer)
}
width = framebuffer.width
height = framebuffer.height
} else {
var colorTexture = true
var colorFormat = 'rgba'
var colorType = 'uint8'
var colorCount = 1
ownsColor = true
framebuffer.width = width = width || gl.drawingBufferWidth
framebuffer.height = height = height || gl.drawingBufferHeight
if ('format' in options) {
colorFormat = options.format
check.parameter(colorFormat, colorFormats, 'invalid color format')
colorTexture = colorFormat in colorTextureFormats
}
if ('type' in options) {
check(colorTexture,
'colorType can not be set for renderbuffer targets')
colorType = options.type
check.parameter(colorType, colorTypes, 'invalid color type')
}
if ('colorCount' in options) {
colorCount = options.colorCount | 0
check(colorCount >= 0, 'color count must be positive')
}
// Reuse color buffer array if we own it
if (framebuffer.ownsColor) {
colorBuffers = framebuffer.colorAttachments
while (colorBuffers.length > colorCount) {
decRef(colorBuffers.pop())
}
} else {
colorBuffers = []
}
// update buffers in place, remove incompatible buffers
for (i = 0; i < colorBuffers.length; ++i) {
if (!tryUpdateAttachment(
colorBuffers[i],
colorTexture,
colorFormat,
colorType,
width,
height)) {
colorBuffers[i--] = colorBuffers[colorBuffers.length - 1]
colorBuffers.pop()
}
}
// Then append new buffers
while (colorBuffers.length < colorCount) {
if (colorTexture) {
colorBuffers.push(new FramebufferAttachment(
GL_TEXTURE_2D,
0,
textureState.create({
format: colorFormat,
type: colorType,
width: width,
height: height
}, GL_TEXTURE_2D),
null))
} else {
colorBuffers.push(new FramebufferAttachment(
GL_RENDERBUFFER,
0,
null,
renderbufferState.create({
format: colorFormat,
width: width,
height: height
})))
}
}
}
check(colorBuffers.length > 0, 'must specify at least one color buffer')
framebuffer.width = width
framebuffer.height = height
var depthBuffer = null
var stencilBuffer = null
var depthStencilBuffer = null
var ownsDepthStencil = false
var depthStencilCount = 0
if ('depthBuffer' in options) {
depthBuffer = parseAttachment(options.depthBuffer)
checkFormat(
depthBuffer,
depthTextureFormatEnums,
depthRenderbufferFormatEnums)
depthStencilCount += 1
}
if ('stencilBuffer' in options) {
stencilBuffer = parseAttachment(options.stencilBuffer)
checkFormat(
stencilBuffer,
stencilTextureFormatEnums,
stencilRenderbufferFormatEnums)
depthStencilCount += 1
}
if ('depthStencilBuffer' in options) {
depthStencilBuffer = parseAttachment(options.depthStencilBuffer)
checkFormat(
depthStencilBuffer,
depthStencilTextureFormatEnums,
depthStencilRenderbufferFormatEnums)
depthStencilCount += 1
}
if (!(depthBuffer || stencilBuffer || depthStencilBuffer)) {
var depth = true
var stencil = false
var useTexture = false
if ('depth' in options) {
depth = !!options.depth
}
if ('stencil' in options) {
stencil = !!options.stencil
}
if ('depthTexture' in options) {
useTexture = !!options.depthTexture
}
var curDepthStencil =
framebuffer.depthAttachment ||
framebuffer.stencilAttachment ||
framebuffer.depthStencilAttachment
var nextDepthStencil = null
if (depth || stencil) {
ownsDepthStencil = true
if (useTexture) {
check(extensions.webgl_depth_texture,
'depth texture extension not supported')
var depthTextureFormat
check(depth, 'stencil only textures not supported')
if (stencil) {
depthTextureFormat = 'depth stencil'
} else {
depthTextureFormat = 'depth'
}
if (framebuffer.ownsDepthStencil && curDepthStencil.texture) {
curDepthStencil.texture({
format: depthTextureFormat,
width: width,
height: height
})
curDepthStencil.texture._texture.refCount += 1
nextDepthStencil = curDepthStencil
} else {
nextDepthStencil = new FramebufferAttachment(
GL_TEXTURE_2D,
0,
textureState.create({
format: depthTextureFormat,
width: width,
height: height
}, GL_TEXTURE_2D),
null)
}
} else {
var depthRenderbufferFormat
if (depth) {
if (stencil) {
depthRenderbufferFormat = 'depth stencil'
} else {
depthRenderbufferFormat = 'depth'
}
} else {
depthRenderbufferFormat = 'stencil'
}
if (framebuffer.ownsDepthStencil && curDepthStencil.renderbuffer) {
curDepthStencil.renderbuffer({
format: depthRenderbufferFormat,
width: width,
height: height
})
curDepthStencil.renderbuffer._renderbuffer.refCount += 1
nextDepthStencil = curDepthStencil
} else {
nextDepthStencil = new FramebufferAttachment(
GL_RENDERBUFFER,
0,
null,
renderbufferState.create({
format: depthRenderbufferFormat,
width: width,
height: height
}))
}
}
if (depth) {
if (stencil) {
depthStencilBuffer = nextDepthStencil
} else {
depthBuffer = nextDepthStencil
}
} else {
stencilBuffer = nextDepthStencil
}
}
} else {
check(depthStencilCount === 1,
'can specify only one of depth, stencil or depthStencil attachment')
incRefAndCheckShape(
depthBuffer ||
stencilBuffer ||
depthStencilBuffer,
framebuffer)
}
decFBORefs(framebuffer)
framebuffer.colorAttachments = colorBuffers
framebuffer.depthAttachment = depthBuffer
framebuffer.stencilAttachment = stencilBuffer
framebuffer.depthStencilAttachment = depthStencilBuffer
framebuffer.ownsColor = ownsColor
framebuffer.ownsDepthStencil = ownsDepthStencil
reglFramebuffer.color = colorBuffers.map(unwrapAttachment)
reglFramebuffer.depth = unwrapAttachment(depthBuffer)
reglFramebuffer.stencil = unwrapAttachment(stencilBuffer)
reglFramebuffer.depthStencil = unwrapAttachment(depthStencilBuffer)
refresh(framebuffer)
reglFramebuffer.width = framebuffer.width
reglFramebuffer.height = framebuffer.height
return reglFramebuffer
}
reglFramebuffer(options)
reglFramebuffer._reglType = 'framebuffer'
reglFramebuffer._framebuffer = framebuffer
reglFramebuffer._destroy = function () {
destroy(framebuffer)
}
return reglFramebuffer
}
function refreshCache () {
values(framebufferSet).forEach(refresh)
}
function clearCache () {
values(framebufferSet).forEach(destroy)
}
function poll () {
if (framebufferDirty) {
var top = framebufferStack[framebufferStack.length - 1]
var ext_drawbuffers = extensions.webgl_draw_buffers
if (top) {
gl.bindFramebuffer(GL_FRAMEBUFFER, top.framebuffer)
if (ext_drawbuffers) {
ext_drawbuffers.drawBuffersWEBGL(DRAW_BUFFERS[top.colorAttachments.length])
}
} else {
gl.bindFramebuffer(GL_FRAMEBUFFER, null)
if (ext_drawbuffers) {
ext_drawbuffers.drawBuffersWEBGL(BACK_BUFFER)
}
}
framebufferDirty = false
}
}
function currentFramebuffer () {
return framebufferStack[framebufferStack.length - 1]
}
return {
top: currentFramebuffer,
dirty: function () {
return framebufferDirty
},
push: function (next_) {
var next = next_ || null
framebufferDirty = framebufferDirty || (next !== currentFramebuffer())
framebufferStack.push(next)
return framebufferDirty
},
pop: function () {
var prev = currentFramebuffer()
framebufferStack.pop()
framebufferDirty = framebufferDirty || (prev !== currentFramebuffer())
return framebufferDirty
},
getFramebuffer: function (object) {
if (typeof object === 'function' && object._reglType === 'framebuffer') {
var fbo = object._framebuffer
if (fbo instanceof REGLFramebuffer) {
return fbo
}
}
return null
},
poll: poll,
create: createFBO,
clear: clearCache,
refresh: refreshCache
}
}
},{"./util/check":21,"./util/extend":24,"./util/values":32}],13:[function(require,module,exports){
var GL_SUBPIXEL_BITS = 0x0D50
var GL_RED_BITS = 0x0D52
var GL_GREEN_BITS = 0x0D53
var GL_BLUE_BITS = 0x0D54
var GL_ALPHA_BITS = 0x0D55
var GL_DEPTH_BITS = 0x0D56
var GL_STENCIL_BITS = 0x0D57
var GL_ALIASED_POINT_SIZE_RANGE = 0x846D
var GL_ALIASED_LINE_WIDTH_RANGE = 0x846E
var GL_MAX_TEXTURE_SIZE = 0x0D33
var GL_MAX_VIEWPORT_DIMS = 0x0D3A
var GL_MAX_VERTEX_ATTRIBS = 0x8869
var GL_MAX_VERTEX_UNIFORM_VECTORS = 0x8DFB
var GL_MAX_VARYING_VECTORS = 0x8DFC
var GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS = 0x8B4D
var GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS = 0x8B4C
var GL_MAX_TEXTURE_IMAGE_UNITS = 0x8872
var GL_MAX_FRAGMENT_UNIFORM_VECTORS = 0x8DFD
var GL_MAX_CUBE_MAP_TEXTURE_SIZE = 0x851C
var GL_MAX_RENDERBUFFER_SIZE = 0x84E8
var GL_VENDOR = 0x1F00
var GL_RENDERER = 0x1F01
var GL_VERSION = 0x1F02
var GL_SHADING_LANGUAGE_VERSION = 0x8B8C
var GL_MAX_TEXTURE_MAX_ANISOTROPY_EXT = 0x84FF
var GL_MAX_COLOR_ATTACHMENTS_WEBGL = 0x8CDF
var GL_MAX_DRAW_BUFFERS_WEBGL = 0x8824
module.exports = function (gl, extensions) {
var maxAnisotropic = 1
if (extensions.ext_texture_filter_anisotropic) {
maxAnisotropic = gl.getParameter(GL_MAX_TEXTURE_MAX_ANISOTROPY_EXT)
}
var maxDrawbuffers = 1
var maxColorAttachments = 1
if (extensions.webgl_draw_buffers) {
maxDrawbuffers = gl.getParameter(GL_MAX_DRAW_BUFFERS_WEBGL)
maxColorAttachments = gl.getParameter(GL_MAX_COLOR_ATTACHMENTS_WEBGL)
}
return {
// drawing buffer bit depth
colorBits: [
gl.getParameter(GL_RED_BITS),
gl.getParameter(GL_GREEN_BITS),
gl.getParameter(GL_BLUE_BITS),
gl.getParameter(GL_ALPHA_BITS)
],
depthBits: gl.getParameter(GL_DEPTH_BITS),
stencilBits: gl.getParameter(GL_STENCIL_BITS),
subpixelBits: gl.getParameter(GL_SUBPIXEL_BITS),
// supported extensions
extensions: Object.keys(extensions).filter(function (ext) {
return !!extensions[ext]
}),
// max aniso samples
maxAnisotropic: maxAnisotropic,
// max draw buffers
maxDrawbuffers: maxDrawbuffers,
maxColorAttachments: maxColorAttachments,
// point and line size ranges
pointSizeDims: gl.getParameter(GL_ALIASED_POINT_SIZE_RANGE),
lineWidthDims: gl.getParameter(GL_ALIASED_LINE_WIDTH_RANGE),
maxViewportDims: gl.getParameter(GL_MAX_VIEWPORT_DIMS),
maxCombinedTextureUnits: gl.getParameter(GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS),
maxCubeMapSize: gl.getParameter(GL_MAX_CUBE_MAP_TEXTURE_SIZE),
maxRenderbufferSize: gl.getParameter(GL_MAX_RENDERBUFFER_SIZE),
maxTextureUnits: gl.getParameter(GL_MAX_TEXTURE_IMAGE_UNITS),
maxTextureSize: gl.getParameter(GL_MAX_TEXTURE_SIZE),
maxAttributes: gl.getParameter(GL_MAX_VERTEX_ATTRIBS),
maxVertexUniforms: gl.getParameter(GL_MAX_VERTEX_UNIFORM_VECTORS),
maxVertexTextureUnits: gl.getParameter(GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS),
maxVaryingVectors: gl.getParameter(GL_MAX_VARYING_VECTORS),
maxFragmentUniforms: gl.getParameter(GL_MAX_FRAGMENT_UNIFORM_VECTORS),
// vendor info
glsl: gl.getParameter(GL_SHADING_LANGUAGE_VERSION),
renderer: gl.getParameter(GL_RENDERER),
vendor: gl.getParameter(GL_VENDOR),
version: gl.getParameter(GL_VERSION)
}
}
},{}],14:[function(require,module,exports){
var check = require('./util/check')
var isTypedArray = require('./util/is-typed-array')
var GL_RGBA = 6408
var GL_UNSIGNED_BYTE = 5121
var GL_PACK_ALIGNMENT = 0x0D05
module.exports = function wrapReadPixels (gl, reglPoll, viewportState) {
function readPixels (input) {
var options = input || {}
if (isTypedArray(input)) {
options = {
data: options
}
} else if (arguments.length === 2) {
options = {
width: arguments[0] | 0,
height: arguments[1] | 0
}
} else if (typeof input !== 'object') {
options = {}
}
// Update WebGL state
reglPoll()
// Read viewport state
var x = options.x || 0
var y = options.y || 0
var width = options.width || viewportState.width
var height = options.height || viewportState.height
// Compute size
var size = width * height * 4
// Allocate data
var data = options.data || new Uint8Array(size)
// Type check
check.isTypedArray(data)
check(data.byteLength >= size, 'data buffer too small')
// Run read pixels
gl.pixelStorei(GL_PACK_ALIGNMENT, 4)
gl.readPixels(x, y, width, height, GL_RGBA, GL_UNSIGNED_BYTE, data)
return data
}
return readPixels
}
},{"./util/check":21,"./util/is-typed-array":26}],15:[function(require,module,exports){
var check = require('./util/check')
var values = require('./util/values')
var GL_RENDERBUFFER = 0x8D41
var GL_RGBA4 = 0x8056
var GL_RGB5_A1 = 0x8057
var GL_RGB565 = 0x8D62
var GL_DEPTH_COMPONENT16 = 0x81A5
var GL_STENCIL_INDEX8 = 0x8D48
var GL_DEPTH_STENCIL = 0x84F9
var GL_SRGB8_ALPHA8_EXT = 0x8C43
var GL_RGBA32F_EXT = 0x8814
var GL_RGBA16F_EXT = 0x881A
var GL_RGB16F_EXT = 0x881B
module.exports = function (gl, extensions, limits) {
var formatTypes = {
'rgba4': GL_RGBA4,
'rgb565': GL_RGB565,
'rgb5 a1': GL_RGB5_A1,
'depth': GL_DEPTH_COMPONENT16,
'stencil': GL_STENCIL_INDEX8,
'depth stencil': GL_DEPTH_STENCIL
}
if (extensions.ext_srgb) {
formatTypes['srgba'] = GL_SRGB8_ALPHA8_EXT
}
if (extensions.ext_color_buffer_half_float) {
formatTypes['rgba16f'] = GL_RGBA16F_EXT
formatTypes['rgb16f'] = GL_RGB16F_EXT
}
if (extensions.webgl_color_buffer_float) {
formatTypes['rgba32f'] = GL_RGBA32F_EXT
}
var renderbufferCount = 0
var renderbufferSet = {}
function REGLRenderbuffer () {
this.id = renderbufferCount++
this.refCount = 1
this.renderbuffer = null
this.format = GL_RGBA4
this.width = 0
this.height = 0
}
REGLRenderbuffer.prototype.decRef = function () {
if (--this.refCount === 0) {
destroy(this)
}
}
function refresh (rb) {
if (!gl.isRenderbuffer(rb.renderbuffer)) {
rb.renderbuffer = gl.createRenderbuffer()
}
gl.bindRenderbuffer(GL_RENDERBUFFER, rb.renderbuffer)
gl.renderbufferStorage(
GL_RENDERBUFFER,
rb.format,
rb.width,
rb.height)
}
function destroy (rb) {
var handle = rb.renderbuffer
check(handle, 'must not double destroy renderbuffer')
gl.bindRenderbuffer(GL_RENDERBUFFER, null)
if (gl.isRenderbuffer(handle)) {
gl.deleteRenderbuffer(handle)
}
rb.renderbuffer = null
rb.refCount = 0
delete renderbufferSet[rb.id]
}
function createRenderbuffer (input) {
var renderbuffer = new REGLRenderbuffer()
renderbufferSet[renderbuffer.id] = renderbuffer
function reglRenderbuffer (input) {
var options = input || {}
var w = 0
var h = 0
if ('shape' in options) {
var shape = options.shape
check(Array.isArray(shape) && shape.length >= 2,
'invalid renderbuffer shape')
w = shape[0] | 0
h = shape[1] | 0
} else {
if ('radius' in options) {
w = h = options.radius | 0
}
if ('width' in options) {
w = options.width | 0
}
if ('height' in options) {
h = options.height | 0
}
}
var s = limits.maxRenderbufferSize
check(w >= 0 && h >= 0 && w <= s && h <= s,
'invalid renderbuffer size')
reglRenderbuffer.width = renderbuffer.width = Math.max(w, 1)
reglRenderbuffer.height = renderbuffer.height = Math.max(h, 1)
renderbuffer.format = GL_RGBA4
if ('format' in options) {
var format = options.format
check.parameter(format, formatTypes, 'invalid render buffer format')
renderbuffer.format = formatTypes[format]
}
refresh(renderbuffer)
return reglRenderbuffer
}
reglRenderbuffer(input)
reglRenderbuffer._reglType = 'renderbuffer'
reglRenderbuffer._renderbuffer = renderbuffer
reglRenderbuffer.destroy = function () {
renderbuffer.decRef()
}
return reglRenderbuffer
}
function refreshRenderbuffers () {
values(renderbufferSet).forEach(refresh)
}
function destroyRenderbuffers () {
values(renderbufferSet).forEach(destroy)
}
return {
create: createRenderbuffer,
refresh: refreshRenderbuffers,
clear: destroyRenderbuffers
}
}
},{"./util/check":21,"./util/values":32}],16:[function(require,module,exports){
var check = require('./util/check')
var values = require('./util/values')
var GL_FRAGMENT_SHADER = 35632
var GL_VERTEX_SHADER = 35633
var GL_ACTIVE_UNIFORMS = 0x8B86
var GL_ACTIVE_ATTRIBUTES = 0x8B89
function ActiveInfo (name, id, location, info) {
this.name = name
this.id = id
this.location = location
this.info = info
}
module.exports = function wrapShaderState (
gl,
attributeState,
uniformState,
compileShaderDraw,
stringStore) {
// ===================================================
// glsl compilation and linking
// ===================================================
var fragShaders = {}
var vertShaders = {}
function getShader (type, id) {
var cache = type === GL_FRAGMENT_SHADER ? fragShaders : vertShaders
var shader = cache[id]
if (!shader) {
var source = stringStore.str(id)
shader = gl.createShader(type)
gl.shaderSource(shader, source)
gl.compileShader(shader)
check.shaderError(gl, shader, source, type)
cache[id] = shader
}
return shader
}
// ===================================================
// program linking
// ===================================================
var programCache = {}
var programList = []
function REGLProgram (fragId, vertId) {
this.fragId = fragId
this.vertId = vertId
this.program = null
this.uniforms = []
this.attributes = []
this.draw = function () {}
this.batchCache = {}
}
function linkProgram (desc) {
var i, info
// -------------------------------
// compile & link
// -------------------------------
var fragShader = getShader(GL_FRAGMENT_SHADER, desc.fragId)
var vertShader = getShader(GL_VERTEX_SHADER, desc.vertId)
var program = desc.program = gl.createProgram()
gl.attachShader(program, fragShader)
gl.attachShader(program, vertShader)
gl.linkProgram(program)
check.linkError(
gl,
program,
stringStore.str(desc.fragId),
stringStore.str(desc.vertId))
// -------------------------------
// grab uniforms
// -------------------------------
var numUniforms = gl.getProgramParameter(program, GL_ACTIVE_UNIFORMS)
var uniforms = desc.uniforms = []
for (i = 0; i < numUniforms; ++i) {
info = gl.getActiveUniform(program, i)
if (info) {
if (info.size > 1) {
for (var j = 0; j < info.size; ++j) {
var name = info.name.replace('[0]', '[' + j + ']')
uniformState.def(name)
uniforms.push(new ActiveInfo(
name,
stringStore.id(name),
gl.getUniformLocation(program, name),
info))
}
} else {
uniformState.def(info.name)
uniforms.push(new ActiveInfo(
info.name,
stringStore.id(info.name),
gl.getUniformLocation(program, info.name),
info))
}
}
}
// -------------------------------
// grab attributes
// -------------------------------
var numAttributes = gl.getProgramParameter(program, GL_ACTIVE_ATTRIBUTES)
var attributes = desc.attributes = []
for (i = 0; i < numAttributes; ++i) {
info = gl.getActiveAttrib(program, i)
if (info) {
attributeState.def(info.name)
attributes.push(new ActiveInfo(
info.name,
stringStore.id(info.name),
gl.getAttribLocation(program, info.name),
info))
}
}
// -------------------------------
// clear cached rendering methods
// -------------------------------
desc.draw = compileShaderDraw(desc)
desc.batchCache = {}
}
var fragShaderStack = [ -1 ]
var vertShaderStack = [ -1 ]
return {
clear: function () {
var deleteShader = gl.deleteShader.bind(gl)
values(fragShaders).forEach(deleteShader)
fragShaders = {}
values(vertShaders).forEach(deleteShader)
vertShaders = {}
programList.forEach(function (desc) {
gl.deleteProgram(desc.program)
})
programList.length = 0
programCache = {}
},
refresh: function () {
fragShaders = {}
vertShaders = {}
programList.forEach(linkProgram)
},
program: function (vertId, fragId) {
check(vertId >= 0, 'missing vertex shader')
check(fragId >= 0, 'missing fragment shader')
var cache = programCache[fragId]
if (!cache) {
cache = programCache[fragId] = {}
}
var program = cache[vertId]
if (!program) {
program = new REGLProgram(fragId, vertId)
linkProgram(program)
cache[vertId] = program
programList.push(program)
}
return program
},
shader: getShader,
frag: fragShaderStack,
vert: vertShaderStack
}
}
},{"./util/check":21,"./util/values":32}],17:[function(require,module,exports){
var createStack = require('./util/stack')
var createEnvironment = require('./util/codegen')
// WebGL constants
var GL_CULL_FACE = 0x0B44
var GL_BLEND = 0x0BE2
var GL_DITHER = 0x0BD0
var GL_STENCIL_TEST = 0x0B90
var GL_DEPTH_TEST = 0x0B71
var GL_SCISSOR_TEST = 0x0C11
var GL_POLYGON_OFFSET_FILL = 0x8037
var GL_SAMPLE_ALPHA_TO_COVERAGE = 0x809E
var GL_SAMPLE_COVERAGE = 0x80A0
var GL_FUNC_ADD = 0x8006
var GL_ZERO = 0
var GL_ONE = 1
var GL_FRONT = 1028
var GL_BACK = 1029
var GL_LESS = 513
var GL_CCW = 2305
var GL_ALWAYS = 519
var GL_KEEP = 7680
module.exports = function wrapContextState (gl, framebufferState, viewportState) {
function capStack (cap, dflt) {
var result = createStack([!!dflt], function (flag) {
if (flag) {
gl.enable(cap)
} else {
gl.disable(cap)
}
})
result.flag = cap
return result
}
// Caps, flags and other random WebGL context state
var contextState = {
// Dithering
'dither': capStack(GL_DITHER),
// Blending
'blend.enable': capStack(GL_BLEND),
'blend.color': createStack([0, 0, 0, 0], function (r, g, b, a) {
gl.blendColor(r, g, b, a)
}),
'blend.equation': createStack([GL_FUNC_ADD, GL_FUNC_ADD], function (rgb, a) {
gl.blendEquationSeparate(rgb, a)
}),
'blend.func': createStack([
GL_ONE, GL_ZERO, GL_ONE, GL_ZERO
], function (srcRGB, dstRGB, srcAlpha, dstAlpha) {
gl.blendFuncSeparate(srcRGB, dstRGB, srcAlpha, dstAlpha)
}),
// Depth
'depth.enable': capStack(GL_DEPTH_TEST, true),
'depth.func': createStack([GL_LESS], function (func) {
gl.depthFunc(func)
}),
'depth.range': createStack([0, 1], function (near, far) {
gl.depthRange(near, far)
}),
'depth.mask': createStack([true], function (m) {
gl.depthMask(m)
}),
// Face culling
'cull.enable': capStack(GL_CULL_FACE),
'cull.face': createStack([GL_BACK], function (mode) {
gl.cullFace(mode)
}),
// Front face orientation
'frontFace': createStack([GL_CCW], function (mode) {
gl.frontFace(mode)
}),
// Write masks
'colorMask': createStack([true, true, true, true], function (r, g, b, a) {
gl.colorMask(r, g, b, a)
}),
// Line width
'lineWidth': createStack([1], function (w) {
gl.lineWidth(w)
}),
// Polygon offset
'polygonOffset.enable': capStack(GL_POLYGON_OFFSET_FILL),
'polygonOffset.offset': createStack([0, 0], function (factor, units) {
gl.polygonOffset(factor, units)
}),
// Sample coverage
'sample.alpha': capStack(GL_SAMPLE_ALPHA_TO_COVERAGE),
'sample.enable': capStack(GL_SAMPLE_COVERAGE),
'sample.coverage': createStack([1, false], function (value, invert) {
gl.sampleCoverage(value, invert)
}),
// Stencil
'stencil.enable': capStack(GL_STENCIL_TEST),
'stencil.mask': createStack([-1], function (mask) {
gl.stencilMask(mask)
}),
'stencil.func': createStack([
GL_ALWAYS, 0, -1
], function (func, ref, mask) {
gl.stencilFunc(func, ref, mask)
}),
'stencil.opFront': createStack([
GL_KEEP, GL_KEEP, GL_KEEP
], function (fail, zfail, pass) {
gl.stencilOpSeparate(GL_FRONT, fail, zfail, pass)
}),
'stencil.opBack': createStack([
GL_KEEP, GL_KEEP, GL_KEEP
], function (fail, zfail, pass) {
gl.stencilOpSeparate(GL_BACK, fail, zfail, pass)
}),
// Scissor
'scissor.enable': capStack(GL_SCISSOR_TEST),
'scissor.box': createStack([0, 0, -1, -1], function (x, y, w, h) {
var w_ = w
var fbo = framebufferState.top()
if (w < 0) {
if (fbo) {
w_ = fbo.width - x
} else {
w_ = gl.drawingBufferWidth - x
}
}
var h_ = h
if (h < 0) {
if (fbo) {
h_ = fbo.height - y
} else {
h_ = gl.drawingBufferHeight - y
}
}
gl.scissor(x, y, w_, h_)
}),
// Viewport
'viewport': createStack([0, 0, -1, -1], function (x, y, w, h) {
var w_ = w
var fbo = framebufferState.top()
if (w < 0) {
if (fbo) {
w_ = fbo.width - x
} else {
w_ = gl.drawingBufferWidth - x
}
}
var h_ = h
if (h < 0) {
if (fbo) {
h_ = fbo.height - y
} else {
h_ = gl.drawingBufferHeight - y
}
}
gl.viewport(x, y, w_, h_)
viewportState.width = w_
viewportState.height = h_
})
}
var env = createEnvironment()
var poll = env.proc('poll')
var refresh = env.proc('refresh')
Object.keys(contextState).forEach(function (prop) {
var STACK = env.link(contextState[prop])
poll(STACK, '.poll();')
refresh(STACK, '.setDirty();')
})
var procs = env.compile()
return {
contextState: contextState,
viewport: viewportState,
poll: procs.poll,
refresh: procs.refresh,
notifyViewportChanged: function () {
contextState.viewport.setDirty()
contextState['scissor.box'].setDirty()
}
}
}
},{"./util/codegen":23,"./util/stack":30}],18:[function(require,module,exports){
module.exports = function createStringStore () {
var stringIds = {'': 0}
var stringValues = ['']
return {
id: function (str) {
var result = stringIds[str]
if (result) {
return result
}
result = stringIds[str] = stringValues.length
stringValues.push(str)
return result
},
str: function (id) {
return stringValues[id]
}
}
}
},{}],19:[function(require,module,exports){
var check = require('./util/check')
var extend = require('./util/extend')
var values = require('./util/values')
var isTypedArray = require('./util/is-typed-array')
var isNDArrayLike = require('./util/is-ndarray')
var loadTexture = require('./util/load-texture')
var convertToHalfFloat = require('./util/to-half-float')
var parseDDS = require('./util/parse-dds')
var GL_COMPRESSED_TEXTURE_FORMATS = 0x86A3
var GL_TEXTURE_2D = 0x0DE1
var GL_TEXTURE_CUBE_MAP = 0x8513
var GL_TEXTURE_CUBE_MAP_POSITIVE_X = 0x8515
var GL_RGBA = 0x1908
var GL_ALPHA = 0x1906
var GL_RGB = 0x1907
var GL_LUMINANCE = 0x1909
var GL_LUMINANCE_ALPHA = 0x190A
var GL_RGBA4 = 0x8056
var GL_RGB5_A1 = 0x8057
var GL_RGB565 = 0x8D62
var GL_UNSIGNED_SHORT_4_4_4_4 = 0x8033
var GL_UNSIGNED_SHORT_5_5_5_1 = 0x8034
var GL_UNSIGNED_SHORT_5_6_5 = 0x8363
var GL_UNSIGNED_INT_24_8_WEBGL = 0x84FA
var GL_DEPTH_COMPONENT = 0x1902
var GL_DEPTH_STENCIL = 0x84F9
var GL_SRGB_EXT = 0x8C40
var GL_SRGB_ALPHA_EXT = 0x8C42
var GL_HALF_FLOAT_OES = 0x8D61
var GL_COMPRESSED_RGB_S3TC_DXT1_EXT = 0x83F0
var GL_COMPRESSED_RGBA_S3TC_DXT1_EXT = 0x83F1
var GL_COMPRESSED_RGBA_S3TC_DXT3_EXT = 0x83F2
var GL_COMPRESSED_RGBA_S3TC_DXT5_EXT = 0x83F3
var GL_COMPRESSED_RGB_ATC_WEBGL = 0x8C92
var GL_COMPRESSED_RGBA_ATC_EXPLICIT_ALPHA_WEBGL = 0x8C93
var GL_COMPRESSED_RGBA_ATC_INTERPOLATED_ALPHA_WEBGL = 0x87EE
var GL_COMPRESSED_RGB_PVRTC_4BPPV1_IMG = 0x8C00
var GL_COMPRESSED_RGB_PVRTC_2BPPV1_IMG = 0x8C01
var GL_COMPRESSED_RGBA_PVRTC_4BPPV1_IMG = 0x8C02
var GL_COMPRESSED_RGBA_PVRTC_2BPPV1_IMG = 0x8C03
var GL_COMPRESSED_RGB_ETC1_WEBGL = 0x8D64
var GL_UNSIGNED_BYTE = 0x1401
var GL_UNSIGNED_SHORT = 0x1403
var GL_UNSIGNED_INT = 0x1405
var GL_FLOAT = 0x1406
var GL_TEXTURE_WRAP_S = 0x2802
var GL_TEXTURE_WRAP_T = 0x2803
var GL_REPEAT = 0x2901
var GL_CLAMP_TO_EDGE = 0x812F
var GL_MIRRORED_REPEAT = 0x8370
var GL_TEXTURE_MAG_FILTER = 0x2800
var GL_TEXTURE_MIN_FILTER = 0x2801
var GL_NEAREST = 0x2600
var GL_LINEAR = 0x2601
var GL_NEAREST_MIPMAP_NEAREST = 0x2700
var GL_LINEAR_MIPMAP_NEAREST = 0x2701
var GL_NEAREST_MIPMAP_LINEAR = 0x2702
var GL_LINEAR_MIPMAP_LINEAR = 0x2703
var GL_GENERATE_MIPMAP_HINT = 0x8192
var GL_DONT_CARE = 0x1100
var GL_FASTEST = 0x1101
var GL_NICEST = 0x1102
var GL_TEXTURE_MAX_ANISOTROPY_EXT = 0x84FE
var GL_UNPACK_ALIGNMENT = 0x0CF5
var GL_UNPACK_FLIP_Y_WEBGL = 0x9240
var GL_UNPACK_PREMULTIPLY_ALPHA_WEBGL = 0x9241
var GL_UNPACK_COLORSPACE_CONVERSION_WEBGL = 0x9243
var GL_BROWSER_DEFAULT_WEBGL = 0x9244
var GL_TEXTURE0 = 0x84C0
var MIPMAP_FILTERS = [
GL_NEAREST_MIPMAP_NEAREST,
GL_NEAREST_MIPMAP_LINEAR,
GL_LINEAR_MIPMAP_NEAREST,
GL_LINEAR_MIPMAP_LINEAR
]
function isPow2 (v) {
return !(v & (v - 1)) && (!!v)
}
function isNumericArray (arr) {
return (
Array.isArray(arr) &&
(arr.length === 0 ||
typeof arr[0] === 'number'))
}
function isRectArray (arr) {
if (!Array.isArray(arr)) {
return false
}
var width = arr.length
if (width === 0 || !Array.isArray(arr[0])) {
return false
}
var height = arr[0].length
for (var i = 1; i < width; ++i) {
if (!Array.isArray(arr[i]) || arr[i].length !== height) {
return false
}
}
return true
}
function classString (x) {
return Object.prototype.toString.call(x)
}
function isCanvasElement (object) {
return classString(object) === '[object HTMLCanvasElement]'
}
function isContext2D (object) {
return classString(object) === '[object CanvasRenderingContext2D]'
}
function isImageElement (object) {
return classString(object) === '[object HTMLImageElement]'
}
function isVideoElement (object) {
return classString(object) === '[object HTMLVideoElement]'
}
function isPendingXHR (object) {
return classString(object) === '[object XMLHttpRequest]'
}
function isPixelData (object) {
return (
typeof object === 'string' ||
(!!object && (
isTypedArray(object) ||
isNumericArray(object) ||
isNDArrayLike(object) ||
isCanvasElement(object) ||
isContext2D(object) ||
isImageElement(object) ||
isVideoElement(object) ||
isRectArray(object))))
}
// Transpose an array of pixels
function transposePixels (data, nx, ny, nc, sx, sy, sc, off) {
var result = new data.constructor(nx * ny * nc)
var ptr = 0
for (var i = 0; i < ny; ++i) {
for (var j = 0; j < nx; ++j) {
for (var k = 0; k < nc; ++k) {
result[ptr++] = data[sy * i + sx * j + sc * k + off]
}
}
}
return result
}
module.exports = function createTextureSet (gl, extensions, limits, reglPoll, viewportState) {
var mipmapHint = {
"don't care": GL_DONT_CARE,
'dont care': GL_DONT_CARE,
'nice': GL_NICEST,
'fast': GL_FASTEST
}
var wrapModes = {
'repeat': GL_REPEAT,
'clamp': GL_CLAMP_TO_EDGE,
'mirror': GL_MIRRORED_REPEAT
}
var magFilters = {
'nearest': GL_NEAREST,
'linear': GL_LINEAR
}
var minFilters = extend({
'nearest mipmap nearest': GL_NEAREST_MIPMAP_NEAREST,
'linear mipmap nearest': GL_LINEAR_MIPMAP_NEAREST,
'nearest mipmap linear': GL_NEAREST_MIPMAP_LINEAR,
'linear mipmap linear': GL_LINEAR_MIPMAP_LINEAR,
'mipmap': GL_LINEAR_MIPMAP_LINEAR
}, magFilters)
var colorSpace = {
'none': 0,
'browser': GL_BROWSER_DEFAULT_WEBGL
}
var textureTypes = {
'uint8': GL_UNSIGNED_BYTE,
'rgba4': GL_UNSIGNED_SHORT_4_4_4_4,
'rgb565': GL_UNSIGNED_SHORT_5_6_5,
'rgb5 a1': GL_UNSIGNED_SHORT_5_5_5_1
}
var textureFormats = {
'alpha': GL_ALPHA,
'luminance': GL_LUMINANCE,
'luminance alpha': GL_LUMINANCE_ALPHA,
'rgb': GL_RGB,
'rgba': GL_RGBA,
'rgba4': GL_RGBA4,
'rgb5 a1': GL_RGB5_A1,
'rgb565': GL_RGB565
}
var compressedTextureFormats = {}
if (extensions.ext_srgb) {
textureFormats.srgb = GL_SRGB_EXT
textureFormats.srgba = GL_SRGB_ALPHA_EXT
}
if (extensions.oes_texture_float) {
textureTypes.float = GL_FLOAT
}
if (extensions.oes_texture_half_float) {
textureTypes['half float'] = GL_HALF_FLOAT_OES
}
if (extensions.webgl_depth_texture) {
extend(textureFormats, {
'depth': GL_DEPTH_COMPONENT,
'depth stencil': GL_DEPTH_STENCIL
})
extend(textureTypes, {
'uint16': GL_UNSIGNED_SHORT,
'uint32': GL_UNSIGNED_INT,
'depth stencil': GL_UNSIGNED_INT_24_8_WEBGL
})
}
if (extensions.webgl_compressed_texture_s3tc) {
extend(compressedTextureFormats, {
'rgb s3tc dxt1': GL_COMPRESSED_RGB_S3TC_DXT1_EXT,
'rgba s3tc dxt1': GL_COMPRESSED_RGBA_S3TC_DXT1_EXT,
'rgba s3tc dxt3': GL_COMPRESSED_RGBA_S3TC_DXT3_EXT,
'rgba s3tc dxt5': GL_COMPRESSED_RGBA_S3TC_DXT5_EXT
})
}
if (extensions.webgl_compressed_texture_atc) {
extend(compressedTextureFormats, {
'rgb arc': GL_COMPRESSED_RGB_ATC_WEBGL,
'rgba atc explicit alpha': GL_COMPRESSED_RGBA_ATC_EXPLICIT_ALPHA_WEBGL,
'rgba atc interpolated alpha': GL_COMPRESSED_RGBA_ATC_INTERPOLATED_ALPHA_WEBGL
})
}
if (extensions.webgl_compressed_texture_pvrtc) {
extend(compressedTextureFormats, {
'rgb pvrtc 4bppv1': GL_COMPRESSED_RGB_PVRTC_4BPPV1_IMG,
'rgb pvrtc 2bppv1': GL_COMPRESSED_RGB_PVRTC_2BPPV1_IMG,
'rgba pvrtc 4bppv1': GL_COMPRESSED_RGBA_PVRTC_4BPPV1_IMG,
'rgba pvrtc 2bppv1': GL_COMPRESSED_RGBA_PVRTC_2BPPV1_IMG
})
}
if (extensions.webgl_compressed_texture_etc1) {
compressedTextureFormats['rgb etc1'] = GL_COMPRESSED_RGB_ETC1_WEBGL
}
// Copy over all texture formats
var supportedCompressedFormats = Array.prototype.slice.call(
gl.getParameter(GL_COMPRESSED_TEXTURE_FORMATS))
Object.keys(compressedTextureFormats).forEach(function (name) {
var format = compressedTextureFormats[name]
if (supportedCompressedFormats.indexOf(format) >= 0) {
textureFormats[name] = format
}
})
var supportedFormats = Object.keys(textureFormats)
limits.textureFormats = supportedFormats
var colorFormats = supportedFormats.reduce(function (color, key) {
var glenum = textureFormats[key]
if (glenum === GL_LUMINANCE ||
glenum === GL_ALPHA ||
glenum === GL_LUMINANCE ||
glenum === GL_LUMINANCE_ALPHA ||
glenum === GL_DEPTH_COMPONENT ||
glenum === GL_DEPTH_STENCIL) {
color[glenum] = glenum
} else if (glenum === GL_RGB5_A1 || key.indexOf('rgba') >= 0) {
color[glenum] = GL_RGBA
} else {
color[glenum] = GL_RGB
}
return color
}, {})
// Pixel storage parsing
function PixelInfo (target) {
// tex target
this.target = target
// pixelStorei info
this.flipY = false
this.premultiplyAlpha = false
this.unpackAlignment = 1
this.colorSpace = 0
// shape
this.width = 0
this.height = 0
this.channels = 0
// format and type
this.format = 0
this.internalformat = 0
this.type = 0
this.compressed = false
// mip level
this.miplevel = 0
// ndarray-like parameters
this.strideX = 0
this.strideY = 0
this.strideC = 0
this.offset = 0
// copy pixels info
this.x = 0
this.y = 0
this.copy = false
// data sources
this.data = null
this.image = null
this.video = null
this.canvas = null
this.xhr = null
// CORS
this.crossOrigin = null
// horrible state flags
this.needsPoll = false
this.needsListeners = false
}
extend(PixelInfo.prototype, {
parseFlags: function (options) {
if (typeof options !== 'object' || !options) {
return
}
if ('premultiplyAlpha' in options) {
check.type(options.premultiplyAlpha, 'boolean',
'invalid premultiplyAlpha')
this.premultiplyAlpha = options.premultiplyAlpha
}
if ('flipY' in options) {
check.type(options.flipY, 'boolean',
'invalid texture flip')
this.flipY = options.flipY
}
if ('alignment' in options) {
check.oneOf(options.alignment, [1, 2, 4, 8],
'invalid texture unpack alignment')
this.unpackAlignment = options.alignment
}
if ('colorSpace' in options) {
check.parameter(options.colorSpace, colorSpace,
'invalid colorSpace')
this.colorSpace = colorSpace[options.colorSpace]
}
if ('format' in options) {
var format = options.format
check.parameter(format, textureFormats,
'invalid texture format')
this.internalformat = textureFormats[format]
if (format in textureTypes) {
this.type = textureTypes[format]
}
if (format in compressedTextureFormats) {
this.compressed = true
}
}
if ('type' in options) {
var type = options.type
check.parameter(type, textureTypes,
'invalid texture type')
this.type = textureTypes[type]
}
var w = this.width
var h = this.height
var c = this.channels
if ('shape' in options) {
check(Array.isArray(options.shape) && options.shape.length >= 2,
'shape must be an array')
w = options.shape[0]
h = options.shape[1]
if (options.shape.length === 3) {
c = options.shape[2]
}
} else {
if ('radius' in options) {
w = h = options.radius
}
if ('width' in options) {
w = options.width
}
if ('height' in options) {
h = options.height
}
if ('channels' in options) {
c = options.channels
}
}
this.width = w | 0
this.height = h | 0
this.channels = c | 0
if ('stride' in options) {
var stride = options.stride
check(Array.isArray(stride) && stride.length >= 2,
'invalid stride vector')
this.strideX = stride[0]
this.strideY = stride[1]
if (stride.length === 3) {
this.strideC = stride[2]
} else {
this.strideC = 1
}
this.needsTranspose = true
} else {
this.strideC = 1
this.strideX = this.strideC * c
this.strideY = this.strideX * w
}
if ('offset' in options) {
this.offset = options.offset | 0
this.needsTranspose = true
}
if ('crossOrigin' in options) {
this.crossOrigin = options.crossOrigin
}
},
parse: function (options, miplevel) {
this.miplevel = miplevel
this.width = this.width >> miplevel
this.height = this.height >> miplevel
var data = options
switch (typeof options) {
case 'string':
break
case 'object':
if (!options) {
return
}
this.parseFlags(options)
if (isPixelData(options.data)) {
data = options.data
}
break
case 'undefined':
return
default:
check.raise('invalid pixel data type')
}
if (typeof data === 'string') {
data = loadTexture(data, this.crossOrigin)
}
var array = null
var needsConvert = false
if (this.compressed) {
check(data instanceof Uint8Array || isPendingXHR(data),
'compressed texture data must be stored in a uint8array')
}
if (data === null) {
// TODO
} else if (isTypedArray(data)) {
this.data = data
} else if (isNumericArray(data)) {
array = data
needsConvert = true
} else if (isNDArrayLike(data)) {
if (Array.isArray(data.data)) {
array = data.data
needsConvert = true
} else {
this.data = data.data
}
var shape = data.shape
this.width = shape[0]
this.height = shape[1]
if (shape.length === 3) {
this.channels = shape[2]
} else {
this.channels = 1
}
var stride = data.stride
this.strideX = data.stride[0]
this.strideY = data.stride[1]
if (stride.length === 3) {
this.strideC = data.stride[2]
} else {
this.strideC = 1
}
this.offset = data.offset
this.needsTranspose = true
} else if (isCanvasElement(data) || isContext2D(data)) {
if (isCanvasElement(data)) {
this.canvas = data
} else {
this.canvas = data.canvas
}
this.width = this.canvas.width
this.height = this.canvas.height
this.setDefaultFormat()
} else if (isImageElement(data)) {
this.image = data
if (!data.complete) {
this.width = this.width || data.naturalWidth
this.height = this.height || data.naturalHeight
this.needsListeners = true
} else {
this.width = data.naturalWidth
this.height = data.naturalHeight
}
this.setDefaultFormat()
} else if (isVideoElement(data)) {
this.video = data
if (data.readyState > 1) {
this.width = data.width
this.height = data.height
} else {
this.width = this.width || data.width
this.height = this.height || data.height
this.needsListeners = true
}
this.needsPoll = true
this.setDefaultFormat()
} else if (isPendingXHR(data)) {
this.xhr = data
this.needsListeners = true
} else if (isRectArray(data)) {
var w = data[0].length
var h = data.length
var c = 1
var i, j, k, p
if (Array.isArray(data[0][0])) {
c = data[0][0].length
check(c >= 0 && c <= 4, 'invalid number of channels for image data')
array = Array(w * h * c)
p = 0
for (j = 0; j < h; ++j) {
for (i = 0; i < w; ++i) {
for (k = 0; k < c; ++k) {
array[p++] = data[j][i][k]
}
}
}
} else {
array = Array(w * h)
p = 0
for (j = 0; j < h; ++j) {
for (i = 0; i < w; ++i) {
array[p++] = data[j][i]
}
}
}
this.width = w
this.height = h
this.channels = c
needsConvert = true
} else if (options.copy) {
this.copy = true
this.x = this.x | 0
this.y = this.y | 0
this.width = (this.width || viewportState.width) | 0
this.height = (this.height || viewportState.height) | 0
this.setDefaultFormat()
}
// Fix up missing type info for typed arrays
if (!this.type && this.data) {
if (this.format === GL_DEPTH_COMPONENT) {
if (this.data instanceof Uint16Array) {
this.type = GL_UNSIGNED_SHORT
} else if (this.data instanceof Uint32Array) {
this.type = GL_UNSIGNED_INT
}
} else if (this.data instanceof Float32Array) {
this.type = GL_FLOAT
}
}
// Infer default format
if (!this.internalformat) {
var channels = this.channels = this.channels || 4
this.internalformat = [
GL_LUMINANCE,
GL_LUMINANCE_ALPHA,
GL_RGB,
GL_RGBA][channels - 1]
check(this.internalformat, 'invalid number of channels')
}
var format = this.internalformat
if (format === GL_DEPTH_COMPONENT || format === GL_DEPTH_STENCIL) {
check(extensions.webgl_depth_texture,
'depth/stencil texture not supported')
if (format === GL_DEPTH_COMPONENT) {
check(this.type === GL_UNSIGNED_SHORT || GL_UNSIGNED_INT,
'depth texture type must be uint16 or uint32')
}
if (format === GL_DEPTH_STENCIL) {
check(this.type === GL_UNSIGNED_INT_24_8_WEBGL,
'depth stencil texture format must match type')
}
check(
!this.data && !array && !this.image && !this.video && !this.canvas,
'depth/stencil textures are for rendering only')
}
// Compute color format and number of channels
var colorFormat = this.format = colorFormats[format]
if (!this.channels) {
switch (colorFormat) {
case GL_LUMINANCE:
case GL_ALPHA:
case GL_DEPTH_COMPONENT:
this.channels = 1
break
case GL_DEPTH_STENCIL:
case GL_LUMINANCE_ALPHA:
this.channels = 2
break
case GL_RGB:
this.channels = 3
break
default:
this.channels = 4
}
}
// Check that texture type is supported
var type = this.type
if (type === GL_FLOAT) {
check(extensions.oes_texture_float,
'float texture not supported')
} else if (type === GL_HALF_FLOAT_OES) {
check(extensions.oes_texture_half_float,
'half float texture not supported')
} else if (!type) {
if (format === GL_DEPTH_COMPONENT) {
type = GL_UNSIGNED_INT
} else {
type = GL_UNSIGNED_BYTE
}
}
this.type = type
// apply conversion
if (needsConvert) {
switch (type) {
case GL_UNSIGNED_BYTE:
this.data = new Uint8Array(array)
break
case GL_UNSIGNED_SHORT:
this.data = new Uint16Array(array)
break
case GL_UNSIGNED_INT:
this.data = new Uint32Array(array)
break
case GL_FLOAT:
this.data = new Float32Array(array)
break
case GL_HALF_FLOAT_OES:
this.data = convertToHalfFloat(array)
break
case GL_UNSIGNED_SHORT_5_6_5:
case GL_UNSIGNED_SHORT_5_5_5_1:
case GL_UNSIGNED_SHORT_4_4_4_4:
case GL_UNSIGNED_INT_24_8_WEBGL:
check.raise('unsupported format for automatic conversion')
break
default:
check.raise('unsupported type conversion')
}
}
if (this.data) {
// apply transpose
if (this.needsTranspose) {
this.data = transposePixels(
this.data,
this.width,
this.height,
this.channels,
this.strideX,
this.strideY,
this.strideC,
this.offset)
}
// check data type
switch (type) {
case GL_UNSIGNED_BYTE:
check(this.data instanceof Uint8Array ||
this.data instanceof Uint8ClampedArray,
'incompatible pixel type')
break
case GL_UNSIGNED_SHORT_5_6_5:
case GL_UNSIGNED_SHORT_5_5_5_1:
case GL_UNSIGNED_SHORT_4_4_4_4:
case GL_UNSIGNED_SHORT:
case GL_HALF_FLOAT_OES:
check(this.data instanceof Uint16Array,
'incompatible pixel type')
break
case GL_UNSIGNED_INT:
check(this.data instanceof Uint32Array,
'incompatible pixel type')
break
case GL_FLOAT:
check(this.data instanceof Float32Array,
'incompatible pixel type')
break
default:
check.raise('bad or missing pixel type')
}
}
this.needsTranspose = false
},
setDefaultFormat: function () {
this.format = this.internalformat = GL_RGBA
this.type = GL_UNSIGNED_BYTE
this.channels = 4
this.compressed = false
},
upload: function (params) {
gl.pixelStorei(GL_UNPACK_FLIP_Y_WEBGL, this.flipY)
gl.pixelStorei(GL_UNPACK_PREMULTIPLY_ALPHA_WEBGL, this.premultiplyAlpha)
gl.pixelStorei(GL_UNPACK_COLORSPACE_CONVERSION_WEBGL, this.colorSpace)
gl.pixelStorei(GL_UNPACK_ALIGNMENT, this.unpackAlignment)
var target = this.target
var miplevel = this.miplevel
var image = this.image
var canvas = this.canvas
var video = this.video
var data = this.data
var internalformat = this.internalformat
var format = this.format
var type = this.type
var width = this.width || Math.max(1, params.width >> miplevel)
var height = this.height || Math.max(1, params.height >> miplevel)
if (video && video.readyState > 2) {
gl.texImage2D(target, miplevel, format, format, type, video)
} else if (image && image.complete) {
gl.texImage2D(target, miplevel, format, format, type, image)
} else if (canvas) {
gl.texImage2D(target, miplevel, format, format, type, canvas)
} else if (this.compressed) {
gl.compressedTexImage2D(target, miplevel, internalformat, width, height, 0, data)
} else if (this.copy) {
reglPoll()
gl.copyTexImage2D(target, miplevel, format, this.x, this.y, width, height, 0)
} else if (data) {
gl.texImage2D(target, miplevel, format, width, height, 0, format, type, data)
} else {
gl.texImage2D(target, miplevel, format, width || 1, height || 1, 0, format, type, null)
}
}
})
function TexParams (target) {
this.target = target
// Default image shape info
this.width = 0
this.height = 0
this.format = 0
this.internalformat = 0
this.type = 0
// wrap mode
this.wrapS = GL_CLAMP_TO_EDGE
this.wrapT = GL_CLAMP_TO_EDGE
// filtering
this.minFilter = 0
this.magFilter = GL_NEAREST
this.anisotropic = 1
// mipmaps
this.genMipmaps = false
this.mipmapHint = GL_DONT_CARE
}
extend(TexParams.prototype, {
parse: function (options) {
if (typeof options !== 'object' || !options) {
return
}
if ('min' in options) {
var minFilter = options.min
check.parameter(minFilter, minFilters)
this.minFilter = minFilters[minFilter]
}
if ('mag' in options) {
var magFilter = options.mag
check.parameter(magFilter, magFilters)
this.magFilter = magFilters[magFilter]
}
var wrapS = this.wrapS
var wrapT = this.wrapT
if ('wrap' in options) {
var wrap = options.wrap
if (typeof wrap === 'string') {
check.parameter(wrap, wrapModes)
wrapS = wrapT = wrapModes[wrap]
} else if (Array.isArray(wrap)) {
check.parameter(wrap[0], wrapModes)
check.parameter(wrap[1], wrapModes)
wrapS = wrapModes[wrap[0]]
wrapT = wrapModes[wrap[1]]
}
} else {
if ('wrapS' in options) {
var optWrapS = options.wrapS
check.parameter(optWrapS, wrapModes)
wrapS = wrapModes[optWrapS]
}
if ('wrapT' in options) {
var optWrapT = options.wrapT
check.parameter(optWrapT, wrapModes)
wrapT = wrapModes[optWrapT]
}
}
this.wrapS = wrapS
this.wrapT = wrapT
if ('anisotropic' in options) {
var anisotropic = options.anisotropic
check(typeof anisotropic === 'number' &&
anisotropic >= 1 && anisotropic <= limits.maxAnisotropic,
'aniso samples must be between 1 and ')
this.anisotropic = options.anisotropic
}
if ('mipmap' in options) {
var mipmap = options.mipmap
switch (typeof mipmap) {
case 'string':
check.parameter(mipmap, mipmapHint,
'invalid mipmap hint')
this.mipmapHint = mipmapHint[mipmap]
this.genMipmaps = true
break
case 'boolean':
this.genMipmaps = !!mipmap
break
case 'object':
break
default:
check.raise('invalid mipmap type')
}
}
},
upload: function () {
var target = this.target
gl.texParameteri(target, GL_TEXTURE_MIN_FILTER, this.minFilter)
gl.texParameteri(target, GL_TEXTURE_MAG_FILTER, this.magFilter)
gl.texParameteri(target, GL_TEXTURE_WRAP_S, this.wrapS)
gl.texParameteri(target, GL_TEXTURE_WRAP_T, this.wrapT)
if (extensions.ext_texture_filter_anisotropic) {
gl.texParameteri(target, GL_TEXTURE_MAX_ANISOTROPY_EXT, this.anisotropic)
}
if (this.genMipmaps) {
gl.hint(GL_GENERATE_MIPMAP_HINT, this.mipmapHint)
gl.generateMipmap(target)
}
}
})
// Final pass to merge params and pixel data
function checkTextureComplete (params, pixels) {
var i, pixmap
var type = 0
var format = 0
var internalformat = 0
var width = 0
var height = 0
var channels = 0
var compressed = false
var needsPoll = false
var needsListeners = false
var mipMask2D = 0
var mipMaskCube = [0, 0, 0, 0, 0, 0]
var cubeMask = 0
var hasMip = false
for (i = 0; i < pixels.length; ++i) {
pixmap = pixels[i]
width = width || (pixmap.width << pixmap.miplevel)
height = height || (pixmap.height << pixmap.miplevel)
type = type || pixmap.type
format = format || pixmap.format
internalformat = internalformat || pixmap.internalformat
channels = channels || pixmap.channels
needsPoll = needsPoll || pixmap.needsPoll
needsListeners = needsListeners || pixmap.needsListeners
compressed = compressed || pixmap.compressed
var miplevel = pixmap.miplevel
var target = pixmap.target
hasMip = hasMip || (miplevel > 0)
if (target === GL_TEXTURE_2D) {
mipMask2D |= (1 << miplevel)
} else {
var face = target - GL_TEXTURE_CUBE_MAP_POSITIVE_X
mipMaskCube[face] |= (1 << miplevel)
cubeMask |= (1 << face)
}
}
params.needsPoll = needsPoll
params.needsListeners = needsListeners
params.width = width
params.height = height
params.format = format
params.internalformat = internalformat
params.type = type
var mipMask = hasMip ? (width << 1) - 1 : 1
if (params.target === GL_TEXTURE_2D) {
check(cubeMask === 0,
'pixmap type must not contain cubemap faces')
check(mipMask2D === mipMask, 'missing mip map data')
} else {
check(cubeMask === ((1 << 6) - 1), 'missing cubemap faces')
for (i = 0; i < 6; ++i) {
check(mipMaskCube[i] === mipMask, 'missing mip map data')
}
}
var mipFilter = (MIPMAP_FILTERS.indexOf(params.minFilter) >= 0)
params.genMipmaps = !hasMip && (params.genMipmaps || mipFilter)
var useMipmaps = hasMip || params.genMipmaps
if (!params.minFilter) {
params.minFilter = useMipmaps
? GL_LINEAR_MIPMAP_LINEAR
: GL_NEAREST
} else {
check(useMipmaps === mipFilter,
'min filter inconsistent with mipmap data')
}
if (useMipmaps) {
check(width === height && isPow2(width),
'must be a square power of 2 to support mipmaps')
}
if (params.genMipmaps) {
check(!compressed, 'mipmap generation not supported for compressed textures')
}
params.wrapS = params.wrapS || GL_CLAMP_TO_EDGE
params.wrapT = params.wrapT || GL_CLAMP_TO_EDGE
if (params.wrapS !== GL_CLAMP_TO_EDGE ||
params.wrapT !== GL_CLAMP_TO_EDGE) {
check(isPow2(width) && isPow2(height) && !cubeMask,
'incompatible size for wrap mode, image must be a power of 2')
}
if ((type === GL_FLOAT && !extensions.oes_texture_float_linear) ||
(type === GL_HALF_FLOAT_OES &&
!extensions.oes_texture_half_float_linear)) {
check(this.magFilter === GL_NEAREST && this.minFilter === GL_NEAREST,
'unsupported filter mode for float texture')
}
for (i = 0; i < pixels.length; ++i) {
pixmap = pixels[i]
var level = pixmap.miplevel
if (pixmap.width) {
check(pixmap.width << level === width, 'inconsistent width')
}
if (pixmap.height) {
check(pixmap.height << level === height, 'inconsistent width')
}
if (pixmap.channels) {
check(pixmap.channels === channels, 'inconsistent channels')
} else {
pixmap.channels = channels
}
if (pixmap.format) {
check(pixmap.format === format, 'inconsistent format')
} else {
pixmap.format = format
}
if (pixmap.internalformat) {
check(pixmap.internalformat === internalformat, 'inconsistent internalformat')
} else {
pixmap.internalformat = internalformat
}
if (pixmap.type) {
check(pixmap.type === type, 'inconsistent type')
} else {
pixmap.type = type
}
if (pixmap.copy) {
check(pixmap.type === GL_UNSIGNED_BYTE &&
pixmap.internalformat === GL_RGBA,
'incompatible format/type for copyTexImage2D')
}
}
}
var activeTexture = 0
var textureCount = 0
var textureSet = {}
var pollSet = []
var numTexUnits = limits.maxTextureUnits
var textureUnits = Array(numTexUnits).map(function () {
return null
})
function REGLTexture (target) {
this.id = textureCount++
this.refCount = 1
this.target = target
this.texture = null
this.pollId = -1
this.unit = -1
this.bindCount = 0
// cancels all pending callbacks
this.cancelPending = null
// parsed user inputs
this.params = new TexParams(target)
this.pixels = []
}
function update (texture, options) {
var i
clearListeners(texture)
// Clear parameters and pixel data
var params = texture.params
TexParams.call(params, texture.target)
var pixels = texture.pixels
pixels.length = 0
// parse parameters
params.parse(options)
// parse pixel data
function parseMip (target, data) {
var mipmap = data.mipmap
var pixmap
if (Array.isArray(mipmap)) {
for (var i = 0; i < mipmap.length; ++i) {
pixmap = new PixelInfo(target)
pixmap.parseFlags(options)
pixmap.parseFlags(data)
pixmap.parse(mipmap[i], i)
pixels.push(pixmap)
}
} else {
pixmap = new PixelInfo(target)
pixmap.parseFlags(options)
pixmap.parse(data, 0)
pixels.push(pixmap)
}
}
if (texture.target === GL_TEXTURE_2D) {
parseMip(GL_TEXTURE_2D, options)
} else {
var faces = options.faces || options
if (Array.isArray(faces)) {
check(faces.length === 6,
'invalid number of faces in cube map')
for (i = 0; i < 6; ++i) {
parseMip(GL_TEXTURE_CUBE_MAP_POSITIVE_X + i, faces[i])
}
} else if (typeof faces === 'string') {
// TODO Read dds
} else {
// Initialize to all empty textures
for (i = 0; i < 6; ++i) {
parseMip(GL_TEXTURE_CUBE_MAP_POSITIVE_X + i, {})
}
}
}
// do a second pass to reconcile defaults
checkTextureComplete(params, pixels)
if (params.needsListeners) {
hookListeners(texture)
}
if (params.needsPoll) {
texture.pollId = pollSet.length
pollSet.push(texture)
}
refresh(texture)
}
function refresh (texture) {
if (!gl.isTexture(texture.texture)) {
texture.texture = gl.createTexture()
}
// Lazy bind
var target = texture.target
var unit = texture.unit
if (unit >= 0) {
gl.activeTexture(GL_TEXTURE0 + unit)
activeTexture = unit
} else {
gl.bindTexture(target, texture.texture)
}
// Upload
var pixels = texture.pixels
var params = texture.params
for (var i = 0; i < pixels.length; ++i) {
pixels[i].upload(params)
}
params.upload()
// Lazy unbind
if (unit < 0) {
var active = textureUnits[activeTexture]
if (active) {
// restore binding state
gl.bindTexture(active.target, active.texture)
} else {
// otherwise become new active
texture.unit = activeTexture
textureUnits[activeTexture] = texture
}
}
}
function hookListeners (texture) {
var params = texture.params
var pixels = texture.pixels
// Appends all the texture data from the buffer to the current
function appendDDS (target, miplevel, buffer) {
var dds = parseDDS(buffer)
check(dds.format in colorFormats, 'unsupported dds texture format')
if (dds.cube) {
check(texture.target === GL_TEXTURE_CUBE_MAP)
// TODO handle cube map DDS
check.raise('cube map DDS not yet implemented')
} else {
check(texture.target === GL_TEXTURE_2D)
}
if (miplevel) {
check(dds.pixels.length === 1, 'number of mip levels inconsistent')
}
dds.pixels.forEach(function (pixmap) {
var info = new PixelInfo(dds.cube ? pixmap.target : target)
info.channels = dds.channels
info.compressed = dds.compressed
info.type = dds.type
info.internalformat = dds.format
info.format = colorFormats[dds.format]
info.width = pixmap.width
info.height = pixmap.height
info.miplevel = pixmap.miplevel || miplevel
info.data = pixmap.data
pixels.push(info)
})
}
function onData () {
// Update size of any newly loaded pixels
for (var i = 0; i < pixels.length; ++i) {
var pixelData = pixels[i]
var image = pixelData.image
var video = pixelData.video
var xhr = pixelData.xhr
if (image && image.complete) {
pixelData.width = image.naturalWidth
pixelData.height = image.naturalHeight
} else if (video && video.readyState > 2) {
pixelData.width = video.width
pixelData.height = video.height
} else if (xhr && xhr.readyState === 4) {
pixels[i] = pixels[pixels.length - 1]
pixels.pop()
xhr.removeEventListener('readystatechange', refresh)
appendDDS(pixelData.target, pixelData.miplevel, xhr.response)
}
}
checkTextureComplete(params, pixels)
refresh(texture)
}
pixels.forEach(function (pixelData) {
if (pixelData.image && !pixelData.image.complete) {
pixelData.image.addEventListener('load', onData)
} else if (pixelData.video && pixelData.readyState < 1) {
pixelData.video.addEventListener('progress', onData)
} else if (pixelData.xhr) {
pixelData.xhr.addEventListener('readystatechange', onData)
}
})
texture.cancelPending = function detachListeners () {
pixels.forEach(function (pixelData) {
if (pixelData.image) {
pixelData.image.removeEventListener('load', onData)
} else if (pixelData.video) {
pixelData.video.removeEventListener('progress', onData)
} else if (pixelData.xhr) {
pixelData.xhr.removeEventListener('readystatechange', onData)
pixelData.xhr.abort()
}
})
}
}
function clearListeners (texture) {
var cancelPending = texture.cancelPending
if (cancelPending) {
cancelPending()
texture.cancelPending = null
}
var id = texture.pollId
if (id >= 0) {
var other = pollSet[id] = pollSet[pollSet.length - 1]
other.id = id
pollSet.pop()
texture.pollId = -1
}
}
function destroy (texture) {
var handle = texture.texture
check(handle, 'must not double destroy texture')
var unit = texture.unit
var target = texture.target
if (unit >= 0) {
gl.activeTexture(GL_TEXTURE0 + unit)
activeTexture = unit
gl.bindTexture(target, null)
textureUnits[unit] = null
}
clearListeners(texture)
if (gl.isTexture(handle)) {
gl.deleteTexture(handle)
}
texture.texture = null
texture.params = null
texture.pixels = null
texture.refCount = 0
delete textureSet[texture.id]
}
extend(REGLTexture.prototype, {
bind: function () {
var texture = this
texture.bindCount += 1
var unit = texture.unit
if (unit < 0) {
for (var i = 0; i < numTexUnits; ++i) {
var other = textureUnits[i]
if (other) {
if (other.bindCount > 0) {
continue
}
other.unit = -1
}
textureUnits[i] = texture
unit = i
break
}
if (unit >= numTexUnits) {
check.raise('insufficient number of texture units')
}
texture.unit = unit
gl.activeTexture(GL_TEXTURE0 + unit)
gl.bindTexture(texture.target, texture.texture)
activeTexture = unit
}
return unit
},
unbind: function () {
this.bindCount -= 1
},
decRef: function () {
if (--this.refCount === 0) {
destroy(this)
}
}
})
function createTexture (options, target) {
var texture = new REGLTexture(target)
textureSet[texture.id] = texture
function reglTexture (a0, a1, a2, a3, a4, a5) {
var options = a0 || {}
if (target === GL_TEXTURE_CUBE_MAP && arguments.length === 6) {
options = [a0, a1, a2, a3, a4, a5]
}
update(texture, options)
reglTexture.width = texture.params.width
reglTexture.height = texture.params.height
return reglTexture
}
reglTexture(options)
reglTexture._reglType = 'texture'
reglTexture._texture = texture
reglTexture.destroy = function () {
texture.decRef()
}
return reglTexture
}
// Called after context restore
function refreshTextures () {
values(textureSet).forEach(refresh)
for (var i = 0; i < numTexUnits; ++i) {
textureUnits[i] = null
}
activeTexture = 0
gl.activeTexture(GL_TEXTURE0)
}
// Called when regl is destroyed
function destroyTextures () {
for (var i = 0; i < numTexUnits; ++i) {
gl.activeTexture(GL_TEXTURE0 + i)
gl.bindTexture(GL_TEXTURE_2D, null)
textureUnits[i] = null
}
gl.activeTexture(GL_TEXTURE0)
activeTexture = 0
values(textureSet).forEach(destroy)
}
// Called once per raf, updates video textures
function pollTextures () {
pollSet.forEach(refresh)
}
return {
create: createTexture,
refresh: refreshTextures,
clear: destroyTextures,
poll: pollTextures,
getTexture: function (wrapper) {
return null
}
}
}
},{"./util/check":21,"./util/extend":24,"./util/is-ndarray":25,"./util/is-typed-array":26,"./util/load-texture":27,"./util/parse-dds":28,"./util/to-half-float":31,"./util/values":32}],20:[function(require,module,exports){
module.exports = function wrapUniformState (stringStore) {
var uniformState = {}
function defUniform (name) {
var id = stringStore.id(name)
var result = uniformState[id]
if (!result) {
result = uniformState[id] = []
}
return result
}
return {
def: defUniform,
uniforms: uniformState
}
}
},{}],21:[function(require,module,exports){
// Error checking and parameter validation
/* globals btoa */
var isTypedArray = require('./is-typed-array')
var extend = require('./extend')
// only used for extracting shader names. if btoa not present, then errors
// will be slightly crappier
function decodeB64 (str) {
if (typeof btoa !== 'undefined') {
return btoa(str)
}
return 'base64:' + str
}
function raise (message) {
var error = new Error('(regl) ' + message)
console.error(error)
throw error
}
function check (pred, message) {
if (!pred) {
raise(message)
}
}
function encolon (message) {
if (message) {
return ': ' + message
}
return ''
}
function checkParameter (param, possibilities, message) {
if (!(param in possibilities)) {
raise('unknown parameter (' + param + ')' + encolon(message) +
'. possible values: ' + Object.keys(possibilities).join())
}
}
function checkIsTypedArray (data, message) {
if (!isTypedArray(data)) {
raise(
'invalid parameter type' + encolon(message) +
'. must be a typed array')
}
}
function checkTypeOf (value, type, message) {
if (typeof value !== type) {
raise(
'invalid parameter type' + encolon(message) +
'. expected ' + type + ', got ' + (typeof value))
}
}
function checkNonNegativeInt (value, message) {
if (!((value >= 0) &&
((value | 0) === value))) {
raise('invalid parameter type, (' + value + ')' + encolon(message) +
'. must be a nonnegative integer')
}
}
function checkOneOf (value, list, message) {
if (list.indexOf(value) < 0) {
raise('invalid value' + encolon(message) + '. must be one of: ' + list)
}
}
function leftPad (str, n) {
str = str + ''
while (str.length < n) {
str = ' ' + str
}
return str
}
function ShaderFile () {
this.name = 'unknown'
this.lines = []
this.index = {}
this.hasErrors = false
}
function ShaderLine (number, line) {
this.number = number
this.line = line
this.errors = []
}
function ShaderError (fileNumber, lineNumber, message) {
this.file = fileNumber
this.line = lineNumber
this.message = message
}
function guessCommand () {
var error = new Error()
var stack = error.stack
var pat = /at compileProcedure.*\n\s+at.*\((.*)\)/.exec(stack)
if (pat) {
return pat[1]
}
var pat2 = /at compileProcedure.*\n\s+at\s+(.*)\n/.exec(stack)
if (pat2) {
return pat2[1]
}
return 'unknown'
}
function guessCallSite () {
var error = new Error()
var stack = error.stack
var pat = /at REGLCommand.*\n\s+at.*\((.*)\)/.exec(stack)
if (pat) {
return pat[1]
}
var pat2 = /at REGLCommand.*\n\s+at\s+(.*)\n/.exec(stack)
if (pat2) {
return pat2[1]
}
return 'unknown'
}
function parseSource (source) {
var lines = source.split('\n')
var lineNumber = 1
var fileNumber = 0
var files = {
unknown: new ShaderFile(),
0: new ShaderFile()
}
files.unknown.name = files[0].name = guessCommand()
files.unknown.lines.push(new ShaderLine(0, ''))
for (var i = 0; i < lines.length; ++i) {
var line = lines[i]
var parts = /^\s*\#\s*(\w+)\s+(.+)\s*$/.exec(line)
if (parts) {
switch (parts[1]) {
case 'line':
var lineNumberInfo = /(\d+)(\s+\d+)?/.exec(parts[2])
if (lineNumberInfo) {
lineNumber = lineNumberInfo[1] | 0
if (lineNumberInfo[2]) {
fileNumber = lineNumberInfo[2] | 0
if (!(fileNumber in files)) {
files[fileNumber] = new ShaderFile()
}
}
}
break
case 'define':
var nameInfo = /SHADER_NAME(_B64)?\s+(.*)$/.exec(parts[2])
if (nameInfo) {
files[fileNumber].name = (nameInfo[1]
? decodeB64(nameInfo[2])
: nameInfo[2])
}
break
}
}
files[fileNumber].lines.push(new ShaderLine(lineNumber++, line))
}
Object.keys(files).forEach(function (fileNumber) {
var file = files[fileNumber]
file.lines.forEach(function (line) {
file.index[line.number] = line
})
})
return files
}
function parseErrorLog (errLog) {
var result = []
errLog.split('\n').forEach(function (errMsg) {
var parts = /^ERROR\:\s+(\d+)\:(\d+)\:\s*(.*)$/.exec(errMsg)
if (parts) {
result.push(new ShaderError(
parts[1] | 0,
parts[2] | 0,
parts[3].trim()))
} else if (errMsg.length > 0) {
result.push(new ShaderError('unknown', 0, errMsg))
}
})
return result
}
function annotateFiles (files, errors) {
errors.forEach(function (error) {
var file = files[error.file]
if (file) {
var line = file.index[error.line]
if (line) {
line.errors.push(error)
file.hasErrors = true
return
}
}
files.unknown.hasErrors = true
files.unknown.lines[0].errors.push(error)
})
}
function checkShaderError (gl, shader, source, type) {
if (!gl.getShaderParameter(shader, gl.COMPILE_STATUS)) {
var errLog = gl.getShaderInfoLog(shader)
var typeName = type === gl.FRAGMENT_SHADER ? 'fragment' : 'vertex'
var files = parseSource(source)
var errors = parseErrorLog(errLog)
annotateFiles(files, errors)
Object.keys(files).forEach(function (fileNumber) {
var file = files[fileNumber]
if (!file.hasErrors) {
return
}
var strings = ['']
var styles = ['']
function push (str, style) {
strings.push(str)
styles.push(style || '')
}
push('file number ' + fileNumber + ': ' + file.name + '\n', 'color:red;text-decoration:underline;font-weight:bold')
file.lines.forEach(function (line) {
if (line.errors.length > 0) {
push(leftPad(line.number, 4) + '| ', 'background-color:yellow; font-weight:bold')
push(line.line + '\n', 'color:red; background-color:yellow; font-weight:bold')
// try to guess token
var offset = 0
line.errors.forEach(function (error) {
var message = error.message
var token = /^\s*\'(.*)\'\s*\:\s*(.*)$/.exec(message)
if (token) {
var tokenPat = token[1]
message = token[2]
switch (tokenPat) {
case 'assign':
tokenPat = '='
break
}
offset = Math.max(line.line.indexOf(tokenPat, offset), 0)
} else {
offset = 0
}
push(leftPad('| ', 6))
push(leftPad('^^^', offset + 3) + '\n', 'font-weight:bold')
push(leftPad('| ', 6))
push(message + '\n', 'font-weight:bold')
})
push(leftPad('| ', 6) + '\n')
} else {
push(leftPad(line.number, 4) + '| ')
push(line.line + '\n', 'color:red')
}
})
if (typeof document !== 'undefined') {
styles[0] = strings.join('%c')
console.log.apply(console, styles)
} else {
console.log(strings.join(''))
}
})
check.raise('Error compiling ' + typeName + ' shader, ' + files[0].name)
}
}
function checkLinkError (gl, program, fragShader, vertShader) {
if (!gl.getProgramParameter(program, gl.LINK_STATUS)) {
var errLog = gl.getProgramInfoLog(program)
var fragParse = parseSource(fragShader)
var vertParse = parseSource(vertShader)
var header = 'Error linking program with vertex shader, "' +
vertParse[0].name + '", and fragment shader "' + fragParse[0].name + '"'
if (typeof document !== 'undefined') {
console.log('%c' + header + '\n%c' + errLog,
'color:red;text-decoration:underline;font-weight:bold',
'color:red')
} else {
console.log(header + '\n' + errLog)
}
check.raise(header)
}
}
function saveCommandRef (object) {
object._commandRef = guessCommand()
}
function saveDrawCommandInfo (opts, uniforms, attributes, stringStore) {
saveCommandRef(opts)
function id (str) {
if (str) {
return stringStore.id(str)
}
return 0
}
opts._fragId = id(opts.static.frag)
opts._vertId = id(opts.static.vert)
function addProps (dict, set) {
Object.keys(set).forEach(function (u) {
dict[stringStore.id(u)] = true
})
}
var uniformSet = opts._uniformSet = {}
addProps(uniformSet, uniforms.static)
addProps(uniformSet, uniforms.dynamic)
var attributeSet = opts._attributeSet = {}
addProps(attributeSet, attributes.static)
addProps(attributeSet, attributes.dynamic)
opts._hasCount = (
'count' in opts.static ||
'count' in opts.dynamic ||
'elements' in opts.static ||
'elements' in opts.dynamic)
}
function checkDrawCommandState (
drawState,
shaderState,
uniformState,
attributeState,
commandRef,
fragId,
vertId,
uniformSet,
attributeSet,
hasCount) {
var i, id
// Check count
if (!hasCount && drawState.count[drawState.count.length - 1] < 0) {
raise('missing vertex count in command ' + commandRef + ' - called from ' + guessCallSite())
}
var program = shaderState.program(
vertId || shaderState.vert[shaderState.vert.length - 1],
fragId || shaderState.frag[shaderState.frag.length - 1])
// Check uniforms
var progUniforms = program.uniforms
for (i = 0; i < progUniforms.length; ++i) {
id = progUniforms[i].id
if (!uniformSet[id] && uniformState.uniforms[id].length <= 0) {
raise('missing uniform "' + progUniforms[i].name + '" in command ' +
commandRef + ' called from ' + guessCallSite())
}
}
var progAttributes = program.attributes
for (i = 0; i < progAttributes.length; ++i) {
id = progAttributes[i].id
if (!attributeSet[id] && attributeState.state[id].records.length <= 0) {
raise('missing attribute "' + progAttributes[i].name + '" in command ' +
commandRef + ' called from ' + guessCallSite())
}
}
}
module.exports = extend(check, {
raise: raise,
parameter: checkParameter,
type: checkTypeOf,
isTypedArray: checkIsTypedArray,
nni: checkNonNegativeInt,
oneOf: checkOneOf,
shaderError: checkShaderError,
linkError: checkLinkError,
callSite: guessCallSite,
saveCommandRef: saveCommandRef,
drawOk: checkDrawCommandState,
saveDrawInfo: saveDrawCommandInfo
})
},{"./extend":24,"./is-typed-array":26}],22:[function(require,module,exports){
/* globals performance */
module.exports =
(typeof performance !== 'undefined' && performance.now)
? function () { return performance.now() }
: function () { return +(new Date()) }
},{}],23:[function(require,module,exports){
var extend = require('./extend')
function slice (x) {
return Array.prototype.slice.call(x)
}
module.exports = function createEnvironment () {
// Unique variable id counter
var varCounter = 0
// Linked values are passed from this scope into the generated code block
// Calling link() passes a value into the generated scope and returns
// the variable name which it is bound to
var linkedNames = []
var linkedValues = []
function link (value) {
var name = 'g' + (varCounter++)
linkedNames.push(name)
linkedValues.push(value)
return name
}
// create a code block
function block () {
var code = []
function push () {
code.push.apply(code, slice(arguments))
}
var vars = []
function def () {
var name = 'v' + (varCounter++)
vars.push(name)
if (arguments.length > 0) {
code.push(name, '=')
code.push.apply(code, slice(arguments))
code.push(';')
}
return name
}
return extend(push, {
def: def,
toString: function () {
return [
(vars.length > 0 ? 'var ' + vars + ';' : ''),
code.join('')
].join('')
}
})
}
// procedure list
var procedures = {}
function proc (name) {
var args = []
function arg () {
var name = 'a' + (varCounter++)
args.push(name)
return name
}
var body = block()
var bodyToString = body.toString
var result = procedures[name] = extend(body, {
arg: arg,
toString: function () {
return [
'function(', args.join(), '){',
bodyToString(),
'}'
].join('')
}
})
return result
}
function compile () {
var code = ['"use strict";return {']
Object.keys(procedures).forEach(function (name) {
code.push('"', name, '":', procedures[name].toString(), ',')
})
code.push('}')
var proc = Function.apply(null, linkedNames.concat([code.join('')]))
return proc.apply(null, linkedValues)
}
return {
link: link,
block: block,
proc: proc,
compile: compile
}
}
},{"./extend":24}],24:[function(require,module,exports){
module.exports = function (base, opts) {
var keys = Object.keys(opts)
for (var i = 0; i < keys.length; ++i) {
base[keys[i]] = opts[keys[i]]
}
return base
}
},{}],25:[function(require,module,exports){
var isTypedArray = require('./is-typed-array')
module.exports = function isNDArrayLike (obj) {
return (
typeof obj === 'object' &&
Array.isArray(obj.shape) &&
Array.isArray(obj.stride) &&
typeof obj.offset === 'number' &&
obj.shape.length === obj.stride.length &&
(Array.isArray(obj.data) ||
isTypedArray(obj.data)))
}
},{"./is-typed-array":26}],26:[function(require,module,exports){
var dtypes = require('../constants/arraytypes.json')
module.exports = function (x) {
return Object.prototype.toString.call(x) in dtypes
}
},{"../constants/arraytypes.json":4}],27:[function(require,module,exports){
/* globals document, Image, XMLHttpRequest */
module.exports = loadTexture
function getExtension (url) {
var parts = /\.(\w+)(\?.*)?$/.exec(url)
if (parts && parts[1]) {
return parts[1].toLowerCase()
}
}
function isVideoExtension (url) {
return [
'avi',
'asf',
'gifv',
'mov',
'qt',
'yuv',
'mpg',
'mpeg',
'm2v',
'mp4',
'm4p',
'm4v',
'ogg',
'ogv',
'vob',
'webm',
'wmv'
].indexOf(url) >= 0
}
function isCompressedExtension (url) {
return [
'dds'
].indexOf(url) >= 0
}
function loadVideo (url, crossOrigin) {
var video = document.createElement('video')
video.autoplay = true
video.loop = true
if (crossOrigin) {
video.crossOrigin = crossOrigin
}
video.src = url
return video
}
function loadCompressedTexture (url, ext, crossOrigin) {
var xhr = new XMLHttpRequest()
xhr.responseType = 'arraybuffer'
xhr.open('GET', url, true)
xhr.send()
return xhr
}
function loadImage (url, crossOrigin) {
var image = new Image()
if (crossOrigin) {
image.crossOrigin = crossOrigin
}
image.src = url
return image
}
// Currently this stuff only works in a DOM environment
function loadTexture (url, crossOrigin) {
if (typeof document !== 'undefined') {
var ext = getExtension(url)
if (isVideoExtension(ext)) {
return loadVideo(url, crossOrigin)
}
if (isCompressedExtension(ext)) {
return loadCompressedTexture(url, ext, crossOrigin)
}
return loadImage(url, crossOrigin)
}
return null
}
},{}],28:[function(require,module,exports){
// References:
//
// http://msdn.microsoft.com/en-us/library/bb943991.aspx/
// http://blog.tojicode.com/2011/12/compressed-textures-in-webgl.html
//
var check = require('./check')
module.exports = parseDDS
var DDS_MAGIC = 0x20534444
var GL_TEXTURE_2D = 0x0DE1
var GL_TEXTURE_CUBE_MAP_POSITIVE_X = 0x8515
var GL_COMPRESSED_RGB_S3TC_DXT1_EXT = 0x83F0
var GL_COMPRESSED_RGBA_S3TC_DXT1_EXT = 0x83F1
var GL_COMPRESSED_RGBA_S3TC_DXT3_EXT = 0x83F2
var GL_COMPRESSED_RGBA_S3TC_DXT5_EXT = 0x83F3
var GL_COMPRESSED_RGB_ETC1_WEBGL = 0x8D64
var GL_UNSIGNED_BYTE = 0x1401
// var GL_HALF_FLOAT_OES = 0x8D61
// var GL_FLOAT = 0x1406
var DDSD_MIPMAPCOUNT = 0x20000
var DDSCAPS2_CUBEMAP = 0x200
var DDSCAPS2_CUBEMAP_POSITIVEX = 0x400
var DDSCAPS2_CUBEMAP_NEGATIVEX = 0x800
var DDSCAPS2_CUBEMAP_POSITIVEY = 0x1000
var DDSCAPS2_CUBEMAP_NEGATIVEY = 0x2000
var DDSCAPS2_CUBEMAP_POSITIVEZ = 0x4000
var DDSCAPS2_CUBEMAP_NEGATIVEZ = 0x8000
var CUBEMAP_COMPLETE_FACES = (
DDSCAPS2_CUBEMAP_POSITIVEX |
DDSCAPS2_CUBEMAP_NEGATIVEX |
DDSCAPS2_CUBEMAP_POSITIVEY |
DDSCAPS2_CUBEMAP_NEGATIVEY |
DDSCAPS2_CUBEMAP_POSITIVEZ |
DDSCAPS2_CUBEMAP_NEGATIVEZ)
var DDPF_FOURCC = 0x4
var DDPF_RGB = 0x40
var FOURCC_DXT1 = 0x31545844
var FOURCC_DXT3 = 0x33545844
var FOURCC_DXT5 = 0x35545844
var FOURCC_ETC1 = 0x31435445
// DDS_HEADER {
var OFF_SIZE = 1 // int32 dwSize
var OFF_FLAGS = 2 // int32 dwFlags
var OFF_HEIGHT = 3 // int32 dwHeight
var OFF_WIDTH = 4 // int32 dwWidth
// var OFF_PITCH = 5 // int32 dwPitchOrLinearSize
// var OFF_DEPTH = 6 // int32 dwDepth
var OFF_MIPMAP = 7 // int32 dwMipMapCount; // offset: 7
// int32[11] dwReserved1
// DDS_PIXELFORMAT {
// var OFF_PF_SIZE = 19 // int32 dwSize; // offset: 19
var OFF_PF_FLAGS = 20 // int32 dwFlags
var OFF_FOURCC = 21 // char[4] dwFourCC
// var OFF_RGBA_BITS = 22 // int32 dwRGBBitCount
// var OFF_RED_MASK = 23 // int32 dwRBitMask
// var OFF_GREEN_MASK = 24 // int32 dwGBitMask
// var OFF_BLUE_MASK = 25 // int32 dwBBitMask
// var OFF_ALPHA_MASK = 26 // int32 dwABitMask; // offset: 26
// }
// var OFF_CAPS = 27 // int32 dwCaps; // offset: 27
var OFF_CAPS2 = 28 // int32 dwCaps2
// var OFF_CAPS3 = 29 // int32 dwCaps3
// var OFF_CAPS4 = 30 // int32 dwCaps4
// int32 dwReserved2 // offset 31
function parseDDS (arrayBuffer) {
var header = new Int32Array(arrayBuffer)
check(header[0] === DDS_MAGIC,
'invalid magic number for dds header')
var flags = header[OFF_FLAGS]
check(flags & DDPF_FOURCC,
'unsupported dds format')
var width = header[OFF_WIDTH]
var height = header[OFF_HEIGHT]
var type = GL_UNSIGNED_BYTE
var format = 0
var blockBytes = 0
var channels = 4
switch (header[OFF_FOURCC]) {
case FOURCC_DXT1:
blockBytes = 8
if (flags & DDPF_RGB) {
channels = 3
format = GL_COMPRESSED_RGB_S3TC_DXT1_EXT
} else {
format = GL_COMPRESSED_RGBA_S3TC_DXT1_EXT
}
break
case FOURCC_DXT3:
blockBytes = 16
format = GL_COMPRESSED_RGBA_S3TC_DXT3_EXT
break
case FOURCC_DXT5:
blockBytes = 16
format = GL_COMPRESSED_RGBA_S3TC_DXT5_EXT
break
case FOURCC_ETC1:
blockBytes = 8
format = GL_COMPRESSED_RGB_ETC1_WEBGL
break
// TODO: Implement hdr and uncompressed textures
default:
// Handle uncompressed data here
check.raise('unsupported dds texture format')
}
var pixelFlags = header[OFF_PF_FLAGS]
var mipmapCount = 1
if (pixelFlags & DDSD_MIPMAPCOUNT) {
mipmapCount = Math.max(1, header[OFF_MIPMAP])
}
var ptr = header[OFF_SIZE] + 4
var result = {
width: width,
height: height,
channels: channels,
format: format,
type: type,
compressed: true,
cube: false,
pixels: []
}
function parseMips (target) {
var mipWidth = width
var mipHeight = height
for (var i = 0; i < mipmapCount; ++i) {
var size =
Math.max(1, (mipWidth + 3) >> 2) *
Math.max(1, (mipHeight + 3) >> 2) *
blockBytes
result.pixels.push({
target: target,
miplevel: i,
width: mipWidth,
height: mipHeight,
data: new Uint8Array(arrayBuffer, ptr, size)
})
ptr += size
mipWidth >>= 1
mipHeight >>= 1
}
}
var caps2 = header[OFF_CAPS2]
var cubemap = !!(caps2 & DDSCAPS2_CUBEMAP)
if (cubemap) {
check(
(caps2 & CUBEMAP_COMPLETE_FACES) === CUBEMAP_COMPLETE_FACES,
'missing cubemap faces')
result.cube = true
for (var i = 0; i < 6; ++i) {
parseMips(GL_TEXTURE_CUBE_MAP_POSITIVE_X + i)
}
} else {
parseMips(GL_TEXTURE_2D)
}
return result
}
},{"./check":21}],29:[function(require,module,exports){
/* globals requestAnimationFrame, cancelAnimationFrame */
if (typeof requestAnimationFrame === 'function' &&
typeof cancelAnimationFrame === 'function') {
module.exports = {
next: function (x) { return requestAnimationFrame(x) },
cancel: function (x) { return cancelAnimationFrame(x) }
}
} else {
module.exports = {
next: function (cb) {
setTimeout(cb, 30)
},
cancel: clearTimeout
}
}
},{}],30:[function(require,module,exports){
// A stack for managing the state of a scalar/vector parameter
module.exports = function createStack (init, onChange) {
var n = init.length
var stack = init.slice()
var current = init.slice()
var dirty = false
var forceDirty = true
function poll () {
var ptr = stack.length - n
if (dirty || forceDirty) {
switch (n) {
case 1:
onChange(stack[ptr])
break
case 2:
onChange(stack[ptr], stack[ptr + 1])
break
case 3:
onChange(stack[ptr], stack[ptr + 1], stack[ptr + 2])
break
case 4:
onChange(stack[ptr], stack[ptr + 1], stack[ptr + 2], stack[ptr + 3])
break
case 5:
onChange(stack[ptr], stack[ptr + 1], stack[ptr + 2], stack[ptr + 3], stack[ptr + 4])
break
case 6:
onChange(stack[ptr], stack[ptr + 1], stack[ptr + 2], stack[ptr + 3], stack[ptr + 4], stack[ptr + 5])
break
default:
onChange.apply(null, stack.slice(ptr, stack.length))
}
for (var i = 0; i < n; ++i) {
current[i] = stack[ptr + i]
}
forceDirty = dirty = false
}
}
return {
push: function () {
dirty = false
for (var i = 0; i < n; ++i) {
var x = arguments[i]
dirty = dirty || (x !== current[i])
stack.push(x)
}
},
pop: function () {
dirty = false
stack.length -= n
for (var i = 0; i < n; ++i) {
dirty = dirty || (stack[stack.length - n + i] !== current[i])
}
},
poll: poll,
setDirty: function () {
forceDirty = true
}
}
}
},{}],31:[function(require,module,exports){
module.exports = function convertToHalfFloat (array) {
var floats = new Float32Array(array)
var uints = new Uint32Array(floats.buffer)
var ushorts = new Uint16Array(array.length)
for (var i = 0; i < array.length; ++i) {
if (isNaN(array[i])) {
ushorts[i] = 0xffff
} else if (array[i] === Infinity) {
ushorts[i] = 0x7c00
} else if (array[i] === -Infinity) {
ushorts[i] = 0xfc00
} else {
var x = uints[i]
var sgn = (x >>> 31) << 15
var exp = ((x << 1) >>> 24) - 127
var frac = (x >> 13) & ((1 << 10) - 1)
if (exp < -24) {
// round non-representable denormals to 0
ushorts[i] = sgn
} else if (exp < -14) {
// handle denormals
var s = -14 - exp
ushorts[i] = sgn + ((frac + (1 << 10)) >> s)
} else if (exp > 15) {
// round overflow to +/- Infinity
ushorts[i] = sgn + 0x7c00
} else {
// otherwise convert directly
ushorts[i] = sgn + ((exp + 15) << 10) + frac
}
}
}
return ushorts
}
},{}],32:[function(require,module,exports){
module.exports = function (obj) {
return Object.keys(obj).map(function (key) { return obj[key] })
}
},{}],"regl":[function(require,module,exports){
var check = require('./lib/util/check')
var extend = require('./lib/util/extend')
var getContext = require('./lib/context')
var createStringStore = require('./lib/strings')
var wrapExtensions = require('./lib/extension')
var wrapLimits = require('./lib/limits')
var wrapBuffers = require('./lib/buffer')
var wrapElements = require('./lib/elements')
var wrapTextures = require('./lib/texture')
var wrapRenderbuffers = require('./lib/renderbuffer')
var wrapFramebuffers = require('./lib/framebuffer')
var wrapUniforms = require('./lib/uniform')
var wrapAttributes = require('./lib/attribute')
var wrapShaders = require('./lib/shader')
var wrapDraw = require('./lib/draw')
var wrapContext = require('./lib/state')
var createCompiler = require('./lib/compile')
var wrapRead = require('./lib/read')
var dynamic = require('./lib/dynamic')
var raf = require('./lib/util/raf')
var clock = require('./lib/util/clock')
var GL_COLOR_BUFFER_BIT = 16384
var GL_DEPTH_BUFFER_BIT = 256
var GL_STENCIL_BUFFER_BIT = 1024
var GL_ARRAY_BUFFER = 34962
var GL_TEXTURE_2D = 0x0DE1
var GL_TEXTURE_CUBE_MAP = 0x8513
var CONTEXT_LOST_EVENT = 'webglcontextlost'
var CONTEXT_RESTORED_EVENT = 'webglcontextrestored'
module.exports = function wrapREGL () {
var args = getContext(Array.prototype.slice.call(arguments))
var gl = args.gl
var options = args.options
var stringStore = createStringStore()
var extensionState = wrapExtensions(gl)
var extensions = extensionState.extensions
var viewportState = {
width: gl.drawingBufferWidth,
height: gl.drawingBufferHeight
}
var limits = wrapLimits(
gl,
extensions)
var bufferState = wrapBuffers(gl)
var elementState = wrapElements(
gl,
extensions,
bufferState)
var uniformState = wrapUniforms(stringStore)
var attributeState = wrapAttributes(
gl,
extensions,
limits,
bufferState,
stringStore)
var shaderState = wrapShaders(
gl,
attributeState,
uniformState,
function (program) {
return compiler.draw(program)
},
stringStore)
var drawState = wrapDraw(
gl,
extensions,
bufferState)
var textureState = wrapTextures(
gl,
extensions,
limits,
poll,
viewportState)
var renderbufferState = wrapRenderbuffers(
gl,
extensions,
limits)
var framebufferState = wrapFramebuffers(
gl,
extensions,
limits,
textureState,
renderbufferState)
var frameState = {
count: 0,
start: clock(),
dt: 0,
t: clock(),
renderTime: 0,
width: gl.drawingBufferWidth,
height: gl.drawingBufferHeight,
pixelRatio: options.pixelRatio
}
var glState = wrapContext(
gl,
framebufferState,
viewportState)
var readPixels = wrapRead(gl, poll, viewportState)
var compiler = createCompiler(
gl,
stringStore,
extensions,
limits,
bufferState,
elementState,
textureState,
framebufferState,
glState,
uniformState,
attributeState,
shaderState,
drawState,
frameState,
poll)
var canvas = gl.canvas
var rafCallbacks = []
var activeRAF = 0
function handleRAF () {
activeRAF = raf.next(handleRAF)
frameState.count += 1
if (frameState.width !== gl.drawingBufferWidth ||
frameState.height !== gl.drawingBufferHeight) {
frameState.width = gl.drawingBufferWidth
frameState.height = gl.drawingBufferHeight
glState.notifyViewportChanged()
}
var now = clock()
frameState.dt = now - frameState.t
frameState.t = now
textureState.poll()
for (var i = 0; i < rafCallbacks.length; ++i) {
var cb = rafCallbacks[i]
cb(frameState.count, frameState.t, frameState.dt)
}
frameState.renderTime = clock() - now
}
function startRAF () {
if (!activeRAF && rafCallbacks.length > 0) {
handleRAF()
}
}
function stopRAF () {
if (activeRAF) {
raf.cancel(handleRAF)
activeRAF = 0
}
}
function handleContextLoss (event) {
stopRAF()
event.preventDefault()
if (options.onContextLost) {
options.onContextLost()
}
}
function handleContextRestored (event) {
gl.getError()
extensionState.refresh()
bufferState.refresh()
textureState.refresh()
renderbufferState.refresh()
framebufferState.refresh()
shaderState.refresh()
glState.refresh()
if (options.onContextRestored) {
options.onContextRestored()
}
handleRAF()
}
if (canvas) {
canvas.addEventListener(CONTEXT_LOST_EVENT, handleContextLoss, false)
canvas.addEventListener(CONTEXT_RESTORED_EVENT, handleContextRestored, false)
}
function destroy () {
stopRAF()
if (canvas) {
canvas.removeEventListener(CONTEXT_LOST_EVENT, handleContextLoss)
canvas.removeEventListener(CONTEXT_RESTORED_EVENT, handleContextRestored)
}
shaderState.clear()
framebufferState.clear()
renderbufferState.clear()
textureState.clear()
bufferState.clear()
if (options.onDestroy) {
options.onDestroy()
}
}
function compileProcedure (options) {
check(!!options, 'invalid args to regl({...})')
check.type(options, 'object', 'invalid args to regl({...})')
var hasDynamic = false
function flattenNestedOptions (options) {
var result = extend({}, options)
delete result.uniforms
delete result.attributes
function merge (name) {
if (name in result) {
var child = result[name]
delete result[name]
Object.keys(child).forEach(function (prop) {
result[name + '.' + prop] = child[prop]
})
}
}
merge('blend')
merge('depth')
merge('cull')
merge('stencil')
merge('polygonOffset')
merge('scissor')
merge('sample')
return result
}
function separateDynamic (object) {
var staticItems = {}
var dynamicItems = {}
Object.keys(object).forEach(function (option) {
var value = object[option]
if (dynamic.isDynamic(value)) {
hasDynamic = true
dynamicItems[option] = dynamic.unbox(value, option)
} else {
staticItems[option] = value
}
})
return {
dynamic: dynamicItems,
static: staticItems
}
}
var uniforms = separateDynamic(options.uniforms || {})
var attributes = separateDynamic(options.attributes || {})
var opts = separateDynamic(flattenNestedOptions(options))
var compiled = compiler.command(
opts.static, uniforms.static, attributes.static,
opts.dynamic, uniforms.dynamic, attributes.dynamic,
hasDynamic)
var draw = compiled.draw
var batch = compiled.batch
var scope = compiled.scope
var EMPTY_ARRAY = []
function reserve (count) {
while (EMPTY_ARRAY.length < count) {
EMPTY_ARRAY.push(null)
}
return EMPTY_ARRAY
}
check.saveDrawInfo(opts, uniforms, attributes, stringStore)
function REGLCommand (args, body) {
if (typeof args === 'function') {
return scope(null, args)
} else if (typeof body === 'function') {
return scope(args, body)
}
// Runtime shader check. Removed in production builds
check.drawOk(
drawState,
shaderState,
uniformState,
attributeState,
opts._commandRef,
opts._fragId,
opts._vertId,
opts._uniformSet,
opts._attributeSet,
opts._hasCount)
if (typeof args === 'number') {
return batch(args | 0, reserve(args | 0))
} else if (Array.isArray(args)) {
return batch(args.length, args)
}
return draw(args)
}
return REGLCommand
}
function poll () {
framebufferState.poll()
glState.poll()
}
function clear (options) {
var clearFlags = 0
poll()
var c = options.color
if (c) {
gl.clearColor(+c[0] || 0, +c[1] || 0, +c[2] || 0, +c[3] || 0)
clearFlags |= GL_COLOR_BUFFER_BIT
}
if ('depth' in options) {
gl.clearDepth(+options.depth)
clearFlags |= GL_DEPTH_BUFFER_BIT
}
if ('stencil' in options) {
gl.clearStencil(options.stencil | 0)
clearFlags |= GL_STENCIL_BUFFER_BIT
}
check(!!clearFlags, 'called regl.clear with no buffer specified')
gl.clear(clearFlags)
}
function frame (cb) {
rafCallbacks.push(cb)
function cancel () {
var index = rafCallbacks.find(function (item) {
return item === cb
})
if (index < 0) {
return
}
rafCallbacks.splice(index, 1)
if (rafCallbacks.length <= 0) {
stopRAF()
}
}
startRAF()
return {
cancel: cancel
}
}
return extend(compileProcedure, {
// Clear current FBO
clear: clear,
// Short cut for prop binding
prop: dynamic.define,
// executes an empty draw command
draw: compileProcedure({}),
// Resources
elements: function (options) {
return elementState.create(options)
},
buffer: function (options) {
return bufferState.create(options, GL_ARRAY_BUFFER)
},
texture: function (options) {
return textureState.create(options, GL_TEXTURE_2D)
},
cube: function (options) {
if (arguments.length === 6) {
return textureState.create(
Array.prototype.slice.call(arguments),
GL_TEXTURE_CUBE_MAP)
} else {
return textureState.create(options, GL_TEXTURE_CUBE_MAP)
}
},
renderbuffer: function (options) {
return renderbufferState.create(options)
},
framebuffer: function (options) {
return framebufferState.create(options)
},
framebufferCube: function (options) {
check.raise('framebuffer cube not yet implemented')
},
// Frame rendering
frame: frame,
stats: frameState,
// System limits
limits: limits,
// Read pixels
read: readPixels,
// Destroy regl and all associated resources
destroy: destroy
})
}
},{"./lib/attribute":1,"./lib/buffer":2,"./lib/compile":3,"./lib/context":7,"./lib/draw":8,"./lib/dynamic":9,"./lib/elements":10,"./lib/extension":11,"./lib/framebuffer":12,"./lib/limits":13,"./lib/read":14,"./lib/renderbuffer":15,"./lib/shader":16,"./lib/state":17,"./lib/strings":18,"./lib/texture":19,"./lib/uniform":20,"./lib/util/check":21,"./lib/util/clock":22,"./lib/util/extend":24,"./lib/util/raf":29}]},{},[])
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../../home/admin/browserify-cdn/node_modules/browserify/node_modules/browser-pack/_prelude.js","lib/attribute.js","lib/buffer.js","lib/compile.js","lib/constants/arraytypes.json","lib/constants/dtypes.json","lib/constants/primitives.json","lib/context.js","lib/draw.js","lib/dynamic.js","lib/elements.js","lib/extension.js","lib/framebuffer.js","lib/limits.js","lib/read.js","lib/renderbuffer.js","lib/shader.js","lib/state.js","lib/strings.js","lib/texture.js","lib/uniform.js","lib/util/check.js","lib/util/clock.js","lib/util/codegen.js","lib/util/extend.js","lib/util/is-ndarray.js","lib/util/is-typed-array.js","lib/util/load-texture.js","lib/util/parse-dds.js","lib/util/raf.js","lib/util/stack.js","lib/util/to-half-float.js","lib/util/values.js","regl.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACviDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACfA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzuBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC15CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpYA;AACA;AACA;AACA;AACA;AACA;;ACLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACZA;AACA;AACA;AACA;AACA;;ACJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACfA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtCA;AACA;AACA;AACA;;ACHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","var glTypes = require('./constants/dtypes.json')\nvar check = require('./util/check')\n\nvar GL_FLOAT = 5126\n\nmodule.exports = function wrapAttributeState (\n  gl,\n  extensions,\n  limits,\n  bufferState,\n  stringStore) {\n  var attributeState = {}\n\n  function AttributeRecord () {\n    this.pointer = false\n\n    this.x = 0.0\n    this.y = 0.0\n    this.z = 0.0\n    this.w = 0.0\n\n    this.buffer = null\n    this.size = 0\n    this.normalized = false\n    this.type = GL_FLOAT\n    this.offset = 0\n    this.stride = 0\n    this.divisor = 0\n  }\n\n  function attributeRecordsEqual (left, right, size) {\n    if (!left.pointer) {\n      return !right.pointer &&\n        left.x === right.x &&\n        left.y === right.y &&\n        left.z === right.z &&\n        left.w === right.w\n    } else {\n      return right.pointer &&\n        left.buffer === right.buffer &&\n        left.size === size &&\n        left.normalized === right.normalized &&\n        left.type === right.type &&\n        left.offset === right.offset &&\n        left.stride === right.stride &&\n        left.divisor === right.divisor\n    }\n  }\n\n  function setAttributeRecord (left, right, size) {\n    var pointer = left.pointer = right.pointer\n    if (pointer) {\n      left.buffer = right.buffer\n      left.size = size\n      left.normalized = right.normalized\n      left.type = right.type\n      left.offset = right.offset\n      left.stride = right.stride\n      left.divisor = right.divisor\n    } else {\n      left.x = right.x\n      left.y = right.y\n      left.z = right.z\n      left.w = right.w\n    }\n  }\n\n  var NUM_ATTRIBUTES = limits.maxAttributes\n  var attributeBindings = new Array(NUM_ATTRIBUTES)\n  for (var i = 0; i < NUM_ATTRIBUTES; ++i) {\n    attributeBindings[i] = new AttributeRecord()\n  }\n\n  function AttributeStack (name) {\n    this.records = []\n    this.name = name\n  }\n\n  function stackTop (stack) {\n    var records = stack.records\n    return records[records.length - 1]\n  }\n\n  // ===================================================\n  // BIND AN ATTRIBUTE\n  // ===================================================\n  function bindAttributeRecord (index, current, next, insize) {\n    var size = next.size || insize\n    if (attributeRecordsEqual(current, next, size)) {\n      return\n    }\n    if (!next.pointer) {\n      if (current.pointer) {\n        gl.disableVertexAttribArray(index)\n      }\n      gl.vertexAttrib4f(index, next.x, next.y, next.z, next.w)\n    } else {\n      if (!current.pointer) {\n        gl.enableVertexAttribArray(index)\n      }\n      if (current.buffer !== next.buffer) {\n        next.buffer.bind()\n      }\n      gl.vertexAttribPointer(\n        index,\n        size,\n        next.type,\n        next.normalized,\n        next.stride,\n        next.offset)\n      var extInstancing = extensions.angle_instanced_arrays\n      if (extInstancing) {\n        extInstancing.vertexAttribDivisorANGLE(index, next.divisor)\n      }\n    }\n    setAttributeRecord(current, next, size)\n  }\n\n  function bindAttribute (index, current, attribStack, size) {\n    bindAttributeRecord(index, current, stackTop(attribStack), size)\n  }\n\n  // ===================================================\n  // DEFINE A NEW ATTRIBUTE\n  // ===================================================\n  function defAttribute (name) {\n    var id = stringStore.id(name)\n    var result = attributeState[id]\n    if (!result) {\n      result = attributeState[id] = new AttributeStack(name)\n    }\n    return result\n  }\n\n  function createAttributeBox (name) {\n    var stack = [new AttributeRecord()]\n    check.saveCommandRef(stack)\n\n    function alloc (data) {\n      var box\n      if (stack.length <= 0) {\n        box = new AttributeRecord()\n      } else {\n        box = stack.pop()\n      }\n      if (typeof data === 'number') {\n        box.pointer = false\n        box.x = data\n        box.y = 0\n        box.z = 0\n        box.w = 0\n      } else if (Array.isArray(data)) {\n        box.pointer = false\n        box.x = data[0]\n        box.y = data[1]\n        box.z = data[2]\n        box.w = data[3]\n      } else {\n        var buffer = bufferState.getBuffer(data)\n        var size = 0\n        var stride = 0\n        var offset = 0\n        var divisor = 0\n        var normalized = false\n        var type = GL_FLOAT\n        if (!buffer) {\n          buffer = bufferState.getBuffer(data.buffer)\n          check(buffer, 'missing or invalid buffer for attribute \"' +\n            name + '\" called from command ' + box._commandRef)\n          size = data.size || 0\n          stride = data.stride || 0\n          offset = data.offset || 0\n          divisor = data.divisor || 0\n          normalized = data.normalized || false\n          type = buffer.dtype\n          if ('type' in data) {\n            type = glTypes[data.type]\n          }\n        } else {\n          type = buffer.dtype\n        }\n        box.pointer = true\n        box.buffer = buffer\n        box.size = size\n        box.offset = offset\n        box.stride = stride\n        box.divisor = divisor\n        box.normalized = normalized\n        box.type = type\n      }\n      return box\n    }\n\n    function free (box) {\n      stack.push(box)\n    }\n\n    return {\n      alloc: alloc,\n      free: free\n    }\n  }\n\n  return {\n    bindings: attributeBindings,\n    bind: bindAttribute,\n    bindRecord: bindAttributeRecord,\n    def: defAttribute,\n    box: createAttributeBox,\n    state: attributeState\n  }\n}\n","// Array and element buffer creation\nvar check = require('./util/check')\nvar isTypedArray = require('./util/is-typed-array')\nvar isNDArrayLike = require('./util/is-ndarray')\nvar arrayTypes = require('./constants/arraytypes.json')\nvar bufferTypes = require('./constants/dtypes.json')\nvar values = require('./util/values')\n\nvar GL_STATIC_DRAW = 35044\n\nvar GL_BYTE = 5120\nvar GL_UNSIGNED_BYTE = 5121\nvar GL_SHORT = 5122\nvar GL_UNSIGNED_SHORT = 5123\nvar GL_INT = 5124\nvar GL_UNSIGNED_INT = 5125\nvar GL_FLOAT = 5126\n\nvar usageTypes = {\n  'static': 35044,\n  'dynamic': 35048,\n  'stream': 35040\n}\n\nfunction typedArrayCode (data) {\n  return arrayTypes[Object.prototype.toString.call(data)] | 0\n}\n\nfunction makeTypedArray (dtype, args) {\n  switch (dtype) {\n    case GL_UNSIGNED_BYTE:\n      return new Uint8Array(args)\n    case GL_UNSIGNED_SHORT:\n      return new Uint16Array(args)\n    case GL_UNSIGNED_INT:\n      return new Uint32Array(args)\n    case GL_BYTE:\n      return new Int8Array(args)\n    case GL_SHORT:\n      return new Int16Array(args)\n    case GL_INT:\n      return new Int32Array(args)\n    case GL_FLOAT:\n      return new Float32Array(args)\n    default:\n      return null\n  }\n}\n\nfunction flatten (result, data, dimension) {\n  var ptr = 0\n  for (var i = 0; i < data.length; ++i) {\n    var v = data[i]\n    for (var j = 0; j < dimension; ++j) {\n      result[ptr++] = v[j]\n    }\n  }\n}\n\nfunction transpose (result, data, shapeX, shapeY, strideX, strideY, offset) {\n  var ptr = 0\n  for (var i = 0; i < shapeX; ++i) {\n    for (var j = 0; j < shapeY; ++j) {\n      result[ptr++] = data[strideX * i + strideY * j + offset]\n    }\n  }\n  return result\n}\n\nmodule.exports = function wrapBufferState (gl) {\n  var bufferCount = 0\n  var bufferSet = {}\n\n  function REGLBuffer (buffer, type) {\n    this.id = bufferCount++\n    this.buffer = buffer\n    this.type = type\n    this.usage = GL_STATIC_DRAW\n    this.byteLength = 0\n    this.dimension = 1\n    this.data = null\n    this.dtype = GL_UNSIGNED_BYTE\n  }\n\n  REGLBuffer.prototype.bind = function () {\n    gl.bindBuffer(this.type, this.buffer)\n  }\n\n  function refresh (buffer) {\n    if (!gl.isBuffer(buffer.buffer)) {\n      buffer.buffer = gl.createBuffer()\n    }\n    buffer.bind()\n    gl.bufferData(buffer.type, buffer.data || buffer.byteLength, buffer.usage)\n  }\n\n  function destroy (buffer) {\n    var handle = buffer.buffer\n    check(handle, 'buffer must not be deleted already')\n    if (gl.isBuffer(handle)) {\n      gl.deleteBuffer(handle)\n    }\n    buffer.buffer = null\n    delete bufferSet[buffer.id]\n  }\n\n  function createBuffer (options, type, deferInit) {\n    var handle = gl.createBuffer()\n\n    var buffer = new REGLBuffer(handle, type)\n    bufferSet[buffer.id] = buffer\n\n    function reglBuffer (input) {\n      var options = input || {}\n      if (Array.isArray(options) ||\n          isTypedArray(options) ||\n          isNDArrayLike(options)) {\n        options = {\n          data: options\n        }\n      } else if (typeof options === 'number') {\n        options = {\n          length: options | 0\n        }\n      } else if (options === null || options === void 0) {\n        options = {}\n      }\n\n      check.type(\n        options, 'object',\n        'buffer arguments must be an object, a number or an array')\n\n      if ('usage' in options) {\n        var usage = options.usage\n        check.parameter(usage, usageTypes, 'invalid buffer usage')\n        buffer.usage = usageTypes[options.usage]\n      } else {\n        buffer.usage = GL_STATIC_DRAW\n      }\n\n      var dtype = 0\n      if ('type' in options) {\n        check.parameter(options.type, bufferTypes, 'invalid buffer type')\n        dtype = bufferTypes[options.type]\n      }\n\n      var dimension = (options.dimension | 0) || 1\n      var byteLength = 0\n      var data = null\n      if ('data' in options) {\n        data = options.data\n        if (data === null) {\n          byteLength = options.length | 0\n        } else {\n          if (isNDArrayLike(data)) {\n            var shape = data.shape\n            var stride = data.stride\n            var offset = data.offset\n\n            var shapeX = 0\n            var shapeY = 0\n            var strideX = 0\n            var strideY = 0\n            if (shape.length === 1) {\n              shapeX = shape[0]\n              shapeY = 1\n              strideX = stride[0]\n              strideY = 0\n            } else if (shape.length === 2) {\n              shapeX = shape[0]\n              shapeY = shape[1]\n              strideX = stride[0]\n              strideY = stride[1]\n            } else {\n              check.raise('invalid shape')\n            }\n\n            dtype = dtype || typedArrayCode(data) || GL_FLOAT\n            dimension = shapeY\n            data = transpose(\n              makeTypedArray(dtype, shapeX * shapeY),\n              data.data,\n              shapeX, shapeY,\n              strideX, strideY,\n              offset)\n          } else if (Array.isArray(data)) {\n            if (data.length > 0 && Array.isArray(data[0])) {\n              dimension = data[0].length\n              dtype = dtype || GL_FLOAT\n              var result = makeTypedArray(dtype, data.length * dimension)\n              data = flatten(result, data, dimension)\n              data = result\n            } else {\n              dtype = dtype || GL_FLOAT\n              data = makeTypedArray(dtype, data)\n            }\n          } else {\n            check.isTypedArray(data, 'invalid data type buffer data')\n            dtype = dtype || typedArrayCode(data)\n          }\n          byteLength = data.byteLength\n        }\n      } else if ('length' in options) {\n        byteLength = options.length | 0\n        check.nni(byteLength, 'buffer length must be a nonnegative integer')\n      }\n\n      buffer.data = data\n      buffer.dtype = dtype || GL_UNSIGNED_BYTE\n      buffer.byteLength = byteLength\n      buffer.dimension = dimension\n\n      refresh(buffer)\n\n      return reglBuffer\n    }\n\n    if (!deferInit) {\n      reglBuffer(options)\n    }\n\n    reglBuffer._reglType = 'buffer'\n    reglBuffer._buffer = buffer\n    reglBuffer.destroy = function () { destroy(buffer) }\n\n    return reglBuffer\n  }\n\n  return {\n    create: createBuffer,\n\n    clear: function () {\n      values(bufferSet).forEach(destroy)\n    },\n\n    refresh: function () {\n      values(bufferSet).forEach(refresh)\n    },\n\n    getBuffer: function (wrapper) {\n      if (wrapper && wrapper._buffer instanceof REGLBuffer) {\n        return wrapper._buffer\n      }\n      return null\n    }\n  }\n}\n","var check = require('./util/check')\nvar createEnvironment = require('./util/codegen')\n\nvar primTypes = require('./constants/primitives.json')\n\nvar GL_ELEMENT_ARRAY_BUFFER = 34963\n\nvar GL_FRAGMENT_SHADER = 35632\nvar GL_VERTEX_SHADER = 35633\n\nvar GL_FLOAT = 5126\nvar GL_FLOAT_VEC2 = 35664\nvar GL_FLOAT_VEC3 = 35665\nvar GL_FLOAT_VEC4 = 35666\nvar GL_INT = 5124\nvar GL_INT_VEC2 = 35667\nvar GL_INT_VEC3 = 35668\nvar GL_INT_VEC4 = 35669\nvar GL_BOOL = 35670\nvar GL_BOOL_VEC2 = 35671\nvar GL_BOOL_VEC3 = 35672\nvar GL_BOOL_VEC4 = 35673\nvar GL_FLOAT_MAT2 = 35674\nvar GL_FLOAT_MAT3 = 35675\nvar GL_FLOAT_MAT4 = 35676\nvar GL_SAMPLER_2D = 35678\nvar GL_SAMPLER_CUBE = 35680\n\nvar GL_TRIANGLES = 4\n\nvar GL_CULL_FACE = 0x0B44\nvar GL_BLEND = 0x0BE2\nvar GL_DITHER = 0x0BD0\nvar GL_STENCIL_TEST = 0x0B90\nvar GL_DEPTH_TEST = 0x0B71\nvar GL_SCISSOR_TEST = 0x0C11\nvar GL_POLYGON_OFFSET_FILL = 0x8037\nvar GL_SAMPLE_ALPHA_TO_COVERAGE = 0x809E\nvar GL_SAMPLE_COVERAGE = 0x80A0\n\nvar GL_FRONT = 1028\nvar GL_BACK = 1029\n\nvar GL_CW = 0x0900\nvar GL_CCW = 0x0901\n\nvar GL_MIN_EXT = 0x8007\nvar GL_MAX_EXT = 0x8008\n\nvar blendFuncs = {\n  '0': 0,\n  '1': 1,\n  'zero': 0,\n  'one': 1,\n  'src color': 768,\n  'one minus src color': 769,\n  'src alpha': 770,\n  'one minus src alpha': 771,\n  'dst color': 774,\n  'one minus dst color': 775,\n  'dst alpha': 772,\n  'one minus dst alpha': 773,\n  'constant color': 32769,\n  'one minus constant color': 32770,\n  'constant alpha': 32771,\n  'one minus constant alpha': 32772,\n  'src alpha saturate': 776\n}\n\nvar compareFuncs = {\n  'never': 512,\n  'less': 513,\n  '<': 513,\n  'equal': 514,\n  '=': 514,\n  '==': 514,\n  '===': 514,\n  'lequal': 515,\n  '<=': 515,\n  'greater': 516,\n  '>': 516,\n  'notequal': 517,\n  '!=': 517,\n  '!==': 517,\n  'gequal': 518,\n  '>=': 518,\n  'always': 519\n}\n\nvar stencilOps = {\n  '0': 0,\n  'zero': 0,\n  'keep': 7680,\n  'replace': 7681,\n  'increment': 7682,\n  'decrement': 7683,\n  'increment wrap': 34055,\n  'decrement wrap': 34056,\n  'invert': 5386\n}\n\nfunction typeLength (x) {\n  switch (x) {\n    case GL_FLOAT_VEC2:\n    case GL_INT_VEC2:\n    case GL_BOOL_VEC2:\n      return 2\n    case GL_FLOAT_VEC3:\n    case GL_INT_VEC3:\n    case GL_BOOL_VEC3:\n      return 3\n    case GL_FLOAT_VEC4:\n    case GL_INT_VEC4:\n    case GL_BOOL_VEC4:\n      return 4\n    default:\n      return 1\n  }\n}\n\nfunction setUniformString (gl, type, location, value) {\n  var infix\n  var separator = ','\n  switch (type) {\n    case GL_FLOAT:\n      infix = '1f'\n      break\n    case GL_FLOAT_VEC2:\n      infix = '2fv'\n      break\n    case GL_FLOAT_VEC3:\n      infix = '3fv'\n      break\n    case GL_FLOAT_VEC4:\n      infix = '4fv'\n      break\n    case GL_BOOL:\n    case GL_INT:\n      infix = '1i'\n      break\n    case GL_BOOL_VEC2:\n    case GL_INT_VEC2:\n      infix = '2iv'\n      break\n    case GL_BOOL_VEC3:\n    case GL_INT_VEC3:\n      infix = '3iv'\n      break\n    case GL_BOOL_VEC4:\n    case GL_INT_VEC4:\n      infix = '4iv'\n      break\n    case GL_FLOAT_MAT2:\n      infix = 'Matrix2fv'\n      separator = ',false,'\n      break\n    case GL_FLOAT_MAT3:\n      infix = 'Matrix3fv'\n      separator = ',false,'\n      break\n    case GL_FLOAT_MAT4:\n      infix = 'Matrix4fv'\n      separator = ',false,'\n      break\n    default:\n      check.raise('unsupported uniform type')\n  }\n  return gl + '.uniform' + infix + '(' + location + separator + value + ');'\n}\n\nfunction stackTop (x) {\n  return x + '[' + x + '.length-1]'\n}\n\n// Need to process framebuffer first in options list\nfunction optionPriority (a, b) {\n  if (a === 'framebuffer') {\n    return -1\n  }\n  if (a < b) {\n    return -1\n  } else if (a > b) {\n    return 1\n  }\n  return 0\n}\n\nmodule.exports = function reglCompiler (\n  gl,\n  stringStore,\n  extensions,\n  limits,\n  bufferState,\n  elementState,\n  textureState,\n  framebufferState,\n  glState,\n  uniformState,\n  attributeState,\n  shaderState,\n  drawState,\n  frameState,\n  reglPoll) {\n  var contextState = glState.contextState\n\n  var blendEquations = {\n    'add': 32774,\n    'subtract': 32778,\n    'reverse subtract': 32779\n  }\n  if (extensions.ext_blend_minmax) {\n    blendEquations.min = GL_MIN_EXT\n    blendEquations.max = GL_MAX_EXT\n  }\n\n  var drawCallCounter = 0\n\n  // ===================================================\n  // ===================================================\n  // SHADER SINGLE DRAW OPERATION\n  // ===================================================\n  // ===================================================\n  function compileShaderDraw (program) {\n    var env = createEnvironment()\n    var link = env.link\n    var draw = env.proc('draw')\n    var def = draw.def\n\n    var GL = link(gl)\n    var PROGRAM = link(program.program)\n    var BIND_ATTRIBUTE = link(attributeState.bind)\n    var DRAW_STATE = {\n      count: link(drawState.count),\n      offset: link(drawState.offset),\n      instances: link(drawState.instances),\n      primitive: link(drawState.primitive)\n    }\n    var ELEMENT_STATE = link(elementState.elements)\n    var TEXTURE_UNIFORMS = []\n\n    // bind the program\n    draw(GL, '.useProgram(', PROGRAM, ');')\n\n    // set up attribute state\n    program.attributes.forEach(function (attribute) {\n      var STACK = link(attributeState.def(attribute.name))\n      draw(BIND_ATTRIBUTE, '(',\n        attribute.location, ',',\n        link(attributeState.bindings[attribute.location]), ',',\n        STACK, ',',\n        typeLength(attribute.info.type), ');')\n    })\n\n    // set up uniforms\n    program.uniforms.forEach(function (uniform) {\n      var LOCATION = link(uniform.location)\n      var STACK = link(uniformState.def(uniform.name))\n      var TOP = STACK + '[' + STACK + '.length-1]'\n      var type = uniform.info.type\n      if (type === GL_SAMPLER_2D || type === GL_SAMPLER_CUBE) {\n        var TEX_VALUE = def(TOP + '._texture')\n        TEXTURE_UNIFORMS.push(TEX_VALUE)\n        draw(setUniformString(GL, GL_INT, LOCATION, TEX_VALUE + '.bind()'))\n      } else {\n        draw(setUniformString(GL, type, LOCATION, TOP))\n      }\n    })\n\n    // unbind textures immediately\n    TEXTURE_UNIFORMS.forEach(function (TEX_VALUE) {\n      draw(TEX_VALUE, '.unbind();')\n    })\n\n    // Execute draw command\n    var CUR_PRIMITIVE = def(stackTop(DRAW_STATE.primitive))\n    var CUR_COUNT = def(stackTop(DRAW_STATE.count))\n    var CUR_OFFSET = def(stackTop(DRAW_STATE.offset))\n    var CUR_ELEMENTS = def(stackTop(ELEMENT_STATE))\n\n    // Only execute draw command if number elements is > 0\n    draw('if(', CUR_COUNT, '){')\n\n    var instancing = extensions.angle_instanced_arrays\n    if (instancing) {\n      var CUR_INSTANCES = def(stackTop(DRAW_STATE.instances))\n      var INSTANCE_EXT = link(instancing)\n      draw(\n        'if(', CUR_ELEMENTS, '){',\n        CUR_ELEMENTS, '.bind();',\n        'if(', CUR_INSTANCES, '>0){',\n        INSTANCE_EXT, '.drawElementsInstancedANGLE(',\n        CUR_PRIMITIVE, ',',\n        CUR_COUNT, ',',\n        CUR_ELEMENTS, '.type,',\n        CUR_OFFSET, ',',\n        CUR_INSTANCES, ');}else{',\n        GL, '.drawElements(',\n        CUR_PRIMITIVE, ',',\n        CUR_COUNT, ',',\n        CUR_ELEMENTS, '.type,',\n        CUR_OFFSET, ');}',\n        '}else if(', CUR_INSTANCES, '>0){',\n        INSTANCE_EXT, '.drawArraysInstancedANGLE(',\n        CUR_PRIMITIVE, ',',\n        CUR_OFFSET, ',',\n        CUR_COUNT, ',',\n        CUR_INSTANCES, ');}else{',\n        GL, '.drawArrays(',\n        CUR_PRIMITIVE, ',',\n        CUR_OFFSET, ',',\n        CUR_COUNT, ');}}')\n    } else {\n      draw(\n        'if(', CUR_ELEMENTS, '){',\n        CUR_ELEMENTS, '.bind();',\n        GL, '.drawElements(',\n        CUR_PRIMITIVE, ',',\n        CUR_COUNT, ',',\n        CUR_ELEMENTS, '.type,',\n        CUR_OFFSET, ');',\n        '}else{',\n        GL, '.drawArrays(',\n        CUR_PRIMITIVE, ',',\n        CUR_OFFSET, ',',\n        CUR_COUNT, ');}}')\n    }\n\n    return env.compile().draw\n  }\n\n  // ===================================================\n  // ===================================================\n  // BATCH DRAW OPERATION\n  // ===================================================\n  // ===================================================\n  function compileBatch (\n    program, options, uniforms, attributes, staticOptions) {\n    // -------------------------------\n    // code generation helpers\n    // -------------------------------\n    var env = createEnvironment()\n    var link = env.link\n    var batch = env.proc('batch')\n    var exit = env.block()\n    var def = batch.def\n    var arg = batch.arg\n\n    // -------------------------------\n    // regl state\n    // -------------------------------\n    var GL = link(gl)\n    var PROGRAM = link(program.program)\n    var BIND_ATTRIBUTE = link(attributeState.bind)\n    var BIND_ATTRIBUTE_RECORD = link(attributeState.bindRecord)\n    var FRAME_STATE = link(frameState)\n    var FRAMEBUFFER_STATE = link(framebufferState)\n    var DRAW_STATE = {\n      count: link(drawState.count),\n      offset: link(drawState.offset),\n      instances: link(drawState.instances),\n      primitive: link(drawState.primitive)\n    }\n    var CONTEXT_STATE = {}\n    var ELEMENTS = link(elementState.elements)\n    var CUR_COUNT = def(stackTop(DRAW_STATE.count))\n    var CUR_OFFSET = def(stackTop(DRAW_STATE.offset))\n    var CUR_PRIMITIVE = def(stackTop(DRAW_STATE.primitive))\n    var CUR_ELEMENTS = def(stackTop(ELEMENTS))\n    var CUR_INSTANCES\n    var INSTANCE_EXT\n    var instancing = extensions.angle_instanced_arrays\n    if (instancing) {\n      CUR_INSTANCES = def(stackTop(DRAW_STATE.instances))\n      INSTANCE_EXT = link(instancing)\n    }\n    var hasDynamicElements = 'elements' in options\n\n    function linkContext (x) {\n      var result = CONTEXT_STATE[x]\n      if (result) {\n        return result\n      }\n      result = CONTEXT_STATE[x] = link(contextState[x])\n      return result\n    }\n\n    // -------------------------------\n    // batch/argument vars\n    // -------------------------------\n    var NUM_ARGS = arg()\n    var ARGS = arg()\n    var ARG = def()\n    var BATCH_ID = def()\n\n    // -------------------------------\n    // load a dynamic variable\n    // -------------------------------\n    var dynamicVars = {}\n    function dyn (x) {\n      var id = x.id\n      var result = dynamicVars[id]\n      if (result) {\n        return result\n      }\n      if (x.func) {\n        result = batch.def(\n          link(x.data), '(', ARG, ',', BATCH_ID, ',', FRAME_STATE, ')')\n      } else {\n        result = batch.def(ARG, '.', x.data)\n      }\n      dynamicVars[id] = result\n      return result\n    }\n\n    // -------------------------------\n    // retrieves the first name-matching record from an ActiveInfo list\n    // -------------------------------\n    function findInfo (list, name) {\n      for (var i = 0; i < list.length; ++i) {\n        if (list[i].name === name) {\n          return list[i]\n        }\n      }\n      return null\n    }\n\n    // -------------------------------\n    // bind shader\n    // -------------------------------\n    batch(GL, '.useProgram(', PROGRAM, ');')\n\n    // -------------------------------\n    // set static uniforms\n    // -------------------------------\n    program.uniforms.forEach(function (uniform) {\n      if (uniform.name in uniforms) {\n        return\n      }\n      var LOCATION = link(uniform.location)\n      var STACK = link(uniformState.def(uniform.name))\n      var TOP = STACK + '[' + STACK + '.length-1]'\n      var type = uniform.info.type\n      if (type === GL_SAMPLER_2D || type === GL_SAMPLER_CUBE) {\n        var TEX_VALUE = def(TOP + '._texture')\n        batch(setUniformString(GL, GL_INT, LOCATION, TEX_VALUE + '.bind()'))\n        exit(TEX_VALUE, '.unbind();')\n      } else {\n        batch(setUniformString(GL, type, LOCATION, TOP))\n      }\n    })\n\n    // -------------------------------\n    // set static attributes\n    // -------------------------------\n    program.attributes.forEach(function (attribute) {\n      if (attribute.name in attributes) {\n        return\n      }\n      var STACK = link(attributeState.def(attribute.name))\n      batch(BIND_ATTRIBUTE, '(',\n        attribute.location, ',',\n        link(attributeState.bindings[attribute.location]), ',',\n        STACK, ',',\n        typeLength(attribute.info.type), ');')\n    })\n\n    // -------------------------------\n    // set static element buffer\n    // -------------------------------\n    if (!hasDynamicElements) {\n      batch(\n        'if(', CUR_ELEMENTS, ')',\n        GL, '.bindBuffer(', GL_ELEMENT_ARRAY_BUFFER, ',', CUR_ELEMENTS, '.buffer.buffer);')\n    }\n\n    // -------------------------------\n    // loop over all arguments\n    // -------------------------------\n    batch(\n      'for(', BATCH_ID, '=0;', BATCH_ID, '<', NUM_ARGS, ';++', BATCH_ID, '){',\n      ARG, '=', ARGS, '[', BATCH_ID, '];')\n\n    // -------------------------------\n    // set dynamic flags\n    // -------------------------------\n    Object.keys(options).sort(optionPriority).forEach(function (option) {\n      var VALUE = dyn(options[option])\n\n      function setCap (flag) {\n        batch(\n          'if(', VALUE, '){',\n          GL, '.enable(', flag, ');}else{',\n          GL, '.disable(', flag, ');}')\n      }\n\n      switch (option) {\n        case 'framebuffer':\n          var VIEWPORT_STATE = linkContext('viewport')\n          var SCISSOR_STATE = linkContext('scissor.box')\n          batch(\n            'if(', FRAMEBUFFER_STATE, '.push(',\n            VALUE, '&&', VALUE, '._framebuffer)){',\n            FRAMEBUFFER_STATE, '.poll();',\n            VIEWPORT_STATE, '.setDirty();',\n            SCISSOR_STATE, '.setDirty();',\n            '}')\n          break\n\n        // Caps\n        case 'cull.enable':\n          setCap(GL_CULL_FACE)\n          break\n        case 'blend.enable':\n          setCap(GL_BLEND)\n          break\n        case 'dither':\n          setCap(GL_DITHER)\n          break\n        case 'stencil.enable':\n          setCap(GL_STENCIL_TEST)\n          break\n        case 'depth.enable':\n          setCap(GL_DEPTH_TEST)\n          break\n        case 'scissor.enable':\n          setCap(GL_SCISSOR_TEST)\n          break\n        case 'polygonOffset.enable':\n          setCap(GL_POLYGON_OFFSET_FILL)\n          break\n        case 'sample.alpha':\n          setCap(GL_SAMPLE_ALPHA_TO_COVERAGE)\n          break\n        case 'sample.enable':\n          setCap(GL_SAMPLE_COVERAGE)\n          break\n\n        case 'depth.mask':\n          batch(GL, '.depthMask(', VALUE, ');')\n          break\n\n        case 'depth.func':\n          var DEPTH_FUNCS = link(compareFuncs)\n          batch(GL, '.depthFunc(', DEPTH_FUNCS, '[', VALUE, ']);')\n          break\n\n        case 'depth.range':\n          batch(GL, '.depthRange(', VALUE, '[0],', VALUE, '[1]);')\n          break\n\n        case 'blend.color':\n          batch(GL, '.blendColor(',\n            VALUE, '[0],',\n            VALUE, '[1],',\n            VALUE, '[2],',\n            VALUE, '[3]);')\n          break\n\n        case 'blend.equation':\n          var BLEND_EQUATIONS = link(blendEquations)\n          batch(\n            'if(typeof ', VALUE, '===\"string\"){',\n            GL, '.blendEquation(', BLEND_EQUATIONS, '[', VALUE, ']);',\n            '}else{',\n            GL, '.blendEquationSeparate(',\n            BLEND_EQUATIONS, '[', VALUE, '.rgb],',\n            BLEND_EQUATIONS, '[', VALUE, '.alpha]);',\n            '}')\n          break\n\n        case 'blend.func':\n          var BLEND_FUNCS = link(blendFuncs)\n          batch(\n            GL, '.blendFuncSeparate(',\n            BLEND_FUNCS,\n            '[\"srcRGB\" in ', VALUE, '?', VALUE, '.srcRGB:', VALUE, '.src],',\n            BLEND_FUNCS,\n            '[\"dstRGB\" in ', VALUE, '?', VALUE, '.dstRGB:', VALUE, '.dst],',\n            BLEND_FUNCS,\n            '[\"srcAlpha\" in ', VALUE, '?', VALUE, '.srcAlpha:', VALUE, '.src],',\n            BLEND_FUNCS,\n            '[\"dstAlpha\" in ', VALUE, '?', VALUE, '.dstAlpha:', VALUE, '.dst]);')\n          break\n\n        case 'stencil.mask':\n          batch(GL, '.stencilMask(', VALUE, ');')\n          break\n\n        case 'stencil.func':\n          var STENCIL_FUNCS = link(compareFuncs)\n          batch(GL, '.stencilFunc(',\n            STENCIL_FUNCS, '[', VALUE, '.cmp||\"always\"],',\n            VALUE, '.ref|0,',\n            '\"mask\" in ', VALUE, '?', VALUE, '.mask:-1);')\n          break\n\n        case 'stencil.opFront':\n        case 'stencil.opBack':\n          var STENCIL_OPS = link(stencilOps)\n          batch(GL, '.stencilOpSeparate(',\n            option === 'stencil.opFront' ? GL_FRONT : GL_BACK, ',',\n            STENCIL_OPS, '[', VALUE, '.fail||\"keep\"],',\n            STENCIL_OPS, '[', VALUE, '.zfail||\"keep\"],',\n            STENCIL_OPS, '[', VALUE, '.pass||\"keep\"]);')\n          break\n\n        case 'polygonOffset.offset':\n          batch(GL, '.polygonOffset(',\n            VALUE, '.factor||0,',\n            VALUE, '.units||0);')\n          break\n\n        case 'cull.face':\n          batch(GL, '.cullFace(',\n            VALUE, '===\"front\"?', GL_FRONT, ':', GL_BACK, ');')\n          break\n\n        case 'lineWidth':\n          batch(GL, '.lineWidth(', VALUE, ');')\n          break\n\n        case 'frontFace':\n          batch(GL, '.frontFace(',\n            VALUE, '===\"cw\"?', GL_CW, ':', GL_CCW, ');')\n          break\n\n        case 'colorMask':\n          batch(GL, '.colorMask(',\n            VALUE, '[0],',\n            VALUE, '[1],',\n            VALUE, '[2],',\n            VALUE, '[3]);')\n          break\n\n        case 'sample.coverage':\n          batch(GL, '.sampleCoverage(',\n            VALUE, '.value,',\n            VALUE, '.invert);')\n          break\n\n        case 'scissor.box':\n        case 'viewport':\n          var BOX_STATE = linkContext(option)\n          batch(BOX_STATE, '.push(',\n            VALUE, '.x||0,',\n            VALUE, '.y||0,',\n            VALUE, '.w||-1,',\n            VALUE, '.h||-1);')\n          break\n\n        case 'primitive':\n        case 'offset':\n        case 'count':\n        case 'elements':\n        case 'instances':\n          break\n\n        default:\n          check.raise('unsupported option for batch', option)\n      }\n    })\n\n    // update viewport/scissor box state and restore framebuffer\n    if ('viewport' in options || 'framebuffer' in options) {\n      batch(linkContext('viewport'), '.poll();')\n    }\n    if ('scissor.box' in options || 'framebuffer' in options) {\n      batch(linkContext('scissor.box'), '.poll();')\n    }\n    if ('framebuffer' in options) {\n      batch(FRAMEBUFFER_STATE, '.pop();')\n    }\n\n    // -------------------------------\n    // set dynamic uniforms\n    // -------------------------------\n    var programUniforms = program.uniforms\n    var DYNAMIC_TEXTURES = []\n    Object.keys(uniforms).forEach(function (uniform) {\n      var data = findInfo(programUniforms, uniform)\n      if (!data) {\n        return\n      }\n      var TYPE = data.info.type\n      var LOCATION = link(data.location)\n      var VALUE = dyn(uniforms[uniform])\n      if (data.info.type === GL_SAMPLER_2D ||\n          data.info.type === GL_SAMPLER_CUBE) {\n        var TEX_VALUE = def(VALUE + '._texture')\n        DYNAMIC_TEXTURES.push(TEX_VALUE)\n        batch(setUniformString(GL, GL_INT, LOCATION, TEX_VALUE + '.bind()'))\n      } else {\n        batch(setUniformString(GL, TYPE, LOCATION, VALUE))\n      }\n    })\n    DYNAMIC_TEXTURES.forEach(function (VALUE) {\n      batch(VALUE, '.unbind();')\n    })\n\n    // -------------------------------\n    // set dynamic attributes\n    // -------------------------------\n    var programAttributes = program.attributes\n    Object.keys(attributes).forEach(function (attribute) {\n      var data = findInfo(programAttributes, attribute)\n      if (!data) {\n        return\n      }\n      var BOX = link(attributeState.box(attribute))\n      var ATTRIB_VALUE = dyn(attributes[attribute])\n      var RECORD = def(BOX + '.alloc(' + ATTRIB_VALUE + ')')\n      batch(BIND_ATTRIBUTE_RECORD, '(',\n        data.location, ',',\n        link(attributeState.bindings[data.location]), ',',\n        RECORD, ',',\n        typeLength(data.info.type), ');')\n      exit(BOX, '.free(', RECORD, ');')\n    })\n\n    // -------------------------------\n    // set dynamic attributes\n    // -------------------------------\n\n    if (options.count) {\n      batch(CUR_COUNT, '=', dyn(options.count), ';')\n    }\n    if (options.offset) {\n      batch(CUR_OFFSET, '=', dyn(options.offset), ';')\n    }\n    if (options.primitive) {\n      batch(\n        CUR_PRIMITIVE, '=', link(primTypes), '[', dyn(options.primitive), '];')\n    }\n    if (instancing && options.instances) {\n      batch(CUR_INSTANCES, '=', dyn(options.instances), ';')\n    }\n\n    function useElementOption (x) {\n      return hasDynamicElements && !(x in options || x in staticOptions)\n    }\n    if (hasDynamicElements) {\n      var dynElements = dyn(options.elements)\n      batch(CUR_ELEMENTS, '=',\n        dynElements, '?', dynElements, '._elements:null;')\n    }\n    if (useElementOption('offset')) {\n      batch(CUR_OFFSET, '=0;')\n    }\n\n    // Emit draw command\n    batch('if(', CUR_ELEMENTS, '){')\n    if (useElementOption('count')) {\n      batch(CUR_COUNT, '=', CUR_ELEMENTS, '.vertCount;')\n    }\n    batch('if(', CUR_COUNT, '>0){')\n    if (useElementOption('primitive')) {\n      batch(CUR_PRIMITIVE, '=', CUR_ELEMENTS, '.primType;')\n    }\n    if (hasDynamicElements) {\n      batch(\n        GL,\n        '.bindBuffer(',\n        GL_ELEMENT_ARRAY_BUFFER, ',',\n        CUR_ELEMENTS, '.buffer.buffer);')\n    }\n    if (instancing) {\n      batch(\n        'if(', CUR_INSTANCES, '>0){',\n        INSTANCE_EXT, '.drawElementsInstancedANGLE(',\n        CUR_PRIMITIVE, ',',\n        CUR_COUNT, ',',\n        CUR_ELEMENTS, '.type,',\n        CUR_OFFSET, ',',\n        CUR_INSTANCES, ');}else ')\n    }\n    batch(\n      GL, '.drawElements(',\n      CUR_PRIMITIVE, ',',\n      CUR_COUNT, ',',\n      CUR_ELEMENTS, '.type,',\n      CUR_OFFSET, ');')\n    batch('}}else if(', CUR_COUNT, '>0){')\n    if (!useElementOption('count')) {\n      if (useElementOption('primitive')) {\n        batch(CUR_PRIMITIVE, '=', GL_TRIANGLES, ';')\n      }\n      if (instancing) {\n        batch(\n          'if(', CUR_INSTANCES, '>0){',\n          INSTANCE_EXT, '.drawArraysInstancedANGLE(',\n          CUR_PRIMITIVE, ',',\n          CUR_OFFSET, ',',\n          CUR_COUNT, ',',\n          CUR_INSTANCES, ');}else{')\n      }\n      batch(\n        GL, '.drawArrays(',\n        CUR_PRIMITIVE, ',',\n        CUR_OFFSET, ',',\n        CUR_COUNT, ');')\n      if (instancing) {\n        batch('}')\n      }\n    }\n    batch('}}', exit)\n\n    // -------------------------------\n    // compile and return\n    // -------------------------------\n    return env.compile().batch\n  }\n\n  // ===================================================\n  // ===================================================\n  // MAIN DRAW COMMAND\n  // ===================================================\n  // ===================================================\n  function compileCommand (\n    staticOptions, staticUniforms, staticAttributes,\n    dynamicOptions, dynamicUniforms, dynamicAttributes,\n    hasDynamic) {\n    // Create code generation environment\n    var env = createEnvironment()\n    var link = env.link\n    var block = env.block\n    var proc = env.proc\n\n    var callId = drawCallCounter++\n\n    // -------------------------------\n    // Common state variables\n    // -------------------------------\n    var GL_POLL = link(reglPoll)\n    var SHADER_STATE = link(shaderState)\n    var FRAMEBUFFER_STATE = link(framebufferState)\n    var DRAW_STATE = {\n      count: link(drawState.count),\n      offset: link(drawState.offset),\n      instances: link(drawState.instances),\n      primitive: link(drawState.primitive)\n    }\n    var ELEMENT_STATE = link(elementState.elements)\n    var PRIM_TYPES = link(primTypes)\n    var COMPARE_FUNCS = link(compareFuncs)\n    var STENCIL_OPS = link(stencilOps)\n\n    var CONTEXT_STATE = {}\n    function linkContext (x) {\n      var result = CONTEXT_STATE[x]\n      if (result) {\n        return result\n      }\n      result = CONTEXT_STATE[x] = link(contextState[x])\n      return result\n    }\n\n    // ==========================================================\n    // STATIC STATE\n    // ==========================================================\n    // Code blocks for the static sections\n    var entry = block()\n    var exit = block()\n\n    // -------------------------------\n    // update default context state variables\n    // -------------------------------\n    function handleStaticOption (param, value) {\n      var STATE_STACK = linkContext(param)\n      entry(STATE_STACK, '.push(', value, ');')\n      exit(STATE_STACK, '.pop();')\n    }\n\n    Object.keys(staticOptions).sort(optionPriority).forEach(function (param) {\n      var value = staticOptions[param]\n      switch (param) {\n        case 'frag':\n        case 'vert':\n          var shaderId = stringStore.id(value)\n          shaderState.shader(\n            param === 'frag' ? GL_FRAGMENT_SHADER : GL_VERTEX_SHADER,\n            shaderId)\n          entry(SHADER_STATE, '.', param, '.push(', shaderId, ');')\n          exit(SHADER_STATE, '.', param, '.pop();')\n          break\n\n        case 'framebuffer':\n          var fbo = framebufferState.getFramebuffer(value)\n          check(value === null || fbo, 'invalid framebuffer object')\n          var VIEWPORT_STATE = linkContext('viewport')\n          var SCISSOR_STATE = linkContext('scissor.box')\n          entry('if(', FRAMEBUFFER_STATE, '.push(', link(\n            value && value._framebuffer), ')){',\n            VIEWPORT_STATE, '.setDirty();',\n            SCISSOR_STATE, '.setDirty();',\n            '}')\n          exit('if(', FRAMEBUFFER_STATE, '.pop()){',\n            VIEWPORT_STATE, '.setDirty();',\n            SCISSOR_STATE, '.setDirty();',\n            '}')\n          break\n\n        // Update draw state\n        case 'count':\n        case 'offset':\n        case 'instances':\n          check.nni(value, param)\n          entry(DRAW_STATE[param], '.push(', value, ');')\n          exit(DRAW_STATE[param], '.pop();')\n          break\n\n        // Update primitive type\n        case 'primitive':\n          check.parameter(value, primTypes, 'not a valid drawing primitive')\n          var primType = primTypes[value]\n          entry(DRAW_STATE.primitive, '.push(', primType, ');')\n          exit(DRAW_STATE.primitive, '.pop();')\n          break\n\n        // Update element buffer\n        case 'elements':\n          var elements = elementState.getElements(value)\n          var hasPrimitive = !('primitive' in staticOptions)\n          var hasCount = !('count' in staticOptions)\n          if (elements) {\n            var ELEMENTS = link(elements)\n            entry(ELEMENT_STATE, '.push(', ELEMENTS, ');')\n            if (hasPrimitive) {\n              entry(DRAW_STATE.primitive, '.push(', ELEMENTS, '.primType);')\n            }\n            if (hasCount) {\n              entry(DRAW_STATE.count, '.push(', ELEMENTS, '.vertCount);')\n            }\n          } else {\n            entry(ELEMENT_STATE, '.push(null);')\n            if (hasPrimitive) {\n              entry(DRAW_STATE.primitive, '.push(', GL_TRIANGLES, ');')\n            }\n            if (hasCount) {\n              entry(DRAW_STATE.count, '.push(0);')\n            }\n          }\n          if (hasPrimitive) {\n            exit(DRAW_STATE.primitive, '.pop();')\n          }\n          if (hasCount) {\n            exit(DRAW_STATE.count, '.pop();')\n          }\n          if (!('offset' in staticOptions)) {\n            entry(DRAW_STATE.offset, '.push(0);')\n            exit(DRAW_STATE.offset, '.pop();')\n          }\n          exit(ELEMENT_STATE, '.pop();')\n          break\n\n        case 'cull.enable':\n        case 'blend.enable':\n        case 'dither':\n        case 'stencil.enable':\n        case 'depth.enable':\n        case 'scissor.enable':\n        case 'polygonOffset.enable':\n        case 'sample.alpha':\n        case 'sample.enable':\n        case 'depth.mask':\n          check.type(value, 'boolean', param)\n          handleStaticOption(param, value)\n          break\n\n        case 'depth.func':\n          check.parameter(value, compareFuncs, param)\n          handleStaticOption(param, compareFuncs[value])\n          break\n\n        case 'depth.range':\n          check(\n            Array.isArray(value) &&\n            value.length === 2 &&\n            value[0] <= value[1],\n            'depth range is 2d array')\n          var DEPTH_RANGE_STACK = linkContext(param)\n          entry(DEPTH_RANGE_STACK, '.push(', value[0], ',', value[1], ');')\n          exit(DEPTH_RANGE_STACK, '.pop();')\n          break\n\n        case 'blend.func':\n          var BLEND_FUNC_STACK = linkContext(param)\n          check.type(value, 'object', 'blend func must be an object')\n          var srcRGB = ('srcRGB' in value ? value.srcRGB : value.src)\n          var srcAlpha = ('srcAlpha' in value ? value.srcAlpha : value.src)\n          var dstRGB = ('dstRGB' in value ? value.dstRGB : value.dst)\n          var dstAlpha = ('dstAlpha' in value ? value.dstAlpha : value.dst)\n          check.parameter(srcRGB, blendFuncs)\n          check.parameter(srcAlpha, blendFuncs)\n          check.parameter(dstRGB, blendFuncs)\n          check.parameter(dstAlpha, blendFuncs)\n          entry(BLEND_FUNC_STACK, '.push(',\n            blendFuncs[srcRGB], ',',\n            blendFuncs[dstRGB], ',',\n            blendFuncs[srcAlpha], ',',\n            blendFuncs[dstAlpha], ');')\n          exit(BLEND_FUNC_STACK, '.pop();')\n          break\n\n        case 'blend.equation':\n          var BLEND_EQUATION_STACK = linkContext(param)\n          if (typeof value === 'string') {\n            check.parameter(value, blendEquations, 'invalid blend equation')\n            entry(BLEND_EQUATION_STACK,\n              '.push(',\n              blendEquations[value], ',',\n              blendEquations[value], ');')\n          } else if (typeof value === 'object') {\n            check.parameter(\n              value.rgb, blendEquations, 'invalid blend equation rgb')\n            check.parameter(\n              value.alpha, blendEquations, 'invalid blend equation alpha')\n            entry(BLEND_EQUATION_STACK,\n              '.push(',\n              blendEquations[value.rgb], ',',\n              blendEquations[value.alpha], ');')\n          } else {\n            check.raise('invalid blend equation')\n          }\n          exit(BLEND_EQUATION_STACK, '.pop();')\n          break\n\n        case 'blend.color':\n          check(\n            Array.isArray(value) &&\n            value.length === 4,\n            'blend color is a 4d array')\n          var BLEND_COLOR_STACK = linkContext(param)\n          entry(BLEND_COLOR_STACK,\n            '.push(',\n            value[0], ',',\n            value[1], ',',\n            value[2], ',',\n            value[3], ');')\n          exit(BLEND_COLOR_STACK, '.pop();')\n          break\n\n        case 'stencil.mask':\n          check.type(value, 'number', 'stencil mask must be an integer')\n          var STENCIL_MASK_STACK = linkContext(param)\n          entry(STENCIL_MASK_STACK, '.push(', value, ');')\n          exit(STENCIL_MASK_STACK, '.pop();')\n          break\n\n        case 'stencil.func':\n          check.type(value, 'object', 'stencil func must be an object')\n          var cmp = value.cmp || 'keep'\n          var ref = value.ref || 0\n          var mask = 'mask' in value ? value.mask : -1\n          check.parameter(cmp, compareFuncs, 'invalid stencil func cmp')\n          check.type(ref, 'number', 'stencil func ref')\n          check.type(mask, 'number', 'stencil func mask')\n          var STENCIL_FUNC_STACK = linkContext(param)\n          entry(STENCIL_FUNC_STACK, '.push(',\n            compareFuncs[cmp], ',',\n            ref, ',',\n            mask, ');')\n          exit(STENCIL_FUNC_STACK, '.pop();')\n          break\n\n        case 'stencil.opFront':\n        case 'stencil.opBack':\n          check.type(value, 'object', param)\n          var fail = value.fail || 'keep'\n          var zfail = value.zfail || 'keep'\n          var pass = value.pass || 'keep'\n          check.parameter(fail, stencilOps, param)\n          check.parameter(zfail, stencilOps, param)\n          check.parameter(pass, stencilOps, param)\n          var STENCIL_OP_STACK = linkContext(param)\n          entry(STENCIL_OP_STACK, '.push(',\n            stencilOps[fail], ',',\n            stencilOps[zfail], ',',\n            stencilOps[pass], ');')\n          exit(STENCIL_OP_STACK, '.pop();')\n          break\n\n        case 'polygonOffset.offset':\n          check.type(value, 'object', param)\n          var factor = value.factor || 0\n          var units = value.units || 0\n          check.type(factor, 'number', 'offset.factor')\n          check.type(units, 'number', 'offset.units')\n          var POLYGON_OFFSET_STACK = linkContext(param)\n          entry(POLYGON_OFFSET_STACK, '.push(',\n            factor, ',', units, ');')\n          exit(POLYGON_OFFSET_STACK, '.pop();')\n          break\n\n        case 'cull.face':\n          var face = 0\n          if (value === 'front') {\n            face = GL_FRONT\n          } else if (value === 'back') {\n            face = GL_BACK\n          }\n          check(!!face, 'cull.face')\n          var CULL_FACE_STACK = linkContext(param)\n          entry(CULL_FACE_STACK, '.push(', face, ');')\n          exit(CULL_FACE_STACK, '.pop();')\n          break\n\n        case 'lineWidth':\n          var lineWidthDims = limits.lineWidthDims\n          check(\n            typeof value === 'number' &&\n            value >= lineWidthDims[0] &&\n            value <= lineWidthDims[1],\n            'invalid line width, must positive number between ' +\n            lineWidthDims[0] + ' and ' + lineWidthDims[1])\n          handleStaticOption(param, value)\n          break\n\n        case 'frontFace':\n          var orientation = 0\n          if (value === 'cw') {\n            orientation = GL_CW\n          } else if (value === 'ccw') {\n            orientation = GL_CCW\n          }\n          check(!!orientation, 'frontFace')\n          var FRONT_FACE_STACK = linkContext(param)\n          entry(FRONT_FACE_STACK, '.push(', orientation, ');')\n          exit(FRONT_FACE_STACK, '.pop();')\n          break\n\n        case 'colorMask':\n          check(Array.isArray(value) && value.length === 4, 'color mask must be length 4 array')\n          var COLOR_MASK_STACK = linkContext(param)\n          entry(COLOR_MASK_STACK, '.push(',\n            value.map(function (v) { return !!v }).join(),\n            ');')\n          exit(COLOR_MASK_STACK, '.pop();')\n          break\n\n        case 'sample.coverage':\n          check.type(value, 'object', param)\n          var sampleValue = 'value' in value ? value.value : 1\n          var sampleInvert = !!value.invert\n          check(\n            typeof sampleValue === 'number' &&\n            sampleValue >= 0 && sampleValue <= 1,\n            'sample value')\n          var SAMPLE_COVERAGE_STACK = linkContext(param)\n          entry(SAMPLE_COVERAGE_STACK, '.push(',\n            sampleValue, ',', sampleInvert, ');')\n          exit(SAMPLE_COVERAGE_STACK, '.pop();')\n          break\n\n        case 'viewport':\n        case 'scissor.box':\n          check(typeof value === 'object' && value, param + ' is an object')\n          var X = value.x || 0\n          var Y = value.y || 0\n          var W = -1\n          var H = -1\n          check(typeof X === 'number' && X >= 0, param + '.x must be a positive int')\n          check(typeof Y === 'number' && Y >= 0, param + '.y must be a positive int')\n          if ('w' in value) {\n            W = value.w\n            check(typeof W === 'number' && W >= 0, param + '.w must be a positive int')\n          }\n          if ('h' in value) {\n            H = value.h\n            check(typeof H === 'number' && H >= 0, param + '.h must be a positive int')\n          }\n          var BOX_STACK = linkContext(param)\n          entry(BOX_STACK, '.push(', X, ',', Y, ',', W, ',', H, ');')\n          exit(BOX_STACK, '.pop();')\n          break\n\n        default:\n          // TODO Should this just be a warning instead?\n          check.raise('unsupported parameter ' + param)\n          break\n      }\n    })\n\n    // -------------------------------\n    // update static uniforms\n    // -------------------------------\n    Object.keys(staticUniforms).forEach(function (uniform) {\n      var STACK = link(uniformState.def(uniform))\n      var VALUE\n      var value = staticUniforms[uniform]\n      if (typeof value === 'function' && value._reglType) {\n        VALUE = link(value)\n      } else if (Array.isArray(value)) {\n        VALUE = link(value.slice())\n      } else {\n        VALUE = +value\n      }\n      entry(STACK, '.push(', VALUE, ');')\n      exit(STACK, '.pop();')\n    })\n\n    // -------------------------------\n    // update default attributes\n    // -------------------------------\n    Object.keys(staticAttributes).forEach(function (attribute) {\n      var data = staticAttributes[attribute]\n      var ATTRIBUTE = link(attributeState.def(attribute))\n      var BOX = link(attributeState.box(attribute).alloc(data))\n      entry(ATTRIBUTE, '.records.push(', BOX, ');')\n      exit(ATTRIBUTE, '.records.pop();')\n    })\n\n    // ==========================================================\n    // DYNAMIC STATE (for scope and draw)\n    // ==========================================================\n    // Generated code blocks for dynamic state flags\n    var dynamicEntry = env.block()\n    var dynamicExit = env.block()\n\n    var FRAMESTATE\n    var DYNARGS\n    if (hasDynamic) {\n      FRAMESTATE = link(frameState)\n      DYNARGS = entry.def()\n    }\n\n    var dynamicVars = {}\n    function dyn (x) {\n      var id = x.id\n      var result = dynamicVars[id]\n      if (result) {\n        return result\n      }\n      if (x.func) {\n        result = dynamicEntry.def(\n          link(x.data), '(', DYNARGS, ',0,', FRAMESTATE, ')')\n      } else {\n        result = dynamicEntry.def(DYNARGS, '.', x.data)\n      }\n      dynamicVars[id] = result\n      return result\n    }\n\n    // -------------------------------\n    // dynamic context state variables\n    // -------------------------------\n    Object.keys(dynamicOptions).sort(optionPriority).forEach(function (param) {\n      // Link in dynamic variable\n      var variable = dyn(dynamicOptions[param])\n\n      switch (param) {\n        case 'framebuffer':\n          var VIEWPORT_STATE = linkContext('viewport')\n          var SCISSOR_STATE = linkContext('scissor.box')\n          dynamicEntry('if(',\n            FRAMEBUFFER_STATE, '.push(',\n            variable, '&&', variable, '._framebuffer)){',\n            VIEWPORT_STATE, '.setDirty();',\n            SCISSOR_STATE, '.setDirty();',\n            '}')\n          dynamicExit('if(',\n            FRAMEBUFFER_STATE, '.pop()){',\n            VIEWPORT_STATE, '.setDirty();',\n            SCISSOR_STATE, '.setDirty();',\n            '}')\n          break\n\n        case 'cull.enable':\n        case 'blend.enable':\n        case 'dither':\n        case 'stencil.enable':\n        case 'depth.enable':\n        case 'scissor.enable':\n        case 'polygonOffset.enable':\n        case 'sample.alpha':\n        case 'sample.enable':\n        case 'lineWidth':\n        case 'depth.mask':\n          var STATE_STACK = linkContext(param)\n          dynamicEntry(STATE_STACK, '.push(', variable, ');')\n          dynamicExit(STATE_STACK, '.pop();')\n          break\n\n        // Draw calls\n        case 'count':\n        case 'offset':\n        case 'instances':\n          var DRAW_STACK = DRAW_STATE[param]\n          dynamicEntry(DRAW_STACK, '.push(', variable, ');')\n          dynamicExit(DRAW_STACK, '.pop();')\n          break\n\n        case 'primitive':\n          var PRIM_STACK = DRAW_STATE.primitive\n          dynamicEntry(PRIM_STACK, '.push(', PRIM_TYPES, '[', variable, ']);')\n          dynamicExit(PRIM_STACK, '.pop();')\n          break\n\n        case 'depth.func':\n          var DEPTH_FUNC_STACK = linkContext(param)\n          dynamicEntry(DEPTH_FUNC_STACK, '.push(', COMPARE_FUNCS, '[', variable, ']);')\n          dynamicExit(DEPTH_FUNC_STACK, '.pop();')\n          break\n\n        case 'blend.func':\n          var BLEND_FUNC_STACK = linkContext(param)\n          var BLEND_FUNCS = link(blendFuncs)\n          dynamicEntry(\n            BLEND_FUNC_STACK, '.push(',\n            BLEND_FUNCS,\n            '[\"srcRGB\" in ', variable, '?', variable, '.srcRGB:', variable, '.src],',\n            BLEND_FUNCS,\n            '[\"dstRGB\" in ', variable, '?', variable, '.dstRGB:', variable, '.dst],',\n            BLEND_FUNCS,\n            '[\"srcAlpha\" in ', variable, '?', variable, '.srcAlpha:', variable, '.src],',\n            BLEND_FUNCS,\n            '[\"dstAlpha\" in ', variable, '?', variable, '.dstAlpha:', variable, '.dst]);')\n          dynamicExit(BLEND_FUNC_STACK, '.pop();')\n          break\n\n        case 'blend.equation':\n          var BLEND_EQUATION_STACK = linkContext(param)\n          var BLEND_EQUATIONS = link(blendEquations)\n          dynamicEntry(\n            'if(typeof ', variable, '===\"string\"){',\n            BLEND_EQUATION_STACK, '.push(',\n            BLEND_EQUATIONS, '[', variable, '],',\n            BLEND_EQUATIONS, '[', variable, ']);',\n            '}else{',\n            BLEND_EQUATION_STACK, '.push(',\n            BLEND_EQUATIONS, '[', variable, '.rgb],',\n            BLEND_EQUATIONS, '[', variable, '.alpha]);',\n            '}')\n          dynamicExit(BLEND_EQUATION_STACK, '.pop();')\n          break\n\n        case 'blend.color':\n          var BLEND_COLOR_STACK = linkContext(param)\n          dynamicEntry(BLEND_COLOR_STACK, '.push(',\n            variable, '[0],',\n            variable, '[1],',\n            variable, '[2],',\n            variable, '[3]);')\n          dynamicExit(BLEND_COLOR_STACK, '.pop();')\n          break\n\n        case 'stencil.mask':\n          var STENCIL_MASK_STACK = linkContext(param)\n          dynamicEntry(STENCIL_MASK_STACK, '.push(', variable, ');')\n          dynamicExit(STENCIL_MASK_STACK, '.pop();')\n          break\n\n        case 'stencil.func':\n          var STENCIL_FUNC_STACK = linkContext(param)\n          dynamicEntry(STENCIL_FUNC_STACK, '.push(',\n            COMPARE_FUNCS, '[', variable, '.cmp],',\n            variable, '.ref|0,',\n            '\"mask\" in ', variable, '?', variable, '.mask:-1);')\n          dynamicExit(STENCIL_FUNC_STACK, '.pop();')\n          break\n\n        case 'stencil.opFront':\n        case 'stencil.opBack':\n          var STENCIL_OP_STACK = linkContext(param)\n          dynamicEntry(STENCIL_OP_STACK, '.push(',\n            STENCIL_OPS, '[', variable, '.fail||\"keep\"],',\n            STENCIL_OPS, '[', variable, '.zfail||\"keep\"],',\n            STENCIL_OPS, '[', variable, '.pass||\"keep\"]);')\n          dynamicExit(STENCIL_OP_STACK, '.pop();')\n          break\n\n        case 'polygonOffset.offset':\n          var POLYGON_OFFSET_STACK = linkContext(param)\n          dynamicEntry(POLYGON_OFFSET_STACK, '.push(',\n            variable, '.factor||0,',\n            variable, '.units||0);')\n          dynamicExit(POLYGON_OFFSET_STACK, '.pop();')\n          break\n\n        case 'cull.face':\n          var CULL_FACE_STACK = linkContext(param)\n          dynamicEntry(CULL_FACE_STACK, '.push(',\n            variable, '===\"front\"?', GL_FRONT, ':', GL_BACK, ');')\n          dynamicExit(CULL_FACE_STACK, '.pop();')\n          break\n\n        case 'frontFace':\n          var FRONT_FACE_STACK = linkContext(param)\n          dynamicEntry(FRONT_FACE_STACK, '.push(',\n            variable, '===\"cw\"?', GL_CW, ':', GL_CCW, ');')\n          dynamicExit(FRONT_FACE_STACK, '.pop();')\n          break\n\n        case 'colorMask':\n          var COLOR_MASK_STACK = linkContext(param)\n          dynamicEntry(COLOR_MASK_STACK, '.push(',\n            variable, '[0],',\n            variable, '[1],',\n            variable, '[2],',\n            variable, '[3]);')\n          dynamicExit(COLOR_MASK_STACK, '.pop();')\n          break\n\n        case 'sample.coverage':\n          var SAMPLE_COVERAGE_STACK = linkContext(param)\n          dynamicEntry(SAMPLE_COVERAGE_STACK, '.push(',\n            variable, '.value,',\n            variable, '.invert);')\n          dynamicExit(SAMPLE_COVERAGE_STACK, '.pop();')\n          break\n\n        case 'scissor.box':\n        case 'viewport':\n          var BOX_STACK = linkContext(param)\n          dynamicEntry(BOX_STACK, '.push(',\n            variable, '.x||0,',\n            variable, '.y||0,',\n            '\"w\" in ', variable, '?', variable, '.w:-1,',\n            '\"h\" in ', variable, '?', variable, '.h:-1);')\n          dynamicExit(BOX_STACK, '.pop();')\n          break\n\n        case 'elements':\n          var hasPrimitive =\n          !('primitive' in dynamicOptions) &&\n            !('primitive' in staticOptions)\n          var hasCount =\n          !('count' in dynamicOptions) &&\n            !('count' in staticOptions)\n          var hasOffset =\n          !('offset' in dynamicOptions) &&\n            !('offset' in staticOptions)\n          var ELEMENTS = dynamicEntry.def()\n          dynamicEntry(\n            'if(', variable, '){',\n            ELEMENTS, '=', variable, '._elements;',\n            ELEMENT_STATE, '.push(', ELEMENTS, ');',\n            !hasPrimitive ? ''\n              : DRAW_STATE.primitive + '.push(' + ELEMENTS + '.primType);',\n            !hasCount ? ''\n              : DRAW_STATE.count + '.push(' + ELEMENTS + '.vertCount);',\n            !hasOffset ? ''\n              : DRAW_STATE.offset + '.push(' + ELEMENTS + '.offset);',\n            '}else{',\n            ELEMENT_STATE, '.push(null);',\n            '}')\n          dynamicExit(\n            ELEMENT_STATE, '.pop();',\n            'if(', variable, '){',\n            hasPrimitive ? DRAW_STATE.primitive + '.pop();' : '',\n            hasCount ? DRAW_STATE.count + '.pop();' : '',\n            hasOffset ? DRAW_STATE.offset + '.pop();' : '',\n            '}')\n          break\n\n        default:\n          check.raise('unsupported dynamic option: ' + param)\n      }\n    })\n\n    // -------------------------------\n    // dynamic uniforms\n    // -------------------------------\n    Object.keys(dynamicUniforms).forEach(function (uniform) {\n      var STACK = link(uniformState.def(uniform))\n      var VALUE = dyn(dynamicUniforms[uniform])\n      dynamicEntry(STACK, '.push(', VALUE, ');')\n      dynamicExit(STACK, '.pop();')\n    })\n\n    // -------------------------------\n    // dynamic attributes\n    // -------------------------------\n    Object.keys(dynamicAttributes).forEach(function (attribute) {\n      var ATTRIBUTE = link(attributeState.def(attribute))\n      var VALUE = dyn(dynamicAttributes[attribute])\n      var BOX = link(attributeState.box(attribute))\n      dynamicEntry(ATTRIBUTE, '.records.push(',\n        BOX, '.alloc(', VALUE, '));')\n      dynamicExit(BOX, '.free(', ATTRIBUTE, '.records.pop());')\n    })\n\n    // ==========================================================\n    // SCOPE PROCEDURE\n    // ==========================================================\n    var scope = proc('scope')\n    var SCOPE_ARGS = scope.arg()\n    var SCOPE_BODY = scope.arg()\n    scope(entry)\n    if (hasDynamic) {\n      scope(\n        DYNARGS, '=', SCOPE_ARGS, ';',\n        dynamicEntry)\n    }\n    scope(\n      SCOPE_BODY, '();',\n      hasDynamic ? dynamicExit : '',\n      exit)\n\n    // -------------------------------\n    // update shader program only for DRAW and batch\n    // -------------------------------\n    var commonDraw = block()\n    var CURRENT_PROGRAM = commonDraw.def()\n    if (staticOptions.frag && staticOptions.vert) {\n      var fragSrc = staticOptions.frag\n      var vertSrc = staticOptions.vert\n      commonDraw(CURRENT_PROGRAM, '=', link(\n        shaderState.program(\n          stringStore.id(vertSrc),\n          stringStore.id(fragSrc))), ';')\n    } else {\n      commonDraw(CURRENT_PROGRAM, '=',\n        SHADER_STATE, '.program', '(',\n        SHADER_STATE, '.vert[', SHADER_STATE, '.vert.length-1]', ',',\n        SHADER_STATE, '.frag[', SHADER_STATE, '.frag.length-1]', ');')\n    }\n\n    // ==========================================================\n    // DRAW PROCEDURE\n    // ==========================================================\n    var draw = proc('draw')\n    draw(entry, commonDraw)\n    if (hasDynamic) {\n      draw(\n        DYNARGS, '=', draw.arg(), ';',\n        dynamicEntry)\n    }\n    draw(\n      GL_POLL, '();',\n      'if(', CURRENT_PROGRAM, ')',\n      CURRENT_PROGRAM, '.draw(', hasDynamic ? DYNARGS : '', ');',\n      hasDynamic ? dynamicExit : '',\n      exit)\n\n    // ==========================================================\n    // BATCH DRAW\n    // ==========================================================\n    var batch = proc('batch')\n    batch(entry, commonDraw)\n    var EXEC_BATCH = link(function (program, count, args) {\n      var proc = program.batchCache[callId]\n      if (!proc) {\n        proc = program.batchCache[callId] = compileBatch(\n          program, dynamicOptions, dynamicUniforms, dynamicAttributes,\n          staticOptions)\n      }\n      return proc(count, args)\n    })\n    batch(\n      'if(', CURRENT_PROGRAM, '){',\n      GL_POLL, '();',\n      EXEC_BATCH, '(',\n      CURRENT_PROGRAM, ',',\n      batch.arg(), ',',\n      batch.arg(), ');')\n    // Set dirty on all dynamic flags\n    Object.keys(dynamicOptions).forEach(function (option) {\n      var STATE = CONTEXT_STATE[option]\n      if (STATE) {\n        batch(STATE, '.setDirty();')\n      }\n    })\n    batch('}', exit)\n\n    // -------------------------------\n    // eval and bind\n    // -------------------------------\n    return env.compile()\n  }\n\n  return {\n    draw: compileShaderDraw,\n    command: compileCommand\n  }\n}\n","module.exports={\n  \"[object Int8Array]\": 5120\n, \"[object Int16Array]\": 5122\n, \"[object Int32Array]\": 5124\n, \"[object Uint8Array]\": 5121\n, \"[object Uint8ClampedArray]\": 5121\n, \"[object Uint16Array]\": 5123\n, \"[object Uint32Array]\": 5125\n, \"[object Float32Array]\": 5126\n, \"[object Float64Array]\": 5121\n, \"[object ArrayBuffer]\": 5121\n}\n","module.exports={\n  \"int8\": 5120\n, \"int16\": 5122\n, \"int32\": 5124\n, \"uint8\": 5121\n, \"uint16\": 5123\n, \"uint32\": 5125\n, \"float\": 5126\n}\n","module.exports={\n  \"points\": 0,\n  \"lines\": 1,\n  \"line loop\": 2,\n  \"line strip\": 3,\n  \"triangles\": 4,\n  \"triangle strip\": 5,\n  \"triangle fan\": 6\n}\n","// Context and canvas creation helper functions\n/*globals HTMLElement,WebGLRenderingContext*/\n\nvar check = require('./util/check')\nvar extend = require('./util/extend')\n\nfunction createCanvas (element, options) {\n  var canvas = document.createElement('canvas')\n  var args = getContext(canvas, options)\n\n  extend(canvas.style, {\n    border: 0,\n    margin: 0,\n    padding: 0,\n    top: 0,\n    left: 0\n  })\n  element.appendChild(canvas)\n\n  if (element === document.body) {\n    canvas.style.position = 'absolute'\n    extend(element.style, {\n      margin: 0,\n      padding: 0\n    })\n  }\n\n  var scale = +args.options.pixelRatio\n  function resize () {\n    var w = window.innerWidth\n    var h = window.innerHeight\n    if (element !== document.body) {\n      var bounds = element.getBoundingClientRect()\n      w = bounds.right - bounds.left\n      h = bounds.top - bounds.bottom\n    }\n    canvas.width = scale * w\n    canvas.height = scale * h\n    extend(canvas.style, {\n      width: w + 'px',\n      height: h + 'px'\n    })\n  }\n\n  window.addEventListener('resize', resize, false)\n\n  var prevDestroy = args.options.onDestroy\n  args.options = extend(extend({}, args.options), {\n    onDestroy: function () {\n      window.removeEventListener('resize', resize)\n      element.removeChild(canvas)\n      prevDestroy && prevDestroy()\n    }\n  })\n\n  resize()\n\n  return args\n}\n\nfunction getContext (canvas, options) {\n  var glOptions = options.glOptions || {}\n\n  function get (name) {\n    try {\n      return canvas.getContext(name, glOptions)\n    } catch (e) {\n      return null\n    }\n  }\n\n  var gl = get('webgl') ||\n           get('experimental-webgl') ||\n           get('webgl-experimental')\n\n  check(gl, 'webgl not supported')\n\n  return {\n    gl: gl,\n    options: extend({\n      pixelRatio: window.devicePixelRatio\n    }, options)\n  }\n}\n\nmodule.exports = function parseArgs (args) {\n  if (typeof document === 'undefined' ||\n      typeof HTMLElement === 'undefined') {\n    return {\n      gl: args[0],\n      options: args[1] || {}\n    }\n  }\n\n  var element = document.body\n  var options = args[1] || {}\n\n  if (typeof args[0] === 'string') {\n    element = document.querySelector(args[0]) || document.body\n  } else if (typeof args[0] === 'object') {\n    if (args[0] instanceof HTMLElement) {\n      element = args[0]\n    } else if (args[0] instanceof WebGLRenderingContext) {\n      return {\n        gl: args[0],\n        options: extend({\n          pixelRatio: 1\n        }, options)\n      }\n    } else {\n      options = args[0]\n    }\n  }\n\n  if (element.nodeName && element.nodeName.toUpperCase() === 'CANVAS') {\n    return getContext(element, options)\n  } else {\n    return createCanvas(element, options)\n  }\n}\n","var GL_TRIANGLES = 4\n\nmodule.exports = function wrapDrawState (gl) {\n  var primitive = [ GL_TRIANGLES ]\n  var count = [ -1 ]\n  var offset = [ 0 ]\n  var instances = [ 0 ]\n\n  return {\n    primitive: primitive,\n    count: count,\n    offset: offset,\n    instances: instances\n  }\n}\n","var VARIABLE_COUNTER = 0\n\nfunction DynamicVariable (isFunc, data) {\n  this.id = (VARIABLE_COUNTER++)\n  this.func = isFunc\n  this.data = data\n}\n\nfunction defineDynamic (data, path) {\n  switch (typeof data) {\n    case 'boolean':\n    case 'number':\n    case 'string':\n      return new DynamicVariable(false, data)\n    case 'function':\n      return new DynamicVariable(true, data)\n    default:\n      return defineDynamic\n  }\n}\n\nfunction isDynamic (x) {\n  return (typeof x === 'function' && !x._reglType) ||\n         x instanceof DynamicVariable\n}\n\nfunction unbox (x, path) {\n  if (x instanceof DynamicVariable) {\n    return x\n  } else if (typeof x === 'function' &&\n             x !== defineDynamic) {\n    return new DynamicVariable(true, x)\n  }\n  return new DynamicVariable(false, path)\n}\n\nmodule.exports = {\n  define: defineDynamic,\n  isDynamic: isDynamic,\n  unbox: unbox\n}\n","var check = require('./util/check')\nvar isTypedArray = require('./util/is-typed-array')\nvar isNDArrayLike = require('./util/is-ndarray')\nvar primTypes = require('./constants/primitives.json')\n\nvar GL_POINTS = 0\nvar GL_LINES = 1\nvar GL_TRIANGLES = 4\n\nvar GL_BYTE = 5120\nvar GL_UNSIGNED_BYTE = 5121\nvar GL_SHORT = 5122\nvar GL_UNSIGNED_SHORT = 5123\nvar GL_INT = 5124\nvar GL_UNSIGNED_INT = 5125\n\nvar GL_ELEMENT_ARRAY_BUFFER = 34963\n\nmodule.exports = function wrapElementsState (gl, extensions, bufferState) {\n  var elements = [ null ]\n\n  function REGLElementBuffer () {\n    this.buffer = null\n    this.primType = GL_TRIANGLES\n    this.vertCount = 0\n    this.type = 0\n  }\n\n  REGLElementBuffer.prototype.bind = function () {\n    this.buffer.bind()\n  }\n\n  function createElements (options) {\n    var elements = new REGLElementBuffer()\n    var buffer = bufferState.create(null, GL_ELEMENT_ARRAY_BUFFER, true)\n    elements.buffer = buffer._buffer\n\n    function reglElements (input) {\n      var options = input\n      var ext32bit = extensions.oes_element_index_uint\n\n      // Upload data to vertex buffer\n      if (!options) {\n        buffer()\n      } else if (typeof options === 'number') {\n        buffer(options)\n      } else {\n        var data = null\n        var usage = 'static'\n        var byteLength = 0\n        if (\n          Array.isArray(options) ||\n          isTypedArray(options) ||\n          isNDArrayLike(options)) {\n          data = options\n        } else {\n          check.type(options, 'object', 'invalid arguments for elements')\n          if ('data' in options) {\n            data = options.data\n          }\n          if ('usage' in options) {\n            usage = options.usage\n          }\n          if ('length' in options) {\n            byteLength = options.length\n          }\n        }\n        if (Array.isArray(data) ||\n            (isNDArrayLike(data) && data.dtype === 'array') ||\n            'type' in options) {\n          buffer({\n            type: options.type ||\n              (ext32bit\n                ? 'uint32'\n                : 'uint16'),\n            usage: usage,\n            data: data,\n            length: byteLength\n          })\n        } else {\n          buffer({\n            usage: usage,\n            data: data,\n            length: byteLength\n          })\n        }\n        if (Array.isArray(data) || isTypedArray(data)) {\n          buffer.dimension = 3\n        }\n      }\n\n      // try to guess default primitive type and arguments\n      var vertCount = elements.buffer.byteLength\n      var type = 0\n      switch (elements.buffer.dtype) {\n        case GL_UNSIGNED_BYTE:\n        case GL_BYTE:\n          type = GL_UNSIGNED_BYTE\n          break\n\n        case GL_UNSIGNED_SHORT:\n        case GL_SHORT:\n          type = GL_UNSIGNED_SHORT\n          vertCount >>= 1\n          break\n\n        case GL_UNSIGNED_INT:\n        case GL_INT:\n          check(ext32bit, '32 bit element buffers not supported')\n          type = GL_UNSIGNED_INT\n          vertCount >>= 2\n          break\n\n        default:\n          check.raise('invalid element buffer type')\n      }\n\n      // try to guess primitive type from cell dimension\n      var primType = GL_TRIANGLES\n      var dimension = elements.buffer.dimension\n      if (dimension === 1) primType = GL_POINTS\n      if (dimension === 2) primType = GL_LINES\n      if (dimension === 3) primType = GL_TRIANGLES\n\n      // if manual override present, use that\n      if (typeof options === 'object') {\n        if ('primitive' in options) {\n          var primitive = options.primitive\n          check.parameter(primitive, primTypes)\n          primType = primTypes[primitive]\n        }\n\n        if ('count' in options) {\n          vertCount = options.vertCount | 0\n        }\n      }\n\n      // update properties for element buffer\n      elements.primType = primType\n      elements.vertCount = vertCount\n      elements.type = type\n\n      return reglElements\n    }\n\n    reglElements(options)\n\n    reglElements._reglType = 'elements'\n    reglElements._elements = elements\n    reglElements.destroy = function () {\n      check(elements.buffer !== null, 'must not double destroy elements')\n      buffer.destroy()\n      elements.buffer = null\n    }\n\n    return reglElements\n  }\n\n  return {\n    create: createElements,\n    elements: elements,\n    getElements: function (elements) {\n      if (elements && elements._elements instanceof REGLElementBuffer) {\n        return elements._elements\n      }\n      return null\n    }\n  }\n}\n","module.exports = function createExtensionCache (gl) {\n  var extensions = {}\n\n  function refreshExtensions () {\n    [\n      'oes_texture_float',\n      'oes_texture_float_linear',\n      'oes_texture_half_float',\n      'oes_texture_half_float_linear',\n      'oes_standard_derivatives',\n      'oes_element_index_uint',\n      'oes_fbo_render_mipmap',\n\n      'webgl_depth_texture',\n      'webgl_draw_buffers',\n      'webgl_color_buffer_float',\n\n      'ext_texture_filter_anisotropic',\n      'ext_frag_depth',\n      'ext_blend_minmax',\n      'ext_shader_texture_lod',\n      'ext_color_buffer_half_float',\n      'ext_srgb',\n\n      'angle_instanced_arrays',\n\n      'webgl_compressed_texture_s3tc',\n      'webgl_compressed_texture_atc',\n      'webgl_compressed_texture_pvrtc',\n      'webgl_compressed_texture_etc1'\n    ].forEach(function (ext) {\n      try {\n        extensions[ext] = gl.getExtension(ext)\n      } catch (e) {}\n    })\n  }\n\n  refreshExtensions()\n\n  return {\n    extensions: extensions,\n    refresh: refreshExtensions\n  }\n}\n","var check = require('./util/check')\nvar values = require('./util/values')\nvar extend = require('./util/extend')\n\n// We store these constants so that the minifier can inline them\nvar GL_FRAMEBUFFER = 0x8D40\nvar GL_RENDERBUFFER = 0x8D41\n\nvar GL_TEXTURE_2D = 0x0DE1\nvar GL_TEXTURE_CUBE_MAP = 0x8513\nvar GL_TEXTURE_CUBE_MAP_POSITIVE_X = 0x8515\n\nvar GL_COLOR_ATTACHMENT0 = 0x8CE0\nvar GL_DEPTH_ATTACHMENT = 0x8D00\nvar GL_STENCIL_ATTACHMENT = 0x8D20\nvar GL_DEPTH_STENCIL_ATTACHMENT = 0x821A\n\nvar GL_UNSIGNED_BYTE = 0x1401\nvar GL_FLOAT = 0x1406\n\nvar GL_HALF_FLOAT_OES = 0x8D61\n\nvar GL_RGBA = 0x1908\n\nvar GL_RGBA4 = 0x8056\nvar GL_RGB5_A1 = 0x8057\nvar GL_RGB565 = 0x8D62\nvar GL_DEPTH_COMPONENT16 = 0x81A5\nvar GL_STENCIL_INDEX8 = 0x8D48\n\nvar GL_DEPTH_COMPONENT = 0x1902\nvar GL_DEPTH_STENCIL = 0x84F9\n\nvar GL_SRGB8_ALPHA8_EXT = 0x8C43\n\nvar GL_RGBA32F_EXT = 0x8814\n\nvar GL_RGBA16F_EXT = 0x881A\nvar GL_RGB16F_EXT = 0x881B\n\nvar GL_FRAMEBUFFER_COMPLETE = 0x8CD5\nvar GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT = 0x8CD6\nvar GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT = 0x8CD7\nvar GL_FRAMEBUFFER_INCOMPLETE_DIMENSIONS = 0x8CD9\nvar GL_FRAMEBUFFER_UNSUPPORTED = 0x8CDD\n\nvar GL_BACK = 1029\n\nvar BACK_BUFFER = [GL_BACK]\n\nmodule.exports = function wrapFBOState (\n  gl,\n  extensions,\n  limits,\n  textureState,\n  renderbufferState) {\n  var statusCode = {}\n  statusCode[GL_FRAMEBUFFER_COMPLETE] = 'complete'\n  statusCode[GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT] = 'incomplete attachment'\n  statusCode[GL_FRAMEBUFFER_INCOMPLETE_DIMENSIONS] = 'incomplete dimensions'\n  statusCode[GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT] = 'incomplete, missing attachment'\n  statusCode[GL_FRAMEBUFFER_UNSUPPORTED] = 'unsupported'\n\n  var colorTextureFormats = {\n    'rgba': GL_RGBA\n  }\n\n  var colorRenderbufferFormats = {\n    'rgba4': GL_RGBA4,\n    'rgb565': GL_RGB565,\n    'rgb5 a1': GL_RGB5_A1\n  }\n\n  if (extensions.ext_srgb) {\n    colorRenderbufferFormats['srgba'] = GL_SRGB8_ALPHA8_EXT\n  }\n\n  if (extensions.ext_color_buffer_half_float) {\n    colorRenderbufferFormats['rgba16f'] = GL_RGBA16F_EXT\n    colorRenderbufferFormats['rgb16f'] = GL_RGB16F_EXT\n  }\n\n  if (extensions.webgl_color_buffer_float) {\n    colorRenderbufferFormats['rgba32f'] = GL_RGBA32F_EXT\n  }\n\n  var depthRenderbufferFormatEnums = [GL_DEPTH_COMPONENT16]\n  var stencilRenderbufferFormatEnums = [GL_STENCIL_INDEX8]\n  var depthStencilRenderbufferFormatEnums = [GL_DEPTH_STENCIL]\n\n  var depthTextureFormatEnums = []\n  var stencilTextureFormatEnums = []\n  var depthStencilTextureFormatEnums = []\n\n  if (extensions.webgl_depth_texture) {\n    depthTextureFormatEnums.push(GL_DEPTH_COMPONENT)\n    depthStencilTextureFormatEnums.push(GL_DEPTH_STENCIL)\n  }\n\n  var colorFormats = extend(extend({},\n    colorTextureFormats),\n    colorRenderbufferFormats)\n\n  var colorTextureFormatEnums = values(colorTextureFormats)\n  var colorRenderbufferFormatEnums = values(colorRenderbufferFormats)\n\n  var highestPrecision = GL_UNSIGNED_BYTE\n  var colorTypes = {\n    'uint8': GL_UNSIGNED_BYTE\n  }\n  if (extensions.oes_texture_half_float) {\n    highestPrecision = colorTypes['half float'] = GL_HALF_FLOAT_OES\n  }\n  if (extensions.oes_texture_float) {\n    highestPrecision = colorTypes.float = GL_FLOAT\n  }\n  colorTypes.best = highestPrecision\n\n  var DRAW_BUFFERS = (function () {\n    var result = new Array(limits.maxDrawbuffers)\n    for (var i = 0; i <= limits.maxDrawbuffers; ++i) {\n      var row = result[i] = new Array(i)\n      for (var j = 0; j < i; ++j) {\n        row[j] = GL_COLOR_ATTACHMENT0 + j\n      }\n    }\n    return result\n  })()\n\n  function FramebufferAttachment (target, level, texture, renderbuffer) {\n    this.target = target\n    this.level = level\n    this.texture = texture\n    this.renderbuffer = renderbuffer\n  }\n\n  function decRef (attachment) {\n    if (attachment) {\n      if (attachment.texture) {\n        attachment.texture._texture.decRef()\n      }\n      if (attachment.renderbuffer) {\n        attachment.renderbuffer._renderbuffer.decRef()\n      }\n    }\n  }\n\n  function checkFormat (attachment, texFormats, rbFormats) {\n    if (attachment.texture) {\n      check.oneOf(attachment.texture._texture.params.internalformat, texFormats,\n        'unsupported texture format for attachment')\n    } else {\n      check.oneOf(attachment.renderbuffer._renderbuffer.format, rbFormats,\n        'unsupported renderbuffer format for attachment')\n    }\n  }\n\n  function incRefAndCheckShape (attachment, framebuffer) {\n    var width = framebuffer.width\n    var height = framebuffer.height\n    if (attachment.texture) {\n      var texture = attachment.texture._texture\n      var tw = Math.max(1, texture.params.width >> attachment.level)\n      var th = Math.max(1, texture.params.height >> attachment.level)\n      width = width || tw\n      height = height || th\n      check(tw === width && th === height,\n        'inconsistent width/height for supplied texture')\n      check(texture.pollId < 0,\n        'polling fbo textures not supported')\n      texture.refCount += 1\n    } else {\n      var renderbuffer = attachment.renderbuffer._renderbuffer\n      width = width || renderbuffer.width\n      height = height || renderbuffer.height\n      check(\n        renderbuffer.width === width && renderbuffer.height === height,\n        'inconsistent width/height for renderbuffer')\n      check(\n        colorRenderbufferFormatEnums.indexOf(renderbuffer.format) >= 0,\n        'renderbuffer format not compatible with color channels')\n      renderbuffer.refCount += 1\n    }\n    framebuffer.width = width\n    framebuffer.height = height\n  }\n\n  function attach (location, attachment) {\n    if (attachment) {\n      if (attachment.texture) {\n        gl.framebufferTexture2D(\n          GL_FRAMEBUFFER,\n          location,\n          attachment.target,\n          attachment.texture._texture.texture,\n          attachment.level)\n      } else {\n        gl.framebufferRenderbuffer(\n          GL_FRAMEBUFFER,\n          location,\n          GL_RENDERBUFFER,\n          attachment.renderbuffer._renderbuffer.renderbuffer)\n      }\n    } else {\n      gl.framebufferTexture2D(\n        GL_FRAMEBUFFER,\n        location,\n        GL_TEXTURE_2D,\n        null,\n        0)\n    }\n  }\n\n  function tryUpdateAttachment (\n    attachment,\n    isTexture,\n    format,\n    type,\n    width,\n    height) {\n    if (attachment.texture) {\n      var texture = attachment.texture\n      if (isTexture) {\n        texture({\n          format: format,\n          type: type,\n          width: width,\n          height: height\n        })\n        texture._texture.refCount += 1\n        return true\n      }\n    } else {\n      var renderbuffer = attachment.renderbuffer\n      if (!isTexture) {\n        renderbuffer({\n          format: format,\n          width: width,\n          height: height\n        })\n        renderbuffer._renderbuffer.refCount += 1\n        return true\n      }\n    }\n    decRef(attachment)\n    return false\n  }\n\n  function parseAttachment (attachment) {\n    var target = GL_TEXTURE_2D\n    var level = 0\n    var texture = null\n    var renderbuffer = null\n\n    var data = attachment\n    if (typeof attachment === 'object') {\n      data = attachment.data\n      if ('level' in attachment) {\n        level = attachment.level | 0\n      }\n      if ('target' in attachment) {\n        target = attachment.target | 0\n      }\n    }\n\n    check.type(data, 'function', 'invalid attachment data')\n\n    var type = attachment._reglType\n    if (type === 'texture') {\n      texture = attachment\n      if (texture._texture.target === GL_TEXTURE_CUBE_MAP) {\n        check(\n          target >= GL_TEXTURE_CUBE_MAP_POSITIVE_X &&\n          target < GL_TEXTURE_CUBE_MAP_POSITIVE_X + 6,\n          'invalid cube map target')\n      } else {\n        check(target === GL_TEXTURE_2D)\n      }\n      // TODO check miplevel is consistent\n    } else if (type === 'renderbuffer') {\n      renderbuffer = attachment\n      target = GL_RENDERBUFFER\n      level = 0\n    } else {\n      check.raise('invalid regl object for attachment')\n    }\n\n    return new FramebufferAttachment(target, level, texture, renderbuffer)\n  }\n\n  function unwrapAttachment (attachment) {\n    return attachment && (attachment.texture || attachment.renderbuffer)\n  }\n\n  var framebufferCount = 0\n  var framebufferSet = {}\n  var framebufferStack = [null]\n  var framebufferDirty = true\n\n  function REGLFramebuffer () {\n    this.id = framebufferCount++\n    framebufferSet[this.id] = this\n\n    this.framebuffer = null\n    this.width = 0\n    this.height = 0\n\n    this.colorAttachments = []\n    this.depthAttachment = null\n    this.stencilAttachment = null\n    this.depthStencilAttachment = null\n\n    this.ownsColor = false\n    this.ownsDepthStencil = false\n  }\n\n  function refresh (framebuffer) {\n    if (!gl.isFramebuffer(framebuffer.framebuffer)) {\n      framebuffer.framebuffer = gl.createFramebuffer()\n    }\n    framebufferDirty = true\n    gl.bindFramebuffer(GL_FRAMEBUFFER, framebuffer.framebuffer)\n\n    var colorAttachments = framebuffer.colorAttachments\n    for (var i = 0; i < colorAttachments.length; ++i) {\n      attach(GL_COLOR_ATTACHMENT0 + i, colorAttachments[i])\n    }\n    for (i = colorAttachments.length; i < limits.maxColorAttachments; ++i) {\n      attach(GL_COLOR_ATTACHMENT0 + i, null)\n    }\n    attach(GL_DEPTH_ATTACHMENT, framebuffer.depthAttachment)\n    attach(GL_STENCIL_ATTACHMENT, framebuffer.stencilAttachment)\n    attach(GL_DEPTH_STENCIL_ATTACHMENT, framebuffer.depthStencilAttachment)\n\n    if (extensions.webgl_draw_buffers) {\n      extensions.webgl_draw_buffers.drawBuffersWEBGL(\n        DRAW_BUFFERS[colorAttachments.length])\n    }\n\n    // Check status code\n    var status = gl.checkFramebufferStatus(GL_FRAMEBUFFER)\n    if (status !== GL_FRAMEBUFFER_COMPLETE) {\n      check.raise('framebuffer configuration not supported, status = ' +\n        statusCode[status])\n    }\n  }\n\n  function decFBORefs (framebuffer) {\n    framebuffer.colorAttachments.forEach(decRef)\n    decRef(framebuffer.depthAttachment)\n    decRef(framebuffer.stencilAttachment)\n    decRef(framebuffer.depthStencilAttachment)\n  }\n\n  function destroy (framebuffer) {\n    var handle = framebuffer.framebuffer\n    check(handle, 'must not double destroy framebuffer')\n    if (gl.isFramebuffer(handle)) {\n      gl.deleteFramebuffer(handle)\n    }\n  }\n\n  function createFBO (options) {\n    var framebuffer = new REGLFramebuffer()\n\n    function reglFramebuffer (input) {\n      var i\n      var options = input || {}\n\n      var extDrawBuffers = extensions.webgl_draw_buffers\n\n      var width = 0\n      var height = 0\n      if ('shape' in options) {\n        var shape = options.shape\n        check(Array.isArray(shape) && shape.length >= 2,\n          'invalid shape for framebuffer')\n        width = shape[0]\n        height = shape[1]\n      } else {\n        if ('radius' in options) {\n          width = height = options.radius\n        }\n        if ('width' in options) {\n          width = options.width\n        }\n        if ('height' in options) {\n          height = options.height\n        }\n      }\n\n      // colorType, numColors\n      var colorBuffers = null\n      var ownsColor = false\n      if ('colorBuffers' in options || 'colorBuffer' in options) {\n        var colorInputs = options.colorBuffers || options.colorBuffer\n        if (!Array.isArray(colorInputs)) {\n          colorInputs = [colorInputs]\n        }\n\n        framebuffer.width = width\n        framebuffer.height = height\n\n        if (colorInputs.length > 1) {\n          check(extDrawBuffers, 'multiple render targets not supported')\n        }\n        check(colorInputs.length >= 0,\n          'must specify at least one color attachment')\n\n        // Wrap color attachments\n        colorBuffers = colorInputs.map(parseAttachment)\n\n        // Check head node\n        for (i = 0; i < colorBuffers.length; ++i) {\n          var colorAttachment = colorBuffers[i]\n          checkFormat(\n            colorAttachment,\n            colorTextureFormatEnums,\n            colorRenderbufferFormatEnums)\n          incRefAndCheckShape(\n            colorAttachment,\n            framebuffer)\n        }\n\n        width = framebuffer.width\n        height = framebuffer.height\n      } else {\n        var colorTexture = true\n        var colorFormat = 'rgba'\n        var colorType = 'uint8'\n        var colorCount = 1\n        ownsColor = true\n\n        framebuffer.width = width = width || gl.drawingBufferWidth\n        framebuffer.height = height = height || gl.drawingBufferHeight\n\n        if ('format' in options) {\n          colorFormat = options.format\n          check.parameter(colorFormat, colorFormats, 'invalid color format')\n          colorTexture = colorFormat in colorTextureFormats\n        }\n\n        if ('type' in options) {\n          check(colorTexture,\n            'colorType can not be set for renderbuffer targets')\n          colorType = options.type\n          check.parameter(colorType, colorTypes, 'invalid color type')\n        }\n\n        if ('colorCount' in options) {\n          colorCount = options.colorCount | 0\n          check(colorCount >= 0, 'color count must be positive')\n        }\n\n        // Reuse color buffer array if we own it\n        if (framebuffer.ownsColor) {\n          colorBuffers = framebuffer.colorAttachments\n          while (colorBuffers.length > colorCount) {\n            decRef(colorBuffers.pop())\n          }\n        } else {\n          colorBuffers = []\n        }\n\n        // update buffers in place, remove incompatible buffers\n        for (i = 0; i < colorBuffers.length; ++i) {\n          if (!tryUpdateAttachment(\n              colorBuffers[i],\n              colorTexture,\n              colorFormat,\n              colorType,\n              width,\n              height)) {\n            colorBuffers[i--] = colorBuffers[colorBuffers.length - 1]\n            colorBuffers.pop()\n          }\n        }\n\n        // Then append new buffers\n        while (colorBuffers.length < colorCount) {\n          if (colorTexture) {\n            colorBuffers.push(new FramebufferAttachment(\n              GL_TEXTURE_2D,\n              0,\n              textureState.create({\n                format: colorFormat,\n                type: colorType,\n                width: width,\n                height: height\n              }, GL_TEXTURE_2D),\n              null))\n          } else {\n            colorBuffers.push(new FramebufferAttachment(\n              GL_RENDERBUFFER,\n              0,\n              null,\n              renderbufferState.create({\n                format: colorFormat,\n                width: width,\n                height: height\n              })))\n          }\n        }\n      }\n\n      check(colorBuffers.length > 0, 'must specify at least one color buffer')\n\n      framebuffer.width = width\n      framebuffer.height = height\n\n      var depthBuffer = null\n      var stencilBuffer = null\n      var depthStencilBuffer = null\n      var ownsDepthStencil = false\n      var depthStencilCount = 0\n\n      if ('depthBuffer' in options) {\n        depthBuffer = parseAttachment(options.depthBuffer)\n        checkFormat(\n          depthBuffer,\n          depthTextureFormatEnums,\n          depthRenderbufferFormatEnums)\n        depthStencilCount += 1\n      }\n      if ('stencilBuffer' in options) {\n        stencilBuffer = parseAttachment(options.stencilBuffer)\n        checkFormat(\n          stencilBuffer,\n          stencilTextureFormatEnums,\n          stencilRenderbufferFormatEnums)\n        depthStencilCount += 1\n      }\n      if ('depthStencilBuffer' in options) {\n        depthStencilBuffer = parseAttachment(options.depthStencilBuffer)\n        checkFormat(\n          depthStencilBuffer,\n          depthStencilTextureFormatEnums,\n          depthStencilRenderbufferFormatEnums)\n        depthStencilCount += 1\n      }\n\n      if (!(depthBuffer || stencilBuffer || depthStencilBuffer)) {\n        var depth = true\n        var stencil = false\n        var useTexture = false\n\n        if ('depth' in options) {\n          depth = !!options.depth\n        }\n        if ('stencil' in options) {\n          stencil = !!options.stencil\n        }\n        if ('depthTexture' in options) {\n          useTexture = !!options.depthTexture\n        }\n\n        var curDepthStencil =\n          framebuffer.depthAttachment ||\n          framebuffer.stencilAttachment ||\n          framebuffer.depthStencilAttachment\n        var nextDepthStencil = null\n\n        if (depth || stencil) {\n          ownsDepthStencil = true\n\n          if (useTexture) {\n            check(extensions.webgl_depth_texture,\n              'depth texture extension not supported')\n            var depthTextureFormat\n            check(depth, 'stencil only textures not supported')\n            if (stencil) {\n              depthTextureFormat = 'depth stencil'\n            } else {\n              depthTextureFormat = 'depth'\n            }\n            if (framebuffer.ownsDepthStencil && curDepthStencil.texture) {\n              curDepthStencil.texture({\n                format: depthTextureFormat,\n                width: width,\n                height: height\n              })\n              curDepthStencil.texture._texture.refCount += 1\n              nextDepthStencil = curDepthStencil\n            } else {\n              nextDepthStencil = new FramebufferAttachment(\n                GL_TEXTURE_2D,\n                0,\n                textureState.create({\n                  format: depthTextureFormat,\n                  width: width,\n                  height: height\n                }, GL_TEXTURE_2D),\n                null)\n            }\n          } else {\n            var depthRenderbufferFormat\n            if (depth) {\n              if (stencil) {\n                depthRenderbufferFormat = 'depth stencil'\n              } else {\n                depthRenderbufferFormat = 'depth'\n              }\n            } else {\n              depthRenderbufferFormat = 'stencil'\n            }\n            if (framebuffer.ownsDepthStencil && curDepthStencil.renderbuffer) {\n              curDepthStencil.renderbuffer({\n                format: depthRenderbufferFormat,\n                width: width,\n                height: height\n              })\n              curDepthStencil.renderbuffer._renderbuffer.refCount += 1\n              nextDepthStencil = curDepthStencil\n            } else {\n              nextDepthStencil = new FramebufferAttachment(\n                GL_RENDERBUFFER,\n                0,\n                null,\n                renderbufferState.create({\n                  format: depthRenderbufferFormat,\n                  width: width,\n                  height: height\n                }))\n            }\n          }\n\n          if (depth) {\n            if (stencil) {\n              depthStencilBuffer = nextDepthStencil\n            } else {\n              depthBuffer = nextDepthStencil\n            }\n          } else {\n            stencilBuffer = nextDepthStencil\n          }\n        }\n      } else {\n        check(depthStencilCount === 1,\n          'can specify only one of depth, stencil or depthStencil attachment')\n\n        incRefAndCheckShape(\n          depthBuffer ||\n          stencilBuffer ||\n          depthStencilBuffer,\n          framebuffer)\n      }\n\n      decFBORefs(framebuffer)\n\n      framebuffer.colorAttachments = colorBuffers\n      framebuffer.depthAttachment = depthBuffer\n      framebuffer.stencilAttachment = stencilBuffer\n      framebuffer.depthStencilAttachment = depthStencilBuffer\n      framebuffer.ownsColor = ownsColor\n      framebuffer.ownsDepthStencil = ownsDepthStencil\n\n      reglFramebuffer.color = colorBuffers.map(unwrapAttachment)\n      reglFramebuffer.depth = unwrapAttachment(depthBuffer)\n      reglFramebuffer.stencil = unwrapAttachment(stencilBuffer)\n      reglFramebuffer.depthStencil = unwrapAttachment(depthStencilBuffer)\n\n      refresh(framebuffer)\n\n      reglFramebuffer.width = framebuffer.width\n      reglFramebuffer.height = framebuffer.height\n\n      return reglFramebuffer\n    }\n\n    reglFramebuffer(options)\n\n    reglFramebuffer._reglType = 'framebuffer'\n    reglFramebuffer._framebuffer = framebuffer\n    reglFramebuffer._destroy = function () {\n      destroy(framebuffer)\n    }\n\n    return reglFramebuffer\n  }\n\n  function refreshCache () {\n    values(framebufferSet).forEach(refresh)\n  }\n\n  function clearCache () {\n    values(framebufferSet).forEach(destroy)\n  }\n\n  function poll () {\n    if (framebufferDirty) {\n      var top = framebufferStack[framebufferStack.length - 1]\n      var ext_drawbuffers = extensions.webgl_draw_buffers\n\n      if (top) {\n        gl.bindFramebuffer(GL_FRAMEBUFFER, top.framebuffer)\n        if (ext_drawbuffers) {\n          ext_drawbuffers.drawBuffersWEBGL(DRAW_BUFFERS[top.colorAttachments.length])\n        }\n      } else {\n        gl.bindFramebuffer(GL_FRAMEBUFFER, null)\n        if (ext_drawbuffers) {\n          ext_drawbuffers.drawBuffersWEBGL(BACK_BUFFER)\n        }\n      }\n\n      framebufferDirty = false\n    }\n  }\n\n  function currentFramebuffer () {\n    return framebufferStack[framebufferStack.length - 1]\n  }\n\n  return {\n    top: currentFramebuffer,\n    dirty: function () {\n      return framebufferDirty\n    },\n    push: function (next_) {\n      var next = next_ || null\n      framebufferDirty = framebufferDirty || (next !== currentFramebuffer())\n      framebufferStack.push(next)\n      return framebufferDirty\n    },\n    pop: function () {\n      var prev = currentFramebuffer()\n      framebufferStack.pop()\n      framebufferDirty = framebufferDirty || (prev !== currentFramebuffer())\n      return framebufferDirty\n    },\n    getFramebuffer: function (object) {\n      if (typeof object === 'function' && object._reglType === 'framebuffer') {\n        var fbo = object._framebuffer\n        if (fbo instanceof REGLFramebuffer) {\n          return fbo\n        }\n      }\n      return null\n    },\n    poll: poll,\n    create: createFBO,\n    clear: clearCache,\n    refresh: refreshCache\n  }\n}\n","var GL_SUBPIXEL_BITS = 0x0D50\nvar GL_RED_BITS = 0x0D52\nvar GL_GREEN_BITS = 0x0D53\nvar GL_BLUE_BITS = 0x0D54\nvar GL_ALPHA_BITS = 0x0D55\nvar GL_DEPTH_BITS = 0x0D56\nvar GL_STENCIL_BITS = 0x0D57\n\nvar GL_ALIASED_POINT_SIZE_RANGE = 0x846D\nvar GL_ALIASED_LINE_WIDTH_RANGE = 0x846E\n\nvar GL_MAX_TEXTURE_SIZE = 0x0D33\nvar GL_MAX_VIEWPORT_DIMS = 0x0D3A\nvar GL_MAX_VERTEX_ATTRIBS = 0x8869\nvar GL_MAX_VERTEX_UNIFORM_VECTORS = 0x8DFB\nvar GL_MAX_VARYING_VECTORS = 0x8DFC\nvar GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS = 0x8B4D\nvar GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS = 0x8B4C\nvar GL_MAX_TEXTURE_IMAGE_UNITS = 0x8872\nvar GL_MAX_FRAGMENT_UNIFORM_VECTORS = 0x8DFD\nvar GL_MAX_CUBE_MAP_TEXTURE_SIZE = 0x851C\nvar GL_MAX_RENDERBUFFER_SIZE = 0x84E8\n\nvar GL_VENDOR = 0x1F00\nvar GL_RENDERER = 0x1F01\nvar GL_VERSION = 0x1F02\nvar GL_SHADING_LANGUAGE_VERSION = 0x8B8C\n\nvar GL_MAX_TEXTURE_MAX_ANISOTROPY_EXT = 0x84FF\n\nvar GL_MAX_COLOR_ATTACHMENTS_WEBGL = 0x8CDF\nvar GL_MAX_DRAW_BUFFERS_WEBGL = 0x8824\n\nmodule.exports = function (gl, extensions) {\n  var maxAnisotropic = 1\n  if (extensions.ext_texture_filter_anisotropic) {\n    maxAnisotropic = gl.getParameter(GL_MAX_TEXTURE_MAX_ANISOTROPY_EXT)\n  }\n\n  var maxDrawbuffers = 1\n  var maxColorAttachments = 1\n  if (extensions.webgl_draw_buffers) {\n    maxDrawbuffers = gl.getParameter(GL_MAX_DRAW_BUFFERS_WEBGL)\n    maxColorAttachments = gl.getParameter(GL_MAX_COLOR_ATTACHMENTS_WEBGL)\n  }\n\n  return {\n    // drawing buffer bit depth\n    colorBits: [\n      gl.getParameter(GL_RED_BITS),\n      gl.getParameter(GL_GREEN_BITS),\n      gl.getParameter(GL_BLUE_BITS),\n      gl.getParameter(GL_ALPHA_BITS)\n    ],\n    depthBits: gl.getParameter(GL_DEPTH_BITS),\n    stencilBits: gl.getParameter(GL_STENCIL_BITS),\n    subpixelBits: gl.getParameter(GL_SUBPIXEL_BITS),\n\n    // supported extensions\n    extensions: Object.keys(extensions).filter(function (ext) {\n      return !!extensions[ext]\n    }),\n\n    // max aniso samples\n    maxAnisotropic: maxAnisotropic,\n\n    // max draw buffers\n    maxDrawbuffers: maxDrawbuffers,\n    maxColorAttachments: maxColorAttachments,\n\n    // point and line size ranges\n    pointSizeDims: gl.getParameter(GL_ALIASED_POINT_SIZE_RANGE),\n    lineWidthDims: gl.getParameter(GL_ALIASED_LINE_WIDTH_RANGE),\n    maxViewportDims: gl.getParameter(GL_MAX_VIEWPORT_DIMS),\n    maxCombinedTextureUnits: gl.getParameter(GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS),\n    maxCubeMapSize: gl.getParameter(GL_MAX_CUBE_MAP_TEXTURE_SIZE),\n    maxRenderbufferSize: gl.getParameter(GL_MAX_RENDERBUFFER_SIZE),\n    maxTextureUnits: gl.getParameter(GL_MAX_TEXTURE_IMAGE_UNITS),\n    maxTextureSize: gl.getParameter(GL_MAX_TEXTURE_SIZE),\n    maxAttributes: gl.getParameter(GL_MAX_VERTEX_ATTRIBS),\n    maxVertexUniforms: gl.getParameter(GL_MAX_VERTEX_UNIFORM_VECTORS),\n    maxVertexTextureUnits: gl.getParameter(GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS),\n    maxVaryingVectors: gl.getParameter(GL_MAX_VARYING_VECTORS),\n    maxFragmentUniforms: gl.getParameter(GL_MAX_FRAGMENT_UNIFORM_VECTORS),\n\n    // vendor info\n    glsl: gl.getParameter(GL_SHADING_LANGUAGE_VERSION),\n    renderer: gl.getParameter(GL_RENDERER),\n    vendor: gl.getParameter(GL_VENDOR),\n    version: gl.getParameter(GL_VERSION)\n  }\n}\n","var check = require('./util/check')\nvar isTypedArray = require('./util/is-typed-array')\n\nvar GL_RGBA = 6408\nvar GL_UNSIGNED_BYTE = 5121\nvar GL_PACK_ALIGNMENT = 0x0D05\n\nmodule.exports = function wrapReadPixels (gl, reglPoll, viewportState) {\n  function readPixels (input) {\n    var options = input || {}\n    if (isTypedArray(input)) {\n      options = {\n        data: options\n      }\n    } else if (arguments.length === 2) {\n      options = {\n        width: arguments[0] | 0,\n        height: arguments[1] | 0\n      }\n    } else if (typeof input !== 'object') {\n      options = {}\n    }\n\n    // Update WebGL state\n    reglPoll()\n\n    // Read viewport state\n    var x = options.x || 0\n    var y = options.y || 0\n    var width = options.width || viewportState.width\n    var height = options.height || viewportState.height\n\n    // Compute size\n    var size = width * height * 4\n\n    // Allocate data\n    var data = options.data || new Uint8Array(size)\n\n    // Type check\n    check.isTypedArray(data)\n    check(data.byteLength >= size, 'data buffer too small')\n\n    // Run read pixels\n    gl.pixelStorei(GL_PACK_ALIGNMENT, 4)\n    gl.readPixels(x, y, width, height, GL_RGBA, GL_UNSIGNED_BYTE, data)\n\n    return data\n  }\n\n  return readPixels\n}\n","var check = require('./util/check')\nvar values = require('./util/values')\n\nvar GL_RENDERBUFFER = 0x8D41\n\nvar GL_RGBA4 = 0x8056\nvar GL_RGB5_A1 = 0x8057\nvar GL_RGB565 = 0x8D62\nvar GL_DEPTH_COMPONENT16 = 0x81A5\nvar GL_STENCIL_INDEX8 = 0x8D48\nvar GL_DEPTH_STENCIL = 0x84F9\n\nvar GL_SRGB8_ALPHA8_EXT = 0x8C43\n\nvar GL_RGBA32F_EXT = 0x8814\n\nvar GL_RGBA16F_EXT = 0x881A\nvar GL_RGB16F_EXT = 0x881B\n\nmodule.exports = function (gl, extensions, limits) {\n  var formatTypes = {\n    'rgba4': GL_RGBA4,\n    'rgb565': GL_RGB565,\n    'rgb5 a1': GL_RGB5_A1,\n    'depth': GL_DEPTH_COMPONENT16,\n    'stencil': GL_STENCIL_INDEX8,\n    'depth stencil': GL_DEPTH_STENCIL\n  }\n\n  if (extensions.ext_srgb) {\n    formatTypes['srgba'] = GL_SRGB8_ALPHA8_EXT\n  }\n\n  if (extensions.ext_color_buffer_half_float) {\n    formatTypes['rgba16f'] = GL_RGBA16F_EXT\n    formatTypes['rgb16f'] = GL_RGB16F_EXT\n  }\n\n  if (extensions.webgl_color_buffer_float) {\n    formatTypes['rgba32f'] = GL_RGBA32F_EXT\n  }\n\n  var renderbufferCount = 0\n  var renderbufferSet = {}\n\n  function REGLRenderbuffer () {\n    this.id = renderbufferCount++\n    this.refCount = 1\n\n    this.renderbuffer = null\n\n    this.format = GL_RGBA4\n    this.width = 0\n    this.height = 0\n  }\n\n  REGLRenderbuffer.prototype.decRef = function () {\n    if (--this.refCount === 0) {\n      destroy(this)\n    }\n  }\n\n  function refresh (rb) {\n    if (!gl.isRenderbuffer(rb.renderbuffer)) {\n      rb.renderbuffer = gl.createRenderbuffer()\n    }\n    gl.bindRenderbuffer(GL_RENDERBUFFER, rb.renderbuffer)\n    gl.renderbufferStorage(\n      GL_RENDERBUFFER,\n      rb.format,\n      rb.width,\n      rb.height)\n  }\n\n  function destroy (rb) {\n    var handle = rb.renderbuffer\n    check(handle, 'must not double destroy renderbuffer')\n    gl.bindRenderbuffer(GL_RENDERBUFFER, null)\n    if (gl.isRenderbuffer(handle)) {\n      gl.deleteRenderbuffer(handle)\n    }\n    rb.renderbuffer = null\n    rb.refCount = 0\n    delete renderbufferSet[rb.id]\n  }\n\n  function createRenderbuffer (input) {\n    var renderbuffer = new REGLRenderbuffer()\n    renderbufferSet[renderbuffer.id] = renderbuffer\n\n    function reglRenderbuffer (input) {\n      var options = input || {}\n\n      var w = 0\n      var h = 0\n      if ('shape' in options) {\n        var shape = options.shape\n        check(Array.isArray(shape) && shape.length >= 2,\n          'invalid renderbuffer shape')\n        w = shape[0] | 0\n        h = shape[1] | 0\n      } else {\n        if ('radius' in options) {\n          w = h = options.radius | 0\n        }\n        if ('width' in options) {\n          w = options.width | 0\n        }\n        if ('height' in options) {\n          h = options.height | 0\n        }\n      }\n      var s = limits.maxRenderbufferSize\n      check(w >= 0 && h >= 0 && w <= s && h <= s,\n        'invalid renderbuffer size')\n      reglRenderbuffer.width = renderbuffer.width = Math.max(w, 1)\n      reglRenderbuffer.height = renderbuffer.height = Math.max(h, 1)\n\n      renderbuffer.format = GL_RGBA4\n      if ('format' in options) {\n        var format = options.format\n        check.parameter(format, formatTypes, 'invalid render buffer format')\n        renderbuffer.format = formatTypes[format]\n      }\n\n      refresh(renderbuffer)\n\n      return reglRenderbuffer\n    }\n\n    reglRenderbuffer(input)\n\n    reglRenderbuffer._reglType = 'renderbuffer'\n    reglRenderbuffer._renderbuffer = renderbuffer\n    reglRenderbuffer.destroy = function () {\n      renderbuffer.decRef()\n    }\n\n    return reglRenderbuffer\n  }\n\n  function refreshRenderbuffers () {\n    values(renderbufferSet).forEach(refresh)\n  }\n\n  function destroyRenderbuffers () {\n    values(renderbufferSet).forEach(destroy)\n  }\n\n  return {\n    create: createRenderbuffer,\n    refresh: refreshRenderbuffers,\n    clear: destroyRenderbuffers\n  }\n}\n","var check = require('./util/check')\nvar values = require('./util/values')\n\nvar GL_FRAGMENT_SHADER = 35632\nvar GL_VERTEX_SHADER = 35633\n\nvar GL_ACTIVE_UNIFORMS = 0x8B86\nvar GL_ACTIVE_ATTRIBUTES = 0x8B89\n\nfunction ActiveInfo (name, id, location, info) {\n  this.name = name\n  this.id = id\n  this.location = location\n  this.info = info\n}\n\nmodule.exports = function wrapShaderState (\n  gl,\n  attributeState,\n  uniformState,\n  compileShaderDraw,\n  stringStore) {\n  // ===================================================\n  // glsl compilation and linking\n  // ===================================================\n  var fragShaders = {}\n  var vertShaders = {}\n\n  function getShader (type, id) {\n    var cache = type === GL_FRAGMENT_SHADER ? fragShaders : vertShaders\n    var shader = cache[id]\n\n    if (!shader) {\n      var source = stringStore.str(id)\n      shader = gl.createShader(type)\n      gl.shaderSource(shader, source)\n      gl.compileShader(shader)\n      check.shaderError(gl, shader, source, type)\n      cache[id] = shader\n    }\n\n    return shader\n  }\n\n  // ===================================================\n  // program linking\n  // ===================================================\n  var programCache = {}\n  var programList = []\n\n  function REGLProgram (fragId, vertId) {\n    this.fragId = fragId\n    this.vertId = vertId\n    this.program = null\n    this.uniforms = []\n    this.attributes = []\n    this.draw = function () {}\n    this.batchCache = {}\n  }\n\n  function linkProgram (desc) {\n    var i, info\n\n    // -------------------------------\n    // compile & link\n    // -------------------------------\n    var fragShader = getShader(GL_FRAGMENT_SHADER, desc.fragId)\n    var vertShader = getShader(GL_VERTEX_SHADER, desc.vertId)\n\n    var program = desc.program = gl.createProgram()\n    gl.attachShader(program, fragShader)\n    gl.attachShader(program, vertShader)\n    gl.linkProgram(program)\n    check.linkError(\n      gl,\n      program,\n      stringStore.str(desc.fragId),\n      stringStore.str(desc.vertId))\n\n    // -------------------------------\n    // grab uniforms\n    // -------------------------------\n    var numUniforms = gl.getProgramParameter(program, GL_ACTIVE_UNIFORMS)\n    var uniforms = desc.uniforms = []\n\n    for (i = 0; i < numUniforms; ++i) {\n      info = gl.getActiveUniform(program, i)\n      if (info) {\n        if (info.size > 1) {\n          for (var j = 0; j < info.size; ++j) {\n            var name = info.name.replace('[0]', '[' + j + ']')\n            uniformState.def(name)\n            uniforms.push(new ActiveInfo(\n              name,\n              stringStore.id(name),\n              gl.getUniformLocation(program, name),\n              info))\n          }\n        } else {\n          uniformState.def(info.name)\n          uniforms.push(new ActiveInfo(\n            info.name,\n            stringStore.id(info.name),\n            gl.getUniformLocation(program, info.name),\n            info))\n        }\n      }\n    }\n\n    // -------------------------------\n    // grab attributes\n    // -------------------------------\n    var numAttributes = gl.getProgramParameter(program, GL_ACTIVE_ATTRIBUTES)\n    var attributes = desc.attributes = []\n    for (i = 0; i < numAttributes; ++i) {\n      info = gl.getActiveAttrib(program, i)\n      if (info) {\n        attributeState.def(info.name)\n        attributes.push(new ActiveInfo(\n          info.name,\n          stringStore.id(info.name),\n          gl.getAttribLocation(program, info.name),\n          info))\n      }\n    }\n\n    // -------------------------------\n    // clear cached rendering methods\n    // -------------------------------\n    desc.draw = compileShaderDraw(desc)\n    desc.batchCache = {}\n  }\n\n  var fragShaderStack = [ -1 ]\n  var vertShaderStack = [ -1 ]\n\n  return {\n    clear: function () {\n      var deleteShader = gl.deleteShader.bind(gl)\n      values(fragShaders).forEach(deleteShader)\n      fragShaders = {}\n      values(vertShaders).forEach(deleteShader)\n      vertShaders = {}\n\n      programList.forEach(function (desc) {\n        gl.deleteProgram(desc.program)\n      })\n      programList.length = 0\n      programCache = {}\n    },\n\n    refresh: function () {\n      fragShaders = {}\n      vertShaders = {}\n      programList.forEach(linkProgram)\n    },\n\n    program: function (vertId, fragId) {\n      check(vertId >= 0, 'missing vertex shader')\n      check(fragId >= 0, 'missing fragment shader')\n\n      var cache = programCache[fragId]\n      if (!cache) {\n        cache = programCache[fragId] = {}\n      }\n      var program = cache[vertId]\n      if (!program) {\n        program = new REGLProgram(fragId, vertId)\n        linkProgram(program)\n        cache[vertId] = program\n        programList.push(program)\n      }\n      return program\n    },\n\n    shader: getShader,\n\n    frag: fragShaderStack,\n    vert: vertShaderStack\n  }\n}\n","var createStack = require('./util/stack')\nvar createEnvironment = require('./util/codegen')\n\n// WebGL constants\nvar GL_CULL_FACE = 0x0B44\nvar GL_BLEND = 0x0BE2\nvar GL_DITHER = 0x0BD0\nvar GL_STENCIL_TEST = 0x0B90\nvar GL_DEPTH_TEST = 0x0B71\nvar GL_SCISSOR_TEST = 0x0C11\nvar GL_POLYGON_OFFSET_FILL = 0x8037\nvar GL_SAMPLE_ALPHA_TO_COVERAGE = 0x809E\nvar GL_SAMPLE_COVERAGE = 0x80A0\nvar GL_FUNC_ADD = 0x8006\nvar GL_ZERO = 0\nvar GL_ONE = 1\nvar GL_FRONT = 1028\nvar GL_BACK = 1029\nvar GL_LESS = 513\nvar GL_CCW = 2305\nvar GL_ALWAYS = 519\nvar GL_KEEP = 7680\n\nmodule.exports = function wrapContextState (gl, framebufferState, viewportState) {\n  function capStack (cap, dflt) {\n    var result = createStack([!!dflt], function (flag) {\n      if (flag) {\n        gl.enable(cap)\n      } else {\n        gl.disable(cap)\n      }\n    })\n    result.flag = cap\n    return result\n  }\n\n  // Caps, flags and other random WebGL context state\n  var contextState = {\n    // Dithering\n    'dither': capStack(GL_DITHER),\n\n    // Blending\n    'blend.enable': capStack(GL_BLEND),\n    'blend.color': createStack([0, 0, 0, 0], function (r, g, b, a) {\n      gl.blendColor(r, g, b, a)\n    }),\n    'blend.equation': createStack([GL_FUNC_ADD, GL_FUNC_ADD], function (rgb, a) {\n      gl.blendEquationSeparate(rgb, a)\n    }),\n    'blend.func': createStack([\n      GL_ONE, GL_ZERO, GL_ONE, GL_ZERO\n    ], function (srcRGB, dstRGB, srcAlpha, dstAlpha) {\n      gl.blendFuncSeparate(srcRGB, dstRGB, srcAlpha, dstAlpha)\n    }),\n\n    // Depth\n    'depth.enable': capStack(GL_DEPTH_TEST, true),\n    'depth.func': createStack([GL_LESS], function (func) {\n      gl.depthFunc(func)\n    }),\n    'depth.range': createStack([0, 1], function (near, far) {\n      gl.depthRange(near, far)\n    }),\n    'depth.mask': createStack([true], function (m) {\n      gl.depthMask(m)\n    }),\n\n    // Face culling\n    'cull.enable': capStack(GL_CULL_FACE),\n    'cull.face': createStack([GL_BACK], function (mode) {\n      gl.cullFace(mode)\n    }),\n\n    // Front face orientation\n    'frontFace': createStack([GL_CCW], function (mode) {\n      gl.frontFace(mode)\n    }),\n\n    // Write masks\n    'colorMask': createStack([true, true, true, true], function (r, g, b, a) {\n      gl.colorMask(r, g, b, a)\n    }),\n\n    // Line width\n    'lineWidth': createStack([1], function (w) {\n      gl.lineWidth(w)\n    }),\n\n    // Polygon offset\n    'polygonOffset.enable': capStack(GL_POLYGON_OFFSET_FILL),\n    'polygonOffset.offset': createStack([0, 0], function (factor, units) {\n      gl.polygonOffset(factor, units)\n    }),\n\n    // Sample coverage\n    'sample.alpha': capStack(GL_SAMPLE_ALPHA_TO_COVERAGE),\n    'sample.enable': capStack(GL_SAMPLE_COVERAGE),\n    'sample.coverage': createStack([1, false], function (value, invert) {\n      gl.sampleCoverage(value, invert)\n    }),\n\n    // Stencil\n    'stencil.enable': capStack(GL_STENCIL_TEST),\n    'stencil.mask': createStack([-1], function (mask) {\n      gl.stencilMask(mask)\n    }),\n    'stencil.func': createStack([\n      GL_ALWAYS, 0, -1\n    ], function (func, ref, mask) {\n      gl.stencilFunc(func, ref, mask)\n    }),\n    'stencil.opFront': createStack([\n      GL_KEEP, GL_KEEP, GL_KEEP\n    ], function (fail, zfail, pass) {\n      gl.stencilOpSeparate(GL_FRONT, fail, zfail, pass)\n    }),\n    'stencil.opBack': createStack([\n      GL_KEEP, GL_KEEP, GL_KEEP\n    ], function (fail, zfail, pass) {\n      gl.stencilOpSeparate(GL_BACK, fail, zfail, pass)\n    }),\n\n    // Scissor\n    'scissor.enable': capStack(GL_SCISSOR_TEST),\n    'scissor.box': createStack([0, 0, -1, -1], function (x, y, w, h) {\n      var w_ = w\n      var fbo = framebufferState.top()\n      if (w < 0) {\n        if (fbo) {\n          w_ = fbo.width - x\n        } else {\n          w_ = gl.drawingBufferWidth - x\n        }\n      }\n      var h_ = h\n      if (h < 0) {\n        if (fbo) {\n          h_ = fbo.height - y\n        } else {\n          h_ = gl.drawingBufferHeight - y\n        }\n      }\n      gl.scissor(x, y, w_, h_)\n    }),\n\n    // Viewport\n    'viewport': createStack([0, 0, -1, -1], function (x, y, w, h) {\n      var w_ = w\n      var fbo = framebufferState.top()\n      if (w < 0) {\n        if (fbo) {\n          w_ = fbo.width - x\n        } else {\n          w_ = gl.drawingBufferWidth - x\n        }\n      }\n      var h_ = h\n      if (h < 0) {\n        if (fbo) {\n          h_ = fbo.height - y\n        } else {\n          h_ = gl.drawingBufferHeight - y\n        }\n      }\n      gl.viewport(x, y, w_, h_)\n      viewportState.width = w_\n      viewportState.height = h_\n    })\n  }\n\n  var env = createEnvironment()\n  var poll = env.proc('poll')\n  var refresh = env.proc('refresh')\n  Object.keys(contextState).forEach(function (prop) {\n    var STACK = env.link(contextState[prop])\n    poll(STACK, '.poll();')\n    refresh(STACK, '.setDirty();')\n  })\n\n  var procs = env.compile()\n\n  return {\n    contextState: contextState,\n    viewport: viewportState,\n    poll: procs.poll,\n    refresh: procs.refresh,\n\n    notifyViewportChanged: function () {\n      contextState.viewport.setDirty()\n      contextState['scissor.box'].setDirty()\n    }\n  }\n}\n","module.exports = function createStringStore () {\n  var stringIds = {'': 0}\n  var stringValues = ['']\n  return {\n    id: function (str) {\n      var result = stringIds[str]\n      if (result) {\n        return result\n      }\n      result = stringIds[str] = stringValues.length\n      stringValues.push(str)\n      return result\n    },\n\n    str: function (id) {\n      return stringValues[id]\n    }\n  }\n}\n","var check = require('./util/check')\nvar extend = require('./util/extend')\nvar values = require('./util/values')\nvar isTypedArray = require('./util/is-typed-array')\nvar isNDArrayLike = require('./util/is-ndarray')\nvar loadTexture = require('./util/load-texture')\nvar convertToHalfFloat = require('./util/to-half-float')\nvar parseDDS = require('./util/parse-dds')\n\nvar GL_COMPRESSED_TEXTURE_FORMATS = 0x86A3\n\nvar GL_TEXTURE_2D = 0x0DE1\nvar GL_TEXTURE_CUBE_MAP = 0x8513\nvar GL_TEXTURE_CUBE_MAP_POSITIVE_X = 0x8515\n\nvar GL_RGBA = 0x1908\nvar GL_ALPHA = 0x1906\nvar GL_RGB = 0x1907\nvar GL_LUMINANCE = 0x1909\nvar GL_LUMINANCE_ALPHA = 0x190A\n\nvar GL_RGBA4 = 0x8056\nvar GL_RGB5_A1 = 0x8057\nvar GL_RGB565 = 0x8D62\n\nvar GL_UNSIGNED_SHORT_4_4_4_4 = 0x8033\nvar GL_UNSIGNED_SHORT_5_5_5_1 = 0x8034\nvar GL_UNSIGNED_SHORT_5_6_5 = 0x8363\nvar GL_UNSIGNED_INT_24_8_WEBGL = 0x84FA\n\nvar GL_DEPTH_COMPONENT = 0x1902\nvar GL_DEPTH_STENCIL = 0x84F9\n\nvar GL_SRGB_EXT = 0x8C40\nvar GL_SRGB_ALPHA_EXT = 0x8C42\n\nvar GL_HALF_FLOAT_OES = 0x8D61\n\nvar GL_COMPRESSED_RGB_S3TC_DXT1_EXT = 0x83F0\nvar GL_COMPRESSED_RGBA_S3TC_DXT1_EXT = 0x83F1\nvar GL_COMPRESSED_RGBA_S3TC_DXT3_EXT = 0x83F2\nvar GL_COMPRESSED_RGBA_S3TC_DXT5_EXT = 0x83F3\n\nvar GL_COMPRESSED_RGB_ATC_WEBGL = 0x8C92\nvar GL_COMPRESSED_RGBA_ATC_EXPLICIT_ALPHA_WEBGL = 0x8C93\nvar GL_COMPRESSED_RGBA_ATC_INTERPOLATED_ALPHA_WEBGL = 0x87EE\n\nvar GL_COMPRESSED_RGB_PVRTC_4BPPV1_IMG = 0x8C00\nvar GL_COMPRESSED_RGB_PVRTC_2BPPV1_IMG = 0x8C01\nvar GL_COMPRESSED_RGBA_PVRTC_4BPPV1_IMG = 0x8C02\nvar GL_COMPRESSED_RGBA_PVRTC_2BPPV1_IMG = 0x8C03\n\nvar GL_COMPRESSED_RGB_ETC1_WEBGL = 0x8D64\n\nvar GL_UNSIGNED_BYTE = 0x1401\nvar GL_UNSIGNED_SHORT = 0x1403\nvar GL_UNSIGNED_INT = 0x1405\nvar GL_FLOAT = 0x1406\n\nvar GL_TEXTURE_WRAP_S = 0x2802\nvar GL_TEXTURE_WRAP_T = 0x2803\n\nvar GL_REPEAT = 0x2901\nvar GL_CLAMP_TO_EDGE = 0x812F\nvar GL_MIRRORED_REPEAT = 0x8370\n\nvar GL_TEXTURE_MAG_FILTER = 0x2800\nvar GL_TEXTURE_MIN_FILTER = 0x2801\n\nvar GL_NEAREST = 0x2600\nvar GL_LINEAR = 0x2601\nvar GL_NEAREST_MIPMAP_NEAREST = 0x2700\nvar GL_LINEAR_MIPMAP_NEAREST = 0x2701\nvar GL_NEAREST_MIPMAP_LINEAR = 0x2702\nvar GL_LINEAR_MIPMAP_LINEAR = 0x2703\n\nvar GL_GENERATE_MIPMAP_HINT = 0x8192\nvar GL_DONT_CARE = 0x1100\nvar GL_FASTEST = 0x1101\nvar GL_NICEST = 0x1102\n\nvar GL_TEXTURE_MAX_ANISOTROPY_EXT = 0x84FE\n\nvar GL_UNPACK_ALIGNMENT = 0x0CF5\nvar GL_UNPACK_FLIP_Y_WEBGL = 0x9240\nvar GL_UNPACK_PREMULTIPLY_ALPHA_WEBGL = 0x9241\nvar GL_UNPACK_COLORSPACE_CONVERSION_WEBGL = 0x9243\n\nvar GL_BROWSER_DEFAULT_WEBGL = 0x9244\n\nvar GL_TEXTURE0 = 0x84C0\n\nvar MIPMAP_FILTERS = [\n  GL_NEAREST_MIPMAP_NEAREST,\n  GL_NEAREST_MIPMAP_LINEAR,\n  GL_LINEAR_MIPMAP_NEAREST,\n  GL_LINEAR_MIPMAP_LINEAR\n]\n\nfunction isPow2 (v) {\n  return !(v & (v - 1)) && (!!v)\n}\n\nfunction isNumericArray (arr) {\n  return (\n    Array.isArray(arr) &&\n    (arr.length === 0 ||\n    typeof arr[0] === 'number'))\n}\n\nfunction isRectArray (arr) {\n  if (!Array.isArray(arr)) {\n    return false\n  }\n\n  var width = arr.length\n  if (width === 0 || !Array.isArray(arr[0])) {\n    return false\n  }\n\n  var height = arr[0].length\n  for (var i = 1; i < width; ++i) {\n    if (!Array.isArray(arr[i]) || arr[i].length !== height) {\n      return false\n    }\n  }\n  return true\n}\n\nfunction classString (x) {\n  return Object.prototype.toString.call(x)\n}\n\nfunction isCanvasElement (object) {\n  return classString(object) === '[object HTMLCanvasElement]'\n}\n\nfunction isContext2D (object) {\n  return classString(object) === '[object CanvasRenderingContext2D]'\n}\n\nfunction isImageElement (object) {\n  return classString(object) === '[object HTMLImageElement]'\n}\n\nfunction isVideoElement (object) {\n  return classString(object) === '[object HTMLVideoElement]'\n}\n\nfunction isPendingXHR (object) {\n  return classString(object) === '[object XMLHttpRequest]'\n}\n\nfunction isPixelData (object) {\n  return (\n    typeof object === 'string' ||\n    (!!object && (\n      isTypedArray(object) ||\n      isNumericArray(object) ||\n      isNDArrayLike(object) ||\n      isCanvasElement(object) ||\n      isContext2D(object) ||\n      isImageElement(object) ||\n      isVideoElement(object) ||\n      isRectArray(object))))\n}\n\n// Transpose an array of pixels\nfunction transposePixels (data, nx, ny, nc, sx, sy, sc, off) {\n  var result = new data.constructor(nx * ny * nc)\n  var ptr = 0\n  for (var i = 0; i < ny; ++i) {\n    for (var j = 0; j < nx; ++j) {\n      for (var k = 0; k < nc; ++k) {\n        result[ptr++] = data[sy * i + sx * j + sc * k + off]\n      }\n    }\n  }\n  return result\n}\n\nmodule.exports = function createTextureSet (gl, extensions, limits, reglPoll, viewportState) {\n  var mipmapHint = {\n    \"don't care\": GL_DONT_CARE,\n    'dont care': GL_DONT_CARE,\n    'nice': GL_NICEST,\n    'fast': GL_FASTEST\n  }\n\n  var wrapModes = {\n    'repeat': GL_REPEAT,\n    'clamp': GL_CLAMP_TO_EDGE,\n    'mirror': GL_MIRRORED_REPEAT\n  }\n\n  var magFilters = {\n    'nearest': GL_NEAREST,\n    'linear': GL_LINEAR\n  }\n\n  var minFilters = extend({\n    'nearest mipmap nearest': GL_NEAREST_MIPMAP_NEAREST,\n    'linear mipmap nearest': GL_LINEAR_MIPMAP_NEAREST,\n    'nearest mipmap linear': GL_NEAREST_MIPMAP_LINEAR,\n    'linear mipmap linear': GL_LINEAR_MIPMAP_LINEAR,\n    'mipmap': GL_LINEAR_MIPMAP_LINEAR\n  }, magFilters)\n\n  var colorSpace = {\n    'none': 0,\n    'browser': GL_BROWSER_DEFAULT_WEBGL\n  }\n\n  var textureTypes = {\n    'uint8': GL_UNSIGNED_BYTE,\n    'rgba4': GL_UNSIGNED_SHORT_4_4_4_4,\n    'rgb565': GL_UNSIGNED_SHORT_5_6_5,\n    'rgb5 a1': GL_UNSIGNED_SHORT_5_5_5_1\n  }\n\n  var textureFormats = {\n    'alpha': GL_ALPHA,\n    'luminance': GL_LUMINANCE,\n    'luminance alpha': GL_LUMINANCE_ALPHA,\n    'rgb': GL_RGB,\n    'rgba': GL_RGBA,\n    'rgba4': GL_RGBA4,\n    'rgb5 a1': GL_RGB5_A1,\n    'rgb565': GL_RGB565\n  }\n\n  var compressedTextureFormats = {}\n\n  if (extensions.ext_srgb) {\n    textureFormats.srgb = GL_SRGB_EXT\n    textureFormats.srgba = GL_SRGB_ALPHA_EXT\n  }\n\n  if (extensions.oes_texture_float) {\n    textureTypes.float = GL_FLOAT\n  }\n\n  if (extensions.oes_texture_half_float) {\n    textureTypes['half float'] = GL_HALF_FLOAT_OES\n  }\n\n  if (extensions.webgl_depth_texture) {\n    extend(textureFormats, {\n      'depth': GL_DEPTH_COMPONENT,\n      'depth stencil': GL_DEPTH_STENCIL\n    })\n\n    extend(textureTypes, {\n      'uint16': GL_UNSIGNED_SHORT,\n      'uint32': GL_UNSIGNED_INT,\n      'depth stencil': GL_UNSIGNED_INT_24_8_WEBGL\n    })\n  }\n\n  if (extensions.webgl_compressed_texture_s3tc) {\n    extend(compressedTextureFormats, {\n      'rgb s3tc dxt1': GL_COMPRESSED_RGB_S3TC_DXT1_EXT,\n      'rgba s3tc dxt1': GL_COMPRESSED_RGBA_S3TC_DXT1_EXT,\n      'rgba s3tc dxt3': GL_COMPRESSED_RGBA_S3TC_DXT3_EXT,\n      'rgba s3tc dxt5': GL_COMPRESSED_RGBA_S3TC_DXT5_EXT\n    })\n  }\n\n  if (extensions.webgl_compressed_texture_atc) {\n    extend(compressedTextureFormats, {\n      'rgb arc': GL_COMPRESSED_RGB_ATC_WEBGL,\n      'rgba atc explicit alpha': GL_COMPRESSED_RGBA_ATC_EXPLICIT_ALPHA_WEBGL,\n      'rgba atc interpolated alpha': GL_COMPRESSED_RGBA_ATC_INTERPOLATED_ALPHA_WEBGL\n    })\n  }\n\n  if (extensions.webgl_compressed_texture_pvrtc) {\n    extend(compressedTextureFormats, {\n      'rgb pvrtc 4bppv1': GL_COMPRESSED_RGB_PVRTC_4BPPV1_IMG,\n      'rgb pvrtc 2bppv1': GL_COMPRESSED_RGB_PVRTC_2BPPV1_IMG,\n      'rgba pvrtc 4bppv1': GL_COMPRESSED_RGBA_PVRTC_4BPPV1_IMG,\n      'rgba pvrtc 2bppv1': GL_COMPRESSED_RGBA_PVRTC_2BPPV1_IMG\n    })\n  }\n\n  if (extensions.webgl_compressed_texture_etc1) {\n    compressedTextureFormats['rgb etc1'] = GL_COMPRESSED_RGB_ETC1_WEBGL\n  }\n\n  // Copy over all texture formats\n  var supportedCompressedFormats = Array.prototype.slice.call(\n    gl.getParameter(GL_COMPRESSED_TEXTURE_FORMATS))\n  Object.keys(compressedTextureFormats).forEach(function (name) {\n    var format = compressedTextureFormats[name]\n    if (supportedCompressedFormats.indexOf(format) >= 0) {\n      textureFormats[name] = format\n    }\n  })\n\n  var supportedFormats = Object.keys(textureFormats)\n  limits.textureFormats = supportedFormats\n\n  var colorFormats = supportedFormats.reduce(function (color, key) {\n    var glenum = textureFormats[key]\n    if (glenum === GL_LUMINANCE ||\n        glenum === GL_ALPHA ||\n        glenum === GL_LUMINANCE ||\n        glenum === GL_LUMINANCE_ALPHA ||\n        glenum === GL_DEPTH_COMPONENT ||\n        glenum === GL_DEPTH_STENCIL) {\n      color[glenum] = glenum\n    } else if (glenum === GL_RGB5_A1 || key.indexOf('rgba') >= 0) {\n      color[glenum] = GL_RGBA\n    } else {\n      color[glenum] = GL_RGB\n    }\n    return color\n  }, {})\n\n  // Pixel storage parsing\n  function PixelInfo (target) {\n    // tex target\n    this.target = target\n\n    // pixelStorei info\n    this.flipY = false\n    this.premultiplyAlpha = false\n    this.unpackAlignment = 1\n    this.colorSpace = 0\n\n    // shape\n    this.width = 0\n    this.height = 0\n    this.channels = 0\n\n    // format and type\n    this.format = 0\n    this.internalformat = 0\n    this.type = 0\n    this.compressed = false\n\n    // mip level\n    this.miplevel = 0\n\n    // ndarray-like parameters\n    this.strideX = 0\n    this.strideY = 0\n    this.strideC = 0\n    this.offset = 0\n\n    // copy pixels info\n    this.x = 0\n    this.y = 0\n    this.copy = false\n\n    // data sources\n    this.data = null\n    this.image = null\n    this.video = null\n    this.canvas = null\n    this.xhr = null\n\n    // CORS\n    this.crossOrigin = null\n\n    // horrible state flags\n    this.needsPoll = false\n    this.needsListeners = false\n  }\n\n  extend(PixelInfo.prototype, {\n    parseFlags: function (options) {\n      if (typeof options !== 'object' || !options) {\n        return\n      }\n\n      if ('premultiplyAlpha' in options) {\n        check.type(options.premultiplyAlpha, 'boolean',\n          'invalid premultiplyAlpha')\n        this.premultiplyAlpha = options.premultiplyAlpha\n      }\n\n      if ('flipY' in options) {\n        check.type(options.flipY, 'boolean',\n          'invalid texture flip')\n        this.flipY = options.flipY\n      }\n\n      if ('alignment' in options) {\n        check.oneOf(options.alignment, [1, 2, 4, 8],\n          'invalid texture unpack alignment')\n        this.unpackAlignment = options.alignment\n      }\n\n      if ('colorSpace' in options) {\n        check.parameter(options.colorSpace, colorSpace,\n          'invalid colorSpace')\n        this.colorSpace = colorSpace[options.colorSpace]\n      }\n\n      if ('format' in options) {\n        var format = options.format\n        check.parameter(format, textureFormats,\n          'invalid texture format')\n        this.internalformat = textureFormats[format]\n        if (format in textureTypes) {\n          this.type = textureTypes[format]\n        }\n        if (format in compressedTextureFormats) {\n          this.compressed = true\n        }\n      }\n\n      if ('type' in options) {\n        var type = options.type\n        check.parameter(type, textureTypes,\n          'invalid texture type')\n        this.type = textureTypes[type]\n      }\n\n      var w = this.width\n      var h = this.height\n      var c = this.channels\n      if ('shape' in options) {\n        check(Array.isArray(options.shape) && options.shape.length >= 2,\n          'shape must be an array')\n        w = options.shape[0]\n        h = options.shape[1]\n        if (options.shape.length === 3) {\n          c = options.shape[2]\n        }\n      } else {\n        if ('radius' in options) {\n          w = h = options.radius\n        }\n        if ('width' in options) {\n          w = options.width\n        }\n        if ('height' in options) {\n          h = options.height\n        }\n        if ('channels' in options) {\n          c = options.channels\n        }\n      }\n      this.width = w | 0\n      this.height = h | 0\n      this.channels = c | 0\n\n      if ('stride' in options) {\n        var stride = options.stride\n        check(Array.isArray(stride) && stride.length >= 2,\n          'invalid stride vector')\n        this.strideX = stride[0]\n        this.strideY = stride[1]\n        if (stride.length === 3) {\n          this.strideC = stride[2]\n        } else {\n          this.strideC = 1\n        }\n        this.needsTranspose = true\n      } else {\n        this.strideC = 1\n        this.strideX = this.strideC * c\n        this.strideY = this.strideX * w\n      }\n\n      if ('offset' in options) {\n        this.offset = options.offset | 0\n        this.needsTranspose = true\n      }\n\n      if ('crossOrigin' in options) {\n        this.crossOrigin = options.crossOrigin\n      }\n    },\n    parse: function (options, miplevel) {\n      this.miplevel = miplevel\n      this.width = this.width >> miplevel\n      this.height = this.height >> miplevel\n\n      var data = options\n      switch (typeof options) {\n        case 'string':\n          break\n        case 'object':\n          if (!options) {\n            return\n          }\n          this.parseFlags(options)\n          if (isPixelData(options.data)) {\n            data = options.data\n          }\n          break\n        case 'undefined':\n          return\n        default:\n          check.raise('invalid pixel data type')\n      }\n\n      if (typeof data === 'string') {\n        data = loadTexture(data, this.crossOrigin)\n      }\n\n      var array = null\n      var needsConvert = false\n\n      if (this.compressed) {\n        check(data instanceof Uint8Array || isPendingXHR(data),\n          'compressed texture data must be stored in a uint8array')\n      }\n\n      if (data === null) {\n        // TODO\n      } else if (isTypedArray(data)) {\n        this.data = data\n      } else if (isNumericArray(data)) {\n        array = data\n        needsConvert = true\n      } else if (isNDArrayLike(data)) {\n        if (Array.isArray(data.data)) {\n          array = data.data\n          needsConvert = true\n        } else {\n          this.data = data.data\n        }\n        var shape = data.shape\n        this.width = shape[0]\n        this.height = shape[1]\n        if (shape.length === 3) {\n          this.channels = shape[2]\n        } else {\n          this.channels = 1\n        }\n        var stride = data.stride\n        this.strideX = data.stride[0]\n        this.strideY = data.stride[1]\n        if (stride.length === 3) {\n          this.strideC = data.stride[2]\n        } else {\n          this.strideC = 1\n        }\n        this.offset = data.offset\n        this.needsTranspose = true\n      } else if (isCanvasElement(data) || isContext2D(data)) {\n        if (isCanvasElement(data)) {\n          this.canvas = data\n        } else {\n          this.canvas = data.canvas\n        }\n        this.width = this.canvas.width\n        this.height = this.canvas.height\n        this.setDefaultFormat()\n      } else if (isImageElement(data)) {\n        this.image = data\n        if (!data.complete) {\n          this.width = this.width || data.naturalWidth\n          this.height = this.height || data.naturalHeight\n          this.needsListeners = true\n        } else {\n          this.width = data.naturalWidth\n          this.height = data.naturalHeight\n        }\n        this.setDefaultFormat()\n      } else if (isVideoElement(data)) {\n        this.video = data\n        if (data.readyState > 1) {\n          this.width = data.width\n          this.height = data.height\n        } else {\n          this.width = this.width || data.width\n          this.height = this.height || data.height\n          this.needsListeners = true\n        }\n        this.needsPoll = true\n        this.setDefaultFormat()\n      } else if (isPendingXHR(data)) {\n        this.xhr = data\n        this.needsListeners = true\n      } else if (isRectArray(data)) {\n        var w = data[0].length\n        var h = data.length\n        var c = 1\n        var i, j, k, p\n        if (Array.isArray(data[0][0])) {\n          c = data[0][0].length\n          check(c >= 0 && c <= 4, 'invalid number of channels for image data')\n          array = Array(w * h * c)\n          p = 0\n          for (j = 0; j < h; ++j) {\n            for (i = 0; i < w; ++i) {\n              for (k = 0; k < c; ++k) {\n                array[p++] = data[j][i][k]\n              }\n            }\n          }\n        } else {\n          array = Array(w * h)\n          p = 0\n          for (j = 0; j < h; ++j) {\n            for (i = 0; i < w; ++i) {\n              array[p++] = data[j][i]\n            }\n          }\n        }\n        this.width = w\n        this.height = h\n        this.channels = c\n        needsConvert = true\n      } else if (options.copy) {\n        this.copy = true\n        this.x = this.x | 0\n        this.y = this.y | 0\n        this.width = (this.width || viewportState.width) | 0\n        this.height = (this.height || viewportState.height) | 0\n        this.setDefaultFormat()\n      }\n\n      // Fix up missing type info for typed arrays\n      if (!this.type && this.data) {\n        if (this.format === GL_DEPTH_COMPONENT) {\n          if (this.data instanceof Uint16Array) {\n            this.type = GL_UNSIGNED_SHORT\n          } else if (this.data instanceof Uint32Array) {\n            this.type = GL_UNSIGNED_INT\n          }\n        } else if (this.data instanceof Float32Array) {\n          this.type = GL_FLOAT\n        }\n      }\n\n      // Infer default format\n      if (!this.internalformat) {\n        var channels = this.channels = this.channels || 4\n        this.internalformat = [\n          GL_LUMINANCE,\n          GL_LUMINANCE_ALPHA,\n          GL_RGB,\n          GL_RGBA][channels - 1]\n        check(this.internalformat, 'invalid number of channels')\n      }\n\n      var format = this.internalformat\n      if (format === GL_DEPTH_COMPONENT || format === GL_DEPTH_STENCIL) {\n        check(extensions.webgl_depth_texture,\n          'depth/stencil texture not supported')\n        if (format === GL_DEPTH_COMPONENT) {\n          check(this.type === GL_UNSIGNED_SHORT || GL_UNSIGNED_INT,\n            'depth texture type must be uint16 or uint32')\n        }\n        if (format === GL_DEPTH_STENCIL) {\n          check(this.type === GL_UNSIGNED_INT_24_8_WEBGL,\n            'depth stencil texture format must match type')\n        }\n        check(\n          !this.data && !array && !this.image && !this.video && !this.canvas,\n          'depth/stencil textures are for rendering only')\n      }\n\n      // Compute color format and number of channels\n      var colorFormat = this.format = colorFormats[format]\n      if (!this.channels) {\n        switch (colorFormat) {\n          case GL_LUMINANCE:\n          case GL_ALPHA:\n          case GL_DEPTH_COMPONENT:\n            this.channels = 1\n            break\n\n          case GL_DEPTH_STENCIL:\n          case GL_LUMINANCE_ALPHA:\n            this.channels = 2\n            break\n\n          case GL_RGB:\n            this.channels = 3\n            break\n\n          default:\n            this.channels = 4\n        }\n      }\n\n      // Check that texture type is supported\n      var type = this.type\n      if (type === GL_FLOAT) {\n        check(extensions.oes_texture_float,\n          'float texture not supported')\n      } else if (type === GL_HALF_FLOAT_OES) {\n        check(extensions.oes_texture_half_float,\n          'half float texture not supported')\n      } else if (!type) {\n        if (format === GL_DEPTH_COMPONENT) {\n          type = GL_UNSIGNED_INT\n        } else {\n          type = GL_UNSIGNED_BYTE\n        }\n      }\n      this.type = type\n\n      // apply conversion\n      if (needsConvert) {\n        switch (type) {\n          case GL_UNSIGNED_BYTE:\n            this.data = new Uint8Array(array)\n            break\n          case GL_UNSIGNED_SHORT:\n            this.data = new Uint16Array(array)\n            break\n          case GL_UNSIGNED_INT:\n            this.data = new Uint32Array(array)\n            break\n          case GL_FLOAT:\n            this.data = new Float32Array(array)\n            break\n          case GL_HALF_FLOAT_OES:\n            this.data = convertToHalfFloat(array)\n            break\n\n          case GL_UNSIGNED_SHORT_5_6_5:\n          case GL_UNSIGNED_SHORT_5_5_5_1:\n          case GL_UNSIGNED_SHORT_4_4_4_4:\n          case GL_UNSIGNED_INT_24_8_WEBGL:\n            check.raise('unsupported format for automatic conversion')\n            break\n\n          default:\n            check.raise('unsupported type conversion')\n        }\n      }\n\n      if (this.data) {\n        // apply transpose\n        if (this.needsTranspose) {\n          this.data = transposePixels(\n            this.data,\n            this.width,\n            this.height,\n            this.channels,\n            this.strideX,\n            this.strideY,\n            this.strideC,\n            this.offset)\n        }\n        // check data type\n        switch (type) {\n          case GL_UNSIGNED_BYTE:\n            check(this.data instanceof Uint8Array ||\n                  this.data instanceof Uint8ClampedArray,\n                  'incompatible pixel type')\n            break\n          case GL_UNSIGNED_SHORT_5_6_5:\n          case GL_UNSIGNED_SHORT_5_5_5_1:\n          case GL_UNSIGNED_SHORT_4_4_4_4:\n          case GL_UNSIGNED_SHORT:\n          case GL_HALF_FLOAT_OES:\n            check(this.data instanceof Uint16Array,\n                  'incompatible pixel type')\n            break\n          case GL_UNSIGNED_INT:\n            check(this.data instanceof Uint32Array,\n                  'incompatible pixel type')\n            break\n\n          case GL_FLOAT:\n            check(this.data instanceof Float32Array,\n                  'incompatible pixel type')\n            break\n\n          default:\n            check.raise('bad or missing pixel type')\n        }\n      }\n\n      this.needsTranspose = false\n    },\n\n    setDefaultFormat: function () {\n      this.format = this.internalformat = GL_RGBA\n      this.type = GL_UNSIGNED_BYTE\n      this.channels = 4\n      this.compressed = false\n    },\n\n    upload: function (params) {\n      gl.pixelStorei(GL_UNPACK_FLIP_Y_WEBGL, this.flipY)\n      gl.pixelStorei(GL_UNPACK_PREMULTIPLY_ALPHA_WEBGL, this.premultiplyAlpha)\n      gl.pixelStorei(GL_UNPACK_COLORSPACE_CONVERSION_WEBGL, this.colorSpace)\n      gl.pixelStorei(GL_UNPACK_ALIGNMENT, this.unpackAlignment)\n\n      var target = this.target\n      var miplevel = this.miplevel\n      var image = this.image\n      var canvas = this.canvas\n      var video = this.video\n      var data = this.data\n      var internalformat = this.internalformat\n      var format = this.format\n      var type = this.type\n      var width = this.width || Math.max(1, params.width >> miplevel)\n      var height = this.height || Math.max(1, params.height >> miplevel)\n      if (video && video.readyState > 2) {\n        gl.texImage2D(target, miplevel, format, format, type, video)\n      } else if (image && image.complete) {\n        gl.texImage2D(target, miplevel, format, format, type, image)\n      } else if (canvas) {\n        gl.texImage2D(target, miplevel, format, format, type, canvas)\n      } else if (this.compressed) {\n        gl.compressedTexImage2D(target, miplevel, internalformat, width, height, 0, data)\n      } else if (this.copy) {\n        reglPoll()\n        gl.copyTexImage2D(target, miplevel, format, this.x, this.y, width, height, 0)\n      } else if (data) {\n        gl.texImage2D(target, miplevel, format, width, height, 0, format, type, data)\n      } else {\n        gl.texImage2D(target, miplevel, format, width || 1, height || 1, 0, format, type, null)\n      }\n    }\n  })\n\n  function TexParams (target) {\n    this.target = target\n\n    // Default image shape info\n    this.width = 0\n    this.height = 0\n    this.format = 0\n    this.internalformat = 0\n    this.type = 0\n\n    // wrap mode\n    this.wrapS = GL_CLAMP_TO_EDGE\n    this.wrapT = GL_CLAMP_TO_EDGE\n\n    // filtering\n    this.minFilter = 0\n    this.magFilter = GL_NEAREST\n    this.anisotropic = 1\n\n    // mipmaps\n    this.genMipmaps = false\n    this.mipmapHint = GL_DONT_CARE\n  }\n\n  extend(TexParams.prototype, {\n    parse: function (options) {\n      if (typeof options !== 'object' || !options) {\n        return\n      }\n\n      if ('min' in options) {\n        var minFilter = options.min\n        check.parameter(minFilter, minFilters)\n        this.minFilter = minFilters[minFilter]\n      }\n\n      if ('mag' in options) {\n        var magFilter = options.mag\n        check.parameter(magFilter, magFilters)\n        this.magFilter = magFilters[magFilter]\n      }\n\n      var wrapS = this.wrapS\n      var wrapT = this.wrapT\n      if ('wrap' in options) {\n        var wrap = options.wrap\n        if (typeof wrap === 'string') {\n          check.parameter(wrap, wrapModes)\n          wrapS = wrapT = wrapModes[wrap]\n        } else if (Array.isArray(wrap)) {\n          check.parameter(wrap[0], wrapModes)\n          check.parameter(wrap[1], wrapModes)\n          wrapS = wrapModes[wrap[0]]\n          wrapT = wrapModes[wrap[1]]\n        }\n      } else {\n        if ('wrapS' in options) {\n          var optWrapS = options.wrapS\n          check.parameter(optWrapS, wrapModes)\n          wrapS = wrapModes[optWrapS]\n        }\n        if ('wrapT' in options) {\n          var optWrapT = options.wrapT\n          check.parameter(optWrapT, wrapModes)\n          wrapT = wrapModes[optWrapT]\n        }\n      }\n      this.wrapS = wrapS\n      this.wrapT = wrapT\n\n      if ('anisotropic' in options) {\n        var anisotropic = options.anisotropic\n        check(typeof anisotropic === 'number' &&\n           anisotropic >= 1 && anisotropic <= limits.maxAnisotropic,\n          'aniso samples must be between 1 and ')\n        this.anisotropic = options.anisotropic\n      }\n\n      if ('mipmap' in options) {\n        var mipmap = options.mipmap\n        switch (typeof mipmap) {\n          case 'string':\n            check.parameter(mipmap, mipmapHint,\n              'invalid mipmap hint')\n            this.mipmapHint = mipmapHint[mipmap]\n            this.genMipmaps = true\n            break\n\n          case 'boolean':\n            this.genMipmaps = !!mipmap\n            break\n\n          case 'object':\n            break\n\n          default:\n            check.raise('invalid mipmap type')\n        }\n      }\n    },\n\n    upload: function () {\n      var target = this.target\n      gl.texParameteri(target, GL_TEXTURE_MIN_FILTER, this.minFilter)\n      gl.texParameteri(target, GL_TEXTURE_MAG_FILTER, this.magFilter)\n      gl.texParameteri(target, GL_TEXTURE_WRAP_S, this.wrapS)\n      gl.texParameteri(target, GL_TEXTURE_WRAP_T, this.wrapT)\n      if (extensions.ext_texture_filter_anisotropic) {\n        gl.texParameteri(target, GL_TEXTURE_MAX_ANISOTROPY_EXT, this.anisotropic)\n      }\n      if (this.genMipmaps) {\n        gl.hint(GL_GENERATE_MIPMAP_HINT, this.mipmapHint)\n        gl.generateMipmap(target)\n      }\n    }\n  })\n\n  // Final pass to merge params and pixel data\n  function checkTextureComplete (params, pixels) {\n    var i, pixmap\n\n    var type = 0\n    var format = 0\n    var internalformat = 0\n    var width = 0\n    var height = 0\n    var channels = 0\n    var compressed = false\n    var needsPoll = false\n    var needsListeners = false\n    var mipMask2D = 0\n    var mipMaskCube = [0, 0, 0, 0, 0, 0]\n    var cubeMask = 0\n    var hasMip = false\n    for (i = 0; i < pixels.length; ++i) {\n      pixmap = pixels[i]\n      width = width || (pixmap.width << pixmap.miplevel)\n      height = height || (pixmap.height << pixmap.miplevel)\n      type = type || pixmap.type\n      format = format || pixmap.format\n      internalformat = internalformat || pixmap.internalformat\n      channels = channels || pixmap.channels\n      needsPoll = needsPoll || pixmap.needsPoll\n      needsListeners = needsListeners || pixmap.needsListeners\n      compressed = compressed || pixmap.compressed\n\n      var miplevel = pixmap.miplevel\n      var target = pixmap.target\n      hasMip = hasMip || (miplevel > 0)\n      if (target === GL_TEXTURE_2D) {\n        mipMask2D |= (1 << miplevel)\n      } else {\n        var face = target - GL_TEXTURE_CUBE_MAP_POSITIVE_X\n        mipMaskCube[face] |= (1 << miplevel)\n        cubeMask |= (1 << face)\n      }\n    }\n\n    params.needsPoll = needsPoll\n    params.needsListeners = needsListeners\n    params.width = width\n    params.height = height\n    params.format = format\n    params.internalformat = internalformat\n    params.type = type\n\n    var mipMask = hasMip ? (width << 1) - 1 : 1\n    if (params.target === GL_TEXTURE_2D) {\n      check(cubeMask === 0,\n        'pixmap type must not contain cubemap faces')\n      check(mipMask2D === mipMask, 'missing mip map data')\n    } else {\n      check(cubeMask === ((1 << 6) - 1), 'missing cubemap faces')\n      for (i = 0; i < 6; ++i) {\n        check(mipMaskCube[i] === mipMask, 'missing mip map data')\n      }\n    }\n\n    var mipFilter = (MIPMAP_FILTERS.indexOf(params.minFilter) >= 0)\n    params.genMipmaps = !hasMip && (params.genMipmaps || mipFilter)\n    var useMipmaps = hasMip || params.genMipmaps\n\n    if (!params.minFilter) {\n      params.minFilter = useMipmaps\n        ? GL_LINEAR_MIPMAP_LINEAR\n        : GL_NEAREST\n    } else {\n      check(useMipmaps === mipFilter,\n        'min filter inconsistent with mipmap data')\n    }\n\n    if (useMipmaps) {\n      check(width === height && isPow2(width),\n        'must be a square power of 2 to support mipmaps')\n    }\n\n    if (params.genMipmaps) {\n      check(!compressed, 'mipmap generation not supported for compressed textures')\n    }\n\n    params.wrapS = params.wrapS || GL_CLAMP_TO_EDGE\n    params.wrapT = params.wrapT || GL_CLAMP_TO_EDGE\n    if (params.wrapS !== GL_CLAMP_TO_EDGE ||\n        params.wrapT !== GL_CLAMP_TO_EDGE) {\n      check(isPow2(width) && isPow2(height) && !cubeMask,\n        'incompatible size for wrap mode, image must be a power of 2')\n    }\n\n    if ((type === GL_FLOAT && !extensions.oes_texture_float_linear) ||\n        (type === GL_HALF_FLOAT_OES &&\n          !extensions.oes_texture_half_float_linear)) {\n      check(this.magFilter === GL_NEAREST && this.minFilter === GL_NEAREST,\n        'unsupported filter mode for float texture')\n    }\n\n    for (i = 0; i < pixels.length; ++i) {\n      pixmap = pixels[i]\n      var level = pixmap.miplevel\n      if (pixmap.width) {\n        check(pixmap.width << level === width, 'inconsistent width')\n      }\n      if (pixmap.height) {\n        check(pixmap.height << level === height, 'inconsistent width')\n      }\n      if (pixmap.channels) {\n        check(pixmap.channels === channels, 'inconsistent channels')\n      } else {\n        pixmap.channels = channels\n      }\n      if (pixmap.format) {\n        check(pixmap.format === format, 'inconsistent format')\n      } else {\n        pixmap.format = format\n      }\n      if (pixmap.internalformat) {\n        check(pixmap.internalformat === internalformat, 'inconsistent internalformat')\n      } else {\n        pixmap.internalformat = internalformat\n      }\n      if (pixmap.type) {\n        check(pixmap.type === type, 'inconsistent type')\n      } else {\n        pixmap.type = type\n      }\n      if (pixmap.copy) {\n        check(pixmap.type === GL_UNSIGNED_BYTE &&\n          pixmap.internalformat === GL_RGBA,\n          'incompatible format/type for copyTexImage2D')\n      }\n    }\n  }\n\n  var activeTexture = 0\n  var textureCount = 0\n  var textureSet = {}\n  var pollSet = []\n  var numTexUnits = limits.maxTextureUnits\n  var textureUnits = Array(numTexUnits).map(function () {\n    return null\n  })\n\n  function REGLTexture (target) {\n    this.id = textureCount++\n    this.refCount = 1\n\n    this.target = target\n    this.texture = null\n\n    this.pollId = -1\n\n    this.unit = -1\n    this.bindCount = 0\n\n    // cancels all pending callbacks\n    this.cancelPending = null\n\n    // parsed user inputs\n    this.params = new TexParams(target)\n    this.pixels = []\n  }\n\n  function update (texture, options) {\n    var i\n    clearListeners(texture)\n\n    // Clear parameters and pixel data\n    var params = texture.params\n    TexParams.call(params, texture.target)\n    var pixels = texture.pixels\n    pixels.length = 0\n\n    // parse parameters\n    params.parse(options)\n\n    // parse pixel data\n    function parseMip (target, data) {\n      var mipmap = data.mipmap\n      var pixmap\n      if (Array.isArray(mipmap)) {\n        for (var i = 0; i < mipmap.length; ++i) {\n          pixmap = new PixelInfo(target)\n          pixmap.parseFlags(options)\n          pixmap.parseFlags(data)\n          pixmap.parse(mipmap[i], i)\n          pixels.push(pixmap)\n        }\n      } else {\n        pixmap = new PixelInfo(target)\n        pixmap.parseFlags(options)\n        pixmap.parse(data, 0)\n        pixels.push(pixmap)\n      }\n    }\n    if (texture.target === GL_TEXTURE_2D) {\n      parseMip(GL_TEXTURE_2D, options)\n    } else {\n      var faces = options.faces || options\n      if (Array.isArray(faces)) {\n        check(faces.length === 6,\n          'invalid number of faces in cube map')\n        for (i = 0; i < 6; ++i) {\n          parseMip(GL_TEXTURE_CUBE_MAP_POSITIVE_X + i, faces[i])\n        }\n      } else if (typeof faces === 'string') {\n        // TODO Read dds\n      } else {\n        // Initialize to all empty textures\n        for (i = 0; i < 6; ++i) {\n          parseMip(GL_TEXTURE_CUBE_MAP_POSITIVE_X + i, {})\n        }\n      }\n    }\n\n    // do a second pass to reconcile defaults\n    checkTextureComplete(params, pixels)\n\n    if (params.needsListeners) {\n      hookListeners(texture)\n    }\n\n    if (params.needsPoll) {\n      texture.pollId = pollSet.length\n      pollSet.push(texture)\n    }\n\n    refresh(texture)\n  }\n\n  function refresh (texture) {\n    if (!gl.isTexture(texture.texture)) {\n      texture.texture = gl.createTexture()\n    }\n\n    // Lazy bind\n    var target = texture.target\n    var unit = texture.unit\n    if (unit >= 0) {\n      gl.activeTexture(GL_TEXTURE0 + unit)\n      activeTexture = unit\n    } else {\n      gl.bindTexture(target, texture.texture)\n    }\n\n    // Upload\n    var pixels = texture.pixels\n    var params = texture.params\n    for (var i = 0; i < pixels.length; ++i) {\n      pixels[i].upload(params)\n    }\n    params.upload()\n\n    // Lazy unbind\n    if (unit < 0) {\n      var active = textureUnits[activeTexture]\n      if (active) {\n        // restore binding state\n        gl.bindTexture(active.target, active.texture)\n      } else {\n        // otherwise become new active\n        texture.unit = activeTexture\n        textureUnits[activeTexture] = texture\n      }\n    }\n  }\n\n  function hookListeners (texture) {\n    var params = texture.params\n    var pixels = texture.pixels\n\n    // Appends all the texture data from the buffer to the current\n    function appendDDS (target, miplevel, buffer) {\n      var dds = parseDDS(buffer)\n\n      check(dds.format in colorFormats, 'unsupported dds texture format')\n\n      if (dds.cube) {\n        check(texture.target === GL_TEXTURE_CUBE_MAP)\n\n        // TODO handle cube map DDS\n        check.raise('cube map DDS not yet implemented')\n      } else {\n        check(texture.target === GL_TEXTURE_2D)\n      }\n\n      if (miplevel) {\n        check(dds.pixels.length === 1, 'number of mip levels inconsistent')\n      }\n\n      dds.pixels.forEach(function (pixmap) {\n        var info = new PixelInfo(dds.cube ? pixmap.target : target)\n\n        info.channels = dds.channels\n        info.compressed = dds.compressed\n        info.type = dds.type\n        info.internalformat = dds.format\n        info.format = colorFormats[dds.format]\n\n        info.width = pixmap.width\n        info.height = pixmap.height\n        info.miplevel = pixmap.miplevel || miplevel\n        info.data = pixmap.data\n\n        pixels.push(info)\n      })\n    }\n\n    function onData () {\n      // Update size of any newly loaded pixels\n      for (var i = 0; i < pixels.length; ++i) {\n        var pixelData = pixels[i]\n        var image = pixelData.image\n        var video = pixelData.video\n        var xhr = pixelData.xhr\n        if (image && image.complete) {\n          pixelData.width = image.naturalWidth\n          pixelData.height = image.naturalHeight\n        } else if (video && video.readyState > 2) {\n          pixelData.width = video.width\n          pixelData.height = video.height\n        } else if (xhr && xhr.readyState === 4) {\n          pixels[i] = pixels[pixels.length - 1]\n          pixels.pop()\n          xhr.removeEventListener('readystatechange', refresh)\n          appendDDS(pixelData.target, pixelData.miplevel, xhr.response)\n        }\n      }\n      checkTextureComplete(params, pixels)\n      refresh(texture)\n    }\n\n    pixels.forEach(function (pixelData) {\n      if (pixelData.image && !pixelData.image.complete) {\n        pixelData.image.addEventListener('load', onData)\n      } else if (pixelData.video && pixelData.readyState < 1) {\n        pixelData.video.addEventListener('progress', onData)\n      } else if (pixelData.xhr) {\n        pixelData.xhr.addEventListener('readystatechange', onData)\n      }\n    })\n\n    texture.cancelPending = function detachListeners () {\n      pixels.forEach(function (pixelData) {\n        if (pixelData.image) {\n          pixelData.image.removeEventListener('load', onData)\n        } else if (pixelData.video) {\n          pixelData.video.removeEventListener('progress', onData)\n        } else if (pixelData.xhr) {\n          pixelData.xhr.removeEventListener('readystatechange', onData)\n          pixelData.xhr.abort()\n        }\n      })\n    }\n  }\n\n  function clearListeners (texture) {\n    var cancelPending = texture.cancelPending\n    if (cancelPending) {\n      cancelPending()\n      texture.cancelPending = null\n    }\n    var id = texture.pollId\n    if (id >= 0) {\n      var other = pollSet[id] = pollSet[pollSet.length - 1]\n      other.id = id\n      pollSet.pop()\n      texture.pollId = -1\n    }\n  }\n\n  function destroy (texture) {\n    var handle = texture.texture\n    check(handle, 'must not double destroy texture')\n    var unit = texture.unit\n    var target = texture.target\n    if (unit >= 0) {\n      gl.activeTexture(GL_TEXTURE0 + unit)\n      activeTexture = unit\n      gl.bindTexture(target, null)\n      textureUnits[unit] = null\n    }\n    clearListeners(texture)\n    if (gl.isTexture(handle)) {\n      gl.deleteTexture(handle)\n    }\n    texture.texture = null\n    texture.params = null\n    texture.pixels = null\n    texture.refCount = 0\n    delete textureSet[texture.id]\n  }\n\n  extend(REGLTexture.prototype, {\n    bind: function () {\n      var texture = this\n      texture.bindCount += 1\n      var unit = texture.unit\n      if (unit < 0) {\n        for (var i = 0; i < numTexUnits; ++i) {\n          var other = textureUnits[i]\n          if (other) {\n            if (other.bindCount > 0) {\n              continue\n            }\n            other.unit = -1\n          }\n          textureUnits[i] = texture\n          unit = i\n          break\n        }\n        if (unit >= numTexUnits) {\n          check.raise('insufficient number of texture units')\n        }\n        texture.unit = unit\n        gl.activeTexture(GL_TEXTURE0 + unit)\n        gl.bindTexture(texture.target, texture.texture)\n        activeTexture = unit\n      }\n      return unit\n    },\n\n    unbind: function () {\n      this.bindCount -= 1\n    },\n\n    decRef: function () {\n      if (--this.refCount === 0) {\n        destroy(this)\n      }\n    }\n  })\n\n  function createTexture (options, target) {\n    var texture = new REGLTexture(target)\n    textureSet[texture.id] = texture\n\n    function reglTexture (a0, a1, a2, a3, a4, a5) {\n      var options = a0 || {}\n      if (target === GL_TEXTURE_CUBE_MAP && arguments.length === 6) {\n        options = [a0, a1, a2, a3, a4, a5]\n      }\n      update(texture, options)\n      reglTexture.width = texture.params.width\n      reglTexture.height = texture.params.height\n      return reglTexture\n    }\n\n    reglTexture(options)\n\n    reglTexture._reglType = 'texture'\n    reglTexture._texture = texture\n    reglTexture.destroy = function () {\n      texture.decRef()\n    }\n\n    return reglTexture\n  }\n\n  // Called after context restore\n  function refreshTextures () {\n    values(textureSet).forEach(refresh)\n    for (var i = 0; i < numTexUnits; ++i) {\n      textureUnits[i] = null\n    }\n    activeTexture = 0\n    gl.activeTexture(GL_TEXTURE0)\n  }\n\n  // Called when regl is destroyed\n  function destroyTextures () {\n    for (var i = 0; i < numTexUnits; ++i) {\n      gl.activeTexture(GL_TEXTURE0 + i)\n      gl.bindTexture(GL_TEXTURE_2D, null)\n      textureUnits[i] = null\n    }\n    gl.activeTexture(GL_TEXTURE0)\n    activeTexture = 0\n    values(textureSet).forEach(destroy)\n  }\n\n  // Called once per raf, updates video textures\n  function pollTextures () {\n    pollSet.forEach(refresh)\n  }\n\n  return {\n    create: createTexture,\n    refresh: refreshTextures,\n    clear: destroyTextures,\n    poll: pollTextures,\n    getTexture: function (wrapper) {\n      return null\n    }\n  }\n}\n","module.exports = function wrapUniformState (stringStore) {\n  var uniformState = {}\n\n  function defUniform (name) {\n    var id = stringStore.id(name)\n    var result = uniformState[id]\n    if (!result) {\n      result = uniformState[id] = []\n    }\n    return result\n  }\n\n  return {\n    def: defUniform,\n    uniforms: uniformState\n  }\n}\n","// Error checking and parameter validation\n/* globals btoa */\nvar isTypedArray = require('./is-typed-array')\nvar extend = require('./extend')\n\n// only used for extracting shader names.  if btoa not present, then errors\n// will be slightly crappier\nfunction decodeB64 (str) {\n  if (typeof btoa !== 'undefined') {\n    return btoa(str)\n  }\n  return 'base64:' + str\n}\n\nfunction raise (message) {\n  var error = new Error('(regl) ' + message)\n  console.error(error)\n  throw error\n}\n\nfunction check (pred, message) {\n  if (!pred) {\n    raise(message)\n  }\n}\n\nfunction encolon (message) {\n  if (message) {\n    return ': ' + message\n  }\n  return ''\n}\n\nfunction checkParameter (param, possibilities, message) {\n  if (!(param in possibilities)) {\n    raise('unknown parameter (' + param + ')' + encolon(message) +\n          '. possible values: ' + Object.keys(possibilities).join())\n  }\n}\n\nfunction checkIsTypedArray (data, message) {\n  if (!isTypedArray(data)) {\n    raise(\n      'invalid parameter type' + encolon(message) +\n      '. must be a typed array')\n  }\n}\n\nfunction checkTypeOf (value, type, message) {\n  if (typeof value !== type) {\n    raise(\n      'invalid parameter type' + encolon(message) +\n      '. expected ' + type + ', got ' + (typeof value))\n  }\n}\n\nfunction checkNonNegativeInt (value, message) {\n  if (!((value >= 0) &&\n        ((value | 0) === value))) {\n    raise('invalid parameter type, (' + value + ')' + encolon(message) +\n          '. must be a nonnegative integer')\n  }\n}\n\nfunction checkOneOf (value, list, message) {\n  if (list.indexOf(value) < 0) {\n    raise('invalid value' + encolon(message) + '. must be one of: ' + list)\n  }\n}\n\nfunction leftPad (str, n) {\n  str = str + ''\n  while (str.length < n) {\n    str = ' ' + str\n  }\n  return str\n}\n\nfunction ShaderFile () {\n  this.name = 'unknown'\n  this.lines = []\n  this.index = {}\n  this.hasErrors = false\n}\n\nfunction ShaderLine (number, line) {\n  this.number = number\n  this.line = line\n  this.errors = []\n}\n\nfunction ShaderError (fileNumber, lineNumber, message) {\n  this.file = fileNumber\n  this.line = lineNumber\n  this.message = message\n}\n\nfunction guessCommand () {\n  var error = new Error()\n  var stack = error.stack\n  var pat = /at compileProcedure.*\\n\\s+at.*\\((.*)\\)/.exec(stack)\n  if (pat) {\n    return pat[1]\n  }\n  var pat2 = /at compileProcedure.*\\n\\s+at\\s+(.*)\\n/.exec(stack)\n  if (pat2) {\n    return pat2[1]\n  }\n  return 'unknown'\n}\n\nfunction guessCallSite () {\n  var error = new Error()\n  var stack = error.stack\n  var pat = /at REGLCommand.*\\n\\s+at.*\\((.*)\\)/.exec(stack)\n  if (pat) {\n    return pat[1]\n  }\n  var pat2 = /at REGLCommand.*\\n\\s+at\\s+(.*)\\n/.exec(stack)\n  if (pat2) {\n    return pat2[1]\n  }\n  return 'unknown'\n}\n\nfunction parseSource (source) {\n  var lines = source.split('\\n')\n  var lineNumber = 1\n  var fileNumber = 0\n  var files = {\n    unknown: new ShaderFile(),\n    0: new ShaderFile()\n  }\n  files.unknown.name = files[0].name = guessCommand()\n  files.unknown.lines.push(new ShaderLine(0, ''))\n  for (var i = 0; i < lines.length; ++i) {\n    var line = lines[i]\n    var parts = /^\\s*\\#\\s*(\\w+)\\s+(.+)\\s*$/.exec(line)\n    if (parts) {\n      switch (parts[1]) {\n        case 'line':\n          var lineNumberInfo = /(\\d+)(\\s+\\d+)?/.exec(parts[2])\n          if (lineNumberInfo) {\n            lineNumber = lineNumberInfo[1] | 0\n            if (lineNumberInfo[2]) {\n              fileNumber = lineNumberInfo[2] | 0\n              if (!(fileNumber in files)) {\n                files[fileNumber] = new ShaderFile()\n              }\n            }\n          }\n          break\n        case 'define':\n          var nameInfo = /SHADER_NAME(_B64)?\\s+(.*)$/.exec(parts[2])\n          if (nameInfo) {\n            files[fileNumber].name = (nameInfo[1]\n                ? decodeB64(nameInfo[2])\n                : nameInfo[2])\n          }\n          break\n      }\n    }\n    files[fileNumber].lines.push(new ShaderLine(lineNumber++, line))\n  }\n  Object.keys(files).forEach(function (fileNumber) {\n    var file = files[fileNumber]\n    file.lines.forEach(function (line) {\n      file.index[line.number] = line\n    })\n  })\n  return files\n}\n\nfunction parseErrorLog (errLog) {\n  var result = []\n  errLog.split('\\n').forEach(function (errMsg) {\n    var parts = /^ERROR\\:\\s+(\\d+)\\:(\\d+)\\:\\s*(.*)$/.exec(errMsg)\n    if (parts) {\n      result.push(new ShaderError(\n        parts[1] | 0,\n        parts[2] | 0,\n        parts[3].trim()))\n    } else if (errMsg.length > 0) {\n      result.push(new ShaderError('unknown', 0, errMsg))\n    }\n  })\n  return result\n}\n\nfunction annotateFiles (files, errors) {\n  errors.forEach(function (error) {\n    var file = files[error.file]\n    if (file) {\n      var line = file.index[error.line]\n      if (line) {\n        line.errors.push(error)\n        file.hasErrors = true\n        return\n      }\n    }\n    files.unknown.hasErrors = true\n    files.unknown.lines[0].errors.push(error)\n  })\n}\n\nfunction checkShaderError (gl, shader, source, type) {\n  if (!gl.getShaderParameter(shader, gl.COMPILE_STATUS)) {\n    var errLog = gl.getShaderInfoLog(shader)\n    var typeName = type === gl.FRAGMENT_SHADER ? 'fragment' : 'vertex'\n    var files = parseSource(source)\n    var errors = parseErrorLog(errLog)\n    annotateFiles(files, errors)\n\n    Object.keys(files).forEach(function (fileNumber) {\n      var file = files[fileNumber]\n      if (!file.hasErrors) {\n        return\n      }\n\n      var strings = ['']\n      var styles = ['']\n\n      function push (str, style) {\n        strings.push(str)\n        styles.push(style || '')\n      }\n\n      push('file number ' + fileNumber + ': ' + file.name + '\\n', 'color:red;text-decoration:underline;font-weight:bold')\n\n      file.lines.forEach(function (line) {\n        if (line.errors.length > 0) {\n          push(leftPad(line.number, 4) + '|  ', 'background-color:yellow; font-weight:bold')\n          push(line.line + '\\n', 'color:red; background-color:yellow; font-weight:bold')\n\n          // try to guess token\n          var offset = 0\n          line.errors.forEach(function (error) {\n            var message = error.message\n            var token = /^\\s*\\'(.*)\\'\\s*\\:\\s*(.*)$/.exec(message)\n            if (token) {\n              var tokenPat = token[1]\n              message = token[2]\n              switch (tokenPat) {\n                case 'assign':\n                  tokenPat = '='\n                  break\n              }\n              offset = Math.max(line.line.indexOf(tokenPat, offset), 0)\n            } else {\n              offset = 0\n            }\n\n            push(leftPad('| ', 6))\n            push(leftPad('^^^', offset + 3) + '\\n', 'font-weight:bold')\n            push(leftPad('| ', 6))\n            push(message + '\\n', 'font-weight:bold')\n          })\n          push(leftPad('| ', 6) + '\\n')\n        } else {\n          push(leftPad(line.number, 4) + '|  ')\n          push(line.line + '\\n', 'color:red')\n        }\n      })\n      if (typeof document !== 'undefined') {\n        styles[0] = strings.join('%c')\n        console.log.apply(console, styles)\n      } else {\n        console.log(strings.join(''))\n      }\n    })\n\n    check.raise('Error compiling ' + typeName + ' shader, ' + files[0].name)\n  }\n}\n\nfunction checkLinkError (gl, program, fragShader, vertShader) {\n  if (!gl.getProgramParameter(program, gl.LINK_STATUS)) {\n    var errLog = gl.getProgramInfoLog(program)\n    var fragParse = parseSource(fragShader)\n    var vertParse = parseSource(vertShader)\n\n    var header = 'Error linking program with vertex shader, \"' +\n      vertParse[0].name + '\", and fragment shader \"' + fragParse[0].name + '\"'\n\n    if (typeof document !== 'undefined') {\n      console.log('%c' + header + '\\n%c' + errLog,\n        'color:red;text-decoration:underline;font-weight:bold',\n        'color:red')\n    } else {\n      console.log(header + '\\n' + errLog)\n    }\n    check.raise(header)\n  }\n}\n\nfunction saveCommandRef (object) {\n  object._commandRef = guessCommand()\n}\n\nfunction saveDrawCommandInfo (opts, uniforms, attributes, stringStore) {\n  saveCommandRef(opts)\n\n  function id (str) {\n    if (str) {\n      return stringStore.id(str)\n    }\n    return 0\n  }\n  opts._fragId = id(opts.static.frag)\n  opts._vertId = id(opts.static.vert)\n\n  function addProps (dict, set) {\n    Object.keys(set).forEach(function (u) {\n      dict[stringStore.id(u)] = true\n    })\n  }\n\n  var uniformSet = opts._uniformSet = {}\n  addProps(uniformSet, uniforms.static)\n  addProps(uniformSet, uniforms.dynamic)\n\n  var attributeSet = opts._attributeSet = {}\n  addProps(attributeSet, attributes.static)\n  addProps(attributeSet, attributes.dynamic)\n\n  opts._hasCount = (\n    'count' in opts.static ||\n    'count' in opts.dynamic ||\n    'elements' in opts.static ||\n    'elements' in opts.dynamic)\n}\n\nfunction checkDrawCommandState (\n  drawState,\n  shaderState,\n  uniformState,\n  attributeState,\n  commandRef,\n  fragId,\n  vertId,\n  uniformSet,\n  attributeSet,\n  hasCount) {\n  var i, id\n\n  // Check count\n  if (!hasCount && drawState.count[drawState.count.length - 1] < 0) {\n    raise('missing vertex count in command ' + commandRef + ' - called from ' + guessCallSite())\n  }\n\n  var program = shaderState.program(\n    vertId || shaderState.vert[shaderState.vert.length - 1],\n    fragId || shaderState.frag[shaderState.frag.length - 1])\n\n  // Check uniforms\n  var progUniforms = program.uniforms\n  for (i = 0; i < progUniforms.length; ++i) {\n    id = progUniforms[i].id\n    if (!uniformSet[id] && uniformState.uniforms[id].length <= 0) {\n      raise('missing uniform \"' + progUniforms[i].name + '\" in command ' +\n        commandRef + ' called from ' + guessCallSite())\n    }\n  }\n\n  var progAttributes = program.attributes\n  for (i = 0; i < progAttributes.length; ++i) {\n    id = progAttributes[i].id\n    if (!attributeSet[id] && attributeState.state[id].records.length <= 0) {\n      raise('missing attribute \"' + progAttributes[i].name + '\" in command ' +\n        commandRef + ' called from ' + guessCallSite())\n    }\n  }\n}\n\nmodule.exports = extend(check, {\n  raise: raise,\n  parameter: checkParameter,\n  type: checkTypeOf,\n  isTypedArray: checkIsTypedArray,\n  nni: checkNonNegativeInt,\n  oneOf: checkOneOf,\n  shaderError: checkShaderError,\n  linkError: checkLinkError,\n  callSite: guessCallSite,\n  saveCommandRef: saveCommandRef,\n  drawOk: checkDrawCommandState,\n  saveDrawInfo: saveDrawCommandInfo\n})\n","/* globals performance */\nmodule.exports =\n  (typeof performance !== 'undefined' && performance.now)\n  ? function () { return performance.now() }\n  : function () { return +(new Date()) }\n","var extend = require('./extend')\n\nfunction slice (x) {\n  return Array.prototype.slice.call(x)\n}\n\nmodule.exports = function createEnvironment () {\n  // Unique variable id counter\n  var varCounter = 0\n\n  // Linked values are passed from this scope into the generated code block\n  // Calling link() passes a value into the generated scope and returns\n  // the variable name which it is bound to\n  var linkedNames = []\n  var linkedValues = []\n  function link (value) {\n    var name = 'g' + (varCounter++)\n    linkedNames.push(name)\n    linkedValues.push(value)\n    return name\n  }\n\n  // create a code block\n  function block () {\n    var code = []\n    function push () {\n      code.push.apply(code, slice(arguments))\n    }\n\n    var vars = []\n    function def () {\n      var name = 'v' + (varCounter++)\n      vars.push(name)\n\n      if (arguments.length > 0) {\n        code.push(name, '=')\n        code.push.apply(code, slice(arguments))\n        code.push(';')\n      }\n\n      return name\n    }\n\n    return extend(push, {\n      def: def,\n      toString: function () {\n        return [\n          (vars.length > 0 ? 'var ' + vars + ';' : ''),\n          code.join('')\n        ].join('')\n      }\n    })\n  }\n\n  // procedure list\n  var procedures = {}\n  function proc (name) {\n    var args = []\n    function arg () {\n      var name = 'a' + (varCounter++)\n      args.push(name)\n      return name\n    }\n\n    var body = block()\n    var bodyToString = body.toString\n\n    var result = procedures[name] = extend(body, {\n      arg: arg,\n      toString: function () {\n        return [\n          'function(', args.join(), '){',\n          bodyToString(),\n          '}'\n        ].join('')\n      }\n    })\n\n    return result\n  }\n\n  function compile () {\n    var code = ['\"use strict\";return {']\n    Object.keys(procedures).forEach(function (name) {\n      code.push('\"', name, '\":', procedures[name].toString(), ',')\n    })\n    code.push('}')\n    var proc = Function.apply(null, linkedNames.concat([code.join('')]))\n    return proc.apply(null, linkedValues)\n  }\n\n  return {\n    link: link,\n    block: block,\n    proc: proc,\n    compile: compile\n  }\n}\n","module.exports = function (base, opts) {\n  var keys = Object.keys(opts)\n  for (var i = 0; i < keys.length; ++i) {\n    base[keys[i]] = opts[keys[i]]\n  }\n  return base\n}\n","var isTypedArray = require('./is-typed-array')\n\nmodule.exports = function isNDArrayLike (obj) {\n  return (\n    typeof obj === 'object' &&\n    Array.isArray(obj.shape) &&\n    Array.isArray(obj.stride) &&\n    typeof obj.offset === 'number' &&\n    obj.shape.length === obj.stride.length &&\n    (Array.isArray(obj.data) ||\n      isTypedArray(obj.data)))\n}\n","var dtypes = require('../constants/arraytypes.json')\nmodule.exports = function (x) {\n  return Object.prototype.toString.call(x) in dtypes\n}\n","/* globals document, Image, XMLHttpRequest */\n\nmodule.exports = loadTexture\n\nfunction getExtension (url) {\n  var parts = /\\.(\\w+)(\\?.*)?$/.exec(url)\n  if (parts && parts[1]) {\n    return parts[1].toLowerCase()\n  }\n}\n\nfunction isVideoExtension (url) {\n  return [\n    'avi',\n    'asf',\n    'gifv',\n    'mov',\n    'qt',\n    'yuv',\n    'mpg',\n    'mpeg',\n    'm2v',\n    'mp4',\n    'm4p',\n    'm4v',\n    'ogg',\n    'ogv',\n    'vob',\n    'webm',\n    'wmv'\n  ].indexOf(url) >= 0\n}\n\nfunction isCompressedExtension (url) {\n  return [\n    'dds'\n  ].indexOf(url) >= 0\n}\n\nfunction loadVideo (url, crossOrigin) {\n  var video = document.createElement('video')\n  video.autoplay = true\n  video.loop = true\n  if (crossOrigin) {\n    video.crossOrigin = crossOrigin\n  }\n  video.src = url\n  return video\n}\n\nfunction loadCompressedTexture (url, ext, crossOrigin) {\n  var xhr = new XMLHttpRequest()\n  xhr.responseType = 'arraybuffer'\n  xhr.open('GET', url, true)\n  xhr.send()\n  return xhr\n}\n\nfunction loadImage (url, crossOrigin) {\n  var image = new Image()\n  if (crossOrigin) {\n    image.crossOrigin = crossOrigin\n  }\n  image.src = url\n  return image\n}\n\n// Currently this stuff only works in a DOM environment\nfunction loadTexture (url, crossOrigin) {\n  if (typeof document !== 'undefined') {\n    var ext = getExtension(url)\n    if (isVideoExtension(ext)) {\n      return loadVideo(url, crossOrigin)\n    }\n    if (isCompressedExtension(ext)) {\n      return loadCompressedTexture(url, ext, crossOrigin)\n    }\n    return loadImage(url, crossOrigin)\n  }\n  return null\n}\n","// References:\n//\n// http://msdn.microsoft.com/en-us/library/bb943991.aspx/\n// http://blog.tojicode.com/2011/12/compressed-textures-in-webgl.html\n//\nvar check = require('./check')\n\nmodule.exports = parseDDS\n\nvar DDS_MAGIC = 0x20534444\n\nvar GL_TEXTURE_2D = 0x0DE1\nvar GL_TEXTURE_CUBE_MAP_POSITIVE_X = 0x8515\n\nvar GL_COMPRESSED_RGB_S3TC_DXT1_EXT = 0x83F0\nvar GL_COMPRESSED_RGBA_S3TC_DXT1_EXT = 0x83F1\nvar GL_COMPRESSED_RGBA_S3TC_DXT3_EXT = 0x83F2\nvar GL_COMPRESSED_RGBA_S3TC_DXT5_EXT = 0x83F3\n\nvar GL_COMPRESSED_RGB_ETC1_WEBGL = 0x8D64\n\nvar GL_UNSIGNED_BYTE = 0x1401\n// var GL_HALF_FLOAT_OES = 0x8D61\n// var GL_FLOAT = 0x1406\n\nvar DDSD_MIPMAPCOUNT = 0x20000\n\nvar DDSCAPS2_CUBEMAP = 0x200\nvar DDSCAPS2_CUBEMAP_POSITIVEX = 0x400\nvar DDSCAPS2_CUBEMAP_NEGATIVEX = 0x800\nvar DDSCAPS2_CUBEMAP_POSITIVEY = 0x1000\nvar DDSCAPS2_CUBEMAP_NEGATIVEY = 0x2000\nvar DDSCAPS2_CUBEMAP_POSITIVEZ = 0x4000\nvar DDSCAPS2_CUBEMAP_NEGATIVEZ = 0x8000\n\nvar CUBEMAP_COMPLETE_FACES = (\n  DDSCAPS2_CUBEMAP_POSITIVEX |\n  DDSCAPS2_CUBEMAP_NEGATIVEX |\n  DDSCAPS2_CUBEMAP_POSITIVEY |\n  DDSCAPS2_CUBEMAP_NEGATIVEY |\n  DDSCAPS2_CUBEMAP_POSITIVEZ |\n  DDSCAPS2_CUBEMAP_NEGATIVEZ)\n\nvar DDPF_FOURCC = 0x4\nvar DDPF_RGB = 0x40\n\nvar FOURCC_DXT1 = 0x31545844\nvar FOURCC_DXT3 = 0x33545844\nvar FOURCC_DXT5 = 0x35545844\nvar FOURCC_ETC1 = 0x31435445\n\n// DDS_HEADER {\nvar OFF_SIZE = 1        // int32 dwSize\nvar OFF_FLAGS = 2       // int32 dwFlags\nvar OFF_HEIGHT = 3      // int32 dwHeight\nvar OFF_WIDTH = 4       // int32 dwWidth\n// var OFF_PITCH = 5       // int32 dwPitchOrLinearSize\n// var OFF_DEPTH = 6       // int32 dwDepth\nvar OFF_MIPMAP = 7      // int32 dwMipMapCount; // offset: 7\n// int32[11] dwReserved1\n// DDS_PIXELFORMAT {\n// var OFF_PF_SIZE = 19    // int32 dwSize; // offset: 19\nvar OFF_PF_FLAGS = 20   // int32 dwFlags\nvar OFF_FOURCC = 21     // char[4] dwFourCC\n// var OFF_RGBA_BITS = 22  // int32 dwRGBBitCount\n// var OFF_RED_MASK = 23   // int32 dwRBitMask\n// var OFF_GREEN_MASK = 24 // int32 dwGBitMask\n// var OFF_BLUE_MASK = 25  // int32 dwBBitMask\n// var OFF_ALPHA_MASK = 26 // int32 dwABitMask; // offset: 26\n// }\n// var OFF_CAPS = 27       // int32 dwCaps; // offset: 27\nvar OFF_CAPS2 = 28      // int32 dwCaps2\n// var OFF_CAPS3 = 29      // int32 dwCaps3\n// var OFF_CAPS4 = 30      // int32 dwCaps4\n// int32 dwReserved2 // offset 31\n\nfunction parseDDS (arrayBuffer) {\n  var header = new Int32Array(arrayBuffer)\n  check(header[0] === DDS_MAGIC,\n    'invalid magic number for dds header')\n\n  var flags = header[OFF_FLAGS]\n  check(flags & DDPF_FOURCC,\n    'unsupported dds format')\n\n  var width = header[OFF_WIDTH]\n  var height = header[OFF_HEIGHT]\n\n  var type = GL_UNSIGNED_BYTE\n  var format = 0\n  var blockBytes = 0\n  var channels = 4\n  switch (header[OFF_FOURCC]) {\n    case FOURCC_DXT1:\n      blockBytes = 8\n      if (flags & DDPF_RGB) {\n        channels = 3\n        format = GL_COMPRESSED_RGB_S3TC_DXT1_EXT\n      } else {\n        format = GL_COMPRESSED_RGBA_S3TC_DXT1_EXT\n      }\n      break\n\n    case FOURCC_DXT3:\n      blockBytes = 16\n      format = GL_COMPRESSED_RGBA_S3TC_DXT3_EXT\n      break\n\n    case FOURCC_DXT5:\n      blockBytes = 16\n      format = GL_COMPRESSED_RGBA_S3TC_DXT5_EXT\n      break\n\n    case FOURCC_ETC1:\n      blockBytes = 8\n      format = GL_COMPRESSED_RGB_ETC1_WEBGL\n      break\n\n    // TODO: Implement hdr and uncompressed textures\n\n    default:\n      // Handle uncompressed data here\n      check.raise('unsupported dds texture format')\n  }\n\n  var pixelFlags = header[OFF_PF_FLAGS]\n\n  var mipmapCount = 1\n  if (pixelFlags & DDSD_MIPMAPCOUNT) {\n    mipmapCount = Math.max(1, header[OFF_MIPMAP])\n  }\n\n  var ptr = header[OFF_SIZE] + 4\n\n  var result = {\n    width: width,\n    height: height,\n    channels: channels,\n    format: format,\n    type: type,\n    compressed: true,\n    cube: false,\n    pixels: []\n  }\n\n  function parseMips (target) {\n    var mipWidth = width\n    var mipHeight = height\n\n    for (var i = 0; i < mipmapCount; ++i) {\n      var size =\n        Math.max(1, (mipWidth + 3) >> 2) *\n        Math.max(1, (mipHeight + 3) >> 2) *\n        blockBytes\n      result.pixels.push({\n        target: target,\n        miplevel: i,\n        width: mipWidth,\n        height: mipHeight,\n        data: new Uint8Array(arrayBuffer, ptr, size)\n      })\n      ptr += size\n      mipWidth >>= 1\n      mipHeight >>= 1\n    }\n  }\n\n  var caps2 = header[OFF_CAPS2]\n  var cubemap = !!(caps2 & DDSCAPS2_CUBEMAP)\n  if (cubemap) {\n    check(\n      (caps2 & CUBEMAP_COMPLETE_FACES) === CUBEMAP_COMPLETE_FACES,\n      'missing cubemap faces')\n    result.cube = true\n    for (var i = 0; i < 6; ++i) {\n      parseMips(GL_TEXTURE_CUBE_MAP_POSITIVE_X + i)\n    }\n  } else {\n    parseMips(GL_TEXTURE_2D)\n  }\n\n  return result\n}\n","/* globals requestAnimationFrame, cancelAnimationFrame */\nif (typeof requestAnimationFrame === 'function' &&\n    typeof cancelAnimationFrame === 'function') {\n  module.exports = {\n    next: function (x) { return requestAnimationFrame(x) },\n    cancel: function (x) { return cancelAnimationFrame(x) }\n  }\n} else {\n  module.exports = {\n    next: function (cb) {\n      setTimeout(cb, 30)\n    },\n    cancel: clearTimeout\n  }\n}\n","// A stack for managing the state of a scalar/vector parameter\n\nmodule.exports = function createStack (init, onChange) {\n  var n = init.length\n  var stack = init.slice()\n  var current = init.slice()\n  var dirty = false\n  var forceDirty = true\n\n  function poll () {\n    var ptr = stack.length - n\n    if (dirty || forceDirty) {\n      switch (n) {\n        case 1:\n          onChange(stack[ptr])\n          break\n        case 2:\n          onChange(stack[ptr], stack[ptr + 1])\n          break\n        case 3:\n          onChange(stack[ptr], stack[ptr + 1], stack[ptr + 2])\n          break\n        case 4:\n          onChange(stack[ptr], stack[ptr + 1], stack[ptr + 2], stack[ptr + 3])\n          break\n        case 5:\n          onChange(stack[ptr], stack[ptr + 1], stack[ptr + 2], stack[ptr + 3], stack[ptr + 4])\n          break\n        case 6:\n          onChange(stack[ptr], stack[ptr + 1], stack[ptr + 2], stack[ptr + 3], stack[ptr + 4], stack[ptr + 5])\n          break\n        default:\n          onChange.apply(null, stack.slice(ptr, stack.length))\n      }\n      for (var i = 0; i < n; ++i) {\n        current[i] = stack[ptr + i]\n      }\n      forceDirty = dirty = false\n    }\n  }\n\n  return {\n    push: function () {\n      dirty = false\n      for (var i = 0; i < n; ++i) {\n        var x = arguments[i]\n        dirty = dirty || (x !== current[i])\n        stack.push(x)\n      }\n    },\n\n    pop: function () {\n      dirty = false\n      stack.length -= n\n      for (var i = 0; i < n; ++i) {\n        dirty = dirty || (stack[stack.length - n + i] !== current[i])\n      }\n    },\n\n    poll: poll,\n\n    setDirty: function () {\n      forceDirty = true\n    }\n  }\n}\n","module.exports = function convertToHalfFloat (array) {\n  var floats = new Float32Array(array)\n  var uints = new Uint32Array(floats.buffer)\n  var ushorts = new Uint16Array(array.length)\n\n  for (var i = 0; i < array.length; ++i) {\n    if (isNaN(array[i])) {\n      ushorts[i] = 0xffff\n    } else if (array[i] === Infinity) {\n      ushorts[i] = 0x7c00\n    } else if (array[i] === -Infinity) {\n      ushorts[i] = 0xfc00\n    } else {\n      var x = uints[i]\n\n      var sgn = (x >>> 31) << 15\n      var exp = ((x << 1) >>> 24) - 127\n      var frac = (x >> 13) & ((1 << 10) - 1)\n\n      if (exp < -24) {\n        // round non-representable denormals to 0\n        ushorts[i] = sgn\n      } else if (exp < -14) {\n        // handle denormals\n        var s = -14 - exp\n        ushorts[i] = sgn + ((frac + (1 << 10)) >> s)\n      } else if (exp > 15) {\n        // round overflow to +/- Infinity\n        ushorts[i] = sgn + 0x7c00\n      } else {\n        // otherwise convert directly\n        ushorts[i] = sgn + ((exp + 15) << 10) + frac\n      }\n    }\n  }\n\n  return ushorts\n}\n","module.exports = function (obj) {\n  return Object.keys(obj).map(function (key) { return obj[key] })\n}\n","var check = require('./lib/util/check')\nvar extend = require('./lib/util/extend')\nvar getContext = require('./lib/context')\nvar createStringStore = require('./lib/strings')\nvar wrapExtensions = require('./lib/extension')\nvar wrapLimits = require('./lib/limits')\nvar wrapBuffers = require('./lib/buffer')\nvar wrapElements = require('./lib/elements')\nvar wrapTextures = require('./lib/texture')\nvar wrapRenderbuffers = require('./lib/renderbuffer')\nvar wrapFramebuffers = require('./lib/framebuffer')\nvar wrapUniforms = require('./lib/uniform')\nvar wrapAttributes = require('./lib/attribute')\nvar wrapShaders = require('./lib/shader')\nvar wrapDraw = require('./lib/draw')\nvar wrapContext = require('./lib/state')\nvar createCompiler = require('./lib/compile')\nvar wrapRead = require('./lib/read')\nvar dynamic = require('./lib/dynamic')\nvar raf = require('./lib/util/raf')\nvar clock = require('./lib/util/clock')\n\nvar GL_COLOR_BUFFER_BIT = 16384\nvar GL_DEPTH_BUFFER_BIT = 256\nvar GL_STENCIL_BUFFER_BIT = 1024\n\nvar GL_ARRAY_BUFFER = 34962\nvar GL_TEXTURE_2D = 0x0DE1\nvar GL_TEXTURE_CUBE_MAP = 0x8513\n\nvar CONTEXT_LOST_EVENT = 'webglcontextlost'\nvar CONTEXT_RESTORED_EVENT = 'webglcontextrestored'\n\nmodule.exports = function wrapREGL () {\n  var args = getContext(Array.prototype.slice.call(arguments))\n  var gl = args.gl\n  var options = args.options\n\n  var stringStore = createStringStore()\n\n  var extensionState = wrapExtensions(gl)\n  var extensions = extensionState.extensions\n\n  var viewportState = {\n    width: gl.drawingBufferWidth,\n    height: gl.drawingBufferHeight\n  }\n\n  var limits = wrapLimits(\n    gl,\n    extensions)\n\n  var bufferState = wrapBuffers(gl)\n\n  var elementState = wrapElements(\n    gl,\n    extensions,\n    bufferState)\n\n  var uniformState = wrapUniforms(stringStore)\n\n  var attributeState = wrapAttributes(\n    gl,\n    extensions,\n    limits,\n    bufferState,\n    stringStore)\n\n  var shaderState = wrapShaders(\n    gl,\n    attributeState,\n    uniformState,\n    function (program) {\n      return compiler.draw(program)\n    },\n    stringStore)\n\n  var drawState = wrapDraw(\n    gl,\n    extensions,\n    bufferState)\n\n  var textureState = wrapTextures(\n    gl,\n    extensions,\n    limits,\n    poll,\n    viewportState)\n\n  var renderbufferState = wrapRenderbuffers(\n    gl,\n    extensions,\n    limits)\n\n  var framebufferState = wrapFramebuffers(\n    gl,\n    extensions,\n    limits,\n    textureState,\n    renderbufferState)\n\n  var frameState = {\n    count: 0,\n    start: clock(),\n    dt: 0,\n    t: clock(),\n    renderTime: 0,\n    width: gl.drawingBufferWidth,\n    height: gl.drawingBufferHeight,\n    pixelRatio: options.pixelRatio\n  }\n\n  var glState = wrapContext(\n    gl,\n    framebufferState,\n    viewportState)\n\n  var readPixels = wrapRead(gl, poll, viewportState)\n\n  var compiler = createCompiler(\n    gl,\n    stringStore,\n    extensions,\n    limits,\n    bufferState,\n    elementState,\n    textureState,\n    framebufferState,\n    glState,\n    uniformState,\n    attributeState,\n    shaderState,\n    drawState,\n    frameState,\n    poll)\n\n  var canvas = gl.canvas\n\n  var rafCallbacks = []\n  var activeRAF = 0\n  function handleRAF () {\n    activeRAF = raf.next(handleRAF)\n    frameState.count += 1\n\n    if (frameState.width !== gl.drawingBufferWidth ||\n        frameState.height !== gl.drawingBufferHeight) {\n      frameState.width = gl.drawingBufferWidth\n      frameState.height = gl.drawingBufferHeight\n      glState.notifyViewportChanged()\n    }\n\n    var now = clock()\n    frameState.dt = now - frameState.t\n    frameState.t = now\n\n    textureState.poll()\n\n    for (var i = 0; i < rafCallbacks.length; ++i) {\n      var cb = rafCallbacks[i]\n      cb(frameState.count, frameState.t, frameState.dt)\n    }\n    frameState.renderTime = clock() - now\n  }\n\n  function startRAF () {\n    if (!activeRAF && rafCallbacks.length > 0) {\n      handleRAF()\n    }\n  }\n\n  function stopRAF () {\n    if (activeRAF) {\n      raf.cancel(handleRAF)\n      activeRAF = 0\n    }\n  }\n\n  function handleContextLoss (event) {\n    stopRAF()\n    event.preventDefault()\n    if (options.onContextLost) {\n      options.onContextLost()\n    }\n  }\n\n  function handleContextRestored (event) {\n    gl.getError()\n    extensionState.refresh()\n    bufferState.refresh()\n    textureState.refresh()\n    renderbufferState.refresh()\n    framebufferState.refresh()\n    shaderState.refresh()\n    glState.refresh()\n    if (options.onContextRestored) {\n      options.onContextRestored()\n    }\n    handleRAF()\n  }\n\n  if (canvas) {\n    canvas.addEventListener(CONTEXT_LOST_EVENT, handleContextLoss, false)\n    canvas.addEventListener(CONTEXT_RESTORED_EVENT, handleContextRestored, false)\n  }\n\n  function destroy () {\n    stopRAF()\n\n    if (canvas) {\n      canvas.removeEventListener(CONTEXT_LOST_EVENT, handleContextLoss)\n      canvas.removeEventListener(CONTEXT_RESTORED_EVENT, handleContextRestored)\n    }\n\n    shaderState.clear()\n    framebufferState.clear()\n    renderbufferState.clear()\n    textureState.clear()\n    bufferState.clear()\n\n    if (options.onDestroy) {\n      options.onDestroy()\n    }\n  }\n\n  function compileProcedure (options) {\n    check(!!options, 'invalid args to regl({...})')\n    check.type(options, 'object', 'invalid args to regl({...})')\n\n    var hasDynamic = false\n\n    function flattenNestedOptions (options) {\n      var result = extend({}, options)\n      delete result.uniforms\n      delete result.attributes\n\n      function merge (name) {\n        if (name in result) {\n          var child = result[name]\n          delete result[name]\n          Object.keys(child).forEach(function (prop) {\n            result[name + '.' + prop] = child[prop]\n          })\n        }\n      }\n      merge('blend')\n      merge('depth')\n      merge('cull')\n      merge('stencil')\n      merge('polygonOffset')\n      merge('scissor')\n      merge('sample')\n\n      return result\n    }\n\n    function separateDynamic (object) {\n      var staticItems = {}\n      var dynamicItems = {}\n      Object.keys(object).forEach(function (option) {\n        var value = object[option]\n        if (dynamic.isDynamic(value)) {\n          hasDynamic = true\n          dynamicItems[option] = dynamic.unbox(value, option)\n        } else {\n          staticItems[option] = value\n        }\n      })\n      return {\n        dynamic: dynamicItems,\n        static: staticItems\n      }\n    }\n\n    var uniforms = separateDynamic(options.uniforms || {})\n    var attributes = separateDynamic(options.attributes || {})\n    var opts = separateDynamic(flattenNestedOptions(options))\n\n    var compiled = compiler.command(\n      opts.static, uniforms.static, attributes.static,\n      opts.dynamic, uniforms.dynamic, attributes.dynamic,\n      hasDynamic)\n\n    var draw = compiled.draw\n    var batch = compiled.batch\n    var scope = compiled.scope\n\n    var EMPTY_ARRAY = []\n    function reserve (count) {\n      while (EMPTY_ARRAY.length < count) {\n        EMPTY_ARRAY.push(null)\n      }\n      return EMPTY_ARRAY\n    }\n\n    check.saveDrawInfo(opts, uniforms, attributes, stringStore)\n\n    function REGLCommand (args, body) {\n      if (typeof args === 'function') {\n        return scope(null, args)\n      } else if (typeof body === 'function') {\n        return scope(args, body)\n      }\n\n      // Runtime shader check.  Removed in production builds\n      check.drawOk(\n        drawState,\n        shaderState,\n        uniformState,\n        attributeState,\n        opts._commandRef,\n        opts._fragId,\n        opts._vertId,\n        opts._uniformSet,\n        opts._attributeSet,\n        opts._hasCount)\n\n      if (typeof args === 'number') {\n        return batch(args | 0, reserve(args | 0))\n      } else if (Array.isArray(args)) {\n        return batch(args.length, args)\n      }\n      return draw(args)\n    }\n\n    return REGLCommand\n  }\n\n  function poll () {\n    framebufferState.poll()\n    glState.poll()\n  }\n\n  function clear (options) {\n    var clearFlags = 0\n\n    poll()\n\n    var c = options.color\n    if (c) {\n      gl.clearColor(+c[0] || 0, +c[1] || 0, +c[2] || 0, +c[3] || 0)\n      clearFlags |= GL_COLOR_BUFFER_BIT\n    }\n    if ('depth' in options) {\n      gl.clearDepth(+options.depth)\n      clearFlags |= GL_DEPTH_BUFFER_BIT\n    }\n    if ('stencil' in options) {\n      gl.clearStencil(options.stencil | 0)\n      clearFlags |= GL_STENCIL_BUFFER_BIT\n    }\n\n    check(!!clearFlags, 'called regl.clear with no buffer specified')\n    gl.clear(clearFlags)\n  }\n\n  function frame (cb) {\n    rafCallbacks.push(cb)\n\n    function cancel () {\n      var index = rafCallbacks.find(function (item) {\n        return item === cb\n      })\n      if (index < 0) {\n        return\n      }\n      rafCallbacks.splice(index, 1)\n      if (rafCallbacks.length <= 0) {\n        stopRAF()\n      }\n    }\n\n    startRAF()\n\n    return {\n      cancel: cancel\n    }\n  }\n\n  return extend(compileProcedure, {\n    // Clear current FBO\n    clear: clear,\n\n    // Short cut for prop binding\n    prop: dynamic.define,\n\n    // executes an empty draw command\n    draw: compileProcedure({}),\n\n    // Resources\n    elements: function (options) {\n      return elementState.create(options)\n    },\n    buffer: function (options) {\n      return bufferState.create(options, GL_ARRAY_BUFFER)\n    },\n    texture: function (options) {\n      return textureState.create(options, GL_TEXTURE_2D)\n    },\n    cube: function (options) {\n      if (arguments.length === 6) {\n        return textureState.create(\n          Array.prototype.slice.call(arguments),\n          GL_TEXTURE_CUBE_MAP)\n      } else {\n        return textureState.create(options, GL_TEXTURE_CUBE_MAP)\n      }\n    },\n    renderbuffer: function (options) {\n      return renderbufferState.create(options)\n    },\n    framebuffer: function (options) {\n      return framebufferState.create(options)\n    },\n    framebufferCube: function (options) {\n      check.raise('framebuffer cube not yet implemented')\n    },\n\n    // Frame rendering\n    frame: frame,\n    stats: frameState,\n\n    // System limits\n    limits: limits,\n\n    // Read pixels\n    read: readPixels,\n\n    // Destroy regl and all associated resources\n    destroy: destroy\n  })\n}\n"]}
require=(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
// shim for using process in browser
var process = module.exports = {};
process.nextTick = (function () {
var canSetImmediate = typeof window !== 'undefined'
&& window.setImmediate;
var canMutationObserver = typeof window !== 'undefined'
&& window.MutationObserver;
var canPost = typeof window !== 'undefined'
&& window.postMessage && window.addEventListener
;
if (canSetImmediate) {
return function (f) { return window.setImmediate(f) };
}
var queue = [];
if (canMutationObserver) {
var hiddenDiv = document.createElement("div");
var observer = new MutationObserver(function () {
var queueList = queue.slice();
queue.length = 0;
queueList.forEach(function (fn) {
fn();
});
});
observer.observe(hiddenDiv, { attributes: true });
return function nextTick(fn) {
if (!queue.length) {
hiddenDiv.setAttribute('yes', 'no');
}
queue.push(fn);
};
}
if (canPost) {
window.addEventListener('message', function (ev) {
var source = ev.source;
if ((source === window || source === null) && ev.data === 'process-tick') {
ev.stopPropagation();
if (queue.length > 0) {
var fn = queue.shift();
fn();
}
}
}, true);
return function nextTick(fn) {
queue.push(fn);
window.postMessage('process-tick', '*');
};
}
return function nextTick(fn) {
setTimeout(fn, 0);
};
})();
process.title = 'browser';
process.browser = true;
process.env = {};
process.argv = [];
function noop() {}
process.on = noop;
process.addListener = noop;
process.once = noop;
process.off = noop;
process.removeListener = noop;
process.removeAllListeners = noop;
process.emit = noop;
process.binding = function (name) {
throw new Error('process.binding is not supported');
};
// TODO(shtylman)
process.cwd = function () { return '/' };
process.chdir = function (dir) {
throw new Error('process.chdir is not supported');
};
},{}],2:[function(require,module,exports){
(function (process){
var keys = require('vkey')
var list = Object.keys(keys)
var down = {}
reset()
module.exports = pressed
if (process.browser) {
window.addEventListener('keydown', keydown, false)
window.addEventListener('keyup', keyup, false)
window.addEventListener('blur', reset, false)
}
function pressed(key) {
return key
? down[key]
: down
}
function reset() {
list.forEach(function(code) {
down[keys[code]] = false
})
}
function keyup(e) {
down[keys[e.keyCode]] = false
}
function keydown(e) {
down[keys[e.keyCode]] = true
}
}).call(this,require('_process'))
//# sourceMappingURL=data:application/json;charset:utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIm5vZGVfbW9kdWxlcy9rZXktcHJlc3NlZC9pbmRleC5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EiLCJmaWxlIjoiZ2VuZXJhdGVkLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXNDb250ZW50IjpbInZhciBrZXlzID0gcmVxdWlyZSgndmtleScpXG52YXIgbGlzdCA9IE9iamVjdC5rZXlzKGtleXMpXG52YXIgZG93biA9IHt9XG5cbnJlc2V0KClcblxubW9kdWxlLmV4cG9ydHMgPSBwcmVzc2VkXG5cbmlmIChwcm9jZXNzLmJyb3dzZXIpIHtcbiAgd2luZG93LmFkZEV2ZW50TGlzdGVuZXIoJ2tleWRvd24nLCBrZXlkb3duLCBmYWxzZSlcbiAgd2luZG93LmFkZEV2ZW50TGlzdGVuZXIoJ2tleXVwJywga2V5dXAsIGZhbHNlKVxuICB3aW5kb3cuYWRkRXZlbnRMaXN0ZW5lcignYmx1cicsIHJlc2V0LCBmYWxzZSlcbn1cblxuZnVuY3Rpb24gcHJlc3NlZChrZXkpIHtcbiAgcmV0dXJuIGtleVxuICAgID8gZG93bltrZXldXG4gICAgOiBkb3duXG59XG5cbmZ1bmN0aW9uIHJlc2V0KCkge1xuICBsaXN0LmZvckVhY2goZnVuY3Rpb24oY29kZSkge1xuICAgIGRvd25ba2V5c1tjb2RlXV0gPSBmYWxzZVxuICB9KVxufVxuXG5mdW5jdGlvbiBrZXl1cChlKSB7XG4gIGRvd25ba2V5c1tlLmtleUNvZGVdXSA9IGZhbHNlXG59XG5cbmZ1bmN0aW9uIGtleWRvd24oZSkge1xuICBkb3duW2tleXNbZS5rZXlDb2RlXV0gPSB0cnVlXG59XG4iXX0=
},{"_process":1,"vkey":3}],3:[function(require,module,exports){
var ua = typeof window !== 'undefined' ? window.navigator.userAgent : ''
, isOSX = /OS X/.test(ua)
, isOpera = /Opera/.test(ua)
, maybeFirefox = !/like Gecko/.test(ua) && !isOpera
var i, output = module.exports = {
0: isOSX ? '<menu>' : '<UNK>'
, 1: '<mouse 1>'
, 2: '<mouse 2>'
, 3: '<break>'
, 4: '<mouse 3>'
, 5: '<mouse 4>'
, 6: '<mouse 5>'
, 8: '<backspace>'
, 9: '<tab>'
, 12: '<clear>'
, 13: '<enter>'
, 16: '<shift>'
, 17: '<control>'
, 18: '<alt>'
, 19: '<pause>'
, 20: '<caps-lock>'
, 21: '<ime-hangul>'
, 23: '<ime-junja>'
, 24: '<ime-final>'
, 25: '<ime-kanji>'
, 27: '<escape>'
, 28: '<ime-convert>'
, 29: '<ime-nonconvert>'
, 30: '<ime-accept>'
, 31: '<ime-mode-change>'
, 27: '<escape>'
, 32: '<space>'
, 33: '<page-up>'
, 34: '<page-down>'
, 35: '<end>'
, 36: '<home>'
, 37: '<left>'
, 38: '<up>'
, 39: '<right>'
, 40: '<down>'
, 41: '<select>'
, 42: '<print>'
, 43: '<execute>'
, 44: '<snapshot>'
, 45: '<insert>'
, 46: '<delete>'
, 47: '<help>'
, 91: '<meta>' // meta-left -- no one handles left and right properly, so we coerce into one.
, 92: '<meta>' // meta-right
, 93: isOSX ? '<meta>' : '<menu>' // chrome,opera,safari all report this for meta-right (osx mbp).
, 95: '<sleep>'
, 106: '<num-*>'
, 107: '<num-+>'
, 108: '<num-enter>'
, 109: '<num-->'
, 110: '<num-.>'
, 111: '<num-/>'
, 144: '<num-lock>'
, 145: '<scroll-lock>'
, 160: '<shift-left>'
, 161: '<shift-right>'
, 162: '<control-left>'
, 163: '<control-right>'
, 164: '<alt-left>'
, 165: '<alt-right>'
, 166: '<browser-back>'
, 167: '<browser-forward>'
, 168: '<browser-refresh>'
, 169: '<browser-stop>'
, 170: '<browser-search>'
, 171: '<browser-favorites>'
, 172: '<browser-home>'
// ff/osx reports '<volume-mute>' for '-'
, 173: isOSX && maybeFirefox ? '-' : '<volume-mute>'
, 174: '<volume-down>'
, 175: '<volume-up>'
, 176: '<next-track>'
, 177: '<prev-track>'
, 178: '<stop>'
, 179: '<play-pause>'
, 180: '<launch-mail>'
, 181: '<launch-media-select>'
, 182: '<launch-app 1>'
, 183: '<launch-app 2>'
, 186: ';'
, 187: '='
, 188: ','
, 189: '-'
, 190: '.'
, 191: '/'
, 192: '`'
, 219: '['
, 220: '\\'
, 221: ']'
, 222: "'"
, 223: '<meta>'
, 224: '<meta>' // firefox reports meta here.
, 226: '<alt-gr>'
, 229: '<ime-process>'
, 231: isOpera ? '`' : '<unicode>'
, 246: '<attention>'
, 247: '<crsel>'
, 248: '<exsel>'
, 249: '<erase-eof>'
, 250: '<play>'
, 251: '<zoom>'
, 252: '<no-name>'
, 253: '<pa-1>'
, 254: '<clear>'
}
for(i = 58; i < 65; ++i) {
output[i] = String.fromCharCode(i)
}
// 0-9
for(i = 48; i < 58; ++i) {
output[i] = (i - 48)+''
}
// A-Z
for(i = 65; i < 91; ++i) {
output[i] = String.fromCharCode(i)
}
// num0-9
for(i = 96; i < 106; ++i) {
output[i] = '<num-'+(i - 96)+'>'
}
// F1-F24
for(i = 112; i < 136; ++i) {
output[i] = 'F'+(i-111)
}
},{}],4:[function(require,module,exports){
var Emitter = require('events/')
module.exports = attach
function attach(element, listener) {
var position = new Emitter
position.x = 0
position.y = 0
position.prevX = 0
position.prevY = 0
position.flush = flush
if (typeof window === 'undefined') {
return position
}
listener = listener || element || window
element = element || document.body
listener.addEventListener('mousemove', (
element === document.body
|| element === window
) ? function(e) {
position.prevX = position.x
position.prevY = position.y
position.x = e.clientX
position.y = e.clientY
position.emit('move', e)
}
: function(e) {
position.prevX = position.x
position.prevY = position.y
var bounds = element.getBoundingClientRect()
position.x = e.clientX - bounds.left
position.y = e.clientY - bounds.top
position.emit('move', e)
}
, false)
return position
function flush() {
this.prevX = this.x
this.prevY = this.y
}
}
},{"events/":5}],5:[function(require,module,exports){
// Copyright Joyent, Inc. and other Node contributors.
//
// Permission is hereby granted, free of charge, to any person obtaining a
// copy of this software and associated documentation files (the
// "Software"), to deal in the Software without restriction, including
// without limitation the rights to use, copy, modify, merge, publish,
// distribute, sublicense, and/or sell copies of the Software, and to permit
// persons to whom the Software is furnished to do so, subject to the
// following conditions:
//
// The above copyright notice and this permission notice shall be included
// in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
// USE OR OTHER DEALINGS IN THE SOFTWARE.
function EventEmitter() {
this._events = this._events || {};
this._maxListeners = this._maxListeners || undefined;
}
module.exports = EventEmitter;
// Backwards-compat with node 0.10.x
EventEmitter.EventEmitter = EventEmitter;
EventEmitter.prototype._events = undefined;
EventEmitter.prototype._maxListeners = undefined;
// By default EventEmitters will print a warning if more than 10 listeners are
// added to it. This is a useful default which helps finding memory leaks.
EventEmitter.defaultMaxListeners = 10;
// Obviously not all Emitters should be limited to 10. This function allows
// that to be increased. Set to zero for unlimited.
EventEmitter.prototype.setMaxListeners = function(n) {
if (!isNumber(n) || n < 0 || isNaN(n))
throw TypeError('n must be a positive number');
this._maxListeners = n;
return this;
};
EventEmitter.prototype.emit = function(type) {
var er, handler, len, args, i, listeners;
if (!this._events)
this._events = {};
// If there is no 'error' event listener then throw.
if (type === 'error') {
if (!this._events.error ||
(isObject(this._events.error) && !this._events.error.length)) {
er = arguments[1];
if (er instanceof Error) {
throw er; // Unhandled 'error' event
}
throw TypeError('Uncaught, unspecified "error" event.');
}
}
handler = this._events[type];
if (isUndefined(handler))
return false;
if (isFunction(handler)) {
switch (arguments.length) {
// fast cases
case 1:
handler.call(this);
break;
case 2:
handler.call(this, arguments[1]);
break;
case 3:
handler.call(this, arguments[1], arguments[2]);
break;
// slower
default:
args = Array.prototype.slice.call(arguments, 1);
handler.apply(this, args);
}
} else if (isObject(handler)) {
args = Array.prototype.slice.call(arguments, 1);
listeners = handler.slice();
len = listeners.length;
for (i = 0; i < len; i++)
listeners[i].apply(this, args);
}
return true;
};
EventEmitter.prototype.addListener = function(type, listener) {
var m;
if (!isFunction(listener))
throw TypeError('listener must be a function');
if (!this._events)
this._events = {};
// To avoid recursion in the case that type === "newListener"! Before
// adding it to the listeners, first emit "newListener".
if (this._events.newListener)
this.emit('newListener', type,
isFunction(listener.listener) ?
listener.listener : listener);
if (!this._events[type])
// Optimize the case of one listener. Don't need the extra array object.
this._events[type] = listener;
else if (isObject(this._events[type]))
// If we've already got an array, just append.
this._events[type].push(listener);
else
// Adding the second element, need to change to array.
this._events[type] = [this._events[type], listener];
// Check for listener leak
if (isObject(this._events[type]) && !this._events[type].warned) {
if (!isUndefined(this._maxListeners)) {
m = this._maxListeners;
} else {
m = EventEmitter.defaultMaxListeners;
}
if (m && m > 0 && this._events[type].length > m) {
this._events[type].warned = true;
console.error('(node) warning: possible EventEmitter memory ' +
'leak detected. %d listeners added. ' +
'Use emitter.setMaxListeners() to increase limit.',
this._events[type].length);
if (typeof console.trace === 'function') {
// not supported in IE 10
console.trace();
}
}
}
return this;
};
EventEmitter.prototype.on = EventEmitter.prototype.addListener;
EventEmitter.prototype.once = function(type, listener) {
if (!isFunction(listener))
throw TypeError('listener must be a function');
var fired = false;
function g() {
this.removeListener(type, g);
if (!fired) {
fired = true;
listener.apply(this, arguments);
}
}
g.listener = listener;
this.on(type, g);
return this;
};
// emits a 'removeListener' event iff the listener was removed
EventEmitter.prototype.removeListener = function(type, listener) {
var list, position, length, i;
if (!isFunction(listener))
throw TypeError('listener must be a function');
if (!this._events || !this._events[type])
return this;
list = this._events[type];
length = list.length;
position = -1;
if (list === listener ||
(isFunction(list.listener) && list.listener === listener)) {
delete this._events[type];
if (this._events.removeListener)
this.emit('removeListener', type, listener);
} else if (isObject(list)) {
for (i = length; i-- > 0;) {
if (list[i] === listener ||
(list[i].listener && list[i].listener === listener)) {
position = i;
break;
}
}
if (position < 0)
return this;
if (list.length === 1) {
list.length = 0;
delete this._events[type];
} else {
list.splice(position, 1);
}
if (this._events.removeListener)
this.emit('removeListener', type, listener);
}
return this;
};
EventEmitter.prototype.removeAllListeners = function(type) {
var key, listeners;
if (!this._events)
return this;
// not listening for removeListener, no need to emit
if (!this._events.removeListener) {
if (arguments.length === 0)
this._events = {};
else if (this._events[type])
delete this._events[type];
return this;
}
// emit removeListener for all listeners on all events
if (arguments.length === 0) {
for (key in this._events) {
if (key === 'removeListener') continue;
this.removeAllListeners(key);
}
this.removeAllListeners('removeListener');
this._events = {};
return this;
}
listeners = this._events[type];
if (isFunction(listeners)) {
this.removeListener(type, listeners);
} else if (listeners) {
// LIFO order
while (listeners.length)
this.removeListener(type, listeners[listeners.length - 1]);
}
delete this._events[type];
return this;
};
EventEmitter.prototype.listeners = function(type) {
var ret;
if (!this._events || !this._events[type])
ret = [];
else if (isFunction(this._events[type]))
ret = [this._events[type]];
else
ret = this._events[type].slice();
return ret;
};
EventEmitter.prototype.listenerCount = function(type) {
if (this._events) {
var evlistener = this._events[type];
if (isFunction(evlistener))
return 1;
else if (evlistener)
return evlistener.length;
}
return 0;
};
EventEmitter.listenerCount = function(emitter, type) {
return emitter.listenerCount(type);
};
function isFunction(arg) {
return typeof arg === 'function';
}
function isNumber(arg) {
return typeof arg === 'number';
}
function isObject(arg) {
return typeof arg === 'object' && arg !== null;
}
function isUndefined(arg) {
return arg === void 0;
}
},{}],6:[function(require,module,exports){
var Emitter = require('events/')
var map = [
'left'
, 'middle'
, 'right'
]
module.exports = pressed
function pressed(element, preventDefault) {
var mouse = new Emitter
mouse.left = false
mouse.right = false
mouse.middle = false
if (typeof window !== 'undefined') {
element = element || window
element.addEventListener('mousedown', mousedown, false)
element.addEventListener('mouseup', mouseup, false)
if (preventDefault) {
element.addEventListener('contextmenu', function(e) {
return e.preventDefault && e.preventDefault() && false
}, false)
}
}
return mouse
function mousedown(e) {
mouse.right = false
mouse[map[e.button]] = true
mouse.emit('down', e)
if (!preventDefault) return
if (!e.preventDefault) return false
e.preventDefault()
e.stopPropagation()
}
function mouseup(e) {
mouse.right = false
mouse[map[e.button]] = false
mouse.emit('up', e)
if (!preventDefault) return
if (!e.preventDefault) return
e.preventDefault()
e.stopPropagation()
}
}
},{"events/":7}],7:[function(require,module,exports){
arguments[4][5][0].apply(exports,arguments)
},{"dup":5}],8:[function(require,module,exports){
/**
* @fileoverview gl-matrix - High performance matrix and vector operations
* @author Brandon Jones
* @author Colin MacKenzie IV
* @version 2.3.2
*/
/* Copyright (c) 2015, Brandon Jones, Colin MacKenzie IV.
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE. */
// END HEADER
exports.glMatrix = require("./gl-matrix/common.js");
exports.mat2 = require("./gl-matrix/mat2.js");
exports.mat2d = require("./gl-matrix/mat2d.js");
exports.mat3 = require("./gl-matrix/mat3.js");
exports.mat4 = require("./gl-matrix/mat4.js");
exports.quat = require("./gl-matrix/quat.js");
exports.vec2 = require("./gl-matrix/vec2.js");
exports.vec3 = require("./gl-matrix/vec3.js");
exports.vec4 = require("./gl-matrix/vec4.js");
},{"./gl-matrix/common.js":9,"./gl-matrix/mat2.js":10,"./gl-matrix/mat2d.js":11,"./gl-matrix/mat3.js":12,"./gl-matrix/mat4.js":13,"./gl-matrix/quat.js":14,"./gl-matrix/vec2.js":15,"./gl-matrix/vec3.js":16,"./gl-matrix/vec4.js":17}],9:[function(require,module,exports){
/* Copyright (c) 2015, Brandon Jones, Colin MacKenzie IV.
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE. */
/**
* @class Common utilities
* @name glMatrix
*/
var glMatrix = {};
// Configuration Constants
glMatrix.EPSILON = 0.000001;
glMatrix.ARRAY_TYPE = (typeof Float32Array !== 'undefined') ? Float32Array : Array;
glMatrix.RANDOM = Math.random;
glMatrix.ENABLE_SIMD = false;
// Capability detection
glMatrix.SIMD_AVAILABLE = (glMatrix.ARRAY_TYPE === Float32Array) && ('SIMD' in this);
glMatrix.USE_SIMD = glMatrix.ENABLE_SIMD && glMatrix.SIMD_AVAILABLE;
/**
* Sets the type of array used when creating new vectors and matrices
*
* @param {Type} type Array type, such as Float32Array or Array
*/
glMatrix.setMatrixArrayType = function(type) {
glMatrix.ARRAY_TYPE = type;
}
var degree = Math.PI / 180;
/**
* Convert Degree To Radian
*
* @param {Number} Angle in Degrees
*/
glMatrix.toRadian = function(a){
return a * degree;
}
/**
* Tests whether or not the arguments have approximately the same value, within an absolute
* or relative tolerance of glMatrix.EPSILON (an absolute tolerance is used for values less
* than or equal to 1.0, and a relative tolerance is used for larger values)
*
* @param {Number} a The first number to test.
* @param {Number} b The second number to test.
* @returns {Boolean} True if the numbers are approximately equal, false otherwise.
*/
glMatrix.equals = function(a, b) {
return Math.abs(a - b) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a), Math.abs(b));
}
module.exports = glMatrix;
},{}],10:[function(require,module,exports){
/* Copyright (c) 2015, Brandon Jones, Colin MacKenzie IV.
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE. */
var glMatrix = require("./common.js");
/**
* @class 2x2 Matrix
* @name mat2
*/
var mat2 = {};
/**
* Creates a new identity mat2
*
* @returns {mat2} a new 2x2 matrix
*/
mat2.create = function() {
var out = new glMatrix.ARRAY_TYPE(4);
out[0] = 1;
out[1] = 0;
out[2] = 0;
out[3] = 1;
return out;
};
/**
* Creates a new mat2 initialized with values from an existing matrix
*
* @param {mat2} a matrix to clone
* @returns {mat2} a new 2x2 matrix
*/
mat2.clone = function(a) {
var out = new glMatrix.ARRAY_TYPE(4);
out[0] = a[0];
out[1] = a[1];
out[2] = a[2];
out[3] = a[3];
return out;
};
/**
* Copy the values from one mat2 to another
*
* @param {mat2} out the receiving matrix
* @param {mat2} a the source matrix
* @returns {mat2} out
*/
mat2.copy = function(out, a) {
out[0] = a[0];
out[1] = a[1];
out[2] = a[2];
out[3] = a[3];
return out;
};
/**
* Set a mat2 to the identity matrix
*
* @param {mat2} out the receiving matrix
* @returns {mat2} out
*/
mat2.identity = function(out) {
out[0] = 1;
out[1] = 0;
out[2] = 0;
out[3] = 1;
return out;
};
/**
* Create a new mat2 with the given values
*
* @param {Number} m00 Component in column 0, row 0 position (index 0)
* @param {Number} m01 Component in column 0, row 1 position (index 1)
* @param {Number} m10 Component in column 1, row 0 position (index 2)
* @param {Number} m11 Component in column 1, row 1 position (index 3)
* @returns {mat2} out A new 2x2 matrix
*/
mat2.fromValues = function(m00, m01, m10, m11) {
var out = new glMatrix.ARRAY_TYPE(4);
out[0] = m00;
out[1] = m01;
out[2] = m10;
out[3] = m11;
return out;
};
/**
* Set the components of a mat2 to the given values
*
* @param {mat2} out the receiving matrix
* @param {Number} m00 Component in column 0, row 0 position (index 0)
* @param {Number} m01 Component in column 0, row 1 position (index 1)
* @param {Number} m10 Component in column 1, row 0 position (index 2)
* @param {Number} m11 Component in column 1, row 1 position (index 3)
* @returns {mat2} out
*/
mat2.set = function(out, m00, m01, m10, m11) {
out[0] = m00;
out[1] = m01;
out[2] = m10;
out[3] = m11;
return out;
};
/**
* Transpose the values of a mat2
*
* @param {mat2} out the receiving matrix
* @param {mat2} a the source matrix
* @returns {mat2} out
*/
mat2.transpose = function(out, a) {
// If we are transposing ourselves we can skip a few steps but have to cache some values
if (out === a) {
var a1 = a[1];
out[1] = a[2];
out[2] = a1;
} else {
out[0] = a[0];
out[1] = a[2];
out[2] = a[1];
out[3] = a[3];
}
return out;
};
/**
* Inverts a mat2
*
* @param {mat2} out the receiving matrix
* @param {mat2} a the source matrix
* @returns {mat2} out
*/
mat2.invert = function(out, a) {
var a0 = a[0], a1 = a[1], a2 = a[2], a3 = a[3],
// Calculate the determinant
det = a0 * a3 - a2 * a1;
if (!det) {
return null;
}
det = 1.0 / det;
out[0] = a3 * det;
out[1] = -a1 * det;
out[2] = -a2 * det;
out[3] = a0 * det;
return out;
};
/**
* Calculates the adjugate of a mat2
*
* @param {mat2} out the receiving matrix
* @param {mat2} a the source matrix
* @returns {mat2} out
*/
mat2.adjoint = function(out, a) {
// Caching this value is nessecary if out == a
var a0 = a[0];
out[0] = a[3];
out[1] = -a[1];
out[2] = -a[2];
out[3] = a0;
return out;
};
/**
* Calculates the determinant of a mat2
*
* @param {mat2} a the source matrix
* @returns {Number} determinant of a
*/
mat2.determinant = function (a) {
return a[0] * a[3] - a[2] * a[1];
};
/**
* Multiplies two mat2's
*
* @param {mat2} out the receiving matrix
* @param {mat2} a the first operand
* @param {mat2} b the second operand
* @returns {mat2} out
*/
mat2.multiply = function (out, a, b) {
var a0 = a[0], a1 = a[1], a2 = a[2], a3 = a[3];
var b0 = b[0], b1 = b[1], b2 = b[2], b3 = b[3];
out[0] = a0 * b0 + a2 * b1;
out[1] = a1 * b0 + a3 * b1;
out[2] = a0 * b2 + a2 * b3;
out[3] = a1 * b2 + a3 * b3;
return out;
};
/**
* Alias for {@link mat2.multiply}
* @function
*/
mat2.mul = mat2.multiply;
/**
* Rotates a mat2 by the given angle
*
* @param {mat2} out the receiving matrix
* @param {mat2} a the matrix to rotate
* @param {Number} rad the angle to rotate the matrix by
* @returns {mat2} out
*/
mat2.rotate = function (out, a, rad) {
var a0 = a[0], a1 = a[1], a2 = a[2], a3 = a[3],
s = Math.sin(rad),
c = Math.cos(rad);
out[0] = a0 * c + a2 * s;
out[1] = a1 * c + a3 * s;
out[2] = a0 * -s + a2 * c;
out[3] = a1 * -s + a3 * c;
return out;
};
/**
* Scales the mat2 by the dimensions in the given vec2
*
* @param {mat2} out the receiving matrix
* @param {mat2} a the matrix to rotate
* @param {vec2} v the vec2 to scale the matrix by
* @returns {mat2} out
**/
mat2.scale = function(out, a, v) {
var a0 = a[0], a1 = a[1], a2 = a[2], a3 = a[3],
v0 = v[0], v1 = v[1];
out[0] = a0 * v0;
out[1] = a1 * v0;
out[2] = a2 * v1;
out[3] = a3 * v1;
return out;
};
/**
* Creates a matrix from a given angle
* This is equivalent to (but much faster than):
*
* mat2.identity(dest);
* mat2.rotate(dest, dest, rad);
*
* @param {mat2} out mat2 receiving operation result
* @param {Number} rad the angle to rotate the matrix by
* @returns {mat2} out
*/
mat2.fromRotation = function(out, rad) {
var s = Math.sin(rad),
c = Math.cos(rad);
out[0] = c;
out[1] = s;
out[2] = -s;
out[3] = c;
return out;
}
/**
* Creates a matrix from a vector scaling
* This is equivalent to (but much faster than):
*
* mat2.identity(dest);
* mat2.scale(dest, dest, vec);
*
* @param {mat2} out mat2 receiving operation result
* @param {vec2} v Scaling vector
* @returns {mat2} out
*/
mat2.fromScaling = function(out, v) {
out[0] = v[0];
out[1] = 0;
out[2] = 0;
out[3] = v[1];
return out;
}
/**
* Returns a string representation of a mat2
*
* @param {mat2} mat matrix to represent as a string
* @returns {String} string representation of the matrix
*/
mat2.str = function (a) {
return 'mat2(' + a[0] + ', ' + a[1] + ', ' + a[2] + ', ' + a[3] + ')';
};
/**
* Returns Frobenius norm of a mat2
*
* @param {mat2} a the matrix to calculate Frobenius norm of
* @returns {Number} Frobenius norm
*/
mat2.frob = function (a) {
return(Math.sqrt(Math.pow(a[0], 2) + Math.pow(a[1], 2) + Math.pow(a[2], 2) + Math.pow(a[3], 2)))
};
/**
* Returns L, D and U matrices (Lower triangular, Diagonal and Upper triangular) by factorizing the input matrix
* @param {mat2} L the lower triangular matrix
* @param {mat2} D the diagonal matrix
* @param {mat2} U the upper triangular matrix
* @param {mat2} a the input matrix to factorize
*/
mat2.LDU = function (L, D, U, a) {
L[2] = a[2]/a[0];
U[0] = a[0];
U[1] = a[1];
U[3] = a[3] - L[2] * U[1];
return [L, D, U];
};
/**
* Adds two mat2's
*
* @param {mat2} out the receiving matrix
* @param {mat2} a the first operand
* @param {mat2} b the second operand
* @returns {mat2} out
*/
mat2.add = function(out, a, b) {
out[0] = a[0] + b[0];
out[1] = a[1] + b[1];
out[2] = a[2] + b[2];
out[3] = a[3] + b[3];
return out;
};
/**
* Subtracts matrix b from matrix a
*
* @param {mat2} out the receiving matrix
* @param {mat2} a the first operand
* @param {mat2} b the second operand
* @returns {mat2} out
*/
mat2.subtract = function(out, a, b) {
out[0] = a[0] - b[0];
out[1] = a[1] - b[1];
out[2] = a[2] - b[2];
out[3] = a[3] - b[3];
return out;
};
/**
* Alias for {@link mat2.subtract}
* @function
*/
mat2.sub = mat2.subtract;
/**
* Returns whether or not the matrices have exactly the same elements in the same position (when compared with ===)
*
* @param {mat2} a The first matrix.
* @param {mat2} b The second matrix.
* @returns {Boolean} True if the matrices are equal, false otherwise.
*/
mat2.exactEquals = function (a, b) {
return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3];
};
/**
* Returns whether or not the matrices have approximately the same elements in the same position.
*
* @param {mat2} a The first matrix.
* @param {mat2} b The second matrix.
* @returns {Boolean} True if the matrices are equal, false otherwise.
*/
mat2.equals = function (a, b) {
var a0 = a[0], a1 = a[1], a2 = a[2], a3 = a[3];
var b0 = b[0], b1 = b[1], b2 = b[2], b3 = b[3];
return (Math.abs(a0 - b0) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a0), Math.abs(b0)) &&
Math.abs(a1 - b1) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a1), Math.abs(b1)) &&
Math.abs(a2 - b2) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a2), Math.abs(b2)) &&
Math.abs(a3 - b3) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a3), Math.abs(b3)));
};
/**
* Multiply each element of the matrix by a scalar.
*
* @param {mat2} out the receiving matrix
* @param {mat2} a the matrix to scale
* @param {Number} b amount to scale the matrix's elements by
* @returns {mat2} out
*/
mat2.multiplyScalar = function(out, a, b) {
out[0] = a[0] * b;
out[1] = a[1] * b;
out[2] = a[2] * b;
out[3] = a[3] * b;
return out;
};
/**
* Adds two mat2's after multiplying each element of the second operand by a scalar value.
*
* @param {mat2} out the receiving vector
* @param {mat2} a the first operand
* @param {mat2} b the second operand
* @param {Number} scale the amount to scale b's elements by before adding
* @returns {mat2} out
*/
mat2.multiplyScalarAndAdd = function(out, a, b, scale) {
out[0] = a[0] + (b[0] * scale);
out[1] = a[1] + (b[1] * scale);
out[2] = a[2] + (b[2] * scale);
out[3] = a[3] + (b[3] * scale);
return out;
};
module.exports = mat2;
},{"./common.js":9}],11:[function(require,module,exports){
/* Copyright (c) 2015, Brandon Jones, Colin MacKenzie IV.
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE. */
var glMatrix = require("./common.js");
/**
* @class 2x3 Matrix
* @name mat2d
*
* @description
* A mat2d contains six elements defined as:
* <pre>
* [a, c, tx,
* b, d, ty]
* </pre>
* This is a short form for the 3x3 matrix:
* <pre>
* [a, c, tx,
* b, d, ty,
* 0, 0, 1]
* </pre>
* The last row is ignored so the array is shorter and operations are faster.
*/
var mat2d = {};
/**
* Creates a new identity mat2d
*
* @returns {mat2d} a new 2x3 matrix
*/
mat2d.create = function() {
var out = new glMatrix.ARRAY_TYPE(6);
out[0] = 1;
out[1] = 0;
out[2] = 0;
out[3] = 1;
out[4] = 0;
out[5] = 0;
return out;
};
/**
* Creates a new mat2d initialized with values from an existing matrix
*
* @param {mat2d} a matrix to clone
* @returns {mat2d} a new 2x3 matrix
*/
mat2d.clone = function(a) {
var out = new glMatrix.ARRAY_TYPE(6);
out[0] = a[0];
out[1] = a[1];
out[2] = a[2];
out[3] = a[3];
out[4] = a[4];
out[5] = a[5];
return out;
};
/**
* Copy the values from one mat2d to another
*
* @param {mat2d} out the receiving matrix
* @param {mat2d} a the source matrix
* @returns {mat2d} out
*/
mat2d.copy = function(out, a) {
out[0] = a[0];
out[1] = a[1];
out[2] = a[2];
out[3] = a[3];
out[4] = a[4];
out[5] = a[5];
return out;
};
/**
* Set a mat2d to the identity matrix
*
* @param {mat2d} out the receiving matrix
* @returns {mat2d} out
*/
mat2d.identity = function(out) {
out[0] = 1;
out[1] = 0;
out[2] = 0;
out[3] = 1;
out[4] = 0;
out[5] = 0;
return out;
};
/**
* Create a new mat2d with the given values
*
* @param {Number} a Component A (index 0)
* @param {Number} b Component B (index 1)
* @param {Number} c Component C (index 2)
* @param {Number} d Component D (index 3)
* @param {Number} tx Component TX (index 4)
* @param {Number} ty Component TY (index 5)
* @returns {mat2d} A new mat2d
*/
mat2d.fromValues = function(a, b, c, d, tx, ty) {
var out = new glMatrix.ARRAY_TYPE(6);
out[0] = a;
out[1] = b;
out[2] = c;
out[3] = d;
out[4] = tx;
out[5] = ty;
return out;
};
/**
* Set the components of a mat2d to the given values
*
* @param {mat2d} out the receiving matrix
* @param {Number} a Component A (index 0)
* @param {Number} b Component B (index 1)
* @param {Number} c Component C (index 2)
* @param {Number} d Component D (index 3)
* @param {Number} tx Component TX (index 4)
* @param {Number} ty Component TY (index 5)
* @returns {mat2d} out
*/
mat2d.set = function(out, a, b, c, d, tx, ty) {
out[0] = a;
out[1] = b;
out[2] = c;
out[3] = d;
out[4] = tx;
out[5] = ty;
return out;
};
/**
* Inverts a mat2d
*
* @param {mat2d} out the receiving matrix
* @param {mat2d} a the source matrix
* @returns {mat2d} out
*/
mat2d.invert = function(out, a) {
var aa = a[0], ab = a[1], ac = a[2], ad = a[3],
atx = a[4], aty = a[5];
var det = aa * ad - ab * ac;
if(!det){
return null;
}
det = 1.0 / det;
out[0] = ad * det;
out[1] = -ab * det;
out[2] = -ac * det;
out[3] = aa * det;
out[4] = (ac * aty - ad * atx) * det;
out[5] = (ab * atx - aa * aty) * det;
return out;
};
/**
* Calculates the determinant of a mat2d
*
* @param {mat2d} a the source matrix
* @returns {Number} determinant of a
*/
mat2d.determinant = function (a) {
return a[0] * a[3] - a[1] * a[2];
};
/**
* Multiplies two mat2d's
*
* @param {mat2d} out the receiving matrix
* @param {mat2d} a the first operand
* @param {mat2d} b the second operand
* @returns {mat2d} out
*/
mat2d.multiply = function (out, a, b) {
var a0 = a[0], a1 = a[1], a2 = a[2], a3 = a[3], a4 = a[4], a5 = a[5],
b0 = b[0], b1 = b[1], b2 = b[2], b3 = b[3], b4 = b[4], b5 = b[5];
out[0] = a0 * b0 + a2 * b1;
out[1] = a1 * b0 + a3 * b1;
out[2] = a0 * b2 + a2 * b3;
out[3] = a1 * b2 + a3 * b3;
out[4] = a0 * b4 + a2 * b5 + a4;
out[5] = a1 * b4 + a3 * b5 + a5;
return out;
};
/**
* Alias for {@link mat2d.multiply}
* @function
*/
mat2d.mul = mat2d.multiply;
/**
* Rotates a mat2d by the given angle
*
* @param {mat2d} out the receiving matrix
* @param {mat2d} a the matrix to rotate
* @param {Number} rad the angle to rotate the matrix by
* @returns {mat2d} out
*/
mat2d.rotate = function (out, a, rad) {
var a0 = a[0], a1 = a[1], a2 = a[2], a3 = a[3], a4 = a[4], a5 = a[5],
s = Math.sin(rad),
c = Math.cos(rad);
out[0] = a0 * c + a2 * s;
out[1] = a1 * c + a3 * s;
out[2] = a0 * -s + a2 * c;
out[3] = a1 * -s + a3 * c;
out[4] = a4;
out[5] = a5;
return out;
};
/**
* Scales the mat2d by the dimensions in the given vec2
*
* @param {mat2d} out the receiving matrix
* @param {mat2d} a the matrix to translate
* @param {vec2} v the vec2 to scale the matrix by
* @returns {mat2d} out
**/
mat2d.scale = function(out, a, v) {
var a0 = a[0], a1 = a[1], a2 = a[2], a3 = a[3], a4 = a[4], a5 = a[5],
v0 = v[0], v1 = v[1];
out[0] = a0 * v0;
out[1] = a1 * v0;
out[2] = a2 * v1;
out[3] = a3 * v1;
out[4] = a4;
out[5] = a5;
return out;
};
/**
* Translates the mat2d by the dimensions in the given vec2
*
* @param {mat2d} out the receiving matrix
* @param {mat2d} a the matrix to translate
* @param {vec2} v the vec2 to translate the matrix by
* @returns {mat2d} out
**/
mat2d.translate = function(out, a, v) {
var a0 = a[0], a1 = a[1], a2 = a[2], a3 = a[3], a4 = a[4], a5 = a[5],
v0 = v[0], v1 = v[1];
out[0] = a0;
out[1] = a1;
out[2] = a2;
out[3] = a3;
out[4] = a0 * v0 + a2 * v1 + a4;
out[5] = a1 * v0 + a3 * v1 + a5;
return out;
};
/**
* Creates a matrix from a given angle
* This is equivalent to (but much faster than):
*
* mat2d.identity(dest);
* mat2d.rotate(dest, dest, rad);
*
* @param {mat2d} out mat2d receiving operation result
* @param {Number} rad the angle to rotate the matrix by
* @returns {mat2d} out
*/
mat2d.fromRotation = function(out, rad) {
var s = Math.sin(rad), c = Math.cos(rad);
out[0] = c;
out[1] = s;
out[2] = -s;
out[3] = c;
out[4] = 0;
out[5] = 0;
return out;
}
/**
* Creates a matrix from a vector scaling
* This is equivalent to (but much faster than):
*
* mat2d.identity(dest);
* mat2d.scale(dest, dest, vec);
*
* @param {mat2d} out mat2d receiving operation result
* @param {vec2} v Scaling vector
* @returns {mat2d} out
*/
mat2d.fromScaling = function(out, v) {
out[0] = v[0];
out[1] = 0;
out[2] = 0;
out[3] = v[1];
out[4] = 0;
out[5] = 0;
return out;
}
/**
* Creates a matrix from a vector translation
* This is equivalent to (but much faster than):
*
* mat2d.identity(dest);
* mat2d.translate(dest, dest, vec);
*
* @param {mat2d} out mat2d receiving operation result
* @param {vec2} v Translation vector
* @returns {mat2d} out
*/
mat2d.fromTranslation = function(out, v) {
out[0] = 1;
out[1] = 0;
out[2] = 0;
out[3] = 1;
out[4] = v[0];
out[5] = v[1];
return out;
}
/**
* Returns a string representation of a mat2d
*
* @param {mat2d} a matrix to represent as a string
* @returns {String} string representation of the matrix
*/
mat2d.str = function (a) {
return 'mat2d(' + a[0] + ', ' + a[1] + ', ' + a[2] + ', ' +
a[3] + ', ' + a[4] + ', ' + a[5] + ')';
};
/**
* Returns Frobenius norm of a mat2d
*
* @param {mat2d} a the matrix to calculate Frobenius norm of
* @returns {Number} Frobenius norm
*/
mat2d.frob = function (a) {
return(Math.sqrt(Math.pow(a[0], 2) + Math.pow(a[1], 2) + Math.pow(a[2], 2) + Math.pow(a[3], 2) + Math.pow(a[4], 2) + Math.pow(a[5], 2) + 1))
};
/**
* Adds two mat2d's
*
* @param {mat2d} out the receiving matrix
* @param {mat2d} a the first operand
* @param {mat2d} b the second operand
* @returns {mat2d} out
*/
mat2d.add = function(out, a, b) {
out[0] = a[0] + b[0];
out[1] = a[1] + b[1];
out[2] = a[2] + b[2];
out[3] = a[3] + b[3];
out[4] = a[4] + b[4];
out[5] = a[5] + b[5];
return out;
};
/**
* Subtracts matrix b from matrix a
*
* @param {mat2d} out the receiving matrix
* @param {mat2d} a the first operand
* @param {mat2d} b the second operand
* @returns {mat2d} out
*/
mat2d.subtract = function(out, a, b) {
out[0] = a[0] - b[0];
out[1] = a[1] - b[1];
out[2] = a[2] - b[2];
out[3] = a[3] - b[3];
out[4] = a[4] - b[4];
out[5] = a[5] - b[5];
return out;
};
/**
* Alias for {@link mat2d.subtract}
* @function
*/
mat2d.sub = mat2d.subtract;
/**
* Multiply each element of the matrix by a scalar.
*
* @param {mat2d} out the receiving matrix
* @param {mat2d} a the matrix to scale
* @param {Number} b amount to scale the matrix's elements by
* @returns {mat2d} out
*/
mat2d.multiplyScalar = function(out, a, b) {
out[0] = a[0] * b;
out[1] = a[1] * b;
out[2] = a[2] * b;
out[3] = a[3] * b;
out[4] = a[4] * b;
out[5] = a[5] * b;
return out;
};
/**
* Adds two mat2d's after multiplying each element of the second operand by a scalar value.
*
* @param {mat2d} out the receiving vector
* @param {mat2d} a the first operand
* @param {mat2d} b the second operand
* @param {Number} scale the amount to scale b's elements by before adding
* @returns {mat2d} out
*/
mat2d.multiplyScalarAndAdd = function(out, a, b, scale) {
out[0] = a[0] + (b[0] * scale);
out[1] = a[1] + (b[1] * scale);
out[2] = a[2] + (b[2] * scale);
out[3] = a[3] + (b[3] * scale);
out[4] = a[4] + (b[4] * scale);
out[5] = a[5] + (b[5] * scale);
return out;
};
/**
* Returns whether or not the matrices have exactly the same elements in the same position (when compared with ===)
*
* @param {mat2d} a The first matrix.
* @param {mat2d} b The second matrix.
* @returns {Boolean} True if the matrices are equal, false otherwise.
*/
mat2d.exactEquals = function (a, b) {
return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3] && a[4] === b[4] && a[5] === b[5];
};
/**
* Returns whether or not the matrices have approximately the same elements in the same position.
*
* @param {mat2d} a The first matrix.
* @param {mat2d} b The second matrix.
* @returns {Boolean} True if the matrices are equal, false otherwise.
*/
mat2d.equals = function (a, b) {
var a0 = a[0], a1 = a[1], a2 = a[2], a3 = a[3], a4 = a[4], a5 = a[5];
var b0 = b[0], b1 = b[1], b2 = b[2], b3 = b[3], b4 = b[4], b5 = b[5];
return (Math.abs(a0 - b0) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a0), Math.abs(b0)) &&
Math.abs(a1 - b1) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a1), Math.abs(b1)) &&
Math.abs(a2 - b2) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a2), Math.abs(b2)) &&
Math.abs(a3 - b3) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a3), Math.abs(b3)) &&
Math.abs(a4 - b4) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a4), Math.abs(b4)) &&
Math.abs(a5 - b5) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a5), Math.abs(b5)));
};
module.exports = mat2d;
},{"./common.js":9}],12:[function(require,module,exports){
/* Copyright (c) 2015, Brandon Jones, Colin MacKenzie IV.
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE. */
var glMatrix = require("./common.js");
/**
* @class 3x3 Matrix
* @name mat3
*/
var mat3 = {};
/**
* Creates a new identity mat3
*
* @returns {mat3} a new 3x3 matrix
*/
mat3.create = function() {
var out = new glMatrix.ARRAY_TYPE(9);
out[0] = 1;
out[1] = 0;
out[2] = 0;
out[3] = 0;
out[4] = 1;
out[5] = 0;
out[6] = 0;
out[7] = 0;
out[8] = 1;
return out;
};
/**
* Copies the upper-left 3x3 values into the given mat3.
*
* @param {mat3} out the receiving 3x3 matrix
* @param {mat4} a the source 4x4 matrix
* @returns {mat3} out
*/
mat3.fromMat4 = function(out, a) {
out[0] = a[0];
out[1] = a[1];
out[2] = a[2];
out[3] = a[4];
out[4] = a[5];
out[5] = a[6];
out[6] = a[8];
out[7] = a[9];
out[8] = a[10];
return out;
};
/**
* Creates a new mat3 initialized with values from an existing matrix
*
* @param {mat3} a matrix to clone
* @returns {mat3} a new 3x3 matrix
*/
mat3.clone = function(a) {
var out = new glMatrix.ARRAY_TYPE(9);
out[0] = a[0];
out[1] = a[1];
out[2] = a[2];
out[3] = a[3];
out[4] = a[4];
out[5] = a[5];
out[6] = a[6];
out[7] = a[7];
out[8] = a[8];
return out;
};
/**
* Copy the values from one mat3 to another
*
* @param {mat3} out the receiving matrix
* @param {mat3} a the source matrix
* @returns {mat3} out
*/
mat3.copy = function(out, a) {
out[0] = a[0];
out[1] = a[1];
out[2] = a[2];
out[3] = a[3];
out[4] = a[4];
out[5] = a[5];
out[6] = a[6];
out[7] = a[7];
out[8] = a[8];
return out;
};
/**
* Create a new mat3 with the given values
*
* @param {Number} m00 Component in column 0, row 0 position (index 0)
* @param {Number} m01 Component in column 0, row 1 position (index 1)
* @param {Number} m02 Component in column 0, row 2 position (index 2)
* @param {Number} m10 Component in column 1, row 0 position (index 3)
* @param {Number} m11 Component in column 1, row 1 position (index 4)
* @param {Number} m12 Component in column 1, row 2 position (index 5)
* @param {Number} m20 Component in column 2, row 0 position (index 6)
* @param {Number} m21 Component in column 2, row 1 position (index 7)
* @param {Number} m22 Component in column 2, row 2 position (index 8)
* @returns {mat3} A new mat3
*/
mat3.fromValues = function(m00, m01, m02, m10, m11, m12, m20, m21, m22) {
var out = new glMatrix.ARRAY_TYPE(9);
out[0] = m00;
out[1] = m01;
out[2] = m02;
out[3] = m10;
out[4] = m11;
out[5] = m12;
out[6] = m20;
out[7] = m21;
out[8] = m22;
return out;
};
/**
* Set the components of a mat3 to the given values
*
* @param {mat3} out the receiving matrix
* @param {Number} m00 Component in column 0, row 0 position (index 0)
* @param {Number} m01 Component in column 0, row 1 position (index 1)
* @param {Number} m02 Component in column 0, row 2 position (index 2)
* @param {Number} m10 Component in column 1, row 0 position (index 3)
* @param {Number} m11 Component in column 1, row 1 position (index 4)
* @param {Number} m12 Component in column 1, row 2 position (index 5)
* @param {Number} m20 Component in column 2, row 0 position (index 6)
* @param {Number} m21 Component in column 2, row 1 position (index 7)
* @param {Number} m22 Component in column 2, row 2 position (index 8)
* @returns {mat3} out
*/
mat3.set = function(out, m00, m01, m02, m10, m11, m12, m20, m21, m22) {
out[0] = m00;
out[1] = m01;
out[2] = m02;
out[3] = m10;
out[4] = m11;
out[5] = m12;
out[6] = m20;
out[7] = m21;
out[8] = m22;
return out;
};
/**
* Set a mat3 to the identity matrix
*
* @param {mat3} out the receiving matrix
* @returns {mat3} out
*/
mat3.identity = function(out) {
out[0] = 1;
out[1] = 0;
out[2] = 0;
out[3] = 0;
out[4] = 1;
out[5] = 0;
out[6] = 0;
out[7] = 0;
out[8] = 1;
return out;
};
/**
* Transpose the values of a mat3
*
* @param {mat3} out the receiving matrix
* @param {mat3} a the source matrix
* @returns {mat3} out
*/
mat3.transpose = function(out, a) {
// If we are transposing ourselves we can skip a few steps but have to cache some values
if (out === a) {
var a01 = a[1], a02 = a[2], a12 = a[5];
out[1] = a[3];
out[2] = a[6];
out[3] = a01;
out[5] = a[7];
out[6] = a02;
out[7] = a12;
} else {
out[0] = a[0];
out[1] = a[3];
out[2] = a[6];
out[3] = a[1];
out[4] = a[4];
out[5] = a[7];
out[6] = a[2];
out[7] = a[5];
out[8] = a[8];
}
return out;
};
/**
* Inverts a mat3
*
* @param {mat3} out the receiving matrix
* @param {mat3} a the source matrix
* @returns {mat3} out
*/
mat3.invert = function(out, a) {
var a00 = a[0], a01 = a[1], a02 = a[2],
a10 = a[3], a11 = a[4], a12 = a[5],
a20 = a[6], a21 = a[7], a22 = a[8],
b01 = a22 * a11 - a12 * a21,
b11 = -a22 * a10 + a12 * a20,
b21 = a21 * a10 - a11 * a20,
// Calculate the determinant
det = a00 * b01 + a01 * b11 + a02 * b21;
if (!det) {
return null;
}
det = 1.0 / det;
out[0] = b01 * det;
out[1] = (-a22 * a01 + a02 * a21) * det;
out[2] = (a12 * a01 - a02 * a11) * det;
out[3] = b11 * det;
out[4] = (a22 * a00 - a02 * a20) * det;
out[5] = (-a12 * a00 + a02 * a10) * det;
out[6] = b21 * det;
out[7] = (-a21 * a00 + a01 * a20) * det;
out[8] = (a11 * a00 - a01 * a10) * det;
return out;
};
/**
* Calculates the adjugate of a mat3
*
* @param {mat3} out the receiving matrix
* @param {mat3} a the source matrix
* @returns {mat3} out
*/
mat3.adjoint = function(out, a) {
var a00 = a[0], a01 = a[1], a02 = a[2],
a10 = a[3], a11 = a[4], a12 = a[5],
a20 = a[6], a21 = a[7], a22 = a[8];
out[0] = (a11 * a22 - a12 * a21);
out[1] = (a02 * a21 - a01 * a22);
out[2] = (a01 * a12 - a02 * a11);
out[3] = (a12 * a20 - a10 * a22);
out[4] = (a00 * a22 - a02 * a20);
out[5] = (a02 * a10 - a00 * a12);
out[6] = (a10 * a21 - a11 * a20);
out[7] = (a01 * a20 - a00 * a21);
out[8] = (a00 * a11 - a01 * a10);
return out;
};
/**
* Calculates the determinant of a mat3
*
* @param {mat3} a the source matrix
* @returns {Number} determinant of a
*/
mat3.determinant = function (a) {
var a00 = a[0], a01 = a[1], a02 = a[2],
a10 = a[3], a11 = a[4], a12 = a[5],
a20 = a[6], a21 = a[7], a22 = a[8];
return a00 * (a22 * a11 - a12 * a21) + a01 * (-a22 * a10 + a12 * a20) + a02 * (a21 * a10 - a11 * a20);
};
/**
* Multiplies two mat3's
*
* @param {mat3} out the receiving matrix
* @param {mat3} a the first operand
* @param {mat3} b the second operand
* @returns {mat3} out
*/
mat3.multiply = function (out, a, b) {
var a00 = a[0], a01 = a[1], a02 = a[2],
a10 = a[3], a11 = a[4], a12 = a[5],
a20 = a[6], a21 = a[7], a22 = a[8],
b00 = b[0], b01 = b[1], b02 = b[2],
b10 = b[3], b11 = b[4], b12 = b[5],
b20 = b[6], b21 = b[7], b22 = b[8];
out[0] = b00 * a00 + b01 * a10 + b02 * a20;
out[1] = b00 * a01 + b01 * a11 + b02 * a21;
out[2] = b00 * a02 + b01 * a12 + b02 * a22;
out[3] = b10 * a00 + b11 * a10 + b12 * a20;
out[4] = b10 * a01 + b11 * a11 + b12 * a21;
out[5] = b10 * a02 + b11 * a12 + b12 * a22;
out[6] = b20 * a00 + b21 * a10 + b22 * a20;
out[7] = b20 * a01 + b21 * a11 + b22 * a21;
out[8] = b20 * a02 + b21 * a12 + b22 * a22;
return out;
};
/**
* Alias for {@link mat3.multiply}
* @function
*/
mat3.mul = mat3.multiply;
/**
* Translate a mat3 by the given vector
*
* @param {mat3} out the receiving matrix
* @param {mat3} a the matrix to translate
* @param {vec2} v vector to translate by
* @returns {mat3} out
*/
mat3.translate = function(out, a, v) {
var a00 = a[0], a01 = a[1], a02 = a[2],
a10 = a[3], a11 = a[4], a12 = a[5],
a20 = a[6], a21 = a[7], a22 = a[8],
x = v[0], y = v[1];
out[0] = a00;
out[1] = a01;
out[2] = a02;
out[3] = a10;
out[4] = a11;
out[5] = a12;
out[6] = x * a00 + y * a10 + a20;
out[7] = x * a01 + y * a11 + a21;
out[8] = x * a02 + y * a12 + a22;
return out;
};
/**
* Rotates a mat3 by the given angle
*
* @param {mat3} out the receiving matrix
* @param {mat3} a the matrix to rotate
* @param {Number} rad the angle to rotate the matrix by
* @returns {mat3} out
*/
mat3.rotate = function (out, a, rad) {
var a00 = a[0], a01 = a[1], a02 = a[2],
a10 = a[3], a11 = a[4], a12 = a[5],
a20 = a[6], a21 = a[7], a22 = a[8],
s = Math.sin(rad),
c = Math.cos(rad);
out[0] = c * a00 + s * a10;
out[1] = c * a01 + s * a11;
out[2] = c * a02 + s * a12;
out[3] = c * a10 - s * a00;
out[4] = c * a11 - s * a01;
out[5] = c * a12 - s * a02;
out[6] = a20;
out[7] = a21;
out[8] = a22;
return out;
};
/**
* Scales the mat3 by the dimensions in the given vec2
*
* @param {mat3} out the receiving matrix
* @param {mat3} a the matrix to rotate
* @param {vec2} v the vec2 to scale the matrix by
* @returns {mat3} out
**/
mat3.scale = function(out, a, v) {
var x = v[0], y = v[1];
out[0] = x * a[0];
out[1] = x * a[1];
out[2] = x * a[2];
out[3] = y * a[3];
out[4] = y * a[4];
out[5] = y * a[5];
out[6] = a[6];
out[7] = a[7];
out[8] = a[8];
return out;
};
/**
* Creates a matrix from a vector translation
* This is equivalent to (but much faster than):
*
* mat3.identity(dest);
* mat3.translate(dest, dest, vec);
*
* @param {mat3} out mat3 receiving operation result
* @param {vec2} v Translation vector
* @returns {mat3} out
*/
mat3.fromTranslation = function(out, v) {
out[0] = 1;
out[1] = 0;
out[2] = 0;
out[3] = 0;
out[4] = 1;
out[5] = 0;
out[6] = v[0];
out[7] = v[1];
out[8] = 1;
return out;
}
/**
* Creates a matrix from a given angle
* This is equivalent to (but much faster than):
*
* mat3.identity(dest);
* mat3.rotate(dest, dest, rad);
*
* @param {mat3} out mat3 receiving operation result
* @param {Number} rad the angle to rotate the matrix by
* @returns {mat3} out
*/
mat3.fromRotation = function(out, rad) {
var s = Math.sin(rad), c = Math.cos(rad);
out[0] = c;
out[1] = s;
out[2] = 0;
out[3] = -s;
out[4] = c;
out[5] = 0;
out[6] = 0;
out[7] = 0;
out[8] = 1;
return out;
}
/**
* Creates a matrix from a vector scaling
* This is equivalent to (but much faster than):
*
* mat3.identity(dest);
* mat3.scale(dest, dest, vec);
*
* @param {mat3} out mat3 receiving operation result
* @param {vec2} v Scaling vector
* @returns {mat3} out
*/
mat3.fromScaling = function(out, v) {
out[0] = v[0];
out[1] = 0;
out[2] = 0;
out[3] = 0;
out[4] = v[1];
out[5] = 0;
out[6] = 0;
out[7] = 0;
out[8] = 1;
return out;
}
/**
* Copies the values from a mat2d into a mat3
*
* @param {mat3} out the receiving matrix
* @param {mat2d} a the matrix to copy
* @returns {mat3} out
**/
mat3.fromMat2d = function(out, a) {
out[0] = a[0];
out[1] = a[1];
out[2] = 0;
out[3] = a[2];
out[4] = a[3];
out[5] = 0;
out[6] = a[4];
out[7] = a[5];
out[8] = 1;
return out;
};
/**
* Calculates a 3x3 matrix from the given quaternion
*
* @param {mat3} out mat3 receiving operation result
* @param {quat} q Quaternion to create matrix from
*
* @returns {mat3} out
*/
mat3.fromQuat = function (out, q) {
var x = q[0], y = q[1], z = q[2], w = q[3],
x2 = x + x,
y2 = y + y,
z2 = z + z,
xx = x * x2,
yx = y * x2,
yy = y * y2,
zx = z * x2,
zy = z * y2,
zz = z * z2,
wx = w * x2,
wy = w * y2,
wz = w * z2;
out[0] = 1 - yy - zz;
out[3] = yx - wz;
out[6] = zx + wy;
out[1] = yx + wz;
out[4] = 1 - xx - zz;
out[7] = zy - wx;
out[2] = zx - wy;
out[5] = zy + wx;
out[8] = 1 - xx - yy;
return out;
};
/**
* Calculates a 3x3 normal matrix (transpose inverse) from the 4x4 matrix
*
* @param {mat3} out mat3 receiving operation result
* @param {mat4} a Mat4 to derive the normal matrix from
*
* @returns {mat3} out
*/
mat3.normalFromMat4 = function (out, a) {
var a00 = a[0], a01 = a[1], a02 = a[2], a03 = a[3],
a10 = a[4], a11 = a[5], a12 = a[6], a13 = a[7],
a20 = a[8], a21 = a[9], a22 = a[10], a23 = a[11],
a30 = a[12], a31 = a[13], a32 = a[14], a33 = a[15],
b00 = a00 * a11 - a01 * a10,
b01 = a00 * a12 - a02 * a10,
b02 = a00 * a13 - a03 * a10,
b03 = a01 * a12 - a02 * a11,
b04 = a01 * a13 - a03 * a11,
b05 = a02 * a13 - a03 * a12,
b06 = a20 * a31 - a21 * a30,
b07 = a20 * a32 - a22 * a30,
b08 = a20 * a33 - a23 * a30,
b09 = a21 * a32 - a22 * a31,
b10 = a21 * a33 - a23 * a31,
b11 = a22 * a33 - a23 * a32,
// Calculate the determinant
det = b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;
if (!det) {
return null;
}
det = 1.0 / det;
out[0] = (a11 * b11 - a12 * b10 + a13 * b09) * det;
out[1] = (a12 * b08 - a10 * b11 - a13 * b07) * det;
out[2] = (a10 * b10 - a11 * b08 + a13 * b06) * det;
out[3] = (a02 * b10 - a01 * b11 - a03 * b09) * det;
out[4] = (a00 * b11 - a02 * b08 + a03 * b07) * det;
out[5] = (a01 * b08 - a00 * b10 - a03 * b06) * det;
out[6] = (a31 * b05 - a32 * b04 + a33 * b03) * det;
out[7] = (a32 * b02 - a30 * b05 - a33 * b01) * det;
out[8] = (a30 * b04 - a31 * b02 + a33 * b00) * det;
return out;
};
/**
* Returns a string representation of a mat3
*
* @param {mat3} mat matrix to represent as a string
* @returns {String} string representation of the matrix
*/
mat3.str = function (a) {
return 'mat3(' + a[0] + ', ' + a[1] + ', ' + a[2] + ', ' +
a[3] + ', ' + a[4] + ', ' + a[5] + ', ' +
a[6] + ', ' + a[7] + ', ' + a[8] + ')';
};
/**
* Returns Frobenius norm of a mat3
*
* @param {mat3} a the matrix to calculate Frobenius norm of
* @returns {Number} Frobenius norm
*/
mat3.frob = function (a) {
return(Math.sqrt(Math.pow(a[0], 2) + Math.pow(a[1], 2) + Math.pow(a[2], 2) + Math.pow(a[3], 2) + Math.pow(a[4], 2) + Math.pow(a[5], 2) + Math.pow(a[6], 2) + Math.pow(a[7], 2) + Math.pow(a[8], 2)))
};
/**
* Adds two mat3's
*
* @param {mat3} out the receiving matrix
* @param {mat3} a the first operand
* @param {mat3} b the second operand
* @returns {mat3} out
*/
mat3.add = function(out, a, b) {
out[0] = a[0] + b[0];
out[1] = a[1] + b[1];
out[2] = a[2] + b[2];
out[3] = a[3] + b[3];
out[4] = a[4] + b[4];
out[5] = a[5] + b[5];
out[6] = a[6] + b[6];
out[7] = a[7] + b[7];
out[8] = a[8] + b[8];
return out;
};
/**
* Subtracts matrix b from matrix a
*
* @param {mat3} out the receiving matrix
* @param {mat3} a the first operand
* @param {mat3} b the second operand
* @returns {mat3} out
*/
mat3.subtract = function(out, a, b) {
out[0] = a[0] - b[0];
out[1] = a[1] - b[1];
out[2] = a[2] - b[2];
out[3] = a[3] - b[3];
out[4] = a[4] - b[4];
out[5] = a[5] - b[5];
out[6] = a[6] - b[6];
out[7] = a[7] - b[7];
out[8] = a[8] - b[8];
return out;
};
/**
* Alias for {@link mat3.subtract}
* @function
*/
mat3.sub = mat3.subtract;
/**
* Multiply each element of the matrix by a scalar.
*
* @param {mat3} out the receiving matrix
* @param {mat3} a the matrix to scale
* @param {Number} b amount to scale the matrix's elements by
* @returns {mat3} out
*/
mat3.multiplyScalar = function(out, a, b) {
out[0] = a[0] * b;
out[1] = a[1] * b;
out[2] = a[2] * b;
out[3] = a[3] * b;
out[4] = a[4] * b;
out[5] = a[5] * b;
out[6] = a[6] * b;
out[7] = a[7] * b;
out[8] = a[8] * b;
return out;
};
/**
* Adds two mat3's after multiplying each element of the second operand by a scalar value.
*
* @param {mat3} out the receiving vector
* @param {mat3} a the first operand
* @param {mat3} b the second operand
* @param {Number} scale the amount to scale b's elements by before adding
* @returns {mat3} out
*/
mat3.multiplyScalarAndAdd = function(out, a, b, scale) {
out[0] = a[0] + (b[0] * scale);
out[1] = a[1] + (b[1] * scale);
out[2] = a[2] + (b[2] * scale);
out[3] = a[3] + (b[3] * scale);
out[4] = a[4] + (b[4] * scale);
out[5] = a[5] + (b[5] * scale);
out[6] = a[6] + (b[6] * scale);
out[7] = a[7] + (b[7] * scale);
out[8] = a[8] + (b[8] * scale);
return out;
};
/*
* Returns whether or not the matrices have exactly the same elements in the same position (when compared with ===)
*
* @param {mat3} a The first matrix.
* @param {mat3} b The second matrix.
* @returns {Boolean} True if the matrices are equal, false otherwise.
*/
mat3.exactEquals = function (a, b) {
return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] &&
a[3] === b[3] && a[4] === b[4] && a[5] === b[5] &&
a[6] === b[6] && a[7] === b[7] && a[8] === b[8];
};
/**
* Returns whether or not the matrices have approximately the same elements in the same position.
*
* @param {mat3} a The first matrix.
* @param {mat3} b The second matrix.
* @returns {Boolean} True if the matrices are equal, false otherwise.
*/
mat3.equals = function (a, b) {
var a0 = a[0], a1 = a[1], a2 = a[2], a3 = a[3], a4 = a[4], a5 = a[5], a6 = a[6], a7 = a[7], a8 = a[8];
var b0 = b[0], b1 = b[1], b2 = b[2], b3 = b[3], b4 = b[4], b5 = b[5], b6 = a[6], b7 = b[7], b8 = b[8];
return (Math.abs(a0 - b0) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a0), Math.abs(b0)) &&
Math.abs(a1 - b1) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a1), Math.abs(b1)) &&
Math.abs(a2 - b2) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a2), Math.abs(b2)) &&
Math.abs(a3 - b3) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a3), Math.abs(b3)) &&
Math.abs(a4 - b4) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a4), Math.abs(b4)) &&
Math.abs(a5 - b5) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a5), Math.abs(b5)) &&
Math.abs(a6 - b6) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a6), Math.abs(b6)) &&
Math.abs(a7 - b7) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a7), Math.abs(b7)) &&
Math.abs(a8 - b8) <= glMatrix.EPSILON*Math.max(1.0, Math.abs(a8), Math.abs(b8)));
};
module.exports = mat3;
},{"./common.js":9}],13:[function(require,module,exports){
/* Copyright (c) 2015, Brandon Jones, Colin MacKenzie IV.
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE. */
var glMatrix = require("./common.js");
/**
* @class 4x4 Matrix
* @name mat4
*/
var mat4 = {
scalar: {},
SIMD: {},
};
/**
* Creates a new identity mat4
*
* @returns {mat4} a new 4x4 matrix
*/
mat4.create = function() {
var out = new glMatrix.ARRAY_TYPE(16);
out[0] = 1;
out[1] = 0;
out[2] = 0;
out[3] = 0;
out[4] = 0;
out[5] = 1;
out[6] = 0;
out[7] = 0;
out[8] = 0;
out[9] = 0;
out[10] = 1;
out[11] = 0;
out[12] = 0;
out[13] = 0;
out[14] = 0;
out[15] = 1;
return out;
};
/**
* Creates a new mat4 initialized with values from an existing matrix
*
* @param {mat4} a matrix to clone
* @returns {mat4} a new 4x4 matrix
*/
mat4.clone = function(a) {
var out = new glMatrix.ARRAY_TYPE(16);
out[0] = a[0];
out[1] = a[1];
out[2] = a[2];
out[3] = a[3];
out[4] = a[4];
out[5] = a[5];
out[6] = a[6];
out[7] = a[7];
out[8] = a[8];
out[9] = a[9];
out[10] = a[10];
out[11] = a[11];
out[12] = a[12];
out[13] = a[13];
out[14] = a[14];
out[15] = a[15];
return out;
};
/**
* Copy the values from one mat4 to another
*
* @param {mat4} out the receiving matrix
* @param {mat4} a the source matrix
* @returns {mat4} out
*/
mat4.copy = function(out, a) {
out[0] = a[0];
out[1] = a[1];
out[2] = a[2];
out[3] = a[3];
out[4] = a[4];
out[5] = a[5];
out[6] = a[6];
out[7] = a[7];
out[8] = a[8];
out[9] = a[9];
out[10] = a[10];
out[11] = a[11];
out[12] = a[12];
out[13] = a[13];
out[14] = a[14];
out[15] = a[15];
return out;
};
/**
* Create a new mat4 with the given values
*
* @param {Number} m00 Component in column 0, row 0 position (index 0)
* @param {Number} m01 Component in column 0, row 1 position (index 1)
* @param {Number} m02 Component in column 0, row 2 position (index 2)
* @param {Number} m03 Component in column 0, row 3 position (index 3)
* @param {Number} m10 Component in column 1, row 0 position (index 4)
* @param {Number} m11 Component in column 1, row 1 position (index 5)
* @param {Number} m12 Component in column 1, row 2 position (index 6)
* @param {Number} m13 Component in column 1, row 3 position (index 7)
* @param {Number} m20 Component in column 2, row 0 position (index 8)
* @param {Number} m21 Component in column 2, row 1 position (index 9)
* @param {Number} m22 Component in column 2, row 2 position (index 10)
* @param {Number} m23 Component in column 2, row 3 position (index 11)
* @param {Number} m30 Component in column 3, row 0 position (index 12)
* @param {Number} m31 Component in column 3, row 1 position (index 13)
* @param {Number} m32 Component in column 3, row 2 position (index 14)
* @param {Number} m33 Component in column 3, row 3 position (index 15)
* @returns {mat4} A new mat4
*/
mat4.fromValues = function(m00, m01, m02, m03, m10, m11, m12, m13, m20, m21, m22, m23, m30, m31, m32, m33) {
var out = new glMatrix.ARRAY_TYPE(16);
out[0] = m00;
out[1] = m01;
out[2] = m02;
out[3] = m03;
out[4] = m10;
out[5] = m11;
out[6] = m12;
out[7] = m13;
out[8] = m20;
out[9] = m21;
out[10] = m22;
out[11] = m23;
out[12] = m30;
out[13] = m31;
out[14] = m32;
out[15] = m33;
return out;
};
/**
* Set the components of a mat4 to the given values
*
* @param {mat4} out the receiving matrix
* @param {Number} m00 Component in column 0, row 0 position (index 0)
* @param {Number} m01 Component in column 0, row 1 position (index 1)
* @param {Number} m02 Component in column 0, row 2 position (index 2)
* @param {Number} m03 Component in column 0, row 3 position (index 3)
* @param {Number} m10 Component in column 1, row 0 position (index 4)
* @param {Number} m11 Component in column 1, row 1 position (index 5)
* @param {Number} m12 Component in column 1, row 2 position (index 6)
* @param {Number} m13 Component in column 1, row 3 position (index 7)
* @param {Number} m20 Component in column 2, row 0 position (index 8)
* @param {Number} m21 Component in column 2, row 1 position (index 9)
* @param {Number} m22 Component in column 2, row 2 position (index 10)
* @param {Number} m23 Component in column 2, row 3 position (index 11)
* @param {Number} m30 Component in column 3, row 0 position (index 12)
* @param {Number} m31 Component in column 3, row 1 position (index 13)
* @param {Number} m32 Component in column 3, row 2 position (index 14)
* @param {Number} m33 Component in column 3, row 3 position (index 15)
* @returns {mat4} out
*/
mat4.set = function(out, m00, m01, m02, m03, m10, m11, m12, m13, m20, m21, m22, m23, m30, m31, m32, m33) {
out[0] = m00;
out[1] = m01;
out[2] = m02;
out[3] = m03;
out[4] = m10;
out[5] = m11;
out[6] = m12;
out[7] = m13;
out[8] = m20;
out[9] = m21;
out[10] = m22;
out[11] = m23;
out[12] = m30;
out[13] = m31;
out[14] = m32;
out[15] = m33;
return out;
};
/**
* Set a mat4 to the identity matrix
*
* @param {mat4} out the receiving matrix
* @returns {mat4} out
*/
mat4.identity = function(out) {
out[0] = 1;
out[1] = 0;
out[2] = 0;
out[3] = 0;
out[4] = 0;
out[5] = 1;
out[6] = 0;
out[7] = 0;
out[8] = 0;
out[9] = 0;
out[10] = 1;
out[11] = 0;
out[12] = 0;
out[13] = 0;
out[14] = 0;
out[15] = 1;
return out;
};
/**
* Transpose the values of a mat4 not using SIMD
*
* @param {mat4} out the receiving matrix
* @param {mat4} a the source matrix
* @returns {mat4} out
*/
mat4.scalar.transpose = function(out, a) {
// If we are transposing ourselves we can skip a few steps but have to cache some values
if (out === a) {
var a01 = a[1], a02 = a[2], a03 = a[3],
a12 = a[6], a13 = a[7],
a23 = a[11];
out[1] = a[4];
out[2] = a[8];
out[3] = a[12];
out[4] = a01;
out[6] = a[9];
out[7] = a[13];
out[8] = a02;
out[9] = a12;
out[11] = a[14];
out[12] = a03;
out[13] = a13;
out[14] = a23;
} else {
out[0] = a[0];
out[1] = a[4];
out[2] = a[8];
out[3] = a[12];
out[4] = a[1];
out[5] = a[5];
out[6] = a[9];
out[7] = a[13];
out[8] = a[2];
out[9] = a[6];
out[10] = a[10];
out[11] = a[14];
out[12] = a[3];
out[13] = a[7];
out[14] = a[11];
out[15] = a[15];
}
return out;
};
/**
* Transpose the values of a mat4 using SIMD
*
* @param {mat4} out the receiving matrix
* @param {mat4} a the source matrix
* @returns {mat4} out
*/
mat4.SIMD.transpose = function(out, a) {
var a0, a1, a2, a3,
tmp01, tmp23,
out0, out1, out2, out3;
a0 = SIMD.Float32x4.load(a, 0);
a1 = SIMD.Float32x4.load(a, 4);
a2 = SIMD.Float32x4.load(a, 8);
a3 = SIMD.Float32x4.load(a, 12);
tmp01 = SIMD.Float32x4.shuffle(a0, a1, 0, 1, 4, 5);
tmp23 = SIMD.Float32x4.shuffle(a2, a3, 0, 1, 4, 5);
out0 = SIMD.Float32x4.shuffle(tmp01, tmp23, 0, 2, 4, 6);
out1 = SIMD.Float32x4.shuffle(tmp01, tmp23, 1, 3, 5, 7);
SIMD.Float32x4.store(out, 0, out0);
SIMD.Float32x4.store(out, 4, out1);
tmp01 = SIMD.Float32x4.shuffle(a0, a1, 2, 3, 6, 7);
tmp23 = SIMD.Float32x4.shuffle(a2, a3, 2, 3, 6, 7);
out2 = SIMD.Float32x4.shuffle(tmp01, tmp23, 0, 2, 4, 6);
out3 = SIMD.Float32x4.shuffle(tmp01, tmp23, 1, 3, 5, 7);
SIMD.Float32x4.store(out, 8, out2);
SIMD.Float32x4.store(out, 12, out3);
return out;
};
/**
* Transpse a mat4 using SIMD if available and enabled
*
* @param {mat4} out the receiving matrix
* @param {mat4} a the source matrix
* @returns {mat4} out
*/
mat4.transpose = glMatrix.USE_SIMD ? mat4.SIMD.transpose : mat4.scalar.transpose;
/**
* Inverts a mat4 not using SIMD
*
* @param {mat4} out the receiving matrix
* @param {mat4} a the source matrix
* @returns {mat4} out
*/
mat4.scalar.invert = function(out, a) {
var a00 = a[0], a01 = a[1], a02 = a[2], a03 = a[3],
a10 = a[4], a11 = a[5], a12 = a[6], a13 = a[7],
a20 = a[8], a21 = a[9], a22 = a[10], a23 = a[11],
a30 = a[12], a31 = a[13], a32 = a[14], a33 = a[15],
b00 = a00 * a11 - a01 * a10,
b01 = a00 * a12 - a02 * a10,
b02 = a00 * a13 - a03 * a10,
b03 = a01 * a12 - a02 * a11,
b04 = a01 * a13 - a03 * a11,
b05 = a02 * a13 - a03 * a12,
b06 = a20 * a31 - a21 * a30,
b07 = a20 * a32 - a22 * a30,
b08 = a20 * a33 - a23 * a30,
b09 = a21 * a32 - a22 * a31,
b10 = a21 * a33 - a23 * a31,
b11 = a22 * a33 - a23 * a32,
// Calculate the determinant
det = b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;
if (!det) {
return null;
}
det = 1.0 / det;
out[0] = (a11 * b11 - a12 * b10 + a13 * b09) * det;
out[1] = (a02 * b10 - a01 * b11 - a03 * b09) * det;
out[2] = (a31 * b05 - a32 * b04 + a33 * b03) * det;
out[3] = (a22 * b04 - a21 * b05 - a23 * b03) * det;
out[4] = (a12 * b08 - a10 * b11 - a13 * b07) * det;
out[5] = (a00 * b11 - a02 * b08 + a03 * b07) * det;
out[6] = (a32 * b02 - a30 * b05 - a33 * b01) * det;
out[7] = (a20 * b05 - a22 * b02 + a23 * b01) * det;
out[8] = (a10 * b10 - a11 * b08 + a13 * b06) * det;
out[9] = (a01 * b08 - a00 * b10 - a03 * b06) * det;
out[10] = (a30 * b04 - a31 * b02 + a33 * b00) * det;
out[11] = (a21 * b02 - a20 * b04 - a23 * b00) * det;
out[12] = (a11 * b07 - a10 * b09 - a12 * b06) * det;
out[13] = (a00 * b09 - a01 * b07 + a02 * b06) * det;
out[14] = (a31 * b01 - a30 * b03 - a32 * b00) * det;
out[15] = (a20 * b03 - a21 * b01 + a22 * b00) * det;
return out;
};
/**
* Inverts a mat4 using SIMD
*
* @param {mat4} out the receiving matrix
* @param {mat4} a the source matrix
* @returns {mat4} out
*/
mat4.SIMD.invert = function(out, a) {
var row0, row1, row2, row3,
tmp1,
minor0, minor1, minor2, minor3,
det,
a0 = SIMD.Float32x4.load(a, 0),
a1 = SIMD.Float32x4.load(a, 4),
a2 = SIMD.Float32x4.load(a, 8),
a3 = SIMD.Float32x4.load(a, 12);
// Compute matrix adjugate
tmp1 = SIMD.Float32x4.shuffle(a0, a1, 0, 1, 4, 5);
row1 = SIMD.Float32x4.shuffle(a2, a3, 0, 1, 4, 5);
row0 = SIMD.Float32x4.shuffle(tmp1, row1, 0, 2, 4, 6);
row1 = SIMD.Float32x4.shuffle(row1, tmp1, 1, 3, 5, 7);
tmp1 = SIMD.Float32x4.shuffle(a0, a1, 2, 3, 6, 7);
row3 = SIMD.Float32x4.shuffle(a2, a3, 2, 3, 6, 7);
row2 = SIMD.Float32x4.shuffle(tmp1, row3, 0, 2, 4, 6);
row3 = SIMD.Float32x4.shuffle(row3, tmp1, 1, 3, 5, 7);
tmp1 = SIMD.Float32x4.mul(row2, row3);
tmp1 = SIMD.Float32x4.swizzle(tmp1, 1, 0, 3, 2);
minor0 = SIMD.Float32x4.mul(row1, tmp1);
minor1 = SIMD.Float32x4.mul(row0, tmp1);
tmp1 = SIMD.Float32x4.swizzle(tmp1, 2, 3, 0, 1);
minor0 = SIMD.Float32x4.sub(SIMD.Float32x4.mul(row1, tmp1), minor0);
minor1 = SIMD.Float32x4.sub(SIMD.Float32x4.mul(row0, tmp1), minor1);
minor1 = SIMD.Float32x4.swizzle(minor1, 2, 3, 0, 1);
tmp1 = SIMD.Float32x4.mul(row1, row2);
tmp1 = SIMD.Float32x4.swizzle(tmp1, 1, 0, 3, 2);
minor0 = SIMD.Float32x4.add(SIMD.Float32x4.mul(row3, tmp1), minor0);
minor3 = SIMD.Float32x4.mul(row0, tmp1);
tmp1 = SIMD.Float32x4.swizzle(tmp1, 2, 3, 0, 1);
minor0 = SIMD.Float32x4.sub(minor0, SIMD.Float32x4.mul(row3, tmp1));
minor3 = SIMD.Float32x4.sub(SIMD.Float32x4.mul(row0, tmp1), minor3);
minor3 = SIMD.Float32x4.swizzle(minor3, 2, 3, 0, 1);
tmp1 = SIMD.Float32x4.mul(SIMD.Float32x4.swizzle(row1, 2, 3, 0, 1), row3);
tmp1 = SIMD.Float32x4.swizzle(tmp1, 1, 0, 3, 2);
row2 = SIMD.Float32x4.swizzle(row2, 2, 3, 0, 1);
minor0 = SIMD.Float32x4.add(SIMD.Float32x4.mul(row2, tmp1), minor0);
minor2 = SIMD.Float32x4.mul(row0, tmp1);
tmp1 = SIMD.Float32x4.swizzle(tmp1, 2, 3, 0, 1);
minor0 = SIMD.Float32x4.sub(minor0, SIMD.Float32x4.mul(row2, tmp1));
minor2 = SIMD.Float32x4.sub(SIMD.Float32x4.mul(row0, tmp1), minor2);
minor2 = SIMD.Float32x4.swizzle(minor2, 2, 3, 0, 1);
tmp1 = SIMD.Float32x4.mul(row0, row1);
tmp1 = SIMD.Float32x4.swizzle(tmp1, 1, 0, 3, 2);
minor2 = SIMD.Float32x4.add(SIMD.Float32x4.mul(row3, tmp1), minor2);
minor3 = SIMD.Float32x4.sub(SIMD.Float32x4.mul(row2, tmp1), minor3);
tmp1 = SIMD.Float32x4.swizzle(tmp1, 2, 3, 0, 1);
minor2 = SIMD.Float32x4.sub(SIMD.Float32x4.mul(row3, tmp1), minor2);
minor3 = SIMD.Float32x4.sub(minor3, SIMD.Float32x4.mul(row2, tmp1));
tmp1 = SIMD.Float32x4.mul(row0, row3);
tmp1 = SIMD.Float32x4.swizzle(tmp1, 1, 0, 3, 2);
minor1 = SIMD.Float32x4.sub(minor1, SIMD.Float32x4.mul(row2, tmp1));
minor2 = SIMD.Float32x4.add(SIMD.Float32x4.mul(row1, tmp1), minor2);
tmp1 = SIMD.Float32x4.swizzle(tmp1, 2, 3, 0, 1);
minor1 = SIMD.Float32x4.add(SIMD.Float32x4.mul(row2, tmp1), minor1);
minor2 = SIMD.Float32x4.sub(minor2, SIMD.Float32x4.mul(row1, tmp1));
tmp1 = SIMD.Float32x4.mul(row0, row2);
tmp1 = SIMD.Float32x4.swizzle(tmp1, 1, 0, 3, 2);
minor1 = SIMD.Float32x4.add(SIMD.Float32x4.mul(row3, tmp1), minor1);
minor3 = SIMD.Float32x4.sub(minor3, SIMD.Float32x4.mul(row1, tmp1));
tmp1 = SIMD.Float32x4.swizzle(tmp1, 2, 3, 0, 1);
minor1 = SIMD.Float32x4.sub(minor1, SIMD.Float32x4.mul(row3, tmp1));
minor3 = SIMD.Float32x4.add(SIMD.Float32x4.mul(row1, tmp1), minor3);
// Compute matrix determinant
det = SIMD.Float32x4.mul(row0, minor0);
det = SIMD.Float32x4.add(SIMD.Float32x4.swizzle(det, 2, 3, 0, 1), det);
det = SIMD.Float32x4.add(SIMD.Float32x4.swizzle(det, 1, 0, 3, 2), det);
tmp1 = SIMD.Float32x4.reciprocalApproximation(det);
det = SIMD.Float32x4.sub(
SIMD.Float32x4.add(tmp1, tmp1),
SIMD.Float32x4.mul(det, SIMD.Float32x4.mul(tmp1, tmp1)));
det = SIMD.Float32x4.swizzle(det, 0, 0, 0, 0);
if (!det) {
return null;
}
// Compute matrix inverse
SIMD.Float32x4.store(out, 0, SIMD.Float32x4.mul(det, minor0));
SIMD.Float32x4.store(out, 4, SIMD.Float32x4.mul(det, minor1));
SIMD.Float32x4.store(out, 8, SIMD.Float32x4.mul(det, minor2));
SIMD.Float32x4.store(out, 12, SIMD.Float32x4.mul(det, minor3));
return out;
}
/**
* Inverts a mat4 using SIMD if available and enabled
*
* @param {mat4} out the receiving matrix
* @param {mat4} a the source matrix
* @returns {mat4} out
*/
mat4.invert = glMatrix.USE_SIMD ? mat4.SIMD.invert : mat4.scalar.invert;
/**
* Calculates the adjugate of a mat4 not using SIMD
*
* @param {mat4} out the receiving matrix
* @param {mat4} a the source matrix
* @returns {mat4} out
*/
mat4.scalar.adjoint = function(out, a) {
var a00 = a[0], a01 = a[1], a02 = a[2], a03 = a[3],
a10 = a[4], a11 = a[5], a12 = a[6], a13 = a[7],
a20 = a[8], a21 = a[9], a22 = a[10], a23 = a[11],
a30 = a[12], a31 = a[13], a32 = a[14], a33 = a[15];
out[0] = (a11 * (a22 * a33 - a23 * a32) - a21 * (a12 * a33 - a13 * a32) + a31 * (a12 * a23 - a13 * a22));
out[1] = -(a01 * (a22 * a33 - a23 * a32) - a21 * (a02 * a33 - a03 * a32) + a31 * (a02 * a23 - a03 * a22));
out[2] = (a01 * (a12 * a33 - a13 * a32) - a11 * (a02 * a33 - a03 * a32) + a31 * (a02 * a13 - a03 * a12));
out[3] = -(a01 * (a12 * a23 - a13 * a22) - a11 * (a02 * a23 - a03 * a22) + a21 * (a02 * a13 - a03 * a12));
out[4] = -(a10 * (a22 * a33 - a23 * a32) - a20 * (a12 * a33 - a13 * a32) + a30 * (a12 * a23 - a
View raw

(Sorry about that, but we can’t show files that are this big right now.)

View raw

(Sorry about that, but we can’t show files that are this big right now.)

View raw

(Sorry about that, but we can’t show files that are this big right now.)

View raw

(Sorry about that, but we can’t show files that are this big right now.)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment