// ==UserScript==
// @name Tank
// @namespace http://www.thenewgroup.com/gmscripts
// @description Destroy your least favorite website with this tank!
// @include *
// @copyright 2010+, The New Group (http://theNewGroup.com)
// @author Kory Paulsen
// @licence GPL version 3 or any later version; http://www.gnu.org/copyleft/gpl.html
// @version 0.2.4
// ==/UserScript==
// Jarett's update checker (https://userscripts-mirror.org/scripts/review/20145)
var SUC_script_num = 66802; // Basecamp Writeboard Sorting UserScripts script id
try{function updateCheck(forced){if ((forced) || (parseInt(GM_getValue('SUC_last_update', '0')) + 86400000 <= (new Date().getTime()))){try{GM_xmlhttpRequest({method: 'GET',url: 'https://userscripts-mirror.org/scripts/source/'+SUC_script_num+'.meta.js?'+new Date().getTime(),headers: {'Cache-Control': 'no-cache'},onload: function(resp){var local_version, remote_version, rt, script_name;rt=resp.responseText;GM_setValue('SUC_last_update', new Date().getTime()+'');remote_version=parseInt(/@uso:version\s*(.*?)\s*$/m.exec(rt)[1]);local_version=parseInt(GM_getValue('SUC_current_version', '-1'));if(local_version!=-1){script_name = (/@name\s*(.*?)\s*$/m.exec(rt))[1];GM_setValue('SUC_target_script_name', script_name);if (remote_version > local_version){if(confirm('There is an update available for the Greasemonkey script "'+script_name+'."\nWould you like to go to the install page now?')){GM_openInTab('https://userscripts-mirror.org/scripts/show/'+SUC_script_num);GM_setValue('SUC_current_version', remote_version);}}else if (forced)alert('No update is available for "'+script_name+'."');}else GM_setValue('SUC_current_version', remote_version+'');}});}catch (err){if (forced)alert('An error occurred while checking for updates:\n'+err);}}}GM_registerMenuCommand(GM_getValue('SUC_target_script_name', '???') + ' - Manual Update Check', function(){updateCheck(true);});updateCheck(false);}catch(err){}
var Tank = function()
{
var delay = 50;
var tank = document.createElement('canvas');
tank.degrees = 0;
tank.speed = 8;
tank.imgWidth = 80;
tank.imgHeight = 80;
tank.canvSize = Math.round(Math.sqrt(tank.imgWidth*tank.imgWidth+tank.imgHeight*tank.imgHeight)); //square canvas as wide as hypotenuse for complete rotation
tank.height = tank.canvSize;
tank.width = tank.canvSize;
tank.x = Math.floor(window.innerWidth / 2);
tank.y = Math.floor(window.innerHeight / 2);
tank.left = tank.x-Math.floor(tank.canvSize/2);
tank.top = tank.y-Math.floor(tank.canvSize/2);
tank.setAttribute('style', 'position:absolute; top:'+tank.top+'px; left:'+tank.left+'px; width:'+tank.canvSize+'px; height:'+tank.canvSize+'px; z-index:998;');
tank.className = 'GM_Tank_aTank';
document.body.appendChild(tank);
var tankCtx = tank.getContext('2d');
tank.img = new Image();
tank.img.src = '';
tank.img.addEventListener('load', drawTank, true);
function drawTank()
{
tankCtx.drawImage(tank.img, (tank.canvSize-tank.imgWidth)/2, (tank.canvSize-tank.imgHeight)/2);
}
var turret = document.createElement('canvas');
turret.degrees = 0;
turret.speed = 8;
turret.imgWidth = tank.imgWidth;
turret.imgHeight = tank.imgHeight;
turret.canvSize = Math.round(Math.sqrt(turret.imgWidth*turret.imgWidth+turret.imgHeight*turret.imgHeight)); //square canvas as wide as hypotenuse for complete rotation
turret.height = turret.canvSize;
turret.width = turret.canvSize;
turret.x = tank.x;
turret.y = tank.y;
turret.left = turret.x-Math.floor(turret.canvSize/2);
turret.top = turret.y-Math.floor(turret.canvSize/2);
turret.setAttribute('style', 'position:absolute; top:'+turret.top+'px; left:'+turret.left+'px; width:'+turret.canvSize+'px; height:'+turret.canvSize+'px; z-index:999;');
turret.className = 'GM_Tank_aTurret';
document.body.appendChild(turret);
var turretCtx = turret.getContext('2d');
turret.img = new Image();
turret.img.src = '';
turret.img.addEventListener('load', drawTurret, true);
function drawTurret()
{
turretCtx.drawImage(turret.img, (turret.canvSize-turret.imgWidth)/2, (turret.canvSize-turret.imgHeight)/2);
}
var mouseObj = turret;
var keybdObj = tank;
function rotate(canvas, deg){
ctx = canvas.getContext('2d');
rad = deg*Math.PI/180
ctx.clearRect(0,0,canvas.canvSize,canvas.canvSize);
ctx.save();
ctx.translate(canvas.canvSize/2,canvas.canvSize/2);
try{
ctx.rotate(rad);
}catch(e){}
ctx.drawImage(canvas.img, -canvas.imgWidth/2, -canvas.imgHeight/2, canvas.imgWidth, canvas.imgHeight);
ctx.restore();
}
function followMouse(e)
{
if (!myTank) {
document.removeEventListener('mousemove', followMouse, true);
return false;
}
mouseX = e.pageX;
mouseY = e.pageY;
offX = (mouseX-mouseObj.x);
if (offX == 0) offX = 0.01;
offY = -1*(mouseY-mouseObj.y);
deg = 90 - (180/Math.PI) * Math.atan( offY/offX );
if (offX < 0)
deg += 180;
mouseObj.degrees = deg;
rotate( mouseObj, deg );
}
function move(forward)
{
maxX = document.body.clientWidth - 15;
minX = tank.imgWidth/2;
maxY = document.body.clientHeight - 15;
minY = tank.imgHeight/2;
deg = tank.degrees;
mX = Math.round( tank.speed * Math.sin( deg*Math.PI/180 ) );
if (!forward) mX *= -1;
turret.x = tank.x += mX; if (tank.x > maxX) tank.x = maxX; else if (tank.x < minX) tank.x = minX;
turret.left = tank.left = tank.x-Math.floor(tank.canvSize/2);
turret.style.left = tank.style.left = tank.left + 'px';
mY = Math.round( tank.speed * Math.cos( deg*Math.PI/180 ) );
if (!forward) mY *= -1;
turret.y = tank.y -= mY; if (tank.y > maxY) tank.y = maxY; else if (tank.y < minY) tank.y = minY;
turret.top = tank.top = tank.y-Math.floor(tank.canvSize/2);
turret.style.top = tank.style.top = tank.top + 'px';
}
function turn(mvLeft)
{
if ( mvLeft )
{
keybdObj.degrees -= keybdObj.speed; if ( keybdObj.degrees < 0 ) keybdObj.degrees += 360;
}
else
{
keybdObj.degrees += keybdObj.speed; if ( keybdObj.degrees > 360 ) keybdObj.degrees -= 360;
}
rotate( keybdObj, keybdObj.degrees );
}
function forward() { move(true); }
function backward() { move(false); }
function left() { turn(true); }
function right() { turn(false); }
function doKey(e)
{
if (!myTank) {
document.removeEventListener('keydown', doKey, true);
return false;
}
switch( e.which )
{
case 87: // 'w' up
if(tank.go==undefined)
tank.go = setInterval(forward, delay);
break;
case 83: // 's' down
if(tank.go==undefined)
tank.go = setInterval(backward, delay);
break;
case 65: // 'a' left
if(turret.go==undefined)
turret.go = setInterval(left, delay);
break;
case 68: // 'd' right
if(turret.go==undefined)
turret.go = setInterval(right, delay);
break;
}
}
function stop(e)
{
if (!myTank) {
document.removeEventListener('keyup', stop, true);
return false;
}
switch( e.which )
{
case 87: // 'w' up
clearInterval(tank.go);
tank.go=undefined;
break;
case 83: // 's' down
clearInterval(tank.go);
tank.go=undefined;
break;
case 65: // 'a' left
clearInterval(turret.go);
turret.go=undefined;
break;
case 68: // 'd' right
clearInterval(turret.go);
turret.go=undefined;
break;
}
}
var cannon = function(x, y, deg, dist)
{
var x = x;
var y = y;
var degrees = deg;
var distance = dist;
var speed = 16;
this.imgWidth = 80;
this.imgHeight = 80;
var canvSize = Math.round(Math.sqrt(this.imgWidth*this.imgWidth+this.imgHeight*this.imgHeight)); //square canvas as wide as hypotenuse for complete rotation
var left = x-Math.floor(this.canvSize/2);
var top = y-Math.floor(this.canvSize/2);
var minX = this.imgWidth/2;
var minY = this.imgHeight/2;
var maxX = document.body.clientWidth - 15;
var maxY = document.body.clientHeight - 15;
var canvas = document.createElement('canvas');
canvas.height = canvSize;
canvas.width = canvSize;
canvas.setAttribute('style', 'position:absolute; top:'+top+'px; left:'+left+'px; width:'+canvSize+'px; height:'+canvSize+'px; z-index:997;');
canvas.className = 'GM_Tank_aCannon';
document.body.appendChild(canvas);
var ctx = canvas.getContext('2d');
ctx.beginPath();
ctx.fillStyle = '#999';
ctx.strokeStyle = '#666';
ctx.arc(canvSize/2,canvSize/2,2,0,Math.PI*2,true);
ctx.fill();
ctx.stroke();
var flying = setInterval( fly, delay );
function fly()
{
var bExp = false;
distance -= speed;
if ( distance < 0 )
{
speed += distance;
explode();
}
var mX = Math.round( speed * Math.sin( degrees*Math.PI/180 ) );
x += mX;
left = x-Math.floor(canvSize/2);
canvas.style.left = left + 'px';
var mY = Math.round( speed * Math.cos( degrees*Math.PI/180 ) );
y -= mY;
top = y-Math.floor(canvSize/2);
canvas.style.top = top + 'px';
if (x > maxX || x < minX || y > maxY || y < minY) explode();
}
function explode()
{
clearInterval(flying);
ctx.beginPath();
ctx.fillStyle = '#000';
ctx.strokeStyle = '#111';
ctx.moveTo(Math.floor(Math.random()*11),Math.floor(Math.random()*11)); // topLeft
ctx.lineTo(canvSize*.33+Math.floor(Math.random()*11)-5,Math.floor(Math.random()*31)); // topMid
ctx.lineTo(canvSize*.66+Math.floor(Math.random()*11)-5,Math.floor(Math.random()*31)); // topMid
ctx.lineTo(canvSize-Math.floor(Math.random()*11),Math.floor(Math.random()*11)); // topRight
ctx.lineTo(canvSize-Math.floor(Math.random()*31),canvSize*.33+Math.floor(Math.random()*11)-5); // midRight
ctx.lineTo(canvSize-Math.floor(Math.random()*31),canvSize*.66+Math.floor(Math.random()*11)-5); // midRight
ctx.lineTo(canvSize-Math.floor(Math.random()*11),canvSize-Math.floor(Math.random()*11)); // bottomRight
ctx.lineTo(canvSize*.66+Math.floor(Math.random()*11)-5,canvSize-Math.floor(Math.random()*31)); // bottomMid
ctx.lineTo(canvSize*.33+Math.floor(Math.random()*11)-5,canvSize-Math.floor(Math.random()*31)); // bottomMid
ctx.lineTo(Math.floor(Math.random()*11),canvSize-Math.floor(Math.random()*11)); // bottomLeft
ctx.lineTo(Math.floor(Math.random()*31),canvSize*.66+Math.floor(Math.random()*11)-5); // midRight
ctx.lineTo(Math.floor(Math.random()*31),canvSize*.33+Math.floor(Math.random()*11)-5); // midRight
ctx.closePath();
ctx.fill();
ctx.stroke();
}
}
function fire(e)
{
if (!myTank) {
document.removeEventListener('click', fire, true);
return false;
}
mouseX = e.pageX;
mouseY = e.pageY;
offX = (mouseX-turret.x);
offY = -1*(mouseY-turret.y);
distance = Math.round(Math.sqrt(offX*offX+offY*offY));
var boom = new cannon( turret.x, turret.y, turret.degrees, distance );
}
document.addEventListener('mousemove', followMouse, true);
document.addEventListener('keydown', doKey, true);
document.addEventListener('keyup', stop, true);
document.addEventListener('click', fire, true);
}
var TankMenu = function()
{
var menuDiv = document.createElement('div');
menuDiv.setAttribute('style', 'position:absolute; top:0px; right:0px; height:16px; z-index:1000; cursor:pointer; margin-top:2px;');
var menuIcon = document.createElement('canvas');
menuIcon.setAttribute('height', '16');
menuIcon.setAttribute('width', '16');
menuDiv.appendChild(menuIcon);
document.body.appendChild(menuDiv);
var menuIconCtx = menuIcon.getContext('2d');
var menuIconImg = new Image();
menuIconImg.src = '';
menuIconImg.addEventListener('load', drawMenuIcon, true);
function drawMenuIcon() { menuIconCtx.drawImage(menuIconImg, 0, 0); }
menuIcon.addEventListener('click', newTank, true);
}
var menu = new TankMenu();
var myTank;
function newTank() {
if (!myTank)
myTank = new Tank();
else
{
// remove holes
var rmCannons = document.evaluate("//*[@class='GM_Tank_aCannon']",
document, null, XPathResult.UNORDERED_NODE_SNAPSHOT_TYPE, null);
for (var i = rmCannons.snapshotLength - 1; i >= 0; i--)
{
var elm = rmCannons.snapshotItem(i);
document.body.removeChild( elm );
}
var rmTank = document.evaluate("//*[@class='GM_Tank_aTank']",
document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null ).singleNodeValue;
document.body.removeChild( rmTank );
var rmTurret = document.evaluate("//*[@class='GM_Tank_aTurret']",
document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null ).singleNodeValue;
document.body.removeChild( rmTurret );
myTank = null;
}
}