!function(a){var b=/iPhone/i,c=/iPod/i,d=/iPad/i,e=/(?=.*\bAndroid\b)(?=.*\bMobile\b)/i,f=/Android/i,g=/(?=.*\bAndroid\b)(?=.*\bSD4930UR\b)/i,h=/(?=.*\bAndroid\b)(?=.*\b(?:KFOT|KFTT|KFJWI|KFJWA|KFSOWI|KFTHWI|KFTHWA|KFAPWI|KFAPWA|KFARWI|KFASWI|KFSAWI|KFSAWA)\b)/i,i=/IEMobile/i,j=/(?=.*\bWindows\b)(?=.*\bARM\b)/i,k=/BlackBerry/i,l=/BB10/i,m=/Opera Mini/i,n=/(CriOS|Chrome)(?=.*\bMobile\b)/i,o=/(?=.*\bFirefox\b)(?=.*\bMobile\b)/i,p=new RegExp("(?:Nexus 7|BNTV250|Kindle Fire|Silk|GT-P1000)","i"),q=function(a,b){return a.test(b)},r=function(a){var r=a||navigator.userAgent,s=r.split("[FBAN");return"undefined"!=typeof s[1]&&(r=s[0]),s=r.split("Twitter"),"undefined"!=typeof s[1]&&(r=s[0]),this.apple={phone:q(b,r),ipod:q(c,r),tablet:!q(b,r)&&q(d,r),device:q(b,r)||q(c,r)||q(d,r)},this.amazon={phone:q(g,r),tablet:!q(g,r)&&q(h,r),device:q(g,r)||q(h,r)},this.android={phone:q(g,r)||q(e,r),tablet:!q(g,r)&&!q(e,r)&&(q(h,r)||q(f,r)),device:q(g,r)||q(h,r)||q(e,r)||q(f,r)},this.windows={phone:q(i,r),tablet:q(j,r),device:q(i,r)||q(j,r)},this.other={blackberry:q(k,r),blackberry10:q(l,r),opera:q(m,r),firefox:q(o,r),chrome:q(n,r),device:q(k,r)||q(l,r)||q(m,r)||q(o,r)||q(n,r)},this.seven_inch=q(p,r),this.any=this.apple.device||this.android.device||this.windows.device||this.other.device||this.seven_inch,this.phone=this.apple.phone||this.android.phone||this.windows.phone,this.tablet=this.apple.tablet||this.android.tablet||this.windows.tablet,"undefined"==typeof window?this:void 0},s=function(){var a=new r;return a.Class=r,a};"undefined"!=typeof module&&module.exports&&"undefined"==typeof window?module.exports=r:"undefined"!=typeof module&&module.exports&&"undefined"!=typeof window?module.exports=s():"function"==typeof define&&define.amd?define("isMobile",[],a.isMobile=s()):a.isMobile=s()}(this);
// import { BufferGeometry } from '../../core/BufferGeometry';
// import { Vector2 } from '../../math/Vector2';
// import { Vector3 } from '../../math/Vector3';
// import { BufferAttribute } from '../../core/BufferAttribute';
/**
* @author Mugen87 / https://github.com/Mugen87
*/
function PRingBufferGeometry( innerRadius, outerRadius, thetaSegments, phiSegments, thetaStart, thetaLength ) {
THREE.BufferGeometry.call( this );
this.type = 'PRingBufferGeometry';
this.parameters = {
innerRadius: innerRadius,
outerRadius: outerRadius,
thetaSegments: thetaSegments,
phiSegments: phiSegments,
thetaStart: thetaStart,
thetaLength: thetaLength
};
innerRadius = innerRadius || 20;
outerRadius = outerRadius || 50;
thetaStart = thetaStart !== undefined ? thetaStart : 0;
thetaLength = thetaLength !== undefined ? thetaLength : Math.PI * 2;
thetaSegments = thetaSegments !== undefined ? Math.max( 3, thetaSegments ) : 8;
phiSegments = phiSegments !== undefined ? Math.max( 1, phiSegments ) : 1;
// these are used to calculate buffer length
var vertexCount = ( thetaSegments + 1 ) * ( phiSegments + 1 );
var indexCount = thetaSegments * phiSegments * 2 * 3;
// buffers
var indices = new THREE.BufferAttribute( new ( indexCount > 65535 ? Uint32Array : Uint16Array )( indexCount ) , 1 );
var vertices = new THREE.BufferAttribute( new Float32Array( vertexCount * 3 ), 3 );
var normals = new THREE.BufferAttribute( new Float32Array( vertexCount * 3 ), 3 );
var uvs = new THREE.BufferAttribute( new Float32Array( vertexCount * 2 ), 2 );
// some helper variables
var index = 0, indexOffset = 0, segment;
var radius = innerRadius;
var radiusStep = ( ( outerRadius - innerRadius ) / phiSegments );
var vertex = new THREE.Vector3();
var uv = new THREE.Vector2();
var j, i;
// generate vertices, normals and uvs
// values are generate from the inside of the ring to the outside
var phiStep = 1/phiSegments;
var thetaStep = 1/thetaSegments;
for ( j = 0; j <= phiSegments; j ++ ) {
for ( i = 0; i <= thetaSegments; i ++ ) {
segment = thetaStart + i / thetaSegments * thetaLength;
// vertex
vertex.x = radius * Math.cos( segment );
vertex.y = radius * Math.sin( segment );
vertices.setXYZ( index, vertex.x, vertex.y, vertex.z );
// normal
normals.setXYZ( index, 0, 0, 1 );
// uv
// uv.x = ( vertex.x / outerRadius + 1 ) / 2;
// uv.y = ( vertex.y / outerRadius + 1 ) / 2;
uv.x = i*thetaStep;
uv.y = j*phiStep;
uvs.setXY( index, uv.x, uv.y );
// increase index
index++;
}
// increase the radius for next row of vertices
radius += radiusStep;
}
// generate indices
for ( j = 0; j < phiSegments; j ++ ) {
var thetaSegmentLevel = j * ( thetaSegments + 1 );
for ( i = 0; i < thetaSegments; i ++ ) {
segment = i + thetaSegmentLevel;
// indices
var a = segment;
var b = segment + thetaSegments + 1;
var c = segment + thetaSegments + 2;
var d = segment + 1;
// face one
indices.setX( indexOffset, a ); indexOffset++;
indices.setX( indexOffset, b ); indexOffset++;
indices.setX( indexOffset, c ); indexOffset++;
// face two
indices.setX( indexOffset, a ); indexOffset++;
indices.setX( indexOffset, c ); indexOffset++;
indices.setX( indexOffset, d ); indexOffset++;
}
}
// build geometry
this.setIndex( indices );
this.addAttribute( 'position', vertices );
this.addAttribute( 'normal', normals );
this.addAttribute( 'uv', uvs );
}
PRingBufferGeometry.prototype = Object.create( THREE.BufferGeometry.prototype );
PRingBufferGeometry.prototype.constructor = PRingBufferGeometry;
THREE.PRingBufferGeometry = PRingBufferGeometry;
// export { RingBufferGeometry };
/**
* @author alteredq / http://alteredqualia.com/
*
* Full-screen textured quad shader
*/
THREE.CopyShader = {
uniforms: {
"tDiffuse": { value: null },
"opacity": { value: 1.0 }
},
vertexShader: [
"varying vec2 vUv;",
"void main() {",
"vUv = uv;",
"gl_Position = projectionMatrix * modelViewMatrix * vec4( position, 1.0 );",
"}"
].join( "\n" ),
fragmentShader: [
"uniform float opacity;",
"uniform sampler2D tDiffuse;",
"varying vec2 vUv;",
"void main() {",
"vec4 texel = texture2D( tDiffuse, vUv );",
"gl_FragColor = opacity * texel;",
"}"
].join( "\n" )
};
/**
* @author felixturner / http://airtight.cc/
*
* RGB Shift Shader
* Shifts red and blue channels from center in opposite directions
* Ported from http://kriss.cx/tom/2009/05/rgb-shift/
* by Tom Butterworth / http://kriss.cx/tom/
*
* amount: shift distance (1 is width of input)
* angle: shift angle in radians
*/
THREE.RGBShiftShader = {
uniforms: {
"tDiffuse": { value: null },
"amount": { value: 0.005 },
"angle": { value: 0.0 }
},
vertexShader: [
"varying vec2 vUv;",
"void main() {",
"vUv = uv;",
"gl_Position = projectionMatrix * modelViewMatrix * vec4( position, 1.0 );",
"}"
].join( "\n" ),
fragmentShader: [
"uniform sampler2D tDiffuse;",
"uniform float amount;",
"uniform float angle;",
"varying vec2 vUv;",
"void main() {",
// "vec2 offset = amount * vec2( cos(angle), sin(angle));",
"vec2 offset = amount * vec2( vUv.x - .5, vUv.y - .5 );",
"vec4 cr = texture2D(tDiffuse, vUv + offset);",
"vec4 cga = texture2D(tDiffuse, vUv);",
"vec4 cb = texture2D(tDiffuse, vUv - offset);",
"gl_FragColor = vec4(cr.r, cga.g, cb.b, cga.a);",
"}"
].join( "\n" )
};
THREE.AlphaColorShader = {
uniforms: {
color: { value: new THREE.Color( 0xffffff ) },
fogType: { value: 1 },
fogNear: { value: 10.0 },
fogFar: { value: 30.0 }
},
vertexShader: [
"attribute float alpha;",
"varying float vAlpha;",
"void main() {",
"vAlpha = alpha;",
"vec4 mvPosition = modelViewMatrix * vec4( position, 1.0 );",
"gl_PointSize = 4.0;",
"gl_Position = projectionMatrix * mvPosition;",
"}",
].join( "\n" ),
fragmentShader: [
// "uniform vec3 color;",
// "varying float vAlpha;",
// "void main() {",
// "gl_FragColor = vec4( color, vAlpha * ( gl_FragCoord.z ) );",
// "}"
"uniform vec3 color;",
"uniform int fogType;",
"uniform float fogNear;",
"uniform float fogFar;",
"varying float vAlpha;",
"void main() {",
// vec4 texture = texture2D( map, vUV );
// if ( texture.a < alphaTest ) discard;
// gl_FragColor = vec4( color * texture.xyz, texture.a * opacity );
"vec3 fogColor = vec3(0,0,0);",
"gl_FragColor = vec4( color, vAlpha * ( gl_FragCoord.z ) );",
"if ( fogType > 0 ) {",
"float depth = gl_FragCoord.z / gl_FragCoord.w;",
"float fogFactor = 0.0;",
"if ( fogType == 1 ) {",
"fogFactor = smoothstep( fogNear, fogFar, depth );",
"} else {",
// "const float LOG2 = 1.442695;",
// "float fogFactor = exp2( - fogDensity * fogDensity * depth * depth * LOG2 );",
// "fogFactor = 1.0 - clamp( fogFactor, 0.0, 1.0 );",
"}",
"gl_FragColor = mix( gl_FragColor, vec4( fogColor, gl_FragColor.w ), fogFactor );",
"}",
"}"
].join( "\n" ),
transparent: true
};
/**
* @author alteredq / http://alteredqualia.com/
*
* Simple fake tilt-shift effect, modulating two pass Gaussian blur (see above) by vertical position
*
* - 9 samples per pass
* - standard deviation 2.7
* - "h" and "v" parameters should be set to "1 / width" and "1 / height"
* - "r" parameter control where "focused" horizontal line lies
*/
THREE.VerticalTiltShiftShader = {
uniforms: {
"tDiffuse": { value: null },
"v": { value: 1.0 / 512.0 },
"r": { value: 0.35 }
},
vertexShader: [
"\
varying vec2 vUv;\
\
void main() {\
\
vUv = uv;\
gl_Position = projectionMatrix * modelViewMatrix * vec4( position, 1.0 );\
\
}"
].join( "\n" ),
fragmentShader: [
"uniform sampler2D tDiffuse;\
uniform float v;\
uniform float r;\
\
varying vec2 vUv;\
\
void main() {\
\
vec4 sum = vec4( 0.0 );\
\
float vv = v * abs( vUv.y - 0.5 ) * 1.5;\
\
sum += texture2D( tDiffuse, vec2( vUv.x, vUv.y - 4.0 * vv ) ) * 0.051;\
sum += texture2D( tDiffuse, vec2( vUv.x, vUv.y - 3.0 * vv ) ) * 0.0918;\
sum += texture2D( tDiffuse, vec2( vUv.x, vUv.y - 2.0 * vv ) ) * 0.12245;\
sum += texture2D( tDiffuse, vec2( vUv.x, vUv.y - 1.0 * vv ) ) * 0.1531;\
sum += texture2D( tDiffuse, vec2( vUv.x, vUv.y ) ) * 0.1633;\
sum += texture2D( tDiffuse, vec2( vUv.x, vUv.y + 1.0 * vv ) ) * 0.1531;\
sum += texture2D( tDiffuse, vec2( vUv.x, vUv.y + 2.0 * vv ) ) * 0.12245;\
sum += texture2D( tDiffuse, vec2( vUv.x, vUv.y + 3.0 * vv ) ) * 0.0918;\
sum += texture2D( tDiffuse, vec2( vUv.x, vUv.y + 4.0 * vv ) ) * 0.051;\
\
gl_FragColor = sum;\
\
}"
].join( "\n" )
};
/**
* @author alteredq / http://alteredqualia.com/
* @author davidedc / http://www.sketchpatch.net/
*
* NVIDIA FXAA by Timothy Lottes
* http://timothylottes.blogspot.com/2011/06/fxaa3-source-released.html
* - WebGL port by @supereggbert
* http://www.glge.org/demos/fxaa/
*/
THREE.FXAAShader = {
uniforms: {
"tDiffuse": { value: null },
"resolution": { value: new THREE.Vector2( 1 / 1024, 1 / 512 ) }
},
vertexShader: [
"void main() {",
"gl_Position = projectionMatrix * modelViewMatrix * vec4( position, 1.0 );",
"}"
].join( "\n" ),
fragmentShader: [
"uniform sampler2D tDiffuse;",
"uniform vec2 resolution;",
"#define FXAA_REDUCE_MIN (1.0/128.0)",
"#define FXAA_REDUCE_MUL (1.0/8.0)",
"#define FXAA_SPAN_MAX 8.0",
"void main() {",
"vec3 rgbNW = texture2D( tDiffuse, ( gl_FragCoord.xy + vec2( -1.0, -1.0 ) ) * resolution ).xyz;",
"vec3 rgbNE = texture2D( tDiffuse, ( gl_FragCoord.xy + vec2( 1.0, -1.0 ) ) * resolution ).xyz;",
"vec3 rgbSW = texture2D( tDiffuse, ( gl_FragCoord.xy + vec2( -1.0, 1.0 ) ) * resolution ).xyz;",
"vec3 rgbSE = texture2D( tDiffuse, ( gl_FragCoord.xy + vec2( 1.0, 1.0 ) ) * resolution ).xyz;",
"vec4 rgbaM = texture2D( tDiffuse, gl_FragCoord.xy * resolution );",
"vec3 rgbM = rgbaM.xyz;",
"vec3 luma = vec3( 0.299, 0.587, 0.114 );",
"float lumaNW = dot( rgbNW, luma );",
"float lumaNE = dot( rgbNE, luma );",
"float lumaSW = dot( rgbSW, luma );",
"float lumaSE = dot( rgbSE, luma );",
"float lumaM = dot( rgbM, luma );",
"float lumaMin = min( lumaM, min( min( lumaNW, lumaNE ), min( lumaSW, lumaSE ) ) );",
"float lumaMax = max( lumaM, max( max( lumaNW, lumaNE) , max( lumaSW, lumaSE ) ) );",
"vec2 dir;",
"dir.x = -((lumaNW + lumaNE) - (lumaSW + lumaSE));",
"dir.y = ((lumaNW + lumaSW) - (lumaNE + lumaSE));",
"float dirReduce = max( ( lumaNW + lumaNE + lumaSW + lumaSE ) * ( 0.25 * FXAA_REDUCE_MUL ), FXAA_REDUCE_MIN );",
"float rcpDirMin = 1.0 / ( min( abs( dir.x ), abs( dir.y ) ) + dirReduce );",
"dir = min( vec2( FXAA_SPAN_MAX, FXAA_SPAN_MAX),",
"max( vec2(-FXAA_SPAN_MAX, -FXAA_SPAN_MAX),",
"dir * rcpDirMin)) * resolution;",
"vec4 rgbA = (1.0/2.0) * (",
"texture2D(tDiffuse, gl_FragCoord.xy * resolution + dir * (1.0/3.0 - 0.5)) +",
"texture2D(tDiffuse, gl_FragCoord.xy * resolution + dir * (2.0/3.0 - 0.5)));",
"vec4 rgbB = rgbA * (1.0/2.0) + (1.0/4.0) * (",
"texture2D(tDiffuse, gl_FragCoord.xy * resolution + dir * (0.0/3.0 - 0.5)) +",
"texture2D(tDiffuse, gl_FragCoord.xy * resolution + dir * (3.0/3.0 - 0.5)));",
"float lumaB = dot(rgbB, vec4(luma, 0.0));",
"if ( ( lumaB < lumaMin ) || ( lumaB > lumaMax ) ) {",
"gl_FragColor = rgbA;",
"} else {",
"gl_FragColor = rgbB;",
"}",
"}"
].join( "\n" )
};
/**
* @author alteredq / http://alteredqualia.com/
*/
THREE.EffectComposer = function ( renderer, renderTarget ) {
this.renderer = renderer;
if ( renderTarget === undefined ) {
var parameters = {
minFilter: THREE.LinearFilter,
magFilter: THREE.LinearFilter,
format: THREE.RGBAFormat,
stencilBuffer: false
};
var size = renderer.getSize();
renderTarget = new THREE.WebGLRenderTarget( size.width, size.height, parameters );
}
this.renderTarget1 = renderTarget;
this.renderTarget2 = renderTarget.clone();
this.writeBuffer = this.renderTarget1;
this.readBuffer = this.renderTarget2;
this.passes = [];
if ( THREE.CopyShader === undefined )
console.error( "THREE.EffectComposer relies on THREE.CopyShader" );
this.copyPass = new THREE.ShaderPass( THREE.CopyShader );
};
Object.assign( THREE.EffectComposer.prototype, {
swapBuffers: function() {
var tmp = this.readBuffer;
this.readBuffer = this.writeBuffer;
this.writeBuffer = tmp;
},
addPass: function ( pass ) {
this.passes.push( pass );
var size = this.renderer.getSize();
pass.setSize( size.width, size.height );
},
insertPass: function ( pass, index ) {
this.passes.splice( index, 0, pass );
},
render: function ( delta ) {
var maskActive = false;
var pass, i, il = this.passes.length;
for ( i = 0; i < il; i ++ ) {
pass = this.passes[ i ];
if ( pass.enabled === false ) continue;
pass.render( this.renderer, this.writeBuffer, this.readBuffer, delta, maskActive );
if ( pass.needsSwap ) {
if ( maskActive ) {
var context = this.renderer.context;
context.stencilFunc( context.NOTEQUAL, 1, 0xffffffff );
this.copyPass.render( this.renderer, this.writeBuffer, this.readBuffer, delta );
context.stencilFunc( context.EQUAL, 1, 0xffffffff );
}
this.swapBuffers();
}
if ( THREE.MaskPass !== undefined ) {
if ( pass instanceof THREE.MaskPass ) {
maskActive = true;
} else if ( pass instanceof THREE.ClearMaskPass ) {
maskActive = false;
}
}
}
},
reset: function ( renderTarget ) {
if ( renderTarget === undefined ) {
var size = this.renderer.getSize();
renderTarget = this.renderTarget1.clone();
renderTarget.setSize( size.width, size.height );
}
this.renderTarget1.dispose();
this.renderTarget2.dispose();
this.renderTarget1 = renderTarget;
this.renderTarget2 = renderTarget.clone();
this.writeBuffer = this.renderTarget1;
this.readBuffer = this.renderTarget2;
},
setSize: function ( width, height ) {
this.renderTarget1.setSize( width, height );
this.renderTarget2.setSize( width, height );
for ( var i = 0; i < this.passes.length; i ++ ) {
this.passes[i].setSize( width, height );
}
}
} );
THREE.Pass = function () {
// if set to true, the pass is processed by the composer
this.enabled = true;
// if set to true, the pass indicates to swap read and write buffer after rendering
this.needsSwap = true;
// if set to true, the pass clears its buffer before rendering
this.clear = false;
// if set to true, the result of the pass is rendered to screen
this.renderToScreen = false;
};
Object.assign( THREE.Pass.prototype, {
setSize: function( width, height ) {},
render: function ( renderer, writeBuffer, readBuffer, delta, maskActive ) {
console.error( "THREE.Pass: .render() must be implemented in derived pass." );
}
} );
/**
* @author alteredq / http://alteredqualia.com/
*/
THREE.RenderPass = function ( scene, camera, overrideMaterial, clearColor, clearAlpha ) {
THREE.Pass.call( this );
this.scene = scene;
this.camera = camera;
this.overrideMaterial = overrideMaterial;
this.clearColor = clearColor;
this.clearAlpha = ( clearAlpha !== undefined ) ? clearAlpha : 0;
this.clear = true;
this.needsSwap = false;
};
THREE.RenderPass.prototype = Object.assign( Object.create( THREE.Pass.prototype ), {
constructor: THREE.RenderPass,
render: function ( renderer, writeBuffer, readBuffer, delta, maskActive ) {
var oldAutoClear = renderer.autoClear;
renderer.autoClear = false;
this.scene.overrideMaterial = this.overrideMaterial;
var oldClearColor, oldClearAlpha;
if ( this.clearColor ) {
oldClearColor = renderer.getClearColor().getHex();
oldClearAlpha = renderer.getClearAlpha();
renderer.setClearColor( this.clearColor, this.clearAlpha );
}
renderer.render( this.scene, this.camera, this.renderToScreen ? null : readBuffer, this.clear );
if ( this.clearColor ) {
renderer.setClearColor( oldClearColor, oldClearAlpha );
}
this.scene.overrideMaterial = null;
renderer.autoClear = oldAutoClear;
}
} );
/**
* @author alteredq / http://alteredqualia.com/
*/
THREE.MaskPass = function ( scene, camera ) {
THREE.Pass.call( this );
this.scene = scene;
this.camera = camera;
this.clear = true;
this.needsSwap = false;
this.inverse = false;
};
THREE.MaskPass.prototype = Object.assign( Object.create( THREE.Pass.prototype ), {
constructor: THREE.MaskPass,
render: function ( renderer, writeBuffer, readBuffer, delta, maskActive ) {
var context = renderer.context;
var state = renderer.state;
// don't update color or depth
state.buffers.color.setMask( false );
state.buffers.depth.setMask( false );
// lock buffers
state.buffers.color.setLocked( true );
state.buffers.depth.setLocked( true );
// set up stencil
var writeValue, clearValue;
if ( this.inverse ) {
writeValue = 0;
clearValue = 1;
} else {
writeValue = 1;
clearValue = 0;
}
state.buffers.stencil.setTest( true );
state.buffers.stencil.setOp( context.REPLACE, context.REPLACE, context.REPLACE );
state.buffers.stencil.setFunc( context.ALWAYS, writeValue, 0xffffffff );
state.buffers.stencil.setClear( clearValue );
// draw into the stencil buffer
renderer.render( this.scene, this.camera, readBuffer, this.clear );
renderer.render( this.scene, this.camera, writeBuffer, this.clear );
// unlock color and depth buffer for subsequent rendering
state.buffers.color.setLocked( false );
state.buffers.depth.setLocked( false );
// only render where stencil is set to 1
state.buffers.stencil.setFunc( context.EQUAL, 1, 0xffffffff ); // draw if == 1
state.buffers.stencil.setOp( context.KEEP, context.KEEP, context.KEEP );
}
} );
THREE.ClearMaskPass = function () {
THREE.Pass.call( this );
this.needsSwap = false;
};
THREE.ClearMaskPass.prototype = Object.create( THREE.Pass.prototype );
Object.assign( THREE.ClearMaskPass.prototype, {
render: function ( renderer, writeBuffer, readBuffer, delta, maskActive ) {
renderer.state.buffers.stencil.setTest( false );
}
} );
/**
* @author alteredq / http://alteredqualia.com/
*/
THREE.ShaderPass = function ( shader, textureID ) {
THREE.Pass.call( this );
this.textureID = ( textureID !== undefined ) ? textureID : "tDiffuse";
if ( shader instanceof THREE.ShaderMaterial ) {
this.uniforms = shader.uniforms;
this.material = shader;
} else if ( shader ) {
this.uniforms = THREE.UniformsUtils.clone( shader.uniforms );
this.material = new THREE.ShaderMaterial( {
defines: shader.defines || {},
uniforms: this.uniforms,
vertexShader: shader.vertexShader,
fragmentShader: shader.fragmentShader
} );
}
this.camera = new THREE.OrthographicCamera( - 1, 1, 1, - 1, 0, 1 );
this.scene = new THREE.Scene();
this.quad = new THREE.Mesh( new THREE.PlaneBufferGeometry( 2, 2 ), null );
this.scene.add( this.quad );
};
THREE.ShaderPass.prototype = Object.assign( Object.create( THREE.Pass.prototype ), {
constructor: THREE.ShaderPass,
render: function( renderer, writeBuffer, readBuffer, delta, maskActive ) {
if ( this.uniforms[ this.textureID ] ) {
this.uniforms[ this.textureID ].value = readBuffer.texture;
}
this.quad.material = this.material;
if ( this.renderToScreen ) {
renderer.render( this.scene, this.camera );
} else {
renderer.render( this.scene, this.camera, writeBuffer, this.clear );
}
}
} );
/**
* @author mpk / http://polko.me/
*/
THREE.SMAAPass = function ( width, height ) {
THREE.Pass.call( this );
// render targets
this.edgesRT = new THREE.WebGLRenderTarget( width, height, {
depthBuffer: false,
stencilBuffer: false,
generateMipmaps: false,
minFilter: THREE.LinearFilter,
format: THREE.RGBFormat
} );
this.weightsRT = new THREE.WebGLRenderTarget( width, height, {
depthBuffer: false,
stencilBuffer: false,
generateMipmaps: false,
minFilter: THREE.LinearFilter,
format: THREE.RGBAFormat
} );
// textures
var areaTextureImage = new Image();
areaTextureImage.src = this.getAreaTexture();
this.areaTexture = new THREE.Texture();
this.areaTexture.image = areaTextureImage;
this.areaTexture.format = THREE.RGBFormat;
this.areaTexture.minFilter = THREE.LinearFilter;
this.areaTexture.generateMipmaps = false;
this.areaTexture.needsUpdate = true;
this.areaTexture.flipY = false;
var searchTextureImage = new Image();
searchTextureImage.src = this.getSearchTexture();
this.searchTexture = new THREE.Texture();
this.searchTexture.image = searchTextureImage;
this.searchTexture.magFilter = THREE.NearestFilter;
this.searchTexture.minFilter = THREE.NearestFilter;
this.searchTexture.generateMipmaps = false;
this.searchTexture.needsUpdate = true;
this.searchTexture.flipY = false;
// materials - pass 1
if ( THREE.SMAAShader === undefined ) {
console.error( "THREE.SMAAPass relies on THREE.SMAAShader" );
}
this.uniformsEdges = THREE.UniformsUtils.clone( THREE.SMAAShader[0].uniforms );
this.uniformsEdges[ "resolution" ].value.set( 1 / width, 1 / height );
this.materialEdges = new THREE.ShaderMaterial( {
defines: THREE.SMAAShader[0].defines,
uniforms: this.uniformsEdges,
vertexShader: THREE.SMAAShader[0].vertexShader,
fragmentShader: THREE.SMAAShader[0].fragmentShader
} );
// materials - pass 2
this.uniformsWeights = THREE.UniformsUtils.clone( THREE.SMAAShader[1].uniforms );
this.uniformsWeights[ "resolution" ].value.set( 1 / width, 1 / height );
this.uniformsWeights[ "tDiffuse" ].value = this.edgesRT.texture;
this.uniformsWeights[ "tArea" ].value = this.areaTexture;
this.uniformsWeights[ "tSearch" ].value = this.searchTexture;
this.materialWeights = new THREE.ShaderMaterial( {
defines: THREE.SMAAShader[1].defines,
uniforms: this.uniformsWeights,
vertexShader: THREE.SMAAShader[1].vertexShader,
fragmentShader: THREE.SMAAShader[1].fragmentShader
} );
// materials - pass 3
this.uniformsBlend = THREE.UniformsUtils.clone( THREE.SMAAShader[2].uniforms );
this.uniformsBlend[ "resolution" ].value.set( 1 / width, 1 / height );
this.uniformsBlend[ "tDiffuse" ].value = this.weightsRT.texture;
this.materialBlend = new THREE.ShaderMaterial( {
uniforms: this.uniformsBlend,
vertexShader: THREE.SMAAShader[2].vertexShader,
fragmentShader: THREE.SMAAShader[2].fragmentShader
} );
this.needsSwap = false;
this.camera = new THREE.OrthographicCamera( -1, 1, 1, -1, 0, 1 );
this.scene = new THREE.Scene();
this.quad = new THREE.Mesh( new THREE.PlaneBufferGeometry( 2, 2 ), null );
this.scene.add( this.quad );
};
THREE.SMAAPass.prototype = Object.assign( Object.create( THREE.Pass.prototype ), {
constructor: THREE.SMAAPass,
render: function ( renderer, writeBuffer, readBuffer, delta, maskActive ) {
// pass 1
this.uniformsEdges[ "tDiffuse" ].value = readBuffer.texture;
this.quad.material = this.materialEdges;
renderer.render( this.scene, this.camera, this.edgesRT, this.clear );
// pass 2
this.quad.material = this.materialWeights;
renderer.render( this.scene, this.camera, this.weightsRT, this.clear );
// pass 3
this.uniformsBlend[ "tColor" ].value = readBuffer.texture;
this.quad.material = this.materialBlend;
if ( this.renderToScreen ) {
renderer.render( this.scene, this.camera );
} else {
renderer.render( this.scene, this.camera, writeBuffer, this.clear );
}
},
setSize: function ( width, height ) {
this.edgesRT.setSize( width, height );
this.weightsRT.setSize( width, height );
this.materialEdges.uniforms[ 'resolution' ].value.set( 1 / width, 1 / height );
this.materialWeights.uniforms[ 'resolution' ].value.set( 1 / width, 1 / height );
this.materialBlend.uniforms[ 'resolution' ].value.set( 1 / width, 1 / height );
},
getAreaTexture: function () {
return '';
},
getSearchTexture: function () {
return '';
}
} );
/**
* @author alteredq / http://alteredqualia.com/
*/
THREE.FilmPass = function ( noiseIntensity, scanlinesIntensity, scanlinesCount, grayscale ) {
THREE.Pass.call( this );
if ( THREE.FilmShader === undefined )
console.error( "THREE.FilmPass relies on THREE.FilmShader" );
var shader = THREE.FilmShader;
this.uniforms = THREE.UniformsUtils.clone( shader.uniforms );
this.material = new THREE.ShaderMaterial( {
uniforms: this.uniforms,
vertexShader: shader.vertexShader,
fragmentShader: shader.fragmentShader
} );
if ( grayscale !== undefined ) this.uniforms.grayscale.value = grayscale;
if ( noiseIntensity !== undefined ) this.uniforms.nIntensity.value = noiseIntensity;
if ( scanlinesIntensity !== undefined ) this.uniforms.sIntensity.value = scanlinesIntensity;
if ( scanlinesCount !== undefined ) this.uniforms.sCount.value = scanlinesCount;
this.camera = new THREE.OrthographicCamera( - 1, 1, 1, - 1, 0, 1 );
this.scene = new THREE.Scene();
this.quad = new THREE.Mesh( new THREE.PlaneBufferGeometry( 2, 2 ), null );
this.scene.add( this.quad );
};
THREE.FilmPass.prototype = Object.assign( Object.create( THREE.Pass.prototype ), {
constructor: THREE.FilmPass,
render: function ( renderer, writeBuffer, readBuffer, delta, maskActive ) {
this.uniforms[ "tDiffuse" ].value = readBuffer.texture;
this.uniforms[ "time" ].value += delta;
this.quad.material = this.material;
if ( this.renderToScreen ) {
renderer.render( this.scene, this.camera );
} else {
renderer.render( this.scene, this.camera, writeBuffer, this.clear );
}
}
} );
THREE.POrbitControls = function ( looking_object, DOMElement ) {
var scope = this;
var is_dragging = false;
var target = new THREE.Vector3();
var lo_spherical = new THREE.Spherical();
var t_spherical = new THREE.Spherical();
var scale = 1;
scope.enabled = true;
scope.enableDamping = true;
scope.dampingFactor = .063;
// How far you can orbit vertically, upper and lower limits.
// Range is 0 to Math.PI radians.
this.minPolarAngle = 0; // radians
this.maxPolarAngle = Math.PI; // radians
// How far you can orbit horizontally, upper and lower limits.
// If set, must be a sub-interval of the interval [ - Math.PI, Math.PI ].
this.minAzimuthAngle = - Infinity; // radians
this.maxAzimuthAngle = Infinity; // radians
// Set to true to automatically rotate around the target
this.autoRotate = false;
this.autoRotateSpeed = 2.0;
// console.log("> distance:", distance );
// This option actually enables dollying in and out; left as "zoom" for backwards compatibility.
// Set to false to disable zooming
this.enableZoom = true;
this.zoomSpeed = 1.0;
this.zoomFactor = .063;
// How far you can dolly in and out ( PerspectiveCamera only )
this.minDistance = 0;
this.maxDistance = Infinity;
//
var mouse = {
x: 0,
y: 0,
prev_x: 0,
prev_y: 0,
delta_x: 0,
delta_y: 0
}
var dollyStart = new THREE.Vector2();
var dollyEnd = new THREE.Vector2();
var dollyDelta = new THREE.Vector2();
scope.attach = function () {
DOMElement.addEventListener( 'mousedown', onMouseDown, false );
DOMElement.addEventListener( 'mousemove', onMouseMove, false );
DOMElement.addEventListener( 'mouseup', onMouseUp, false );
DOMElement.addEventListener( 'touchstart', onTouchStart, false );
DOMElement.addEventListener( 'touchend', onTouchEnd, false );
DOMElement.addEventListener( 'touchmove', onTouchMove, false );
window.addEventListener( 'mouseout', onMouseUp, false );
DOMElement.addEventListener( 'wheel', onMouseWheel, false );
t_spherical.setFromVector3( looking_object.position );
lo_spherical.setFromVector3( looking_object.position );
}
scope.detach = function() {
DOMElement.removeEventListener( 'mousedown', onMouseDown );
DOMElement.removeEventListener( 'mousemove', onMouseMove );
DOMElement.removeEventListener( 'mouseup', onMouseUp );
DOMElement.removeEventListener( 'touchstart', onTouchStart );
DOMElement.removeEventListener( 'touchend', onTouchEnd );
DOMElement.removeEventListener( 'touchmove', onTouchMove );
window.removeEventListener( 'mouseout', onMouseUp );
DOMElement.removeEventListener( 'wheel', onMouseWheel );
}
/*
██████╗ ██████╗ ████████╗ █████╗ ████████╗███████╗
██╔══██╗██╔═══██╗╚══██╔══╝██╔══██╗╚══██╔══╝██╔════╝
██████╔╝██║ ██║ ██║ ███████║ ██║ █████╗
██╔══██╗██║ ██║ ██║ ██╔══██║ ██║ ██╔══╝
██║ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ███████╗
╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚══════╝
*/
function setMousePosition( x, y, start ) {
mouse.x = x;
mouse.y = y;
mouse.delta_x = start ? 0 : mouse.x - mouse.prev_x;
mouse.delta_y = start ? 0 : mouse.y - mouse.prev_y;
mouse.prev_x = mouse.x;
mouse.prev_y = mouse.y;
if( is_dragging ){
t_spherical.phi -= mouse.delta_y * .002;
t_spherical.theta -= mouse.delta_x * .002;
t_spherical.makeSafe();
}
}
// ROTATE
function onMouseDown ( e ) {
if ( scope.enabled === false ) return;
e.preventDefault();
is_dragging = true;
setMousePosition( e.clientX, e.clientY, true );
}
function onMouseMove ( e ) {
if ( scope.enabled === false ) return;
setMousePosition( e.clientX, e.clientY );
}
function onMouseUp ( e ) {
if ( scope.enabled === false ) return;
e.preventDefault();
is_dragging = false;
}
/*
███████╗ ██████╗ ██████╗ ███╗ ███╗
╚══███╔╝██╔═══██╗██╔═══██╗████╗ ████║
███╔╝ ██║ ██║██║ ██║██╔████╔██║
███╔╝ ██║ ██║██║ ██║██║╚██╔╝██║
███████╗╚██████╔╝╚██████╔╝██║ ╚═╝ ██║
╚══════╝ ╚═════╝ ╚═════╝ ╚═╝ ╚═╝
*/
// WHEEL
function onMouseWheel( e ) {
if ( scope.enabled === false || scope.enableZoom === false ) return;
e.preventDefault();
e.stopPropagation();
if ( e.deltaY < 0 ) {
// dollyOut( getZoomScale() );
changeZoom( -scope.zoomSpeed );
} else if ( e.deltaY > 0 ) {
// dollyIn( getZoomScale() );
changeZoom( scope.zoomSpeed );
}
}
function changeZoom ( _add ) {
t_spherical.radius += _add;
// restrict radius to be between desired limits
t_spherical.radius = Math.max( scope.minDistance, Math.min( scope.maxDistance, t_spherical.radius ) );
}
/*
████████╗ ██████╗ ██╗ ██╗ ██████╗██╗ ██╗
╚══██╔══╝██╔═══██╗██║ ██║██╔════╝██║ ██║
██║ ██║ ██║██║ ██║██║ ███████║
██║ ██║ ██║██║ ██║██║ ██╔══██║
██║ ╚██████╔╝╚██████╔╝╚██████╗██║ ██║
╚═╝ ╚═════╝ ╚═════╝ ╚═════╝╚═╝ ╚═╝
*/
// TOUCH
function onTouchStart( event ) {
if ( scope.enabled === false ) return;
var touches = event.touches;
switch ( touches.length ) {
case 1: // one-fingered touch: rotate
is_dragging = true;
setMousePosition( touches[ 0 ].pageX, touches[ 0 ].pageY, true );
break;
case 2: // two-fingered touch: dolly
if ( scope.enableZoom === false ) return;
var dx = touches[ 0 ].pageX - touches[ 1 ].pageX;
var dy = touches[ 0 ].pageY - touches[ 1 ].pageY;
var distance = Math.sqrt( dx * dx + dy * dy );
dollyStart.set( 0, distance );
break;
}
}
function onTouchMove( event ) {
if ( scope.enabled === false ) return;
event.preventDefault();
event.stopPropagation();
var touches = event.touches;
switch ( touches.length ) {
case 1: // one-fingered touch: rotate
setMousePosition( touches[ 0 ].pageX, touches[ 0 ].pageY );
break;
case 2: // two-fingered touch: dolly
if ( scope.enableZoom === false ) return;
var dx = touches[ 0 ].pageX - touches[ 1 ].pageX;
var dy = touches[ 0 ].pageY - touches[ 1 ].pageY;
var distance = Math.sqrt( dx * dx + dy * dy );
dollyEnd.set( 0, distance );
dollyDelta.subVectors( dollyEnd, dollyStart );
if ( dollyDelta.y > 0 ) {
changeZoom( -scope.zoomSpeed );
} else if ( dollyDelta.y < 0 ) {
changeZoom( scope.zoomSpeed );
}
dollyStart.copy( dollyEnd );
break;
}
}
function onTouchEnd( event ) {
if ( scope.enabled === false ) return;
is_dragging = false;
}
/*
██╗ ██╗██████╗ ██████╗ █████╗ ████████╗███████╗
██║ ██║██╔══██╗██╔══██╗██╔══██╗╚══██╔══╝██╔════╝
██║ ██║██████╔╝██║ ██║███████║ ██║ █████╗
██║ ██║██╔═══╝ ██║ ██║██╔══██║ ██║ ██╔══╝
╚██████╔╝██║ ██████╔╝██║ ██║ ██║ ███████╗
╚═════╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚══════╝
*/
// UPDATE
scope.update = function () {
// zoom
lo_spherical.radius += (t_spherical.radius - lo_spherical.radius) * scope.zoomFactor;
// console.log( "> ", t_spherical.radius, scale);
// restrict theta to be between desired limits
t_spherical.theta = Math.max( scope.minAzimuthAngle, Math.min( scope.maxAzimuthAngle, t_spherical.theta ) );
// restrict phi to be between desired limits
t_spherical.phi = Math.max( scope.minPolarAngle, Math.min( scope.maxPolarAngle, t_spherical.phi ) );
if( !is_dragging && scope.autoRotate ) {
t_spherical.theta += scope.autoRotateSpeed;
t_spherical.makeSafe();
}
// console.log( t_spherical.radius, scale );
if( scope.enableDamping ){
lo_spherical.phi += ( t_spherical.phi - lo_spherical.phi ) * scope.dampingFactor;
lo_spherical.theta += ( t_spherical.theta - lo_spherical.theta ) * scope.dampingFactor;
lo_spherical.makeSafe();
}else{
lo_spherical.radius = t_spherical.radius;
lo_spherical.phi = t_spherical.phi;
lo_spherical.theta = t_spherical.theta;
}
looking_object.position.setFromSpherical( lo_spherical );
looking_object.lookAt( target );
}
scope.attach();
}
// d.potekhin
// TODO: stop animation of the grid effect
THREE.XRayMaterial = function( params ){
var uniforms = {
uTex: { type:"t", value: params.map || new THREE.Texture() },
offsetRepeat: { value: new THREE.Vector4( 0, 0, 1, 1 ) },
alphaProportion: { type: "1f", value: params.alphaProportion || 0.5 },
diffuse: { value: params.color || new THREE.Color( 0xffffff ) },
opacity: { value: params.opacity || 1 },
gridOffset: { value: 0 }
}
setInterval(function(){
uniforms.gridOffset.value += params.gridOffsetSpeed || 1;
// m.needsUpdate = true;
}, 40 );
var m = new THREE.ShaderMaterial( {
uniforms: uniforms,
// attributes: {
// vertexOpacity: { value: [] }
// },
vertexShader:
"\
varying float _alpha;\
varying vec2 vUv;\
uniform vec4 offsetRepeat;\
uniform float alphaProportion;\
\
void main() {\
gl_Position = projectionMatrix * modelViewMatrix * vec4( position, 1.0 );\
vUv = uv * offsetRepeat.zw + offsetRepeat.xy;\
\
vec4 worldPosition = modelMatrix * vec4( vec3( position ), 1.0 );\
vec3 cameraToVertex = normalize( cameraPosition - worldPosition.xyz);\
_alpha = 1.0 - max( 0.0, dot( normal, cameraToVertex ) );\
_alpha = max( 0.0, (_alpha - alphaProportion) / (1.0 - alphaProportion) );\
}"
,
//alpha = alphaProportion + (alpha - 0.0) * (1.0 - alphaProportion) / (1.0 - 0.0);\
fragmentShader:
"uniform sampler2D uTex;\
uniform vec3 diffuse;\
uniform float opacity;\
uniform float gridOffset;\
\
varying float _alpha;\
varying vec2 vUv;\
\
void main() {\
vec4 texColor = texture2D( uTex, vUv );\
float _a = _alpha * opacity;\
if( _a <= 0.0 ) discard;\
_a = _a * ( sin( vUv.y * 2000.0 + gridOffset ) * .5 + .5 );\
gl_FragColor = vec4( texColor.rgb * diffuse, _a );\
\
}"
,
//if ( alpha < .5 ) discard;\
transparent: true,
blending: THREE.AdditiveBlending,
depthTest: false
} );
return m;
};
(function (kaspersky){
/*
// !! In some cause Detector doesn't see webGL content on my configuration with nVidia GTS250
console.log("Detector.webgl", Detector, Detector.webgl);
if ( ! Detector.webgl ){
console.log("Your graphics card does not seem to support WebGL. Find out how to get it here.");
Detector.addGetWebGLMessage()
return;
}
*/
var scope = this;
var $container = scope.$container = $(".main-container");
if ( !Detector.webgl ){
$container.hide();
return;
}
var enable_screenshot = !true;// !!! Flag enables making Screenshot functionality
var dpr = window.devicePixelRatio !== undefined ? window.devicePixelRatio : 1;
// WEBGL
var scene = scope.scene = new THREE.Scene();
scene.fog = new THREE.Fog( 0x000000, 7, enable_screenshot ? 60 : 40 );
var renderer = scope.renderer = new THREE.WebGLRenderer({
preserveDrawingBuffer: enable_screenshot
});
renderer.autoClearColor = new THREE.Color( 0, 0, 0, 0 );
renderer.setPixelRatio( dpr );
$container.append( renderer.domElement );
// CAMERA
var camera = scope.camera = new THREE.PerspectiveCamera(60, window.innerWidth / window.innerHeight, 0.1, 700);
camera.position.set(-16,11,0);
camera.lookAt( new THREE.Vector3(0,0,0));
// !!! SCREENSHOT
if( enable_screenshot ){
$("body").css("background-color","#ff0000");
$(".main-container").css("background-color","rgba(0,0,0,0)");
$("MAKE SCREENSHOT
").appendTo($("body")).click(function(){
planet.bg.visible = false;
if( composer ) composer.render();
else renderer.render( scene, camera );
var dataURL = renderer.domElement.toDataURL();
$("").attr("src", dataURL ).appendTo($("body"));
});
}
// !!! SCREENSHOT
// EFFECTS
var composer;
if( !isMobile.any ){
var effect, pass;
composer = new THREE.EffectComposer(renderer);
effect = new THREE.RenderPass(scene, camera);
composer.addPass(effect);
// RGB shift
effect = new THREE.ShaderPass( THREE.RGBShiftShader );
effect.uniforms[ 'amount' ].value = 0.0025;//0.005;
composer.addPass( effect );
// FXAA
var FXAAShaderPass = effect = new THREE.ShaderPass( THREE.FXAAShader );
composer.addPass(effect);
var VerticalTiltShiftShader = effect = new THREE.ShaderPass( THREE.VerticalTiltShiftShader );
effect.uniforms['r'].value = .35;
effect.renderToScreen = true;
composer.addPass( effect );
}
// STATS
var stats;
// RESIZE
window.addEventListener('resize', onWindowResize, false );
function onWindowResize() {
var w = window.innerWidth > 1000 ? window.innerWidth - 302 : window.innerWidth;
var h = window.innerHeight;
window.canvasWidth = w;
renderer.domElement.originalSize = { width: w, height: h};
camera.aspect = w / h;
camera.updateProjectionMatrix();
renderer.setPixelRatio( dpr );
renderer.setSize( w, h );
if(composer) {
composer.setSize( w, h );
FXAAShaderPass.uniforms['resolution'].value.set( 1 / w , 1 / h );
VerticalTiltShiftShader.uniforms["v"].value = 1 / h;
}
}
onWindowResize();
var controls = scope.controls = new THREE.POrbitControls( camera, renderer.domElement );
controls.minPolarAngle = Math.PI * .15;
controls.maxPolarAngle = Math.PI * .85;
controls.enableDamping = true;
controls.dampingFactor = .063;
controls.autoRotate = true;
controls.autoRotateSpeed = -.002*3;
controls.enableZoom = !false;
controls.minDistance = 18;
controls.maxDistance = enable_screenshot ? 40 : 20;
controls.zoomSpeed = .2;
controls.zoomFactor = .1;
// PLanet
var planet;
var def = kaspersky.ready.addWait();
kaspersky.planetMain = function (api, locationAPI) {
api()
.then(function( data ) {
setTimeout(function(){
planet = new Planet( scope, data, function () {
def.resolve();
def = undefined;
}, locationAPI );
animate();
}, 500);
})
.fail(function() {
console.log( "error" );
})
.always(function() {
})
};
$(document).trigger("panetMainLoaded");
// ANIMATION LOOP
function animate() {
window.requestAnimationFrame( animate );
if( planet.state != planet.ANIMATED ) controls.update();
planet.update();
if( composer ) composer.render();
else renderer.render( scene, camera );
if ( stats ) stats.update(renderer);
}
this.animateIn = function( t, onComplete ) {
t = t || 3;
TweenLite.to( controls, t,{
autoRotateSpeed: -.002,
ease: Sine.easeOut
});
camera.updateProjectionMatrix();
TweenLite.to( camera, t,{
fov: 60,
ease: Sine.easeInOut,
onUpdate:function() {
camera.updateProjectionMatrix();
},
onComplete: function() {
if( onComplete ) onComplete();
}
})
};
this.animateOut = function( t, onComplete ) {
t = t || 2;
TweenLite.to( camera, t,{
fov: 10,
ease: Sine.easeInOut,
onUpdate:function() {
camera.updateProjectionMatrix();
},
onComplete: function() {
if( onComplete ) onComplete();
}
})
};
})(kaspersky = kaspersky || {});
//------------------------------------------------------------
var kaspersky;
/*
██████╗ ██╗ █████╗ ███╗ ██╗███████╗████████╗ ██████╗ █████╗ ███████╗███████╗
██╔══██╗██║ ██╔══██╗████╗ ██║██╔════╝╚══██╔══╝ ██╔══██╗██╔══██╗██╔════╝██╔════╝
██████╔╝██║ ███████║██╔██╗ ██║█████╗ ██║ ██████╔╝███████║███████╗█████╗
██╔═══╝ ██║ ██╔══██║██║╚██╗██║██╔══╝ ██║ ██╔══██╗██╔══██║╚════██║██╔══╝
██║ ███████╗██║ ██║██║ ╚████║███████╗ ██║ ██████╔╝██║ ██║███████║███████╗
╚═╝ ╚══════╝╚═╝ ╚═╝╚═╝ ╚═══╝╚══════╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝╚══════╝
*/
function Planet( main, data, onReady, locationAPI ){
var $debug = $(".debug");
var scope = window.PLANET = this;
scope.main = main;
var DOMElement = main.renderer.domElement;
var scene = scope.scene = main.scene;
var camera = main.camera;
var radius = scope.radius = 10; // Radius used to calculate position of tiles
var ratio = scope.ratio = window.devicePixelRatio || 1;
// PLANET 3D CONTAINER
var container = this.container = new THREE.Object3D();
scene.add(container);
// 3D CONTAINER FOR OBJECTS LOOKING TO THE CAMERA
var static_container = this.static_container = new THREE.Object3D();
scene.add(static_container);
static_container.matrixAutoUpdate = false;
var textureLoader = this.textureLoader = new THREE.TextureLoader();
this.planet_color = 0xa6f5a3;
this.water_color = 0x81915b;
// STATES
scope.LOADING = "loading";
scope.IDLE = "idle";
scope.ANIMATED = "animated";
scope.state = scope.LOADING;
scope.data = data;
// sort countries by id
data.countries_by_id = {};
for (var i = 0; i < data.countries.length; i++) {
var l = data.countries[i];
data.countries_by_id[ l.country_id ] = l;
};
// sort locations by id
data.locations_by_id = {};
for (var i = 0; i < data.locations.length; i++) {
var l = data.locations[i];
data.locations_by_id[ l.location_id ] = l;
}
//--------------------------------------------
/*
██████╗ ██████╗ ███╗ ███╗███╗ ███╗███████╗███╗ ██╗████████╗ ████████╗██╗██████╗
██╔════╝██╔═══██╗████╗ ████║████╗ ████║██╔════╝████╗ ██║╚══██╔══╝ ╚══██╔══╝██║██╔══██╗
██║ ██║ ██║██╔████╔██║██╔████╔██║█████╗ ██╔██╗ ██║ ██║ ██║ ██║██████╔╝
██║ ██║ ██║██║╚██╔╝██║██║╚██╔╝██║██╔══╝ ██║╚██╗██║ ██║ ██║ ██║██╔═══╝
╚██████╗╚██████╔╝██║ ╚═╝ ██║██║ ╚═╝ ██║███████╗██║ ╚████║ ██║ ██║ ██║██║
╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═╝╚══════╝╚═╝ ╚═══╝ ╚═╝ ╚═╝ ╚═╝╚═╝
*/
// COMMENTARY TIP
this._PlanetCommentPopup = PlanetCommentPopup( this.main.$container );
scope.commentToggle = function( _show ){
if( !scope._PlanetCommentPopup ) return;
if( _show ) scope._PlanetCommentPopup.showRandom();
else scope._PlanetCommentPopup.hide();
}
this.getLocationBriefs = function( location_id, onComplete ){
var loc = scope.data.locations_by_id[ location_id ];
var brief_list = loc.brief_list;
if( brief_list ){
if(!!brief_list[ PlanetData.YEAR_ID ] !== false)
onComplete( brief_list[ PlanetData.YEAR_ID ].briefs );
return;
}else{
locationAPI( { location_id: location_id } )
.done( function( locationData ){
brief_list = loc.brief_list = locationData;
if(!!brief_list[ PlanetData.YEAR_ID ] !== false)
onComplete( brief_list[ PlanetData.YEAR_ID ].briefs );
} );
}
}
// ELEMENTS
this.drawPoints( radius * 1.1 );
this.drawParticles( radius );
this.drawBG();
this.drawSputniks();
this.drawOrbitas();
this.planetLocations = new PlanetLocations( this );
this.planetPointed = new PlanetPointed( this );
this.planetContour = new PlanetContour( this );
/*
██╗ ██╗██████╗ ██████╗ █████╗ ████████╗███████╗
██║ ██║██╔══██╗██╔══██╗██╔══██╗╚══██╔══╝██╔════╝
██║ ██║██████╔╝██║ ██║███████║ ██║ █████╗
██║ ██║██╔═══╝ ██║ ██║██╔══██║ ██║ ██╔══╝
╚██████╔╝██║ ██████╔╝██║ ██║ ██║ ███████╗
╚═════╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚══════╝
*/
var camera_dist_to_center;
this.update = function() {
camera_dist_to_center = camera.position.distanceToSquared( THREE.Vector3.ZERO ) - 100;
static_container.lookAt( camera.getWorldPosition() );
scope.static_container.updateMatrix();
// Rotation of the grid sphere
this.grid_shpere.rotation.y += .0005;
// MOVEMENT OF DECOR ORBITS
if( this.orbits ){
for (var i = 0; i < this.orbits.length; i++) {
var orbit = this.orbits[i];
orbit.rotation.x += orbit._increment.x;
orbit.rotation.y += orbit._increment.y;
orbit.rotation.z += orbit._increment.z;
};
}
// Movement of space stations
if( this.sputniks ){
for (var i = 0; i < this.sputniks.length; i++) {
var sputnik = this.sputniks[i];
sputnik.rotation.y += .0020;
}
}
// SPACE JUNK FLOATING AROUND THE PLANET
if( this.particles ){
this.particles.rotation.y += .001;
}
// PLANET LANDSCAPE MADE OF POINTS
this.planetPointed.update();
if( this.updateBg ) this.updateBg();
// Active hexagon update state
if( active_hexagon ) {
active_hex_dummy.updateMatrix();
var pos = active_hex_dummy.getWorldPosition();
var v1 = pos.clone().project( camera );
v1.x = (v1.x + 1) / 2 * window.canvasWidth;
v1.y = -(v1.y - 1) / 2 * window.innerHeight;
active_hexagon.style.transform = "translate("+ v1.x +"px, "+v1.y+"px)";
// Deactivate picked hexagons when it moves to the negative side of the planet
if( !this.isInFrontOfPlanet( active_hex_dummy.position ) ){
scope.deactivateHexagon();
}
}
if( this._PlanetCommentPopup ) this._PlanetCommentPopup.update();
if( this.state == this.IDLE && !isMobile.any ) {
chekHover()
}
}
/*
██╗███╗ ██╗████████╗███████╗██████╗ █████╗ ██████╗████████╗██╗██╗ ██╗███████╗
██║████╗ ██║╚══██╔══╝██╔════╝██╔══██╗██╔══██╗██╔════╝╚══██╔══╝██║██║ ██║██╔════╝
██║██╔██╗ ██║ ██║ █████╗ ██████╔╝███████║██║ ██║ ██║██║ ██║█████╗
██║██║╚██╗██║ ██║ ██╔══╝ ██╔══██╗██╔══██║██║ ██║ ██║╚██╗ ██╔╝██╔══╝
██║██║ ╚████║ ██║ ███████╗██║ ██║██║ ██║╚██████╗ ██║ ██║ ╚████╔╝ ███████╗
╚═╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝╚═╝ ╚═╝╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═══╝ ╚══════╝
*/
// INTERACTIVE
scope.tile_current = null;
var mouse = new THREE.Vector2(-2,-2);
var mouse_dir = mouse.clone();PlanetPointed
var mouse_moving_distance = 0;
var intersect_sphere = new THREE.Sphere( scope.container.position.clone(), radius );
var raycaster = new THREE.Raycaster();
DOMElement.addEventListener( 'touchstart', onMouseDown, false );
DOMElement.addEventListener( 'touchmove', onMouseMove, false );
DOMElement.addEventListener( 'touchend', onMouseUp, false );
DOMElement.addEventListener( 'mousedown', onMouseDown, false );
DOMElement.addEventListener( 'mousemove', onMouseMove, false );
DOMElement.addEventListener( 'mouseup', onMouseUp, false );
var mx, my, dx, dy;
function onMouseDown (e) {
e.preventDefault();
if( e.touches ){
var touch = e.touches[0];
mx = touch.clientX;
my = touch.clientY;
}else{
mx = e.clientX * ratio;
my = e.clientY * ratio;
}
updateMouse( mx, my );
mouse_moving_distance = 0;
mouse_is_down = true;
}
function onMouseUp (e) {
mouse_is_down = false;
if( scope.state != scope.IDLE ) return;
// is it was a click?
if( mouse_moving_distance>6 ) return;
if( isMobile.any ){
checkTapOnObject();
}
}
function onMouseMove (e) {
e.preventDefault();
if( e.touches ){
var touch = e.touches[0];
mx = touch.clientX;
my = touch.clientY;
}else{
mx = e.clientX * ratio;
my = e.clientY * ratio;
}
updateMouse(mx, my);
}
function updateMouse ( mx, my ) {
dx = Math.abs(mouse.x - mx);
dy = Math.abs(mouse.y - my);
mouse_moving_distance += dx*dx + dy*dy;
mouse.x = mx;
mouse.y = my;
if( isMobile.any){
mouse_dir.x = (mouse.x / DOMElement.originalSize.width - .5) * 2;
mouse_dir.y = -(mouse.y / DOMElement.originalSize.height - .5) * 2;
}else{
mouse_dir.x = (mouse.x / DOMElement.width - .5) * 2;
mouse_dir.y = -(mouse.y / DOMElement.height - .5) * 2;
}
}
function chekHover () {
if( scope.state != scope.IDLE) return;
var intersection, _location_current;
raycaster.setFromCamera( mouse_dir, camera );
// intersection with sphere
intersection = raycaster.ray.intersectSphere( intersect_sphere );
if( intersection ){
_location_current = scope.getClosestLocation( intersection );
}
if( _location_current ){
if( _location_current != scope.location_current ){
activateHexagon( _location_current );
return true;
}
}
return false;
}
function checkTapOnObject () {
var intersection, _location_current;
raycaster.setFromCamera( mouse_dir, camera );
// intersection with sphere
intersection = raycaster.ray.intersectSphere( intersect_sphere );
if( intersection ){
_location_current = scope.getClosestLocation( intersection );
}
if( _location_current ) {
activateHexagon( _location_current , true);
}else{
scope.deactivateHexagon();
}
}
/*
██╗ ██╗ ██████╗ ██╗ ██╗███████╗██████╗ ███████╗
██║ ██║██╔═══██╗██║ ██║██╔════╝██╔══██╗██╔════╝
███████║██║ ██║██║ ██║█████╗ ██████╔╝███████╗
██╔══██║██║ ██║╚██╗ ██╔╝██╔══╝ ██╔══██╗╚════██║
██║ ██║╚██████╔╝ ╚████╔╝ ███████╗██║ ██║███████║
╚═╝ ╚═╝ ╚═════╝ ╚═══╝ ╚══════╝╚═╝ ╚═╝╚══════╝
*/
var location_picked = null;// hexagon_hovered and picked hexagon
/*
█████╗ ██████╗████████╗██╗██╗ ██╗███████╗ ██╗ ██╗███████╗██╗ ██╗ █████╗ ██████╗ ██████╗ ███╗ ██╗
██╔══██╗██╔════╝╚══██╔══╝██║██║ ██║██╔════╝ ██║ ██║██╔════╝╚██╗██╔╝██╔══██╗██╔════╝ ██╔═══██╗████╗ ██║
███████║██║ ██║ ██║██║ ██║█████╗ ███████║█████╗ ╚███╔╝ ███████║██║ ███╗██║ ██║██╔██╗ ██║
██╔══██║██║ ██║ ██║╚██╗ ██╔╝██╔══╝ ██╔══██║██╔══╝ ██╔██╗ ██╔══██║██║ ██║██║ ██║██║╚██╗██║
██║ ██║╚██████╗ ██║ ██║ ╚████╔╝ ███████╗ ██║ ██║███████╗██╔╝ ██╗██║ ██║╚██████╔╝╚██████╔╝██║ ╚████║
╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═══╝ ╚══════╝ ╚═╝ ╚═╝╚══════╝╚═╝ ╚═╝╚═╝ ╚═╝ ╚═════╝ ╚═════╝ ╚═╝ ╚═══╝
*/
// TODO: separate placing of additive prview hexagons
var active_hex_dummy = new THREE.Object3D();
active_hex_dummy.matrixAutoUpdate = false;
this.container.add( active_hex_dummy );
var active_hexagon = null;//
var extActiveHexagonHide;// External method to hide DOM element of the active hexagon
// ACTIVE HEXAGON
function activateHexagon( _location, mobile){
scope.deactivateHexagon();
main.controls.autoRotate = false;
scope.location_current = _location;
// Define an array of nearby locations to view
location_picked = _location;
_location.briefs = PlanetData.hasLocationAnyBriefs( location_picked, true );
_location.briefs_text = PlanetData.hasLocationAnyBriefTexts( location_picked, true );
_location.works_list = PlanetData.getWorkList( location_picked, true );
_location.more = PlanetData.getMoreText( location_picked, true );
;
var _locations = [ _location ];
// Hit an external method to show defined hexagons
if( kaspersky.getHexaspherePopup ) {
var o = kaspersky.getHexaspherePopup( _locations, moveToActiveHexagon, mobile);
active_hexagon = o.element;
extActiveHexagonHide = o.close;
var $active_hexagon = $(active_hexagon);
$active_hexagon.on( "mouseleave", onActiveHexagonMouseLeave );
function onActiveHexagonMouseLeave(){
scope.deactivateHexagon();
$active_hexagon.off( "mouseleave", onActiveHexagonMouseLeave );
}
}
active_hex_dummy.position.copy( _location.position );
// Disable Earth constant rotation
main.controls.autoRotate = false;
_location.light.visible = false;
// Hide current commentary
scope.commentToggle( false );
}
this.deactivateHexagon = function() {
if( !location_picked ) return;
if (typeof extActiveHexagonHide === "function") extActiveHexagonHide();
active_hexagon = null;
scope.location_current = null;
location_picked.light.visible = true;
location_picked = null;
main.controls.autoRotate = true;
// show a commentary
scope.commentToggle( true );
}
var camera_rotator = new THREE.Object3D();
camera_rotator.up = new THREE.Vector3(0,1,0);
main.scene.add(camera_rotator);
camera_rotator.matrixAutoUpdate = false;
function moveToActiveHexagon( location ) {
var navigateUrl;
var $info = location.years[PlanetData.YEAR_ID];
if ($info.type === "brief") {
navigateUrl = langUrl + $info.brief_id;
} else {
navigateUrl = langUrl + "locations/" + PlanetData.YEAR_ID + "/" + location.location_id;
}
scope.state = scope.ANIMATED;
main.controls.detach();
DOMElement.style.cursor = "default";
scope.deactivateHexagon();
// hide current commentary
scope.commentToggle( false );
// target rotation
camera_rotator.lookAt( location.position );
var qend = camera_rotator.quaternion.clone();
camera_rotator.lookAt( main.camera.position );
var qstart = camera_rotator.quaternion.clone();
camera_rotator.updateMatrix();
var d = main.camera.position.length();
camera_rotator.add( main.camera );
main.camera.position.set( 0,0,d );
main.camera.rotation.set( 0,0,0 );
scene.updateMatrixWorld();
// ANIMATE SPHERICAL ROTATION TO ACTIVE HEXAGON
camera_rotator._anim_progress_ = 0;
TweenLite.to( camera_rotator, 1.5, {
_anim_progress_: 1,
ease: Sine.easeInOut,
onUpdate: function () {
THREE.Quaternion.slerp( qstart, qend, camera_rotator.quaternion, camera_rotator._anim_progress_ );
camera_rotator.updateMatrix();
},
onComplete: function() {
TweenLite.to( main.camera.position, 1.2, {
z: radius*1.15,
ease: Sine.easeIn,
onComplete: function() {
}
});
// show Transition
setTimeout( kaspersky.Transition.fadeIn, 400);
setTimeout(function () {
kaspersky.navigateTo( navigateUrl );
}, 400);
}
});
}
// UTILS
this.isInFrontOfPlanet = function( position ) {
var dist = camera.position.distanceToSquared( position );
if( dist > camera_dist_to_center ) return false;
return true;
}
this.googlePosToUV = function( _u, _v ){
if( _u===undefined ) {
var a = ('40.705565,-74.1180857').split(",");
_u = a[0];
_v = a[1];
}
var u = 1-(parseFloat( _u ) + 90 )/180;
u += .014; if( u>1 ) u = u-1;
var v = (parseFloat( _v ) + 180 )/360;
v -= .031; if( v<0 ) v = 1+v;
return {u:u,v:v};
}
// INIT THE PLANET
var _first_year = window.currentYear || PlanetData.year_ids[0];
onReady();
main.animateOut( .05 , function(){
scope.showYear( _first_year, undefined, 6 );
});
}
Planet.prototype.getClosestLocation = function( intersection ) {
var _location_current;
var min = isMobile.any ? .15 : .04;
var _locations = this.planetLocations.current_locations;
for (var i = 0; i < _locations.length; i++) {
var l = _locations[i];
// if the location object has no briefs
if( !l.position ) continue;
if( !PlanetData.hasLocationAnyBriefs( l, true ) ) continue;
var d = intersection.distanceToSquared( l.position );
if( d>>", position.y, p );
p += Math.random()>.8 ? Math.random()*.7 : Math.random() * .2;
var rotation = new THREE.Vector3( Math.random() * Math.PI, 0, Math.random() * Math.PI );
// this.drawRing( position, radius * (Math.random()*.4 + .9), Math.random()*.1 + .02, rotation );
position.y = 0;
var orbit = this.drawRing( position, this.radius*1.1, Math.random()*.05 + .02, rotation );
var speed_x = Utils.getRandSides( .002 );
var speed_y = Utils.getRandSides( .002 );
var speed_z = Utils.getRandSides( .002 );
orbit._increment = new THREE.Vector3( speed_x, speed_y, speed_z );
this.orbits.push(orbit);
};
// console.log("> ", this.orbits.length);
}
/*
██████╗ ██████╗
██╔══██╗██╔════╝
██████╔╝██║ ███╗
██╔══██╗██║ ██║
██████╔╝╚██████╔╝
╚═════╝ ╚═════╝
*/
Planet.prototype.drawBG = function() {
// draw BG
var scope = this;
var wo = 2560;
var ho = 1440;
var scale_coef = 590;
// var w = this.radius*.5 / 9*16 *5;
// var h = this.radius*.5 *5;
var w = wo / scale_coef * this.radius;
var h = ho / scale_coef * this.radius;
var g = new THREE.PlaneGeometry( w, h, 1 );
var m = createMaterial( "bg.jpg", function(){
TweenLite.to( m, 3,{
opacity: 1,
ease: Sine.easeInOut
})
});
var bg = this.bg = new THREE.Mesh( g, m );
bg.matrixAutoUpdate = false;
// bg.position.z = radius*1.5;
this.static_container.add( bg );
var patches = [];
addPatch( "bg_fragment_left.jpg", 933, 1011, 219, 0, 0 );
addPatch( "bg_fragment_left_1.jpg", 933, 1011, 219, 0, Math.PI );
addPatch( "bg_fragment_right.jpg", 1074, 1133, 1516, 8, Math.PI/3 );
addPatch( "bg_fragment_right_1.jpg", 1074, 1133, 1516, 8, Math.PI/3 + Math.PI );
this.updateBg = function( delta ){
for (var i = 0; i < patches.length; i++) {
var m = patches[i];
m.anim_offset += .03;
m.opacity = Math.sin( m.anim_offset ) * .4 + .2;
};
}
function addPatch( img, ww, hh, xx, yy, anim_offset ){
ww = ww/wo*w;
hh = hh/ho*h;
var g = new THREE.PlaneGeometry( ww, hh, 1 );
var m = createMaterial( img );
// m.blending = 0;
m.opacity = 1;
m.anim_offset = anim_offset || 0;
patches.push( m );
var mesh = new THREE.Mesh( g, m );
mesh.position.x = -w/2 + ww/2 + xx/wo*w;
// console.log( -w/2, mesh.position.x );
mesh.position.y = h/2 - hh/2 - yy/ho*h;
mesh.position.z = .01;
mesh.matrixAutoUpdate = false;
mesh.updateMatrix();
bg.add( mesh );
}
function createMaterial( img, onLoad ){
var m = new THREE.MeshBasicMaterial();
m.fog = !false;
m.transparent = true;
m.opacity = 0;
m.blending = THREE.AdditiveBlending;
m.depthWrite = false;
// m.color = new THREE.Color( 0xff0000 );
// m.wireframe = true;
m.map = scope.textureLoader.load( PlanetData.textures_path+img, onLoad );
m.map.generateMipalphaMaps = false;
m.map.magFilter = THREE.LinearFilter;
m.map.minFilter = THREE.LinearFilter;
return m;
}
}
/*
███████╗██████╗ ██╗ ██╗████████╗███╗ ██╗██╗██╗ ██╗
██╔════╝██╔══██╗██║ ██║╚══██╔══╝████╗ ██║██║██║ ██╔╝
███████╗██████╔╝██║ ██║ ██║ ██╔██╗ ██║██║█████╔╝
╚════██║██╔═══╝ ██║ ██║ ██║ ██║╚██╗██║██║██╔═██╗
███████║██║ ╚██████╔╝ ██║ ██║ ╚████║██║██║ ██╗
╚══════╝╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═══╝╚═╝╚═╝ ╚═╝
*/
Planet.prototype.drawSputniks = function () {
// SPUTNIKS
this.sputniks = [];
var sputnik1 = this.addSputnik( 'station_b.js', this.radius*1.2, this.container, new THREE.Vector3(-.5,0,0) );
var sputnik2 = this.addSputnik( 'station_c.js', this.radius*1.25, this.container, new THREE.Vector3(.6,3,-1.2) );
}
Planet.prototype.addSputnik = function ( name, radius, container, rotation ) {
var scope = this;
var loader = new THREE.JSONLoader();
var ring = scope.drawRing( new THREE.Vector3(), radius, .05, new THREE.Vector3() );
var m = ring.material;
m.opacity = .3;
loader.load(
// resource URL
PlanetData.textures_path + name,
// Function when resource is loaded
function ( geometry, materials ) {
var m = new THREE.MeshBasicMaterial( { color: scope.planet_color });
m.wireframe = true;
// m.transparent = true;
// m.blending = THREE.AdditiveBlending;
// m.depthWrite = false;
var object = new THREE.Mesh( geometry, m );
object.position.x = -radius;
object.rotation.x = .5;
object.scale.set( .03, .03, .03 );
object.matrixAutoUpdate = false;
object.updateMatrix();
ring.add( object );
}
);
scope.sputniks.push( ring );
var ring_container = new THREE.Object3D();
scope.container.add( ring_container );
ring_container.add( ring );
ring_container.rotation.x = rotation.x;
ring_container.rotation.y = rotation.y;
ring_container.rotation.z = rotation.z;
ring_container.matrixAutoUpdate = false;
ring_container.updateMatrix();
return ring;
}
function PlanetPointed( planet ) {
var scope = this;
/// INIT
var parts_count = 2;
var u_grid = 250;
var v_grid = 250;
var texture = planet.textureLoader.load( PlanetData.textures_path + "dot.png" );
texture.generateMipalphaMaps = false;
texture.magFilter = THREE.LinearFilter;
texture.minFilter = THREE.LinearFilter;
var materials = [];
for (var i = 0; i < parts_count; i++) {
var m = new THREE.PointsMaterial( { size: .15 / planet.ratio } );//
m.color = new THREE.Color(0x31b477);
m.map = texture;
m.depthWrite = false;
m.transparent = true;
m.opacity = 0;
m.blending = THREE.AdditiveBlending;
// m.side = THREE.BackSide;//FrontSide;
var prop = i/parts_count;
m.t_ = prop*Math.PI*2;
m.speed_ = .04;
m.min_ = Math.random()*.2+.5;
m.delta_ = Math.random()*.1+.1;
m.opacity_coef_ = 1;
materials.push( m );
}
var current_container;
var sea_hex_material;
// METHODS
this.init = function( _year ) {
var container = new THREE.Object3D();
container.matrixAutoUpdate = false;
// LAND
var sphere_parts = [];
for (var i = 0; i < parts_count; i++) {
sphere_parts[i] = {
positions: [],
// colors: [],
};
};
var positions = [];
// var colors = [];
spherical = new THREE.Spherical();
spherical.radius = planet.radius;
var pos = new THREE.Vector3();
var projectiveImage = PlanetData.years[_year].earth_image;
// var col = new THREE.Color(0xb1f7a0);
for (var sv = 0; sv < v_grid; sv++) {
var st = (u_grid * ( 1 - Math.sin( sv/v_grid*Math.PI) ))/u_grid+.5;
// console.log(sv,st);
for (var su = 0; su < u_grid; su+=st) {
var u = su/u_grid;
var v = sv/v_grid;
var is_land = projectiveImage.isLandByUV( u, v );
if( is_land ){
var o = sphere_parts[ Math.floor(Math.random()*parts_count)];
spherical.theta = u * Math.PI *2 - Math.PI/2;
spherical.phi = v * Math.PI;
pos.setFromSpherical( spherical );
o.positions.push( pos.x );
o.positions.push( pos.y );
o.positions.push( pos.z );
// o.colors.push( col.r );
// o.colors.push( col.g );
// o.colors.push( col.b );
}
}
};
// POINTS
for (var i = 0; i < sphere_parts.length; i++) {
var o = sphere_parts[i];
var geometry = new THREE.BufferGeometry();
var positions_ = new Float32Array( o.positions.length );
for (var j = 0; j < o.positions.length; j++) {
positions_[j] = o.positions[j];
};
// var colors_ = Float32Array.from( o.colors );
geometry.addAttribute( 'position', new THREE.BufferAttribute( positions_, 3 ) );
// geometry.addAttribute( 'color', new THREE.BufferAttribute( colors_, 3 ) );
geometry.computeBoundingSphere();
o.material = materials[i];
var points = o.mesh = new THREE.Points( geometry, o.material );
points.matrixAutoUpdate = false;
container.add( points );
};
// SEA HEXS
var r = planet.radius * .97;
sea_hex_material = new THREE.MeshBasicMaterial({
transparent : true,
opacity : 0,
color : new THREE.Color( 0x31b477 ),
blending : THREE.AdditiveBlending,
// side : THREE.DoubleSide,
depthWrite : false
// ,wireframe: true
});
// Utils.drawFunction( undefined, 0 );
var hex = Utils.createHexagon( .12, false, sea_hex_material, .025 );
var geo = new THREE.Geometry();//hex.geometry.clone();
var step = .01;
var stp = step;
var end, xx, y, yy;
var pi2 = Math.PI*2;
var check_offset = .02;
var row = false;
for (var _y = -Math.PI + .5; _y < -.6; _y+= .04 ) {
y = Math.cos( _y ) * .5 + .5;
// console.log(">>", _y, y );
row = !row;
stp = Math.abs( step / Math.sin( y * Math.PI ) );
end = Math.floor( 1 / stp );
// console.log( y, stp, end );
for (var x = 0; x < end; x++ ) {
xx = .5 + ( end/2 - x - (row ? .5 : 0) ) * stp;
yy = y;
if(
Math.random()>.25 ||
projectiveImage.isLandByUV( xx, yy )
|| projectiveImage.isLandByUV( xx - check_offset, yy )
|| projectiveImage.isLandByUV( xx + check_offset, yy )
|| projectiveImage.isLandByUV( xx, yy - check_offset )
|| projectiveImage.isLandByUV( xx, yy + check_offset )
) continue;
Utils.setFromSpherical( planet.radius * (.97 - Math.random()*.01), xx, yy, hex.position );
hex.lookAt( THREE.Vector3.ZERO );
hex.updateMatrix();
hex.updateMatrixWorld();
geo.merge( hex.geometry, hex.matrixWorld );
}
};
// console.log("=>", geo );
var hexs = new THREE.Mesh( geo, sea_hex_material );
hexs.matrixAutoUpdate = false;
container.add( hexs );
return container;
}
this.update = function(){
for (var i = 0; i < materials.length; i++) {
var m = materials[i];
m.t_ += m.speed_;
// o.material.opacity = Math.sin(o.t)*.2+.5;
m.opacity = (Math.sin( m.t_ ) * m.delta_ + m.min_) * m.opacity_coef_;
};
}
this.show = function( _year, t ){
var year_data = PlanetData.years[_year];
if( !year_data.pointed_sphere ) year_data.pointed_sphere = this.init( _year );
current_container = year_data.pointed_sphere;
planet.container.add( current_container );
for (var i = 0; i < materials.length; i++) {
var m = materials[i];
TweenLite.killTweensOf(m);
TweenLite.to( m, t, {
opacity_coef_: 1,
ease: Sine.easeInOut,
onComplete: function(){
}
})
};
TweenLite.killTweensOf(sea_hex_material);
TweenLite.to( sea_hex_material, t, {
opacity: .4,
ease: Sine.easeInOut
});
}
this.hide = function ( t ) {
for (var i = 0; i < materials.length; i++) {
var m = materials[i];
TweenLite.killTweensOf(m);
TweenLite.to( m, t, {
opacity_coef_: 0,
ease: Sine.easeInOut,
onComplete: function(){
planet.container.remove( current_container );
}
})
};
TweenLite.killTweensOf(sea_hex_material);
TweenLite.to( sea_hex_material, t, {
opacity: 0,
ease: Sine.easeInOut
});
}
}
// LAND CONTOUR
PlanetContour = function ( planet ) {
var max_opacity = .8;//.65;
var _geometry = new THREE.SphereGeometry( planet.radius * .995, 32, 32, 0, Math.PI );
var m1 = new THREE.MeshBasicMaterial();
m1.color = new THREE.Color( 0x31b477 );
m1.fog = false;
m1.transparent = true;
m1.blending = THREE.AdditiveBlending;
m1.depthWrite = false;
var sphere1 = new THREE.Mesh( _geometry, m1 );
sphere1.visible = false;
sphere1.matrixAutoUpdate = false;
// sphere1.rotation.y = Math.PI * .999;
sphere1.updateMatrix();
planet.container.add( sphere1 );
sphere1.matrixAutoUpdate = false;
// 2nd halfsphere
var m2 = m1.clone();
var sphere2 = sphere1.clone();
sphere2.material = m2;
planet.container.add( sphere2 );
sphere2.rotation.y = Math.PI;
sphere2.updateMatrix();
// METHODS
this.show = function( _year, t ){
var year_data = PlanetData.years[_year];
if( !year_data.contour_texture ){
var urls = year_data.contour_url;
year_data.contour_texture = [ getTexture( urls[0] ), getTexture( urls[1] ) ];
}
m1.map = year_data.contour_texture[0];
m1.map.generateMipalphaMaps = false;
m1.map.magFilter = THREE.LinearFilter;
m1.map.minFilter = THREE.LinearFilter;
m1.needsUpdate = true;
m2.map = year_data.contour_texture[1];
m2.map.generateMipalphaMaps = false;
m2.map.magFilter = THREE.LinearFilter;
m2.map.minFilter = THREE.LinearFilter;
m2.needsUpdate = true;
fadeIn( sphere1, t );
fadeIn( sphere2, t );
}
this.hide = function( t ){
fadeOut( sphere1, t );
fadeOut( sphere2, t );
}
function getTexture( path ){
var map = planet.textureLoader.load( path );// PlanetData.textures_path + "contour1_1.png"
map.generateMipalphaMaps = false;
map.magFilter = THREE.LinearFilter;
map.minFilter = THREE.LinearFilter;
return map;
}
function fadeIn ( o, t ){
o.visible = true;
var m = o.material;
m.opacity = 0;
TweenLite.killTweensOf( m );
TweenLite.to( m, t,{
opacity: max_opacity,
ease: Sine.easeInOut
// ,onComplete: function(){
// }
})
}
function fadeOut ( o, t ){
var m = o.material;
TweenLite.killTweensOf( m );
TweenLite.to( m, t,{
opacity: 0,
ease: Sine.easeInOut,
onComplete: function(){
// o.visible = false;
}
})
}
}
// COMMENT POPUP
function PlanetCommentPopup( $container ) {
var comment_is_shown = false;
var planet = window.PLANET;
var shownext_timer;
var $line = $("