Skip to content

Instantly share code, notes, and snippets.

@njcray
Created January 23, 2018 00:10
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save njcray/60000735c813d617c94197e96749246b to your computer and use it in GitHub Desktop.
Save njcray/60000735c813d617c94197e96749246b to your computer and use it in GitHub Desktop.
QVM 3D Contingency Table Visualisation
<html><head>
<TITLE>QVM Contingency Table</TITLE>
</head>
<body onload="rgl.start();">
<div align="center">
<script>/*
* Copyright (C) 2009 Apple Inc. All Rights Reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
* Copyright (2016) Duncan Murdoch - fixed CanvasMatrix4.ortho,
* cleaned up.
*/
/*
CanvasMatrix4 class
This class implements a 4x4 matrix. It has functions which
duplicate the functionality of the OpenGL matrix stack and
glut functions.
IDL:
[
Constructor(in CanvasMatrix4 matrix), // copy passed matrix into new CanvasMatrix4
Constructor(in sequence<float> array) // create new CanvasMatrix4 with 16 floats (row major)
Constructor() // create new CanvasMatrix4 with identity matrix
]
interface CanvasMatrix4 {
attribute float m11;
attribute float m12;
attribute float m13;
attribute float m14;
attribute float m21;
attribute float m22;
attribute float m23;
attribute float m24;
attribute float m31;
attribute float m32;
attribute float m33;
attribute float m34;
attribute float m41;
attribute float m42;
attribute float m43;
attribute float m44;
void load(in CanvasMatrix4 matrix); // copy the values from the passed matrix
void load(in sequence<float> array); // copy 16 floats into the matrix
sequence<float> getAsArray(); // return the matrix as an array of 16 floats
WebGLFloatArray getAsCanvasFloatArray(); // return the matrix as a WebGLFloatArray with 16 values
void makeIdentity(); // replace the matrix with identity
void transpose(); // replace the matrix with its transpose
void invert(); // replace the matrix with its inverse
void translate(in float x, in float y, in float z); // multiply the matrix by passed translation values on the right
void scale(in float x, in float y, in float z); // multiply the matrix by passed scale values on the right
void rotate(in float angle, // multiply the matrix by passed rotation values on the right
in float x, in float y, in float z); // (angle is in degrees)
void multRight(in CanvasMatrix matrix); // multiply the matrix by the passed matrix on the right
void multLeft(in CanvasMatrix matrix); // multiply the matrix by the passed matrix on the left
void ortho(in float left, in float right, // multiply the matrix by the passed ortho values on the right
in float bottom, in float top,
in float near, in float far);
void frustum(in float left, in float right, // multiply the matrix by the passed frustum values on the right
in float bottom, in float top,
in float near, in float far);
void perspective(in float fovy, in float aspect, // multiply the matrix by the passed perspective values on the right
in float zNear, in float zFar);
void lookat(in float eyex, in float eyey, in float eyez, // multiply the matrix by the passed lookat
in float ctrx, in float ctry, in float ctrz, // values on the right
in float upx, in float upy, in float upz);
}
*/
CanvasMatrix4 = function(m)
{
if (typeof m == 'object') {
if ("length" in m && m.length >= 16) {
this.load(m[0], m[1], m[2], m[3], m[4], m[5], m[6], m[7], m[8], m[9], m[10], m[11], m[12], m[13], m[14], m[15]);
return;
}
else if (m instanceof CanvasMatrix4) {
this.load(m);
return;
}
}
this.makeIdentity();
};
CanvasMatrix4.prototype.load = function()
{
if (arguments.length == 1 && typeof arguments[0] == 'object') {
var matrix = arguments[0];
if ("length" in matrix && matrix.length == 16) {
this.m11 = matrix[0];
this.m12 = matrix[1];
this.m13 = matrix[2];
this.m14 = matrix[3];
this.m21 = matrix[4];
this.m22 = matrix[5];
this.m23 = matrix[6];
this.m24 = matrix[7];
this.m31 = matrix[8];
this.m32 = matrix[9];
this.m33 = matrix[10];
this.m34 = matrix[11];
this.m41 = matrix[12];
this.m42 = matrix[13];
this.m43 = matrix[14];
this.m44 = matrix[15];
return;
}
if (arguments[0] instanceof CanvasMatrix4) {
this.m11 = matrix.m11;
this.m12 = matrix.m12;
this.m13 = matrix.m13;
this.m14 = matrix.m14;
this.m21 = matrix.m21;
this.m22 = matrix.m22;
this.m23 = matrix.m23;
this.m24 = matrix.m24;
this.m31 = matrix.m31;
this.m32 = matrix.m32;
this.m33 = matrix.m33;
this.m34 = matrix.m34;
this.m41 = matrix.m41;
this.m42 = matrix.m42;
this.m43 = matrix.m43;
this.m44 = matrix.m44;
return;
}
}
this.makeIdentity();
};
CanvasMatrix4.prototype.getAsArray = function()
{
return [
this.m11, this.m12, this.m13, this.m14,
this.m21, this.m22, this.m23, this.m24,
this.m31, this.m32, this.m33, this.m34,
this.m41, this.m42, this.m43, this.m44
];
};
CanvasMatrix4.prototype.getAsWebGLFloatArray = function()
{
return new WebGLFloatArray(this.getAsArray());
};
CanvasMatrix4.prototype.makeIdentity = function()
{
this.m11 = 1;
this.m12 = 0;
this.m13 = 0;
this.m14 = 0;
this.m21 = 0;
this.m22 = 1;
this.m23 = 0;
this.m24 = 0;
this.m31 = 0;
this.m32 = 0;
this.m33 = 1;
this.m34 = 0;
this.m41 = 0;
this.m42 = 0;
this.m43 = 0;
this.m44 = 1;
};
CanvasMatrix4.prototype.transpose = function()
{
var tmp = this.m12;
this.m12 = this.m21;
this.m21 = tmp;
tmp = this.m13;
this.m13 = this.m31;
this.m31 = tmp;
tmp = this.m14;
this.m14 = this.m41;
this.m41 = tmp;
tmp = this.m23;
this.m23 = this.m32;
this.m32 = tmp;
tmp = this.m24;
this.m24 = this.m42;
this.m42 = tmp;
tmp = this.m34;
this.m34 = this.m43;
this.m43 = tmp;
};
CanvasMatrix4.prototype.invert = function()
{
// Calculate the 4x4 determinant
// If the determinant is zero,
// then the inverse matrix is not unique.
var det = this._determinant4x4();
if (Math.abs(det) < 1e-8)
return null;
this._makeAdjoint();
// Scale the adjoint matrix to get the inverse
this.m11 /= det;
this.m12 /= det;
this.m13 /= det;
this.m14 /= det;
this.m21 /= det;
this.m22 /= det;
this.m23 /= det;
this.m24 /= det;
this.m31 /= det;
this.m32 /= det;
this.m33 /= det;
this.m34 /= det;
this.m41 /= det;
this.m42 /= det;
this.m43 /= det;
this.m44 /= det;
};
CanvasMatrix4.prototype.translate = function(x,y,z)
{
if (x === undefined)
x = 0;
if (y === undefined)
y = 0;
if (z === undefined)
z = 0;
var matrix = new CanvasMatrix4();
matrix.m41 = x;
matrix.m42 = y;
matrix.m43 = z;
this.multRight(matrix);
};
CanvasMatrix4.prototype.scale = function(x,y,z)
{
if (x === undefined)
x = 1;
if (z === undefined) {
if (y === undefined) {
y = x;
z = x;
}
else
z = 1;
}
else if (y === undefined)
y = x;
var matrix = new CanvasMatrix4();
matrix.m11 = x;
matrix.m22 = y;
matrix.m33 = z;
this.multRight(matrix);
};
CanvasMatrix4.prototype.rotate = function(angle,x,y,z)
{
// angles are in degrees. Switch to radians
angle = angle / 180 * Math.PI;
angle /= 2;
var sinA = Math.sin(angle);
var cosA = Math.cos(angle);
var sinA2 = sinA * sinA;
// normalize
var length = Math.sqrt(x * x + y * y + z * z);
if (length === 0) {
// bad vector, just use something reasonable
x = 0;
y = 0;
z = 1;
} else if (length != 1) {
x /= length;
y /= length;
z /= length;
}
var mat = new CanvasMatrix4();
// optimize case where axis is along major axis
if (x == 1 && y === 0 && z === 0) {
mat.m11 = 1;
mat.m12 = 0;
mat.m13 = 0;
mat.m21 = 0;
mat.m22 = 1 - 2 * sinA2;
mat.m23 = 2 * sinA * cosA;
mat.m31 = 0;
mat.m32 = -2 * sinA * cosA;
mat.m33 = 1 - 2 * sinA2;
mat.m14 = mat.m24 = mat.m34 = 0;
mat.m41 = mat.m42 = mat.m43 = 0;
mat.m44 = 1;
} else if (x === 0 && y == 1 && z === 0) {
mat.m11 = 1 - 2 * sinA2;
mat.m12 = 0;
mat.m13 = -2 * sinA * cosA;
mat.m21 = 0;
mat.m22 = 1;
mat.m23 = 0;
mat.m31 = 2 * sinA * cosA;
mat.m32 = 0;
mat.m33 = 1 - 2 * sinA2;
mat.m14 = mat.m24 = mat.m34 = 0;
mat.m41 = mat.m42 = mat.m43 = 0;
mat.m44 = 1;
} else if (x === 0 && y === 0 && z == 1) {
mat.m11 = 1 - 2 * sinA2;
mat.m12 = 2 * sinA * cosA;
mat.m13 = 0;
mat.m21 = -2 * sinA * cosA;
mat.m22 = 1 - 2 * sinA2;
mat.m23 = 0;
mat.m31 = 0;
mat.m32 = 0;
mat.m33 = 1;
mat.m14 = mat.m24 = mat.m34 = 0;
mat.m41 = mat.m42 = mat.m43 = 0;
mat.m44 = 1;
} else {
var x2 = x*x;
var y2 = y*y;
var z2 = z*z;
mat.m11 = 1 - 2 * (y2 + z2) * sinA2;
mat.m12 = 2 * (x * y * sinA2 + z * sinA * cosA);
mat.m13 = 2 * (x * z * sinA2 - y * sinA * cosA);
mat.m21 = 2 * (y * x * sinA2 - z * sinA * cosA);
mat.m22 = 1 - 2 * (z2 + x2) * sinA2;
mat.m23 = 2 * (y * z * sinA2 + x * sinA * cosA);
mat.m31 = 2 * (z * x * sinA2 + y * sinA * cosA);
mat.m32 = 2 * (z * y * sinA2 - x * sinA * cosA);
mat.m33 = 1 - 2 * (x2 + y2) * sinA2;
mat.m14 = mat.m24 = mat.m34 = 0;
mat.m41 = mat.m42 = mat.m43 = 0;
mat.m44 = 1;
}
this.multRight(mat);
};
CanvasMatrix4.prototype.multRight = function(mat)
{
var m11 = (this.m11 * mat.m11 + this.m12 * mat.m21 +
this.m13 * mat.m31 + this.m14 * mat.m41);
var m12 = (this.m11 * mat.m12 + this.m12 * mat.m22 +
this.m13 * mat.m32 + this.m14 * mat.m42);
var m13 = (this.m11 * mat.m13 + this.m12 * mat.m23 +
this.m13 * mat.m33 + this.m14 * mat.m43);
var m14 = (this.m11 * mat.m14 + this.m12 * mat.m24 +
this.m13 * mat.m34 + this.m14 * mat.m44);
var m21 = (this.m21 * mat.m11 + this.m22 * mat.m21 +
this.m23 * mat.m31 + this.m24 * mat.m41);
var m22 = (this.m21 * mat.m12 + this.m22 * mat.m22 +
this.m23 * mat.m32 + this.m24 * mat.m42);
var m23 = (this.m21 * mat.m13 + this.m22 * mat.m23 +
this.m23 * mat.m33 + this.m24 * mat.m43);
var m24 = (this.m21 * mat.m14 + this.m22 * mat.m24 +
this.m23 * mat.m34 + this.m24 * mat.m44);
var m31 = (this.m31 * mat.m11 + this.m32 * mat.m21 +
this.m33 * mat.m31 + this.m34 * mat.m41);
var m32 = (this.m31 * mat.m12 + this.m32 * mat.m22 +
this.m33 * mat.m32 + this.m34 * mat.m42);
var m33 = (this.m31 * mat.m13 + this.m32 * mat.m23 +
this.m33 * mat.m33 + this.m34 * mat.m43);
var m34 = (this.m31 * mat.m14 + this.m32 * mat.m24 +
this.m33 * mat.m34 + this.m34 * mat.m44);
var m41 = (this.m41 * mat.m11 + this.m42 * mat.m21 +
this.m43 * mat.m31 + this.m44 * mat.m41);
var m42 = (this.m41 * mat.m12 + this.m42 * mat.m22 +
this.m43 * mat.m32 + this.m44 * mat.m42);
var m43 = (this.m41 * mat.m13 + this.m42 * mat.m23 +
this.m43 * mat.m33 + this.m44 * mat.m43);
var m44 = (this.m41 * mat.m14 + this.m42 * mat.m24 +
this.m43 * mat.m34 + this.m44 * mat.m44);
this.m11 = m11;
this.m12 = m12;
this.m13 = m13;
this.m14 = m14;
this.m21 = m21;
this.m22 = m22;
this.m23 = m23;
this.m24 = m24;
this.m31 = m31;
this.m32 = m32;
this.m33 = m33;
this.m34 = m34;
this.m41 = m41;
this.m42 = m42;
this.m43 = m43;
this.m44 = m44;
};
CanvasMatrix4.prototype.multLeft = function(mat)
{
var m11 = (mat.m11 * this.m11 + mat.m12 * this.m21 +
mat.m13 * this.m31 + mat.m14 * this.m41);
var m12 = (mat.m11 * this.m12 + mat.m12 * this.m22 +
mat.m13 * this.m32 + mat.m14 * this.m42);
var m13 = (mat.m11 * this.m13 + mat.m12 * this.m23 +
mat.m13 * this.m33 + mat.m14 * this.m43);
var m14 = (mat.m11 * this.m14 + mat.m12 * this.m24 +
mat.m13 * this.m34 + mat.m14 * this.m44);
var m21 = (mat.m21 * this.m11 + mat.m22 * this.m21 +
mat.m23 * this.m31 + mat.m24 * this.m41);
var m22 = (mat.m21 * this.m12 + mat.m22 * this.m22 +
mat.m23 * this.m32 + mat.m24 * this.m42);
var m23 = (mat.m21 * this.m13 + mat.m22 * this.m23 +
mat.m23 * this.m33 + mat.m24 * this.m43);
var m24 = (mat.m21 * this.m14 + mat.m22 * this.m24 +
mat.m23 * this.m34 + mat.m24 * this.m44);
var m31 = (mat.m31 * this.m11 + mat.m32 * this.m21 +
mat.m33 * this.m31 + mat.m34 * this.m41);
var m32 = (mat.m31 * this.m12 + mat.m32 * this.m22 +
mat.m33 * this.m32 + mat.m34 * this.m42);
var m33 = (mat.m31 * this.m13 + mat.m32 * this.m23 +
mat.m33 * this.m33 + mat.m34 * this.m43);
var m34 = (mat.m31 * this.m14 + mat.m32 * this.m24 +
mat.m33 * this.m34 + mat.m34 * this.m44);
var m41 = (mat.m41 * this.m11 + mat.m42 * this.m21 +
mat.m43 * this.m31 + mat.m44 * this.m41);
var m42 = (mat.m41 * this.m12 + mat.m42 * this.m22 +
mat.m43 * this.m32 + mat.m44 * this.m42);
var m43 = (mat.m41 * this.m13 + mat.m42 * this.m23 +
mat.m43 * this.m33 + mat.m44 * this.m43);
var m44 = (mat.m41 * this.m14 + mat.m42 * this.m24 +
mat.m43 * this.m34 + mat.m44 * this.m44);
this.m11 = m11;
this.m12 = m12;
this.m13 = m13;
this.m14 = m14;
this.m21 = m21;
this.m22 = m22;
this.m23 = m23;
this.m24 = m24;
this.m31 = m31;
this.m32 = m32;
this.m33 = m33;
this.m34 = m34;
this.m41 = m41;
this.m42 = m42;
this.m43 = m43;
this.m44 = m44;
};
CanvasMatrix4.prototype.ortho = function(left, right, bottom, top, near, far)
{
var tx = (left + right) / (left - right);
var ty = (top + bottom) / (bottom - top);
var tz = (far + near) / (near - far);
var matrix = new CanvasMatrix4();
matrix.m11 = 2 / (right - left);
matrix.m12 = 0;
matrix.m13 = 0;
matrix.m14 = 0;
matrix.m21 = 0;
matrix.m22 = 2 / (top - bottom);
matrix.m23 = 0;
matrix.m24 = 0;
matrix.m31 = 0;
matrix.m32 = 0;
matrix.m33 = -2 / (far - near);
matrix.m34 = 0;
matrix.m41 = tx;
matrix.m42 = ty;
matrix.m43 = tz;
matrix.m44 = 1;
this.multRight(matrix);
};
CanvasMatrix4.prototype.frustum = function(left, right, bottom, top, near, far)
{
var matrix = new CanvasMatrix4();
var A = (right + left) / (right - left);
var B = (top + bottom) / (top - bottom);
var C = -(far + near) / (far - near);
var D = -(2 * far * near) / (far - near);
matrix.m11 = (2 * near) / (right - left);
matrix.m12 = 0;
matrix.m13 = 0;
matrix.m14 = 0;
matrix.m21 = 0;
matrix.m22 = 2 * near / (top - bottom);
matrix.m23 = 0;
matrix.m24 = 0;
matrix.m31 = A;
matrix.m32 = B;
matrix.m33 = C;
matrix.m34 = -1;
matrix.m41 = 0;
matrix.m42 = 0;
matrix.m43 = D;
matrix.m44 = 0;
this.multRight(matrix);
};
CanvasMatrix4.prototype.perspective = function(fovy, aspect, zNear, zFar)
{
var top = Math.tan(fovy * Math.PI / 360) * zNear;
var bottom = -top;
var left = aspect * bottom;
var right = aspect * top;
this.frustum(left, right, bottom, top, zNear, zFar);
};
CanvasMatrix4.prototype.lookat = function(eyex, eyey, eyez, centerx, centery, centerz, upx, upy, upz)
{
var matrix = new CanvasMatrix4();
// Make rotation matrix
// Z vector
var zx = eyex - centerx;
var zy = eyey - centery;
var zz = eyez - centerz;
var mag = Math.sqrt(zx * zx + zy * zy + zz * zz);
if (mag) {
zx /= mag;
zy /= mag;
zz /= mag;
}
// Y vector
var yx = upx;
var yy = upy;
var yz = upz;
// X vector = Y cross Z
xx = yy * zz - yz * zy;
xy = -yx * zz + yz * zx;
xz = yx * zy - yy * zx;
// Recompute Y = Z cross X
yx = zy * xz - zz * xy;
yy = -zx * xz + zz * xx;
yx = zx * xy - zy * xx;
// cross product gives area of parallelogram, which is < 1.0 for
// non-perpendicular unit-length vectors; so normalize x, y here
mag = Math.sqrt(xx * xx + xy * xy + xz * xz);
if (mag) {
xx /= mag;
xy /= mag;
xz /= mag;
}
mag = Math.sqrt(yx * yx + yy * yy + yz * yz);
if (mag) {
yx /= mag;
yy /= mag;
yz /= mag;
}
matrix.m11 = xx;
matrix.m12 = xy;
matrix.m13 = xz;
matrix.m14 = 0;
matrix.m21 = yx;
matrix.m22 = yy;
matrix.m23 = yz;
matrix.m24 = 0;
matrix.m31 = zx;
matrix.m32 = zy;
matrix.m33 = zz;
matrix.m34 = 0;
matrix.m41 = 0;
matrix.m42 = 0;
matrix.m43 = 0;
matrix.m44 = 1;
matrix.translate(-eyex, -eyey, -eyez);
this.multRight(matrix);
};
// Support functions
CanvasMatrix4.prototype._determinant2x2 = function(a, b, c, d)
{
return a * d - b * c;
};
CanvasMatrix4.prototype._determinant3x3 = function(a1, a2, a3, b1, b2, b3, c1, c2, c3)
{
return a1 * this._determinant2x2(b2, b3, c2, c3) -
b1 * this._determinant2x2(a2, a3, c2, c3) +
c1 * this._determinant2x2(a2, a3, b2, b3);
};
CanvasMatrix4.prototype._determinant4x4 = function()
{
var a1 = this.m11;
var b1 = this.m12;
var c1 = this.m13;
var d1 = this.m14;
var a2 = this.m21;
var b2 = this.m22;
var c2 = this.m23;
var d2 = this.m24;
var a3 = this.m31;
var b3 = this.m32;
var c3 = this.m33;
var d3 = this.m34;
var a4 = this.m41;
var b4 = this.m42;
var c4 = this.m43;
var d4 = this.m44;
return a1 * this._determinant3x3(b2, b3, b4, c2, c3, c4, d2, d3, d4) -
b1 * this._determinant3x3(a2, a3, a4, c2, c3, c4, d2, d3, d4) +
c1 * this._determinant3x3(a2, a3, a4, b2, b3, b4, d2, d3, d4) -
d1 * this._determinant3x3(a2, a3, a4, b2, b3, b4, c2, c3, c4);
};
CanvasMatrix4.prototype._makeAdjoint = function()
{
var a1 = this.m11;
var b1 = this.m12;
var c1 = this.m13;
var d1 = this.m14;
var a2 = this.m21;
var b2 = this.m22;
var c2 = this.m23;
var d2 = this.m24;
var a3 = this.m31;
var b3 = this.m32;
var c3 = this.m33;
var d3 = this.m34;
var a4 = this.m41;
var b4 = this.m42;
var c4 = this.m43;
var d4 = this.m44;
// Row column labeling reversed since we transpose rows & columns
this.m11 = this._determinant3x3(b2, b3, b4, c2, c3, c4, d2, d3, d4);
this.m21 = - this._determinant3x3(a2, a3, a4, c2, c3, c4, d2, d3, d4);
this.m31 = this._determinant3x3(a2, a3, a4, b2, b3, b4, d2, d3, d4);
this.m41 = - this._determinant3x3(a2, a3, a4, b2, b3, b4, c2, c3, c4);
this.m12 = - this._determinant3x3(b1, b3, b4, c1, c3, c4, d1, d3, d4);
this.m22 = this._determinant3x3(a1, a3, a4, c1, c3, c4, d1, d3, d4);
this.m32 = - this._determinant3x3(a1, a3, a4, b1, b3, b4, d1, d3, d4);
this.m42 = this._determinant3x3(a1, a3, a4, b1, b3, b4, c1, c3, c4);
this.m13 = this._determinant3x3(b1, b2, b4, c1, c2, c4, d1, d2, d4);
this.m23 = - this._determinant3x3(a1, a2, a4, c1, c2, c4, d1, d2, d4);
this.m33 = this._determinant3x3(a1, a2, a4, b1, b2, b4, d1, d2, d4);
this.m43 = - this._determinant3x3(a1, a2, a4, b1, b2, b4, c1, c2, c4);
this.m14 = - this._determinant3x3(b1, b2, b3, c1, c2, c3, d1, d2, d3);
this.m24 = this._determinant3x3(a1, a2, a3, c1, c2, c3, d1, d2, d3);
this.m34 = - this._determinant3x3(a1, a2, a3, b1, b2, b3, d1, d2, d3);
this.m44 = this._determinant3x3(a1, a2, a3, b1, b2, b3, c1, c2, c3);
};</script>
<script>
rglwidgetClass = function() {
this.canvas = null;
this.userMatrix = new CanvasMatrix4();
this.types = [];
this.prMatrix = new CanvasMatrix4();
this.mvMatrix = new CanvasMatrix4();
this.vp = null;
this.prmvMatrix = null;
this.origs = null;
this.gl = null;
this.scene = null;
};
(function() {
this.multMV = function(M, v) {
return [ M.m11 * v[0] + M.m12 * v[1] + M.m13 * v[2] + M.m14 * v[3],
M.m21 * v[0] + M.m22 * v[1] + M.m23 * v[2] + M.m24 * v[3],
M.m31 * v[0] + M.m32 * v[1] + M.m33 * v[2] + M.m34 * v[3],
M.m41 * v[0] + M.m42 * v[1] + M.m43 * v[2] + M.m44 * v[3]
];
};
this.vlen = function(v) {
return Math.sqrt(this.dotprod(v, v));
};
this.dotprod = function(a, b) {
return a[0]*b[0] + a[1]*b[1] + a[2]*b[2];
};
this.xprod = function(a, b) {
return [a[1]*b[2] - a[2]*b[1],
a[2]*b[0] - a[0]*b[2],
a[0]*b[1] - a[1]*b[0]];
};
this.cbind = function(a, b) {
if (b.length < a.length)
b = this.repeatToLen(b, a.length);
else if (a.length < b.length)
a = this.repeatToLen(a, b.length);
return a.map(function(currentValue, index, array) {
return currentValue.concat(b[index]);
});
};
this.swap = function(a, i, j) {
var temp = a[i];
a[i] = a[j];
a[j] = temp;
};
this.flatten = function(a) {
return [].concat.apply([], a);
};
/* set element of 1d or 2d array as if it was flattened. Column major, zero based! */
this.setElement = function(a, i, value) {
if (Array.isArray(a[0])) {
var dim = a.length,
col = Math.floor(i/dim),
row = i % dim;
a[row][col] = value;
} else {
a[i] = value;
}
};
this.transpose = function(a) {
var newArray = [],
n = a.length,
m = a[0].length,
i;
for(i = 0; i < m; i++){
newArray.push([]);
}
for(i = 0; i < n; i++){
for(var j = 0; j < m; j++){
newArray[j].push(a[i][j]);
}
}
return newArray;
};
this.sumsq = function(x) {
var result = 0, i;
for (i=0; i < x.length; i++)
result += x[i]*x[i];
return result;
};
this.toCanvasMatrix4 = function(mat) {
if (mat instanceof CanvasMatrix4)
return mat;
var result = new CanvasMatrix4();
mat = this.flatten(this.transpose(mat));
result.load(mat);
return result;
};
this.stringToRgb = function(s) {
s = s.replace("#", "");
var bigint = parseInt(s, 16);
return [((bigint >> 16) & 255)/255,
((bigint >> 8) & 255)/255,
(bigint & 255)/255];
};
this.componentProduct = function(x, y) {
if (typeof y === "undefined") {
this.alertOnce("Bad arg to componentProduct");
}
var result = new Float32Array(3), i;
for (i = 0; i<3; i++)
result[i] = x[i]*y[i];
return result;
};
this.getPowerOfTwo = function(value) {
var pow = 1;
while(pow<value) {
pow *= 2;
}
return pow;
};
this.unique = function(arr) {
arr = [].concat(arr);
return arr.filter(function(value, index, self) {
return self.indexOf(value) === index;
});
};
this.repeatToLen = function(arr, len) {
arr = [].concat(arr);
while (arr.length < len/2)
arr = arr.concat(arr);
return arr.concat(arr.slice(0, len - arr.length));
};
this.alertOnce = function(msg) {
if (typeof this.alerted !== "undefined")
return;
this.alerted = true;
alert(msg);
};
this.f_is_lit = 1;
this.f_is_smooth = 2;
this.f_has_texture = 4;
this.f_is_indexed = 8;
this.f_depth_sort = 16;
this.f_fixed_quads = 32;
this.f_is_transparent = 64;
this.f_is_lines = 128;
this.f_sprites_3d = 256;
this.f_sprite_3d = 512;
this.f_is_subscene = 1024;
this.f_is_clipplanes = 2048;
this.f_fixed_size = 4096;
this.f_is_points = 8192;
this.f_is_twosided = 16384;
this.whichList = function(id) {
var obj = this.getObj(id),
flags = obj.flags;
if (obj.type === "light")
return "lights";
if (flags & this.f_is_subscene)
return "subscenes";
if (flags & this.f_is_clipplanes)
return "clipplanes";
if (flags & this.f_is_transparent)
return "transparent";
return "opaque";
};
this.getObj = function(id) {
if (typeof id !== "number") {
this.alertOnce("getObj id is "+typeof id);
}
return this.scene.objects[id];
};
this.getIdsByType = function(type, subscene) {
var
result = [], i, self = this;
if (typeof subscene === "undefined") {
Object.keys(this.scene.objects).forEach(
function(key) {
key = parseInt(key, 10);
if (self.getObj(key).type === type)
result.push(key);
});
} else {
ids = this.getObj(subscene).objects;
for (i=0; i < ids.length; i++) {
if (this.getObj(ids[i]).type === type) {
result.push(ids[i]);
}
}
}
return result;
};
this.getMaterial = function(id, property) {
var obj = this.getObj(id),
mat = obj.material[property];
if (typeof mat === "undefined")
mat = this.scene.material[property];
return mat;
};
this.inSubscene = function(id, subscene) {
return this.getObj(subscene).objects.indexOf(id) > -1;
};
this.addToSubscene = function(id, subscene) {
var thelist,
thesub = this.getObj(subscene),
ids = [id],
obj = this.getObj(id), i;
if (typeof obj.newIds !== "undefined") {
ids = ids.concat(obj.newIds);
}
for (i = 0; i < ids.length; i++) {
id = ids[i];
if (thesub.objects.indexOf(id) == -1) {
thelist = this.whichList(id);
thesub.objects.push(id);
thesub[thelist].push(id);
}
}
};
this.delFromSubscene = function(id, subscene) {
var thelist,
thesub = this.getObj(subscene),
obj = this.getObj(id),
ids = [id], i;
if (typeof obj.newIds !== "undefined")
ids = ids.concat(obj.newIds);
for (j=0; j<ids.length;j++) {
id = ids[j];
i = thesub.objects.indexOf(id);
if (i > -1) {
thesub.objects.splice(i, 1);
thelist = this.whichList(id);
i = thesub[thelist].indexOf(id);
thesub[thelist].splice(i, 1);
}
}
};
this.setSubsceneEntries = function(ids, subsceneid) {
var sub = this.getObj(subsceneid);
sub.objects = ids;
this.initSubscene(subsceneid);
};
this.getSubsceneEntries = function(subscene) {
return this.getObj(subscene).objects;
};
this.getChildSubscenes = function(subscene) {
return this.getObj(subscene).subscenes;
};
this.startDrawing = function() {
var value = this.drawing;
this.drawing = true;
return value;
};
this.stopDrawing = function(saved) {
this.drawing = saved;
if (!saved && this.gl && this.gl.isContextLost())
this.restartCanvas();
};
this.getVertexShader = function(id) {
var obj = this.getObj(id),
userShader = obj.userVertexShader,
flags = obj.flags,
type = obj.type,
is_lit = flags & this.f_is_lit,
has_texture = flags & this.f_has_texture,
fixed_quads = flags & this.f_fixed_quads,
sprites_3d = flags & this.f_sprites_3d,
sprite_3d = flags & this.f_sprite_3d,
nclipplanes = this.countClipplanes(),
fixed_size = flags & this.f_fixed_size,
is_points = flags & this.f_is_points,
is_twosided = flags & this.f_is_twosided,
result;
if (type === "clipplanes" || sprites_3d) return;
if (typeof userShader !== "undefined") return userShader;
result = " /* ****** "+type+" object "+id+" vertex shader ****** */\n"+
" attribute vec3 aPos;\n"+
" attribute vec4 aCol;\n"+
" uniform mat4 mvMatrix;\n"+
" uniform mat4 prMatrix;\n"+
" varying vec4 vCol;\n"+
" varying vec4 vPosition;\n";
if ((is_lit && !fixed_quads) || sprite_3d)
result = result + " attribute vec3 aNorm;\n"+
" uniform mat4 normMatrix;\n"+
" varying vec3 vNormal;\n";
if (has_texture || type === "text")
result = result + " attribute vec2 aTexcoord;\n"+
" varying vec2 vTexcoord;\n";
if (fixed_size)
result = result + " uniform vec2 textScale;\n";
if (fixed_quads)
result = result + " attribute vec2 aOfs;\n";
else if (sprite_3d)
result = result + " uniform vec3 uOrig;\n"+
" uniform float uSize;\n"+
" uniform mat4 usermat;\n";
if (is_twosided)
result = result + " attribute vec3 aPos1;\n"+
" attribute vec3 aPos2;\n"+
" varying float normz;\n";
result = result + " void main(void) {\n";
if (nclipplanes || (!fixed_quads && !sprite_3d))
result = result + " vPosition = mvMatrix * vec4(aPos, 1.);\n";
if (!fixed_quads && !sprite_3d)
result = result + " gl_Position = prMatrix * vPosition;\n";
if (is_points) {
var size = this.getMaterial(id, "size");
result = result + " gl_PointSize = "+size.toFixed(1)+";\n";
}
result = result + " vCol = aCol;\n";
if (is_lit && !fixed_quads && !sprite_3d)
result = result + " vNormal = normalize((normMatrix * vec4(aNorm, 1.)).xyz);\n";
if (has_texture || type == "text")
result = result + " vTexcoord = aTexcoord;\n";
if (fixed_size)
result = result + " vec4 pos = prMatrix * mvMatrix * vec4(aPos, 1.);\n"+
" pos = pos/pos.w;\n"+
" gl_Position = pos + vec4(aOfs*textScale, 0.,0.);\n";
if (type == "sprites" && !fixed_size)
result = result + " vec4 pos = mvMatrix * vec4(aPos, 1.);\n"+
" pos = pos/pos.w + vec4(aOfs, 0., 0.);\n"+
" gl_Position = prMatrix*pos;\n";
if (sprite_3d)
result = result + " vNormal = normalize((normMatrix * vec4(aNorm, 1.)).xyz);\n"+
" vec4 pos = mvMatrix * vec4(uOrig, 1.);\n"+
" vPosition = pos/pos.w + vec4(uSize*(vec4(aPos, 1.)*usermat).xyz,0.);\n"+
" gl_Position = prMatrix * vPosition;\n";
if (is_twosided)
result = result + " vec4 pos1 = prMatrix*(mvMatrix*vec4(aPos1, 1.));\n"+
" pos1 = pos1/pos1.w - gl_Position/gl_Position.w;\n"+
" vec4 pos2 = prMatrix*(mvMatrix*vec4(aPos2, 1.));\n"+
" pos2 = pos2/pos2.w - gl_Position/gl_Position.w;\n"+
" normz = pos1.x*pos2.y - pos1.y*pos2.x;\n";
result = result + " }\n";
// console.log(result);
return result;
};
this.getFragmentShader = function(id) {
var obj = this.getObj(id),
userShader = obj.userFragmentShader,
flags = obj.flags,
type = obj.type,
is_lit = flags & this.f_is_lit,
has_texture = flags & this.f_has_texture,
fixed_quads = flags & this.f_fixed_quads,
sprites_3d = flags & this.f_sprites_3d,
is_twosided = (flags & this.f_is_twosided) > 0,
nclipplanes = this.countClipplanes(), i,
texture_format, nlights,
result;
if (type === "clipplanes" || sprites_3d) return;
if (typeof userShader !== "undefined") return userShader;
if (has_texture)
texture_format = this.getMaterial(id, "textype");
result = "/* ****** "+type+" object "+id+" fragment shader ****** */\n"+
"#ifdef GL_ES\n"+
" precision highp float;\n"+
"#endif\n"+
" varying vec4 vCol; // carries alpha\n"+
" varying vec4 vPosition;\n";
if (has_texture || type === "text")
result = result + " varying vec2 vTexcoord;\n"+
" uniform sampler2D uSampler;\n";
if (is_lit && !fixed_quads)
result = result + " varying vec3 vNormal;\n";
for (i = 0; i < nclipplanes; i++)
result = result + " uniform vec4 vClipplane"+i+";\n";
if (is_lit) {
nlights = this.countLights();
if (nlights)
result = result + " uniform mat4 mvMatrix;\n";
else
is_lit = false;
}
if (is_lit) {
result = result + " uniform vec3 emission;\n"+
" uniform float shininess;\n";
for (i=0; i < nlights; i++) {
result = result + " uniform vec3 ambient" + i + ";\n"+
" uniform vec3 specular" + i +"; // light*material\n"+
" uniform vec3 diffuse" + i + ";\n"+
" uniform vec3 lightDir" + i + ";\n"+
" uniform bool viewpoint" + i + ";\n"+
" uniform bool finite" + i + ";\n";
}
}
if (is_twosided)
result = result + " uniform bool front;\n"+
" varying float normz;\n";
result = result + " void main(void) {\n";
for (i=0; i < nclipplanes;i++)
result = result + " if (dot(vPosition, vClipplane"+i+") < 0.0) discard;\n";
if (fixed_quads) {
result = result + " vec3 n = vec3(0., 0., 1.);\n";
} else if (is_lit) {
result = result + " vec3 n = normalize(vNormal);\n";
}
if (is_twosided) {
result = result + " if ((normz <= 0.) != front) discard;";
}
if (is_lit) {
result = result + " vec3 eye = normalize(-vPosition.xyz);\n"+
" vec3 lightdir;\n"+
" vec4 colDiff;\n"+
" vec3 halfVec;\n"+
" vec4 lighteffect = vec4(emission, 0.);\n"+
" vec3 col;\n"+
" float nDotL;\n";
if (!fixed_quads) {
result = result + " n = -faceforward(n, n, eye);\n";
}
for (i=0; i < nlights; i++) {
result = result + " colDiff = vec4(vCol.rgb * diffuse" + i + ", vCol.a);\n"+
" lightdir = lightDir" + i + ";\n"+
" if (!viewpoint" + i +")\n"+
" lightdir = (mvMatrix * vec4(lightdir, 1.)).xyz;\n"+
" if (!finite" + i + ") {\n"+
" halfVec = normalize(lightdir + eye);\n"+
" } else {\n"+
" lightdir = normalize(lightdir - vPosition.xyz);\n"+
" halfVec = normalize(lightdir + eye);\n"+
" }\n"+
" col = ambient" + i + ";\n"+
" nDotL = dot(n, lightdir);\n"+
" col = col + max(nDotL, 0.) * colDiff.rgb;\n"+
" col = col + pow(max(dot(halfVec, n), 0.), shininess) * specular" + i + ";\n"+
" lighteffect = lighteffect + vec4(col, colDiff.a);\n";
}
} else {
result = result + " vec4 colDiff = vCol;\n"+
" vec4 lighteffect = colDiff;\n";
}
if (type === "text")
result = result + " vec4 textureColor = lighteffect*texture2D(uSampler, vTexcoord);\n";
if (has_texture) {
result = result + {
rgb: " vec4 textureColor = lighteffect*vec4(texture2D(uSampler, vTexcoord).rgb, 1.);\n",
rgba: " vec4 textureColor = lighteffect*texture2D(uSampler, vTexcoord);\n",
alpha: " vec4 textureColor = texture2D(uSampler, vTexcoord);\n"+
" float luminance = dot(vec3(1.,1.,1.), textureColor.rgb)/3.;\n"+
" textureColor = vec4(lighteffect.rgb, lighteffect.a*luminance);\n",
luminance: " vec4 textureColor = vec4(lighteffect.rgb*dot(texture2D(uSampler, vTexcoord).rgb, vec3(1.,1.,1.))/3., lighteffect.a);\n",
"luminance.alpha":" vec4 textureColor = texture2D(uSampler, vTexcoord);\n"+
" float luminance = dot(vec3(1.,1.,1.),textureColor.rgb)/3.;\n"+
" textureColor = vec4(lighteffect.rgb*luminance, lighteffect.a*textureColor.a);\n"
}[texture_format]+
" gl_FragColor = textureColor;\n";
} else if (type === "text") {
result = result + " if (textureColor.a < 0.1)\n"+
" discard;\n"+
" else\n"+
" gl_FragColor = textureColor;\n";
} else
result = result + " gl_FragColor = lighteffect;\n";
result = result + " }\n";
// console.log(result);
return result;
};
this.getShader = function(shaderType, code) {
var gl = this.gl, shader;
shader = gl.createShader(shaderType);
gl.shaderSource(shader, code);
gl.compileShader(shader);
if (!gl.getShaderParameter(shader, gl.COMPILE_STATUS) && !gl.isContextLost())
alert(gl.getShaderInfoLog(shader));
return shader;
};
this.handleLoadedTexture = function(texture, textureCanvas) {
var gl = this.gl || this.initGL();
gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, true);
gl.bindTexture(gl.TEXTURE_2D, texture);
gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, textureCanvas);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR_MIPMAP_NEAREST);
gl.generateMipmap(gl.TEXTURE_2D);
gl.bindTexture(gl.TEXTURE_2D, null);
};
this.loadImageToTexture = function(uri, texture) {
var canvas = this.textureCanvas,
ctx = canvas.getContext("2d"),
image = new Image(),
self = this;
image.onload = function() {
var w = image.width,
h = image.height,
canvasX = self.getPowerOfTwo(w),
canvasY = self.getPowerOfTwo(h),
gl = self.gl || self.initGL(),
maxTexSize = gl.getParameter(gl.MAX_TEXTURE_SIZE);
if (maxTexSize > 4096) maxTexSize = 4096;
while (canvasX > 1 && canvasY > 1 && (canvasX > maxTexSize || canvasY > maxTexSize)) {
canvasX /= 2;
canvasY /= 2;
}
canvas.width = canvasX;
canvas.height = canvasY;
ctx.imageSmoothingEnabled = true;
ctx.drawImage(image, 0, 0, canvasX, canvasY);
self.handleLoadedTexture(texture, canvas);
self.drawScene();
};
image.src = uri;
};
this.drawTextToCanvas = function(text, cex, family, font) {
var canvasX, canvasY,
textY,
scaling = 20,
textColour = "white",
backgroundColour = "rgba(0,0,0,0)",
canvas = this.textureCanvas,
ctx = canvas.getContext("2d"),
i, textHeights = [], widths = [], offset = 0, offsets = [],
fontStrings = [],
getFontString = function(i) {
textHeights[i] = scaling*cex[i];
var fontString = textHeights[i] + "px",
family0 = family[i],
font0 = font[i];
if (family0 === "sans")
family0 = "sans-serif";
else if (family0 === "mono")
family0 = "monospace";
fontString = fontString + " " + family0;
if (font0 === 2 || font0 === 4)
fontString = "bold " + fontString;
if (font0 === 3 || font0 === 4)
fontString = "italic " + fontString;
return fontString;
};
cex = this.repeatToLen(cex, text.length);
family = this.repeatToLen(family, text.length);
font = this.repeatToLen(font, text.length);
canvasX = 1;
for (i = 0; i < text.length; i++) {
ctx.font = fontStrings[i] = getFontString(i);
widths[i] = ctx.measureText(text[i]).width;
offset = offsets[i] = offset + 2*textHeights[i];
canvasX = (widths[i] > canvasX) ? widths[i] : canvasX;
}
canvasX = this.getPowerOfTwo(canvasX);
canvasY = this.getPowerOfTwo(offset);
canvas.width = canvasX;
canvas.height = canvasY;
ctx.fillStyle = backgroundColour;
ctx.fillRect(0, 0, ctx.canvas.width, ctx.canvas.height);
ctx.textBaseline = "alphabetic";
for(i = 0; i < text.length; i++) {
textY = offsets[i];
ctx.font = fontStrings[i];
ctx.fillStyle = textColour;
ctx.textAlign = "left";
ctx.fillText(text[i], 0, textY);
}
return {canvasX:canvasX, canvasY:canvasY,
widths:widths, textHeights:textHeights,
offsets:offsets};
};
this.setViewport = function(id) {
var gl = this.gl || this.initGL(),
vp = this.getObj(id).par3d.viewport,
x = vp.x*this.canvas.width,
y = vp.y*this.canvas.height,
width = vp.width*this.canvas.width,
height = vp.height*this.canvas.height;
this.vp = {x:x, y:y, width:width, height:height};
gl.viewport(x, y, width, height);
gl.scissor(x, y, width, height);
gl.enable(gl.SCISSOR_TEST);
};
this.setprMatrix = function(id) {
var subscene = this.getObj(id),
embedding = subscene.embeddings.projection;
if (embedding === "replace")
this.prMatrix.makeIdentity();
else
this.setprMatrix(subscene.parent);
if (embedding === "inherit")
return;
// This is based on the Frustum::enclose code from geom.cpp
var bbox = subscene.par3d.bbox,
scale = subscene.par3d.scale,
ranges = [(bbox[1]-bbox[0])*scale[0]/2,
(bbox[3]-bbox[2])*scale[1]/2,
(bbox[5]-bbox[4])*scale[2]/2],
radius = Math.sqrt(this.sumsq(ranges))*1.1; // A bit bigger to handle labels
if (radius <= 0) radius = 1;
var observer = subscene.par3d.observer,
distance = observer[2],
FOV = subscene.par3d.FOV, ortho = FOV === 0,
t = ortho ? 1 : Math.tan(FOV*Math.PI/360),
near = distance - radius,
far = distance + radius,
hlen,
aspect = this.vp.width/this.vp.height,
z = subscene.par3d.zoom;
if (far < 0.)
far = 1.;
if (near < far/100.)
near = far/100.;
hlen = t*near;
if (ortho) {
if (aspect > 1)
this.prMatrix.ortho(-hlen*aspect*z, hlen*aspect*z,
-hlen*z, hlen*z, near, far);
else
this.prMatrix.ortho(-hlen*z, hlen*z,
-hlen*z/aspect, hlen*z/aspect,
near, far);
} else {
if (aspect > 1)
this.prMatrix.frustum(-hlen*aspect*z, hlen*aspect*z,
-hlen*z, hlen*z, near, far);
else
this.prMatrix.frustum(-hlen*z, hlen*z,
-hlen*z/aspect, hlen*z/aspect,
near, far);
}
};
this.setmvMatrix = function(id) {
var observer = this.getObj(id).par3d.observer;
this.mvMatrix.makeIdentity();
this.setmodelMatrix(id);
this.mvMatrix.translate(-observer[0], -observer[1], -observer[2]);
};
this.setmodelMatrix = function(id) {
var subscene = this.getObj(id),
embedding = subscene.embeddings.model;
if (embedding !== "inherit") {
var scale = subscene.par3d.scale,
bbox = subscene.par3d.bbox,
center = [(bbox[0]+bbox[1])/2,
(bbox[2]+bbox[3])/2,
(bbox[4]+bbox[5])/2];
this.mvMatrix.translate(-center[0], -center[1], -center[2]);
this.mvMatrix.scale(scale[0], scale[1], scale[2]);
this.mvMatrix.multRight( subscene.par3d.userMatrix );
}
if (embedding !== "replace")
this.setmodelMatrix(subscene.parent);
};
this.setnormMatrix = function(subsceneid) {
var self = this,
recurse = function(id) {
var sub = self.getObj(id),
embedding = sub.embeddings.model;
if (embedding !== "inherit") {
var scale = sub.par3d.scale;
self.normMatrix.scale(1/scale[0], 1/scale[1], 1/scale[2]);
self.normMatrix.multRight(sub.par3d.userMatrix);
}
if (embedding !== "replace")
recurse(sub.parent);
};
self.normMatrix.makeIdentity();
recurse(subsceneid);
};
this.setprmvMatrix = function() {
this.prmvMatrix = new CanvasMatrix4( this.mvMatrix );
this.prmvMatrix.multRight( this.prMatrix );
};
this.countClipplanes = function() {
return this.countObjs("clipplanes");
};
this.countLights = function() {
return this.countObjs("light");
};
this.countObjs = function(type) {
var self = this,
bound = 0;
Object.keys(this.scene.objects).forEach(
function(key) {
if (self.getObj(parseInt(key, 10)).type === type)
bound = bound + 1;
});
return bound;
};
this.initSubscene = function(id) {
var sub = this.getObj(id),
i, obj;
if (sub.type !== "subscene")
return;
sub.par3d.userMatrix = this.toCanvasMatrix4(sub.par3d.userMatrix);
sub.par3d.listeners = [].concat(sub.par3d.listeners);
sub.backgroundId = undefined;
sub.subscenes = [];
sub.clipplanes = [];
sub.transparent = [];
sub.opaque = [];
sub.lights = [];
for (i=0; i < sub.objects.length; i++) {
obj = this.getObj(sub.objects[i]);
if (typeof obj === "undefined") {
sub.objects.splice(i, 1);
i--;
} else if (obj.type === "background")
sub.backgroundId = obj.id;
else
sub[this.whichList(obj.id)].push(obj.id);
}
};
this.copyObj = function(id, reuse) {
var obj = this.getObj(id),
prev = document.getElementById(reuse);
if (prev !== null) {
prev = prev.rglinstance;
var
prevobj = prev.getObj(id),
fields = ["flags", "type",
"colors", "vertices", "centers",
"normals", "offsets",
"texts", "cex", "family", "font", "adj",
"material",
"radii",
"texcoords",
"userMatrix", "ids",
"dim",
"par3d", "userMatrix",
"viewpoint", "finite"],
i;
for (i = 0; i < fields.length; i++) {
if (typeof prevobj[fields[i]] !== "undefined")
obj[fields[i]] = prevobj[fields[i]];
}
} else
console.warn("copyObj failed");
};
this.planeUpdateTriangles = function(id, bbox) {
var perms = [[0,0,1], [1,2,2], [2,1,0]],
x, xrow, elem, A, d, nhits, i, j, k, u, v, w, intersect, which, v0, v2, vx, reverse,
face1 = [], face2 = [], normals = [],
obj = this.getObj(id),
nPlanes = obj.normals.length;
obj.bbox = bbox;
obj.vertices = [];
obj.initialized = false;
for (elem = 0; elem < nPlanes; elem++) {
// Vertex Av = normal.getRecycled(elem);
x = [];
A = obj.normals[elem];
d = obj.offsets[elem][0];
nhits = 0;
for (i=0; i<3; i++)
for (j=0; j<2; j++)
for (k=0; k<2; k++) {
u = perms[0][i];
v = perms[1][i];
w = perms[2][i];
if (A[w] !== 0.0) {
intersect = -(d + A[u]*bbox[j+2*u] + A[v]*bbox[k+2*v])/A[w];
if (bbox[2*w] < intersect && intersect < bbox[1+2*w]) {
xrow = [];
xrow[u] = bbox[j+2*u];
xrow[v] = bbox[k+2*v];
xrow[w] = intersect;
x.push(xrow);
face1[nhits] = j + 2*u;
face2[nhits] = k + 2*v;
nhits++;
}
}
}
if (nhits > 3) {
/* Re-order the intersections so the triangles work */
for (i=0; i<nhits-2; i++) {
which = 0; /* initialize to suppress warning */
for (j=i+1; j<nhits; j++) {
if (face1[i] == face1[j] || face1[i] == face2[j] ||
face2[i] == face1[j] || face2[i] == face2[j] ) {
which = j;
break;
}
}
if (which > i+1) {
this.swap(x, i+1, which);
this.swap(face1, i+1, which);
this.swap(face2, i+1, which);
}
}
}
if (nhits >= 3) {
/* Put in order so that the normal points out the FRONT of the faces */
v0 = [x[0][0] - x[1][0] , x[0][1] - x[1][1], x[0][2] - x[1][2]];
v2 = [x[2][0] - x[1][0] , x[2][1] - x[1][1], x[2][2] - x[1][2]];
/* cross-product */
vx = this.xprod(v0, v2);
reverse = this.dotprod(vx, A) > 0;
for (i=0; i<nhits-2; i++) {
obj.vertices.push(x[0]);
normals.push(A);
for (j=1; j<3; j++) {
obj.vertices.push(x[i + (reverse ? 3-j : j)]);
normals.push(A);
}
}
}
}
obj.pnormals = normals;
};
this.initObj = function(id) {
var obj = this.getObj(id),
flags = obj.flags,
type = obj.type,
is_indexed = flags & this.f_is_indexed,
is_lit = flags & this.f_is_lit,
has_texture = flags & this.f_has_texture,
fixed_quads = flags & this.f_fixed_quads,
depth_sort = flags & this.f_depth_sort,
sprites_3d = flags & this.f_sprites_3d,
sprite_3d = flags & this.f_sprite_3d,
fixed_size = flags & this.f_fixed_size,
is_twosided = (flags & this.f_is_twosided) > 0,
gl = this.gl || this.initGL(),
texinfo, drawtype, nclipplanes, f, nrows,
i,j,v,v1,v2, mat, uri, matobj, pass, pmode,
dim, nx, nz, attr;
if (typeof id !== "number") {
this.alertOnce("initObj id is "+typeof id);
}
obj.initialized = true;
if (type === "bboxdeco" || type === "subscene")
return;
if (type === "light") {
obj.ambient = new Float32Array(obj.colors[0].slice(0,3));
obj.diffuse = new Float32Array(obj.colors[1].slice(0,3));
obj.specular = new Float32Array(obj.colors[2].slice(0,3));
obj.lightDir = new Float32Array(obj.vertices[0]);
return;
}
if (type === "clipplanes") {
obj.vClipplane = this.flatten(this.cbind(obj.normals, obj.offsets));
return;
}
if (type == "background" && typeof obj.ids !== "undefined") {
obj.quad = this.flatten([].concat(obj.ids));
return;
}
if (typeof obj.vertices === "undefined")
obj.vertices = [];
v = obj.vertices;
obj.vertexCount = v.length;
if (!obj.vertexCount) return;
if (is_twosided) {
if (typeof obj.userAttributes === "undefined")
obj.userAttributes = {};
v1 = Array(v.length);
v2 = Array(v.length);
if (obj.type == "triangles" || obj.type == "quads") {
if (obj.type == "triangles")
nrow = 3;
else
nrow = 4;
for (i=0; i<Math.floor(v.length/nrow); i++)
for (j=0; j<nrow; j++) {
v1[nrow*i + j] = v[nrow*i + ((j+1) % nrow)];
v2[nrow*i + j] = v[nrow*i + ((j+2) % nrow)];
}
} else if (obj.type == "surface") {
dim = obj.dim[0];
nx = dim[0];
nz = dim[1];
for (j=0; j<nx; j++) {
for (i=0; i<nz; i++) {
if (i+1 < nz && j+1 < nx) {
v2[j + nx*i] = v[j + nx*(i+1)];
v1[j + nx*i] = v[j+1 + nx*(i+1)];
} else if (i+1 < nz) {
v2[j + nx*i] = v[j-1 + nx*i];
v1[j + nx*i] = v[j + nx*(i+1)];
} else {
v2[j + nx*i] = v[j + nx*(i-1)];
v1[j + nx*i] = v[j-1 + nx*(i-1)];
}
}
}
}
obj.userAttributes.aPos1 = v1;
obj.userAttributes.aPos2 = v2;
}
if (!sprites_3d) {
if (gl.isContextLost()) return;
obj.prog = gl.createProgram();
gl.attachShader(obj.prog, this.getShader( gl.VERTEX_SHADER,
this.getVertexShader(id) ));
gl.attachShader(obj.prog, this.getShader( gl.FRAGMENT_SHADER,
this.getFragmentShader(id) ));
// Force aPos to location 0, aCol to location 1
gl.bindAttribLocation(obj.prog, 0, "aPos");
gl.bindAttribLocation(obj.prog, 1, "aCol");
gl.linkProgram(obj.prog);
var linked = gl.getProgramParameter(obj.prog, gl.LINK_STATUS);
if (!linked) {
// An error occurred while linking
var lastError = gl.getProgramInfoLog(obj.prog);
console.warn("Error in program linking:" + lastError);
gl.deleteProgram(obj.prog);
return;
}
}
if (type === "text") {
texinfo = this.drawTextToCanvas(obj.texts,
this.flatten(obj.cex),
this.flatten(obj.family),
this.flatten(obj.family));
}
if (fixed_quads && !sprites_3d) {
obj.ofsLoc = gl.getAttribLocation(obj.prog, "aOfs");
}
if (sprite_3d) {
obj.origLoc = gl.getUniformLocation(obj.prog, "uOrig");
obj.sizeLoc = gl.getUniformLocation(obj.prog, "uSize");
obj.usermatLoc = gl.getUniformLocation(obj.prog, "usermat");
}
if (has_texture || type == "text") {
obj.texture = gl.createTexture();
obj.texLoc = gl.getAttribLocation(obj.prog, "aTexcoord");
obj.sampler = gl.getUniformLocation(obj.prog, "uSampler");
}
if (has_texture) {
mat = obj.material;
if (typeof mat.uri !== "undefined")
uri = mat.uri;
else if (typeof mat.uriElementId === "undefined") {
matobj = this.getObj(mat.uriId);
if (typeof matobj !== "undefined") {
uri = matobj.material.uri;
} else {
uri = "";
}
} else
uri = document.getElementById(mat.uriElementId).rglinstance.getObj(mat.uriId).material.uri;
this.loadImageToTexture(uri, obj.texture);
}
if (type === "text") {
this.handleLoadedTexture(obj.texture, this.textureCanvas);
}
var stride = 3, nc, cofs, nofs, radofs, oofs, tofs, vnew;
nc = obj.colorCount = obj.colors.length;
if (nc > 1) {
cofs = stride;
stride = stride + 4;
v = this.cbind(v, obj.colors);
} else {
cofs = -1;
obj.onecolor = this.flatten(obj.colors);
}
if (typeof obj.normals !== "undefined") {
nofs = stride;
stride = stride + 3;
v = this.cbind(v, typeof obj.pnormals !== "undefined" ? obj.pnormals : obj.normals);
} else
nofs = -1;
if (typeof obj.radii !== "undefined") {
radofs = stride;
stride = stride + 1;
// FIXME: always concat the radii?
if (obj.radii.length === v.length) {
v = this.cbind(v, obj.radii);
} else if (obj.radii.length === 1) {
v = v.map(function(row, i, arr) { return row.concat(obj.radii[0]);});
}
} else
radofs = -1;
if (type == "sprites" && !sprites_3d) {
tofs = stride;
stride += 2;
oofs = stride;
stride += 2;
vnew = new Array(4*v.length);
var rescale = fixed_size ? 72 : 1,
size = obj.radii, s = rescale*size[0]/2;
for (i=0; i < v.length; i++) {
if (size.length > 1)
s = rescale*size[i]/2;
vnew[4*i] = v[i].concat([0,0,-s,-s]);
vnew[4*i+1]= v[i].concat([1,0, s,-s]);
vnew[4*i+2]= v[i].concat([1,1, s, s]);
vnew[4*i+3]= v[i].concat([0,1,-s, s]);
}
v = vnew;
obj.vertexCount = v.length;
} else if (type === "text") {
tofs = stride;
stride += 2;
oofs = stride;
stride += 2;
vnew = new Array(4*v.length);
for (i=0; i < v.length; i++) {
vnew[4*i] = v[i].concat([0,-0.5]).concat(obj.adj[0]);
vnew[4*i+1]= v[i].concat([1,-0.5]).concat(obj.adj[0]);
vnew[4*i+2]= v[i].concat([1, 1.5]).concat(obj.adj[0]);
vnew[4*i+3]= v[i].concat([0, 1.5]).concat(obj.adj[0]);
for (j=0; j < 4; j++) {
v1 = vnew[4*i+j];
v1[tofs+2] = 2*(v1[tofs]-v1[tofs+2])*texinfo.widths[i];
v1[tofs+3] = 2*(v1[tofs+1]-v1[tofs+3])*texinfo.textHeights[i];
v1[tofs] *= texinfo.widths[i]/texinfo.canvasX;
v1[tofs+1] = 1.0-(texinfo.offsets[i] -
v1[tofs+1]*texinfo.textHeights[i])/texinfo.canvasY;
vnew[4*i+j] = v1;
}
}
v = vnew;
obj.vertexCount = v.length;
} else if (typeof obj.texcoords !== "undefined") {
tofs = stride;
stride += 2;
oofs = -1;
v = this.cbind(v, obj.texcoords);
} else {
tofs = -1;
oofs = -1;
}
if (typeof obj.userAttributes !== "undefined") {
obj.userAttribOffsets = {};
obj.userAttribLocations = {};
obj.userAttribSizes = {};
for (attr in obj.userAttributes) {
obj.userAttribLocations[attr] = gl.getAttribLocation(obj.prog, attr);
if (obj.userAttribLocations[attr] >= 0) { // Attribute may not have been used
obj.userAttribOffsets[attr] = stride;
v = this.cbind(v, obj.userAttributes[attr]);
stride = v[0].length;
obj.userAttribSizes[attr] = stride - obj.userAttribOffsets[attr];
}
}
}
if (typeof obj.userUniforms !== "undefined") {
obj.userUniformLocations = {};
for (attr in obj.userUniforms)
obj.userUniformLocations[attr] = gl.getUniformLocation(obj.prog, attr);
}
if (stride !== v[0].length) {
this.alertOnce("problem in stride calculation");
}
obj.vOffsets = {vofs:0, cofs:cofs, nofs:nofs, radofs:radofs, oofs:oofs, tofs:tofs, stride:stride};
obj.values = new Float32Array(this.flatten(v));
if (sprites_3d) {
obj.userMatrix = new CanvasMatrix4(obj.userMatrix);
obj.objects = this.flatten([].concat(obj.ids));
is_lit = false;
}
if (is_lit && !fixed_quads) {
obj.normLoc = gl.getAttribLocation(obj.prog, "aNorm");
}
nclipplanes = this.countClipplanes();
if (nclipplanes && !sprites_3d) {
obj.clipLoc = [];
for (i=0; i < nclipplanes; i++)
obj.clipLoc[i] = gl.getUniformLocation(obj.prog,"vClipplane" + i);
}
if (is_lit) {
obj.emissionLoc = gl.getUniformLocation(obj.prog, "emission");
obj.emission = new Float32Array(this.stringToRgb(this.getMaterial(id, "emission")));
obj.shininessLoc = gl.getUniformLocation(obj.prog, "shininess");
obj.shininess = this.getMaterial(id, "shininess");
obj.nlights = this.countLights();
obj.ambientLoc = [];
obj.ambient = new Float32Array(this.stringToRgb(this.getMaterial(id, "ambient")));
obj.specularLoc = [];
obj.specular = new Float32Array(this.stringToRgb(this.getMaterial(id, "specular")));
obj.diffuseLoc = [];
obj.lightDirLoc = [];
obj.viewpointLoc = [];
obj.finiteLoc = [];
for (i=0; i < obj.nlights; i++) {
obj.ambientLoc[i] = gl.getUniformLocation(obj.prog, "ambient" + i);
obj.specularLoc[i] = gl.getUniformLocation(obj.prog, "specular" + i);
obj.diffuseLoc[i] = gl.getUniformLocation(obj.prog, "diffuse" + i);
obj.lightDirLoc[i] = gl.getUniformLocation(obj.prog, "lightDir" + i);
obj.viewpointLoc[i] = gl.getUniformLocation(obj.prog, "viewpoint" + i);
obj.finiteLoc[i] = gl.getUniformLocation(obj.prog, "finite" + i);
}
}
if (is_indexed) {
obj.f = Array(2);
for (pass = 0; pass < is_twosided + 1; pass++) {
if (type === "triangles" || type === "quads" || type === "surface")
pmode = this.getMaterial(id, (pass === 0) ? "front" : "back");
else pmode = "filled";
if (pmode === "culled")
continue;
if (pmode === "points") {
nrows = obj.vertexCount;
f = Array(nrows);
for (i=0; i < nrows; i++)
f[i] = i;
} else if ((type === "quads" || type === "text" ||
type === "sprites") && !sprites_3d) {
nrows = Math.floor(obj.vertexCount/4);
if (pmode === "filled") {
f = Array(6*nrows);
for (i=0; i < nrows; i++) {
f[6*i] = 4*i;
f[6*i+1] = 4*i + 1;
f[6*i+2] = 4*i + 2;
f[6*i+3] = 4*i;
f[6*i+4] = 4*i + 2;
f[6*i+5] = 4*i + 3;
}
} else {
f = Array(8*nrows);
for (i=0; i < nrows; i++) {
f[8*i] = 4*i;
f[8*i+1] = 4*i + 1;
f[8*i+2] = 4*i + 1;
f[8*i+3] = 4*i + 2;
f[8*i+4] = 4*i + 2;
f[8*i+5] = 4*i + 3;
f[8*i+6] = 4*i + 3;
f[8*i+7] = 4*i;
}
}
} else if (type === "triangles") {
nrows = Math.floor(obj.vertexCount/3);
if (pmode === "filled") {
f = Array(3*nrows);
for (i=0; i < f.length; i++) {
f[i] = i;
}
} else if (pmode === "lines") {
f = Array(6*nrows);
for (i=0; i < nrows; i++) {
f[6*i] = 3*i;
f[6*i + 1] = 3*i + 1;
f[6*i + 2] = 3*i + 1;
f[6*i + 3] = 3*i + 2;
f[6*i + 4] = 3*i + 2;
f[6*i + 5] = 3*i;
}
}
} else if (type === "spheres") {
nrows = obj.vertexCount;
f = Array(nrows);
for (i=0; i < f.length; i++) {
f[i] = i;
}
} else if (type === "surface") {
dim = obj.dim[0];
nx = dim[0];
nz = dim[1];
if (pmode === "filled") {
f = [];
for (j=0; j<nx-1; j++) {
for (i=0; i<nz-1; i++) {
f.push(j + nx*i,
j + nx*(i+1),
j + 1 + nx*(i+1),
j + nx*i,
j + 1 + nx*(i+1),
j + 1 + nx*i);
}
}
} else if (pmode === "lines") {
f = [];
for (j=0; j<nx; j++) {
for (i=0; i<nz; i++) {
if (i+1 < nz)
f.push(j + nx*i,
j + nx*(i+1));
if (j+1 < nx)
f.push(j + nx*i,
j+1 + nx*i);
}
}
}
}
obj.f[pass] = new Uint16Array(f);
if (depth_sort) {
drawtype = "DYNAMIC_DRAW";
} else {
drawtype = "STATIC_DRAW";
}
}
}
if (type !== "spheres" && !sprites_3d) {
obj.buf = gl.createBuffer();
gl.bindBuffer(gl.ARRAY_BUFFER, obj.buf);
gl.bufferData(gl.ARRAY_BUFFER, obj.values, gl.STATIC_DRAW); //
}
if (is_indexed && type !== "spheres" && !sprites_3d) {
obj.ibuf = Array(is_twosided + 1);
obj.ibuf[0] = gl.createBuffer();
gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, obj.ibuf[0]);
gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, obj.f[0], gl[drawtype]);
if (is_twosided) {
obj.ibuf[1] = gl.createBuffer();
gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, obj.ibuf[1]);
gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, obj.f[1], gl[drawtype]);
}
}
if (!sprites_3d) {
obj.mvMatLoc = gl.getUniformLocation(obj.prog, "mvMatrix");
obj.prMatLoc = gl.getUniformLocation(obj.prog, "prMatrix");
}
if (fixed_size) {
obj.textScaleLoc = gl.getUniformLocation(obj.prog, "textScale");
}
if (is_lit && !sprites_3d) {
obj.normMatLoc = gl.getUniformLocation(obj.prog, "normMatrix");
}
if (is_twosided) {
obj.frontLoc = gl.getUniformLocation(obj.prog, "front");
}
};
this.setDepthTest = function(id) {
var gl = this.gl || this.initGL(),
tests = {never: gl.NEVER,
less: gl.LESS,
equal: gl.EQUAL,
lequal:gl.LEQUAL,
greater: gl.GREATER,
notequal: gl.NOTEQUAL,
gequal: gl.GEQUAL,
always: gl.ALWAYS},
test = tests[this.getMaterial(id, "depth_test")];
gl.depthFunc(test);
};
this.mode4type = {points : "POINTS",
linestrip : "LINE_STRIP",
abclines : "LINES",
lines : "LINES",
sprites : "TRIANGLES",
planes : "TRIANGLES",
text : "TRIANGLES",
quads : "TRIANGLES",
surface : "TRIANGLES",
triangles : "TRIANGLES"};
this.drawObj = function(id, subsceneid) {
var obj = this.getObj(id),
subscene = this.getObj(subsceneid),
flags = obj.flags,
type = obj.type,
is_indexed = flags & this.f_is_indexed,
is_lit = flags & this.f_is_lit,
has_texture = flags & this.f_has_texture,
fixed_quads = flags & this.f_fixed_quads,
depth_sort = flags & this.f_depth_sort,
sprites_3d = flags & this.f_sprites_3d,
sprite_3d = flags & this.f_sprite_3d,
is_lines = flags & this.f_is_lines,
is_points = flags & this.f_is_points,
fixed_size = flags & this.f_fixed_size,
is_twosided = (flags & this.f_is_twosided) > 0,
gl = this.gl || this.initGL(),
mat,
sphereMV, baseofs, ofs, sscale, i, count, light,
faces, pass, mode, pmode, attr,
depthsort = function(i,j) { return depths[j] - depths[i]; };
if (typeof id !== "number") {
this.alertOnce("drawObj id is "+typeof id);
}
if (type === "planes") {
if (obj.bbox !== subscene.par3d.bbox || !obj.initialized) {
this.planeUpdateTriangles(id, subscene.par3d.bbox);
}
}
if (!obj.initialized)
this.initObj(id);
if (type === "clipplanes") {
count = obj.offsets.length;
var IMVClip = [];
for (i=0; i < count; i++) {
IMVClip[i] = this.multMV(this.invMatrix, obj.vClipplane.slice(4*i, 4*(i+1)));
}
obj.IMVClip = IMVClip;
return;
}
if (type === "light" || type === "bboxdeco" || !obj.vertexCount)
return;
this.setDepthTest(id);
if (sprites_3d) {
var norigs = obj.vertices.length,
savenorm = new CanvasMatrix4(this.normMatrix);
this.origs = obj.vertices;
this.usermat = new Float32Array(obj.userMatrix.getAsArray());
this.radii = obj.radii;
this.normMatrix = subscene.spriteNormmat;
for (this.iOrig=0; this.iOrig < norigs; this.iOrig++) {
for (i=0; i < obj.objects.length; i++) {
this.drawObj(obj.objects[i], subsceneid);
}
}
this.normMatrix = savenorm;
return;
} else {
gl.useProgram(obj.prog);
}
if (sprite_3d) {
gl.uniform3fv(obj.origLoc, new Float32Array(this.origs[this.iOrig]));
if (this.radii.length > 1) {
gl.uniform1f(obj.sizeLoc, this.radii[this.iOrig][0]);
} else {
gl.uniform1f(obj.sizeLoc, this.radii[0][0]);
}
gl.uniformMatrix4fv(obj.usermatLoc, false, this.usermat);
}
if (type === "spheres") {
gl.bindBuffer(gl.ARRAY_BUFFER, this.sphere.buf);
} else {
gl.bindBuffer(gl.ARRAY_BUFFER, obj.buf);
}
gl.uniformMatrix4fv( obj.prMatLoc, false, new Float32Array(this.prMatrix.getAsArray()) );
gl.uniformMatrix4fv( obj.mvMatLoc, false, new Float32Array(this.mvMatrix.getAsArray()) );
var clipcheck = 0,
clipplaneids = subscene.clipplanes,
clip, j;
for (i=0; i < clipplaneids.length; i++) {
clip = this.getObj(clipplaneids[i]);
for (j=0; j < clip.offsets.length; j++) {
gl.uniform4fv(obj.clipLoc[clipcheck + j], clip.IMVClip[j]);
}
clipcheck += clip.offsets.length;
}
if (typeof obj.clipLoc !== "undefined")
for (i=clipcheck; i < obj.clipLoc.length; i++)
gl.uniform4f(obj.clipLoc[i], 0,0,0,0);
if (is_lit) {
gl.uniformMatrix4fv( obj.normMatLoc, false, new Float32Array(this.normMatrix.getAsArray()) );
gl.uniform3fv( obj.emissionLoc, obj.emission);
gl.uniform1f( obj.shininessLoc, obj.shininess);
for (i=0; i < subscene.lights.length; i++) {
light = this.getObj(subscene.lights[i]);
gl.uniform3fv( obj.ambientLoc[i], this.componentProduct(light.ambient, obj.ambient));
gl.uniform3fv( obj.specularLoc[i], this.componentProduct(light.specular, obj.specular));
gl.uniform3fv( obj.diffuseLoc[i], light.diffuse);
gl.uniform3fv( obj.lightDirLoc[i], light.lightDir);
gl.uniform1i( obj.viewpointLoc[i], light.viewpoint);
gl.uniform1i( obj.finiteLoc[i], light.finite);
}
for (i=subscene.lights.length; i < obj.nlights; i++) {
gl.uniform3f( obj.ambientLoc[i], 0,0,0);
gl.uniform3f( obj.specularLoc[i], 0,0,0);
gl.uniform3f( obj.diffuseLoc[i], 0,0,0);
}
}
if (fixed_size) {
gl.uniform2f( obj.textScaleLoc, 0.75/this.vp.width, 0.75/this.vp.height);
}
gl.enableVertexAttribArray( this.posLoc );
var nc = obj.colorCount;
count = obj.vertexCount;
if (type === "spheres") {
subscene = this.getObj(subsceneid);
var scale = subscene.par3d.scale,
scount = count;
gl.vertexAttribPointer(this.posLoc, 3, gl.FLOAT, false, 4*this.sphere.vOffsets.stride, 0);
gl.enableVertexAttribArray(obj.normLoc );
gl.vertexAttribPointer(obj.normLoc, 3, gl.FLOAT, false, 4*this.sphere.vOffsets.stride, 0);
gl.disableVertexAttribArray( this.colLoc );
var sphereNorm = new CanvasMatrix4();
sphereNorm.scale(scale[0], scale[1], scale[2]);
sphereNorm.multRight(this.normMatrix);
gl.uniformMatrix4fv( obj.normMatLoc, false, new Float32Array(sphereNorm.getAsArray()) );
if (nc == 1) {
gl.vertexAttrib4fv( this.colLoc, new Float32Array(obj.onecolor));
}
if (has_texture) {
gl.enableVertexAttribArray( obj.texLoc );
gl.vertexAttribPointer(obj.texLoc, 2, gl.FLOAT, false, 4*this.sphere.vOffsets.stride,
4*this.sphere.vOffsets.tofs);
gl.activeTexture(gl.TEXTURE0);
gl.bindTexture(gl.TEXTURE_2D, obj.texture);
gl.uniform1i( obj.sampler, 0);
}
for (i = 0; i < scount; i++) {
sphereMV = new CanvasMatrix4();
if (depth_sort) {
baseofs = faces[i]*obj.vOffsets.stride;
} else {
baseofs = i*obj.vOffsets.stride;
}
ofs = baseofs + obj.vOffsets.radofs;
sscale = obj.values[ofs];
sphereMV.scale(sscale/scale[0], sscale/scale[1], sscale/scale[2]);
sphereMV.translate(obj.values[baseofs],
obj.values[baseofs+1],
obj.values[baseofs+2]);
sphereMV.multRight(this.mvMatrix);
gl.uniformMatrix4fv( obj.mvMatLoc, false, new Float32Array(sphereMV.getAsArray()) );
if (nc > 1) {
ofs = baseofs + obj.vOffsets.cofs;
gl.vertexAttrib4f( this.colLoc, obj.values[ofs],
obj.values[ofs+1],
obj.values[ofs+2],
obj.values[ofs+3] );
}
gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.sphere.ibuf);
gl.drawElements(gl.TRIANGLES, this.sphere.sphereCount, gl.UNSIGNED_SHORT, 0);
}
return;
} else {
if (obj.colorCount === 1) {
gl.disableVertexAttribArray( this.colLoc );
gl.vertexAttrib4fv( this.colLoc, new Float32Array(obj.onecolor));
} else {
gl.enableVertexAttribArray( this.colLoc );
gl.vertexAttribPointer(this.colLoc, 4, gl.FLOAT, false, 4*obj.vOffsets.stride, 4*obj.vOffsets.cofs);
}
}
if (is_lit && obj.vOffsets.nofs > 0) {
gl.enableVertexAttribArray( obj.normLoc );
gl.vertexAttribPointer(obj.normLoc, 3, gl.FLOAT, false, 4*obj.vOffsets.stride, 4*obj.vOffsets.nofs);
}
if (has_texture || type === "text") {
gl.enableVertexAttribArray( obj.texLoc );
gl.vertexAttribPointer(obj.texLoc, 2, gl.FLOAT, false, 4*obj.vOffsets.stride, 4*obj.vOffsets.tofs);
gl.activeTexture(gl.TEXTURE0);
gl.bindTexture(gl.TEXTURE_2D, obj.texture);
gl.uniform1i( obj.sampler, 0);
}
if (fixed_quads) {
gl.enableVertexAttribArray( obj.ofsLoc );
gl.vertexAttribPointer(obj.ofsLoc, 2, gl.FLOAT, false, 4*obj.vOffsets.stride, 4*obj.vOffsets.oofs);
}
if (typeof obj.userAttributes !== "undefined") {
for (attr in obj.userAttribSizes) { // Not all attributes may have been used
gl.enableVertexAttribArray( obj.userAttribLocations[attr] );
gl.vertexAttribPointer( obj.userAttribLocations[attr], obj.userAttribSizes[attr],
gl.FLOAT, false, 4*obj.vOffsets.stride, 4*obj.userAttribOffsets[attr]);
}
}
if (typeof obj.userUniforms !== "undefined") {
for (attr in obj.userUniformLocations) {
var loc = obj.userUniformLocations[attr];
if (loc !== null) {
var uniform = obj.userUniforms[attr];
if (typeof uniform.length === "undefined")
gl.uniform1f(loc, uniform);
else if (typeof uniform[0].length === "undefined") {
uniform = new Float32Array(uniform);
switch(uniform.length) {
case 2: gl.uniform2fv(loc, uniform); break;
case 3: gl.uniform3fv(loc, uniform); break;
case 4: gl.uniform4fv(loc, uniform); break;
default: console.warn("bad uniform length");
}
} else if (uniform.length == 4 && uniform[0].length == 4)
gl.uniformMatrix4fv(loc, false, new Float32Array(uniform.getAsArray()));
else
console.warn("unsupported uniform matrix");
}
}
}
for (pass = 0; pass < is_twosided + 1; pass++) {
if (type === "triangles" || type === "quads" || type === "surface")
pmode = this.getMaterial(id, (pass === 0) ? "front" : "back");
else pmode = "filled";
if (pmode === "culled")
continue;
mode = this.mode4type[type];
if (depth_sort && pmode == "filled") {// Don't try depthsorting on wireframe or points
var nfaces = obj.centers.length,
z, w, frowsize;
frowsize = Math.floor(obj.f[pass].length/nfaces);
var depths = new Float32Array(nfaces);
faces = new Array(nfaces);
for(i=0; i<nfaces; i++) {
z = this.prmvMatrix.m13*obj.centers[3*i] +
this.prmvMatrix.m23*obj.centers[3*i+1] +
this.prmvMatrix.m33*obj.centers[3*i+2] +
this.prmvMatrix.m43;
w = this.prmvMatrix.m14*obj.centers[3*i] +
this.prmvMatrix.m24*obj.centers[3*i+1] +
this.prmvMatrix.m34*obj.centers[3*i+2] +
this.prmvMatrix.m44;
depths[i] = z/w;
faces[i] = i;
}
faces.sort(depthsort);
if (type !== "spheres") {
var f = new Uint16Array(obj.f[pass].length);
for (i=0; i<nfaces; i++) {
for (j=0; j<frowsize; j++) {
f[frowsize*i + j] = obj.f[pass][frowsize*faces[i] + j];
}
}
gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, obj.ibuf[pass]);
gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, f, gl.DYNAMIC_DRAW);
}
}
if (is_twosided)
gl.uniform1i(obj.frontLoc, pass !== 0);
if (is_indexed && type !== "spheres") {
gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, obj.ibuf[pass]);
} else if (type === "spheres") {
// FIX ME!
}
if (type === "sprites" || type === "text" || type === "quads") {
count = count * 6/4;
} else if (type === "surface") {
count = obj.f[pass].length;
}
if (is_indexed) {
count = obj.f[pass].length;
if (pmode === "lines") {
mode = "LINES";
is_lines = true;
} else if (pmode === "points") {
mode = "POINTS";
}
}
if (is_lines) {
gl.lineWidth( this.getMaterial(id, "lwd") );
}
gl.vertexAttribPointer(this.posLoc, 3, gl.FLOAT, false, 4*obj.vOffsets.stride, 4*obj.vOffsets.vofs);
if (is_indexed) {
gl.drawElements(gl[mode], count, gl.UNSIGNED_SHORT, 0);
} else {
gl.drawArrays(gl[mode], 0, count);
}
}
};
this.drawBackground = function(id, subsceneid) {
var gl = this.gl || this.initGL(),
obj = this.getObj(id),
bg, i;
if (!obj.initialized)
this.initObj(id);
if (obj.colors.length) {
bg = obj.colors[0];
gl.clearColor(bg[0], bg[1], bg[2], bg[3]);
gl.depthMask(true);
gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);
}
if (typeof obj.quad !== "undefined") {
this.prMatrix.makeIdentity();
this.mvMatrix.makeIdentity();
gl.disable(gl.BLEND);
gl.disable(gl.DEPTH_TEST);
gl.depthMask(false);
for (i=0; i < obj.quad.length; i++)
this.drawObj(obj.quad[i], subsceneid);
}
};
this.drawSubscene = function(subsceneid) {
var gl = this.gl || this.initGL(),
obj = this.getObj(subsceneid),
objects = this.scene.objects,
subids = obj.objects,
subscene_has_faces = false,
subscene_needs_sorting = false,
flags, i;
if (obj.par3d.skipRedraw)
return;
for (i=0; i < subids.length; i++) {
flags = objects[subids[i]].flags;
if (typeof flags !== "undefined") {
subscene_has_faces |= (flags & this.f_is_lit)
& !(flags & this.f_fixed_quads);
subscene_needs_sorting |= (flags & this.f_depth_sort);
}
}
this.setViewport(subsceneid);
if (typeof obj.backgroundId !== "undefined")
this.drawBackground(obj.backgroundId, subsceneid);
if (subids.length) {
this.setprMatrix(subsceneid);
this.setmvMatrix(subsceneid);
if (subscene_has_faces) {
this.setnormMatrix(subsceneid);
if ((obj.flags & this.f_sprites_3d) &&
typeof obj.spriteNormmat === "undefined") {
obj.spriteNormmat = new CanvasMatrix4(this.normMatrix);
}
}
if (subscene_needs_sorting)
this.setprmvMatrix();
gl.enable(gl.DEPTH_TEST);
gl.depthMask(true);
gl.disable(gl.BLEND);
var clipids = obj.clipplanes;
if (typeof clipids === "undefined") {
console.warn("bad clipids");
}
if (clipids.length > 0) {
this.invMatrix = new CanvasMatrix4(this.mvMatrix);
this.invMatrix.invert();
for (i = 0; i < clipids.length; i++)
this.drawObj(clipids[i], subsceneid);
}
subids = obj.opaque;
if (subids.length > 0) {
for (i = 0; i < subids.length; i++) {
this.drawObj(subids[i], subsceneid);
}
}
subids = obj.transparent;
if (subids.length > 0) {
gl.depthMask(false);
gl.blendFuncSeparate(gl.SRC_ALPHA, gl.ONE_MINUS_SRC_ALPHA,
gl.ONE, gl.ONE);
gl.enable(gl.BLEND);
for (i = 0; i < subids.length; i++) {
this.drawObj(subids[i], subsceneid);
}
}
subids = obj.subscenes;
for (i = 0; i < subids.length; i++) {
this.drawSubscene(subids[i]);
}
}
};
this.relMouseCoords = function(event) {
var totalOffsetX = 0,
totalOffsetY = 0,
currentElement = this.canvas;
do {
totalOffsetX += currentElement.offsetLeft;
totalOffsetY += currentElement.offsetTop;
currentElement = currentElement.offsetParent;
}
while(currentElement);
var canvasX = event.pageX - totalOffsetX,
canvasY = event.pageY - totalOffsetY;
return {x:canvasX, y:canvasY};
};
this.setMouseHandlers = function() {
var self = this, activeSubscene, handler,
handlers = {}, drag = 0;
handlers.rotBase = 0;
this.screenToVector = function(x, y) {
var viewport = this.getObj(activeSubscene).par3d.viewport,
width = viewport.width*this.canvas.width,
height = viewport.height*this.canvas.height,
radius = Math.max(width, height)/2.0,
cx = width/2.0,
cy = height/2.0,
px = (x-cx)/radius,
py = (y-cy)/radius,
plen = Math.sqrt(px*px+py*py);
if (plen > 1.e-6) {
px = px/plen;
py = py/plen;
}
var angle = (Math.SQRT2 - plen)/Math.SQRT2*Math.PI/2,
z = Math.sin(angle),
zlen = Math.sqrt(1.0 - z*z);
px = px * zlen;
py = py * zlen;
return [px, py, z];
};
handlers.trackballdown = function(x,y) {
var activeSub = this.getObj(activeSubscene),
activeModel = this.getObj(this.useid(activeSub.id, "model")),
i, l = activeModel.par3d.listeners;
handlers.rotBase = this.screenToVector(x, y);
this.saveMat = [];
for (i = 0; i < l.length; i++) {
activeSub = this.getObj(l[i]);
activeSub.saveMat = new CanvasMatrix4(activeSub.par3d.userMatrix);
}
};
handlers.trackballmove = function(x,y) {
var rotCurrent = this.screenToVector(x,y),
rotBase = handlers.rotBase,
dot = rotBase[0]*rotCurrent[0] +
rotBase[1]*rotCurrent[1] +
rotBase[2]*rotCurrent[2],
angle = Math.acos( dot/this.vlen(rotBase)/this.vlen(rotCurrent) )*180.0/Math.PI,
axis = this.xprod(rotBase, rotCurrent),
objects = this.scene.objects,
activeSub = this.getObj(activeSubscene),
activeModel = this.getObj(this.useid(activeSub.id, "model")),
l = activeModel.par3d.listeners,
i;
for (i = 0; i < l.length; i++) {
activeSub = this.getObj(l[i]);
activeSub.par3d.userMatrix.load(objects[l[i]].saveMat);
activeSub.par3d.userMatrix.rotate(angle, axis[0], axis[1], axis[2]);
}
this.drawScene();
};
handlers.trackballend = 0;
handlers.axisdown = function(x,y) {
handlers.rotBase = this.screenToVector(x, this.canvas.height/2);
var activeSub = this.getObj(activeSubscene),
activeModel = this.getObj(this.useid(activeSub.id, "model")),
i, l = activeModel.par3d.listeners;
for (i = 0; i < l.length; i++) {
activeSub = this.getObj(l[i]);
activeSub.saveMat = new CanvasMatrix4(activeSub.par3d.userMatrix);
}
};
handlers.axismove = function(x,y) {
var rotCurrent = this.screenToVector(x, this.canvas.height/2),
rotBase = handlers.rotBase,
angle = (rotCurrent[0] - rotBase[0])*180/Math.PI,
rotMat = new CanvasMatrix4();
rotMat.rotate(angle, handlers.axis[0], handlers.axis[1], handlers.axis[2]);
var activeSub = this.getObj(activeSubscene),
activeModel = this.getObj(this.useid(activeSub.id, "model")),
i, l = activeModel.par3d.listeners;
for (i = 0; i < l.length; i++) {
activeSub = this.getObj(l[i]);
activeSub.par3d.userMatrix.load(activeSub.saveMat);
activeSub.par3d.userMatrix.multLeft(rotMat);
}
this.drawScene();
};
handlers.axisend = 0;
handlers.y0zoom = 0;
handlers.zoom0 = 0;
handlers.zoomdown = function(x, y) {
var activeSub = this.getObj(activeSubscene),
activeProjection = this.getObj(this.useid(activeSub.id, "projection")),
i, l = activeProjection.par3d.listeners;
handlers.y0zoom = y;
for (i = 0; i < l.length; i++) {
activeSub = this.getObj(l[i]);
activeSub.zoom0 = Math.log(activeSub.par3d.zoom);
}
};
handlers.zoommove = function(x, y) {
var activeSub = this.getObj(activeSubscene),
activeProjection = this.getObj(this.useid(activeSub.id, "projection")),
i, l = activeProjection.par3d.listeners;
for (i = 0; i < l.length; i++) {
activeSub = this.getObj(l[i]);
activeSub.par3d.zoom = Math.exp(activeSub.zoom0 + (y-handlers.y0zoom)/this.canvas.height);
}
this.drawScene();
};
handlers.zoomend = 0;
handlers.y0fov = 0;
handlers.fovdown = function(x, y) {
handlers.y0fov = y;
var activeSub = this.getObj(activeSubscene),
activeProjection = this.getObj(this.useid(activeSub.id, "projection")),
i, l = activeProjection.par3d.listeners;
for (i = 0; i < l.length; i++) {
activeSub = this.getObj(l[i]);
activeSub.fov0 = activeSub.par3d.FOV;
}
};
handlers.fovmove = function(x, y) {
var activeSub = this.getObj(activeSubscene),
activeProjection = this.getObj(this.useid(activeSub.id, "projection")),
i, l = activeProjection.par3d.listeners;
for (i = 0; i < l.length; i++) {
activeSub = this.getObj(l[i]);
activeSub.par3d.FOV = Math.max(1, Math.min(179, activeSub.fov0 +
180*(y-handlers.y0fov)/this.canvas.height));
}
this.drawScene();
};
handlers.fovend = 0;
this.canvas.onmousedown = function ( ev ){
if (!ev.which) // Use w3c defns in preference to MS
switch (ev.button) {
case 0: ev.which = 1; break;
case 1:
case 4: ev.which = 2; break;
case 2: ev.which = 3;
}
drag = ["left", "middle", "right"][ev.which-1];
var coords = self.relMouseCoords(ev);
coords.y = self.canvas.height-coords.y;
activeSubscene = self.whichSubscene(coords);
var sub = self.getObj(activeSubscene), f;
handler = sub.par3d.mouseMode[drag];
switch (handler) {
case "xAxis":
handler = "axis";
handlers.axis = [1.0, 0.0, 0.0];
break;
case "yAxis":
handler = "axis";
handlers.axis = [0.0, 1.0, 0.0];
break;
case "zAxis":
handler = "axis";
handlers.axis = [0.0, 0.0, 1.0];
break;
}
f = handlers[handler + "down"];
if (f) {
coords = self.translateCoords(activeSubscene, coords);
f.call(self, coords.x, coords.y);
ev.preventDefault();
}
};
this.canvas.onmouseup = function ( ev ){
if ( drag === 0 ) return;
var f = handlers[handler + "up"];
if (f)
f();
drag = 0;
};
this.canvas.onmouseout = this.canvas.onmouseup;
this.canvas.onmousemove = function ( ev ) {
if ( drag === 0 ) return;
var f = handlers[handler + "move"];
if (f) {
var coords = self.relMouseCoords(ev);
coords.y = self.canvas.height - coords.y;
coords = self.translateCoords(activeSubscene, coords);
f.call(self, coords.x, coords.y);
}
};
handlers.wheelHandler = function(ev) {
var del = 1.02, i;
if (ev.shiftKey) del = 1.002;
var ds = ((ev.detail || ev.wheelDelta) > 0) ? del : (1 / del);
if (typeof activeSubscene === "undefined")
activeSubscene = self.scene.rootSubscene;
var activeSub = self.getObj(activeSubscene),
activeProjection = self.getObj(self.useid(activeSub.id, "projection")),
l = activeProjection.par3d.listeners;
for (i = 0; i < l.length; i++) {
activeSub = self.getObj(l[i]);
activeSub.par3d.zoom *= ds;
}
self.drawScene();
ev.preventDefault();
};
this.canvas.addEventListener("DOMMouseScroll", handlers.wheelHandler, false);
this.canvas.addEventListener("mousewheel", handlers.wheelHandler, false);
};
this.useid = function(subsceneid, type) {
var sub = this.getObj(subsceneid);
if (sub.embeddings[type] === "inherit")
return(this.useid(sub.parent, type));
else
return subsceneid;
};
this.inViewport = function(coords, subsceneid) {
var viewport = this.getObj(subsceneid).par3d.viewport,
x0 = coords.x - viewport.x*this.canvas.width,
y0 = coords.y - viewport.y*this.canvas.height;
return 0 <= x0 && x0 <= viewport.width*this.canvas.width &&
0 <= y0 && y0 <= viewport.height*this.canvas.height;
};
this.whichSubscene = function(coords) {
var self = this,
recurse = function(subsceneid) {
var subscenes = self.getChildSubscenes(subsceneid), i, id;
for (i=0; i < subscenes.length; i++) {
id = recurse(subscenes[i]);
if (typeof(id) !== "undefined")
return(id);
}
if (self.inViewport(coords, subsceneid))
return(subsceneid);
else
return undefined;
},
rootid = this.scene.rootSubscene,
result = recurse(rootid);
if (typeof(result) === "undefined")
result = rootid;
return result;
};
this.translateCoords = function(subsceneid, coords) {
var viewport = this.getObj(subsceneid).par3d.viewport;
return {x: coords.x - viewport.x*this.canvas.width,
y: coords.y - viewport.y*this.canvas.height};
};
this.initSphere = function() {
var verts = this.scene.sphereVerts,
reuse = verts.reuse, result;
if (typeof reuse !== "undefined") {
var prev = document.getElementById(reuse).rglinstance.sphere;
result = {values: prev.values, vOffsets: prev.vOffsets, it: prev.it};
} else
result = {values: new Float32Array(this.flatten(this.cbind(this.transpose(verts.vb),
this.transpose(verts.texcoords)))),
it: new Uint16Array(this.flatten(this.transpose(verts.it))),
vOffsets: {vofs:0, cofs:-1, nofs:-1, radofs:-1, oofs:-1,
tofs:3, stride:5}};
result.sphereCount = result.it.length;
this.sphere = result;
};
this.initSphereGL = function() {
var gl = this.gl || this.initGL(), sphere = this.sphere;
if (gl.isContextLost()) return;
sphere.buf = gl.createBuffer();
gl.bindBuffer(gl.ARRAY_BUFFER, sphere.buf);
gl.bufferData(gl.ARRAY_BUFFER, sphere.values, gl.STATIC_DRAW);
sphere.ibuf = gl.createBuffer();
gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, sphere.ibuf);
gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, sphere.it, gl.STATIC_DRAW);
return;
};
this.initialize = function(el, x) {
this.textureCanvas = document.createElement("canvas");
this.textureCanvas.style.display = "block";
this.scene = x;
this.normMatrix = new CanvasMatrix4();
this.saveMat = {};
this.distance = null;
this.posLoc = 0;
this.colLoc = 1;
if (el) {
el.rglinstance = this;
this.el = el;
this.webGLoptions = el.rglinstance.scene.webGLoptions;
this.initCanvas();
}
};
this.restartCanvas = function() {
var newcanvas = document.createElement("canvas");
newcanvas.width = this.el.width;
newcanvas.height = this.el.height;
newcanvas.addEventListener("webglcontextrestored",
this.onContextRestored, false);
newcanvas.addEventListener("webglcontextlost",
this.onContextLost, false);
while (this.el.firstChild) {
this.el.removeChild(this.el.firstChild);
}
this.el.appendChild(newcanvas);
this.canvas = newcanvas;
this.gl = null;
};
this.initCanvas = function() {
this.restartCanvas();
var objs = this.scene.objects,
self = this;
Object.keys(objs).forEach(function(key){
var id = parseInt(key, 10),
obj = self.getObj(id);
if (typeof obj.reuse !== "undefined")
self.copyObj(id, obj.reuse);
});
Object.keys(objs).forEach(function(key){
self.initSubscene(parseInt(key, 10));
});
this.setMouseHandlers();
this.initSphere();
this.onContextRestored = function(event) {
self.initGL();
self.drawScene();
// console.log("restored context for "+self.scene.rootSubscene);
};
this.onContextLost = function(event) {
if (!self.drawing)
self.restartCanvas();
event.preventDefault();
};
this.initGL0();
lazyLoadScene = function() {
if (self.isInBrowserViewport()) {
if (!self.gl) {
self.initGL();
}
self.drawScene();
}
};
window.addEventListener("DOMContentLoaded", lazyLoadScene, false);
window.addEventListener("load", lazyLoadScene, false);
window.addEventListener("resize", lazyLoadScene, false);
window.addEventListener("scroll", lazyLoadScene, false);
};
/* this is only used by writeWebGL; rglwidget has
no debug element and does the drawing in rglwidget.js */
this.start = function() {
if (typeof this.prefix !== "undefined") {
this.debugelement = document.getElementById(this.prefix + "debug");
this.debug("");
}
this.drag = 0;
this.drawScene();
};
this.debug = function(msg, img) {
if (typeof this.debugelement !== "undefined" && this.debugelement !== null) {
this.debugelement.innerHTML = msg;
if (typeof img !== "undefined") {
this.debugelement.insertBefore(img, this.debugelement.firstChild);
}
} else if (msg !== "")
alert(msg);
};
this.getSnapshot = function() {
var img;
if (typeof this.scene.snapshot !== "undefined") {
img = document.createElement("img");
img.src = this.scene.snapshot;
img.alt = "Snapshot";
}
return img;
};
this.initGL0 = function() {
if (!window.WebGLRenderingContext){
alert("Your browser does not support WebGL. See http://get.webgl.org");
return;
}
};
this.isInBrowserViewport = function() {
var rect = this.canvas.getBoundingClientRect(),
windHeight = (window.innerHeight || document.documentElement.clientHeight),
windWidth = (window.innerWidth || document.documentElement.clientWidth);
return (
rect.top >= -windHeight &&
rect.left >= -windWidth &&
rect.bottom <= 2*windHeight &&
rect.right <= 2*windWidth);
};
this.initGL = function() {
var self = this;
if (this.gl) {
if (!this.drawing && this.gl.isContextLost())
this.restartCanvas();
else
return this.gl;
}
// if (!this.isInBrowserViewport()) return; Return what??? At this point we know this.gl is null.
this.canvas.addEventListener("webglcontextrestored",
this.onContextRestored, false);
this.canvas.addEventListener("webglcontextlost",
this.onContextLost, false);
this.gl = this.canvas.getContext("webgl", this.webGLoptions) ||
this.canvas.getContext("experimental-webgl", this.webGLoptions);
var save = this.startDrawing();
this.initSphereGL();
Object.keys(this.scene.objects).forEach(function(key){
self.initObj(parseInt(key, 10));
});
this.stopDrawing(save);
return this.gl;
};
this.resize = function(el) {
this.canvas.width = el.width;
this.canvas.height = el.height;
};
this.drawScene = function() {
var gl = this.gl || this.initGL(),
save = this.startDrawing();
gl.enable(gl.DEPTH_TEST);
gl.depthFunc(gl.LEQUAL);
gl.clearDepth(1.0);
gl.clearColor(1,1,1,1);
gl.depthMask(true); // Must be true before clearing depth buffer
gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);
this.drawSubscene(this.scene.rootSubscene);
this.stopDrawing(save);
};
this.subsetSetter = function(el, control) {
if (typeof control.subscenes === "undefined" ||
control.subscenes === null)
control.subscenes = this.scene.rootSubscene;
var value = Math.round(control.value),
subscenes = [].concat(control.subscenes),
fullset = [].concat(control.fullset),
i, j, entries, subsceneid,
adds = [], deletes = [],
ismissing = function(x) {
return fullset.indexOf(x) < 0;
},
tointeger = function(x) {
return parseInt(x, 10);
};
if (control.accumulate)
for (i=0; i <= value; i++)
adds = adds.concat(control.subsets[i]);
else
adds = adds.concat(control.subsets[value]);
deletes = fullset.filter(function(x) { return adds.indexOf(x) < 0 });
for (i = 0; i < subscenes.length; i++) {
subsceneid = subscenes[i];
if (typeof this.getObj(subsceneid) === "undefined")
this.alertOnce("typeof object is undefined");
for (j = 0; j < adds.length; j++)
this.addToSubscene(adds[j], subsceneid);
for (j = 0; j < deletes.length; j++)
this.delFromSubscene(deletes[j], subsceneid);
}
};
this.propertySetter = function(el, control) {
var value = control.value,
values = [].concat(control.values),
svals = [].concat(control.param),
direct = values[0] === null,
entries = [].concat(control.entries),
ncol = entries.length,
nrow = values.length/ncol,
properties = this.repeatToLen(control.properties, ncol),
objids = this.repeatToLen(control.objids, ncol),
property, objid = objids[0],
obj = this.getObj(objid),
propvals, i, v1, v2, p, entry, gl, needsBinding,
newprop, newid,
getPropvals = function() {
if (property === "userMatrix")
return obj.par3d.userMatrix.getAsArray();
else if (property === "scale" || property === "FOV" || property === "zoom")
return [].concat(obj.par3d[property]);
else
return [].concat(obj[property]);
};
putPropvals = function(newvals) {
if (newvals.length == 1)
newvals = newvals[0];
if (property === "userMatrix")
obj.par3d.userMatrix.load(newvals);
else if (property === "scale" || property === "FOV" || property === "zoom")
obj.par3d[property] = newvals;
else
obj[property] = newvals;
}
if (direct && typeof value === "undefined")
return;
if (control.interp) {
values = values.slice(0, ncol).concat(values).
concat(values.slice(ncol*(nrow-1), ncol*nrow));
svals = [-Infinity].concat(svals).concat(Infinity);
for (i = 1; i < svals.length; i++) {
if (value <= svals[i]) {
if (svals[i] === Infinity)
p = 1;
else
p = (svals[i] - value)/(svals[i] - svals[i-1]);
break;
}
}
} else if (!direct) {
value = Math.round(value);
}
for (j=0; j<entries.length; j++) {
entry = entries[j];
newprop = properties[j];
newid = objids[j];
if (newprop !== property || newid != objid) {
if (typeof property !== "undefined")
putPropvals(propvals);
property = newprop;
objid = newid;
obj = this.getObj(objid);
propvals = getPropvals();
}
if (control.interp) {
v1 = values[ncol*(i-1) + j];
v2 = values[ncol*i + j];
this.setElement(propvals, entry, p*v1 + (1-p)*v2);
} else if (!direct) {
this.setElement(propvals, entry, values[ncol*value + j]);
} else {
this.setElement(propvals, entry, value[j]);
}
}
putPropvals(propvals);
needsBinding = [];
for (j=0; j < entries.length; j++) {
if (properties[j] === "values" &&
needsBinding.indexOf(objids[j]) === -1) {
needsBinding.push(objids[j]);
}
}
for (j=0; j < needsBinding.length; j++) {
gl = this.gl || this.initGL();
obj = this.getObj(needsBinding[j]);
gl.bindBuffer(gl.ARRAY_BUFFER, obj.buf);
gl.bufferData(gl.ARRAY_BUFFER, obj.values, gl.STATIC_DRAW);
}
};
this.vertexSetter = function(el, control) {
var svals = [].concat(control.param),
j, k, p, propvals, stride, ofs, obj,
attrib,
ofss = {x:"vofs", y:"vofs", z:"vofs",
red:"cofs", green:"cofs", blue:"cofs",
alpha:"cofs", radii:"radofs",
nx:"nofs", ny:"nofs", nz:"nofs",
ox:"oofs", oy:"oofs", oz:"oofs",
ts:"tofs", tt:"tofs"},
pos = {x:0, y:1, z:2,
red:0, green:1, blue:2,
alpha:3,radii:0,
nx:0, ny:1, nz:2,
ox:0, oy:1, oz:2,
ts:0, tt:1},
values = control.values,
direct = values === null,
ncol,
interp = control.interp,
vertices = [].concat(control.vertices),
attributes = [].concat(control.attributes),
value = control.value;
ncol = Math.max(vertices.length, attributes.length);
if (!ncol)
return;
vertices = this.repeatToLen(vertices, ncol);
attributes = this.repeatToLen(attributes, ncol);
if (direct)
interp = false;
/* JSON doesn't pass Infinity */
svals[0] = -Infinity;
svals[svals.length - 1] = Infinity;
for (j = 1; j < svals.length; j++) {
if (value <= svals[j]) {
if (interp) {
if (svals[j] === Infinity)
p = 1;
else
p = (svals[j] - value)/(svals[j] - svals[j-1]);
} else {
if (svals[j] - value > value - svals[j-1])
j = j - 1;
}
break;
}
}
obj = this.getObj(control.objid);
propvals = obj.values;
for (k=0; k<ncol; k++) {
attrib = attributes[k];
vertex = vertices[k];
ofs = obj.vOffsets[ofss[attrib]];
if (ofs < 0)
this.alertOnce("Attribute '"+attrib+"' not found in object "+control.objid);
else {
stride = obj.vOffsets.stride;
ofs = vertex*stride + ofs + pos[attrib];
if (direct) {
propvals[ofs] = value;
} else if (interp) {
propvals[ofs] = p*values[j-1][k] + (1-p)*values[j][k];
} else {
propvals[ofs] = values[j][k];
}
}
}
if (typeof obj.buf !== "undefined") {
var gl = this.gl || this.initGL();
gl.bindBuffer(gl.ARRAY_BUFFER, obj.buf);
gl.bufferData(gl.ARRAY_BUFFER, propvals, gl.STATIC_DRAW);
}
};
this.ageSetter = function(el, control) {
var objids = [].concat(control.objids),
nobjs = objids.length,
time = control.value,
births = [].concat(control.births),
ages = [].concat(control.ages),
steps = births.length,
j = Array(steps),
p = Array(steps),
i, k, age, j0, propvals, stride, ofs, objid, obj,
attrib, dim,
attribs = ["colors", "alpha", "radii", "vertices",
"normals", "origins", "texcoords",
"x", "y", "z",
"red", "green", "blue"],
ofss = ["cofs", "cofs", "radofs", "vofs",
"nofs", "oofs", "tofs",
"vofs", "vofs", "vofs",
"cofs", "cofs", "cofs"],
dims = [3,1,1,3,
3,2,2,
1,1,1,
1,1,1],
pos = [0,3,0,0,
0,0,0,
0,1,2,
0,1,2];
/* Infinity doesn't make it through JSON */
ages[0] = -Infinity;
ages[ages.length-1] = Infinity;
for (i = 0; i < steps; i++) {
if (births[i] !== null) { // NA in R becomes null
age = time - births[i];
for (j0 = 1; age > ages[j0]; j0++);
if (ages[j0] == Infinity)
p[i] = 1;
else if (ages[j0] > ages[j0-1])
p[i] = (ages[j0] - age)/(ages[j0] - ages[j0-1]);
else
p[i] = 0;
j[i] = j0;
}
}
for (l = 0; l < nobjs; l++) {
objid = objids[l];
obj = this.getObj(objid);
if (typeof obj.vOffsets === "undefined")
continue;
propvals = obj.values;
stride = obj.vOffsets.stride;
for (k = 0; k < attribs.length; k++) {
attrib = control[attribs[k]];
if (typeof attrib !== "undefined") {
ofs = obj.vOffsets[ofss[k]];
if (ofs >= 0) {
dim = dims[k];
ofs = ofs + pos[k];
for (i = 0; i < steps; i++) {
if (births[i] !== null) {
for (d=0; d < dim; d++) {
propvals[i*stride + ofs + d] = p[i]*attrib[dim*(j[i]-1) + d] + (1-p[i])*attrib[dim*j[i] + d];
}
}
}
} else
this.alertOnce("\'"+attribs[k]+"\' property not found in object "+objid);
}
}
obj.values = propvals;
if (typeof obj.buf !== "undefined") {
gl = this.gl || this.initGL();
gl.bindBuffer(gl.ARRAY_BUFFER, obj.buf);
gl.bufferData(gl.ARRAY_BUFFER, obj.values, gl.STATIC_DRAW);
}
}
};
this.oldBridge = function(el, control) {
var attrname, global = window[control.prefix + "rgl"];
if (typeof global !== "undefined")
for (attrname in global)
this[attrname] = global[attrname];
window[control.prefix + "rgl"] = this;
};
this.Player = function(el, control) {
var
self = this,
components = [].concat(control.components),
buttonLabels = [].concat(control.buttonLabels),
Tick = function() { /* "this" will be a timer */
var i,
nominal = this.value,
slider = this.Slider,
labels = this.outputLabels,
output = this.Output,
step;
if (typeof slider !== "undefined" && nominal != slider.value)
slider.value = nominal;
if (typeof output !== "undefined") {
step = Math.round((nominal - output.sliderMin)/output.sliderStep);
if (labels !== null) {
output.innerHTML = labels[step];
} else {
step = step*output.sliderStep + output.sliderMin;
output.innerHTML = step.toPrecision(output.outputPrecision);
}
}
for (i=0; i < this.actions.length; i++) {
this.actions[i].value = nominal;
}
self.applyControls(el, this.actions, false);
self.drawScene();
},
OnSliderInput = function() { /* "this" will be the slider */
this.rgltimer.value = Number(this.value);
this.rgltimer.Tick();
},
addSlider = function(min, max, step, value) {
var slider = document.createElement("input");
slider.type = "range";
slider.min = min;
slider.max = max;
slider.step = step;
slider.value = value;
slider.oninput = OnSliderInput;
slider.sliderActions = control.actions;
slider.sliderScene = this;
slider.className = "rgl-slider";
slider.id = el.id + "-slider";
el.rgltimer.Slider = slider;
slider.rgltimer = el.rgltimer;
el.appendChild(slider);
},
addLabel = function(labels, min, step, precision) {
var output = document.createElement("output");
output.sliderMin = min;
output.sliderStep = step;
output.outputPrecision = precision;
output.className = "rgl-label";
output.id = el.id + "-label";
el.rgltimer.Output = output;
el.rgltimer.outputLabels = labels;
el.appendChild(output);
},
addButton = function(which, label, active) {
var button = document.createElement("input"),
onclicks = {Reverse: function() { this.rgltimer.reverse();},
Play: function() { this.rgltimer.play();
this.value = this.rgltimer.enabled ? this.inactiveValue : this.activeValue; },
Slower: function() { this.rgltimer.slower(); },
Faster: function() { this.rgltimer.faster(); },
Reset: function() { this.rgltimer.reset(); },
Step: function() { this.rgltimer.step(); }
};
button.rgltimer = el.rgltimer;
button.type = "button";
button.value = label;
button.activeValue = label;
button.inactiveValue = active;
if (which === "Play")
button.rgltimer.PlayButton = button;
button.onclick = onclicks[which];
button.className = "rgl-button";
button.id = el.id + "-" + which;
el.appendChild(button);
};
if (typeof control.reinit !== "undefined" && control.reinit !== null) {
control.actions.reinit = control.reinit;
}
el.rgltimer = new rgltimerClass(Tick, control.start, control.interval, control.stop,
control.step, control.value, control.rate, control.loop, control.actions);
for (var i=0; i < components.length; i++) {
switch(components[i]) {
case "Slider": addSlider(control.start, control.stop,
control.step, control.value);
break;
case "Label": addLabel(control.labels, control.start,
control.step, control.precision);
break;
default:
addButton(components[i], buttonLabels[i], control.pause);
}
}
el.rgltimer.Tick();
};
this.applyControls = function(el, x, draw) {
var self = this, reinit = x.reinit, i, control, type;
for (i = 0; i < x.length; i++) {
control = x[i];
type = control.type;
self[type](el, control);
}
if (typeof reinit !== "undefined" && reinit !== null) {
reinit = [].concat(reinit);
for (i = 0; i < reinit.length; i++)
self.getObj(reinit[i]).initialized = false;
}
if (typeof draw === "undefined" || draw)
self.drawScene();
};
this.sceneChangeHandler = function(message) {
var self = document.getElementById(message.elementId).rglinstance,
objs = message.objects, mat = message.material,
root = message.rootSubscene,
initSubs = message.initSubscenes,
redraw = message.redrawScene,
skipRedraw = message.skipRedraw,
deletes, subs, allsubs = [], i,j;
if (typeof message.delete !== "undefined") {
deletes = [].concat(message.delete);
if (typeof message.delfromSubscenes !== "undefined")
subs = [].concat(message.delfromSubscenes);
else
subs = [];
for (i = 0; i < deletes.length; i++) {
for (j = 0; j < subs.length; j++) {
self.delFromSubscene(deletes[i], subs[j]);
}
delete self.scene.objects[deletes[i]];
}
}
if (typeof objs !== "undefined") {
Object.keys(objs).forEach(function(key){
key = parseInt(key, 10);
self.scene.objects[key] = objs[key];
self.initObj(key);
var obj = self.getObj(key),
subs = [].concat(obj.inSubscenes), k;
allsubs = allsubs.concat(subs);
for (k = 0; k < subs.length; k++)
self.addToSubscene(key, subs[k]);
});
}
if (typeof mat !== "undefined") {
self.scene.material = mat;
}
if (typeof root !== "undefined") {
self.scene.rootSubscene = root;
}
if (typeof initSubs !== "undefined")
allsubs = allsubs.concat(initSubs);
allsubs = self.unique(allsubs);
for (i = 0; i < allsubs.length; i++) {
self.initSubscene(allsubs[i]);
}
if (typeof skipRedraw !== "undefined") {
root = self.getObj(self.scene.rootSubscene);
root.par3d.skipRedraw = skipRedraw;
}
if (redraw)
self.drawScene();
};
}).call(rglwidgetClass.prototype);
rgltimerClass = function(Tick, startTime, interval, stopTime, stepSize, value, rate, loop, actions) {
this.enabled = false;
this.timerId = 0;
this.startTime = startTime; /* nominal start time in seconds */
this.value = value; /* current nominal time */
this.interval = interval; /* seconds between updates */
this.stopTime = stopTime; /* nominal stop time */
this.stepSize = stepSize; /* nominal step size */
this.rate = rate; /* nominal units per second */
this.loop = loop; /* "none", "cycle", or "oscillate" */
this.realStart = undefined; /* real world start time */
this.multiplier = 1; /* multiplier for fast-forward
or reverse */
this.actions = actions;
this.Tick = Tick;
};
(function() {
this.play = function() {
if (this.enabled) {
this.enabled = false;
window.clearInterval(this.timerId);
this.timerId = 0;
return;
}
var tick = function(self) {
var now = new Date();
self.value = self.multiplier*self.rate*(now - self.realStart)/1000 + self.startTime;
self.forceToRange();
if (typeof self.Tick !== "undefined") {
self.Tick(self.value);
}
};
this.realStart = new Date() - 1000*(this.value - this.startTime)/this.rate/this.multiplier;
this.timerId = window.setInterval(tick, 1000*this.interval, this);
this.enabled = true;
};
this.forceToRange = function() {
if (this.value > this.stopTime + this.stepSize/2 || this.value < this.startTime - this.stepSize/2) {
if (!this.loop) {
this.reset();
} else {
var cycle = this.stopTime - this.startTime + this.stepSize,
newval = (this.value - this.startTime) % cycle + this.startTime;
if (newval < this.startTime) {
newval += cycle;
}
this.realStart += (this.value - newval)*1000/this.multiplier/this.rate;
this.value = newval;
}
}
}
this.reset = function() {
this.value = this.startTime;
this.newmultiplier(1);
if (typeof this.Tick !== "undefined") {
this.Tick(this.value);
}
if (this.enabled)
this.play(); /* really pause... */
if (typeof this.PlayButton !== "undefined")
this.PlayButton.value = "Play";
};
this.faster = function() {
this.newmultiplier(Math.SQRT2*this.multiplier);
};
this.slower = function() {
this.newmultiplier(this.multiplier/Math.SQRT2);
};
this.reverse = function() {
this.newmultiplier(-this.multiplier);
};
this.newmultiplier = function(newmult) {
if (newmult != this.multiplier) {
this.realStart += 1000*(this.value - this.startTime)/this.rate*(1/this.multiplier - 1/newmult);
this.multiplier = newmult;
}
};
this.step = function() {
this.value += this.rate*this.multiplier;
this.forceToRange();
if (typeof this.Tick !== "undefined")
this.Tick(this.value);
}
}).call(rgltimerClass.prototype);</script>
<div id="div" class="rglWebGL"></div>
<script type="text/javascript">
var div = document.getElementById("div"),
rgl = new rglwidgetClass();
div.width = 1000;
div.height = 1000;
rgl.initialize(div,
{"material":{"color":"#000000","alpha":1,"lit":true,"ambient":"#000000","specular":"#FFFFFF","emission":"#000000","shininess":50,"smooth":true,"front":"filled","back":"filled","size":3,"lwd":1,"fog":false,"point_antialias":false,"line_antialias":false,"texture":null,"textype":"rgb","texmipmap":false,"texminfilter":"linear","texmagfilter":"linear","texenvmap":false,"depth_mask":true,"depth_test":"less","isTransparent":false},"rootSubscene":66,"objects":{"72":{"id":72,"type":"spheres","material":{},"vertices":[[1,1,1],[1,8,2],[2,7,4],[1,1,10],[8,9,11],[8,10,11]],"colors":[[1,0,0,1]],"radii":[[0.006859576]],"centers":[[1,1,1],[1,8,2],[2,7,4],[1,1,10],[8,9,11],[8,10,11]],"ignoreExtent":false,"flags":3},"73":{"id":73,"type":"spheres","material":{},"vertices":[[1,8,1],[4,11,7],[7,9,10],[7,10,11]],"colors":[[1,0.1215686,0,1]],"radii":[[0.01575917]],"centers":[[1,8,1],[4,11,7],[7,9,10],[7,10,11]],"ignoreExtent":false,"flags":3},"74":{"id":74,"type":"spheres","material":{},"vertices":[[2,1,1],[4,6,1],[3,1,2],[5,1,2],[3,2,2],[1,7,2],[1,2,3],[2,7,3],[2,6,4],[4,11,5],[4,11,6],[8,9,10],[7,9,11],[9,9,11],[9,10,11]],"colors":[[1,0.2431373,0,1]],"radii":[[0.02563556]],"centers":[[2,1,1],[4,6,1],[3,1,2],[5,1,2],[3,2,2],[1,7,2],[1,2,3],[2,7,3],[2,6,4],[4,11,5],[4,11,6],[8,9,10],[7,9,11],[9,9,11],[9,10,11]],"ignoreExtent":false,"flags":3},"75":{"id":75,"type":"spheres","material":{},"vertices":[[3,1,1],[1,2,1],[3,2,1],[5,2,1],[1,7,1],[1,9,1],[1,1,2],[5,2,2],[5,1,3],[3,2,3],[8,3,3],[1,4,3],[1,7,3],[1,8,3],[2,8,4],[4,11,4],[5,11,4],[6,11,4],[2,6,5],[2,7,5],[3,9,5],[4,9,5],[3,9,6],[3,9,7],[4,10,7],[5,11,7],[1,1,9],[7,8,10],[7,10,10],[6,10,11]],"colors":[[1,0.3686275,0,1]],"radii":[[0.03620506]],"centers":[[3,1,1],[1,2,1],[3,2,1],[5,2,1],[1,7,1],[1,9,1],[1,1,2],[5,2,2],[5,1,3],[3,2,3],[8,3,3],[1,4,3],[1,7,3],[1,8,3],[2,8,4],[4,11,4],[5,11,4],[6,11,4],[2,6,5],[2,7,5],[3,9,5],[4,9,5],[3,9,6],[3,9,7],[4,10,7],[5,11,7],[1,1,9],[7,8,10],[7,10,10],[6,10,11]],"ignoreExtent":false,"flags":3},"76":{"id":76,"type":"spheres","material":{},"vertices":[[7,3,1],[3,4,1],[4,5,1],[4,7,1],[2,1,2],[4,1,2],[1,2,2],[3,3,2],[4,5,2],[1,6,2],[4,6,2],[1,9,2],[3,1,3],[2,2,3],[1,3,3],[7,3,3],[1,6,3],[2,6,3],[4,6,3],[6,11,3],[1,1,4],[2,1,4],[4,1,4],[1,2,4],[1,6,4],[1,7,4],[4,9,4],[3,8,5],[4,10,5],[5,11,5],[4,9,6],[4,10,6],[5,11,6],[3,11,7],[1,1,8],[3,9,8],[8,10,10],[8,8,11]],"colors":[[1,0.4901961,0,1]],"radii":[[0.0473218]],"centers":[[7,3,1],[3,4,1],[4,5,1],[4,7,1],[2,1,2],[4,1,2],[1,2,2],[3,3,2],[4,5,2],[1,6,2],[4,6,2],[1,9,2],[3,1,3],[2,2,3],[1,3,3],[7,3,3],[1,6,3],[2,6,3],[4,6,3],[6,11,3],[1,1,4],[2,1,4],[4,1,4],[1,2,4],[1,6,4],[1,7,4],[4,9,4],[3,8,5],[4,10,5],[5,11,5],[4,9,6],[4,10,6],[5,11,6],[3,11,7],[1,1,8],[3,9,8],[8,10,10],[8,8,11]],"ignoreExtent":false,"flags":3},"77":{"id":77,"type":"spheres","material":{},"vertices":[[6,2,1],[1,3,1],[3,3,1],[5,3,1],[1,4,2],[3,4,2],[1,1,3],[2,1,3],[2,3,3],[3,3,3],[3,4,3],[2,8,3],[5,11,3],[2,2,4],[8,3,4],[1,4,4],[1,5,4],[2,5,4],[1,8,4],[3,9,4],[4,10,4],[4,1,5],[3,7,5],[4,7,5],[2,8,5],[4,8,5],[6,11,5],[6,11,6],[6,4,7],[8,11,7],[6,3,8],[4,11,8],[5,11,8],[8,8,10],[6,10,10],[1,1,11],[9,8,11],[8,11,11]],"colors":[[1,0.6117647,0,1]],"radii":[[0.05889504]],"centers":[[6,2,1],[1,3,1],[3,3,1],[5,3,1],[1,4,2],[3,4,2],[1,1,3],[2,1,3],[2,3,3],[3,3,3],[3,4,3],[2,8,3],[5,11,3],[2,2,4],[8,3,4],[1,4,4],[1,5,4],[2,5,4],[1,8,4],[3,9,4],[4,10,4],[4,1,5],[3,7,5],[4,7,5],[2,8,5],[4,8,5],[6,11,5],[6,11,6],[6,4,7],[8,11,7],[6,3,8],[4,11,8],[5,11,8],[8,8,10],[6,10,10],[1,1,11],[9,8,11],[8,11,11]],"ignoreExtent":false,"flags":3},"78":{"id":78,"type":"spheres","material":{},"vertices":[[5,1,1],[7,2,1],[2,3,1],[6,3,1],[1,4,1],[2,5,1],[1,6,1],[10,10,1],[2,2,2],[4,2,2],[2,3,2],[1,5,2],[4,1,3],[2,4,3],[1,5,3],[2,5,3],[4,11,3],[3,1,4],[3,2,4],[1,3,4],[3,7,4],[5,10,4],[2,5,5],[3,11,5],[2,6,6],[2,7,6],[3,7,6],[4,9,7],[3,10,7],[5,10,7],[6,11,7],[6,4,8],[8,9,8],[9,9,8],[4,10,8],[8,9,9],[9,9,9],[9,9,10],[7,8,11],[6,11,11]],"colors":[[1,0.7333333,0,1]],"radii":[[0.07086225]],"centers":[[5,1,1],[7,2,1],[2,3,1],[6,3,1],[1,4,1],[2,5,1],[1,6,1],[10,10,1],[2,2,2],[4,2,2],[2,3,2],[1,5,2],[4,1,3],[2,4,3],[1,5,3],[2,5,3],[4,11,3],[3,1,4],[3,2,4],[1,3,4],[3,7,4],[5,10,4],[2,5,5],[3,11,5],[2,6,6],[2,7,6],[3,7,6],[4,9,7],[3,10,7],[5,10,7],[6,11,7],[6,4,8],[8,9,8],[9,9,8],[4,10,8],[8,9,9],[9,9,9],[9,9,10],[7,8,11],[6,11,11]],"ignoreExtent":false,"flags":3},"79":{"id":79,"type":"spheres","material":{},"vertices":[[2,4,1],[4,4,1],[1,5,1],[3,5,1],[9,10,1],[6,1,2],[2,4,2],[4,4,2],[3,5,2],[2,6,2],[2,7,2],[5,10,2],[6,1,3],[5,2,3],[7,2,3],[3,5,3],[4,5,3],[4,7,3],[3,9,3],[4,9,3],[4,10,3],[5,10,3],[7,11,3],[7,3,4],[2,4,4],[3,6,4],[4,7,4],[3,8,4],[6,10,4],[1,1,5],[2,1,5],[3,1,5],[1,2,5],[3,2,5],[8,3,5],[1,5,5],[3,6,5],[5,10,5],[6,10,5],[1,1,6],[2,1,6],[3,2,6],[3,8,6],[5,9,6],[5,10,6],[6,10,6],[3,11,6],[8,11,6],[6,3,7],[9,9,7],[2,11,7],[3,7,8],[6,11,8],[6,3,9],[7,9,9],[7,10,9],[1,5,10],[9,10,10],[7,11,11],[9,11,11]],"colors":[[1,0.8588235,0,1]],"radii":[[0.08317738]],"centers":[[2,4,1],[4,4,1],[1,5,1],[3,5,1],[9,10,1],[6,1,2],[2,4,2],[4,4,2],[3,5,2],[2,6,2],[2,7,2],[5,10,2],[6,1,3],[5,2,3],[7,2,3],[3,5,3],[4,5,3],[4,7,3],[3,9,3],[4,9,3],[4,10,3],[5,10,3],[7,11,3],[7,3,4],[2,4,4],[3,6,4],[4,7,4],[3,8,4],[6,10,4],[1,1,5],[2,1,5],[3,1,5],[1,2,5],[3,2,5],[8,3,5],[1,5,5],[3,6,5],[5,10,5],[6,10,5],[1,1,6],[2,1,6],[3,2,6],[3,8,6],[5,9,6],[5,10,6],[6,10,6],[3,11,6],[8,11,6],[6,3,7],[9,9,7],[2,11,7],[3,7,8],[6,11,8],[6,3,9],[7,9,9],[7,10,9],[1,5,10],[9,10,10],[7,11,11],[9,11,11]],"ignoreExtent":false,"flags":3},"80":{"id":80,"type":"spheres","material":{},"vertices":[[4,1,1],[6,1,1],[2,2,1],[4,2,1],[2,6,1],[1,10,1],[6,2,2],[1,3,2],[7,3,2],[8,3,2],[8,4,2],[2,5,2],[4,7,2],[2,8,2],[5,11,2],[8,11,2],[6,2,3],[8,2,3],[6,3,3],[8,4,3],[8,11,3],[5,1,4],[6,1,4],[6,2,4],[2,3,4],[3,3,4],[6,3,4],[3,4,4],[3,5,4],[4,8,4],[8,2,5],[3,3,5],[1,4,5],[1,6,5],[5,9,5],[3,10,5],[7,11,5],[3,3,6],[3,6,6],[4,7,6],[2,8,6],[4,8,6],[11,9,6],[3,10,6],[3,2,7],[3,3,7],[5,3,7],[5,4,7],[3,7,7],[5,9,7],[1,11,7],[7,11,7],[2,7,8],[4,9,8],[5,10,8],[8,10,8],[1,11,8],[3,11,8],[1,6,10],[9,8,10],[6,9,10],[6,11,10],[6,9,11],[1,10,11]],"colors":[[1,0.9803922,0,1]],"radii":[[0.09580502]],"centers":[[4,1,1],[6,1,1],[2,2,1],[4,2,1],[2,6,1],[1,10,1],[6,2,2],[1,3,2],[7,3,2],[8,3,2],[8,4,2],[2,5,2],[4,7,2],[2,8,2],[5,11,2],[8,11,2],[6,2,3],[8,2,3],[6,3,3],[8,4,3],[8,11,3],[5,1,4],[6,1,4],[6,2,4],[2,3,4],[3,3,4],[6,3,4],[3,4,4],[3,5,4],[4,8,4],[8,2,5],[3,3,5],[1,4,5],[1,6,5],[5,9,5],[3,10,5],[7,11,5],[3,3,6],[3,6,6],[4,7,6],[2,8,6],[4,8,6],[11,9,6],[3,10,6],[3,2,7],[3,3,7],[5,3,7],[5,4,7],[3,7,7],[5,9,7],[1,11,7],[7,11,7],[2,7,8],[4,9,8],[5,10,8],[8,10,8],[1,11,8],[3,11,8],[1,6,10],[9,8,10],[6,9,10],[6,11,10],[6,9,11],[1,10,11]],"ignoreExtent":false,"flags":3},"81":{"id":81,"type":"spheres","material":{},"vertices":[[4,3,1],[5,7,1],[4,8,1],[8,10,1],[8,11,1],[7,2,2],[5,3,2],[4,9,2],[5,9,2],[4,10,2],[6,11,2],[7,1,3],[9,1,3],[3,6,3],[3,7,3],[1,9,3],[8,2,4],[4,6,4],[3,10,4],[3,11,4],[7,11,4],[2,2,5],[1,3,5],[6,3,5],[7,3,5],[3,4,5],[1,7,5],[3,1,6],[8,2,6],[6,3,6],[6,4,6],[1,5,6],[1,6,6],[7,10,6],[7,11,6],[1,1,7],[4,5,7],[3,6,7],[2,7,7],[3,8,7],[11,9,7],[6,10,7],[2,1,8],[5,3,8],[4,5,8],[1,7,8],[3,8,8],[7,9,8],[3,10,8],[6,10,8],[7,10,8],[7,3,9],[3,9,9],[2,1,10],[6,8,10],[1,11,11],[10,11,11]],"colors":[[0.8980392,1,0,1]],"radii":[[0.1087169]],"centers":[[4,3,1],[5,7,1],[4,8,1],[8,10,1],[8,11,1],[7,2,2],[5,3,2],[4,9,2],[5,9,2],[4,10,2],[6,11,2],[7,1,3],[9,1,3],[3,6,3],[3,7,3],[1,9,3],[8,2,4],[4,6,4],[3,10,4],[3,11,4],[7,11,4],[2,2,5],[1,3,5],[6,3,5],[7,3,5],[3,4,5],[1,7,5],[3,1,6],[8,2,6],[6,3,6],[6,4,6],[1,5,6],[1,6,6],[7,10,6],[7,11,6],[1,1,7],[4,5,7],[3,6,7],[2,7,7],[3,8,7],[11,9,7],[6,10,7],[2,1,8],[5,3,8],[4,5,8],[1,7,8],[3,8,8],[7,9,8],[3,10,8],[6,10,8],[7,10,8],[7,3,9],[3,9,9],[2,1,10],[6,8,10],[1,11,11],[10,11,11]],"ignoreExtent":false,"flags":3},"82":{"id":82,"type":"spheres","material":{},"vertices":[[5,4,1],[2,7,1],[2,8,1],[3,9,1],[7,1,2],[4,3,2],[2,9,2],[3,9,2],[4,11,2],[7,11,2],[3,8,3],[2,9,3],[7,4,4],[8,4,4],[2,3,5],[2,4,5],[3,5,5],[4,6,5],[8,11,5],[4,1,6],[1,2,6],[2,2,6],[5,3,6],[1,4,6],[2,5,6],[3,5,6],[1,7,6],[9,9,6],[1,3,7],[1,4,7],[4,4,7],[5,5,7],[10,9,7],[8,10,7],[7,3,8],[5,4,8],[2,6,8],[3,6,8],[1,9,8],[5,9,8],[2,11,8],[8,11,8],[6,10,9],[9,10,9],[1,11,9],[5,8,10],[2,10,11],[10,10,11]],"colors":[[0.7764706,1,0,1]],"radii":[[0.1218901]],"centers":[[5,4,1],[2,7,1],[2,8,1],[3,9,1],[7,1,2],[4,3,2],[2,9,2],[3,9,2],[4,11,2],[7,11,2],[3,8,3],[2,9,3],[7,4,4],[8,4,4],[2,3,5],[2,4,5],[3,5,5],[4,6,5],[8,11,5],[4,1,6],[1,2,6],[2,2,6],[5,3,6],[1,4,6],[2,5,6],[3,5,6],[1,7,6],[9,9,6],[1,3,7],[1,4,7],[4,4,7],[5,5,7],[10,9,7],[8,10,7],[7,3,8],[5,4,8],[2,6,8],[3,6,8],[1,9,8],[5,9,8],[2,11,8],[8,11,8],[6,10,9],[9,10,9],[1,11,9],[5,8,10],[2,10,11],[10,10,11]],"ignoreExtent":false,"flags":3},"83":{"id":83,"type":"spheres","material":{},"vertices":[[8,4,1],[3,6,1],[3,8,1],[5,8,1],[2,9,1],[4,9,1],[9,9,1],[9,11,1],[6,3,2],[1,10,2],[8,1,3],[4,2,3],[9,3,3],[4,4,3],[7,4,3],[4,8,3],[5,9,3],[3,10,3],[6,10,3],[7,1,4],[8,1,4],[4,2,4],[5,2,4],[2,9,4],[5,1,5],[7,4,5],[2,9,5],[11,9,5],[7,10,5],[4,2,6],[1,3,6],[2,4,6],[3,4,6],[5,4,6],[2,9,6],[8,10,6],[2,1,7],[3,1,7],[1,2,7],[8,2,7],[2,3,7],[4,3,7],[6,5,7],[2,6,7],[2,9,7],[8,9,7],[9,10,7],[8,3,8],[4,4,8],[2,5,8],[5,5,8],[11,9,8],[9,10,8],[7,11,8],[5,3,9],[7,8,9],[9,8,9],[4,10,9],[5,10,9],[8,10,9],[5,11,9],[6,11,9],[6,3,10],[1,7,10],[5,9,10],[5,10,10],[8,11,10],[10,11,10],[6,8,11],[10,8,11],[10,9,11],[4,10,11]],"colors":[[0.654902,1,0,1]],"radii":[[0.1353053]],"centers":[[8,4,1],[3,6,1],[3,8,1],[5,8,1],[2,9,1],[4,9,1],[9,9,1],[9,11,1],[6,3,2],[1,10,2],[8,1,3],[4,2,3],[9,3,3],[4,4,3],[7,4,3],[4,8,3],[5,9,3],[3,10,3],[6,10,3],[7,1,4],[8,1,4],[4,2,4],[5,2,4],[2,9,4],[5,1,5],[7,4,5],[2,9,5],[11,9,5],[7,10,5],[4,2,6],[1,3,6],[2,4,6],[3,4,6],[5,4,6],[2,9,6],[8,10,6],[2,1,7],[3,1,7],[1,2,7],[8,2,7],[2,3,7],[4,3,7],[6,5,7],[2,6,7],[2,9,7],[8,9,7],[9,10,7],[8,3,8],[4,4,8],[2,5,8],[5,5,8],[11,9,8],[9,10,8],[7,11,8],[5,3,9],[7,8,9],[9,8,9],[4,10,9],[5,10,9],[8,10,9],[5,11,9],[6,11,9],[6,3,10],[1,7,10],[5,9,10],[5,10,10],[8,11,10],[10,11,10],[6,8,11],[10,8,11],[10,9,11],[4,10,11]],"ignoreExtent":false,"flags":3},"84":{"id":84,"type":"spheres","material":{},"vertices":[[5,5,1],[3,10,1],[8,2,2],[3,6,2],[8,10,2],[9,1,4],[7,2,4],[5,9,4],[8,11,4],[6,2,5],[8,4,5],[1,8,5],[2,3,6],[4,3,6],[8,3,6],[7,4,6],[4,5,6],[5,5,6],[4,6,6],[2,11,6],[2,2,7],[3,4,7],[3,5,7],[4,6,7],[1,9,7],[2,10,7],[7,10,7],[9,11,7],[1,2,8],[8,2,8],[7,4,8],[6,5,8],[1,6,8],[2,9,8],[10,9,8],[2,1,9],[8,3,9],[6,4,9],[1,9,9],[4,9,9],[5,9,9],[11,9,9],[2,11,9],[4,11,9],[7,3,10],[4,10,10],[1,11,10],[5,10,11],[5,11,11]],"colors":[[0.5294118,1,0,1]],"radii":[[0.1489462]],"centers":[[5,5,1],[3,10,1],[8,2,2],[3,6,2],[8,10,2],[9,1,4],[7,2,4],[5,9,4],[8,11,4],[6,2,5],[8,4,5],[1,8,5],[2,3,6],[4,3,6],[8,3,6],[7,4,6],[4,5,6],[5,5,6],[4,6,6],[2,11,6],[2,2,7],[3,4,7],[3,5,7],[4,6,7],[1,9,7],[2,10,7],[7,10,7],[9,11,7],[1,2,8],[8,2,8],[7,4,8],[6,5,8],[1,6,8],[2,9,8],[10,9,8],[2,1,9],[8,3,9],[6,4,9],[1,9,9],[4,9,9],[5,9,9],[11,9,9],[2,11,9],[4,11,9],[7,3,10],[4,10,10],[1,11,10],[5,10,11],[5,11,11]],"ignoreExtent":false,"flags":3},"85":{"id":85,"type":"spheres","material":{},"vertices":[[8,3,1],[5,6,1],[9,8,1],[5,9,1],[1,11,1],[7,11,1],[10,11,1],[9,1,2],[5,7,2],[3,10,2],[9,10,2],[9,2,3],[5,3,3],[8,10,3],[3,11,3],[1,9,4],[7,10,4],[8,10,4],[6,1,5],[8,1,5],[4,2,5],[5,3,5],[6,4,5],[9,9,5],[2,11,5],[5,2,6],[4,4,6],[8,4,6],[6,5,6],[11,8,6],[7,9,6],[8,9,6],[9,10,6],[4,2,7],[2,4,7],[1,5,7],[2,5,7],[1,6,7],[4,7,7],[4,8,7],[7,9,7],[9,2,8],[1,3,8],[4,3,8],[1,5,8],[3,5,8],[9,8,8],[11,8,8],[1,10,8],[1,2,9],[9,2,9],[1,5,9],[3,7,9],[5,8,9],[8,8,9],[6,9,9],[10,9,9],[3,11,9],[10,11,9],[10,10,10],[7,11,10],[9,11,10]],"colors":[[0.4078431,1,0,1]],"radii":[[0.1627987]],"centers":[[8,3,1],[5,6,1],[9,8,1],[5,9,1],[1,11,1],[7,11,1],[10,11,1],[9,1,2],[5,7,2],[3,10,2],[9,10,2],[9,2,3],[5,3,3],[8,10,3],[3,11,3],[1,9,4],[7,10,4],[8,10,4],[6,1,5],[8,1,5],[4,2,5],[5,3,5],[6,4,5],[9,9,5],[2,11,5],[5,2,6],[4,4,6],[8,4,6],[6,5,6],[11,8,6],[7,9,6],[8,9,6],[9,10,6],[4,2,7],[2,4,7],[1,5,7],[2,5,7],[1,6,7],[4,7,7],[4,8,7],[7,9,7],[9,2,8],[1,3,8],[4,3,8],[1,5,8],[3,5,8],[9,8,8],[11,8,8],[1,10,8],[1,2,9],[9,2,9],[1,5,9],[3,7,9],[5,8,9],[8,8,9],[6,9,9],[10,9,9],[3,11,9],[10,11,9],[10,10,10],[7,11,10],[9,11,10]],"ignoreExtent":false,"flags":3},"86":{"id":86,"type":"spheres","material":{},"vertices":[[7,4,1],[3,7,1],[5,10,1],[9,2,2],[5,4,2],[7,4,2],[5,5,2],[8,5,2],[4,8,2],[5,8,2],[6,10,2],[10,1,3],[5,3,4],[6,4,4],[4,5,4],[7,5,4],[8,7,4],[10,1,5],[5,2,5],[5,4,5],[4,5,5],[7,5,5],[6,9,5],[8,10,5],[6,2,6],[6,9,6],[10,9,6],[2,10,6],[1,11,6],[9,11,6],[5,2,7],[8,3,7],[1,7,7],[2,8,7],[11,8,7],[1,10,7],[2,2,8],[3,2,8],[3,3,8],[4,6,8],[6,9,8],[8,2,9],[5,4,9],[2,5,9],[1,6,9],[3,8,9],[11,8,9],[2,9,9],[3,10,9],[7,11,9],[1,4,10],[2,5,10],[1,9,10],[4,9,10],[10,9,10],[1,10,10],[5,11,10],[1,5,11],[1,9,11]],"colors":[[0.2862745,1,0,1]],"radii":[[0.1768507]],"centers":[[7,4,1],[3,7,1],[5,10,1],[9,2,2],[5,4,2],[7,4,2],[5,5,2],[8,5,2],[4,8,2],[5,8,2],[6,10,2],[10,1,3],[5,3,4],[6,4,4],[4,5,4],[7,5,4],[8,7,4],[10,1,5],[5,2,5],[5,4,5],[4,5,5],[7,5,5],[6,9,5],[8,10,5],[6,2,6],[6,9,6],[10,9,6],[2,10,6],[1,11,6],[9,11,6],[5,2,7],[8,3,7],[1,7,7],[2,8,7],[11,8,7],[1,10,7],[2,2,8],[3,2,8],[3,3,8],[4,6,8],[6,9,8],[8,2,9],[5,4,9],[2,5,9],[1,6,9],[3,8,9],[11,8,9],[2,9,9],[3,10,9],[7,11,9],[1,4,10],[2,5,10],[1,9,10],[4,9,10],[10,9,10],[1,10,10],[5,11,10],[1,5,11],[1,9,11]],"ignoreExtent":false,"flags":3},"87":{"id":87,"type":"spheres","material":{},"vertices":[[4,10,1],[5,11,1],[3,7,2],[9,7,2],[9,8,2],[1,11,2],[9,11,2],[4,3,3],[6,4,3],[6,7,3],[1,10,3],[10,1,4],[4,3,4],[9,3,4],[9,7,4],[8,8,4],[9,8,4],[6,9,4],[1,10,4],[9,1,5],[7,2,5],[6,5,5],[5,8,5],[7,9,5],[2,10,5],[9,10,5],[9,11,5],[7,3,6],[7,5,6],[5,7,6],[5,8,6],[9,8,6],[1,9,6],[4,1,7],[9,2,7],[7,3,7],[6,9,7],[10,10,7],[3,1,8],[9,1,8],[2,3,8],[1,4,8],[2,4,8],[4,7,8],[4,8,8],[7,8,8],[8,8,8],[2,10,8],[1,4,9],[4,5,9],[5,5,9],[1,7,9],[1,10,9],[8,11,9],[9,11,9],[1,2,10],[1,8,10],[10,8,10],[11,8,10],[3,9,10],[2,1,11],[1,6,11],[4,9,11],[5,9,11],[3,10,11]],"colors":[[0.1647059,1,0,1]],"radii":[[0.1910914]],"centers":[[4,10,1],[5,11,1],[3,7,2],[9,7,2],[9,8,2],[1,11,2],[9,11,2],[4,3,3],[6,4,3],[6,7,3],[1,10,3],[10,1,4],[4,3,4],[9,3,4],[9,7,4],[8,8,4],[9,8,4],[6,9,4],[1,10,4],[9,1,5],[7,2,5],[6,5,5],[5,8,5],[7,9,5],[2,10,5],[9,10,5],[9,11,5],[7,3,6],[7,5,6],[5,7,6],[5,8,6],[9,8,6],[1,9,6],[4,1,7],[9,2,7],[7,3,7],[6,9,7],[10,10,7],[3,1,8],[9,1,8],[2,3,8],[1,4,8],[2,4,8],[4,7,8],[4,8,8],[7,8,8],[8,8,8],[2,10,8],[1,4,9],[4,5,9],[5,5,9],[1,7,9],[1,10,9],[8,11,9],[9,11,9],[1,2,10],[1,8,10],[10,8,10],[11,8,10],[3,9,10],[2,1,11],[1,6,11],[4,9,11],[5,9,11],[3,10,11]],"ignoreExtent":false,"flags":3},"88":{"id":88,"type":"spheres","material":{},"vertices":[[6,7,1],[8,9,1],[9,3,2],[5,6,2],[6,7,2],[3,8,2],[3,11,2],[5,4,3],[1,11,3],[4,4,4],[10,8,4],[8,9,4],[1,11,4],[9,2,5],[4,3,5],[9,3,5],[9,8,5],[10,8,5],[8,9,5],[10,9,5],[11,10,5],[9,2,6],[1,8,6],[10,8,6],[7,4,7],[9,8,7],[11,10,7],[5,2,8],[6,2,8],[7,2,8],[9,3,8],[3,4,8],[5,6,8],[2,8,8],[10,10,8],[11,10,8],[1,3,9],[9,3,9],[7,4,9],[2,7,9],[2,10,9],[11,10,9],[5,7,10],[4,8,10],[11,9,10],[1,4,11],[1,8,11],[5,8,11],[11,9,11]],"colors":[[0.03921569,1,0,1]],"radii":[[0.2055114]],"centers":[[6,7,1],[8,9,1],[9,3,2],[5,6,2],[6,7,2],[3,8,2],[3,11,2],[5,4,3],[1,11,3],[4,4,4],[10,8,4],[8,9,4],[1,11,4],[9,2,5],[4,3,5],[9,3,5],[9,8,5],[10,8,5],[8,9,5],[10,9,5],[11,10,5],[9,2,6],[1,8,6],[10,8,6],[7,4,7],[9,8,7],[11,10,7],[5,2,8],[6,2,8],[7,2,8],[9,3,8],[3,4,8],[5,6,8],[2,8,8],[10,10,8],[11,10,8],[1,3,9],[9,3,9],[7,4,9],[2,7,9],[2,10,9],[11,10,9],[5,7,10],[4,8,10],[11,9,10],[1,4,11],[1,8,11],[5,8,11],[11,9,11]],"ignoreExtent":false,"flags":3},"89":{"id":89,"type":"spheres","material":{},"vertices":[[7,1,1],[6,4,1],[9,7,1],[10,9,1],[2,10,1],[4,11,1],[6,11,1],[8,1,2],[6,4,2],[6,8,2],[7,10,2],[5,5,3],[5,7,3],[7,10,3],[9,2,4],[5,4,4],[8,5,4],[8,6,4],[6,8,4],[5,5,5],[5,1,6],[8,1,6],[9,1,6],[1,10,6],[11,10,6],[9,1,7],[6,2,7],[5,6,7],[5,7,7],[8,1,8],[4,2,8],[8,4,8],[1,8,8],[5,8,8],[9,11,8],[10,11,8],[8,1,9],[2,4,9],[3,6,9],[4,8,9],[10,8,9],[10,10,9],[8,2,10],[5,3,10],[3,8,10],[2,10,10],[3,10,10],[11,10,10],[4,8,11],[11,8,11],[2,9,11],[3,9,11],[11,10,11],[4,11,11],[11,11,11]],"colors":[[0,1,0.08235294,1]],"radii":[[0.2201021]],"centers":[[7,1,1],[6,4,1],[9,7,1],[10,9,1],[2,10,1],[4,11,1],[6,11,1],[8,1,2],[6,4,2],[6,8,2],[7,10,2],[5,5,3],[5,7,3],[7,10,3],[9,2,4],[5,4,4],[8,5,4],[8,6,4],[6,8,4],[5,5,5],[5,1,6],[8,1,6],[9,1,6],[1,10,6],[11,10,6],[9,1,7],[6,2,7],[5,6,7],[5,7,7],[8,1,8],[4,2,8],[8,4,8],[1,8,8],[5,8,8],[9,11,8],[10,11,8],[8,1,9],[2,4,9],[3,6,9],[4,8,9],[10,8,9],[10,10,9],[8,2,10],[5,3,10],[3,8,10],[2,10,10],[3,10,10],[11,10,10],[4,8,11],[11,8,11],[2,9,11],[3,9,11],[11,10,11],[4,11,11],[11,11,11]],"ignoreExtent":false,"flags":3},"90":{"id":90,"type":"spheres","material":{},"vertices":[[9,3,1],[9,4,1],[9,6,1],[11,10,1],[9,4,2],[9,6,2],[9,9,2],[10,10,2],[9,4,3],[6,5,3],[8,5,3],[5,6,3],[8,7,3],[9,7,3],[5,8,3],[6,8,3],[8,8,3],[9,8,3],[6,9,3],[9,9,4],[2,10,4],[2,11,4],[5,7,5],[11,8,5],[1,9,5],[1,10,5],[10,10,5],[1,11,5],[10,1,6],[9,3,6],[10,10,6],[8,4,7],[5,8,7],[10,11,7],[10,8,8],[9,1,9],[2,2,9],[7,2,9],[2,3,9],[3,3,9],[4,4,9],[10,4,9],[3,5,9],[2,6,9],[4,7,9],[1,8,9],[6,8,9],[3,1,10],[9,2,10],[1,3,10],[8,3,10],[6,4,10],[2,6,10],[2,9,10],[4,11,10],[11,11,10]],"colors":[[0,1,0.2039216,1]],"radii":[[0.2348559]],"centers":[[9,3,1],[9,4,1],[9,6,1],[11,10,1],[9,4,2],[9,6,2],[9,9,2],[10,10,2],[9,4,3],[6,5,3],[8,5,3],[5,6,3],[8,7,3],[9,7,3],[5,8,3],[6,8,3],[8,8,3],[9,8,3],[6,9,3],[9,9,4],[2,10,4],[2,11,4],[5,7,5],[11,8,5],[1,9,5],[1,10,5],[10,10,5],[1,11,5],[10,1,6],[9,3,6],[10,10,6],[8,4,7],[5,8,7],[10,11,7],[10,8,8],[9,1,9],[2,2,9],[7,2,9],[2,3,9],[3,3,9],[4,4,9],[10,4,9],[3,5,9],[2,6,9],[4,7,9],[1,8,9],[6,8,9],[3,1,10],[9,2,10],[1,3,10],[8,3,10],[6,4,10],[2,6,10],[2,9,10],[4,11,10],[11,11,10]],"ignoreExtent":false,"flags":3},"91":{"id":91,"type":"spheres","material":{},"vertices":[[6,5,1],[9,5,1],[6,8,1],[8,8,1],[3,11,1],[10,1,2],[6,5,2],[8,6,2],[7,8,2],[8,8,2],[6,9,2],[11,1,3],[7,5,3],[6,6,3],[9,11,3],[11,1,4],[9,4,4],[6,5,4],[9,6,4],[6,7,4],[4,4,5],[6,7,5],[9,7,5],[10,7,5],[7,2,6],[7,2,7],[10,2,7],[7,5,7],[10,8,7],[10,2,8],[5,7,8],[3,4,9],[6,5,9],[5,6,9],[5,7,9],[11,11,9],[2,4,10],[7,4,10],[8,7,11],[3,8,11],[2,11,11]],"colors":[[0,1,0.3254902,1]],"radii":[[0.249766]],"centers":[[6,5,1],[9,5,1],[6,8,1],[8,8,1],[3,11,1],[10,1,2],[6,5,2],[8,6,2],[7,8,2],[8,8,2],[6,9,2],[11,1,3],[7,5,3],[6,6,3],[9,11,3],[11,1,4],[9,4,4],[6,5,4],[9,6,4],[6,7,4],[4,4,5],[6,7,5],[9,7,5],[10,7,5],[7,2,6],[7,2,7],[10,2,7],[7,5,7],[10,8,7],[10,2,8],[5,7,8],[3,4,9],[6,5,9],[5,6,9],[5,7,9],[11,11,9],[2,4,10],[7,4,10],[8,7,11],[3,8,11],[2,11,11]],"ignoreExtent":false,"flags":3},"92":{"id":92,"type":"spheres","material":{},"vertices":[[8,2,1],[8,5,1],[7,8,1],[7,10,1],[9,5,2],[6,6,2],[8,7,2],[8,9,2],[8,9,3],[5,5,4],[7,7,4],[10,7,4],[5,8,4],[9,10,4],[7,1,5],[10,2,5],[7,7,5],[8,7,5],[6,8,5],[10,2,6],[5,6,6],[7,7,6],[9,7,6],[10,7,6],[10,11,6],[10,1,7],[1,8,7],[10,3,8],[3,1,9],[3,2,9],[10,2,9],[4,3,9],[8,4,9],[11,4,9],[4,6,9],[8,1,10],[7,2,10],[11,4,10],[3,5,10],[7,7,10],[2,11,10],[3,1,11],[3,2,11],[3,3,11],[5,3,11],[11,4,11],[9,7,11],[10,7,11]],"colors":[[0,1,0.4470588,1]],"radii":[[0.2648259]],"centers":[[8,2,1],[8,5,1],[7,8,1],[7,10,1],[9,5,2],[6,6,2],[8,7,2],[8,9,2],[8,9,3],[5,5,4],[7,7,4],[10,7,4],[5,8,4],[9,10,4],[7,1,5],[10,2,5],[7,7,5],[8,7,5],[6,8,5],[10,2,6],[5,6,6],[7,7,6],[9,7,6],[10,7,6],[10,11,6],[10,1,7],[1,8,7],[10,3,8],[3,1,9],[3,2,9],[10,2,9],[4,3,9],[8,4,9],[11,4,9],[4,6,9],[8,1,10],[7,2,10],[11,4,10],[3,5,10],[7,7,10],[2,11,10],[3,1,11],[3,2,11],[3,3,11],[5,3,11],[11,4,11],[9,7,11],[10,7,11]],"ignoreExtent":false,"flags":3},"93":{"id":93,"type":"spheres","material":{},"vertices":[[9,2,1],[6,6,1],[11,11,1],[2,10,2],[10,2,3],[7,7,3],[2,10,3],[9,10,3],[7,9,4],[11,1,5],[8,8,5],[6,7,6],[6,8,6],[7,8,6],[5,1,7],[8,1,7],[9,3,7],[10,3,7],[11,7,7],[10,1,8],[10,4,8],[7,5,8],[5,2,9],[6,2,9],[10,3,9],[2,2,10],[3,6,10],[6,7,10],[1,2,11],[10,6,11],[11,7,11],[2,8,11]],"colors":[[0,1,0.572549,1]],"radii":[[0.28003]],"centers":[[9,2,1],[6,6,1],[11,11,1],[2,10,2],[10,2,3],[7,7,3],[2,10,3],[9,10,3],[7,9,4],[11,1,5],[8,8,5],[6,7,6],[6,8,6],[7,8,6],[5,1,7],[8,1,7],[9,3,7],[10,3,7],[11,7,7],[10,1,8],[10,4,8],[7,5,8],[5,2,9],[6,2,9],[10,3,9],[2,2,10],[3,6,10],[6,7,10],[1,2,11],[10,6,11],[11,7,11],[2,8,11]],"ignoreExtent":false,"flags":3},"94":{"id":94,"type":"spheres","material":{},"vertices":[[2,11,1],[11,1,2],[10,2,2],[7,5,2],[8,6,3],[9,9,3],[11,9,4],[9,4,5],[8,5,5],[10,11,5],[9,4,6],[8,5,6],[7,6,6],[8,7,6],[11,7,6],[7,8,7],[4,1,8],[11,4,8],[6,8,8],[9,4,9],[10,2,10],[5,4,10],[10,4,10],[5,5,10],[5,6,10],[2,7,10],[8,7,10],[2,8,10],[6,3,11],[3,4,11],[11,5,11]],"colors":[[0,1,0.6941177,1]],"radii":[[0.2953729]],"centers":[[2,11,1],[11,1,2],[10,2,2],[7,5,2],[8,6,3],[9,9,3],[11,9,4],[9,4,5],[8,5,5],[10,11,5],[9,4,6],[8,5,6],[7,6,6],[8,7,6],[11,7,6],[7,8,7],[4,1,8],[11,4,8],[6,8,8],[9,4,9],[10,2,10],[5,4,10],[10,4,10],[5,5,10],[5,6,10],[2,7,10],[8,7,10],[2,8,10],[6,3,11],[3,4,11],[11,5,11]],"ignoreExtent":false,"flags":3},"95":{"id":95,"type":"spheres","material":{},"vertices":[[10,8,1],[7,9,1],[7,7,2],[7,9,2],[10,11,2],[9,6,3],[2,11,3],[10,2,4],[7,6,4],[5,7,4],[7,8,4],[10,3,5],[7,8,5],[10,3,6],[8,8,6],[10,4,7],[7,7,7],[5,1,8],[11,2,8],[2,8,9],[3,3,10],[3,4,10],[4,5,10],[3,7,10],[4,7,10],[2,2,11],[2,4,11],[6,7,11],[7,7,11]],"colors":[[0,1,0.8156863,1]],"radii":[[0.3108499]],"centers":[[10,8,1],[7,9,1],[7,7,2],[7,9,2],[10,11,2],[9,6,3],[2,11,3],[10,2,4],[7,6,4],[5,7,4],[7,8,4],[10,3,5],[7,8,5],[10,3,6],[8,8,6],[10,4,7],[7,7,7],[5,1,8],[11,2,8],[2,8,9],[3,3,10],[3,4,10],[4,5,10],[3,7,10],[4,7,10],[2,2,11],[2,4,11],[6,7,11],[7,7,11]],"ignoreExtent":false,"flags":3},"96":{"id":96,"type":"spheres","material":{},"vertices":[[11,1,1],[6,10,1],[9,5,3],[7,8,3],[10,8,3],[7,9,3],[9,5,4],[6,6,4],[10,6,4],[9,11,4],[5,6,5],[7,6,5],[11,7,5],[11,11,5],[6,6,6],[9,7,7],[8,8,7],[9,4,8],[11,11,8],[10,1,9],[4,2,9],[11,2,9],[9,1,10],[6,2,10],[2,3,10],[9,3,10],[3,11,10],[4,3,11],[1,7,11],[5,7,11]],"colors":[[0,1,0.9372549,1]],"radii":[[0.3264565]],"centers":[[11,1,1],[6,10,1],[9,5,3],[7,8,3],[10,8,3],[7,9,3],[9,5,4],[6,6,4],[10,6,4],[9,11,4],[5,6,5],[7,6,5],[11,7,5],[11,11,5],[6,6,6],[9,7,7],[8,8,7],[9,4,8],[11,11,8],[10,1,9],[4,2,9],[11,2,9],[9,1,10],[6,2,10],[2,3,10],[9,3,10],[3,11,10],[4,3,11],[1,7,11],[5,7,11]],"ignoreExtent":false,"flags":3},"97":{"id":97,"type":"spheres","material":{},"vertices":[[7,5,1],[10,6,1],[2,11,2],[10,3,3],[10,9,4],[8,6,5],[6,1,6],[10,4,6],[8,6,6],[11,1,7],[11,2,7],[10,7,7],[11,3,8],[7,7,8],[4,1,9],[5,1,9],[10,3,10],[4,6,10],[8,1,11],[8,2,11],[1,3,11],[7,3,11],[4,4,11],[5,4,11],[10,5,11],[11,6,11]],"colors":[[0,0.9372549,1,1]],"radii":[[0.3421884]],"centers":[[7,5,1],[10,6,1],[2,11,2],[10,3,3],[10,9,4],[8,6,5],[6,1,6],[10,4,6],[8,6,6],[11,1,7],[11,2,7],[10,7,7],[11,3,8],[7,7,8],[4,1,9],[5,1,9],[10,3,10],[4,6,10],[8,1,11],[8,2,11],[1,3,11],[7,3,11],[4,4,11],[5,4,11],[10,5,11],[11,6,11]],"ignoreExtent":false,"flags":3},"98":{"id":98,"type":"spheres","material":{},"vertices":[[6,9,1],[10,3,2],[10,8,2],[5,6,4],[11,8,4],[6,6,5],[11,1,6],[8,5,7],[8,7,7],[8,7,8],[11,7,8],[11,3,9],[10,1,10],[9,2,11]],"colors":[[0,0.8156863,1,1]],"radii":[[0.3580418]],"centers":[[6,9,1],[10,3,2],[10,8,2],[5,6,4],[11,8,4],[6,6,5],[11,1,6],[8,5,7],[8,7,7],[8,7,8],[11,7,8],[11,3,9],[10,1,10],[9,2,11]],"ignoreExtent":false,"flags":3},"99":{"id":99,"type":"spheres","material":{},"vertices":[[8,6,1],[7,7,1],[11,10,2],[11,11,2],[10,3,4],[11,10,4],[9,6,5],[10,6,5],[11,4,7],[11,1,8],[8,5,8],[9,7,8],[7,1,9],[7,5,9],[10,5,9],[11,7,9],[3,2,10],[5,2,10],[9,7,10],[10,2,11],[6,4,11],[2,5,11],[3,11,11]],"colors":[[0,0.6941177,1,1]],"radii":[[0.3740132]],"centers":[[8,6,1],[7,7,1],[11,10,2],[11,11,2],[10,3,4],[11,10,4],[9,6,5],[10,6,5],[11,4,7],[11,1,8],[8,5,8],[9,7,8],[7,1,9],[7,5,9],[10,5,9],[11,7,9],[3,2,10],[5,2,10],[9,7,10],[10,2,11],[6,4,11],[2,5,11],[3,11,11]],"ignoreExtent":false,"flags":3},"100":{"id":100,"type":"spheres","material":{},"vertices":[[8,7,1],[10,9,2],[10,7,3],[10,10,3],[11,2,4],[10,10,4],[10,4,5],[7,1,6],[11,3,7],[11,5,7],[7,6,7],[6,7,7],[6,8,7],[7,1,8],[10,7,8],[7,7,9],[4,1,10],[5,1,10],[7,1,10],[8,4,10],[4,2,11],[2,3,11],[10,4,11],[4,7,11]],"colors":[[0,0.572549,1,1]],"radii":[[0.390099]],"centers":[[8,7,1],[10,9,2],[10,7,3],[10,10,3],[11,2,4],[10,10,4],[10,4,5],[7,1,6],[11,3,7],[11,5,7],[7,6,7],[6,7,7],[6,8,7],[7,1,8],[10,7,8],[7,7,9],[4,1,10],[5,1,10],[7,1,10],[8,4,10],[4,2,11],[2,3,11],[10,4,11],[4,7,11]],"ignoreExtent":false,"flags":3},"101":{"id":101,"type":"spheres","material":{},"vertices":[[10,5,1],[11,9,1],[10,6,3],[11,11,4],[11,2,5],[11,3,6],[10,6,6],[11,11,6],[9,4,7],[6,6,7],[10,5,8],[11,5,8],[6,6,8],[7,6,8],[6,7,8],[9,7,9],[10,7,9],[4,4,10],[10,5,10],[7,2,11]],"colors":[[0,0.4470588,1,1]],"radii":[[0.4062962]],"centers":[[10,5,1],[11,9,1],[10,6,3],[11,11,4],[11,2,5],[11,3,6],[10,6,6],[11,11,6],[9,4,7],[6,6,7],[10,5,8],[11,5,8],[6,6,8],[7,6,8],[6,7,8],[9,7,9],[10,7,9],[4,4,10],[10,5,10],[7,2,11]],"ignoreExtent":false,"flags":3},"102":{"id":102,"type":"spheres","material":{},"vertices":[[10,1,1],[10,7,1],[11,2,2],[10,6,2],[11,2,3],[7,6,3],[11,3,5],[9,5,5],[11,4,6],[10,5,6],[11,5,6],[10,5,7],[11,11,7],[6,1,9],[6,7,9],[8,7,9],[6,1,10],[4,3,10],[9,4,10],[6,5,10],[11,7,10],[4,5,11],[5,5,11],[4,6,11],[5,6,11]],"colors":[[0,0.3254902,1,1]],"radii":[[0.4226018]],"centers":[[10,1,1],[10,7,1],[11,2,2],[10,6,2],[11,2,3],[7,6,3],[11,3,5],[9,5,5],[11,4,6],[10,5,6],[11,5,6],[10,5,7],[11,11,7],[6,1,9],[6,7,9],[8,7,9],[6,1,10],[4,3,10],[9,4,10],[6,5,10],[11,7,10],[4,5,11],[5,5,11],[4,6,11],[5,6,11]],"ignoreExtent":false,"flags":3},"103":{"id":103,"type":"spheres","material":{},"vertices":[[10,4,1],[11,11,3],[11,3,4],[11,2,6],[9,6,6],[11,6,7],[11,5,9],[11,2,10],[11,3,10],[11,5,10],[10,7,10],[4,1,11],[5,2,11],[10,3,11],[3,5,11]],"colors":[[0,0.2039216,1,1]],"radii":[[0.4390129]],"centers":[[10,4,1],[11,11,3],[11,3,4],[11,2,6],[9,6,6],[11,6,7],[11,5,9],[11,2,10],[11,3,10],[11,5,10],[10,7,10],[4,1,11],[5,2,11],[10,3,11],[3,5,11]],"ignoreExtent":false,"flags":3},"104":{"id":104,"type":"spheres","material":{},"vertices":[[11,8,1],[10,4,2],[7,6,2],[10,4,4],[10,5,5],[9,5,6],[6,1,8],[11,1,9],[10,6,10],[9,1,11],[2,6,11],[6,6,11],[8,6,11]],"colors":[[0,0.08235294,1,1]],"radii":[[0.4555269]],"centers":[[11,8,1],[10,4,2],[7,6,2],[10,4,4],[10,5,5],[9,5,6],[6,1,8],[11,1,9],[10,6,10],[9,1,11],[2,6,11],[6,6,11],[8,6,11]],"ignoreExtent":false,"flags":3},"105":{"id":105,"type":"spheres","material":{},"vertices":[[10,2,1],[10,3,1],[10,7,2],[10,9,3],[11,10,3],[11,7,4],[10,11,4],[8,5,9],[4,2,10],[7,5,10],[6,6,10],[7,1,11],[8,3,11],[11,3,11],[9,6,11]],"colors":[[0.03921569,0,1,1]],"radii":[[0.4721413]],"centers":[[10,2,1],[10,3,1],[10,7,2],[10,9,3],[11,10,3],[11,7,4],[10,11,4],[8,5,9],[4,2,10],[7,5,10],[6,6,10],[7,1,11],[8,3,11],[11,3,11],[9,6,11]],"ignoreExtent":false,"flags":3},"106":{"id":106,"type":"spheres","material":{},"vertices":[[11,2,1],[11,3,2],[11,4,5],[11,6,6],[7,1,7]],"colors":[[0.1647059,0,1,1]],"radii":[[0.4888537]],"centers":[[11,2,1],[11,3,2],[11,4,5],[11,6,6],[7,1,7]],"ignoreExtent":false,"flags":3},"107":{"id":107,"type":"spheres","material":{},"vertices":[[10,11,3],[11,6,8],[5,1,11],[10,1,11],[11,2,11],[9,3,11],[7,4,11],[3,6,11]],"colors":[[0.2862745,0,1,1]],"radii":[[0.5056619]],"centers":[[10,11,3],[11,6,8],[5,1,11],[10,1,11],[11,2,11],[9,3,11],[7,4,11],[3,6,11]],"ignoreExtent":false,"flags":3},"108":{"id":108,"type":"spheres","material":{},"vertices":[[9,1,1],[11,3,1],[11,6,1],[11,7,1],[10,4,3],[11,4,4],[11,5,5],[11,6,5],[6,1,7],[9,5,9],[6,6,9],[6,2,11],[6,5,11]],"colors":[[0.4078431,0,1,1]],"radii":[[0.5225638]],"centers":[[9,1,1],[11,3,1],[11,6,1],[11,7,1],[10,4,3],[11,4,4],[11,5,5],[11,6,5],[6,1,7],[9,5,9],[6,6,9],[6,2,11],[6,5,11]],"ignoreExtent":false,"flags":3},"109":{"id":109,"type":"spheres","material":{},"vertices":[[7,6,1],[11,8,2],[11,9,2],[11,3,3],[11,6,3],[11,8,3],[10,5,4],[8,6,7],[7,6,9],[10,6,9],[11,6,9]],"colors":[[0.5294118,0,1,1]],"radii":[[0.5395573]],"centers":[[7,6,1],[11,8,2],[11,9,2],[11,3,3],[11,6,3],[11,8,3],[10,5,4],[8,6,7],[7,6,9],[10,6,9],[11,6,9]],"ignoreExtent":false,"flags":3},"110":{"id":110,"type":"spheres","material":{},"vertices":[[10,5,2],[11,9,3],[11,6,4],[7,6,10],[11,6,10],[9,5,11]],"colors":[[0.654902,0,1,1]],"radii":[[0.5566405]],"centers":[[10,5,2],[11,9,3],[11,6,4],[7,6,10],[11,6,10],[9,5,11]],"ignoreExtent":false,"flags":3},"111":{"id":111,"type":"spheres","material":{},"vertices":[[9,5,7],[10,6,7],[11,1,10],[7,6,11]],"colors":[[0.7764706,0,1,1]],"radii":[[0.5738115]],"centers":[[9,5,7],[10,6,7],[11,1,10],[7,6,11]],"ignoreExtent":false,"flags":3},"112":{"id":112,"type":"spheres","material":{},"vertices":[[11,5,1],[11,7,3],[9,5,8],[8,6,10],[2,7,11]],"colors":[[0.8980392,0,1,1]],"radii":[[0.5910686]],"centers":[[11,5,1],[11,7,3],[9,5,8],[8,6,10],[2,7,11]],"ignoreExtent":false,"flags":3},"113":{"id":113,"type":"spheres","material":{},"vertices":[[10,5,3],[8,5,10],[6,1,11],[3,7,11]],"colors":[[1,0,0.9803922,1]],"radii":[[0.6084101]],"centers":[[10,5,3],[8,5,10],[6,1,11],[3,7,11]],"ignoreExtent":false,"flags":3},"114":{"id":114,"type":"spheres","material":{},"vertices":[[8,6,8],[9,5,10],[7,5,11]],"colors":[[1,0,0.8588235,1]],"radii":[[0.6258343]],"centers":[[8,6,8],[9,5,10],[7,5,11]],"ignoreExtent":false,"flags":3},"115":{"id":115,"type":"spheres","material":{},"vertices":[[8,1,1],[11,5,4],[10,6,8],[9,6,9],[9,6,10]],"colors":[[1,0,0.7333333,1]],"radii":[[0.6433399]],"centers":[[8,1,1],[11,5,4],[10,6,8],[9,6,9],[9,6,10]],"ignoreExtent":false,"flags":3},"116":{"id":116,"type":"spheres","material":{},"vertices":[[11,4,1],[11,6,2],[9,6,7],[8,5,11]],"colors":[[1,0,0.6117647,1]],"radii":[[0.6609251]],"centers":[[11,4,1],[11,6,2],[9,6,7],[8,5,11]],"ignoreExtent":false,"flags":3},"117":{"id":117,"type":"spheres","material":{},"vertices":[[11,4,2],[11,4,3],[8,4,11],[9,4,11]],"colors":[[1,0,0.4901961,1]],"radii":[[0.6785887]],"centers":[[11,4,2],[11,4,3],[8,4,11],[9,4,11]],"ignoreExtent":false,"flags":3},"118":{"id":118,"type":"spheres","material":{},"vertices":[[8,6,9]],"colors":[[1,0,0.3686275,1]],"radii":[[0.6963294]],"centers":[[8,6,9]],"ignoreExtent":false,"flags":3},"119":{"id":119,"type":"spheres","material":{},"vertices":[[11,7,2],[11,5,3]],"colors":[[1,0,0.2431373,1]],"radii":[[0.7141455]],"centers":[[11,7,2],[11,5,3]],"ignoreExtent":false,"flags":3},"120":{"id":120,"type":"spheres","material":{},"vertices":[[11,5,2],[9,6,8],[11,1,11]],"colors":[[1,0,0.1215686,1]],"radii":[[0.7320362]],"centers":[[11,5,2],[9,6,8],[11,1,11]],"ignoreExtent":false,"flags":3},"121":{"id":121,"type":"text","material":{"lit":false},"vertices":[[6.194924,13.41045,13.66314]],"colors":[[0,0,1,1]],"texts":[["QVM Contingency"]],"cex":[[1]],"adj":[[0.5,0.5]],"centers":[[6.194924,13.41045,13.66314]],"family":[["sans"]],"font":[[1]],"ignoreExtent":true,"flags":4136},"122":{"id":122,"type":"text","material":{"lit":false},"vertices":[[6.194924,-1.636826,-1.592032]],"colors":[[0,0,1,1]],"texts":[["Quality.Rank"]],"cex":[[1]],"adj":[[0.5,0.5]],"centers":[[6.194924,-1.636826,-1.592032]],"family":[["sans"]],"font":[[1]],"ignoreExtent":true,"flags":4136},"123":{"id":123,"type":"text","material":{"lit":false},"vertices":[[-1.21927,5.886813,-1.592032]],"colors":[[0,0,1,1]],"texts":[["Value.Rank"]],"cex":[[1]],"adj":[[0.5,0.5]],"centers":[[-1.21927,5.886813,-1.592032]],"family":[["sans"]],"font":[[1]],"ignoreExtent":true,"flags":4136},"124":{"id":124,"type":"text","material":{"lit":false},"vertices":[[-1.21927,-1.636826,6.035556]],"colors":[[0,0,1,1]],"texts":[["Momentum.Rank"]],"cex":[[1]],"adj":[[0.5,0.5]],"centers":[[-1.21927,-1.636826,6.035556]],"family":[["sans"]],"font":[[1]],"ignoreExtent":true,"flags":4136},"125":{"id":125,"type":"lines","material":{"lit":false},"vertices":[[1,0.09939835,0.1681804],[11,0.09939835,0.1681804],[1,0.09939835,0.1681804],[1,-0.1899724,-0.1251883],[2,0.09939835,0.1681804],[2,-0.1899724,-0.1251883],[3,0.09939835,0.1681804],[3,-0.1899724,-0.1251883],[4,0.09939835,0.1681804],[4,-0.1899724,-0.1251883],[6,0.09939835,0.1681804],[6,-0.1899724,-0.1251883],[7,0.09939835,0.1681804],[7,-0.1899724,-0.1251883],[8,0.09939835,0.1681804],[8,-0.1899724,-0.1251883],[9,0.09939835,0.1681804],[9,-0.1899724,-0.1251883],[11,0.09939835,0.1681804],[11,-0.1899724,-0.1251883]],"colors":[[0,0,0,1]],"centers":[[6,0.09939835,0.1681804],[1,-0.04528701,0.02149604],[2,-0.04528701,0.02149604],[3,-0.04528701,0.02149604],[4,-0.04528701,0.02149604],[6,-0.04528701,0.02149604],[7,-0.04528701,0.02149604],[8,-0.04528701,0.02149604],[9,-0.04528701,0.02149604],[11,-0.04528701,0.02149604]],"ignoreExtent":true,"flags":128},"126":{"id":126,"type":"text","material":{"lit":false},"vertices":[[1,-0.7687138,-0.7119259],[2,-0.7687138,-0.7119259],[3,-0.7687138,-0.7119259],[4,-0.7687138,-0.7119259],[6,-0.7687138,-0.7119259],[7,-0.7687138,-0.7119259],[8,-0.7687138,-0.7119259],[9,-0.7687138,-0.7119259],[11,-0.7687138,-0.7119259]],"colors":[[0,0,0,1]],"texts":[["(19,30]"],["(30,40]"],["(40,47]"],["(47,54]"],["(60,66]"],["(66,72]"],["(72,78]"],["(78,84]"],["(89,95]"]],"cex":[[1]],"adj":[[0.5,0.5]],"centers":[[1,-0.7687138,-0.7119259],[2,-0.7687138,-0.7119259],[3,-0.7687138,-0.7119259],[4,-0.7687138,-0.7119259],[6,-0.7687138,-0.7119259],[7,-0.7687138,-0.7119259],[8,-0.7687138,-0.7119259],[9,-0.7687138,-0.7119259],[11,-0.7687138,-0.7119259]],"family":[["sans"]],"font":[[1]],"ignoreExtent":true,"flags":4136},"127":{"id":127,"type":"lines","material":{"lit":false},"vertices":[[0.4916983,1,0.1681804],[0.4916983,11,0.1681804],[0.4916983,1,0.1681804],[0.206537,1,-0.1251883],[0.4916983,2,0.1681804],[0.206537,2,-0.1251883],[0.4916983,3,0.1681804],[0.206537,3,-0.1251883],[0.4916983,4,0.1681804],[0.206537,4,-0.1251883],[0.4916983,6,0.1681804],[0.206537,6,-0.1251883],[0.4916983,7,0.1681804],[0.206537,7,-0.1251883],[0.4916983,8,0.1681804],[0.206537,8,-0.1251883],[0.4916983,9,0.1681804],[0.206537,9,-0.1251883],[0.4916983,11,0.1681804],[0.206537,11,-0.1251883]],"colors":[[0,0,0,1]],"centers":[[0.4916983,6,0.1681804],[0.3491176,1,0.02149604],[0.3491176,2,0.02149604],[0.3491176,3,0.02149604],[0.3491176,4,0.02149604],[0.3491176,6,0.02149604],[0.3491176,7,0.02149604],[0.3491176,8,0.02149604],[0.3491176,9,0.02149604],[0.3491176,11,0.02149604]],"ignoreExtent":true,"flags":128},"128":{"id":128,"type":"text","material":{"lit":false},"vertices":[[-0.3637856,1,-0.7119259],[-0.3637856,2,-0.7119259],[-0.3637856,3,-0.7119259],[-0.3637856,4,-0.7119259],[-0.3637856,6,-0.7119259],[-0.3637856,7,-0.7119259],[-0.3637856,8,-0.7119259],[-0.3637856,9,-0.7119259],[-0.3637856,11,-0.7119259]],"colors":[[0,0,0,1]],"texts":[["(10,17]"],["(17,24]"],["(24,31]"],["(31,38]"],["(45,52]"],["(52,59]"],["(59,66]"],["(66,73]"],["(80,88]"]],"cex":[[1]],"adj":[[0.5,0.5]],"centers":[[-0.3637856,1,-0.7119259],[-0.3637856,2,-0.7119259],[-0.3637856,3,-0.7119259],[-0.3637856,4,-0.7119259],[-0.3637856,6,-0.7119259],[-0.3637856,7,-0.7119259],[-0.3637856,8,-0.7119259],[-0.3637856,9,-0.7119259],[-0.3637856,11,-0.7119259]],"family":[["sans"]],"font":[[1]],"ignoreExtent":true,"flags":4136},"129":{"id":129,"type":"lines","material":{"lit":false},"vertices":[[0.4916983,0.09939835,1],[0.4916983,0.09939835,11],[0.4916983,0.09939835,1],[0.206537,-0.1899724,1],[0.4916983,0.09939835,2],[0.206537,-0.1899724,2],[0.4916983,0.09939835,3],[0.206537,-0.1899724,3],[0.4916983,0.09939835,4],[0.206537,-0.1899724,4],[0.4916983,0.09939835,6],[0.206537,-0.1899724,6],[0.4916983,0.09939835,7],[0.206537,-0.1899724,7],[0.4916983,0.09939835,8],[0.206537,-0.1899724,8],[0.4916983,0.09939835,9],[0.206537,-0.1899724,9],[0.4916983,0.09939835,11],[0.206537,-0.1899724,11]],"colors":[[0,0,0,1]],"centers":[[0.4916983,0.09939835,6],[0.3491176,-0.04528701,1],[0.3491176,-0.04528701,2],[0.3491176,-0.04528701,3],[0.3491176,-0.04528701,4],[0.3491176,-0.04528701,6],[0.3491176,-0.04528701,7],[0.3491176,-0.04528701,8],[0.3491176,-0.04528701,9],[0.3491176,-0.04528701,11]],"ignoreExtent":true,"flags":128},"130":{"id":130,"type":"text","material":{"lit":false},"vertices":[[-0.3637856,-0.7687138,1],[-0.3637856,-0.7687138,2],[-0.3637856,-0.7687138,3],[-0.3637856,-0.7687138,4],[-0.3637856,-0.7687138,6],[-0.3637856,-0.7687138,7],[-0.3637856,-0.7687138,8],[-0.3637856,-0.7687138,9],[-0.3637856,-0.7687138,11]],"colors":[[0,0,0,1]],"texts":[["(17,29]"],["(29,38]"],["(38,46]"],["(46,53]"],["(61,68]"],["(68,74]"],["(74,80]"],["(80,85]"],["(90,95]"]],"cex":[[1]],"adj":[[0.5,0.5]],"centers":[[-0.3637856,-0.7687138,1],[-0.3637856,-0.7687138,2],[-0.3637856,-0.7687138,3],[-0.3637856,-0.7687138,4],[-0.3637856,-0.7687138,6],[-0.3637856,-0.7687138,7],[-0.3637856,-0.7687138,8],[-0.3637856,-0.7687138,9],[-0.3637856,-0.7687138,11]],"family":[["sans"]],"font":[[1]],"ignoreExtent":true,"flags":4136},"70":{"id":70,"type":"light","vertices":[[0,0,1]],"colors":[[1,1,1,1],[1,1,1,1],[1,1,1,1]],"viewpoint":true,"finite":false},"69":{"id":69,"type":"background","material":{"fog":true},"colors":[[0.2980392,0.2980392,0.2980392,1]],"centers":[[0,0,0]],"sphere":false,"fogtype":"none","flags":0},"71":{"id":71,"type":"background","material":{"lit":false,"back":"lines"},"colors":[[1,1,1,1]],"centers":[[0,0,0]],"sphere":false,"fogtype":"none","flags":0},"66":{"id":66,"type":"subscene","par3d":{"antialias":0,"FOV":30,"ignoreExtent":false,"listeners":66,"mouseMode":{"left":"trackball","right":"zoom","middle":"zoom","wheel":"push"},"observer":[0,0,37.59544],"modelMatrix":[[1,0,0,-6.194924],[0,0.3420202,0.9396926,-7.684976],[0,-0.9396926,0.3420202,-34.12792],[0,0,0,1]],"projMatrix":[[3.732051,0,0,0],[0,3.732051,0,0],[0,0,-3.863703,-135.5272],[0,0,-1,0]],"skipRedraw":false,"userMatrix":[[1,0,0,0],[0,0.3420201,0.9396926,0],[0,-0.9396926,0.3420201,0],[0,0,0,1]],"scale":[1,1,1],"viewport":{"x":0,"y":0,"width":1,"height":1},"zoom":1,"bbox":[0.6578116,11.73204,0.2679638,11.50566,0.3390749,11.73204],"windowRect":[268,291,524,547],"family":"sans","font":1,"cex":1,"useFreeType":false,"fontname":"TT Arial","maxClipPlanes":6,"glVersion":1.1},"embeddings":{"viewport":"replace","projection":"replace","model":"replace"},"objects":[71,72,73,74,75,76,77,78,79,80,81,82,83,84,85,86,87,88,89,90,91,92,93,94,95,96,97,98,99,100,101,102,103,104,105,106,107,108,109,110,111,112,113,114,115,116,117,118,119,120,121,122,123,124,125,126,127,128,129,130,70],"subscenes":[],"flags":5291}},"snapshot":"","width":1000,"height":1000,"sphereVerts":{"vb":[[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0.07465783,0.1464466,0.2126075,0.2705981,0.3181896,0.3535534,0.3753303,0.3826834,0.3753303,0.3535534,0.3181896,0.2705981,0.2126075,0.1464466,0.07465783,0,0,0.1379497,0.2705981,0.3928475,0.5,0.5879378,0.6532815,0.6935199,0.7071068,0.6935199,0.6532815,0.5879378,0.5,0.3928475,0.2705981,0.1379497,0,0,0.18024,0.3535534,0.51328,0.6532815,0.7681778,0.8535534,0.9061274,0.9238795,0.9061274,0.8535534,0.7681778,0.6532815,0.51328,0.3535534,0.18024,0,0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1,0.9807853,0.9238795,0.8314696,0.7071068,0.5555702,0.3826834,0.1950903,0,0,0.18024,0.3535534,0.51328,0.6532815,0.7681778,0.8535534,0.9061274,0.9238795,0.9061274,0.8535534,0.7681778,0.6532815,0.51328,0.3535534,0.18024,0,0,0.1379497,0.2705981,0.3928475,0.5,0.5879378,0.6532815,0.6935199,0.7071068,0.6935199,0.6532815,0.5879378,0.5,0.3928475,0.2705981,0.1379497,0,0,0.07465783,0.1464466,0.2126075,0.2705981,0.3181896,0.3535534,0.3753303,0.3826834,0.3753303,0.3535534,0.3181896,0.2705981,0.2126075,0.1464466,0.07465783,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,-0,-0.07465783,-0.1464466,-0.2126075,-0.2705981,-0.3181896,-0.3535534,-0.3753303,-0.3826834,-0.3753303,-0.3535534,-0.3181896,-0.2705981,-0.2126075,-0.1464466,-0.07465783,-0,-0,-0.1379497,-0.2705981,-0.3928475,-0.5,-0.5879378,-0.6532815,-0.6935199,-0.7071068,-0.6935199,-0.6532815,-0.5879378,-0.5,-0.3928475,-0.2705981,-0.1379497,-0,-0,-0.18024,-0.3535534,-0.51328,-0.6532815,-0.7681778,-0.8535534,-0.9061274,-0.9238795,-0.9061274,-0.8535534,-0.7681778,-0.6532815,-0.51328,-0.3535534,-0.18024,-0,-0,-0.1950903,-0.3826834,-0.5555702,-0.7071068,-0.8314696,-0.9238795,-0.9807853,-1,-0.9807853,-0.9238795,-0.8314696,-0.7071068,-0.5555702,-0.3826834,-0.1950903,-0,-0,-0.18024,-0.3535534,-0.51328,-0.6532815,-0.7681778,-0.8535534,-0.9061274,-0.9238795,-0.9061274,-0.8535534,-0.7681778,-0.6532815,-0.51328,-0.3535534,-0.18024,-0,-0,-0.1379497,-0.2705981,-0.3928475,-0.5,-0.5879378,-0.6532815,-0.6935199,-0.7071068,-0.6935199,-0.6532815,-0.5879378,-0.5,-0.3928475,-0.2705981,-0.1379497,-0,-0,-0.07465783,-0.1464466,-0.2126075,-0.2705981,-0.3181896,-0.3535534,-0.3753303,-0.3826834,-0.3753303,-0.3535534,-0.3181896,-0.2705981,-0.2126075,-0.1464466,-0.07465783,-0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],[-1,-0.9807853,-0.9238795,-0.8314696,-0.7071068,-0.5555702,-0.3826834,-0.1950903,0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1,-1,-0.9807853,-0.9238795,-0.8314696,-0.7071068,-0.5555702,-0.3826834,-0.1950903,0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1,-1,-0.9807853,-0.9238795,-0.8314696,-0.7071068,-0.5555702,-0.3826834,-0.1950903,0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1,-1,-0.9807853,-0.9238795,-0.8314696,-0.7071068,-0.5555702,-0.3826834,-0.1950903,0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1,-1,-0.9807853,-0.9238795,-0.8314696,-0.7071068,-0.5555702,-0.3826834,-0.1950903,0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1,-1,-0.9807853,-0.9238795,-0.8314696,-0.7071068,-0.5555702,-0.3826834,-0.1950903,0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1,-1,-0.9807853,-0.9238795,-0.8314696,-0.7071068,-0.5555702,-0.3826834,-0.1950903,0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1,-1,-0.9807853,-0.9238795,-0.8314696,-0.7071068,-0.5555702,-0.3826834,-0.1950903,0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1,-1,-0.9807853,-0.9238795,-0.8314696,-0.7071068,-0.5555702,-0.3826834,-0.1950903,0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1,-1,-0.9807853,-0.9238795,-0.8314696,-0.7071068,-0.5555702,-0.3826834,-0.1950903,0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1,-1,-0.9807853,-0.9238795,-0.8314696,-0.7071068,-0.5555702,-0.3826834,-0.1950903,0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1,-1,-0.9807853,-0.9238795,-0.8314696,-0.7071068,-0.5555702,-0.3826834,-0.1950903,0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1,-1,-0.9807853,-0.9238795,-0.8314696,-0.7071068,-0.5555702,-0.3826834,-0.1950903,0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1,-1,-0.9807853,-0.9238795,-0.8314696,-0.7071068,-0.5555702,-0.3826834,-0.1950903,0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1,-1,-0.9807853,-0.9238795,-0.8314696,-0.7071068,-0.5555702,-0.3826834,-0.1950903,0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1,-1,-0.9807853,-0.9238795,-0.8314696,-0.7071068,-0.5555702,-0.3826834,-0.1950903,0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1,-1,-0.9807853,-0.9238795,-0.8314696,-0.7071068,-0.5555702,-0.3826834,-0.1950903,0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1],[0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1,0.9807853,0.9238795,0.8314696,0.7071068,0.5555702,0.3826834,0.1950903,0,0,0.18024,0.3535534,0.51328,0.6532815,0.7681778,0.8535534,0.9061274,0.9238795,0.9061274,0.8535534,0.7681778,0.6532815,0.51328,0.3535534,0.18024,0,0,0.1379497,0.2705981,0.3928475,0.5,0.5879378,0.6532815,0.6935199,0.7071068,0.6935199,0.6532815,0.5879378,0.5,0.3928475,0.2705981,0.1379497,0,0,0.07465783,0.1464466,0.2126075,0.2705981,0.3181896,0.3535534,0.3753303,0.3826834,0.3753303,0.3535534,0.3181896,0.2705981,0.2126075,0.1464466,0.07465783,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,-0,-0.07465783,-0.1464466,-0.2126075,-0.2705981,-0.3181896,-0.3535534,-0.3753303,-0.3826834,-0.3753303,-0.3535534,-0.3181896,-0.2705981,-0.2126075,-0.1464466,-0.07465783,-0,-0,-0.1379497,-0.2705981,-0.3928475,-0.5,-0.5879378,-0.6532815,-0.6935199,-0.7071068,-0.6935199,-0.6532815,-0.5879378,-0.5,-0.3928475,-0.2705981,-0.1379497,-0,-0,-0.18024,-0.3535534,-0.51328,-0.6532815,-0.7681778,-0.8535534,-0.9061274,-0.9238795,-0.9061274,-0.8535534,-0.7681778,-0.6532815,-0.51328,-0.3535534,-0.18024,-0,-0,-0.1950903,-0.3826834,-0.5555702,-0.7071068,-0.8314696,-0.9238795,-0.9807853,-1,-0.9807853,-0.9238795,-0.8314696,-0.7071068,-0.5555702,-0.3826834,-0.1950903,-0,-0,-0.18024,-0.3535534,-0.51328,-0.6532815,-0.7681778,-0.8535534,-0.9061274,-0.9238795,-0.9061274,-0.8535534,-0.7681778,-0.6532815,-0.51328,-0.3535534,-0.18024,-0,-0,-0.1379497,-0.2705981,-0.3928475,-0.5,-0.5879378,-0.6532815,-0.6935199,-0.7071068,-0.6935199,-0.6532815,-0.5879378,-0.5,-0.3928475,-0.2705981,-0.1379497,-0,-0,-0.07465783,-0.1464466,-0.2126075,-0.2705981,-0.3181896,-0.3535534,-0.3753303,-0.3826834,-0.3753303,-0.3535534,-0.3181896,-0.2705981,-0.2126075,-0.1464466,-0.07465783,-0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0.07465783,0.1464466,0.2126075,0.2705981,0.3181896,0.3535534,0.3753303,0.3826834,0.3753303,0.3535534,0.3181896,0.2705981,0.2126075,0.1464466,0.07465783,0,0,0.1379497,0.2705981,0.3928475,0.5,0.5879378,0.6532815,0.6935199,0.7071068,0.6935199,0.6532815,0.5879378,0.5,0.3928475,0.2705981,0.1379497,0,0,0.18024,0.3535534,0.51328,0.6532815,0.7681778,0.8535534,0.9061274,0.9238795,0.9061274,0.8535534,0.7681778,0.6532815,0.51328,0.3535534,0.18024,0,0,0.1950903,0.3826834,0.5555702,0.7071068,0.8314696,0.9238795,0.9807853,1,0.9807853,0.9238795,0.8314696,0.7071068,0.5555702,0.3826834,0.1950903,0]],"it":[[0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,17,18,19,20,21,22,23,24,25,26,27,28,29,30,31,32,34,35,36,37,38,39,40,41,42,43,44,45,46,47,48,49,51,52,53,54,55,56,57,58,59,60,61,62,63,64,65,66,68,69,70,71,72,73,74,75,76,77,78,79,80,81,82,83,85,86,87,88,89,90,91,92,93,94,95,96,97,98,99,100,102,103,104,105,106,107,108,109,110,111,112,113,114,115,116,117,119,120,121,122,123,124,125,126,127,128,129,130,131,132,133,134,136,137,138,139,140,141,142,143,144,145,146,147,148,149,150,151,153,154,155,156,157,158,159,160,161,162,163,164,165,166,167,168,170,171,172,173,174,175,176,177,178,179,180,181,182,183,184,185,187,188,189,190,191,192,193,194,195,196,197,198,199,200,201,202,204,205,206,207,208,209,210,211,212,213,214,215,216,217,218,219,221,222,223,224,225,226,227,228,229,230,231,232,233,234,235,236,238,239,240,241,242,243,244,245,246,247,248,249,250,251,252,253,255,256,257,258,259,260,261,262,263,264,265,266,267,268,269,270,0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,17,18,19,20,21,22,23,24,25,26,27,28,29,30,31,32,34,35,36,37,38,39,40,41,42,43,44,45,46,47,48,49,51,52,53,54,55,56,57,58,59,60,61,62,63,64,65,66,68,69,70,71,72,73,74,75,76,77,78,79,80,81,82,83,85,86,87,88,89,90,91,92,93,94,95,96,97,98,99,100,102,103,104,105,106,107,108,109,110,111,112,113,114,115,116,117,119,120,121,122,123,124,125,126,127,128,129,130,131,132,133,134,136,137,138,139,140,141,142,143,144,145,146,147,148,149,150,151,153,154,155,156,157,158,159,160,161,162,163,164,165,166,167,168,170,171,172,173,174,175,176,177,178,179,180,181,182,183,184,185,187,188,189,190,191,192,193,194,195,196,197,198,199,200,201,202,204,205,206,207,208,209,210,211,212,213,214,215,216,217,218,219,221,222,223,224,225,226,227,228,229,230,231,232,233,234,235,236,238,239,240,241,242,243,244,245,246,247,248,249,250,251,252,253,255,256,257,258,259,260,261,262,263,264,265,266,267,268,269,270],[17,18,19,20,21,22,23,24,25,26,27,28,29,30,31,32,34,35,36,37,38,39,40,41,42,43,44,45,46,47,48,49,51,52,53,54,55,56,57,58,59,60,61,62,63,64,65,66,68,69,70,71,72,73,74,75,76,77,78,79,80,81,82,83,85,86,87,88,89,90,91,92,93,94,95,96,97,98,99,100,102,103,104,105,106,107,108,109,110,111,112,113,114,115,116,117,119,120,121,122,123,124,125,126,127,128,129,130,131,132,133,134,136,137,138,139,140,141,142,143,144,145,146,147,148,149,150,151,153,154,155,156,157,158,159,160,161,162,163,164,165,166,167,168,170,171,172,173,174,175,176,177,178,179,180,181,182,183,184,185,187,188,189,190,191,192,193,194,195,196,197,198,199,200,201,202,204,205,206,207,208,209,210,211,212,213,214,215,216,217,218,219,221,222,223,224,225,226,227,228,229,230,231,232,233,234,235,236,238,239,240,241,242,243,244,245,246,247,248,249,250,251,252,253,255,256,257,258,259,260,261,262,263,264,265,266,267,268,269,270,272,273,274,275,276,277,278,279,280,281,282,283,284,285,286,287,18,19,20,21,22,23,24,25,26,27,28,29,30,31,32,33,35,36,37,38,39,40,41,42,43,44,45,46,47,48,49,50,52,53,54,55,56,57,58,59,60,61,62,63,64,65,66,67,69,70,71,72,73,74,75,76,77,78,79,80,81,82,83,84,86,87,88,89,90,91,92,93,94,95,96,97,98,99,100,101,103,104,105,106,107,108,109,110,111,112,113,114,115,116,117,118,120,121,122,123,124,125,126,127,128,129,130,131,132,133,134,135,137,138,139,140,141,142,143,144,145,146,147,148,149,150,151,152,154,155,156,157,158,159,160,161,162,163,164,165,166,167,168,169,171,172,173,174,175,176,177,178,179,180,181,182,183,184,185,186,188,189,190,191,192,193,194,195,196,197,198,199,200,201,202,203,205,206,207,208,209,210,211,212,213,214,215,216,217,218,219,220,222,223,224,225,226,227,228,229,230,231,232,233,234,235,236,237,239,240,241,242,243,244,245,246,247,248,249,250,251,252,253,254,256,257,258,259,260,261,262,263,264,265,266,267,268,269,270,271,273,274,275,276,277,278,279,280,281,282,283,284,285,286,287,288],[18,19,20,21,22,23,24,25,26,27,28,29,30,31,32,33,35,36,37,38,39,40,41,42,43,44,45,46,47,48,49,50,52,53,54,55,56,57,58,59,60,61,62,63,64,65,66,67,69,70,71,72,73,74,75,76,77,78,79,80,81,82,83,84,86,87,88,89,90,91,92,93,94,95,96,97,98,99,100,101,103,104,105,106,107,108,109,110,111,112,113,114,115,116,117,118,120,121,122,123,124,125,126,127,128,129,130,131,132,133,134,135,137,138,139,140,141,142,143,144,145,146,147,148,149,150,151,152,154,155,156,157,158,159,160,161,162,163,164,165,166,167,168,169,171,172,173,174,175,176,177,178,179,180,181,182,183,184,185,186,188,189,190,191,192,193,194,195,196,197,198,199,200,201,202,203,205,206,207,208,209,210,211,212,213,214,215,216,217,218,219,220,222,223,224,225,226,227,228,229,230,231,232,233,234,235,236,237,239,240,241,242,243,244,245,246,247,248,249,250,251,252,253,254,256,257,258,259,260,261,262,263,264,265,266,267,268,269,270,271,273,274,275,276,277,278,279,280,281,282,283,284,285,286,287,288,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,18,19,20,21,22,23,24,25,26,27,28,29,30,31,32,33,35,36,37,38,39,40,41,42,43,44,45,46,47,48,49,50,52,53,54,55,56,57,58,59,60,61,62,63,64,65,66,67,69,70,71,72,73,74,75,76,77,78,79,80,81,82,83,84,86,87,88,89,90,91,92,93,94,95,96,97,98,99,100,101,103,104,105,106,107,108,109,110,111,112,113,114,115,116,117,118,120,121,122,123,124,125,126,127,128,129,130,131,132,133,134,135,137,138,139,140,141,142,143,144,145,146,147,148,149,150,151,152,154,155,156,157,158,159,160,161,162,163,164,165,166,167,168,169,171,172,173,174,175,176,177,178,179,180,181,182,183,184,185,186,188,189,190,191,192,193,194,195,196,197,198,199,200,201,202,203,205,206,207,208,209,210,211,212,213,214,215,216,217,218,219,220,222,223,224,225,226,227,228,229,230,231,232,233,234,235,236,237,239,240,241,242,243,244,245,246,247,248,249,250,251,252,253,254,256,257,258,259,260,261,262,263,264,265,266,267,268,269,270,271]],"primitivetype":"triangle","material":null,"normals":null,"texcoords":[[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0.0625,0.0625,0.0625,0.0625,0.0625,0.0625,0.0625,0.0625,0.0625,0.0625,0.0625,0.0625,0.0625,0.0625,0.0625,0.0625,0.0625,0.125,0.125,0.125,0.125,0.125,0.125,0.125,0.125,0.125,0.125,0.125,0.125,0.125,0.125,0.125,0.125,0.125,0.1875,0.1875,0.1875,0.1875,0.1875,0.1875,0.1875,0.1875,0.1875,0.1875,0.1875,0.1875,0.1875,0.1875,0.1875,0.1875,0.1875,0.25,0.25,0.25,0.25,0.25,0.25,0.25,0.25,0.25,0.25,0.25,0.25,0.25,0.25,0.25,0.25,0.25,0.3125,0.3125,0.3125,0.3125,0.3125,0.3125,0.3125,0.3125,0.3125,0.3125,0.3125,0.3125,0.3125,0.3125,0.3125,0.3125,0.3125,0.375,0.375,0.375,0.375,0.375,0.375,0.375,0.375,0.375,0.375,0.375,0.375,0.375,0.375,0.375,0.375,0.375,0.4375,0.4375,0.4375,0.4375,0.4375,0.4375,0.4375,0.4375,0.4375,0.4375,0.4375,0.4375,0.4375,0.4375,0.4375,0.4375,0.4375,0.5,0.5,0.5,0.5,0.5,0.5,0.5,0.5,0.5,0.5,0.5,0.5,0.5,0.5,0.5,0.5,0.5,0.5625,0.5625,0.5625,0.5625,0.5625,0.5625,0.5625,0.5625,0.5625,0.5625,0.5625,0.5625,0.5625,0.5625,0.5625,0.5625,0.5625,0.625,0.625,0.625,0.625,0.625,0.625,0.625,0.625,0.625,0.625,0.625,0.625,0.625,0.625,0.625,0.625,0.625,0.6875,0.6875,0.6875,0.6875,0.6875,0.6875,0.6875,0.6875,0.6875,0.6875,0.6875,0.6875,0.6875,0.6875,0.6875,0.6875,0.6875,0.75,0.75,0.75,0.75,0.75,0.75,0.75,0.75,0.75,0.75,0.75,0.75,0.75,0.75,0.75,0.75,0.75,0.8125,0.8125,0.8125,0.8125,0.8125,0.8125,0.8125,0.8125,0.8125,0.8125,0.8125,0.8125,0.8125,0.8125,0.8125,0.8125,0.8125,0.875,0.875,0.875,0.875,0.875,0.875,0.875,0.875,0.875,0.875,0.875,0.875,0.875,0.875,0.875,0.875,0.875,0.9375,0.9375,0.9375,0.9375,0.9375,0.9375,0.9375,0.9375,0.9375,0.9375,0.9375,0.9375,0.9375,0.9375,0.9375,0.9375,0.9375,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1],[0,0.0625,0.125,0.1875,0.25,0.3125,0.375,0.4375,0.5,0.5625,0.625,0.6875,0.75,0.8125,0.875,0.9375,1,0,0.0625,0.125,0.1875,0.25,0.3125,0.375,0.4375,0.5,0.5625,0.625,0.6875,0.75,0.8125,0.875,0.9375,1,0,0.0625,0.125,0.1875,0.25,0.3125,0.375,0.4375,0.5,0.5625,0.625,0.6875,0.75,0.8125,0.875,0.9375,1,0,0.0625,0.125,0.1875,0.25,0.3125,0.375,0.4375,0.5,0.5625,0.625,0.6875,0.75,0.8125,0.875,0.9375,1,0,0.0625,0.125,0.1875,0.25,0.3125,0.375,0.4375,0.5,0.5625,0.625,0.6875,0.75,0.8125,0.875,0.9375,1,0,0.0625,0.125,0.1875,0.25,0.3125,0.375,0.4375,0.5,0.5625,0.625,0.6875,0.75,0.8125,0.875,0.9375,1,0,0.0625,0.125,0.1875,0.25,0.3125,0.375,0.4375,0.5,0.5625,0.625,0.6875,0.75,0.8125,0.875,0.9375,1,0,0.0625,0.125,0.1875,0.25,0.3125,0.375,0.4375,0.5,0.5625,0.625,0.6875,0.75,0.8125,0.875,0.9375,1,0,0.0625,0.125,0.1875,0.25,0.3125,0.375,0.4375,0.5,0.5625,0.625,0.6875,0.75,0.8125,0.875,0.9375,1,0,0.0625,0.125,0.1875,0.25,0.3125,0.375,0.4375,0.5,0.5625,0.625,0.6875,0.75,0.8125,0.875,0.9375,1,0,0.0625,0.125,0.1875,0.25,0.3125,0.375,0.4375,0.5,0.5625,0.625,0.6875,0.75,0.8125,0.875,0.9375,1,0,0.0625,0.125,0.1875,0.25,0.3125,0.375,0.4375,0.5,0.5625,0.625,0.6875,0.75,0.8125,0.875,0.9375,1,0,0.0625,0.125,0.1875,0.25,0.3125,0.375,0.4375,0.5,0.5625,0.625,0.6875,0.75,0.8125,0.875,0.9375,1,0,0.0625,0.125,0.1875,0.25,0.3125,0.375,0.4375,0.5,0.5625,0.625,0.6875,0.75,0.8125,0.875,0.9375,1,0,0.0625,0.125,0.1875,0.25,0.3125,0.375,0.4375,0.5,0.5625,0.625,0.6875,0.75,0.8125,0.875,0.9375,1,0,0.0625,0.125,0.1875,0.25,0.3125,0.375,0.4375,0.5,0.5625,0.625,0.6875,0.75,0.8125,0.875,0.9375,1,0,0.0625,0.125,0.1875,0.25,0.3125,0.375,0.4375,0.5,0.5625,0.625,0.6875,0.75,0.8125,0.875,0.9375,1]]}});
rgl.prefix = "";
</script>
<p id="debug">
You must enable Javascript to view this page properly.</p>
</div>
<br>Drag mouse to rotate model. Use mouse wheel or middle button
to zoom it.
<hr>
<br>
Object written from rgl 0.98.1 by writeWebGL.
</body>
</html>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment