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)");

        $("<div id='screenshot-button' style='\
        display : inline-block;\
        font-family: Tahoma;\
        font-size : 18px;\
        color: white;\
        width: auto;\
        height: auto;\
        padding:10px;\
        background-color: red;\
        opacity: .99;\
        z-index: 100;\
        position: absolute;\
        bottom: 50px;\
        left: 50%;\
        cursor: pointer;\
        '>MAKE SCREENSHOT</div>").appendTo($("body")).click(function(){
            
            planet.bg.visible = false;

            if( composer ) composer.render();
            else renderer.render( scene, camera );

            var dataURL = renderer.domElement.toDataURL();
            $("<img id='ready-screenshot' style='position:fixed;opacity:.99;z-index:999999;top:0;left:0;'>").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<min ){
            min = d;
            _location_current = l;
            break;
        }
    };
    
    return _location_current;
}


Planet.prototype.getAnyClosestLocation = function( intersection, except_locations ) {
    
    var _location_current;
    var min = 10000;
    var _locations = this.planetLocations.current_locations;

    for (var i = 0; i < _locations.length; i++) {
        
        var l = _locations[i];
        
        if( except_locations && checkExceptions( l ) ) continue;

        var d = intersection.distanceToSquared( l.position );
        if( d<min ){
            min = d;
            _location_current = l;
        }
    };
    
    function checkExceptions ( l ) {
        for (var j = 0; j < except_locations.length; j++) {
            if( except_locations[j] == l) return true;
        };
        return false;
    }

    return _location_current;
}

