daily pastebin goal
43%
SHARE
TWEET

Untitled

a guest Sep 24th, 2018 72 Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
  1.  
  2. <!DOCTYPE html>
  3. <html lang="en" >
  4.  
  5. <head>
  6.  
  7.  
  8.   <title>Happy Birthday Hmway</title>
  9.  
  10.  
  11.  
  12.  
  13.       <style>
  14.       html, body {
  15.             margin: 0;
  16.             overflow: hidden;
  17.             cursor: none;
  18.         }
  19.    
  20.     /*FOCUS*/
  21. .focus {
  22.   -webkit-transition: all 1s ease;
  23.      -moz-transition: all 1s ease;
  24.        -o-transition: all 1s ease;
  25.       -ms-transition: all 1s ease;
  26.           transition: all 1s ease;
  27. }
  28.  
  29. .focus:hover {
  30.   border: 70px solid #000;
  31.   border-radius: 50%;
  32. }
  33.    
  34.     </style>
  35.  
  36.   <script>
  37.   window.console = window.console || function(t) {};
  38. </script>
  39.  
  40.  
  41.  
  42.   <script>
  43.   if (document.location.search.match(/type=embed/gi)) {
  44.     window.parent.postMessage("resize", "*");
  45.   }
  46. </script>
  47.  
  48.  
  49. </head>
  50.  
  51. <body translate="no" >
  52.  
  53.  
  54.  
  55.   <canvas id="canvas"></canvas>
  56.     <script src="//static.codepen.io/assets/common/stopExecutionOnTimeout-41c52890748cd7143004e05d3c5f786c66b19939c4500ce446314d1748483e13.js"></script>
  57.  
  58.   <script src='https://cdnjs.cloudflare.com/ajax/libs/three.js/r83/three.min.js'></script>
  59. <script>
  60. /**
  61.  * @author qiao / https://github.com/qiao
  62.  * @author mrdoob / http://mrdoob.com
  63.  * @author alteredq / http://alteredqualia.com/
  64.  * @author WestLangley / https://github.com/WestLangley
  65.  * @author erich666 / http://erichaines.com
  66.  */
  67.  
  68. // This set of controls performs orbiting, dollying (zooming), and panning.
  69. // Unlike TrackballControls, it maintains the "up" direction object.up (+Y by default).
  70. //
  71. //    Orbit - left mouse / touch: one finger move
  72. //    Zoom - middle mouse, or mousewheel / touch: two finger spread or squish
  73. //    Pan - right mouse, or arrow keys / touch: three finger swipe
  74.  
  75. THREE.OrbitControls = function ( object, domElement ) {
  76.  
  77.     this.object = object;
  78.  
  79.     this.domElement = ( domElement !== undefined ) ? domElement : document;
  80.  
  81.     // Set to false to disable this control
  82.     this.enabled = true;
  83.  
  84.     // "target" sets the location of focus, where the object orbits around
  85.     this.target = new THREE.Vector3();
  86.  
  87.     // How far you can dolly in and out ( PerspectiveCamera only )
  88.     this.minDistance = 0;
  89.     this.maxDistance = Infinity;
  90.  
  91.     // How far you can zoom in and out ( OrthographicCamera only )
  92.     this.minZoom = 0;
  93.     this.maxZoom = Infinity;
  94.  
  95.     // How far you can orbit vertically, upper and lower limits.
  96.     // Range is 0 to Math.PI radians.
  97.     this.minPolarAngle = 0; // radians
  98.     this.maxPolarAngle = Math.PI; // radians
  99.  
  100.     // How far you can orbit horizontally, upper and lower limits.
  101.     // If set, must be a sub-interval of the interval [ - Math.PI, Math.PI ].
  102.     this.minAzimuthAngle = - Infinity; // radians
  103.     this.maxAzimuthAngle = Infinity; // radians
  104.  
  105.     // Set to true to enable damping (inertia)
  106.     // If damping is enabled, you must call controls.update() in your animation loop
  107.     this.enableDamping = false;
  108.     this.dampingFactor = 0.25;
  109.  
  110.     // This option actually enables dollying in and out; left as "zoom" for backwards compatibility.
  111.     // Set to false to disable zooming
  112.     this.enableZoom = true;
  113.     this.zoomSpeed = 1.0;
  114.  
  115.     // Set to false to disable rotating
  116.     this.enableRotate = true;
  117.     this.rotateSpeed = 1.0;
  118.  
  119.     // Set to false to disable panning
  120.     this.enablePan = true;
  121.     this.keyPanSpeed = 7.0; // pixels moved per arrow key push
  122.  
  123.     // Set to true to automatically rotate around the target
  124.     // If auto-rotate is enabled, you must call controls.update() in your animation loop
  125.     this.autoRotate = false;
  126.     this.autoRotateSpeed = 2.0; // 30 seconds per round when fps is 60
  127.  
  128.     // Set to false to disable use of the keys
  129.     this.enableKeys = true;
  130.  
  131.     // The four arrow keys
  132.     this.keys = { LEFT: 37, UP: 38, RIGHT: 39, BOTTOM: 40 };
  133.  
  134.     // Mouse buttons
  135.     this.mouseButtons = { ORBIT: THREE.MOUSE.LEFT, ZOOM: THREE.MOUSE.MIDDLE, PAN: THREE.MOUSE.RIGHT };
  136.  
  137.     // for reset
  138.     this.target0 = this.target.clone();
  139.     this.position0 = this.object.position.clone();
  140.     this.zoom0 = this.object.zoom;
  141.  
  142.     //
  143.     // public methods
  144.     //
  145.  
  146.     this.getPolarAngle = function () {
  147.  
  148.         return spherical.phi;
  149.  
  150.     };
  151.  
  152.     this.getAzimuthalAngle = function () {
  153.  
  154.         return spherical.theta;
  155.  
  156.     };
  157.  
  158.     this.reset = function () {
  159.  
  160.         scope.target.copy( scope.target0 );
  161.         scope.object.position.copy( scope.position0 );
  162.         scope.object.zoom = scope.zoom0;
  163.  
  164.         scope.object.updateProjectionMatrix();
  165.         scope.dispatchEvent( changeEvent );
  166.  
  167.         scope.update();
  168.  
  169.         state = STATE.NONE;
  170.  
  171.     };
  172.  
  173.     // this method is exposed, but perhaps it would be better if we can make it private...
  174.     this.update = function () {
  175.  
  176.         var offset = new THREE.Vector3();
  177.  
  178.         // so camera.up is the orbit axis
  179.         var quat = new THREE.Quaternion().setFromUnitVectors( object.up, new THREE.Vector3( 0, 1, 0 ) );
  180.         var quatInverse = quat.clone().inverse();
  181.  
  182.         var lastPosition = new THREE.Vector3();
  183.         var lastQuaternion = new THREE.Quaternion();
  184.  
  185.         return function update() {
  186.  
  187.             var position = scope.object.position;
  188.  
  189.             offset.copy( position ).sub( scope.target );
  190.  
  191.             // rotate offset to "y-axis-is-up" space
  192.             offset.applyQuaternion( quat );
  193.  
  194.             // angle from z-axis around y-axis
  195.             spherical.setFromVector3( offset );
  196.  
  197.             if ( scope.autoRotate && state === STATE.NONE ) {
  198.  
  199.                 rotateLeft( getAutoRotationAngle() );
  200.  
  201.             }
  202.  
  203.             spherical.theta += sphericalDelta.theta;
  204.             spherical.phi += sphericalDelta.phi;
  205.  
  206.             // restrict theta to be between desired limits
  207.             spherical.theta = Math.max( scope.minAzimuthAngle, Math.min( scope.maxAzimuthAngle, spherical.theta ) );
  208.  
  209.             // restrict phi to be between desired limits
  210.             spherical.phi = Math.max( scope.minPolarAngle, Math.min( scope.maxPolarAngle, spherical.phi ) );
  211.  
  212.             spherical.makeSafe();
  213.  
  214.  
  215.             spherical.radius *= scale;
  216.  
  217.             // restrict radius to be between desired limits
  218.             spherical.radius = Math.max( scope.minDistance, Math.min( scope.maxDistance, spherical.radius ) );
  219.  
  220.             // move target to panned location
  221.             scope.target.add( panOffset );
  222.  
  223.             offset.setFromSpherical( spherical );
  224.  
  225.             // rotate offset back to "camera-up-vector-is-up" space
  226.             offset.applyQuaternion( quatInverse );
  227.  
  228.             position.copy( scope.target ).add( offset );
  229.  
  230.             scope.object.lookAt( scope.target );
  231.  
  232.             if ( scope.enableDamping === true ) {
  233.  
  234.                 sphericalDelta.theta *= ( 1 - scope.dampingFactor );
  235.                 sphericalDelta.phi *= ( 1 - scope.dampingFactor );
  236.  
  237.             } else {
  238.  
  239.                 sphericalDelta.set( 0, 0, 0 );
  240.  
  241.             }
  242.  
  243.             scale = 1;
  244.             panOffset.set( 0, 0, 0 );
  245.  
  246.             // update condition is:
  247.             // min(camera displacement, camera rotation in radians)^2 > EPS
  248.             // using small-angle approximation cos(x/2) = 1 - x^2 / 8
  249.  
  250.             if ( zoomChanged ||
  251.                 lastPosition.distanceToSquared( scope.object.position ) > EPS ||
  252.                 8 * ( 1 - lastQuaternion.dot( scope.object.quaternion ) ) > EPS ) {
  253.  
  254.                 scope.dispatchEvent( changeEvent );
  255.  
  256.                 lastPosition.copy( scope.object.position );
  257.                 lastQuaternion.copy( scope.object.quaternion );
  258.                 zoomChanged = false;
  259.  
  260.                 return true;
  261.  
  262.             }
  263.  
  264.             return false;
  265.  
  266.         };
  267.  
  268.     }();
  269.  
  270.     this.dispose = function () {
  271.  
  272.         scope.domElement.removeEventListener( 'contextmenu', onContextMenu, false );
  273.         scope.domElement.removeEventListener( 'mousedown', onMouseDown, false );
  274.         scope.domElement.removeEventListener( 'wheel', onMouseWheel, false );
  275.  
  276.         scope.domElement.removeEventListener( 'touchstart', onTouchStart, false );
  277.         scope.domElement.removeEventListener( 'touchend', onTouchEnd, false );
  278.         scope.domElement.removeEventListener( 'touchmove', onTouchMove, false );
  279.  
  280.         document.removeEventListener( 'mousemove', onMouseMove, false );
  281.         document.removeEventListener( 'mouseup', onMouseUp, false );
  282.  
  283.         window.removeEventListener( 'keydown', onKeyDown, false );
  284.  
  285.         //scope.dispatchEvent( { type: 'dispose' } ); // should this be added here?
  286.  
  287.     };
  288.  
  289.     //
  290.     // internals
  291.     //
  292.  
  293.     var scope = this;
  294.  
  295.     var changeEvent = { type: 'change' };
  296.     var startEvent = { type: 'start' };
  297.     var endEvent = { type: 'end' };
  298.  
  299.     var STATE = { NONE: - 1, ROTATE: 0, DOLLY: 1, PAN: 2, TOUCH_ROTATE: 3, TOUCH_DOLLY: 4, TOUCH_PAN: 5 };
  300.  
  301.     var state = STATE.NONE;
  302.  
  303.     var EPS = 0.000001;
  304.  
  305.     // current position in spherical coordinates
  306.     var spherical = new THREE.Spherical();
  307.     var sphericalDelta = new THREE.Spherical();
  308.  
  309.     var scale = 1;
  310.     var panOffset = new THREE.Vector3();
  311.     var zoomChanged = false;
  312.  
  313.     var rotateStart = new THREE.Vector2();
  314.     var rotateEnd = new THREE.Vector2();
  315.     var rotateDelta = new THREE.Vector2();
  316.  
  317.     var panStart = new THREE.Vector2();
  318.     var panEnd = new THREE.Vector2();
  319.     var panDelta = new THREE.Vector2();
  320.  
  321.     var dollyStart = new THREE.Vector2();
  322.     var dollyEnd = new THREE.Vector2();
  323.     var dollyDelta = new THREE.Vector2();
  324.  
  325.     function getAutoRotationAngle() {
  326.  
  327.         return 2 * Math.PI / 60 / 60 * scope.autoRotateSpeed;
  328.  
  329.     }
  330.  
  331.     function getZoomScale() {
  332.  
  333.         return Math.pow( 0.95, scope.zoomSpeed );
  334.  
  335.     }
  336.  
  337.     function rotateLeft( angle ) {
  338.  
  339.         sphericalDelta.theta -= angle;
  340.  
  341.     }
  342.  
  343.     function rotateUp( angle ) {
  344.  
  345.         sphericalDelta.phi -= angle;
  346.  
  347.     }
  348.  
  349.     var panLeft = function () {
  350.  
  351.         var v = new THREE.Vector3();
  352.  
  353.         return function panLeft( distance, objectMatrix ) {
  354.  
  355.             v.setFromMatrixColumn( objectMatrix, 0 ); // get X column of objectMatrix
  356.             v.multiplyScalar( - distance );
  357.  
  358.             panOffset.add( v );
  359.  
  360.         };
  361.  
  362.     }();
  363.  
  364.     var panUp = function () {
  365.  
  366.         var v = new THREE.Vector3();
  367.  
  368.         return function panUp( distance, objectMatrix ) {
  369.  
  370.             v.setFromMatrixColumn( objectMatrix, 1 ); // get Y column of objectMatrix
  371.             v.multiplyScalar( distance );
  372.  
  373.             panOffset.add( v );
  374.  
  375.         };
  376.  
  377.     }();
  378.  
  379.     // deltaX and deltaY are in pixels; right and down are positive
  380.     var pan = function () {
  381.  
  382.         var offset = new THREE.Vector3();
  383.  
  384.         return function pan( deltaX, deltaY ) {
  385.  
  386.             var element = scope.domElement === document ? scope.domElement.body : scope.domElement;
  387.  
  388.             if ( scope.object instanceof THREE.PerspectiveCamera ) {
  389.  
  390.                 // perspective
  391.                 var position = scope.object.position;
  392.                 offset.copy( position ).sub( scope.target );
  393.                 var targetDistance = offset.length();
  394.  
  395.                 // half of the fov is center to top of screen
  396.                 targetDistance *= Math.tan( ( scope.object.fov / 2 ) * Math.PI / 180.0 );
  397.  
  398.                 // we actually don't use screenWidth, since perspective camera is fixed to screen height
  399.                 panLeft( 2 * deltaX * targetDistance / element.clientHeight, scope.object.matrix );
  400.                 panUp( 2 * deltaY * targetDistance / element.clientHeight, scope.object.matrix );
  401.  
  402.             } else if ( scope.object instanceof THREE.OrthographicCamera ) {
  403.  
  404.                 // orthographic
  405.                 panLeft( deltaX * ( scope.object.right - scope.object.left ) / scope.object.zoom / element.clientWidth, scope.object.matrix );
  406.                 panUp( deltaY * ( scope.object.top - scope.object.bottom ) / scope.object.zoom / element.clientHeight, scope.object.matrix );
  407.  
  408.             } else {
  409.  
  410.                 // camera neither orthographic nor perspective
  411.                 console.warn( 'WARNING: OrbitControls.js encountered an unknown camera type - pan disabled.' );
  412.                 scope.enablePan = false;
  413.  
  414.             }
  415.  
  416.         };
  417.  
  418.     }();
  419.  
  420.     function dollyIn( dollyScale ) {
  421.  
  422.         if ( scope.object instanceof THREE.PerspectiveCamera ) {
  423.  
  424.             scale /= dollyScale;
  425.  
  426.         } else if ( scope.object instanceof THREE.OrthographicCamera ) {
  427.  
  428.             scope.object.zoom = Math.max( scope.minZoom, Math.min( scope.maxZoom, scope.object.zoom * dollyScale ) );
  429.             scope.object.updateProjectionMatrix();
  430.             zoomChanged = true;
  431.  
  432.         } else {
  433.  
  434.             console.warn( 'WARNING: OrbitControls.js encountered an unknown camera type - dolly/zoom disabled.' );
  435.             scope.enableZoom = false;
  436.  
  437.         }
  438.  
  439.     }
  440.  
  441.     function dollyOut( dollyScale ) {
  442.  
  443.         if ( scope.object instanceof THREE.PerspectiveCamera ) {
  444.  
  445.             scale *= dollyScale;
  446.  
  447.         } else if ( scope.object instanceof THREE.OrthographicCamera ) {
  448.  
  449.             scope.object.zoom = Math.max( scope.minZoom, Math.min( scope.maxZoom, scope.object.zoom / dollyScale ) );
  450.             scope.object.updateProjectionMatrix();
  451.             zoomChanged = true;
  452.  
  453.         } else {
  454.  
  455.             console.warn( 'WARNING: OrbitControls.js encountered an unknown camera type - dolly/zoom disabled.' );
  456.             scope.enableZoom = false;
  457.  
  458.         }
  459.  
  460.     }
  461.  
  462.     //
  463.     // event callbacks - update the object state
  464.     //
  465.  
  466.     function handleMouseDownRotate( event ) {
  467.  
  468.         //console.log( 'handleMouseDownRotate' );
  469.  
  470.         rotateStart.set( event.clientX, event.clientY );
  471.  
  472.     }
  473.  
  474.     function handleMouseDownDolly( event ) {
  475.  
  476.         //console.log( 'handleMouseDownDolly' );
  477.  
  478.         dollyStart.set( event.clientX, event.clientY );
  479.  
  480.     }
  481.  
  482.     function handleMouseDownPan( event ) {
  483.  
  484.         //console.log( 'handleMouseDownPan' );
  485.  
  486.         panStart.set( event.clientX, event.clientY );
  487.  
  488.     }
  489.  
  490.     function handleMouseMoveRotate( event ) {
  491.  
  492.         //console.log( 'handleMouseMoveRotate' );
  493.  
  494.         rotateEnd.set( event.clientX, event.clientY );
  495.         rotateDelta.subVectors( rotateEnd, rotateStart );
  496.  
  497.         var element = scope.domElement === document ? scope.domElement.body : scope.domElement;
  498.  
  499.         // rotating across whole screen goes 360 degrees around
  500.         rotateLeft( 2 * Math.PI * rotateDelta.x / element.clientWidth * scope.rotateSpeed );
  501.  
  502.         // rotating up and down along whole screen attempts to go 360, but limited to 180
  503.         rotateUp( 2 * Math.PI * rotateDelta.y / element.clientHeight * scope.rotateSpeed );
  504.  
  505.         rotateStart.copy( rotateEnd );
  506.  
  507.         scope.update();
  508.  
  509.     }
  510.  
  511.     function handleMouseMoveDolly( event ) {
  512.  
  513.         //console.log( 'handleMouseMoveDolly' );
  514.  
  515.         dollyEnd.set( event.clientX, event.clientY );
  516.  
  517.         dollyDelta.subVectors( dollyEnd, dollyStart );
  518.  
  519.         if ( dollyDelta.y > 0 ) {
  520.  
  521.             dollyIn( getZoomScale() );
  522.  
  523.         } else if ( dollyDelta.y < 0 ) {
  524.  
  525.             dollyOut( getZoomScale() );
  526.  
  527.         }
  528.  
  529.         dollyStart.copy( dollyEnd );
  530.  
  531.         scope.update();
  532.  
  533.     }
  534.  
  535.     function handleMouseMovePan( event ) {
  536.  
  537.         //console.log( 'handleMouseMovePan' );
  538.  
  539.         panEnd.set( event.clientX, event.clientY );
  540.  
  541.         panDelta.subVectors( panEnd, panStart );
  542.  
  543.         pan( panDelta.x, panDelta.y );
  544.  
  545.         panStart.copy( panEnd );
  546.  
  547.         scope.update();
  548.  
  549.     }
  550.  
  551.     function handleMouseUp( event ) {
  552.  
  553.         // console.log( 'handleMouseUp' );
  554.  
  555.     }
  556.  
  557.     function handleMouseWheel( event ) {
  558.  
  559.         // console.log( 'handleMouseWheel' );
  560.  
  561.         if ( event.deltaY < 0 ) {
  562.  
  563.             dollyOut( getZoomScale() );
  564.  
  565.         } else if ( event.deltaY > 0 ) {
  566.  
  567.             dollyIn( getZoomScale() );
  568.  
  569.         }
  570.  
  571.         scope.update();
  572.  
  573.     }
  574.  
  575.     function handleKeyDown( event ) {
  576.  
  577.         //console.log( 'handleKeyDown' );
  578.  
  579.         switch ( event.keyCode ) {
  580.  
  581.             case scope.keys.UP:
  582.                 pan( 0, scope.keyPanSpeed );
  583.                 scope.update();
  584.                 break;
  585.  
  586.             case scope.keys.BOTTOM:
  587.                 pan( 0, - scope.keyPanSpeed );
  588.                 scope.update();
  589.                 break;
  590.  
  591.             case scope.keys.LEFT:
  592.                 pan( scope.keyPanSpeed, 0 );
  593.                 scope.update();
  594.                 break;
  595.  
  596.             case scope.keys.RIGHT:
  597.                 pan( - scope.keyPanSpeed, 0 );
  598.                 scope.update();
  599.                 break;
  600.  
  601.         }
  602.  
  603.     }
  604.  
  605.     function handleTouchStartRotate( event ) {
  606.  
  607.         //console.log( 'handleTouchStartRotate' );
  608.  
  609.         rotateStart.set( event.touches[ 0 ].pageX, event.touches[ 0 ].pageY );
  610.  
  611.     }
  612.  
  613.     function handleTouchStartDolly( event ) {
  614.  
  615.         //console.log( 'handleTouchStartDolly' );
  616.  
  617.         var dx = event.touches[ 0 ].pageX - event.touches[ 1 ].pageX;
  618.         var dy = event.touches[ 0 ].pageY - event.touches[ 1 ].pageY;
  619.  
  620.         var distance = Math.sqrt( dx * dx + dy * dy );
  621.  
  622.         dollyStart.set( 0, distance );
  623.  
  624.     }
  625.  
  626.     function handleTouchStartPan( event ) {
  627.  
  628.         //console.log( 'handleTouchStartPan' );
  629.  
  630.         panStart.set( event.touches[ 0 ].pageX, event.touches[ 0 ].pageY );
  631.  
  632.     }
  633.  
  634.     function handleTouchMoveRotate( event ) {
  635.  
  636.         //console.log( 'handleTouchMoveRotate' );
  637.  
  638.         rotateEnd.set( event.touches[ 0 ].pageX, event.touches[ 0 ].pageY );
  639.         rotateDelta.subVectors( rotateEnd, rotateStart );
  640.  
  641.         var element = scope.domElement === document ? scope.domElement.body : scope.domElement;
  642.  
  643.         // rotating across whole screen goes 360 degrees around
  644.         rotateLeft( 2 * Math.PI * rotateDelta.x / element.clientWidth * scope.rotateSpeed );
  645.  
  646.         // rotating up and down along whole screen attempts to go 360, but limited to 180
  647.         rotateUp( 2 * Math.PI * rotateDelta.y / element.clientHeight * scope.rotateSpeed );
  648.  
  649.         rotateStart.copy( rotateEnd );
  650.  
  651.         scope.update();
  652.  
  653.     }
  654.  
  655.     function handleTouchMoveDolly( event ) {
  656.  
  657.         //console.log( 'handleTouchMoveDolly' );
  658.  
  659.         var dx = event.touches[ 0 ].pageX - event.touches[ 1 ].pageX;
  660.         var dy = event.touches[ 0 ].pageY - event.touches[ 1 ].pageY;
  661.  
  662.         var distance = Math.sqrt( dx * dx + dy * dy );
  663.  
  664.         dollyEnd.set( 0, distance );
  665.  
  666.         dollyDelta.subVectors( dollyEnd, dollyStart );
  667.  
  668.         if ( dollyDelta.y > 0 ) {
  669.  
  670.             dollyOut( getZoomScale() );
  671.  
  672.         } else if ( dollyDelta.y < 0 ) {
  673.  
  674.             dollyIn( getZoomScale() );
  675.  
  676.         }
  677.  
  678.         dollyStart.copy( dollyEnd );
  679.  
  680.         scope.update();
  681.  
  682.     }
  683.  
  684.     function handleTouchMovePan( event ) {
  685.  
  686.         //console.log( 'handleTouchMovePan' );
  687.  
  688.         panEnd.set( event.touches[ 0 ].pageX, event.touches[ 0 ].pageY );
  689.  
  690.         panDelta.subVectors( panEnd, panStart );
  691.  
  692.         pan( panDelta.x, panDelta.y );
  693.  
  694.         panStart.copy( panEnd );
  695.  
  696.         scope.update();
  697.  
  698.     }
  699.  
  700.     function handleTouchEnd( event ) {
  701.  
  702.         //console.log( 'handleTouchEnd' );
  703.  
  704.     }
  705.  
  706.     //
  707.     // event handlers - FSM: listen for events and reset state
  708.     //
  709.  
  710.     function onMouseDown( event ) {
  711.  
  712.         if ( scope.enabled === false ) return;
  713.  
  714.         event.preventDefault();
  715.  
  716.         if ( event.button === scope.mouseButtons.ORBIT ) {
  717.  
  718.             if ( scope.enableRotate === false ) return;
  719.  
  720.             handleMouseDownRotate( event );
  721.  
  722.             state = STATE.ROTATE;
  723.  
  724.         } else if ( event.button === scope.mouseButtons.ZOOM ) {
  725.  
  726.             if ( scope.enableZoom === false ) return;
  727.  
  728.             handleMouseDownDolly( event );
  729.  
  730.             state = STATE.DOLLY;
  731.  
  732.         } else if ( event.button === scope.mouseButtons.PAN ) {
  733.  
  734.             if ( scope.enablePan === false ) return;
  735.  
  736.             handleMouseDownPan( event );
  737.  
  738.             state = STATE.PAN;
  739.  
  740.         }
  741.  
  742.         if ( state !== STATE.NONE ) {
  743.  
  744.             document.addEventListener( 'mousemove', onMouseMove, false );
  745.             document.addEventListener( 'mouseup', onMouseUp, false );
  746.  
  747.             scope.dispatchEvent( startEvent );
  748.  
  749.         }
  750.  
  751.     }
  752.  
  753.     function onMouseMove( event ) {
  754.  
  755.         if ( scope.enabled === false ) return;
  756.  
  757.         event.preventDefault();
  758.  
  759.         if ( state === STATE.ROTATE ) {
  760.  
  761.             if ( scope.enableRotate === false ) return;
  762.  
  763.             handleMouseMoveRotate( event );
  764.  
  765.         } else if ( state === STATE.DOLLY ) {
  766.  
  767.             if ( scope.enableZoom === false ) return;
  768.  
  769.             handleMouseMoveDolly( event );
  770.  
  771.         } else if ( state === STATE.PAN ) {
  772.  
  773.             if ( scope.enablePan === false ) return;
  774.  
  775.             handleMouseMovePan( event );
  776.  
  777.         }
  778.  
  779.     }
  780.  
  781.     function onMouseUp( event ) {
  782.  
  783.         if ( scope.enabled === false ) return;
  784.  
  785.         handleMouseUp( event );
  786.  
  787.         document.removeEventListener( 'mousemove', onMouseMove, false );
  788.         document.removeEventListener( 'mouseup', onMouseUp, false );
  789.  
  790.         scope.dispatchEvent( endEvent );
  791.  
  792.         state = STATE.NONE;
  793.  
  794.     }
  795.  
  796.     function onMouseWheel( event ) {
  797.  
  798.         if ( scope.enabled === false || scope.enableZoom === false || ( state !== STATE.NONE && state !== STATE.ROTATE ) ) return;
  799.  
  800.         event.preventDefault();
  801.         event.stopPropagation();
  802.  
  803.         handleMouseWheel( event );
  804.  
  805.         scope.dispatchEvent( startEvent ); // not sure why these are here...
  806.         scope.dispatchEvent( endEvent );
  807.  
  808.     }
  809.  
  810.     function onKeyDown( event ) {
  811.  
  812.         if ( scope.enabled === false || scope.enableKeys === false || scope.enablePan === false ) return;
  813.  
  814.         handleKeyDown( event );
  815.  
  816.     }
  817.  
  818.     function onTouchStart( event ) {
  819.  
  820.         if ( scope.enabled === false ) return;
  821.  
  822.         switch ( event.touches.length ) {
  823.  
  824.             case 1: // one-fingered touch: rotate
  825.  
  826.                 if ( scope.enableRotate === false ) return;
  827.  
  828.                 handleTouchStartRotate( event );
  829.  
  830.                 state = STATE.TOUCH_ROTATE;
  831.  
  832.                 break;
  833.  
  834.             case 2: // two-fingered touch: dolly
  835.  
  836.                 if ( scope.enableZoom === false ) return;
  837.  
  838.                 handleTouchStartDolly( event );
  839.  
  840.                 state = STATE.TOUCH_DOLLY;
  841.  
  842.                 break;
  843.  
  844.             case 3: // three-fingered touch: pan
  845.  
  846.                 if ( scope.enablePan === false ) return;
  847.  
  848.                 handleTouchStartPan( event );
  849.  
  850.                 state = STATE.TOUCH_PAN;
  851.  
  852.                 break;
  853.  
  854.             default:
  855.  
  856.                 state = STATE.NONE;
  857.  
  858.         }
  859.  
  860.         if ( state !== STATE.NONE ) {
  861.  
  862.             scope.dispatchEvent( startEvent );
  863.  
  864.         }
  865.  
  866.     }
  867.  
  868.     function onTouchMove( event ) {
  869.  
  870.         if ( scope.enabled === false ) return;
  871.  
  872.         event.preventDefault();
  873.         event.stopPropagation();
  874.  
  875.         switch ( event.touches.length ) {
  876.  
  877.             case 1: // one-fingered touch: rotate
  878.  
  879.                 if ( scope.enableRotate === false ) return;
  880.                 if ( state !== STATE.TOUCH_ROTATE ) return; // is this needed?...
  881.  
  882.                 handleTouchMoveRotate( event );
  883.  
  884.                 break;
  885.  
  886.             case 2: // two-fingered touch: dolly
  887.  
  888.                 if ( scope.enableZoom === false ) return;
  889.                 if ( state !== STATE.TOUCH_DOLLY ) return; // is this needed?...
  890.  
  891.                 handleTouchMoveDolly( event );
  892.  
  893.                 break;
  894.  
  895.             case 3: // three-fingered touch: pan
  896.  
  897.                 if ( scope.enablePan === false ) return;
  898.                 if ( state !== STATE.TOUCH_PAN ) return; // is this needed?...
  899.  
  900.                 handleTouchMovePan( event );
  901.  
  902.                 break;
  903.  
  904.             default:
  905.  
  906.                 state = STATE.NONE;
  907.  
  908.         }
  909.  
  910.     }
  911.  
  912.     function onTouchEnd( event ) {
  913.  
  914.         if ( scope.enabled === false ) return;
  915.  
  916.         handleTouchEnd( event );
  917.  
  918.         scope.dispatchEvent( endEvent );
  919.  
  920.         state = STATE.NONE;
  921.  
  922.     }
  923.  
  924.     function onContextMenu( event ) {
  925.  
  926.         event.preventDefault();
  927.  
  928.     }
  929.  
  930.     //
  931.  
  932.     scope.domElement.addEventListener( 'contextmenu', onContextMenu, false );
  933.  
  934.     scope.domElement.addEventListener( 'mousedown', onMouseDown, false );
  935.     scope.domElement.addEventListener( 'wheel', onMouseWheel, false );
  936.  
  937.     scope.domElement.addEventListener( 'touchstart', onTouchStart, false );
  938.     scope.domElement.addEventListener( 'touchend', onTouchEnd, false );
  939.     scope.domElement.addEventListener( 'touchmove', onTouchMove, false );
  940.  
  941.     window.addEventListener( 'keydown', onKeyDown, false );
  942.  
  943.     // force an update at start
  944.  
  945.     this.update();
  946.  
  947. };
  948.  
  949. THREE.OrbitControls.prototype = Object.create( THREE.EventDispatcher.prototype );
  950. THREE.OrbitControls.prototype.constructor = THREE.OrbitControls;
  951.  
  952. Object.defineProperties( THREE.OrbitControls.prototype, {
  953.  
  954.     center: {
  955.  
  956.         get: function () {
  957.  
  958.             console.warn( 'THREE.OrbitControls: .center has been renamed to .target' );
  959.             return this.target;
  960.  
  961.         }
  962.  
  963.     },
  964.  
  965.     // backward compatibility
  966.  
  967.     noZoom: {
  968.  
  969.         get: function () {
  970.  
  971.             console.warn( 'THREE.OrbitControls: .noZoom has been deprecated. Use .enableZoom instead.' );
  972.             return ! this.enableZoom;
  973.  
  974.         },
  975.  
  976.         set: function ( value ) {
  977.  
  978.             console.warn( 'THREE.OrbitControls: .noZoom has been deprecated. Use .enableZoom instead.' );
  979.             this.enableZoom = ! value;
  980.  
  981.         }
  982.  
  983.     },
  984.  
  985.     noRotate: {
  986.  
  987.         get: function () {
  988.  
  989.             console.warn( 'THREE.OrbitControls: .noRotate has been deprecated. Use .enableRotate instead.' );
  990.             return ! this.enableRotate;
  991.  
  992.         },
  993.  
  994.         set: function ( value ) {
  995.  
  996.             console.warn( 'THREE.OrbitControls: .noRotate has been deprecated. Use .enableRotate instead.' );
  997.             this.enableRotate = ! value;
  998.  
  999.         }
  1000.  
  1001.     },
  1002.  
  1003.     noPan: {
  1004.  
  1005.         get: function () {
  1006.  
  1007.             console.warn( 'THREE.OrbitControls: .noPan has been deprecated. Use .enablePan instead.' );
  1008.             return ! this.enablePan;
  1009.  
  1010.         },
  1011.  
  1012.         set: function ( value ) {
  1013.  
  1014.             console.warn( 'THREE.OrbitControls: .noPan has been deprecated. Use .enablePan instead.' );
  1015.             this.enablePan = ! value;
  1016.  
  1017.         }
  1018.  
  1019.     },
  1020.  
  1021.     noKeys: {
  1022.  
  1023.         get: function () {
  1024.  
  1025.             console.warn( 'THREE.OrbitControls: .noKeys has been deprecated. Use .enableKeys instead.' );
  1026.             return ! this.enableKeys;
  1027.  
  1028.         },
  1029.  
  1030.         set: function ( value ) {
  1031.  
  1032.             console.warn( 'THREE.OrbitControls: .noKeys has been deprecated. Use .enableKeys instead.' );
  1033.             this.enableKeys = ! value;
  1034.  
  1035.         }
  1036.  
  1037.     },
  1038.  
  1039.     staticMoving: {
  1040.  
  1041.         get: function () {
  1042.  
  1043.             console.warn( 'THREE.OrbitControls: .staticMoving has been deprecated. Use .enableDamping instead.' );
  1044.             return ! this.enableDamping;
  1045.  
  1046.         },
  1047.  
  1048.         set: function ( value ) {
  1049.  
  1050.             console.warn( 'THREE.OrbitControls: .staticMoving has been deprecated. Use .enableDamping instead.' );
  1051.             this.enableDamping = ! value;
  1052.  
  1053.         }
  1054.  
  1055.     },
  1056.  
  1057.     dynamicDampingFactor: {
  1058.  
  1059.         get: function () {
  1060.  
  1061.             console.warn( 'THREE.OrbitControls: .dynamicDampingFactor has been renamed. Use .dampingFactor instead.' );
  1062.             return this.dampingFactor;
  1063.  
  1064.         },
  1065.  
  1066.         set: function ( value ) {
  1067.  
  1068.             console.warn( 'THREE.OrbitControls: .dynamicDampingFactor has been renamed. Use .dampingFactor instead.' );
  1069.             this.dampingFactor = value;
  1070.  
  1071.         }
  1072.  
  1073.     }
  1074.  
  1075. } );
  1076. </script>
  1077. <script>
  1078. /*
  1079.  * GPU Particle System
  1080.  * @author flimshaw - Charlie Hoey - http://charliehoey.com
  1081.  *
  1082.  * A simple to use, general purpose GPU system. Particles are spawn-and-forget with
  1083.  * several options available, and do not require monitoring or cleanup after spawning.
  1084.  * Because the paths of all particles are completely deterministic once spawned, the scale
  1085.  * and direction of time is also variable.
  1086.  *
  1087.  * Currently uses a static wrapping perlin noise texture for turbulence, and a small png texture for
  1088.  * particles, but adding support for a particle texture atlas or changing to a different type of turbulence
  1089.  * would be a fairly light day's work.
  1090.  *
  1091.  * Shader and javascript packing code derrived from several Stack Overflow examples.
  1092.  *
  1093.  */
  1094.  
  1095. THREE.GPUParticleSystem = function( options ) {
  1096.  
  1097.     THREE.Object3D.apply( this, arguments );
  1098.  
  1099.     options = options || {};
  1100.  
  1101.     // parse options and use defaults
  1102.  
  1103.     this.PARTICLE_COUNT = options.maxParticles || 1000000;
  1104.     this.PARTICLE_CONTAINERS = options.containerCount || 1;
  1105.  
  1106.     this.PARTICLE_NOISE_TEXTURE = options.particleNoiseTex || null;
  1107.     this.PARTICLE_SPRITE_TEXTURE = options.particleSpriteTex || null;
  1108.  
  1109.     this.PARTICLES_PER_CONTAINER = Math.ceil( this.PARTICLE_COUNT / this.PARTICLE_CONTAINERS );
  1110.     this.PARTICLE_CURSOR = 0;
  1111.     this.time = 0;
  1112.     this.particleContainers = [];
  1113.     this.rand = [];
  1114.  
  1115.     // custom vertex and fragement shader
  1116.  
  1117.     var GPUParticleShader = {
  1118.  
  1119.         vertexShader: [
  1120.  
  1121.             'uniform float uTime;',
  1122.             'uniform float uScale;',
  1123.             'uniform sampler2D tNoise;',
  1124.  
  1125.             'attribute vec3 positionStart;',
  1126.             'attribute float startTime;',
  1127.             'attribute vec3 velocity;',
  1128.             'attribute float turbulence;',
  1129.             'attribute vec3 color;',
  1130.             'attribute float size;',
  1131.             'attribute float lifeTime;',
  1132.  
  1133.             'varying vec4 vColor;',
  1134.             'varying float lifeLeft;',
  1135.  
  1136.             'void main() {',
  1137.  
  1138.             // unpack things from our attributes'
  1139.  
  1140.             '   vColor = vec4( color, 1.0 );',
  1141.  
  1142.             // convert our velocity back into a value we can use'
  1143.  
  1144.             '   vec3 newPosition;',
  1145.             '   vec3 v;',
  1146.  
  1147.             '   float timeElapsed = uTime - startTime;',
  1148.  
  1149.             '   lifeLeft = 1.0 - ( timeElapsed / lifeTime );',
  1150.  
  1151.             '   gl_PointSize = ( uScale * size ) * lifeLeft;',
  1152.  
  1153.             '   v.x = ( velocity.x - 0.5 ) * 3.0;',
  1154.             '   v.y = ( velocity.y - 0.5 ) * 3.0;',
  1155.             '   v.z = ( velocity.z - 0.5 ) * 3.0;',
  1156.  
  1157.             '   newPosition = positionStart + ( v * 10.0 ) * ( uTime - startTime );',
  1158.  
  1159.             '   vec3 noise = texture2D( tNoise, vec2( newPosition.x * 0.015 + ( uTime * 0.05 ), newPosition.y * 0.02 + ( uTime * 0.015 ) ) ).rgb;',
  1160.             '   vec3 noiseVel = ( noise.rgb - 0.5 ) * 30.0;',
  1161.  
  1162.             '   newPosition = mix( newPosition, newPosition + vec3( noiseVel * ( turbulence * 5.0 ) ), ( timeElapsed / lifeTime ) );',
  1163.  
  1164.             '   if( v.y > 0. && v.y < .05 ) {',
  1165.  
  1166.             '       lifeLeft = 0.0;',
  1167.  
  1168.             '   }',
  1169.  
  1170.             '   if( v.x < - 1.45 ) {',
  1171.  
  1172.             '       lifeLeft = 0.0;',
  1173.  
  1174.             '   }',
  1175.  
  1176.             '   if( timeElapsed > 0.0 ) {',
  1177.  
  1178.             '       gl_Position = projectionMatrix * modelViewMatrix * vec4( newPosition, 1.0 );',
  1179.  
  1180.             '   } else {',
  1181.  
  1182.             '       gl_Position = projectionMatrix * modelViewMatrix * vec4( position, 1.0 );',
  1183.             '       lifeLeft = 0.0;',
  1184.             '       gl_PointSize = 0.;',
  1185.  
  1186.             '   }',
  1187.  
  1188.             '}'
  1189.  
  1190.         ].join( '\n' ),
  1191.  
  1192.         fragmentShader: [
  1193.  
  1194.             'float scaleLinear( float value, vec2 valueDomain ) {',
  1195.  
  1196.             '   return ( value - valueDomain.x ) / ( valueDomain.y - valueDomain.x );',
  1197.  
  1198.             '}',
  1199.  
  1200.             'float scaleLinear( float value, vec2 valueDomain, vec2 valueRange ) {',
  1201.  
  1202.             '   return mix( valueRange.x, valueRange.y, scaleLinear( value, valueDomain ) );',
  1203.  
  1204.             '}',
  1205.  
  1206.             'varying vec4 vColor;',
  1207.             'varying float lifeLeft;',
  1208.  
  1209.             'uniform sampler2D tSprite;',
  1210.  
  1211.             'void main() {',
  1212.  
  1213.             '   float alpha = 0.;',
  1214.  
  1215.             '   if( lifeLeft > 0.995 ) {',
  1216.  
  1217.             '       alpha = scaleLinear( lifeLeft, vec2( 1.0, 0.995 ), vec2( 0.0, 1.0 ) );',
  1218.  
  1219.             '   } else {',
  1220.  
  1221.             '       alpha = lifeLeft * 0.75;',
  1222.  
  1223.             '   }',
  1224.  
  1225.             '   vec4 tex = texture2D( tSprite, gl_PointCoord );',
  1226.             '   gl_FragColor = vec4( vColor.rgb * tex.a, alpha * tex.a );',
  1227.  
  1228.             '}'
  1229.  
  1230.         ].join( '\n' )
  1231.  
  1232.     };
  1233.  
  1234.     // preload a million random numbers
  1235.  
  1236.     var i;
  1237.  
  1238.     for ( i = 1e5; i > 0; i-- ) {
  1239.  
  1240.         this.rand.push( Math.random() - 0.5 );
  1241.  
  1242.     }
  1243.  
  1244.     this.random = function() {
  1245.  
  1246.         return ++ i >= this.rand.length ? this.rand[ i = 1 ] : this.rand[ i ];
  1247.  
  1248.     };
  1249.  
  1250.     var textureLoader = new THREE.TextureLoader();
  1251.  
  1252. this.particleNoiseTex = this.PARTICLE_NOISE_TEXTURE || textureLoader.load( '' );
  1253.     this.particleNoiseTex.wrapS = this.particleNoiseTex.wrapT = THREE.RepeatWrapping;
  1254.  
  1255.     this.particleSpriteTex = this.PARTICLE_SPRITE_TEXTURE || textureLoader.load( '' );
  1256.     this.particleSpriteTex.wrapS = this.particleSpriteTex.wrapT = THREE.RepeatWrapping;
  1257.  
  1258.     this.particleShaderMat = new THREE.ShaderMaterial( {
  1259.         transparent: true,
  1260.         depthWrite: false,
  1261.         uniforms: {
  1262.             'uTime': {
  1263.                 value: 0.0
  1264.             },
  1265.             'uScale': {
  1266.                 value: 1.0
  1267.             },
  1268.             'tNoise': {
  1269.                 value: this.particleNoiseTex
  1270.             },
  1271.             'tSprite': {
  1272.                 value: this.particleSpriteTex
  1273.             }
  1274.         },
  1275.         blending: THREE.AdditiveBlending,
  1276.         vertexShader: GPUParticleShader.vertexShader,
  1277.         fragmentShader: GPUParticleShader.fragmentShader
  1278.     } );
  1279.  
  1280.     // define defaults for all values
  1281.  
  1282.     this.particleShaderMat.defaultAttributeValues.particlePositionsStartTime = [ 0, 0, 0, 0 ];
  1283.     this.particleShaderMat.defaultAttributeValues.particleVelColSizeLife = [ 0, 0, 0, 0 ];
  1284.  
  1285.     this.init = function() {
  1286.  
  1287.         for ( var i = 0; i < this.PARTICLE_CONTAINERS; i ++ ) {
  1288.  
  1289.             var c = new THREE.GPUParticleContainer( this.PARTICLES_PER_CONTAINER, this );
  1290.             this.particleContainers.push( c );
  1291.             this.add( c );
  1292.  
  1293.         }
  1294.  
  1295.     };
  1296.  
  1297.     this.spawnParticle = function( options ) {
  1298.  
  1299.         this.PARTICLE_CURSOR ++;
  1300.  
  1301.         if ( this.PARTICLE_CURSOR >= this.PARTICLE_COUNT ) {
  1302.  
  1303.             this.PARTICLE_CURSOR = 1;
  1304.  
  1305.         }
  1306.  
  1307.         var currentContainer = this.particleContainers[ Math.floor( this.PARTICLE_CURSOR / this.PARTICLES_PER_CONTAINER ) ];
  1308.  
  1309.         currentContainer.spawnParticle( options );
  1310.  
  1311.     };
  1312.  
  1313.     this.update = function( time ) {
  1314.  
  1315.         for ( var i = 0; i < this.PARTICLE_CONTAINERS; i ++ ) {
  1316.  
  1317.             this.particleContainers[ i ].update( time );
  1318.  
  1319.         }
  1320.  
  1321.     };
  1322.  
  1323.     this.dispose = function() {
  1324.  
  1325.         this.particleShaderMat.dispose();
  1326.         this.particleNoiseTex.dispose();
  1327.         this.particleSpriteTex.dispose();
  1328.  
  1329.         for ( var i = 0; i < this.PARTICLE_CONTAINERS; i ++ ) {
  1330.  
  1331.             this.particleContainers[ i ].dispose();
  1332.  
  1333.         }
  1334.  
  1335.     };
  1336.  
  1337.     this.init();
  1338.  
  1339. };
  1340.  
  1341. THREE.GPUParticleSystem.prototype = Object.create( THREE.Object3D.prototype );
  1342. THREE.GPUParticleSystem.prototype.constructor = THREE.GPUParticleSystem;
  1343.  
  1344.  
  1345. // Subclass for particle containers, allows for very large arrays to be spread out
  1346.  
  1347. THREE.GPUParticleContainer = function( maxParticles, particleSystem ) {
  1348.  
  1349.     THREE.Object3D.apply( this, arguments );
  1350.  
  1351.     this.PARTICLE_COUNT = maxParticles || 100000;
  1352.     this.PARTICLE_CURSOR = 0;
  1353.     this.time = 0;
  1354.     this.offset = 0;
  1355.     this.count = 0;
  1356.     this.DPR = window.devicePixelRatio;
  1357.     this.GPUParticleSystem = particleSystem;
  1358.     this.particleUpdate = false;
  1359.  
  1360.     // geometry
  1361.  
  1362.     this.particleShaderGeo = new THREE.BufferGeometry();
  1363.  
  1364.     this.particleShaderGeo.addAttribute( 'position', new THREE.BufferAttribute( new Float32Array( this.PARTICLE_COUNT * 3 ), 3 ).setDynamic( true ) );
  1365.     this.particleShaderGeo.addAttribute( 'positionStart', new THREE.BufferAttribute( new Float32Array( this.PARTICLE_COUNT * 3 ), 3 ).setDynamic( true ) );
  1366.     this.particleShaderGeo.addAttribute( 'startTime', new THREE.BufferAttribute( new Float32Array( this.PARTICLE_COUNT ), 1 ).setDynamic( true ) );
  1367.     this.particleShaderGeo.addAttribute( 'velocity', new THREE.BufferAttribute( new Float32Array( this.PARTICLE_COUNT * 3 ), 3 ).setDynamic( true ) );
  1368.     this.particleShaderGeo.addAttribute( 'turbulence', new THREE.BufferAttribute( new Float32Array( this.PARTICLE_COUNT ), 1 ).setDynamic( true ) );
  1369.     this.particleShaderGeo.addAttribute( 'color', new THREE.BufferAttribute( new Float32Array( this.PARTICLE_COUNT * 3 ), 3 ).setDynamic( true ) );
  1370.     this.particleShaderGeo.addAttribute( 'size', new THREE.BufferAttribute( new Float32Array( this.PARTICLE_COUNT ), 1 ).setDynamic( true ) );
  1371.     this.particleShaderGeo.addAttribute( 'lifeTime', new THREE.BufferAttribute( new Float32Array( this.PARTICLE_COUNT ), 1 ).setDynamic( true ) );
  1372.  
  1373.     // material
  1374.  
  1375.     this.particleShaderMat = this.GPUParticleSystem.particleShaderMat;
  1376.  
  1377.     var position = new THREE.Vector3();
  1378.     var velocity = new THREE.Vector3();
  1379.     var color = new THREE.Color();
  1380.  
  1381.     this.spawnParticle = function( options ) {
  1382.  
  1383.         var positionStartAttribute = this.particleShaderGeo.getAttribute( 'positionStart' );
  1384.         var startTimeAttribute = this.particleShaderGeo.getAttribute( 'startTime' );
  1385.         var velocityAttribute = this.particleShaderGeo.getAttribute( 'velocity' );
  1386.         var turbulenceAttribute = this.particleShaderGeo.getAttribute( 'turbulence' );
  1387.         var colorAttribute = this.particleShaderGeo.getAttribute( 'color' );
  1388.         var sizeAttribute = this.particleShaderGeo.getAttribute( 'size' );
  1389.         var lifeTimeAttribute = this.particleShaderGeo.getAttribute( 'lifeTime' );
  1390.  
  1391.         options = options || {};
  1392.  
  1393.         // setup reasonable default values for all arguments
  1394.  
  1395.         position = options.position !== undefined ? position.copy( options.position ) : position.set( 0, 0, 0 );
  1396.         velocity = options.velocity !== undefined ? velocity.copy( options.velocity ) : velocity.set( 0, 0, 0 );
  1397.         color = options.color !== undefined ? color.set( options.color ) : color.set( 0xffffff );
  1398.  
  1399.         var positionRandomness = options.positionRandomness !== undefined ? options.positionRandomness : 0;
  1400.         var velocityRandomness = options.velocityRandomness !== undefined ? options.velocityRandomness : 0;
  1401.         var colorRandomness = options.colorRandomness !== undefined ? options.colorRandomness : 1;
  1402.         var turbulence = options.turbulence !== undefined ? options.turbulence : 1;
  1403.         var lifetime = options.lifetime !== undefined ? options.lifetime : 5;
  1404.         var size = options.size !== undefined ? options.size : 10;
  1405.         var sizeRandomness = options.sizeRandomness !== undefined ? options.sizeRandomness : 0;
  1406.         var smoothPosition = options.smoothPosition !== undefined ? options.smoothPosition : false;
  1407.  
  1408.         if ( this.DPR !== undefined ) size *= this.DPR;
  1409.  
  1410.         i = this.PARTICLE_CURSOR;
  1411.  
  1412.         // position
  1413.  
  1414.         positionStartAttribute.array[ i * 3 + 0 ] = position.x + ( particleSystem.random() * positionRandomness );
  1415.         positionStartAttribute.array[ i * 3 + 1 ] = position.y + ( particleSystem.random() * positionRandomness );
  1416.         positionStartAttribute.array[ i * 3 + 2 ] = position.z + ( particleSystem.random() * positionRandomness );
  1417.  
  1418.         if ( smoothPosition === true ) {
  1419.  
  1420.             positionStartAttribute.array[ i * 3 + 0 ] += - ( velocity.x * particleSystem.random() );
  1421.             positionStartAttribute.array[ i * 3 + 1 ] += - ( velocity.y * particleSystem.random() );
  1422.             positionStartAttribute.array[ i * 3 + 2 ] += - ( velocity.z * particleSystem.random() );
  1423.  
  1424.         }
  1425.  
  1426.         // velocity
  1427.  
  1428.         var maxVel = 2;
  1429.  
  1430.         var velX = velocity.x + particleSystem.random() * velocityRandomness;
  1431.         var velY = velocity.y + particleSystem.random() * velocityRandomness;
  1432.         var velZ = velocity.z + particleSystem.random() * velocityRandomness;
  1433.  
  1434.         velX = THREE.Math.clamp( ( velX - ( - maxVel ) ) / ( maxVel - ( - maxVel ) ), 0, 1 );
  1435.         velY = THREE.Math.clamp( ( velY - ( - maxVel ) ) / ( maxVel - ( - maxVel ) ), 0, 1 );
  1436.         velZ = THREE.Math.clamp( ( velZ - ( - maxVel ) ) / ( maxVel - ( - maxVel ) ), 0, 1 );
  1437.  
  1438.         velocityAttribute.array[ i * 3 + 0 ] = velX;
  1439.         velocityAttribute.array[ i * 3 + 1 ] = velY;
  1440.         velocityAttribute.array[ i * 3 + 2 ] = velZ;
  1441.  
  1442.         // color
  1443.  
  1444.         color.r = THREE.Math.clamp( color.r + particleSystem.random() * colorRandomness, 0, 1 );
  1445.         color.g = THREE.Math.clamp( color.g + particleSystem.random() * colorRandomness, 0, 1 );
  1446.         color.b = THREE.Math.clamp( color.b + particleSystem.random() * colorRandomness, 0, 1 );
  1447.  
  1448.         colorAttribute.array[ i * 3 + 0 ] = color.r;
  1449.         colorAttribute.array[ i * 3 + 1 ] = color.g;
  1450.         colorAttribute.array[ i * 3 + 2 ] = color.b;
  1451.  
  1452.         // turbulence, size, lifetime and starttime
  1453.  
  1454.         turbulenceAttribute.array[ i ] = turbulence;
  1455.         sizeAttribute.array[ i ] = size + particleSystem.random() * sizeRandomness;
  1456.         lifeTimeAttribute.array[ i ] = lifetime;
  1457.         startTimeAttribute.array[ i ] = this.time + particleSystem.random() * 2e-2;
  1458.  
  1459.         // offset
  1460.  
  1461.         if ( this.offset === 0 ) {
  1462.  
  1463.             this.offset = this.PARTICLE_CURSOR;
  1464.  
  1465.         }
  1466.  
  1467.         // counter and cursor
  1468.  
  1469.         this.count ++;
  1470.         this.PARTICLE_CURSOR ++;
  1471.  
  1472.         if ( this.PARTICLE_CURSOR >= this.PARTICLE_COUNT ) {
  1473.  
  1474.             this.PARTICLE_CURSOR = 0;
  1475.  
  1476.         }
  1477.  
  1478.         this.particleUpdate = true;
  1479.  
  1480.     };
  1481.  
  1482.     this.init = function() {
  1483.  
  1484.         this.particleSystem = new THREE.Points( this.particleShaderGeo, this.particleShaderMat );
  1485.         this.particleSystem.frustumCulled = false;
  1486.         this.add( this.particleSystem );
  1487.  
  1488.     };
  1489.  
  1490.     this.update = function( time ) {
  1491.  
  1492.         this.time = time;
  1493.         this.particleShaderMat.uniforms.uTime.value = time;
  1494.  
  1495.         this.geometryUpdate();
  1496.  
  1497.     };
  1498.  
  1499.     this.geometryUpdate = function() {
  1500.  
  1501.         if ( this.particleUpdate === true ) {
  1502.  
  1503.             this.particleUpdate = false;
  1504.  
  1505.             var positionStartAttribute = this.particleShaderGeo.getAttribute( 'positionStart' );
  1506.             var startTimeAttribute = this.particleShaderGeo.getAttribute( 'startTime' );
  1507.             var velocityAttribute = this.particleShaderGeo.getAttribute( 'velocity' );
  1508.             var turbulenceAttribute = this.particleShaderGeo.getAttribute( 'turbulence' );
  1509.             var colorAttribute = this.particleShaderGeo.getAttribute( 'color' );
  1510.             var sizeAttribute = this.particleShaderGeo.getAttribute( 'size' );
  1511.             var lifeTimeAttribute = this.particleShaderGeo.getAttribute( 'lifeTime' );
  1512.  
  1513.             if ( this.offset + this.count < this.PARTICLE_COUNT ) {
  1514.  
  1515.                 positionStartAttribute.updateRange.offset = this.offset * positionStartAttribute.itemSize;
  1516.                 startTimeAttribute.updateRange.offset = this.offset * startTimeAttribute.itemSize;
  1517.                 velocityAttribute.updateRange.offset = this.offset * velocityAttribute.itemSize;
  1518.                 turbulenceAttribute.updateRange.offset = this.offset * turbulenceAttribute.itemSize;
  1519.                 colorAttribute.updateRange.offset = this.offset * colorAttribute.itemSize;
  1520.                 sizeAttribute.updateRange.offset = this.offset * sizeAttribute.itemSize;
  1521.                 lifeTimeAttribute.updateRange.offset = this.offset * lifeTimeAttribute.itemSize;
  1522.  
  1523.                 positionStartAttribute.updateRange.count = this.count * positionStartAttribute.itemSize;
  1524.                 startTimeAttribute.updateRange.count = this.count * startTimeAttribute.itemSize;
  1525.                 velocityAttribute.updateRange.count = this.count * velocityAttribute.itemSize;
  1526.                 turbulenceAttribute.updateRange.count = this.count * turbulenceAttribute.itemSize;
  1527.                 colorAttribute.updateRange.count = this.count * colorAttribute.itemSize;
  1528.                 sizeAttribute.updateRange.count = this.count * sizeAttribute.itemSize;
  1529.                 lifeTimeAttribute.updateRange.count = this.count * lifeTimeAttribute.itemSize;
  1530.  
  1531.             } else {
  1532.  
  1533.                 positionStartAttribute.updateRange.offset = 0;
  1534.                 startTimeAttribute.updateRange.offset = 0;
  1535.                 velocityAttribute.updateRange.offset = 0;
  1536.                 turbulenceAttribute.updateRange.offset = 0;
  1537.                 colorAttribute.updateRange.offset = 0;
  1538.                 sizeAttribute.updateRange.offset = 0;
  1539.                 lifeTimeAttribute.updateRange.offset = 0;
  1540.  
  1541.                 positionStartAttribute.updateRange.count = positionStartAttribute.count;
  1542.                 startTimeAttribute.updateRange.count = startTimeAttribute.count;
  1543.                 velocityAttribute.updateRange.count = velocityAttribute.count;
  1544.                 turbulenceAttribute.updateRange.count = turbulenceAttribute.count;
  1545.                 colorAttribute.updateRange.count = colorAttribute.count;
  1546.                 sizeAttribute.updateRange.count = sizeAttribute.count;
  1547.                 lifeTimeAttribute.updateRange.count = lifeTimeAttribute.count;
  1548.  
  1549.             }
  1550.  
  1551.             positionStartAttribute.needsUpdate = true;
  1552.             startTimeAttribute.needsUpdate = true;
  1553.             velocityAttribute.needsUpdate = true;
  1554.             turbulenceAttribute.needsUpdate = true;
  1555.             colorAttribute.needsUpdate = true;
  1556.             sizeAttribute.needsUpdate = true;
  1557.             lifeTimeAttribute.needsUpdate = true;
  1558.  
  1559.             this.offset = 0;
  1560.             this.count = 0;
  1561.  
  1562.         }
  1563.  
  1564.     };
  1565.  
  1566.     this.dispose = function() {
  1567.  
  1568.         this.particleShaderGeo.dispose();
  1569.  
  1570.     };
  1571.  
  1572.     this.init();
  1573.  
  1574. };
  1575.  
  1576. THREE.GPUParticleContainer.prototype = Object.create( THREE.Object3D.prototype );
  1577. THREE.GPUParticleContainer.prototype.constructor = THREE.GPUParticleContainer;
  1578.  
  1579. </script>
  1580.  
  1581.  
  1582.  
  1583.     <script >
  1584.       var mouse = { x: 0, y: 0 },
  1585.   positionFactor = 100,
  1586.   tick = 0,
  1587.   particleSystem,
  1588.   options = {
  1589.     spawnRate: 2000,
  1590.     maxParticles: 100000,
  1591.     particleLifetime: 1,
  1592.     turbulence: 3,
  1593.     size: 2,
  1594.     positionRandomness: 20,
  1595.     sizeRandomness: 1
  1596.   },
  1597.   mouseMoving = false,
  1598.   sizeSign = 1,
  1599.   sizeRandomness = 0.5;
  1600.  
  1601. /**
  1602.  * ====================================
  1603.  * SCENE
  1604.  * ====================================
  1605.  */
  1606.  
  1607. var scene = new THREE.Scene();
  1608. scene.background = new THREE.Color(0x000000);
  1609.  
  1610. var canvas = document.getElementById("canvas"),
  1611.   renderer = new THREE.WebGLRenderer({ canvas: canvas, alpha: false }),
  1612.   canvasWidth = document.documentElement.clientWidth,
  1613.   canvasHeight = document.documentElement.clientHeight;
  1614.  
  1615. renderer.setSize(canvasWidth, canvasHeight);
  1616.  
  1617. /**
  1618.  * ====================================
  1619.  * CAMERA
  1620.  * ====================================
  1621.  */
  1622.  
  1623. var camera = new THREE.PerspectiveCamera(
  1624.   30,
  1625.   canvasWidth / canvasHeight,
  1626.   0.1,
  1627.   10000
  1628. );
  1629. camera.position.z = 150;
  1630. scene.add(camera);
  1631.  
  1632. controls = new THREE.OrbitControls(camera, renderer.domElement);
  1633. controls.enableDamping = true;
  1634. controls.dampingFactor = 0.3;
  1635. controls.enableZoom = true;
  1636. controls.autoRotate = false;
  1637. controls.autoRotateSpeed = 0.5;
  1638. controls.minDistance = 100;
  1639. controls.maxDistance = 300;
  1640. controls.minAzimuthAngle = -0.5;
  1641. controls.maxAzimuthAngle = 0.5;
  1642.  
  1643. window.addEventListener("resize", function() {
  1644.   canvasWidth = window.innerWidth;
  1645.   canvasHeight = window.innerHeight;
  1646.   renderer.setSize(canvasWidth, canvasHeight);
  1647.   camera.aspect = canvasWidth / canvasHeight;
  1648.   camera.updateProjectionMatrix();
  1649. });
  1650.  
  1651. /*
  1652. =======================
  1653. Geometry
  1654. =======================
  1655. */
  1656.  
  1657. var geometry = new THREE.CubeGeometry(5000, 5000, 5000),
  1658.   cubeMaterials = new THREE.MultiMaterial([
  1659.     new THREE.MeshBasicMaterial({
  1660.       map: new THREE.TextureLoader().load("https://projects.thibautfoussard.com/particles2/1/textures/DarkStormyFront2048.jpg"),
  1661.       side: THREE.DoubleSide
  1662.     }),
  1663.     new THREE.MeshBasicMaterial({
  1664.       map: new THREE.TextureLoader().load("textures/DarkStormyBack2048.jpg"),
  1665.       side: THREE.DoubleSide
  1666.     }),
  1667.     new THREE.MeshBasicMaterial({
  1668.       map: new THREE.TextureLoader().load("textures/DarkStormyUp2048.jpg"),
  1669.       side: THREE.DoubleSide
  1670.     }),
  1671.     new THREE.MeshBasicMaterial({
  1672.       map: new THREE.TextureLoader().load("textures/DarkStormyDown2048.jpg"),
  1673.       side: THREE.DoubleSide
  1674.     }),
  1675.     new THREE.MeshBasicMaterial({
  1676.       map: new THREE.TextureLoader().load("textures/DarkStormyRight2048.jpg"),
  1677.       side: THREE.DoubleSide
  1678.     }),
  1679.     new THREE.MeshBasicMaterial({
  1680.       map: new THREE.TextureLoader().load("textures/DarkStormyLeft2048.jpg"),
  1681.       side: THREE.DoubleSide
  1682.     })
  1683.   ]),
  1684.   cube = new THREE.Mesh(geometry, cubeMaterials);
  1685. scene.add(cube);
  1686.  
  1687. particleSystem = new THREE.GPUParticleSystem({
  1688.   maxParticles: options.maxParticles
  1689. });
  1690. scene.add(particleSystem);
  1691.  
  1692. particleOptions = {
  1693.   velocityRandomness: 0,
  1694.   color: "rgb(255,255,255)",
  1695.   lifetime: options.particleLifetime,
  1696.   size: options.size,
  1697.   turbulence: options.turbulence,
  1698.   positionRandomness: options.positionRandomness,
  1699.   sizeRandomness: options.sizeRandomness,
  1700.   velocity: new THREE.Vector3(),
  1701.   position: new THREE.Vector3(),
  1702.   colorRandomness: 0.2,
  1703.   smoothPosition: true,
  1704.   sizeAttenuation: false
  1705. };
  1706.  
  1707. /*
  1708. =======================
  1709. Logic
  1710. =======================
  1711. */
  1712.  
  1713. camera.lookAt(scene.position);
  1714. renderer.render(scene, camera);
  1715.  
  1716. var tack = 0;
  1717. (function animate() {
  1718.   tick += 0.01;
  1719.   tack += Math.random() / 10;
  1720.   controls.update();
  1721.   particleOptions.color = "rgb(255,255,255)";
  1722.   particleOptions.lifetime = options.particleLifetime;
  1723.   particleOptions.turbulence = options.turbulence;
  1724.   particleOptions.size = options.size;
  1725.   particleOptions.positionRandomness = options.positionRandomness;
  1726.   particleOptions.sizeRandomness = changeSizeRandomness();
  1727.   if (mouseMoving) {
  1728.     particleOptions.position.x = mouse.x;
  1729.     particleOptions.position.y = mouse.y;
  1730.   } else {
  1731.     var movingFactor = 0.2;
  1732.     particleOptions.position.x += Math.cos(tack) * movingFactor;
  1733.     particleOptions.position.y += Math.sin(tack) * movingFactor;
  1734.   }
  1735.   for (var x = 0; x < options.spawnRate; x++) {
  1736.     particleSystem.spawnParticle(particleOptions);
  1737.   }
  1738.   particleSystem.update(tick);
  1739.  
  1740.   requestAnimationFrame(animate);
  1741.   renderer.render(scene, camera);
  1742. })();
  1743.  
  1744. var mouseMovingTimeout;
  1745. window.addEventListener("mousemove", function(event) {
  1746.   clearTimeout(mouseMovingTimeout);
  1747.   mouseMoving = true;
  1748.   mouseMovingTimeout = setTimeout(function() {
  1749.     mouseMoving = false;
  1750.   }, 20);
  1751.   mouse.x = (event.clientX / canvasWidth * 2 - 1) * positionFactor;
  1752.   mouse.y = (-(event.clientY / canvasHeight) * 2 + 1) * positionFactor;
  1753. });
  1754.  
  1755. function changeSizeRandomness() {
  1756.   sizeRandomness += 0.02 * sizeSign;
  1757.   if (sizeRandomness >= 10) {
  1758.     sizeSign = -1;
  1759.   }
  1760.   if (sizeRandomness <= 0.4) {
  1761.     sizeSign = 1;
  1762.   }
  1763.   return sizeRandomness;
  1764. }
  1765.       //# sourceURL=pen.js
  1766.     </script>
  1767.  
  1768.  
  1769.  
  1770.  
  1771.  
  1772.   <div style="z-index: 1000;position: absolute;top:20%;left :50%;margin: 0 0 0 -190px;">
  1773.         <div style="margin-left: auto;margin-right: auto;width: 300px;text-align: center;padding: 20px 40px 60px 40px;position: relative;">
  1774.            
  1775. <div class="selectLog" style="position: absolute;left: 0px;top: -44px;width: 380px;">
  1776.     <div id="phoneLog" onclick="show_div()"><font face='Courier'><center><div id="example1"> <h1 style=color:#FFD700;><center><b><b>&hearts; Happy Birthday &hearts;</b></b></center></h1> </div>
  1777. <div><h2><FONT color=#FF0000> Sweet 20 Baby </h2></div>
  1778. <div><h3><FONT color="green"> I'll wish you all the best. </h3></div>
  1779. </center></div>
  1780.     <div class="v-line"></div>
  1781.  
  1782. <div style="margin-top: 5px;margin-bottom: 20px;color:#ccc;font-size: 14px;font-weight: 600;"><div id="example3"><h4> <FONT color=#FFFF00>
  1783. <center>
  1784. <table width="100%">
  1785.  
  1786. <tr>
  1787. <td class="focus pic" >
  1788. <img src="https://i.imgur.com/ylS9kDu.jpg" width="100" height="100">
  1789. </td>
  1790. <td class="focus pic" >
  1791. <img src="https://i.imgur.com/VY6wNUo.jpg" width="100" height="100">
  1792. </td>
  1793. <td class="focus pic" >
  1794. <img src="https://i.imgur.com/GxwXwrR.jpg" width="100" height="100">
  1795. </td>
  1796. </tr>
  1797.  
  1798. <tr>
  1799. <td class="focus pic" >
  1800. <img src="https://i.imgur.com/g5sMQq5.jpg" width="100" height="100">
  1801. </td>
  1802. <td class="focus pic" >
  1803. <img src="https://i.imgur.com/Q7xvmE3.jpg" width="100" height="100">
  1804. </td>
  1805. <td class="focus pic" >
  1806. <img src="https://i.imgur.com/Im7F6mr.jpg" width="100" height="100">
  1807. </td>
  1808. </tr>
  1809.  
  1810. </table>
  1811. </center>
  1812. </div></div>
  1813.  
  1814.  
  1815. <div style="margin-left: 0px;margin-bottom: 5px;color:#ec3627">
  1816.    
  1817.        
  1818.            
  1819.            
  1820.        
  1821.    
  1822. </div>
  1823.  
  1824.  
  1825. </body>
  1826.  
  1827. </html>
RAW Paste Data
We use cookies for various purposes including analytics. By continuing to use Pastebin, you agree to our use of cookies as described in the Cookies Policy. OK, I Understand
 
Top