卡巴斯基 2050.earth 源码分享(完整)

!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 = $("
") .appendTo( $container ) .hide(); var $comment = $("
") .appendTo( $container ) .hide(); var $text = $("
") .appendTo( $comment ) var camera = planet.main.camera; var comment_dummy = new THREE.Object3D(); planet.container.add( comment_dummy ); this.show = function( text, _position, delay, onFinish ) { if( planet.state != planet.IDLE ){ setTimeout( show.bind( this, text.name, _position, delay, onFinish ), 2000 ); return; } $line.show(); $text.text( text.name ); $text.attr("href", text.id); $text.on("click",function(){ location.href = $(this).attr("href"); }); $comment.stop().show(200); comment_is_shown = true; comment_dummy.position.copy( _position ); if(delay){ clearTimeout( shownext_timer ); shownext_timer = setTimeout( function(){ hide( onFinish ); }, delay ); } } this.hide = function( onFinish ) { clearTimeout( shownext_timer ); $comment.stop().hide(200, function() { $line.hide(); comment_is_shown = false; if(onFinish) onFinish(); }); } var _timeout; this.showRandom = function () { var _location; var _locations = planet.planetLocations.current_locations; clearTimeout( _timeout ); if ( !_locations ) { _timeout = setTimeout( this.showRandom, 100 ); return; } for( var i=0; _locations.length > 0 && i<10; i++){ _location = _locations[ ~~(Math.random()*_locations.length) ]; if( _location.position && PlanetData.hasLocationAnyBriefs( _location, true ) && planet.isInFrontOfPlanet( _location.position ) ) { break; } _location = null; } if( !_location ){ _timeout = setTimeout( this.showRandom, 200 ); return; } planet.getLocationBriefs( _location.location_id, function( briefs ){ if( briefs && briefs.length ){ var brief = briefs[ ~~(Math.random()*briefs.length) ]; show( brief, _location.position, 3000, showRandom ); }else{ _timeout = setTimeout( this.showRandom, 100 ); } }); } this.update = function() { if( !comment_is_shown ) return; var pos = comment_dummy.getWorldPosition(); var v1 = pos.clone().project( camera ); v1.x = (v1.x + 1) / 2 * window.canvasWidth; v1.y = -(v1.y - 1) / 2 * window.innerHeight; var v2 = pos.clone().multiplyScalar( 1.1 ).project( camera ); v2.x = (v2.x + 1) / 2 * window.canvasWidth; v2.y = -(v2.y - 1) / 2 * window.innerHeight; $comment.css({ transform: "translate("+ v2.x +"px, "+v2.y+"px)", height: "auto", width: "auto", }); var w = v1.x - v2.x; var h = v1.y - v2.y; var scl = Math.sqrt( w*w + h*h )+1; var a = Math.atan2( h + 1, w ); $line.css({ transform: "translate("+ (v2.x) +"px, "+ (v2.y) +"px) rotate("+a+"rad)", width: scl+"px" }); if( !planet.isInFrontOfPlanet( pos ) ) showRandom(); } return this; } var PlanetData = (function(){ var scope = this; this.textures_path = "/assets/earth/"; this.YEAR = null;// Current YEAR data container this.YEAR_ID = null;// Current year id container // YEARS DATA this.years = { "2030":{ "earth_url" : this.textures_path + "earth_1.png", "contour_url" : [ this.textures_path + "contour1_1.png", this.textures_path + "contour2_1.png" ] ,min_locations: 100 }, "2040":{ "earth_url" : this.textures_path + "earth_1.png", "contour_url" : [ this.textures_path + "contour1_1.png", this.textures_path + "contour2_1.png" ] ,min_locations: 50 }, "2050":{ "earth_url" : this.textures_path + "earth_1.png", "contour_url" : [ this.textures_path + "contour1_1.png", this.textures_path + "contour2_1.png" ] ,min_locations: 10 }, }; // this.year_ids = []; for (var i in this.years) { this.year_ids.push( i ); }; this.hasLocationAnyBriefs = function( _location, year ){ // current year if( year === true && !!_location.years[ scope.YEAR_ID ]){ return _location.years[ scope.YEAR_ID ].briefs; // specified year }else if( year !== undefined && !!_location.years[ year ]){ return _location.years[ year ].briefs; } // any year for (var i in _location.years) { var briefs = _location.years[ i ].briefs; if( briefs ) return briefs; }; return 0; } this.hasLocationAnyBriefTexts = function( _location, year ){ // current year if( year === true ){ return _location.years[ scope.YEAR_ID ].briefs_text; // specified year }else if( year !== undefined ){ return _location.years[ year ].briefs_text; } // any year for (var i in _location.years) { var briefs = _location.years[ i ].briefs; if( briefs ) return _location.years[ i ].briefs_text; }; return 0; } this.getWorkList = function( _location, year ) { if(year == true){ return _location.years[ scope.YEAR_ID ].works_list; } else if( year !== undefined ){ return _location.years[ year ].works_list; } return 0; } this.getMoreText = function( _location, year ) { if(year == true){ return _location.years[ scope.YEAR_ID ].more_btn; } else if( year !== undefined ){ return _location.years[ year ].more_btn; } return 0; } return this; })(); /* ██╗ ██╗███████╗ █████╗ ██████╗ ███████╗ ╚██╗ ██╔╝██╔════╝██╔══██╗██╔══██╗██╔════╝ ╚████╔╝ █████╗ ███████║██████╔╝███████╗ ╚██╔╝ ██╔══╝ ██╔══██║██╔══██╗╚════██║ ██║ ███████╗██║ ██║██║ ██║███████║ ╚═╝ ╚══════╝╚═╝ ╚═╝╚═╝ ╚═╝╚══════╝ */ Planet.prototype.showYear = function ( _year, onReady ) { var scope = this; // console.log(">>>! ", this.main.controls.autoRotate ); if( PlanetData.YEAR_ID ){ this.hidePlanet( _year, onReady ); return; } PlanetData.YEAR_ID = _year; PlanetData.YEAR = PlanetData.years[ _year ]; if( PlanetData.YEAR.earth_image ){ show(); }else{ PlanetData.YEAR.earth_image = this.projectiveImage = new ProjectiveImage( PlanetData.YEAR.earth_url, show ); } $(".timeline").css( "pointer-events", "none" ); function show () { var t = 2.25; scope.main.animateIn( t, function(){ scope.state = scope.IDLE; scope.commentToggle( true ); $(".timeline").css( "pointer-events", "auto" ); }); t = t/3*2; scope.planetLocations.show( _year, t ); scope.planetPointed.show( _year, t ); scope.planetContour.show( _year, t ); if( onReady ) onReady(); // show commentary tip if( scope._PlanetCommentPopup ) scope._PlanetCommentPopup.showRandom(); } } Planet.prototype.hidePlanet = function ( _year, onReady, t ) { this.state = this.ANIMATED; var scope = this; // console.log("hidePlanet"); this.deactivateHexagon(); $(".timeline").css( "pointer-events", "none" ); scope.commentToggle( false ); if( t===undefined ) t = 1.5; this.main.animateOut( t, function(){ PlanetData.YEAR_ID = undefined; scope.showYear( _year, onReady ); // $(".timeline").css( "pointer-events", "auto" ); }); scope.planetLocations.hide( t/2 ); scope.planetPointed.hide( t/2 ); scope.planetContour.hide( t/2 ); if( this._PlanetCommentPopup ) this._PlanetCommentPopup.hide(); } THREE.Text = function ( params ) { // params.text // params.font_propotion // params.font // params.width // params.height // params.transparent // params.offset_x // params.offset_y // params.multiplier // params.fillStyle // params.debug var scope = this; scope.offset_x = params.offset_x || 0; scope.border_x = params.border_x || 0; scope.offset_y = params.offset_y || 0; scope.font = params.font || "Arial"; scope.font_propotion = params.font_propotion || 20; // scope.multiplier = params.multiplier || 1; scope.fillStyle = params.fillStyle || "rgba(255,0,0, 0.95)"; scope.fontStyle = params.fontStyle || ""; // scope.pow_coef_x = 1; var canvas = scope.canvas = document.createElement('canvas'); var context = scope.context = canvas.getContext('2d'); if( params.debug ){ document.body.appendChild( canvas ); // $(canvas).css({ position: "absolute", top: 0, left: 0 }); canvas.style.position = "absolute"; canvas.style.top = 0; canvas.style.left = 0; } var texture = new THREE.Texture(canvas) texture.needsUpdate = true; texture.generateMipmaps = false; texture.magFilter = THREE.LinearFilter; texture.minFilter = THREE.LinearFilter; texture.wrapS = THREE.ClampToEdgeWrapping; // texture.repeat.x = - 1; texture.needsUpdate = true; var material = new THREE.MeshBasicMaterial( { map: texture } ); material.transparent = params.transparent || false; material.side = params.side==undefined ? THREE.DoubleSide : params.side; material.depthWrite = !true; // material.wireframe = true; var g = new THREE.PlaneGeometry( 1, 1 ); var m = new THREE.Matrix4(); m.setPosition( new THREE.Vector3( .5, .5, 0 ) ); g.applyMatrix(m); // var material_test = new THREE.MeshBasicMaterial( { color: 0xff0000 } ); var mesh = scope.mesh = new THREE.Mesh( g, material ); mesh.matrixAutoUpdate = false; mesh.visible = false; mesh.setSize = function ( width, height) { // console.log("setSize: ", width); mesh.width = width; mesh.height = height; var w = mesh.canvas_vis_width = Math.floor( width * 100 ); var h = mesh.canvas_vis_height = Math.floor( height * 100 ); var wc = Utils.nearestPow2( w ); var hc = Utils.nearestPow2( h ); canvas.width = wc; canvas.height = hc; texture.repeat.x = mesh.pow_coef_x = w / wc; texture.repeat.y = mesh.pow_coef_y = h / hc; texture.offset.y = ( 1 - texture.repeat.y ); // console.log( "setSize>> W:", width, w, wc, mesh.pow_coef_x ); // console.log( "setSize>> H:", height, h, hc, mesh.pow_coef_y ); mesh.scale.set( width, height, 1 ); mesh.updateMatrix(); } mesh.setWidth = function ( text ) { var w; if( typeof text == "string" ){// get width from text setupText(); w = ( context.measureText( text ).width + scope.offset_x + scope.border_x ) / 100; // console.log("setWidth: ", text, w ); // console.log(">: ", scope.offset_x, scope.border_x ); }else w = text; mesh.setSize( w, params.height ); return w; } mesh.setText = function ( text, width ) { // console.log("\n"); // console.log("setText: ", scope.offset_x ); mesh.setWidth( width || text ); context.clearRect( 0, 0, canvas.width, canvas.height ); if( mesh.prefill ) mesh.prefill.apply( scope ); // console.log( ">>> ", context.font ); // debug !!! // context.fillStyle = "#FF0000"; // // context.fillRect( 0, 0, mesh.canvas_vis_width, mesh.canvas_vis_height ); // context.fillRect( 0, 0, canvas.width, canvas.height ); // debug !!! setupText(); // context.fillStyle = "#000000"; context.fillText( text, scope.offset_x, mesh.canvas_vis_height / 2 ); texture.needsUpdate = true; } function setupText () { scope.font_height = scope.font_propotion * mesh.canvas_vis_height; context.font = scope.fontStyle +" "+ ( scope.font_height ) + 'px '+scope.font;//"Bold 40px Arial"; context.fillStyle = scope.fillStyle; context.textBaseline = "middle"; } // mesh.setText( "init" ); mesh.show = function( _delay ){ mesh.scale.x = .001; mesh.matrixAutoUpdate = true; TweenLite.killTweensOf( mesh.scale ); TweenLite.to( mesh.scale, .2, { delay: _delay, x: Math.floor( mesh.width ), ease: Sine.easeOut, onStart: function() { mesh.visible = true; }, onComplete: function() { mesh.updateMatrix(); mesh.matrixAutoUpdate = false; } }); } mesh.hide = function( _delay ){ if( !mesh.visible ) return; mesh.scale.x = 0; mesh.matrixAutoUpdate = true; TweenLite.killTweensOf( mesh.scale ); TweenLite.to( mesh.scale, .2, { delay: _delay, x: .001, ease: Sine.easeIn, onComplete: function() { mesh.visible = false; mesh.updateMatrix(); mesh.matrixAutoUpdate = false; } }); } return mesh; } // IMAGE function ProjectiveImage(_img_url, _onload) { var projectionContext; var img = document.createElement("img"); img.src = _img_url; img.onload = function(){ var projectionCanvas = document.createElement('canvas'); projectionContext = projectionCanvas.getContext('2d'); projectionCanvas.width = img.width; projectionCanvas.height = img.height; projectionContext.drawImage(img, 0, 0, img.width, img.height); if(_onload) _onload(); } var pixelData = null; var maxLat = -100; var maxLon = 0; var minLat = 0; var minLon = 0; this.isLand = function (lat, lon) { var x = parseInt(img.width * ( lon + 180 ) / 360); var y = parseInt(img.height * ( lat + 90 ) / 180); if(pixelData == null){ pixelData = projectionContext.getImageData(0,0,img.width, img.height); } return pixelData.data[(y * pixelData.width + x) * 4] === 0; }; this.isLandByUV = function (u, v) { if(pixelData == null){ pixelData = projectionContext.getImageData(0,0,img.width, img.height); } var x = parseInt(img.width * u ); var y = parseInt(img.height * v ); return pixelData.data[(y * pixelData.width + x) * 4] === 0; }; this.getUV = function ( lat, lon ) { return { x: ( lon + 180 ) / 360, y: ( lat + 90 ) / 180 } } this.getUVOfPI = function ( lat, lon ) { return { x: ( lon + Math.PI ) / Utils.PI2, y: ( lat + Utils.PIh ) / Math.PI } } } var Utils = new function(){ this.getHalfPoint = function (p1, p2){ var p1 = new THREE.Vector3(p1.x,p1.y,p1.z); var p2 = new THREE.Vector3(p2.x,p2.y,p2.z); var b2 = p2.sub(p1).divideScalar(2).add(p1); return b2; } this.getRandSides = function (value) { return Math.random()*value*2-value; } this.PIh = Math.PI/2; this.PI2 = Math.PI*2; THREE.Vector3.ZERO = new THREE.Vector3(); this.createHexagon = function ( h, orient, material, ring_width ) { var g = ring_width ? new THREE.RingGeometry( h, h + ring_width, 32, 1 ) : new THREE.CircleGeometry( h, 32 ); if( orient ){ var m = new THREE.Matrix4(); m.identity().makeRotationX( Math.PI/2 ); g.applyMatrix(m); m.identity().makeRotationY( Utils.PI2/6/2 ); g.applyMatrix(m); } if(!material) var material = new THREE.MeshBasicMaterial( { color: 0xffffff } ); material.side = THREE.BackSide; var mesh = new THREE.Mesh( g, material ); mesh.matrixAutoUpdate = false; return mesh; } this.orientHexagon = function( mesh, t, look_at_corner ) { var up_coef = 10; mesh.position.copy( t.centerPoint ); // mesh.scale.set( t.width, t.width, t.width ); // mesh.scale.set( .001, .001, .001 ); mesh.up.set( t.centerPoint.x*up_coef, t.centerPoint.y*up_coef, t.centerPoint.z*up_coef ); if( look_at_corner ) mesh.lookAt( t.boundary[0] ); // mesh.updateMatrix(); } this.nearestPow2 = function( aSize ){ return Math.pow( 2, Math.ceil( Math.log( aSize ) / Math.log( 2 ) ) ); } this.setFromSpherical = function ( radius, u, v, vec3 ) { var spherical = new THREE.Spherical(); spherical.radius = radius; spherical.theta = u * Math.PI *2 - Math.PI/2; spherical.phi = v * Math.PI; if( !vec3 ) vec3 = new THREE.Vector3(); vec3.setFromSpherical( spherical ); return vec3; } } Utils.getEventCursorPosition = function( e, mouse_obj ){ var pos; var touches = e.originalEvent.touches; if(touches) pos = touches.length ? touches[0] : e.originalEvent.changedTouches[0]; else pos = e; if( mouse_obj ){ mouse_obj.x = pos.clientX; mouse_obj.y = pos.clientY; return; } return { x : pos.clientX, y : pos.clientY} ; } Utils.getEventCursorPositionOrientation = function( e, mouse_obj ){ Util.getEventCursorPosition( e, mouse_obj ); mouse_obj.x = ( mouse_obj.x / window.innerWidth ) * 2 - 1; mouse_obj.y = - ( mouse_obj.y / window.innerHeight ) * 2 + 1; } Utils.get2dPosition = function( position, camera, dom_element ){ var v = position.clone().project( camera ); v.x = (v.x + 1) / 2 * dom_element.innerWidth; v.y = -(v.y - 1) / 2 * dom_element.innerHeight;// - window.innerHeight * .08; return v; } Utils.drawFunction = function( start, end, foo, step, amplitude ){ console.log( '====================================' ); start = start==undefined ? -Math.PI : start; end = end==undefined ? Math.PI : end; step = step==undefined ? .1 : step; amplitude = amplitude==undefined ? 30 : amplitude; foo = foo==undefined ? function( v ){ return Math.cos(v) } : foo; var count = 0; for (var y = start ; y < end; y+= step ) { var str = ''; for (var i = 0; i < amplitude*2; i++) { str += i==amplitude ? ':' : "-"; }; var p = foo( y ); var x = Math.round( p * amplitude ) + amplitude; str = str.substr(0, x) + "*" + str.substr(x+1); console.log( count+" |"+ str, x ); count++; if(count>=10) count = 0; } console.log( '====================================' ); } // FAKE EXTERNAL METHODS /* var kaspersky = kaspersky || {}; (function(){ var $DOMElement = $("
").appendTo( $("html") ); if( !kaspersky.getHexaspherePopup ) kaspersky.getHexaspherePopup = function( data, onClick ){ console.log( "getHexaspherePopup> ", data ); $DOMElement.show(); $DOMElement.off().click( function(){ onClick( data[0] ); }); return { "element": $DOMElement[0], "close": function(){ $DOMElement.hide(); console.log( "The active hexagon close!" ); } //закрытие окна } } })(); */

 

你可能感兴趣的:(Threejs)