function Hexagon ( scene ) {
    
    var active_color = new THREE.Color( 0xffffff );
    var inactive_color = new THREE.Color( 0x507e50 );
    
    var mesh = Utils.createHexagon( .5 );
    // mesh.up = new THREE.Vector3(0,1,0);
    var material = mesh.material;
    material.transparent = true;
    material.opacity = .4;
    material.blending = THREE.AdditiveBlending;
    material.depthWrite = false;

    mesh._anim_progress = .001;
    mesh.scale.set( mesh._anim_progress, mesh._anim_progress, mesh._anim_progress );
    
    mesh.setToLocation = function ( l ) {
        // console.log("setToTile: ", l);
        mesh.location = l;

        mesh.position.copy( l.position );
        mesh.lookAt( new THREE.Vector3() );

        material.color = active_color;
        scene.add( mesh );
        mesh.matrixAutoUpdate = true;
        TweenLite.killTweensOf( mesh );
        TweenLite.to( mesh, .2, {
            _anim_progress: .8,// * .8,
            // ease: Sine.easeOut,
            onUpdate: function () {
                mesh.scale.set( mesh._anim_progress, mesh._anim_progress, mesh._anim_progress );
            },
            onComplete:function() {
                mesh.matrixAutoUpdate = false;
            }
        });
    }

    mesh.hide = function( _assets ) {
        mesh.matrixAutoUpdate = true;
        TweenLite.killTweensOf( mesh );
        TweenLite.to( mesh, 1, {
            _anim_progress: .001,
            // ease: Sine.easeIn,
            onUpdate: function () {
                mesh.scale.set( mesh._anim_progress, mesh._anim_progress, mesh._anim_progress );
            },
            onComplete: function () {
                scene.remove( mesh );
                _assets.push( mesh );
                mesh.matrixAutoUpdate = false;
            }
        });
    }
/*
    //
    mesh.setActive = function() {
    	// console.log("setActive", mesh.tile);
    	if( !mesh.tile ) return;
    	mesh.is_active = true;

    	TweenLite.killTweensOf( mesh );
    	mesh._anim_progress = 0;
        TweenLite.to( mesh, .2, {
            _anim_progress: 1,
            // ease: Sine.easeOut,
            onUpdate: function () {
            	var scl = mesh._anim_progress*1+1;
                mesh.scale.set( scl, scl, scl );
                mesh.position.copy( mesh.tile.centerPoint ).multiplyScalar( 1 + scl*.05 );
            }
        });
    }

    mesh.setInactive = function( _assets ) {
    	// _assets.push( mesh );
    	mesh.is_active = false;

    	TweenLite.killTweensOf( mesh );
    	// mesh._anim_progress = 0;
        TweenLite.to( mesh, .5, {
            _anim_progress: .01,
            // ease: Sine.easeOut,
            onUpdate: function () {
                mesh.scale.set( mesh._anim_progress, mesh._anim_progress, mesh._anim_progress );
            },
            onComplete: function () {
                scene.remove( mesh );
                _assets.push( mesh );
            }
        });
    }
*/
    return mesh;
}
function PlanetLocations ( planet ) {
    
	var data = planet.data;
	var data_locations = data.locations;
	var current_container;

	this.current_locations;// a list of locations corresponding to the current year

	var max_opacity = .8;
    var material_yellow = new THREE.MeshBasicMaterial({
    	transparent : true,
	    opacity : 0,
	    color : new THREE.Color( 0xffde00 ),
	    blending : THREE.AdditiveBlending,
	    side : THREE.DoubleSide,
	    depthWrite : false	
    });

    var material_white = new THREE.MeshBasicMaterial({
    	transparent : true,
	    opacity : 0,
	    color : new THREE.Color( 0xffffff ),
	    blending : THREE.AdditiveBlending,
	    side : THREE.DoubleSide,
	    depthWrite : false,
    });


    // !!! fake locations

    // get image links
    var f_images = [];
    for (var i = 0; i < data_locations.length; i++) {
        f_images.push( data_locations[i].preview );
    }

    

    // make base location shape
    var hex = Utils.createHexagon( .15, false, undefined );
    // additional shape - the ring
    var hex_ring = Utils.createHexagon( .2, false, undefined, .02 );
    hex.add( hex_ring );


    var r = planet.radius*1.005;
    var zero = new THREE.Vector3();

    
    // METHODS

    this.init = function ( _year ){

    	var container = new THREE.Object3D();
	    container.matrixAutoUpdate = false;

	    var _locations = container._locations = [];
	    for (var i = 0; i < data_locations.length; i++) {
	    	var l = data_locations[i];
	    	_locations.push( l );/// All Locations are available in all Years
	    }
	    var geo;
	    var geo_yellow = new THREE.Geometry();//hex.geometry.clone();
	    var geo_white = new THREE.Geometry();//hex.geometry.clone();
	    
    	for (var i = 0; i < _locations.length; i++) {
	        
			var l = _locations[i];
			if(!!l.years[_year] === false) continue;
	        var l_year_data = l.years[_year];
	        
	        // skip a point creation if the location doesn't have any briefs
	        if( !l_year_data.briefs ) continue;

	        // get shorthand to a country object of the location
	        l.country = data.countries_by_id[ l.country_id ];
	        
	        var uv = planet.googlePosToUV( l.planet_u, l.planet_v );
	        Utils.setFromSpherical( r, uv.v, uv.u, hex.position );
	        hex.lookAt( zero );
	        hex.updateMatrix();
	        hex.updateMatrixWorld();

	        geo = l_year_data.works ? geo_yellow : geo_white;
	        geo.merge( hex.geometry, hex.matrixWorld );
	        geo.merge( hex_ring.geometry, hex_ring.matrixWorld );

	        l.position = hex.position.clone();

	        // and finaly the ray
	        l.light = addLightRay( hex.position, l_year_data.works );
	        container.add( l.light );

		};
		
	    var hexs_yellow = new THREE.Mesh( geo_yellow, material_yellow );
	    container.add( hexs_yellow );

	    var hexs_white = new THREE.Mesh( geo_white, material_white );
	    container.add( hexs_white );

	    return container;
    }



    this.show = function( _year, t ){
		var year_data = PlanetData.years[_year];

		if( !year_data.locations_container ) year_data.locations_container = this.init( _year );
		current_container = year_data.locations_container;
		this.current_locations = current_container._locations;

		planet.container.add( current_container );

		_showMaterial( material_yellow, t );
		_showMaterial( material_white, t );
		showRays( t/2, t/2 );
	}

	function _showMaterial( mat, t ){

		var _anim = 0;
		var t1 = .3;

		mat.opacity = 0;
		
		TweenLite.killTweensOf( mat );
		TweenLite.to( mat, t1, {
			delay: t-t1,
			ease: Sine.easeOut
			,onUpdate:function(){
				_anim += 1;
				mat.opacity = Math.round( _anim%2 ) * max_opacity;
			}
			,onComplete: function(){
				mat.opacity = max_opacity;
			}
		});

	}
	
	this.hide = function ( t ) {
		
		_hideMaterial( material_yellow, t );
		_hideMaterial( material_white, t );
		hideRays( t/2 );
	}

	function _hideMaterial( mat, t ){
		
		var _anim = 0;

		TweenLite.killTweensOf( mat );
		TweenLite.to( mat, .3, {
			ease: Sine.easeInOut
			,onUpdate:function(){
				_anim += 1;
				mat.opacity = Math.round( _anim%2 ) * max_opacity;
			}
			,onComplete: function(){
				planet.container.remove( current_container );	
			}
		});
	}


	/*
	██████╗  █████╗ ██╗   ██╗███████╗
	██╔══██╗██╔══██╗╚██╗ ██╔╝██╔════╝
	██████╔╝███████║ ╚████╔╝ ███████╗
	██╔══██╗██╔══██║  ╚██╔╝  ╚════██║
	██║  ██║██║  ██║   ██║   ███████║
	╚═╝  ╚═╝╚═╝  ╚═╝   ╚═╝   ╚══════╝
	*/

	var _light_material = new THREE.MeshBasicMaterial({
	    transparent: true,
	    opacity: .9,
	   	blending: THREE.AdditiveBlending,
	    side: THREE.DoubleSide,
	    depthWrite: false,
	    fog: true
	});
	var m = _light_material;
    m.map = planet.textureLoader.load( PlanetData.textures_path + "lightray.jpg" );//,
	m.map.wrapT = THREE.ClampToEdgeWrapping;
   

    var _light_material_yellow = _light_material.clone();
    _light_material_yellow.map = planet.textureLoader.load( PlanetData.textures_path + "lightray_yellow.jpg" );//,

	function addLightRay( _pos, has_works ) {

		var h = Math.random()*2 + 1;
		var geometry = new THREE.PlaneBufferGeometry( .3, h, 1 );
		var plane = new THREE.Mesh( geometry, has_works ? _light_material_yellow : _light_material );
	    var m = new THREE.Matrix4();
	    m.makeRotationX( Math.PI/2 );
	    m.setPosition( new THREE.Vector3(0, 0, -h/2) );
	    geometry.applyMatrix(m);
	    // cross plane of ray
	    var plane2 = plane.clone();
	    plane.add( plane2 );
	    plane2.rotation.z = Math.PI/2;
	    plane2.matrixAutoUpdate = false;
	    plane2.updateMatrix();

	    // main ray plane
	    plane.position.copy( _pos );
	    plane.lookAt( planet.container.position );
	    plane.matrixAutoUpdate = false;
	    plane.updateMatrix();

	    return plane;
	}


	function showRays( t, d ){
		var val = _light_material.map.offset;
		TweenLite.to( val, t, {
			delay: d,
			y: 0,
			ease: Sine.easeInOut
		});
	}

	function hideRays( t ){
		var val = _light_material.map.offset;
		TweenLite.to( val, t, {
			y: -1,
			ease: Sine.easeInOut
		});
	}

	
}

