Another classic simple game. You probably have played this in one form or another.
A feature which I dont recall seeing before is the food decaying and becoming toxic. It adds that little extra dimension where you must make a choice of chasing after that food before it becomes toxic.
This game lets you choose the difficulty level which just translates to how fast the snake moves.
Also a couple of simple sounds. You might recognise what games I based the sounds on.
Have fun.
<html> <head> <meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=yes"> <title>Snake</title> <style> Body { background-color: #181818; color: #ffffff; margin: 0px; } Canvas { position:fixed; } #counter { position:absolute; bottom:0; right:0; color:#ffffff; display:none; } </style> </head> <body> <input id="focus" type="text" style="position:fixed; width:0px;height=0px;" /> <canvas id="myCanvas">Sorry... your browser does not support the canvas element.</canvas> <p id="counter">counter</p> </body> <script> // Global variables // used for calculating and showing the frames per second var dwFrames=0; var dwCurrentTime=0; var dwLastUpdateTime=0; var dwElapsedTime=0; var fpsText; // The game engine needs to persist var GE; var LAYER_TILE = 1; var LAYER_MENU = 2; var LAYER_BUTTON = 3; var SOUND_SNAKE_EAT = 0; var SOUND_SNAKE_DIED = 1; var skipSplash = false; var currentStage; init = function() { var canvas = document.getElementById('myCanvas'); if (canvas.getContext) { // got to have the game engine GE = GameEngine16Colour(canvas,Update,Draw); // GE.CustomResolution(240,240); if ( skipSplash ) currentStage = GameInitBackground(); else currentStage = Splash(GameData()); // when the window resizes we want to resize the canvas window.onresize = function() { doResize(); }; window.onmouseover = function () { getFocus(); }; doResize(); // Init sounds sound = new customSound(24000); sound.add(.1, 500, 200, 50, 50, .5, .5,0,0,0,0,true ); sound.add(.1, 200, 1000, 50, 00, .5, .5,0,0,0,0,true ); sound.createSound(SOUND_SNAKE_EAT); sound = new customSound(24000); for ( var f=1000; f > 200; f-=100) { sound.add(.1, f, f+200, 50, 50, .5, .5,0,0,0,0,true ); sound.add(.1, f+200, f-200, 50, 50, .5, .5,0,0,0,0,true ); } sound.createSound(SOUND_SNAKE_DIED); } } function getFocus() { var f = document.getElementById("focus"); f.focus(); } // This gets called every time the engine redraws the canvas // For slow machines this may be less that the actual fps Draw = function() { currentStage.Draw(); // calculate and show the fps dwFrames++; dwCurrentTime = (new Date).getTime(); dwElapsedTime = dwCurrentTime - dwLastUpdateTime; if(dwElapsedTime >= 1000) { fpsText = "FPS " + dwFrames; dwLastUpdateTime = dwCurrentTime; dwFrames = 0; } var l = document.getElementById('counter'); l.textContent = fpsText; } // This gets called every game update frame Update = function() { currentStage = currentStage.Update(); } // This just makes sure the canvas is as big as possible // given the window size and centered function doResize() { var canvas = document.getElementById('myCanvas'); var screenHeight = window.innerHeight; var screenWidth = window.innerWidth; var ratioX = GE.CanvasWidth / screenWidth; var ratioY = GE.CanvasHeight / screenHeight; var ratio = ratioX if ( ratioX < ratioY ) ratio = ratioY; var width = GE.CanvasWidth / ratio; var height = GE.CanvasHeight / ratio; canvas.style.height = height+"px"; canvas.style.width = width+"px"; canvas.style.marginLeft = (screenWidth-width)/2; canvas.style.marginTop = (screenHeight-height)/2; } // Splash screen stages function GameData() { var obj = {}; obj.score = 0; obj.hiscore = 0; obj.speedSlow = 0; return obj; } function SplashData() { var obj = {}; obj.finished = false; return obj; } function PlayData() { var obj = {}; obj.finished = false; obj.lives; obj.PlayArea; obj.snakeData; obj.score=0; return obj; } function SnakeData() { var obj = {}; obj.finished = false; obj.turnLeft = false; obj.turnRight = false; obj.goLeft = false; obj.goRight = false; obj.goUp = false; obj.goDown = false; obj.snake; obj.food; obj.playArea; obj.headposx=0; obj.headposy=0; obj.headdirection=0; obj.tailposx=0; obj.tailposy=0; obj.taildirection=0; obj.foodHit = false; return obj; } function PlayArea(width, height,snakeData) { var obj = {}; obj.GRID_EMPTY = 0; obj.GRID_OUT_OF_BOUNDS = 1; obj.GRID_SNAKE = 2; obj.GRID_SNAKE_HEAD = 3; obj.GRID_SNAKE_TAIL = 4; obj.GRID_FOOD = 5; obj.GRID_FOOD_EXPIRING = 6; obj.GRID_FOOD_BAD = 7; obj.width = width; obj.height = height; var grid = new Array(); for ( var y = 0; y < height; y++ ) { var row = new Array(); for ( var x = 0; x < width; x++ ) { if ( y == 0 || y == height-1 || x == 0 || x == width-1 ) row.push(obj.GRID_OUT_OF_BOUNDS); else row.push(obj.GRID_EMPTY); } grid.push(row); } var headCostume = CreateSnakeHeadCostume(); var tailCostume = CreateSnakeTailCostume(); var bodyCostume = CreateSnakeBodyCostume(); var foodCostume = CreateFoodCostume(); obj.Draw=function() { GE.DrawRect(0,0,GE.CanvasWidth,GE.CanvasHeight,0); for ( var y = 0; y < height; y++ ) { var row = new Array(); for ( var x = 0; x < width; x++ ) { var cell=obj.Get(x,y); if (cell == obj.GRID_EMPTY ) { // GE.DrawRect(x*8,y*8,8,8,8); // GE.DrawRect(x*8,y*8,7,7,0); } if (cell == obj.GRID_OUT_OF_BOUNDS ) { GE.DrawRect(x*8,y*8,8,8,6+8); } if (cell == obj.GRID_FOOD ) { GE.Paste(x*8,y*8, foodCostume.GetItem(0)); } if (cell == obj.GRID_FOOD_EXPIRING ) { GE.Paste(x*8,y*8, foodCostume.GetItem(1)); } if (cell == obj.GRID_FOOD_BAD ) { GE.Paste(x*8,y*8, foodCostume.GetItem(2)); } if (cell == obj.GRID_SNAKE ) { GE.Paste(x*8, y*8, bodyCostume.GetItem(0)); } GE.DrawRect(snakeData.headposx*8,snakeData.headposy*8,8,8,0); // GE.DrawRect(snakeData.headposx*8,snakeData.headposy*8,7,7,0); GE.DrawRect(snakeData.tailposx*8,snakeData.tailposy*8,8,8,0); // GE.DrawRect(snakeData.tailposx*8,snakeData.tailposy*8,7,7,0); GE.Paste(snakeData.headposx*8,snakeData.headposy*8, headCostume.GetItem(snakeData.headdirection)); GE.Paste(snakeData.tailposx*8,snakeData.tailposy*8, tailCostume.GetItem(snakeData.taildirection)); } } } obj.Get=function(x,y) { if ( x < 0 || x >= width ) return obj.GRID_OUT_OF_BOUNDS; if ( y < 0 || y >= height ) return obj.GRID_OUT_OF_BOUNDS; return grid[y][x]; } obj.Set=function(x,y,value) { if ( x < 0 || x >= width ) return; if ( y < 0 || y >= height ) return; grid[y][x] = value; } return obj; } function Splash(data) { var data = GameData(); var splashData = SplashData(); var obj = GameSatePattern(splashData); obj.AddInnerState( SplashStatic(splashData) ); obj.SetNextState(GameInitBackground(data)); obj.UpdateCustom = function() { if (splashData.finished) obj.SetFinished(); } return obj; } // the following are inner states for the splash function SplashStatic(data) { var obj = GameSatePattern(data); var countFrames = 0; obj.UpdateCustom = function() { countFrames++; if ( countFrames > 30*2 ) obj.SetFinished(); } obj.DrawCustom=function() { for ( var y = 0; y < GE.CanvasHeight; y++) { for ( var x = 0; x < GE.CanvasWidth; x++ ) { GE.DrawPixel(x,y,Math.floor(Math.random()*16)); } } } obj.SetNextState(SplashLogo(data)); return obj; } function SplashLogo(data) { var obj = GameSatePattern(data); var countFrames = 0; obj.Init = function () { data.finished = false; GE.DrawRect(0,0,GE.CanvasWidth, GE.CanvasHeight,0); GE.WriteStringCentered("QuietBloke Productions",GE.CanvasHeight/2-32-8,15,false,true); GE.WriteStringCentered("Presents",GE.CanvasHeight/2,15); GE.WriteStringCentered("Snake",GE.CanvasHeight/2+32,15,false,true); } obj.UpdateCustom=function() { countFrames++; if ( countFrames > 30*3 ) { data.finished = true; } } return obj; } function GameInitBackground(data) { var obj = GameSatePattern(data); obj.Init = function () { if (!data) data = GameData(); obj.SetNextState(MainMenu(data)); obj.SetFinished(); } return obj; } function MainMenu(data) { var obj = GameSatePattern(data); // var buttonSprites = new Array(); var buttonActive = 1; var leftKeyDown = false; var rightKeyDown = false; var activeButton; obj.Init = function () { var menuBitmap = Bitmap(220,20); var menuBitmap2 = Bitmap(220,160); menuBitmap.DrawRect(0,0,menuBitmap.width, menuBitmap.height,6+8); menuBitmap2.DrawRect(0,0,menuBitmap2.width, menuBitmap2.height,6); menuBitmap.WriteStringCentered("SNAKE",GE.CharSet(),4,0,true,true); var textColour = 15; var food = CreateFoodCostume(); menuBitmap2.Paste(4,20,food.GetItem(0)); menuBitmap2.WriteString("Eat Fresh Food 2PTS",GE.CharSet(),28,20,textColour); menuBitmap2.WriteString("Eat old Food 1PT",GE.CharSet(),28,30,textColour); menuBitmap2.Paste(4,30,food.GetItem(1)); menuBitmap2.WriteString("Rotten food is fatal",GE.CharSet(),28,40,textColour); menuBitmap2.Paste(4,40,food.GetItem(2)); menuBitmap2.WriteString("Controls",GE.CharSet(),4,60,textColour); menuBitmap2.WriteString("Cursor Keys",GE.CharSet(),28,70,textColour); menuBitmap2.WriteStringCentered("SpaceBar or click to Begin",GE.CharSet(),110,textColour,false,true); var buttonEasy = Bitmap(60,16); buttonEasy.DrawRect(0,0,60,16,3); buttonEasy.WriteStringCentered("Easy",GE.CharSet(),4,15); var btn = Sprite(60,200,buttonEasy); GE.SpriteAdd("ButtonEasy",btn, LAYER_BUTTON); var buttonNormal = Bitmap(60,16); buttonNormal.DrawRect(0,0,60,16,3); buttonNormal.WriteStringCentered("Normal",GE.CharSet(),4,15); var btn2 = Sprite(130,200,buttonNormal); GE.SpriteAdd("ButtonNormal",btn2, LAYER_BUTTON); var buttonHard = Bitmap(60,16); buttonHard.DrawRect(0,0,60,16,3); buttonHard.WriteStringCentered("Hard",GE.CharSet(),4,15); var btn3 = Sprite(200,200,buttonHard); GE.SpriteAdd("ButtonHard",btn3, LAYER_BUTTON); // buttonSprites.push(buttonHard); // buttonSprites.push(buttonNormal); // buttonSprites.push(buttonEasy); var bmp = Bitmap(64,20); bmp.DrawRect(0,0,64,20,1); bmp.DrawRect(2,2,60,16,-1); activeButton = Sprite(128,198,bmp); GE.SpriteAdd("ButtonActive",activeButton, LAYER_BUTTON); var sp = Sprite((GE.CanvasWidth-menuBitmap.width)/2, 20, menuBitmap ); GE.SpriteAdd("MainMenu", sp, LAYER_MENU); sp = Sprite((GE.CanvasWidth-menuBitmap2.width)/2, 65, menuBitmap2 ); GE.SpriteAdd("MainMenu2", sp, LAYER_MENU); DisplayScore(data); } obj.UpdateCustom=function() { if ( GE.GetKeyState(GE.KEY_LEFT)) { if ( !leftKeyPressed) { leftKeyPressed = true; buttonActive--; if ( buttonActive < 0 ) buttonActive = 2; } } else leftKeyPressed = false; if ( GE.GetKeyState(GE.KEY_RIGHT)) { if ( !rightKeyPressed) { rightKeyPressed = true; buttonActive++; if ( buttonActive > 2 ) buttonActive = 0; } } else rightKeyPressed = false; if ( FirePressed() ) { if ( GE.mouseClicked ) { // find which button was clicked var buttonClicked = -1; if ( GE.mousePos.y >= 200 && GE.mousePos.y <= 200+16 ) { for ( var b = 0; b <= 2; b++ ) { var xpos = 60 + 70*b; if ( GE.mousePos.x >= xpos && GE.mousePos.x <= xpos + 60 ) { buttonClicked = b; break; } } } if ( buttonClicked >= 0 ) { data.speedSlow = 2-buttonClicked; obj.SetFinished(); obj.SetNextState(Play(data)); } } else { data.speedSlow = 2-buttonActive; obj.SetFinished(); obj.SetNextState(Play(data)); } } var xoff = buttonActive*70+58; activeButton.pos.x = xoff; } obj.TearDown = function() { GE.SpriteDelete("MainMenu"); GE.SpriteDelete("MainMenu2"); GE.SpriteDelete("ButtonEasy"); GE.SpriteDelete("ButtonNormal"); GE.SpriteDelete("ButtonHard"); GE.SpriteDelete("ButtonActive"); } return obj; } function Play(data) { var obj = GameSatePattern(data); var playData = PlayData(); playData.speedSlow = data.speedSlow; obj.AddInnerState(PlayInit(playData)); obj.Init=function() { data.score = 0; DisplayScore(data); } obj.DrawCustom = function() { DisplayScore(data); } obj.UpdateCustom = function() { if ( playData.finished ) obj.SetFinished(); obj.SetNextState(MainMenu(data)); if ( playData.score != data.score ) { data.score = playData.score; if ( data.score > data.hiscore ) data.hiscore = data.score; } } return obj; } // the following are inner states to the play state function PlayInit(data) { var obj = GameSatePattern(data); obj.Init=function() { data.score = 0; data.snakeData = SnakeData(); data.playArea = PlayArea(GE.CanvasWidth/8,GE.CanvasHeight/8,data.snakeData); data.snake = Snake(data.snakeData,GE.CanvasWidth/16,GE.CanvasHeight/16,data.speedSlow); data.snakeData.playArea = data.playArea; data.snake.Update(); data.playArea.Draw(); if ( GE.Tiltable()) GE.TiltCalibrate(); data.food = new Array(); for ( var f=0; f < 10;f ++ ) data.food.push(FoodItem()); } obj.SetNextState(PlayGetReady(data)); obj.SetFinished(); return obj; } function PlayGetReady(data) { var obj = GameSatePattern(data); var waitTimer = 4*30; obj.Init = function() { var cs=Costume() var menuBitmap = Bitmap(8*10,16); menuBitmap.DrawRect(0,0,8*10,16,0); var blankBitmap = menuBitmap.Clone(); menuBitmap.WriteString("GET READY",GE.CharSet(),4,4,15); cs.SetItem(0,menuBitmap); cs.SetItem(1,blankBitmap); var sp = Sprite((GE.CanvasWidth-menuBitmap.width)/2, 32, cs ); sp.SetAnimatorLoop(); sp.SetAnimatorUpdateRate(.1); GE.SpriteAdd("GetReady", sp, LAYER_MENU); obj.SetNextState(PlayGame(data)); } obj.UpdateCustom=function() { waitTimer--; if ( waitTimer == 0 ) obj.SetFinished(); } obj.TearDown=function() { GE.SpriteDelete("GetReady"); } return obj; } function FoodItem() { var obj = {}; var lifeSpan = 0; var x = -1; var y = -1; obj.IsExpiring=function() { if (lifeSpan < 4*30 ) return true; return false; } obj.Update=function() { if ( obj.IsAlive()) lifeSpan--; } obj.Set=function(xp,yp,l) { x = xp; y = yp; lifeSpan = l; } obj.xPos=function() { return x; } obj.yPos=function() { return y; } obj.Eat=function() { lifeSpan = 0; } obj.IsAlive=function() { if (lifeSpan > 0 ) return true; return false; } return obj; } function Snake(data,startposx,startposy,speedSlow) { var obj = GameSatePattern(data); var length = 0; var headCostume; var bodyCostume; var tailCostume; var direction; // 0 = up, 1=right, 2=down, 3 = left var segmentBuffer; var headpos = 2; var tailpos = 0; var length = 0; var framePause = speedSlow; var prevtailx; var prevtaily; var MAX_LENGTH = 500; obj.Init = function () { headCostume = CreateSnakeHeadCostume(); tailCostume = CreateSnakeTailCostume(); bodyCostume = CreateSnakeBodyCostume(); // Create the buffer segmentBuffer = new Array(); for ( var i = 0; i < MAX_LENGTH; i++ ) segmentBuffer.push(Vector()); segmentBuffer[0].Set(startposx,startposy); segmentBuffer[1].Set(startposx,startposy-1); headpos = 1; tailpos = 0; length = 2; direction=0; prevtailx = -1; prevtaily = -1; data.headposx = startposx; data.headposy = startposy-1; data.tailposx = startposx; data.tailposy = startposy; } obj.UpdateCustom=function() { if ( framePause > 0 ) { framePause--; return; } framePause=speedSlow; var nextHead = nextPos(headpos); if ( data.turnLeft ) { direction--; if ( direction < 0 ) direction = 3; } if ( data.turnRight ) { direction++; if ( direction > 3 ) direction = 0; } if ( data.goUp && direction != 2 ) direction = 0; else if ( data.goDown && direction != 0 ) direction = 2; else if ( data.goLeft && direction != 1 ) direction = 3; else if ( data.goRight && direction != 3 ) direction = 1; if ( direction == 0 ) { segmentBuffer[nextHead].Set(segmentBuffer[headpos].x,segmentBuffer[headpos].y-1); } else if ( direction == 1 ) { segmentBuffer[nextHead].Set(segmentBuffer[headpos].x+1,segmentBuffer[headpos].y); } else if ( direction == 2 ) { segmentBuffer[nextHead].Set(segmentBuffer[headpos].x,segmentBuffer[headpos].y+1); } else if ( direction == 3 ) { segmentBuffer[nextHead].Set(segmentBuffer[headpos].x-1,segmentBuffer[headpos].y); } headpos = nextHead; if ( length < MAX_LENGTH-1 && data.foodHit) length++; else { prevtailx = segmentBuffer[tailpos].x; prevtaily = segmentBuffer[tailpos].y data.playArea.Set(prevtailx,prevtaily,data.playArea.GRID_EMPTY); tailpos = nextPos(tailpos); var nexttailpos = nextPos(tailpos); var xdiff = segmentBuffer[nexttailpos].x - segmentBuffer[tailpos].x; var ydiff = segmentBuffer[nexttailpos].y - segmentBuffer[tailpos].y; data.taildirection = 0; if ( xdiff == 1 ) data.taildirection = 1; if ( xdiff == -1 ) data.taildirection = 3; if ( ydiff == 1 ) data.taildirection = 2; } data.foodHit = false; data.headposx = segmentBuffer[headpos].x data.headposy = segmentBuffer[headpos].y data.tailposx = segmentBuffer[tailpos].x data.tailposy = segmentBuffer[tailpos].y data.headdirection = direction; data.turnLeft = false; data.turnRight = false; data.goLeft = false; data.goRight = false; data.goUp = false; data.goDown = false; } // obj.DrawCustom = function() // { // // figure out the direction of the tail // var nexttail = nextPos(tailpos); // var xdiff = segmentBuffer[nexttail].x - segmentBuffer[tailpos].x; // var ydiff = segmentBuffer[nexttail].y - segmentBuffer[tailpos].y; // var taildirection = 0; // if ( xdiff == 1 ) // taildirection = 1; // if ( xdiff == -1 ) // taildirection = 3; // if ( ydiff == 1 ) // taildirection = 2; // // segment = nextPos (tailpos); // } function nextPos (pos) { var next = pos + 1; if ( next >= segmentBuffer.length ) next = 0; return next; } return obj; } function PlayGame(data) { var obj = GameSatePattern(data); var player; var aKeyPressed = true; var snakeData; var snakeHeadx=0; var snakeHeady=0; var foodTimer = .1*30; obj.Init=function() { obj.AddInnerState( data.snake ); } obj.DrawCustom=function() { data.playArea.Draw(); } obj.UpdateCustom=function() { if ( data.snakeData.headposx != snakeHeadx || data.snakeData.headposy != snakeHeady ) { var belowHead = data.playArea.Get(data.snakeData.headposx, data.snakeData.headposy ) if (belowHead != data.playArea.GRID_EMPTY && belowHead != data.playArea.GRID_FOOD && belowHead != data.playArea.GRID_FOOD_EXPIRING) { obj.SetNextState(PlayGameOver(data)); obj.SetFinished(); soundPlay(SOUND_SNAKE_DIED); } if (belowHead == data.playArea.GRID_FOOD || belowHead == data.playArea.GRID_FOOD_EXPIRING) { data.snakeData.foodHit=true; for(var l = 0; l < data.food.length; l++ ) { if ( data.food[l].IsAlive()) { if ( data.food[l].xPos() == data.snakeData.headposx && data.food[l].yPos() == data.snakeData.headposy ) { if ( belowHead == data.playArea.GRID_FOOD ) data.score+=2; else data.score+=1; data.food[l].Eat(); soundPlay(SOUND_SNAKE_EAT); } } } } snakeHeadx = data.snakeData.headposx; snakeHeady = data.snakeData.headposy; data.playArea.Set(data.snakeData.headposx, data.snakeData.headposy,data.playArea.GRID_SNAKE); } foodTimer--; if ( foodTimer < 0 ) { // see if we have a spare food item var foodIndex = -1; for ( var f = 0; f < data.food.length; f++ ) { if (!data.food[f].IsAlive()) { foodIndex = f; break; } } var foodx = Math.floor(Math.random() * data.playArea.width); var foody = Math.floor(Math.random() * data.playArea.height); var griditem = data.playArea.Get(foodx, foody ); if ( griditem == data.playArea.GRID_EMPTY ) { data.food[foodIndex].Set(foodx, foody, 10*30 ); data.playArea.Set(foodx, foody, data.playArea.GRID_FOOD); foodTimer = 2*30; } } // draw all the food items for ( var f = 0; f < data.food.length; f++ ) { if (data.food[f].IsAlive()) { data.food[f].Update(); if ( !data.food[f].IsAlive()) { data.playArea.Set(data.food[f].xPos(), data.food[f].yPos(), data.playArea.GRID_FOOD_BAD); } else if ( data.food[f].IsExpiring()) { data.playArea.Set(data.food[f].xPos(), data.food[f].yPos(), data.playArea.GRID_FOOD_EXPIRING); } else data.playArea.Set(data.food[f].xPos(), data.food[f].yPos(), data.playArea.GRID_FOOD); } } if ( GE.GetKeyState(GE.KEY_LEFT)) { data.snakeData.goLeft = true; aKeyPressed = true; } if ( GE.GetKeyState(GE.KEY_RIGHT)) { data.snakeData.goRight = true; aKeyPressed = true; } if ( GE.GetKeyState(GE.KEY_UP)) { data.snakeData.goUp = true; aKeyPressed = true; } if ( GE.GetKeyState(GE.KEY_DOWN)) { data.snakeData.goDown = true; aKeyPressed = true; } } return obj; } function PlayGameOver(data) { var obj = GameSatePattern(data); var countFrames = 0; obj.Init = function () { var cs=Costume() var menuBitmap = Bitmap(8*10,16); menuBitmap.DrawRect(0,0,8*10,16,0); var blankBitmap = menuBitmap.Clone(); menuBitmap.WriteString("GAME OVER",GE.CharSet(),4,4,15); cs.SetItem(0,menuBitmap); cs.SetItem(1,blankBitmap); var sp = Sprite((GE.CanvasWidth-menuBitmap.width)/2, 32, cs ); sp.SetAnimatorLoop(); sp.SetAnimatorUpdateRate(.1); GE.SpriteAdd("GameOver", sp, LAYER_MENU); } obj.UpdateCustom=function() { countFrames++; if ( countFrames > 30*3 ) { obj.SetFinished(); data.finished = true; } } obj.TearDown = function() { GE.SpriteDelete("GameOver"); } return obj; } // helper functions function DisplayScore(data) { var tmpStr = "00000" + data.score; tmpStr = tmpStr.substring(tmpStr.length-5); GE.DrawRect(32-8,0,11*8+16,8,0); GE.WriteString("SCORE:"+tmpStr,32,0,15,false,false); tmpStr = "00000" + data.hiscore; tmpStr = tmpStr.substring(tmpStr.length-5); GE.DrawRect(GE.CanvasWidth-14*8-32-8,0,16*8,8,0); GE.WriteString("HI-SCORE:"+tmpStr,GE.CanvasWidth-14*8-32,0,15,false,false); } function CreateSnakeBodyCostume() { var cs=Costume(); var bm = Bitmap(8,8); bm.DrawRect(0,1,8,6,2); bm.DrawRect(1,0,6,8,2); cs.SetItem(0,bm); return cs; } function CreateSnakeHeadCostume() { var cs=Costume(); var bm = Bitmap(8,8).PixelsFromArray([-1,-1,-1,9,9,-1,-1,-1 ,-1,-1,-1,9,9,-1,-1,-1 ,-1,-1,2,2,2,2,-1,-1 ,-1,2,2,2,2,2,2,-1 ,2,14,0,2,2,0,14,2 ,2,14,14,2,2,14,14,2 ,2,2,2,2,2,2,2,2 ,-1,2,2,2,2,2,2,-1 ]); cs.SetItem(0,bm); var bm = bm.RotateRight(); cs.SetItem(1,bm); var bm = bm.RotateRight(); cs.SetItem(2,bm); var bm = bm.RotateRight(); cs.SetItem(3,bm); return cs; } function CreateSnakeTailCostume() { var cs=Costume(); var bm = Bitmap(8,8).PixelsFromArray([-1,10,10,10,10,10,10,-1 ,-1,12,12,12,12,12,12,-1 ,-1,10,10,10,10,10,10,-1 ,-1,12,12,12,12,12,12,-1 ,-1,-1,10,10,10,10,-1,-1 ,-1,-1,12,12,12,12,-1,-1 ,-1,-1,-1,10,10,-1,-1,-1 ,-1,-1,-1,12,12,-1,-1,-1 ]) cs.SetItem(0,bm); var bm = bm.RotateRight(); cs.SetItem(1,bm); var bm = bm.RotateRight(); cs.SetItem(2,bm); var bm = bm.RotateRight(); cs.SetItem(3,bm); return cs; } function CreateFoodCostume() { var cs=Costume(); cs.SetItem(0, Bitmap(8,8).PixelsFromArray([-1,-1,-1,-1,-1,10,10,10 ,-1,-1,-1,10,10,10,10,-1 ,-1,-1,10,10,-1,10,-1,-1 ,-1,9,9,-1,-1,10,-1,-1 ,9,15,9,9,-1,9,9,-1 ,9,9,9,9,9,15,9,9 ,-1,9,9,-1,9,9,9,9 ,-1,-1,-1,-1,-1,9,9,-1 ])); cs.SetItem(1, Bitmap(8,8).PixelsFromArray([-1,-1,-1,-1,-1,2,2,2 ,-1,-1,-1,2,2,2,2,-1 ,-1,-1,2,2,-1,2,-1,-1 ,-1,1,1,-1,-1,2,-1,-1 ,1,7,1,1,-1,1,1,-1 ,1,1,1,1,1,7,1,1 ,-1,1,1,-1,1,1,1,1 ,-1,-1,-1,-1,-1,1,1,-1 ])); cs.SetItem(2, Bitmap(8,8).PixelsFromArray([9,-1,-1,-1,-1,-1,-1,9 ,-1,9,15,15,15,15,9,-1 ,-1,15,9,15,15,9,15,-1 ,-1,15,0,9,9,0,15,-1 ,-1,0,15,9,9,15,-1,-1 ,-1,-1,9,0,0,9,-1,-1 ,-1,9,-1,15,15,-1,9,-1 ,9,-1,-1,-1,-1,-1,-1,9 ])); return cs; } // Handle various ways to press fire var firePressed = false; var spacekeyDown = false; function FirePressed () { if (GE.mouseClicked ) return true; if ( GE.GetKeyState(GE.KEY_SPACE)) { if ( !spacekeyDown ) { spacekeyDown = true; return true; } } else { spacekeyDown = false; } return false; } GameEngine16Colour = function(canvasIn,updateFunction,drawFunction) { "use strict"; var obj = {}; var useWebFont = false; if ( typeof(updateFunction) != "function" ) updateFunction = function(){}; if ( typeof(drawFunction) != "function" ) drawFunction = function(){}; var CANVAS_WIDTH = 320; //320; var CANVAS_HEIGHT = 240; // 240; var FPS = 30; var updateRate = Math.floor(1000 / FPS); var canvas = canvasIn; var colours = new Array(); var charSet = {}; obj.mousePos = new Vector(); obj.mouseClicked = false; obj.mouseTouchFire = false; obj.mouseUnclicked = false; obj.mouseDown = false; obj.CanvasWidth = CANVAS_WIDTH; obj.CanvasHeight = CANVAS_HEIGHT; canvas.width = CANVAS_WIDTH; canvas.height = CANVAS_HEIGHT; obj.sprites = new Array(); // key = spritename, value = layer obj.layers = new Array(); // key = spritename, value = sprite obj.collisionRules = new Array(); var ctx = null; if(canvas.getContext ) { ctx = canvas.getContext('2d'); } if (ctx === null ) { return null; } var screenImageData = ctx.getImageData(0,0,CANVAS_WIDTH,CANVAS_HEIGHT); for ( var pixel = 0; pixel < CANVAS_WIDTH*CANVAS_HEIGHT; pixel+=1) { screenImageData.data[pixel*4+3] = 255; } var screenBitmap = Bitmap(CANVAS_WIDTH,CANVAS_HEIGHT); screenBitmap.DrawRect(0,0,CANVAS_WIDTH,CANVAS_HEIGHT,0); var tempScreenBitmap= Bitmap(CANVAS_WIDTH,CANVAS_HEIGHT); var prevScreenBitmap= Bitmap(CANVAS_WIDTH,CANVAS_HEIGHT); function InitColours() { colours[0] = Colour(0,0,0); colours[1] = Colour(128,0,0); colours[2] = Colour(0,128,0); colours[3] = Colour(0,0,128); colours[4] = Colour(128,128,0); colours[5] = Colour(128,0,128); colours[6] = Colour(0,128,128); colours[7] = Colour(128,128,128); colours[8] = Colour(64,64,64); colours[9] = Colour(255,0,0); colours[10] = Colour(0,255,0); colours[11] = Colour(0,0,255); colours[12] = Colour(255,255,0); colours[13] = Colour(255,0,255); colours[14] = Colour(0,255,255); colours[15] = Colour(255,255,255); // experimental colours // mid black and dark primary colours[16] = TertiaryMix(colours[7],colours[7],32); colours[17] = TertiaryMix(colours[1],colours[1],64); colours[18] = TertiaryMix(colours[2],colours[2],64); colours[19] = TertiaryMix(colours[3],colours[3],64); // secondary colours[20] = TertiaryMix(colours[4],colours[4],64); colours[21] = TertiaryMix(colours[5],colours[5],64); colours[22] = TertiaryMix(colours[6],colours[6],64); // tertiary colours colours[23] = TertiaryMix(colours[1],colours[4],128); colours[24] = TertiaryMix(colours[1],colours[5],128); // colours[18] = TertiaryMix(colours[1],colours[6],128); colours[25] = TertiaryMix(colours[2],colours[4],128); // colours[20] = TertiaryMix(colours[2],colours[5],128); colours[26] = TertiaryMix(colours[2],colours[6],128); // colours[22] = TertiaryMix(colours[3],colours[4],128); colours[27] = TertiaryMix(colours[3],colours[5],128); colours[28] = TertiaryMix(colours[3],colours[6],128); colours[29] = TertiaryMix(colours[4],colours[5],128); colours[30] = TertiaryMix(colours[4],colours[6],128); // colours[28] = TertiaryMix(colours[5],colours[4],128); // colours[29] = TertiaryMix(colours[5],colours[6],128); // colours[30] = TertiaryMix(colours[6],colours[4],128); colours[31] = TertiaryMix(colours[6],colours[5],128); // bright colours[32] = TertiaryMix(colours[15],colours[15],192); colours[33] = TertiaryMix(colours[9],colours[9],192); colours[34] = TertiaryMix(colours[10],colours[10],192); colours[35] = TertiaryMix(colours[11],colours[11],192); // secondary colours[36] = TertiaryMix(colours[12],colours[12],192); colours[37] = TertiaryMix(colours[13],colours[13],192); colours[38] = TertiaryMix(colours[14],colours[14],192); // colours[31] = TertiaryMix(colours[7],colours[7],96); colours[39] = TertiaryMix(colours[9],colours[12],255); colours[40] = TertiaryMix(colours[9],colours[13],255); // colours[34] = TertiaryMix(colours[9],colours[14],255); colours[41] = TertiaryMix(colours[10],colours[12],255); // colours[36] = TertiaryMix(colours[10],colours[13],255); colours[42] = TertiaryMix(colours[10],colours[14],255); // colours[38] = TertiaryMix(colours[11],colours[12],255); colours[43] = TertiaryMix(colours[11],colours[13],255); colours[44] = TertiaryMix(colours[11],colours[14],255); colours[45] = TertiaryMix(colours[12],colours[13],255); colours[46] = TertiaryMix(colours[12],colours[14],255); // colours[44] = TertiaryMix(colours[13],colours[12],255); // colours[45] = TertiaryMix(colours[13],colours[14],255); // colours[46] = TertiaryMix(colours[14],colours[12],255); colours[47] = TertiaryMix(colours[14],colours[13],255); // colours[47] = TertiaryMix(colours[15],colours[15],192); function TertiaryMix (col1,col2,maxValue) { var red = col1.red + col2.red; var green = col1.green + col2.green; var blue = col1.blue + col2.blue; var br = red; if(green > br) br = green; if(blue > br) br = blue; red = Math.floor(red / br * maxValue); green = Math.floor(green / br * maxValue); blue = Math.floor(blue / br * maxValue); return Colour(red,green,blue); } } InitColours(); function InitCharSet() { charSet = CharSet(8,8); for ( var charVal = 32; charVal < 128; charVal++) { var charArray = new Array(); var character = String.fromCharCode(charVal); ctx.fillStyle = "#000000"; ctx.fillRect(0,0,10,10); ctx.font = "7px lucida console"; // ctx.font = "14px impact"; ctx.fillStyle = "#FFFFFF"; ctx.font = "Bold 9px monospace"; // ctx.font = "Normal 9px Arial"; ctx.fillText(character,0,7); var lineZeroBlank = true; for ( var y = 0; y < 10; y++) { var charLine = ""; for ( var x = 0; x < 8; x++) { var p = ctx.getImageData(x, y, 2, 2).data; if ( p[0] > 125 ) charLine += "*"; else charLine += " "; } charArray.push(charLine); // if ( y == 0 && charLine != " ") // lineZeroBlank = false; if ( y == 8 && charLine != " ") { var tmp = new Array(); for ( var l = 1; l < 9; l++) tmp.push(charArray[l]); charArray = tmp; } } charSet.SetChar(character,charArray); } if ( !useWebFont ) { charSet.SetChar("!",[ " * ", " * ", " * ", " * ", " * ", " ", " * ", " "]); charSet.SetChar("?",[ " ***** ", "* * ", " ** ", " * ", " * ", " ", " * ", " "]); charSet.SetChar("\\",[ "* ", " * ", " * ", " * ", " * ", " * ", " * ", " "]); charSet.SetChar("\"",[ " * * ", " * * ", " ", " ", " ", " ", " ", " "]); charSet.SetChar("#",[ " * * ", "******* ", " * * ", " * * ", " * * ", "******* ", " * * ", " "]); charSet.SetChar("$",[ " * ", " ***** ", "* * ", " ***** ", " * * ", " ***** ", " * ", " "]); charSet.SetChar("%",[ " * * ", "* * * ", " * * ", " * ", " * * ", " * * * ", "* * ", " "]); charSet.SetChar("&",[ " *** ", " * * ", " * * ", " ** ", " * * ", "* * ", " **** * ", " "]); charSet.SetChar("'",[ " * ", " * ", " ", " ", " ", " ", " ", " "]); charSet.SetChar("(",[ " * ", " * ", " * ", " * ", " * ", " * ", " * ", " "]); charSet.SetChar(")",[ " * ", " * ", " * ", " * ", " * ", " * ", " * ", " "]); charSet.SetChar("*",[ " ", "* * ", " * * ", " * ", " * * ", "* * ", " ", " "]); charSet.SetChar("+",[ " ", " * ", " * ", " ***** ", " * ", " * ", " ", " "]); charSet.SetChar(",",[ " ", " ", " ", " ", " ", " * ", " * ", " * "]); charSet.SetChar("-",[ " ", " ", " ", " ***** ", " ", " ", " ", " "]); charSet.SetChar(".",[ " ", " ", " ", " ", " ", " ", " * ", " "]); charSet.SetChar("/",[ " * ", " * ", " * ", " * ", " * ", " * ", "* ", " "]); charSet.SetChar("0",[ " ***** ", "* * ", "* * * ", "* * *", "* * * ", "* * ", " ***** ", " "]); charSet.SetChar("1",[ " * ", " ** ", " * ", " * ", " * ", " * ", " *** ", " "]); charSet.SetChar("2",[ " ***** ", "* * ", " * ", " * ", " ** ", " * ", "******* ", " "]); charSet.SetChar("3",[ " ***** ", "* * ", " * ", " **** ", " * ", "* * ", " ***** ", " "]); charSet.SetChar("4",[ " * ", " * ", " * ", " * * ", " ****** ", " * ", " * ", " "]); charSet.SetChar("5",[ "******* ", "* ", "* ", "****** ", " * ", "* * ", " ***** ", " "]); charSet.SetChar("6",[ " ***** ", "* * ", "* ", "****** ", "* * ", "* * ", " ***** ", " "]); charSet.SetChar("7",[ "******* ", " * ", " * ", " * ", " * ", " * ", " * ", " "]); charSet.SetChar("8",[ " ***** ", "* * ", "* * ", " ***** ", "* * ", "* * ", " ***** ", " "]); charSet.SetChar("9",[ " ***** ", "* * ", "* * ", " ****** ", " * ", "* * ", " ***** ", " "]); charSet.SetChar(":",[ " ", " ", " * ", " ", " * ", " ", " ", " "]); charSet.SetChar(";",[ " ", " ", " * ", " ", " * ", " * ", " ", " "]); charSet.SetChar("<",[ " ", " ** ", " ** ", " * ", " ** ", " ** ", " ", " "]); charSet.SetChar("=",[ " ", " ", " ***** ", " ", " ***** ", " ", " ", " "]); charSet.SetChar(">",[ " ", " ** ", " ** ", " * ", " ** ", " ** ", " ", " "]); charSet.SetChar("A",[ " ***** ", "* * ", "* * ", "******* ", "* * ", "* * ", "* * ", " "]); charSet.SetChar("B",[ "****** ", "* * ", "* * ", "******* ", "* * ", "* * ", "****** ", " "]); charSet.SetChar("C",[ " ***** ", "* * ", "* ", "* ", "* ", "* * ", " ***** ", " "]); charSet.SetChar("D",[ "****** ", "* * ", "* * ", "* * ", "* * ", "* * ", "****** ", " "]); charSet.SetChar("E",[ "******* ", "* ", "* ", "***** ", "* ", "* ", "******* ", " "]); charSet.SetChar("F",[ "******* ", "* ", "* ", "***** ", "* ", "* ", "* ", " "]); charSet.SetChar("G",[ " ***** ", "* * ", "* ", "* **** ", "* * ", "* * ", " ***** ", " "]); charSet.SetChar("H",[ "* * ", "* * ", "* * ", "******* ", "* * ", "* * ", "* * ", " "]); charSet.SetChar("I",[ " *** ", " * ", " * ", " * ", " * ", " * ", " *** ", " "]); charSet.SetChar("J",[ "******* ", " * ", " * ", " * ", "* * ", "* * ", " ***** ", " "]); charSet.SetChar("K",[ "* * ", "* ** ", "* ** ", "** ", "* ** ", "* ** ", "* * ", " "]); charSet.SetChar("L",[ "* ", "* ", "* ", "* ", "* ", "* ", "******* ", " "]); charSet.SetChar("M",[ "* * ", "** ** ", "* * * * ", "* * * ", "* * ", "* * ", "* * ", " "]); charSet.SetChar("N",[ "* * ", "** * ", "* * * ", "* * * ", "* * * ", "* ** ", "* * ", " "]); charSet.SetChar("O",[ " ***** ", "* * ", "* * ", "* * ", "* * ", "* * ", " ***** ", " "]); charSet.SetChar("P",[ " ***** ", "* * ", "* * ", "****** ", "* ", "* ", "* ", " "]); charSet.SetChar("Q",[ " ***** ", "* * ", "* * ", "* * ", "* * * ", "** * ", " **** * ", " "]); charSet.SetChar("R",[ "****** ", "* * ", "* * ", "****** ", "* * ", "* * ", "* * ", " "]); charSet.SetChar("S",[ " ***** ", "* * ", "* ", " ***** ", " * ", "* * ", " ***** ", " "]); charSet.SetChar("T",[ "******* ", " * ", " * ", " * ", " * ", " * ", " * ", " "]); charSet.SetChar("U",[ "* * ", "* * ", "* * ", "* * ", "* * ", "* * ", "* * ", " ****** "]); charSet.SetChar("V",[ "* * ", "* * ", " * * ", " * * ", " * * ", " * * ", " * ", " "]); charSet.SetChar("W",[ "* * ", "* * ", "* * ", "* * ", "* * * ", "* * * * ", "** ** ", " "]); charSet.SetChar("X",[ "* * ", " * * ", " * * ", " * ", " * * ", " * * ", "* * ", " "]); charSet.SetChar("Y",[ "* * ", " * * ", " * * ", " * ", " * ", " * ", " * ", " "]); charSet.SetChar("Z",[ "******* ", " * ", " * ", " * ", " * ", " * ", "******* ", " "]); // [\]^_`abcdefg.....z{|}~ charSet.SetChar("[",[ " *** ", " * ", " * ", " * ", " * ", " * ", " *** ", " "]); charSet.SetChar("\\",[ "* ", " * ", " * ", " * ", " * ", " * ", " * ", " "]); charSet.SetChar("]",[ " *** ", " * ", " * ", " * ", " * ", " * ", " *** ", " "]); charSet.SetChar("^",[ " * ", " * * ", " * * ", " ", " ", " ", " ", " "]); charSet.SetChar("_",[ " ", " ", " ", " ", " ", " ", " ", "********"]); charSet.SetChar("`",[ " * ", " * ", " ", " ", " ", " ", " ", " "]); charSet.SetChar("a",[ " ", " ", " **** ", " * ", " ***** ", "* * ", " **** * ", " "]); charSet.SetChar("b",[ "* ", "* ", "***** ", "* * ", "* * ", "* * ", "***** ", " "]); charSet.SetChar("c",[ " ", " ", " **** ", "* ", "* ", "* ", " **** ", " "]); charSet.SetChar("d",[ " * ", " * ", " ***** ", "* * ", "* * ", "* * ", " ***** ", " "]); charSet.SetChar("e",[ " ", " ", " **** ", "* * ", "****** ", "* ", " **** ", " "]); charSet.SetChar("f",[ " *** ", " * ", " * ", "**** ", " * ", " * ", " * ", " "]); charSet.SetChar("g",[ " ", " ", " **** ", "* * ", "* * ", " ***** ", " * ", " **** "]); charSet.SetChar("h",[ "* ", "* ", "* ", "***** ", "* * ", "* * ", "* * ", " "]); charSet.SetChar("i",[ " ", " * ", " ", " * ", " * ", " * ", " * ", " "]); charSet.SetChar("j",[ " ", " * ", " ", " ** ", " * ", " * ", " * ", " **** "]); charSet.SetChar("k",[ "* ", "* ", "* * ", "* * ", "*** ", "* * ", "* * ", " "]); charSet.SetChar("l",[ "* ", "* ", "* ", "* ", "* ", "* ", " *** ", " "]); charSet.SetChar("m",[ " ", " ", "*** ** ", "* * * ", "* * * ", "* * * ", "* * * ", " "]); charSet.SetChar("n",[ " ", " ", "***** ", "* * ", "* * ", "* * ", "* * ", " "]); charSet.SetChar("o",[ " ", " ", " **** ", "* * ", "* * ", "* * ", " **** ", " "]); charSet.SetChar("p",[ " ", " ", "***** ", "* * ", "* * ", "***** ", "* ", "* "]); charSet.SetChar("q",[ " ", " ", " ***** ", "* * ", "* * ", " ***** ", " * ", " * "]); charSet.SetChar("r",[ " ", " ", " **** ", "* ", "* ", "* ", "* ", " "]); charSet.SetChar("s",[ " ", " ", " **** ", "* ", " **** ", " * ", " **** ", " "]); charSet.SetChar("t",[ " ", " * ", "*** ", " * ", " * ", " * ", " *** ", " "]); charSet.SetChar("u",[ " ", " ", "* * ", "* * ", "* * ", "* * ", " ***** ", " "]); charSet.SetChar("v",[ " ", " ", "* * ", "* * ", " * * ", " * * ", " ** ", " "]); charSet.SetChar("w",[ " ", " ", "* * * ", "* * * ", "* * * ", "* * * ", " ** ** ", " "]); charSet.SetChar("x",[ " ", " ", "* * ", " * * ", " ** ", " * * ", "* * ", " "]); charSet.SetChar("y",[ " ", " ", "* * ", "* * ", "* * ", " ***** ", " * ", " **** "]); charSet.SetChar("z",[ " ", " ", "***** ", " * ", " * ", " * ", "***** ", " "]); charSet.SetChar("{",[ " * ", " * ", " * ", " * ", " * ", " * ", " * ", " "]); charSet.SetChar("|",[ " * ", " * ", " * ", " * ", " * ", " * ", " * ", " "]); charSet.SetChar("}",[ " * ", " * ", " * ", " * ", " * ", " * ", " * ", " "]); charSet.SetChar("~",[ " ", " ", " * ", " ***** ", "* ", " ", " ", " "]); } } obj.CharSet = function() { return charSet; } InitCharSet(); var dwCurrentTime=0; var dwLastUpdateTime=(new Date).getTime(); var dwTimeCarriedOver = 0; var dwElapsedTime=0; obj.Main = function() { dwCurrentTime = (new Date).getTime(); dwElapsedTime = dwCurrentTime - dwLastUpdateTime + dwTimeCarriedOver; if(dwElapsedTime >= updateRate) { drawFunction(); obj.Draw(); var frames = Math.floor(dwElapsedTime / updateRate ); dwTimeCarriedOver = dwElapsedTime % updateRate; dwLastUpdateTime = dwCurrentTime; for ( var f = 0; f < frames; f++) { updateFunction(); obj.Update(); obj.mouseClicked = false; obj.mouseUnclicked = false; obj.mouseTouchFire = false; obj.mouseSwipe.x = 0; obj.mouseSwipe.y = 0; obj.CheckCollisions(); } } } var timer = Timer(FPS, obj.Main); obj.Stop = function() { timer.Stop(); } obj.writeToCanvas = function(bitmap) { var sourceArray = bitmap.GetPixelArray(); var prevArray = prevScreenBitmap.GetPixelArray(); var offset = 0; for ( var p = 0; p < sourceArray.length; p++ ) { if ( sourceArray[p] != prevArray[p]) { prevArray[p] = sourceArray[p]; var colour = colours[sourceArray[p]]; screenImageData.data[offset] = colour.red; screenImageData.data[offset+1] = colour.green; screenImageData.data[offset+2] = colour.blue; } // else // { // screenImageData.data[offset] = 0; // screenImageData.data[offset+1] = 0; // screenImageData.data[offset+2] = 0; // } offset+=4; } ctx.putImageData(screenImageData,0,0); } obj.Update = function() { for ( var l in obj.layers ) { var currentLayer = obj.layers[l]; for ( var i in currentLayer ) { var s = currentLayer[i]; if ( s ) { s.Update(tempScreenBitmap.width, tempScreenBitmap.height); if( s.died ) SpriteDeleteForReal(s.name); } } } } obj.CustomResolution=function(width, height) { CANVAS_WIDTH = width; CANVAS_HEIGHT = height; obj.CanvasWidth = CANVAS_WIDTH; obj.CanvasHeight = CANVAS_HEIGHT; canvas.width = CANVAS_WIDTH; canvas.height = CANVAS_HEIGHT; screenImageData = ctx.getImageData(0,0,CANVAS_WIDTH,CANVAS_HEIGHT); for ( var pixel = 0; pixel < CANVAS_WIDTH*CANVAS_HEIGHT; pixel+=1) { screenImageData.data[pixel*4+3] = 255; } screenBitmap = Bitmap(CANVAS_WIDTH,CANVAS_HEIGHT); screenBitmap.DrawRect(0,0,CANVAS_WIDTH,CANVAS_HEIGHT,0); tempScreenBitmap= Bitmap(CANVAS_WIDTH,CANVAS_HEIGHT); prevScreenBitmap= Bitmap(CANVAS_WIDTH,CANVAS_HEIGHT); } obj.PortraitMode = function() { if (CANVAS_WIDTH > CANVAS_HEIGHT ) { var tmp = CANVAS_WIDTH; CANVAS_WIDTH = CANVAS_HEIGHT; CANVAS_HEIGHT = tmp; obj.CanvasWidth = CANVAS_WIDTH; obj.CanvasHeight = CANVAS_HEIGHT; canvas.width = CANVAS_WIDTH; canvas.height = CANVAS_HEIGHT; screenImageData = ctx.getImageData(0,0,CANVAS_WIDTH,CANVAS_HEIGHT); for ( var pixel = 0; pixel < CANVAS_WIDTH*CANVAS_HEIGHT; pixel+=1) { screenImageData.data[pixel*4+3] = 255; } screenBitmap = Bitmap(CANVAS_WIDTH,CANVAS_HEIGHT); screenBitmap.DrawRect(0,0,CANVAS_WIDTH,CANVAS_HEIGHT,0); tempScreenBitmap= Bitmap(CANVAS_WIDTH,CANVAS_HEIGHT); prevScreenBitmap= Bitmap(CANVAS_WIDTH,CANVAS_HEIGHT); } } obj.CheckCollisions = function() { for ( var r in obj.collisionRules ) { var currentLayer = obj.layers[r]; CheckCollisionsForLayer(currentLayer, obj.collisionRules[r]); } } function CheckCollisionsForLayer(layer, collisionRule) { for ( var spritePos in layer ) { var sprite = layer[spritePos]; if ( sprite ) { sprite.collided = false; for ( var targetLayer in collisionRule) { CheckSpriteCollisionWithLayer(sprite, obj.layers[targetLayer]); } } } } function CheckSpriteCollisionWithLayer(sprite, layer) { if ( layer ) { for ( var spritePos in layer ) { var targetSprite = layer[spritePos]; if( sprite.IsCollidedWith(targetSprite.name)) { sprite.collided = true; targetSprite.collided = true; } } } } obj.CountSpritesInLayer=function(layerName) { var count = 0; var layer = obj.layers[layerName]; for ( var spritePos in layer ) { var sprite = layer[spritePos]; if ( sprite ) { count++; } } return count; } obj.Draw = function() { // copy the screenBitmap and place all the sprites on it // Cloning has been replaced with a permanent temp bitmap and // contents just copied over. This seems to yield significant // speed improvement in chrome. // var bitmap = screenBitmap.Clone(); var size = screenBitmap.width * screenBitmap.height; var targetArray = tempScreenBitmap.GetPixelArray(); var sourceArray = screenBitmap.GetPixelArray(); for ( var p = 0; p < size; p++) { targetArray[p] = sourceArray[p]; } for ( var l in obj.layers ) { var currentLayer = obj.layers[l]; for ( var i in currentLayer ) { var spr = currentLayer[i]; if ( spr ) tempScreenBitmap.Paste(spr.pos.x,spr.pos.y, spr.bitmap ) // bitmap.Paste(s.pos.x,s.pos.y, s.bitmap ) } } obj.writeToCanvas(tempScreenBitmap); // obj.writeToCanvas(bitmap); } obj.DrawPixel = function(pixelX, pixelY,colour) { screenBitmap.SetPixel(Math.floor(pixelX),Math.floor(pixelY),colour); } obj.Clear = function(colour) { screenBitmap.Clear(colour); } obj.DrawLine = function(startX, startY, endX, endY, colour) { screenBitmap.DrawLine(startX, startY, endX, endY, colour) } obj.DrawRect = function(xPos,yPos,width,height,colour) { screenBitmap.DrawRect(xPos, yPos, width, height, colour) } obj.DrawCircle = function(xPos,yPos,radius,colour) { screenBitmap.DrawCircle(xPos,yPos,radius,colour) } obj.Scroll = function(xMovement,yMovement) { screenBitmap.Scroll(xMovement,yMovement); } obj.FlipX = function() { screenBitmap = screenBitmap.FlipX(); } obj.FlipY = function() { screenBitmap = screenBitmap.FlipY(); } obj.Paste = function(xPos,yPos,bitmap) { screenBitmap.Paste(xPos,yPos,bitmap); } obj.DefineColour = function(colour,r,g,b) { colours[colour] = Colour(r,g,b); } function CleanUpArray(arrayObj) { var returnArray = new Array(); for ( var i in arrayObj) { if ( arrayObj[i] != null ) { returnArray[i] = arrayObj[i]; } } return returnArray; } function CountItemsInArray(arrayObj) { var count = 0; for ( var i in arrayObj ) { count++; } return count; } obj.SpriteAdd = function(name,sprite, layer) { obj.SpriteDelete(name); obj.sprites[name] = layer; if (obj.layers[layer] == null ) { obj.layers[layer] = new Array(); } var newLayer = obj.layers[layer]; newLayer[name] = sprite; sprite.name = name; } function SpriteDeleteForReal(name) { var spriteLayer = obj.sprites[name]; if ( spriteLayer ) { obj.layers[spriteLayer][name] = null; obj.layers[spriteLayer] = CleanUpArray(obj.layers[spriteLayer]); if ( CountItemsInArray(obj.layers[spriteLayer]) == 0 ) { obj.layers[spriteLayer] = null; obj.layers = CleanUpArray(obj.layers); } obj.sprites[name] = null; obj.sprites = CleanUpArray(obj.sprites); } } obj.SpriteDelete=function(name) { var spriteLayer = obj.sprites[name]; if ( spriteLayer ) { if ( obj.layers[spriteLayer] ) obj.layers[spriteLayer][name].died = true; } } obj.DeleteSpritesInLayer=function(spriteLayer) { if ( spriteLayer ) { if ( obj.layers[spriteLayer] ) { for ( var i in obj.layers[spriteLayer] ) { obj.layers[spriteLayer][i].died = true; } } } } // obj.GetSpritesInLayer = function(spriteLayer) // { // if ( spriteLayer ) // { // if ( obj.layers[spriteLayer] ) // { // return obj.layers[spriteLayer] // } // } // return null; // } /* obj.WriteChar = function(char, charset, xPos,yPos,colour) { screenBitmap.WriteChar(char, charset, xPos,yPos,colour); } obj.WriteString = function(text,charset, xPos,yPos,colour) { screenBitmap.WriteString(text,charset, xPos,yPos,colour); } */ obj.SpriteGet = function(name) { var spriteLayer = obj.layers[obj.sprites[name]]; if ( spriteLayer ) return spriteLayer[name]; return null; } obj.CollisionRuleAdd = function(sourceLayer, targetLayer) { if ( obj.collisionRules[sourceLayer] == null ) { obj.collisionRules[sourceLayer] = new Array(); } if (obj.collisionRules[sourceLayer][targetLayer] == null ) { obj.collisionRules[sourceLayer][targetLayer] = targetLayer; } } obj.CollisionRuleDelete = function(sourceLayer, targetLayer) { if ( obj.collisionRules[sourceLayer] ) { if ( obj.collisionRules[sourceLayer][targetLayer] ) { obj.collisionRules[sourceLayer][targetLayer] = null; obj.collisionRules[sourceLayer] = CleanUpArray(obj.collisionRules[sourceLayer]); } if ( CountItemsInArray(obj.collisionRules[sourceLayer]) == 0 ) { obj.collisionRules[sourceLayer] = null; obj.collisionRules = CleanUpArray(obj.collisionRules); } } } obj.WriteChar = function(char,xPos,yPos,colour,xbig,ybig) { charSet.WriteChar(char,screenBitmap,xPos,yPos,colour,xbig,ybig) } obj.WriteString = function(text,xPos,yPos,colour,xbig,ybig) { charSet.WriteString(text,screenBitmap,xPos,yPos,colour,xbig,ybig) } obj.WriteStringCentered=function(text,yPos,colour,xbig,ybig) { screenBitmap.WriteStringCentered(text,charSet, yPos,colour,xbig,ybig); } var mouseDownPos = Vector(0,0); var mouseUpPos = Vector(0,0); obj.mouseSwipe = Vector(0,0); function SetMousePos(xp,yp) { var bbox = canvas.getBoundingClientRect(); obj.mousePos.x = (xp - bbox.left) * ( canvas.width / bbox.width); obj.mousePos.y = (yp - bbox.top) * ( canvas.height / bbox.height); } canvas.addEventListener("mousemove", function(evt){ SetMousePos(evt.clientX,evt.clientY); evt.preventDefault(); }); canvas.addEventListener("touchmove", function(evt){ SetMousePos(evt.targetTouches[0].pageX,evt.targetTouches[0].pageY); evt.preventDefault(); }); canvas.addEventListener("mousedown", function(evt){ obj.mouseClicked = true; obj.mouseDown = true; mouseDownPos.x = obj.mousePos.x; mouseDownPos.y = obj.mousePos.y; evt.preventDefault(); }); canvas.addEventListener("touchstart", function(evt){ SetMousePos(evt.targetTouches[0].pageX,evt.targetTouches[0].pageY); obj.mouseClicked = true; obj.mouseDown = true; mouseDownPos.x = obj.mousePos.x; mouseDownPos.y = obj.mousePos.y; if ( evt.targetTouches > 1 ) obj.mouseTouchFire = true; evt.preventDefault(); }); canvas.addEventListener("mouseup", function(evt){ obj.mouseDown = false; obj.mouseUnclicked = true; mouseUpPos.x = obj.mousePos.x; mouseUpPos.y = obj.mousePos.y; obj.mouseSwipe.x = mouseUpPos.x - mouseDownPos.x; obj.mouseSwipe.y = mouseUpPos.y - mouseDownPos.y; evt.preventDefault(); }); canvas.addEventListener("touchend", function(evt){ obj.mouseDown = false; obj.mouseUnclicked = true; mouseUpPos.x = obj.mousePos.x; mouseUpPos.y = obj.mousePos.y; obj.mouseSwipe.x = mouseUpPos.x - mouseDownPos.x; obj.mouseSwipe.y = mouseUpPos.y - mouseDownPos.y; evt.preventDefault(); }); var tiltable = false; obj.tiltX = null; obj.tiltY = null; obj.tiltZ = null; var tiltXBase = 0; var tiltYBase = 0; var tiltZBase = 0; obj.Tilt=function(values) { obj.tiltX = values[0]; obj.tiltY = values[1]; obj.tiltZ = values[2]; if ( tiltable == false && obj.tiltX != null ) { tiltable = true; obj.TiltCalibrate(); } } obj.TiltCalibrate = function() { tiltXBase = obj.tiltX; tiltYBase = obj.tiltY; tiltZBase = obj.tiltZ; } obj.Tiltable = function() { return tiltable; } obj.GetTiltX=function() { var upsidedown = 1; if(obj.tiltZ > 0 ) upsidedown = -1; if ( window.innerWidth < window.innerHeight ) return (obj.tiltX-tiltXBase); else return ( obj.tiltY - tiltYBase )*upsidedown; } // Tilt events if (window.DeviceOrientationEvent) { window.addEventListener("deviceorientation", function () { obj.Tilt([event.gamma,event.beta, event.gamma]); }, true); } // keyboard handler var keycodes = []; obj.GetKeyState=function(keycode) { return keycodes[keycode]; } document.addEventListener("keydown", function(evt){ keycodes[evt.keyCode] = true; }); document.addEventListener("keyup", function(evt){ keycodes[evt.keyCode] = false; }); obj.KEY_ENTER = 13; obj.KEY_ESCAPE = 27; obj.KEY_SPACE = 32; obj.KEY_LEFT = 37; obj.KEY_UP = 38; obj.KEY_RIGHT = 39; obj.KEY_DOWN = 40; obj.KEY_0 = 48; obj.KEY_1 = 49; obj.KEY_2 = 50; obj.KEY_3 = 51; obj.KEY_4 = 52; obj.KEY_5 = 53; obj.KEY_6 = 54; obj.KEY_7 = 55; obj.KEY_8 = 56; obj.KEY_9 = 57; obj.KEY_A = 65; obj.KEY_B = 66; obj.KEY_C = 67; obj.KEY_D = 68; obj.KEY_E = 69; obj.KEY_F = 70; obj.KEY_G = 71; obj.KEY_H = 72; obj.KEY_I = 73; obj.KEY_J = 74; obj.KEY_K = 75; obj.KEY_L = 76; obj.KEY_M = 77; obj.KEY_N = 78; obj.KEY_O = 79; obj.KEY_P = 80; obj.KEY_Q = 81; obj.KEY_R = 82; obj.KEY_S = 83; obj.KEY_T = 84; obj.KEY_U = 85; obj.KEY_V = 86; obj.KEY_W = 87; obj.KEY_X = 88; obj.KEY_Y = 89; obj.KEY_Z = 90; return obj; } Timer = function(fps,callFunction) { var obj = {} var fps = fps; var callFunction = callFunction var timerId = setInterval ( callFunction, Math.floor(1000/fps)); obj.Stop = function() { clearInterval(timerId); } return obj; } Colour = function(red,green,blue) { obj = {} obj.red = red; obj.green = green; obj.blue = blue; return obj; } var BaseBitmap = {}; Bitmap = function(width,height) { var obj = Object.create(BaseBitmap); obj.width = width; obj.height = height; var pixels = new Array() var scrollXFraction = 0.0; var scrollYFraction = 0.0; obj.GetPixelArray = function () { return pixels; } obj.SetPixelArray = function (newArray) { pixels = newArray; } obj.GetPixel = function (xpos, ypos) { if (xpos >= 0 && xpos < obj.width && ypos >= 0 && ypos < obj.height) { return pixels[ypos*width+xpos]; } return 0; } obj.SetPixel = function (xpos,ypos,colour) { if (xpos >= 0 && xpos < obj.width && ypos >= 0 && ypos < obj.height) { pixels[Math.floor(ypos)*width+Math.floor(xpos)] = colour; } } obj.Clear = function (colour) { for ( var ypos = 0; ypos < obj.height; ypos++) { for ( var xpos = 0; xpos < obj.width; xpos++) { obj.SetPixel(xpos,ypos,colour); } } } obj.PixelsFromArray = function(pixelsIn) { var minPixels = Math.min(obj.width*obj.height,pixelsIn.length); for(var p = 0; p < minPixels; p++) { pixels[p] = pixelsIn[p]; } return obj; } obj.DrawLine = function(startX, startY, endX, endY, colour) { var xLen = endX - startX; var yLen = endY - startY; var maxPoints = Math.max(Math.abs(xLen), Math.abs(yLen)) var xDisp = xLen / maxPoints; var yDisp = yLen / maxPoints; for ( var p = 0; p <= maxPoints; p++) { obj.SetPixel(startX + p*xDisp+0.5, startY + p*yDisp+0.5, colour) } } obj.DrawRect = function(xPos,yPos,width,height,colour) { for(var yDisp = 0; yDisp < height; yDisp++ ) { obj.DrawLine(xPos, yPos+yDisp, xPos+width-1, yPos+yDisp,colour) } } obj.DrawCircle = function(xPos,yPos,radius,colour) { var radiusSquared = radius * radius; var xOff; for( var yOff = 0.0; yOff < radius; yOff+= 0.5) { xOff = Math.sqrt(radiusSquared-(yOff*yOff)); obj.DrawLine(xPos-xOff+.5,yPos+yOff+.5,xPos+xOff+.5,yPos+yOff,colour) obj.DrawLine(xPos-xOff+.5,yPos-yOff+.5,xPos+xOff+.5,yPos-yOff,colour) } } obj.Scroll = function(xMovement,yMovement) { scrollXFraction += xMovement; scrollYFraction += yMovement; var xMove = Math.round(scrollXFraction); var yMove = Math.round(scrollYFraction); if(xMove == 0 && yMove == 0) { return; } scrollXFraction -= xMove; scrollYFraction -= yMove; var newPixels = new Array(); var targetXPos = xMove; if(targetXPos < 0) { targetXPos += obj.width; } var targetYPos = yMove if(targetYPos < 0) { targetYPos += obj.height; } for(var yPos = 0; yPos < obj.height; yPos++) { for(var xPos = 0; xPos < obj.width; xPos++) { newPixels[yPos*obj.width+xPos] = pixels[targetYPos*obj.width+targetXPos]; targetXPos += 1 if(targetXPos >= obj.width) { targetXPos -= obj.width; } } targetYPos += 1 if(targetYPos >= obj.height) { targetYPos -= obj.height; } } pixels = newPixels; } // Not ready for relacing scroll function yet. Does not wrap pixels. obj.ScrollNew = function(xMovement,yMovement) { scrollXFraction -= xMovement; scrollYFraction -= yMovement; var xMove = Math.round(scrollXFraction); var yMove = Math.round(scrollYFraction); if(xMove == 0 && yMove == 0) { return; } scrollXFraction -= xMove; scrollYFraction -= yMove; var newBitmap = obj.Clone(); obj.Clear(-1); var targetXPos = xMove; if(targetXPos < 0) { targetXPos += obj.width; } var targetYPos = yMove if(targetYPos < 0) { targetYPos += obj.height; } obj.Paste(xMove,yMove,newBitmap); if(targetYPos > 0 ) obj.Paste(xMove,yMove-obj.height,newBitmap); if(targetYPos < 0 ) obj.Paste(xMove,yMove+obj.height,newBitmap); // obj.Paste(0,0,newBitmap); } obj.FlipX = function () { var newBitmap = Bitmap(obj.width,obj.height); for(var yPos = 0; yPos < obj.height; yPos++) { for(var xPos = 0; xPos < obj.width; xPos++) { newBitmap.SetPixel(xPos,yPos,obj.GetPixel(obj.width-1-xPos,yPos)); } } return newBitmap; } obj.FlipY = function() { var newBitmap = Bitmap(obj.width,obj.height); for(var yPos = 0; yPos < obj.height; yPos++) { for(var xPos = 0; xPos < obj.width; xPos++) { newBitmap.SetPixel(xPos,yPos,obj.GetPixel(xPos,obj.height-1-yPos)); } } return newBitmap; } obj.RotateRight = function() { var newBitmap = Bitmap(obj.width,obj.height); for(var yPos = 0; yPos < obj.height; yPos++) { for(var xPos = 0; xPos < obj.width; xPos++) { newBitmap.SetPixel(yPos,xPos,obj.GetPixel(xPos,obj.height-1-yPos)); } } return newBitmap; } obj.RotateLeft = function() { var newBitmap = Bitmap(obj.width,obj.height); for(var yPos = 0; yPos < obj.height; yPos++) { for(var xPos = 0; xPos < obj.width; xPos++) { newBitmap.SetPixel(yPos,xPos,obj.GetPixel(obj.width-1-xPos,yPos)); } } return newBitmap; } obj.Clone = function() { var newBitmap = Bitmap(obj.width,obj.height); newBitmap.SetPixelArray([].concat(pixels)); // for(var yPos = 0; yPos < obj.height; yPos++) // { // for(var xPos = 0; xPos < obj.width; xPos++) // { // newBitmap.SetPixel(xPos,yPos,obj.GetPixel(xPos,yPos)); // } // } return newBitmap; } obj.Paste = function(xp,yp,bitmap) { var xdisp = Math.round(xp); var ydisp = Math.round(yp); // find the area of overlap // calculate the start y pixel in source and target var srcY = 0; var destY = 0; if (ydisp>0) destY = ydisp; if (ydisp<0) srcY = -ydisp; // calculate the start x pixel in source and target var srcX = 0; var destX = 0; if (xdisp>0) destX = xdisp; if (xdisp<0) srcX = -xdisp; // calculate the height of the overlap var ydist = obj.height-destY; if(bitmap.height-srcY < ydist) ydist = bitmap.height-srcY; // calculate the width of the overlap var xdist = obj.width-destX; if(bitmap.width-srcX < xdist) xdist = bitmap.width-srcX; if ( ydist > 0 && xdist > 0) { for ( var y = 0; y < ydist; y++) { for(var xPos = 0; xPos < bitmap.width; xPos++) { if ( bitmap.GetPixel(xPos,srcY) >= 0 ) { obj.SetPixel(xPos+xdisp,destY,bitmap.GetPixel(xPos,srcY)); } } srcY++; destY++; } } /* for(var yPos = 0; yPos < bitmap.height; yPos++) { if ( yPos +ydisp >= 0 && yPos+ydisp < obj.height) for(var xPos = 0; xPos < bitmap.width; xPos++) { if ( bitmap.GetPixel(xPos,yPos) >= 0 ) obj.SetPixel(xPos+xdisp,yPos+ydisp,bitmap.GetPixel(xPos,yPos)); } } */ } obj.WriteChar = function(char, charset, xPos,yPos,colour,xbig,ybig) { charSet.WriteChar(char,obj,xPos,yPos,colour,xbig,ybig); } obj.WriteString = function(text,charSet, xPos,yPos,colour,xbig,ybig) { charSet.WriteString(text,obj,xPos,yPos,colour,xbig,ybig); } obj.WriteStringCentered = function(text, charset, yPos,colour,xbig,ybig) { if (xbig == undefined) xbig = false; if ( xbig ) charset.WriteString(text,obj,width/2-charset.GetWidth()/2*text.length*2,yPos, colour,xbig,ybig); else charset.WriteString(text,obj,width/2-charset.GetWidth()/2*text.length,yPos, colour,xbig,ybig); } obj.Clear(-1); return obj; } CharSet = function (width, height) { if (width == undefined) width = 8; if (height == undefined) height = 8; var obj = {}; var charData = {}; obj.GetWidth = function () { return width; } obj.GetHeight = function () { return height; } obj.SetChar = function(char, chardata ) { charData[char] = chardata; } obj.WriteChar = function(char,bitmap,xPos,yPos,colour,xbig,ybig) { if(typeof(xbig)==='undefined') xbig = false; if(typeof(ybig)==='undefined') ybig = false; var rowData = charData[char]; if ( rowData != undefined ) { for ( var row = 0; row < rowData.length; row++) { var line = rowData[row]; for ( var charPos = 0; charPos < line.length; charPos++) { var chr = line.charAt(charPos); if ( chr != " " ) { if ( !xbig && !ybig) bitmap.SetPixel(charPos+xPos, row+yPos, colour ); if ( xbig && !ybig) { bitmap.SetPixel(charPos*2+xPos, row+yPos, colour ); bitmap.SetPixel(charPos*2+xPos+1, row+yPos, colour ); } if ( !xbig && ybig) { bitmap.SetPixel(charPos+xPos, row*2+yPos, colour ); bitmap.SetPixel(charPos+xPos, row*2+yPos+1, colour ); } if ( xbig && ybig) { bitmap.SetPixel(charPos*2+xPos, row*2+yPos, colour ); bitmap.SetPixel(charPos*2+xPos+1, row*2+yPos, colour ); bitmap.SetPixel(charPos*2+xPos, row*2+yPos+1, colour ); bitmap.SetPixel(charPos*2+xPos+1, row*2+yPos+1, colour ); } } } } } } obj.WriteString = function(text,bitmap,xPos,yPos,colour,xbig,ybig) { if(typeof(xbig)==='undefined') xbig = false; for ( var charPos = 0; charPos < text.length; charPos++) { var chr = text.charAt(charPos); if ( xbig) obj.WriteChar(chr,bitmap,xPos+charPos*2*width, yPos, colour,xbig,ybig ); else obj.WriteChar(chr,bitmap,xPos+charPos*width, yPos, colour,xbig,ybig ); } } return obj; } Sprite = function (xp,yp,costumeOrBitmap) { if (costumeOrBitmap === null ) return null; var costume = costumeOrBitmap; if ( BaseBitmap.isPrototypeOf(costumeOrBitmap)) { costume = Costume(); costume.SetItem(0,costumeOrBitmap); } else { if ( !BaseCostume.isPrototypeOf(costumeOrBitmap)) return null; } var obj = {}; var lifeSpan = -1; obj.died = false; obj.name = ""; obj.pos = Vector(xp,yp); obj.vel = Vector(0,0); obj.costume = costume; obj.costumeAnimator = Animator([0]); obj.bitmap = costume.GetItem(0); obj.wrap = false; obj.collided = false; obj.collidedList = new Array(); obj.SetLifeSpan = function(newLifeSpan) { lifeSpan = newLifeSpan; } obj.SetCostume = function(costume) { obj.costume = costume; } obj.SetWrapOn = function() { obj.wrap = true; } obj.SetWrapOff = function() { obj.wrap = false; } obj.SetAnimator = function(frameList) { obj.costumeAnimator = Animator(frameList); } obj.SetAnimatorBounce = function () { obj.costumeAnimator.Bounce(obj.costume.Length()); } obj.SetAnimatorLoop = function () { obj.costumeAnimator.Loop(obj.costume.Length()); } obj.SetAnimatorFrame = function(frame) { obj.costumeAnimator.Set(frame); obj.bitmap = obj.costume.GetItem(obj.costumeAnimator.Get()); } obj.GetAnimatorFrame = function(frame) { return obj.costumeAnimator.Get(); } obj.SetCostumeAnimator = function(animator) { obj.costumeAnimator = animator; } obj.SetAnimatorUpdateRate = function (updateRate) { obj.costumeAnimator.SetUpdateRate(updateRate); } obj.SetAnimatorStop = function () { obj.costumeAnimator.Stop(); } obj.SetMovement = function(xm,ym) { obj.vel.x = xm; obj.vel.y = ym; return obj; } obj.IsCollidedWith = function(targetSpriteName) { var target = GE.SpriteGet(targetSpriteName); if ( target == undefined ) return false; // simple bounding box check will do for now if ( obj.pos.x < target.pos.x + target.bitmap.width && obj.pos.x + obj.bitmap.width > target.pos.x && obj.pos.y < target.pos.y + target.bitmap.height && obj.pos.y + obj.bitmap.height > target.pos.y ) { // return true; var collided = false; // now lets do a pixel perfect collision check // get the biggest left var boxLeftPos = obj.pos.x; if ( boxLeftPos < target.pos.x ) boxLeftPos = target.pos.x; // get the smallest right var boxRightPos = obj.pos.x + obj.bitmap.width; if ( boxRightPos > target.pos.x + target.bitmap.width) boxRightPos = target.pos.x + target.bitmap.width; // get the biggest top var boxTopPos = obj.pos.y; if ( boxTopPos < target.pos.y ) boxTopPos = target.pos.y; // get the smallest bottom var boxBottomPos = obj.pos.y + obj.bitmap.height; if ( boxBottomPos > target.pos.y + target.bitmap.height) boxBottomPos = target.pos.y + target.bitmap.height; var overlapWidth = boxRightPos - boxLeftPos; var overlapHeight = boxBottomPos - boxTopPos; var sx = Math.floor(boxLeftPos - obj.pos.x); var sy = Math.floor(boxTopPos - obj.pos.y); var tx = Math.floor(boxLeftPos - target.pos.x); var ty = Math.floor(boxTopPos - target.pos.y); for ( var y = 0; y < overlapHeight; y++) { for ( var x = 0; x < overlapWidth; x++ ) { var p1 = obj.bitmap.GetPixel(sx+x,sy+y); var p2 = target.bitmap.GetPixel(tx+x,ty+y); if (p1 != -1 && p2 != -1 ) collided = true; } } return collided; } return false; } obj.Update = function (scrWidth, scrHeight) { obj.pos.x += obj.vel.x; obj.pos.y += obj.vel.y; if ( obj.wrap ) { if ( obj.pos.x < 0 ) obj.pos.x += scrWidth; if ( obj.pos.y < 0 ) obj.pos.y += scrHeight; if ( obj.pos.x >= scrWidth ) obj.pos.x -= scrWidth; if ( obj.pos.y >= scrHeight ) obj.pos.y -= scrHeight; } obj.costumeAnimator.Update(); obj.bitmap = obj.costume.GetItem(obj.costumeAnimator.Get()); obj.layerChanged = false; if ( lifeSpan != -1 ) { lifeSpan--; if (lifeSpan == 0 ) obj.died = true; } } // obj.Draw = function () // { // // } return obj; } Vector = function(x,y) { var obj = {}; obj.x = x; obj.y = y; obj.Set = function(x,y) { obj.x = x; obj.y = y; } obj.Add = function(v1) { return Vector(obj.x+v1.x, obj.y+v1.y); } obj.Subtract = function(v1) { return Vector(obj.x-v1.x, obj.y-v1.y); } obj.Multiply = function(v1) { return Vector(obj.x*v1.x, obj.y*v1.y); } obj.Magnitude = function() { return Math.sqrt(obj.x*obj.x + obj.y*obj.y); } obj.Normalise = function() { var mag = obj.Magnitude(); obj.x /= mag; obj.y /= mag; } obj.SetMagnitude = function(newMagnitude) { obj.Normalise(); obj.x *= newMagnitude; obj.y *= newMagnitude; } obj.DotProduct = function(v) { return obj.x * v.x + obj.y * v.y; } obj.ScalarMultiply = function(scalar) { obj.x *= scalar; obj.y *= scalar; } return obj; } var BaseCostume = {}; Costume = function() { var obj = Object.create(BaseCostume); obj.bitmaps = Array(); obj.SetItem = function (frame, bitmap) { // try to set a frame to something other than a bitmap will result in nothing happening if ( !BaseBitmap.isPrototypeOf(bitmap)) return; obj.bitmaps[frame] = bitmap; }; obj.SetItems = function (bitmaps,startFrame) { var frame = startFrame; for(var bitmapPos = 0; bitmapPos < bitmaps.length; bitmapPos++) { obj.SetItem(frame, bitmaps[bitmapPos]); frame++; } }; obj.Length = function () { return obj.bitmaps.length; } obj.GetItem = function (frame) { return obj.bitmaps[frame]; } obj.Update = function() { } return obj; } Animator = function (frameList) { var obj = {}; obj.frame = 0; obj.updateRate = 1; obj.repeat = true; if (frameList == null ) { obj.frameList = new Array(); } else { obj.frameList = frameList; } var paused = false; obj.Bounce = function (frames) { var list = new Array(); var f = 0; for ( var i = 0; i < frames; i++) { list[f] = i; f++; } for ( var i = frames-2; i > 0; i--) { list[f] = i; f++; } obj.frameList = list; } obj.Loop = function (frames) { var list = new Array(); var f = 0; for ( var i = 0; i < frames; i++) { list[f] = i; f++; } obj.frameList = list; } obj.Update = function () { if ( paused == false ) { var prevFrame = obj.frame; obj.frame = (obj.frame + obj.updateRate) % obj.frameList.length; if ( obj.frame < prevFrame ) if ( !obj.repeat ) { obj.frame = prevFrame; obj.Pause(); } } } obj.Get = function() { return obj.frameList[Math.floor(obj.frame)]; } obj.Set = function(frame) { obj.frame = Math.floor(frame % obj.frameList.length); } obj.SetUpdateRate = function(updateRate) { obj.updateRate = updateRate; obj.waitTime = 0; } obj.Start = function () { frame = 0; paused = false; } obj.Resume = function () { paused = false; } obj.Stop = function () { frame = 0; paused = true; } obj.Pause = function () { paused = true; } obj.Resume = function () { paused = false; } return obj; } // Experimental code to create sounds. // Once this is stable the GameEngine will assimilate it var channels = new Array(); var soundOn = true; function customSound(bitrate) { this.bitrate = bitrate; this.angle = 0; this.customSoundData=[]; this.totalDuration = 0; this.dataPos = 0; customSound.prototype.add = function(duration, startFreq, endFreq, startVol, endVol, startNoise, endNoise, randomFreq, randomRate, square ) { var vol = startVol; var freq = startFreq; var angleInc = (2*Math.PI)/this.bitrate * freq; var noise = startNoise; var freqRand = 0; var randomCount = 0; if ( randomFreq == undefined ) randomFreq = 0; for ( var x=0; x < this.bitrate*duration; x++) { vol = (startVol + (endVol - startVol)*x/(this.bitrate*duration)); // this.customSoundData[this.dataPos] = (Math.sin(this.angle) + (Math.random()*noise)) * vol; this.customSoundData[this.dataPos] = (Math.sin(this.angle) ) * vol; if (square ) if ( this.customSoundData[this.dataPos] < 0) this.customSoundData[this.dataPos] = -vol; else this.customSoundData[this.dataPos] = vol freq = startFreq + (endFreq - startFreq)*x/(this.bitrate*duration); noise = startNoise + (endNoise - startNoise)*x/(this.bitrate*duration); angleInc = (2*Math.PI)/this.bitrate * (freq+freqRand); this.angle+=angleInc; if ( this.angle > Math.PI * 2 ) { if ( randomRate > 0 ) { randomCount--; if ( randomCount <= 0 ) { freqRand = Math.random()*randomFreq; randomCount=randomRate; } } this.angle -= Math.PI * 2; } this.dataPos++; } } customSound.prototype.createSound = function(channel) { var n = this.customSoundData.length; var header = "RIFF****WAVEfmt \x10\x00\x00\x00\x01\x00\x01\x00********\x01\x00\x08\x00data****"; // ChunkSize var numval = n + 36; header=insertLong(header,4,numval); numval = this.bitrate; // byterate header = insertLong(header,24,numval); // BitRate numval = this.bitrate * 8; //44100; header = insertLong(header,28,numval); // Subchunk2Size // insertLong(n); numval = n; header = insertLong(header,40,numval); // Output sound data for (var i = 0; i < n; ++i) { var charCode = Math.round(Math.min(127, Math.max(-127, this.customSoundData[i]))+127); header += String.fromCharCode(charCode); } // var o= document.getElementById('output'); var h = btoa(header); h = 'data:audio/wav;base64,' + h; var a = new Audio(); a.src = h; if ( channels[channel] ) channels[channel].pause(); channels[channel] = a; return a; } } function soundPlay(channel,loop) { if ( !soundOn ) return; if ( loop ) channels[channel].loop = loop; if (channels[channel].readyState < 1 ) return; // channels[channel].pause(); channels[channel].currentTime=0; channels[channel].play(); } function soundStop(channel) { if (channels[channel].readyState < 1 ) return; channels[channel].pause(); } function insertLong(inString, index, inValue) { var retString = inString.substr(0,index); for (i = 0; i < 4; ++i) { retString += String.fromCharCode(inValue & 255); inValue = inValue >> 8; } retString += inString.substr(index+4); return retString; } GameSatePattern = function (data) { var obj = {}; var innerStates = new Array(); var finished = false; var nextState; var doInit = true; obj.Draw = function() { for( var state in innerStates ) if ( innerStates[state] ) innerStates[state].Draw(); if ( obj.DrawCustom ) obj.DrawCustom(); } obj.Update = function() { if ( doInit ) { doInit = false; if ( obj.Init ) obj.Init(); } if ( obj.UpdateCustom) obj.UpdateCustom(); if ( finished ) { for( var state in innerStates ) { if ( innerStates[state] ) { innerStates[state].SetFinished(); innerStates[state] = innerStates[state].Update(); } } if ( obj.TearDown ) obj.TearDown(); if ( nextState) return nextState; return undefined; } else for( var state in innerStates ) if ( innerStates[state] ) innerStates[state] = innerStates[state].Update(); return this; } obj.AddInnerState=function(state) { innerStates.push(state); } obj.SetNextState=function(state) { nextState = state; } obj.SetFinished=function() { finished = true; } return obj; } // last piece of code is to start the game init(); </script> </html>