Permalink
Switch branches/tags
Nothing to show
Find file Copy path
Fetching contributors…
Cannot retrieve contributors at this time
836 lines (551 sloc) 39.2 KB
<!doctype html>
<html>
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, user-scalable=no, minimum-scale=1.0, maximum-scale=1.0">
<title>Google Gravity | PhysicsJS</title>
<style>
body {
margin: 0;
overflow: hidden;
background-color: #ffffff;
font-family: Arial, sans-serif;
user-select: none;
-webkit-user-select: none;
-moz-user-select: none;
-o-user-select: none;
-ms-user-select: none;
}
button {
height: 29px;
padding: 0 8px 1px 8px;
color: #444;
background-color: #f4f4f4;
border: 1px solid #d9d9d9;
border-radius: 2px;
font-family: Arial, sans-serif;
font-size: 11px;
font-weight: bold;
text-align: center;
cursor: pointer;
}
button:hover {
color: #222;
background-color: #f8f8f8;
border: 1px solid #c6c6c6;
-webkit-box-shadow: 0 1px 0 rgba(0,0,0,.15);
box-shadow: 0 1px 0 rgba(0,0,0,.15);
}
#header {
position: absolute;
top: 15px;
right: 0px;
}
#header a {
color: #444;
font-size: 13px;
margin-right: 15px;
text-decoration: none;
}
#header a:hover {
text-decoration: underline;
}
#header img {
vertical-align: middle;
}
#header button {
margin-right: 15px;
}
#content {
width: 100%;
text-align: center;
margin-top: 192px;
white-space: nowrap;
}
#content #logo {
margin-bottom: 17px;
}
#content input {
width: 560px;
padding: 5px 8px;
margin-bottom: 16px;
font: 16px Arial, sans-serif;
border: 1px solid rgba(0,0,0,.15);
color: #000000;
}
#content input:hover {
border-color: rgba(0,0,0,.3);
}
#content input:focus {
outline: 0; /* this removes browser-side outline */
border-color: #4d90fe;
}
#footer {
position: absolute;
bottom: 0px;
width: 100%;
font-size: 13px;
border-spacing:0;
white-space: nowrap;
}
#footer td {
border-top: 1px solid #e4e4e4;
padding: 12px 17px;
background-color: #f2f2f2;
}
#footer td.empty {
padding: 10px 0;
}
#footer a {
color: #666;
text-decoration: none;
}
#footer a:hover {
text-decoration: underline;
}
.result {
width: 546px;
}
.result .title {
color: #2200C1;
font-size: 16px;
font-weight: normal;
}
.result .title a:active {
color: #2200C1;
}
.result .url {
color: #0e774a;
font-size: 13px;
}
.result .content {
font-size: 13px;
}
</style>
</head>
<body>
<script src="http://cdn.webglstats.com/stat.js" defer="defer" async="async"></script>
<script type="text/javascript">
var _gaq = _gaq || [];
_gaq.push(['_setAccount', 'UA-28341644-1']);
_gaq.push(['_trackPageview']);
(function() {
var ga = document.createElement('script'); ga.type = 'text/javascript'; ga.async = true;
ga.src = ('https:' == document.location.protocol ? 'https://ssl' : 'http://www') + '.google-analytics.com/ga.js';
(document.getElementsByTagName('head')[0] || document.getElementsByTagName('body')[0]).appendChild(ga);
})();
</script>
<div id="header">
<a class="physics-element" href="https://plus.google.com/?gpsrc=ogpy0&tab=wX">+You</a>
<a class="physics-element" href="https://mail.google.com/mail/?tab=wm">Gmail</a>
<a class="physics-element" href="https://www.google.com/imghp?hl=en&tab=wi&ei=ljNXUojfAqqCyAHQw4CYAQ&ved=0CAQQqi4oAg">Images</a>
<a href="http://www.google.com/intl/en/options/"><img src="" width="16" height="16" class="physics-element"></a>
<button id="btnG" class="physics-element" style="height:30px;color:#fff;font-size:13px;font-weight:bold;padding:0 12px 1px 12px;border:1px solid #4285f4;background:-webkit-linear-gradient(top,#4387fd,#4683ea);background:webkit-linear-gradient(top,#4387fd,#4683ea);">Sign in</button>
</div>
<div id="content">
<img id="logo" src="" width="269" height="95" class="physics-element"><br />
<input id="q" value="" title="Search" autocomplete="off" type="text" class="physics-element"><br />
<span style="margin: 0 4px;"><button id="btnG" class="physics-element">Google Search</button></span>
<span style="margin: 0 4px;"><button id="btnI" class="physics-element">I'm Feeling Lucky</button></span>
</div>
<table id="footer">
<tr>
<td class="physics-element empty" style="width:1%"><span>&nbsp;</span></td>
<td class="physics-element"><a href="https://www.google.com/intl/en/ads/?fg=1">Advertising</a></td>
<td class="physics-element"><a href="https://www.google.com/services/?fg=1">Business</a></td>
<td class="physics-element"><a href="https://www.google.com/intl/en/about.html?fg=1">About</a></td>
<td class="physics-element empty" style="width:10%"><span>&nbsp;</span></td>
<td class="physics-element empty" style="width:20%"><span>&nbsp;</span></td>
<td class="physics-element empty" style="width:10%"><span>&nbsp;</span></td>
<td class="physics-element empty" style="width:20%"><span>&nbsp;</span></td>
<td class="physics-element empty" style="width:10%"><span>&nbsp;</span></td>
<td class="physics-element empty" style="width:10%"><span>&nbsp;</span></td>
<td class="physics-element empty" style="width:20%"><span>&nbsp;</span></td>
<td class="physics-element"><a href="https://www.google.com/intl/en/policies/?fg=1">Privacy & Terms</a></td>
<td class="physics-element"><a href="https://www.google.com/preferences?hl=en">Settings</a></td>
<td class="physics-element empty" style="width:1%"><span>&nbsp;</span></td>
</tr>
</table>
<script src="http://www.google.com/uds/api?file=uds.js&v=0.1" type="text/javascript"></script>
<script src="http://wellcaffeinated.net/PhysicsJS/examples/physicsjs-full.js"></script>
<script>
/**
* Original code by mrdoob
* http://www.mrdoob.com/projects/chromeexperiments/google-gravity/2013.html
*
* This code modified by @wellcaffeinated http://wellcaffeinated.net
*/
var delta = [ 0, 0 ];
var stage = [ window.screenX, window.screenY, window.innerWidth, window.innerHeight ];
getBrowserDimensions();
var isRunning = false;
var isMouseDown = false;
var worldAABB;
var world;
var edgeBounce;
var iterations = 16;
var timeStep = 1000 / 260;
var constraints;
var mouseJoint;
var mouse = { x: 0, y: 0 };
var mouseOnClick = [];
var elements = [];
var bodies = [];
var properties = [];
var query, page = 0;
var gWebSearch, gImageSearch;
var imFeelingLuckyMode = false;
var resultBodies = [];
var gravity = { x: 0, y: 1 };
var gravityBehavior;
init();
if ( location.search != "" ) {
var params = location.search.substr(1).split("&")
for (var i = 0; i < params.length; i++) {
var param = params[i].split("=");
if (param[0] == "q") {
document.getElementById('q').value = param[1];
run();
break;
}
}
}
//
function init() {
gWebSearch = new google.search.WebSearch();
gWebSearch.setResultSetSize( google.search.Search.SMALL_RESULTSET );
gWebSearch.setSearchCompleteCallback( null, onWebSearch );
gImageSearch = new google.search.ImageSearch();
gImageSearch.setResultSetSize( google.search.Search.SMALL_RESULTSET );
gImageSearch.setSearchCompleteCallback( null, onImageSearch );
document.addEventListener( 'mousedown', onDocumentMouseDown, false );
document.addEventListener( 'mouseup', onDocumentMouseUp, false );
document.addEventListener( 'mousemove', onDocumentMouseMove, false );
document.addEventListener( 'keyup', onDocumentKeyUp, false );
document.addEventListener( 'touchstart', onDocumentTouchStart, false );
document.addEventListener( 'touchmove', onDocumentTouchMove, false );
document.addEventListener( 'touchend', onDocumentTouchEnd, false );
window.addEventListener( 'deviceorientation', onWindowDeviceOrientation, false );
// init PhysicsJS
worldAABB = Physics.aabb.apply(null, stage);
world = Physics({
timestep: timeStep,
maxIPF: iterations
});
// walls
edgeBounce = Physics.behavior('edge-collision-detection', {
aabb: worldAABB,
restitution: 0.4,
cof: 0.5
});
world.add( edgeBounce );
world.add( Physics.behavior('body-collision-detection', { checkAll: false }) );
world.add( Physics.behavior('sweep-prune') );
world.add( Physics.behavior('body-impulse-response') );
// constraints
constraints = Physics.behavior('verlet-constraints', {
iterations: 2
});
world.add( constraints );
// add gravity
gravityBehavior = Physics.behavior('constant-acceleration', { acc: gravity });
world.add( gravityBehavior );
// Get physics-element elements
elements = getElementsByClass("physics-element");
for ( var i = 0; i < elements.length; i ++ ) {
properties[i] = getElementProperties( elements[i] );
}
for ( var i = 0; i < elements.length; i ++ ) {
var element = elements[ i ];
element.style.position = 'absolute';
element.style.left = ( - properties[i][2]/2) + 'px'; // will be set by renderer
element.style.top = ( - properties[i][3]/2) + 'px';
element.style.width = properties[i][2] + 'px';
element.addEventListener( 'mousedown', onElementMouseDown, false );
element.addEventListener( 'mouseup', onElementMouseUp, false );
element.addEventListener( 'click', onElementClick, false );
bodies[i] = Physics.body('convex-polygon', {
x: properties[i][0] + properties[i][2]/2,
y: properties[i][1] + properties[i][3]/2,
vertices: [
{ x: 0, y: 0 },
{ x: properties[i][2], y: 0 },
{ x: properties[i][2], y: properties[i][3] },
{ x: 0, y: properties[i][3] }
]
});
bodies[i].view = element;
// Clean position dependencies
while ( element.offsetParent ) {
element = element.offsetParent;
element.style.position = 'static';
}
}
// add the bodies to the world
world.add( bodies );
// renderer
var renderer = Physics.renderer('dom', {
el: document.body,
width: window.innerWidth,
height: window.innerHeight,
});
world.add( renderer );
// position the views
world.render();
world.pause();
Physics.util.ticker.subscribe( loop );
Physics.util.ticker.start();
// magic to trigger GPU
world.subscribe('render', function( data ){
var style;
for ( var i = 0, l = data.bodies.length; i < l; ++i ){
style = data.bodies[ i ].view.style;
style.WebkitTransform += ' translateZ(0)';
style.MozTransform += ' translateZ(0)';
style.MsTransform += ' translateZ(0)';
style.transform += ' translateZ(0)';
}
});
}
function run() {
world.unpause();
}
//
function onDocumentMouseDown( event ) {
isMouseDown = true;
}
function onDocumentMouseUp( event ) {
isMouseDown = false;
}
function onDocumentMouseMove( event ) {
if ( world.isPaused() ) run();
mouse.x = event.clientX;
mouse.y = event.clientY;
}
function onDocumentKeyUp( event ) {
if ( event.keyCode == 13 ) search();
}
function onDocumentTouchStart( event ) {
if ( event.touches.length == 1 ) {
if ( !isRunning ) {
run();
}
mouse.x = event.touches[0].pageX;
mouse.y = event.touches[0].pageY;
isMouseDown = true;
}
}
function onDocumentTouchMove( event ) {
if ( event.touches.length == 1 ) {
event.preventDefault();
mouse.x = event.touches[0].pageX;
mouse.y = event.touches[0].pageY;
}
}
function onDocumentTouchEnd( event ) {
if ( event.touches.length == 0 ) {
isMouseDown = false;
}
}
function onWindowDeviceOrientation( event ) {
if ( event.beta ) {
gravity.x = Math.sin( event.gamma * Math.PI / 180 );
gravity.y = Math.sin( ( Math.PI / 4 ) + event.beta * Math.PI / 180 );
}
}
//
function onElementMouseDown( event ) {
event.preventDefault();
mouseOnClick[0] = event.clientX;
mouseOnClick[1] = event.clientY;
}
function onElementMouseUp( event ) {
event.preventDefault();
}
function onElementClick( event ) {
var range = 5;
if ( mouseOnClick[0] > event.clientX + range || mouseOnClick[0] < event.clientX - range &&
mouseOnClick[1] > event.clientY + range || mouseOnClick[1] < event.clientY - range ) {
event.preventDefault();
}
if ( event.target == document.getElementById( 'btnG' ) ) search();
if ( event.target == document.getElementById( 'btnI' ) ) imFeelingLucky();
if ( event.target == document.getElementById( 'q' ) ) document.getElementById('q').focus();
}
// API STUFF
function search() {
if ( !isRunning ) {
run();
}
if ( query == document.getElementById('q').value ) {
page ++;
gWebSearch.gotoPage( page );
gImageSearch.gotoPage( page );
} else {
page = 0;
query = document.getElementById('q').value;
gWebSearch.execute( query );
gImageSearch.execute( query );
}
return false;
}
function imFeelingLucky() {
imFeelingLuckyMode = true;
gWebSearch.execute( document.getElementById('q').value );
return false;
}
function onWebSearch() {
if ( imFeelingLuckyMode ) {
location.href = gWebSearch.results[0].unescapedUrl;
return;
}
for ( var i = 0; i < gWebSearch.results.length; i ++ ) {
addWeb( gWebSearch.results[i] );
}
}
function onImageSearch() {
for ( var i = 0; i < gImageSearch.results.length; i ++ ) {
addImage( gImageSearch.results[i] );
}
}
function addWeb( data ) {
var element = document.createElement('div');
element.innerHTML = '<div class="result"><div class="title"><a href="' + data.unescapedUrl + '" target="_blank">' + data.title + '</a></div><div class="url">' + data.visibleUrl + '</div><div class="content">' + data.content + '</div>';
document.body.appendChild( element );
properties.push( [ Math.random() * ( window.innerWidth / 2 ), - 200, 546, element.offsetHeight ] );
var i = properties.length - 1;
element.style.position = 'absolute';
element.style.left = 0 + 'px';
element.style.top = - 100 + 'px';
element.style.backgroundColor = '#ffffff';
element.addEventListener( 'mousedown', onElementMouseDown, false );
element.addEventListener( 'mouseup', onElementMouseUp, false );
element.addEventListener( 'click', onElementClick, false );
elements[i] = element;
resultBodies.push( bodies[i] = createBox( world, properties[i][0] + ( properties[i][2] >> 1 ), properties[i][1] + ( properties[i][3] >> 1 ), properties[i][2] / 2, properties[i][3] / 2, false, element ) );
}
function addImage( data ) {
var element = document.createElement( 'img' );
element.style.display = 'none';
element.style.cursor = 'pointer';
element.addEventListener( 'load', function () {
properties.push( [ Math.random() * ( window.innerWidth / 2 ), - 200, element.width, element.height ] );
var i = properties.length - 1;
element.style.display = 'block';
element.style.position = 'absolute';
element.style.left = 0 + 'px';
element.style.top = - 200 + 'px';
element.style.backgroundColor = '#ffffff';
element.addEventListener( 'mousedown', onElementMouseDown, false );
element.addEventListener( 'mouseup', onElementMouseUp, false );
element.addEventListener( 'click', onElementClick, false );
element.addEventListener( 'click', function ( event ) {
var range = 5;
if ( mouseOnClick[0] < event.clientX + range && mouseOnClick[0] > event.clientX - range &&
mouseOnClick[1] < event.clientY + range && mouseOnClick[1] > event.clientY - range ) {
window.open( data.unescapedUrl );
}
}, false );
elements[i] = element;
resultBodies.push( bodies[i] = createBox( world, properties[i][0] + ( properties[i][2] >> 1 ), properties[i][1] + ( properties[i][3] >> 1 ), properties[i][2] / 2, properties[i][3] / 2, false, element ) );
}, false );
element.src = data.tbUrl;
document.body.appendChild( element );
}
//
function loop( time ) {
if (getBrowserDimensions())
setWalls();
delta[0] += (0 - delta[0]) * .5;
delta[1] += (0 - delta[1]) * .5;
gravityBehavior.setAcceleration({
x: gravity.x * 5e-4 + delta[0],
y: gravity.y * 5e-4 + delta[1]
});
mouseDrag();
world.step( time );
// only render if not paused
if ( !world.isPaused() ){
world.render();
}
}
function mouseDrag() {
// mouse press
if (isMouseDown && !mouseJoint) {
var body = getBodyAtMouse();
if (body) {
var md = Physics.body('point', {
x: mouse.x,
y: mouse.y
});
mouseJoint = constraints.distanceConstraint(md, body, 0.2);
}
}
// mouse release
if (!isMouseDown) {
if (mouseJoint) {
constraints.remove( mouseJoint );
mouseJoint = null;
}
}
// mouse move
if (mouseJoint) {
mouseJoint.bodyA.state.pos.set(mouse.x, mouse.y);
}
}
function getBodyAtMouse() {
return world.findOne({ $at: Physics.vector(mouse.x, mouse.y) });
}
function setWalls() {
worldAABB = Physics.aabb.apply(null, stage);
edgeBounce.setAABB( worldAABB );
}
// .. UTILS
function getElementsByClass( searchClass ) {
var classElements = [];
var els = document.getElementsByTagName('*');
var elsLen = els.length
for (i = 0, j = 0; i < elsLen; i++) {
var classes = els[i].className.split(' ');
for (k = 0; k < classes.length; k++)
if ( classes[k] == searchClass )
classElements[j++] = els[i];
}
return classElements;
}
function getElementProperties( element ) {
var x = 0;
var y = 0;
var width = element.offsetWidth;
var height = element.offsetHeight;
do {
x += element.offsetLeft;
y += element.offsetTop;
} while ( element = element.offsetParent );
return [ x, y, width, height ];
}
function getBrowserDimensions() {
var changed = false;
if ( stage[0] != window.screenX ) {
delta[0] = (window.screenX - stage[0]) * 50;
stage[0] = window.screenX;
changed = true;
}
if ( stage[1] != window.screenY ) {
delta[1] = (window.screenY - stage[1]) * 50;
stage[1] = window.screenY;
changed = true;
}
if ( stage[2] != window.innerWidth ) {
stage[2] = window.innerWidth;
changed = true;
}
if ( stage[3] != window.innerHeight ) {
stage[3] = window.innerHeight;
changed = true;
}
return changed;
}
</script>
</body>
</html>