/*
██████╗  ██████╗ ██╗███╗   ██╗████████╗███████╗
██╔══██╗██╔═══██╗██║████╗  ██║╚══██╔══╝██╔════╝
██████╔╝██║   ██║██║██╔██╗ ██║   ██║   ███████╗
██╔═══╝ ██║   ██║██║██║╚██╗██║   ██║   ╚════██║
██║     ╚██████╔╝██║██║ ╚████║   ██║   ███████║
╚═╝      ╚═════╝ ╚═╝╚═╝  ╚═══╝   ╚═╝   ╚══════╝
*/
// POINTS
Planet.prototype.drawPoints = function (radius) {
    this.grid_shpere = new THREE.Object3D();
    // XRAY EARTH
    var _geometry = new THREE.SphereGeometry( radius*1.1, 66, 44 );
    
    var _material = new THREE.XRayMaterial({
        map: this.textureLoader.load( PlanetData.textures_path+"clouds.jpg" ),
        alphaProportion: .5,
        color: new THREE.Color( 0xfb2f2c5 ),
        opacity: 1,
        gridOffsetSpeed: .6
    });

    var clouds_mesh = new THREE.Mesh(_geometry, _material);
    clouds_mesh.matrixAutoUpdate = false;
    this.container.add( clouds_mesh );

    var total = _geometry.vertices.length*3;
    var positions = new Float32Array( total );
    var colors = new Float32Array( total );
    var alphas = new Float32Array( total );
    var color = new THREE.Color( 0xffffff );

    for (var i = 0; i < _geometry.vertices.length; i++) {
        var v = _geometry.vertices[i];
        var ii = i*3;
        positions[ii] = v.x;
        positions[ii+1] = v.y;
        positions[ii+2] = v.z;
        colors[ii] = color.r ;
        colors[ii+1] = color.g ;
        colors[ii+2] = color.b ;
        alphas[ Math.floor(ii/3) ] = Math.random()*.3+.5;
    };
    
    
    
    var geometry = new THREE.BufferGeometry();
    geometry.addAttribute( 'position', new THREE.BufferAttribute( positions, 3 ) );
    geometry.addAttribute( 'color', new THREE.BufferAttribute( colors, 3 ) );
    geometry.addAttribute( 'alpha', new THREE.BufferAttribute( alphas, 1 ) );
    geometry.computeBoundingSphere();


    var shaderMaterial = new THREE.ShaderMaterial( THREE.AlphaColorShader );

    shaderMaterial.depthWrite = false;
    shaderMaterial.uniforms.fogNear.value = 10.0;
    shaderMaterial.uniforms.fogFar.value = 20.0;

    var points = new THREE.Points( geometry, shaderMaterial );
    points.matrixAutoUpdate = false;
    this.grid_shpere.add( points );
}



