forked from erkie/erkie.github.com
-
Notifications
You must be signed in to change notification settings - Fork 0
/
asteroids.min.js
56 lines (55 loc) · 16.5 KB
/
asteroids.min.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
(function(){function Asteroids(){function Vector(x,y){if(typeof x=='Object'){this.x=x.x;this.y=x.y;}else{this.x=x;this.y=y;}};Vector.prototype={cp:function(){return new Vector(this.x,this.y);},mul:function(factor){this.x*=factor;this.y*=factor;return this;},mulNew:function(factor){return new Vector(this.x*factor,this.y*factor);},add:function(vec){this.x+=vec.x;this.y+=vec.y;return this;},addNew:function(vec){return new Vector(this.x+vec.x,this.y+vec.y);},sub:function(vec){this.x-=vec.x;this.y-=vec.y;return this;},subNew:function(vec){return new Vector(this.x-vec.x,this.y-vec.y);},rotate:function(angle){var x=this.x,y=this.y;this.x=x*Math.cos(angle)-Math.sin(angle)*y;this.y=x*Math.sin(angle)+Math.cos(angle)*y;return this;},rotateNew:function(angle){return this.cp().rotate(angle);},setAngle:function(angle){var l=this.len();this.x=Math.cos(angle)*l;this.y=Math.sin(angle)*l;return this;},setAngleNew:function(angle){return this.cp().setAngle(angle);},setLength:function(length){var l=this.len();if(l)this.mul(length/l);else this.x=this.y=length;return this;},setLengthNew:function(length){return this.cp().setLength(length);},normalize:function(){var l=this.len();this.x/=l;this.y/=l;return this;},normalizeNew:function(){return this.cp().normalize();},angle:function(){return Math.atan2(this.x,this.y);},collidesWith:function(rect){return this.x>rect.x&&this.y>rect.y&&this.x<rect.x+rect.width&&this.y<rect.y+rect.height;},len:function(){var l=Math.sqrt(this.x*this.x+this.y*this.y);if(l<0.005&&l>-0.005)return 0;return l;},is:function(test){return typeof test=='object'&&this.x==test.x&&this.y==test.y;},toString:function(){return'[Vector('+this.x+', '+this.y+') angle: '+this.angle()+', length: '+this.len()+']';}};function Line(p1,p2){this.p1=p1;this.p2=p2;};Line.prototype={shift:function(pos){this.p1.add(pos);this.p2.add(pos);},intersectsWithRect:function(rect){var LL=new Vector(rect.x,rect.y+rect.height);var UL=new Vector(rect.x,rect.y);var LR=new Vector(rect.x+rect.width,rect.y+rect.height);var UR=new Vector(rect.x+rect.width,rect.y);if(this.p1.x>LL.x&&this.p1.x<UR.x&&this.p1.y<LL.y&&this.p1.y>UR.y&&this.p2.x>LL.x&&this.p2.x<UR.x&&this.p2.y<LL.y&&this.p2.y>UR.y)return true;if(this.intersectsLine(new Line(UL,LL)))return true;if(this.intersectsLine(new Line(LL,LR)))return true;if(this.intersectsLine(new Line(UL,UR)))return true;if(this.intersectsLine(new Line(UR,LR)))return true;return false;},intersectsLine:function(line2){var v1=this.p1,v2=this.p2;var v3=line2.p1,v4=line2.p2;var denom=((v4.y-v3.y)*(v2.x-v1.x))-((v4.x-v3.x)*(v2.y-v1.y));var numerator=((v4.x-v3.x)*(v1.y-v3.y))-((v4.y-v3.y)*(v1.x-v3.x));var numerator2=((v2.x-v1.x)*(v1.y-v3.y))-((v2.y-v1.y)*(v1.x-v3.x));if(denom==0.0){return false;}
var ua=numerator/denom;var ub=numerator2/denom;return(ua>=0.0&&ua<=1.0&&ub>=0.0&&ub<=1.0);}};var that=this;var isIE=!!window.ActiveXObject;var w=document.documentElement.clientWidth,h=document.documentElement.clientHeight;var playerWidth=20,playerHeight=30;var playerVerts=[[-1*playerWidth/2,-15],[playerWidth/2,-15],[0,15]];var ignoredTypes=['HTML','HEAD','BODY','SCRIPT','TITLE','CANVAS','META','STYLE','LINK'];var hiddenTypes=['BR','HR'];var FPS=isIE?30:50;var acc=300;var maxSpeed=600;var rotSpeed=360;var bulletSpeed=700;var particleSpeed=400;var timeBetweenFire=150;var timeBetweenBlink=250;var timeBetweenEnemyUpdate=400;var bulletRadius=2;var maxParticles=isIE?10:40;var maxBullets=isIE?10:20;this.enemiesKilled=0;this.flame={r:[],y:[]};function addBlinkStyle(){addStylesheet("ASTEROIDSYEAH",".ASTEROIDSYEAHENEMY { outline: 2px dotted red; }");};this.pos=new Vector(100,100);this.lastPos=false;this.vel=new Vector(0,0);this.dir=new Vector(0,1);this.keysPressed={};this.firedAt=false;this.updated={enemies:new Date().getTime(),flame:new Date().getTime(),blink:{time:0,isActive:false}};this.scrollPos=new Vector(0,0);this.bullets=[];this.enemies=[];this.dieing=[];this.totalEnemies=0;this.particles=[];function updateEnemyIndex(){for(var i=0,enemy;enemy=that.enemies[i];i++){removeClass(enemy,"ASTEROIDSYEAHENEMY");}
var all=document.body.getElementsByTagName('*');that.enemies=[];for(var i=0;i<all.length;i++){if(indexOf(ignoredTypes,all[i].tagName)==-1&&hasOnlyTextualChildren(all[i])&&all[i].className!="ASTEROIDSYEAH"){all[i].aSize=size(all[i]);that.enemies.push(all[i]);addClass(all[i],"ASTEROIDSYEAHENEMY");if(!all[i].aAdded){all[i].aAdded=true;that.totalEnemies++;}}}};updateEnemyIndex();function createFlames(){that.flame.r=[[0,0]];that.flame.y=[[0,0]];var rWidth=playerWidth;var rIncrease=playerWidth*0.1;for(var x=0;x<rWidth;x+=rIncrease)
that.flame.r.push([x,-random(2,7)]);that.flame.r.push([rWidth,0]);var yWidth=playerWidth*0.6;var yIncrease=yWidth*0.2;for(var x=0;x<yWidth;x+=yIncrease)
that.flame.y.push([x,-random(1,4)]);that.flame.y.push([yWidth,0]);var halfR=rWidth/2,halfY=yWidth/2;for(var i=0;i<that.flame.r.length;i++){that.flame.r[i][0]-=halfR;that.flame.r[i][1]-=playerHeight/2;}
for(var i=0;i<that.flame.y.length;i++){that.flame.y[i][0]-=halfY;that.flame.y[i][1]-=playerHeight/2;}};createFlames();function radians(deg){return deg*0.0174532925;};function degrees(rad){return rad*57.2957795;};function random(from,to){return Math.floor(Math.random()*(to+1)+from);};function code(name){var table={'up':38,'down':40,'left':37,'right':39,'esc':27};if(table[name])return table[name];return name.charCodeAt(0);};function boundsCheck(vec){if(vec.x>w)
vec.x=0;else if(vec.x<0)
vec.x=w;if(vec.y>h)
vec.y=0;else if(vec.y<0)
vec.y=h;};function size(element){var el=element,left=0,top=0;do{left+=el.offsetLeft||0;top+=el.offsetTop||0;el=el.offsetParent;}while(el);return{x:left,y:top,width:element.offsetWidth||10,height:element.offsetHeight||10};};function addEvent(obj,type,fn){if(obj.addEventListener)
obj.addEventListener(type,fn,false);else if(obj.attachEvent){obj["e"+type+fn]=fn;obj[type+fn]=function(){obj["e"+type+fn](window.event);}
obj.attachEvent("on"+type,obj[type+fn]);}}
function removeEvent(obj,type,fn){if(obj.removeEventListener)
obj.removeEventListener(type,fn,false);else if(obj.detachEvent){obj.detachEvent("on"+type,obj[type+fn]);obj[type+fn]=null;obj["e"+type+fn]=null;}}
function arrayRemove(array,from,to){var rest=array.slice((to||from)+1||array.length);array.length=from<0?array.length+from:from;return array.push.apply(array,rest);};function addParticles(startPos){var time=new Date().getTime();var amount=maxParticles;for(var i=0;i<amount;i++){that.particles.push({dir:(new Vector(Math.random()*20-10,Math.random()*20-10)).normalize(),pos:startPos.cp(),cameAlive:time});}};function hasOnlyTextualChildren(element){if(indexOf(hiddenTypes,element.tagName)!=-1)return false;if(element.offsetWidth==0&&element.offsetHeight==0)return false;for(var i=0;i<element.childNodes.length;i++){if(element.childNodes[i].nodeType!=3&&indexOf(hiddenTypes,element.childNodes[i].tagName)==-1&&element.childNodes[i].childNodes.length!=0)return false;}
return true;};window.hasOnlyTextualChildren=hasOnlyTextualChildren;function indexOf(arr,item,from){if(arr.indexOf)return arr.indexOf(item,from);var len=arr.length;for(var i=(from<0)?Math.max(0,len+from):from||0;i<len;i++){if(arr[i]===item)return i;}
return-1;};function addClass(element,className){if(element.className.indexOf(className)==-1)
element.className=(element.className+' '+className).replace(/\s+/g,' ').replace(/^\s+|\s+$/g,'');};function removeClass(element,className){element.className=element.className.replace(new RegExp('(^|\\s)'+className+'(?:\\s|$)'),'$1');};function addStylesheet(name,rules){var stylesheet=document.getElementById(name);if(!stylesheet){var stylesheet=document.createElement('style');stylesheet.type='text/css';stylesheet.rel='stylesheet';stylesheet.id=name;document.getElementsByTagName("head")[0].appendChild(stylesheet);}
stylesheet.innerHTML+=rules;};function removeStylesheet(name){var stylesheet=document.getElementById(name);if(stylesheet){stylesheet.parentNode.removeChild(stylesheet);}};this.canvas=document.createElement('canvas');this.canvas.setAttribute('width',w);this.canvas.setAttribute('height',h);with(this.canvas.style){width=w+"px";height=h+"px";position="fixed";top="0px";left="0px";bottom="0px";right="0px";zIndex="10000";}
if(typeof G_vmlCanvasManager!='undefined'){this.canvas=G_vmlCanvasManager.initElement(this.canvas);if(!this.canvas.getContext){alert("So... you're using IE? Please join me at http://github.com/erkie/erkie.github.com if you think you can help");}}else{if(!this.canvas.getContext){alert('This program does not yet support your browser. Please join me at http://github.com/erkie/erkie.github.com if you think you can help');}}
addEvent(this.canvas,'mousedown',function(){destroy.apply(that);});document.body.appendChild(this.canvas);this.ctx=this.canvas.getContext("2d");this.ctx.fillStyle="black";this.ctx.strokeStyle="black";this.navigation=document.createElement('div');this.navigation.className="ASTEROIDSYEAH";with(this.navigation.style){font="Arial,sans-serif";position="fixed";zIndex="10001";bottom="10px";right="10px";textAlign="right";}
this.navigation.innerHTML="(click anywhere to exit/press esc to quit) ";document.body.appendChild(this.navigation);this.points=document.createElement('span');this.points.style.font="28pt Arial, sans-serif";this.points.style.fontWeight="bold";this.points.className="ASTEROIDSYEAH";this.navigation.appendChild(this.points);this.points.innerHTML="0";this.debug=document.createElement('span');this.debug.className="ASTEROIDSYEAH";this.debug.innerHTML="";this.navigation.appendChild(this.debug);var eventKeydown=function(event){event=event||window.event;that.keysPressed[event.keyCode]=true;switch(event.keyCode){case code(' '):that.firedAt=1;break;}
if(indexOf([code('up'),code('down'),code('right'),code('left'),code(' '),code('B')],event.keyCode)!=-1){if(event.preventDefault)
event.preventDefault();return false;}};addEvent(window,'keydown',eventKeydown);var eventKeypress=function(event){event=event||window.event;if(indexOf([code('up'),code('down'),code('right'),code('left'),code(' ')],event.keyCode||event.which)!=-1){if(event.preventDefault)
event.preventDefault();return false;}};addEvent(window,'keypress',eventKeypress);var eventKeyup=function(event){event=event||window.event;that.keysPressed[event.keyCode]=false;switch(event.keyCode){case code('B'):removeStylesheet("ASTEROIDSYEAH");break;}
if(indexOf([code('up'),code('down'),code('right'),code('left'),code(' '),code('B')],event.keyCode)!=-1){if(event.preventDefault)
event.preventDefault();return false;}};addEvent(window,'keyup',eventKeyup);this.ctx.clear=function(){this.clearRect(0,0,w,h);};this.ctx.clear();this.ctx.drawLine=function(xFrom,yFrom,xTo,yTo){this.beginPath();this.moveTo(xFrom,yFrom);this.lineTo(xTo,yTo);this.closePath();this.stroke();};this.ctx.drawRect=function(rect){var old=this.strokeStyle;this.strokeStyle="red";this.strokeRect(rect.x,rect.y,rect.width,rect.height);this.strokeStyle=old;};this.ctx.drawLineFromLine=function(line){var oldC=this.strokeStyle;this.strokeStyle="green";this.drawLine(line.p1.x,line.p1.y,line.p2.x,line.p2.y);this.strokeStyle=oldC;};this.ctx.tracePoly=function(verts){this.beginPath();this.moveTo(verts[0][0],verts[0][1]);for(var i=1;i<verts.length;i++)
this.lineTo(verts[i][0],verts[i][1]);this.closePath();};this.ctx.drawPlayer=function(){this.save();this.translate(that.pos.x,that.pos.y);this.rotate(-that.dir.angle());this.tracePoly(playerVerts);this.stroke();this.restore();};this.ctx.drawBullet=function(pos){this.beginPath();this.arc(pos.x,pos.y,bulletRadius,0,Math.PI*2,true);this.closePath();this.fill();};var randomParticleColor=function(){return(['red','yellow'])[random(0,1)];};this.ctx.drawParticle=function(particle){var oldColor=this.strokeStyle;this.strokeStyle=randomParticleColor();this.drawLine(particle.pos.x,particle.pos.y,particle.pos.x-particle.dir.x*10,particle.pos.y-particle.dir.y*10);this.strokeStyle=oldColor;};this.ctx.drawFlames=function(flame){this.save();this.translate(that.pos.x,that.pos.y);this.rotate(-that.dir.angle());var oldColor=this.strokeStyle;this.strokeStyle="red";this.tracePoly(flame.r);this.stroke();this.strokeStyle="yellow";this.tracePoly(flame.y);this.stroke();this.strokeStyle=oldColor;this.restore();}
addParticles(this.pos);addClass(document.body,'ASTEROIDSYEAH');var isRunning=true;var lastUpdate=new Date().getTime();var hasKilled=false;this.update=function(){var forceChange=false;var nowTime=new Date().getTime();var tDelta=(nowTime-lastUpdate)/1000;lastUpdate=nowTime;if(nowTime-this.updated.enemies>timeBetweenEnemyUpdate&&hasKilled){updateEnemyIndex();this.updated.enemies=nowTime;hasKilled=false;}
var drawFlame=false;if(nowTime-this.updated.flame>50){createFlames();this.updated.flame=nowTime;}
this.scrollPos.x=window.pageXOffset||document.documentElement.scrollLeft;this.scrollPos.y=window.pageYOffset||document.documentElement.scrollTop;if(this.keysPressed[code('up')]){this.vel.add(this.dir.mulNew(acc*tDelta));drawFlame=true;}else{this.vel.mul(0.96);}
if(this.keysPressed[code('left')]){forceChange=true;this.dir.rotate(radians(rotSpeed*tDelta*-1));}
if(this.keysPressed[code('right')]){forceChange=true;this.dir.rotate(radians(rotSpeed*tDelta));}
if(this.keysPressed[code(' ')]&&nowTime-this.firedAt>timeBetweenFire){this.bullets.push({'dir':this.dir.cp(),'pos':this.pos.cp(),'startVel':this.vel.cp(),'cameAlive':nowTime});this.firedAt=nowTime;if(this.bullets.length>maxBullets){arrayRemove(this.bullets,0);}}
if(this.keysPressed[code('B')]){forceChange=true;this.updated.blink.time+=tDelta*1000;if(this.updated.blink.time>timeBetweenBlink){if(this.updated.blink.isActive){removeStylesheet("ASTEROIDSYEAH");this.updated.blink.isActive=false;}else{addBlinkStyle();this.updated.blink.isActive=true;}
this.updated.blink.time=0;}}
if(this.keysPressed[code('esc')]){destroy.apply(this);return;}
if(this.vel.len()>maxSpeed){this.vel.setLength(maxSpeed);}
this.pos.add(this.vel.mulNew(tDelta));if(this.pos.x>w){window.scrollTo(this.scrollPos.x+50,this.scrollPos.y);this.pos.x=0;}else if(this.pos.x<0){window.scrollTo(this.scrollPos.x-50,this.scrollPos.y);this.pos.x=w;}
if(this.pos.y>h){window.scrollTo(this.scrollPos.x,this.scrollPos.y+h*0.75);this.pos.y=0;}else if(this.pos.y<0){window.scrollTo(this.scrollPos.x,this.scrollPos.y-h*0.75);this.pos.y=h;}
for(var i=0;i<this.bullets.length;i++){if(nowTime-this.bullets[i].cameAlive>2000){arrayRemove(this.bullets,i);i--;continue;}
var bulletVel=this.bullets[i].dir.setLengthNew(bulletSpeed*tDelta).add(this.bullets[i].startVel.mulNew(tDelta));var ray=new Line(this.bullets[i].pos.cp(),this.bullets[i].pos.addNew(bulletVel));ray.shift(this.scrollPos);this.bullets[i].pos.add(bulletVel);boundsCheck(this.bullets[i].pos);var didKill=false;for(var x=0,enemy;enemy=this.enemies[x];x++){if(ray.intersectsWithRect(enemy.aSize)){hasKilled=true;this.dieing.push(enemy);arrayRemove(this.enemies,x);didKill=true;addParticles(this.bullets[i].pos);break;}}
if(didKill){arrayRemove(this.bullets,i);i--;continue;}}
for(var i=0,enemy;enemy=this.dieing[i];i++){this.enemiesKilled++;try{enemy.parentNode.removeChild(enemy);}catch(e){}
this.points.innerHTML=this.enemiesKilled*10;this.points.title=this.enemiesKilled+"/"+this.totalEnemies;arrayRemove(this.dieing,i);i--;}
for(var i=0;i<this.particles.length;i++){this.particles[i].pos.add(this.particles[i].dir.mulNew(particleSpeed*tDelta*Math.random()));if(nowTime-this.particles[i].cameAlive>1000){arrayRemove(this.particles,i);i--;forceChange=true;continue;}}
if(forceChange||this.bullets.length!=0||this.particles.length!=0||!this.pos.is(this.lastPos)||this.vel.len()>0){this.ctx.clear();this.ctx.drawPlayer();if(drawFlame)
this.ctx.drawFlames(that.flame);for(var i=0;i<this.bullets.length;i++)
this.ctx.drawBullet(this.bullets[i].pos);for(var i=0;i<this.particles.length;i++)
this.ctx.drawParticle(this.particles[i]);}
this.lastPos=this.pos;setTimeout(updateFunc,1000/FPS);}
var updateFunc=function(){that.update.call(that);};setTimeout(updateFunc,1000/FPS);function destroy(){removeEvent(window,'keydown',eventKeydown);removeEvent(window,'keypress',eventKeypress);removeEvent(window,'keyup',eventKeyup);isRunning=false;removeStylesheet("ASTEROIDSYEAH");removeClass(document.body,'ASTEROIDSYEAH');this.canvas.parentNode.removeChild(this.canvas);this.navigation.parentNode.removeChild(this.navigation);};}
if(window.ActiveXObject){try{var xamlScript=document.createElement('script');xamlScript.setAttribute('type','text/xaml');xamlScript.textContent='<?xml version="1.0"?><Canvas xmlns="http://schemas.microsoft.com/client/2007"></Canvas>';document.getElementsByTagName('head')[0].appendChild(xamlScript);}catch(e){}
var script=document.createElement("script");script.setAttribute('type','text/javascript');script.onreadystatechange=function(){if(script.readyState=='loaded'||script.readyState=='completed'){new Asteroids();}};script.src="excanvas.js";document.getElementsByTagName('head')[0].appendChild(script);}
else new Asteroids();})();