/*
██████╗  █████╗ ██████╗ ████████╗██╗ ██████╗██╗     ███████╗███████╗
██╔══██╗██╔══██╗██╔══██╗╚══██╔══╝██║██╔════╝██║     ██╔════╝██╔════╝
██████╔╝███████║██████╔╝   ██║   ██║██║     ██║     █████╗  ███████╗
██╔═══╝ ██╔══██║██╔══██╗   ██║   ██║██║     ██║     ██╔══╝  ╚════██║
██║     ██║  ██║██║  ██║   ██║   ██║╚██████╗███████╗███████╗███████║
╚═╝     ╚═╝  ╚═╝╚═╝  ╚═╝   ╚═╝   ╚═╝ ╚═════╝╚══════╝╚══════╝╚══════╝
*/

Planet.prototype.drawParticles = function ( radius ) {

    var total = 600;
    var positions = new Float32Array( total*3 );
    var color = new THREE.Color( 0xfb2f2c5 );

    var spherical = new THREE.Spherical();
    var vec3 = new THREE.Vector3();  

    // ADDITIVE POINTS
    for (var i = 0; i < total; i++) {
        var ii = i*3;

        spherical.radius = radius * ( 1 + Math.random()*.6 );
        spherical.theta = Math.random()*8;
        spherical.phi = .3 + Math.random()*2.2;
        vec3.setFromSpherical( spherical );

        positions[ii] = vec3.x;
        positions[ii+1] = vec3.y;
        positions[ii+2] = vec3.z;
    };
    
    var geometry = new THREE.BufferGeometry();
    geometry.addAttribute( 'position', new THREE.BufferAttribute( positions, 3 ) );
    // geometry.addAttribute( 'color', new THREE.BufferAttribute( colors, 3 ) );
    // geometry.addAttribute( 'alpha', new THREE.BufferAttribute( alphas, 1 ) );
    geometry.computeBoundingSphere();

    // var material = new THREE.ShaderMaterial( THREE.AlphaColorShader );
    var material = new THREE.PointsMaterial();
    material.size = .1 / this.ratio;
    material.color = color;
    material.transparent = true;
    material.opacity = .6;
    material.blending = THREE.AdditiveBlending;
    // console.log("shaderMaterial: ", shaderMaterial);
    material.depthWrite = false;
    // shaderMaterial.uniforms.fogNear.value = 0.1;
    // shaderMaterial.uniforms.fogFar.value = 20.0;

    var points = new THREE.Points( geometry, material );
    points.matrixAutoUpdate = false;

    this.particles = new THREE.Object3D();
    this.scene.add( this.particles );
    this.particles.add( points );
}


/*
██████╗ ██╗███╗   ██╗ ██████╗ 
██╔══██╗██║████╗  ██║██╔════╝ 
██████╔╝██║██╔██╗ ██║██║  ███╗
██╔══██╗██║██║╚██╗██║██║   ██║
██║  ██║██║██║ ╚████║╚██████╔╝
╚═╝  ╚═╝╚═╝╚═╝  ╚═══╝ ╚═════╝ 
*/
// 
Planet.prototype.drawRing = function ( position, radius, width, rotation, dont_orient ) {
    
    var geometry = new THREE.RingGeometry( radius, radius + width, 64, 1 );
    var m = new THREE.Matrix4();
    if(!dont_orient){
        m.makeRotationX( Math.PI/2 );
    }else{
        // m.makeRotationY( Math.PI/2 );
        m.setPosition( new THREE.Vector3(0, 0, radius*.29) );
    }
    geometry.applyMatrix(m);

    var material = new THREE.MeshBasicMaterial( { color: this.planet_color, side: THREE.DoubleSide } );
    material.transparent = true;
    material.opacity = Math.random()*.2+.1;
    material.blending = THREE.AdditiveBlending;
    material.depthWrite = false;
    // material.fog = false;
    var cylinder = new THREE.Mesh( geometry, material );    
    cylinder.rotation.set( rotation.x, rotation.y, rotation.z );
    cylinder.position.set( position.x, position.y, position.z );

    // cylinder.matrixAutoUpdate = false;

    this.container.add( cylinder ); 
    return cylinder;
}


// ORBITAS
Planet.prototype.drawOrbitas  = function() {
    // draw ORBITAS
    this.orbits = [];

    for (var i = 0; i < this.radius; i+=3) {
        // var position = new THREE.Vector3( 0, Math.random() * radius * 2 - radius , 0 );
        var position = new THREE.Vector3( 0, i  , 0 );
        var p = Math.cos( position.y / this.radius );
        // console.log( ">>>", 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 = $("<div class='planet-comment-line'>")
        .appendTo( $container )
        .hide();

    var $comment = $("<div class='planet-comment'>")
        .appendTo( $container )
        .hide();

    var $text = $("<div class='planet-comment-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 = $("<div id='active_hexagon' style='\
        position:fixed;\
        top: 0;\
        left: 0;\
        width: 100px;\
        height: 100px;\
        background: red;\
        display: none;\
    '>").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!" );
            } //закрытие окна
        }

    }

})();
*/

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章