(function(){ 'use strict';
var vendors = [ 'webkit', 'moz' ];
for( var x = 0; x < vendors.length && !window.requestAnimationFrame; ++x ) {
window.requestAnimationFrame = window[ vendors[ x ] + 'RequestAnimationFrame' ];
window.cancelAnimationFrame = window[ vendors[ x ] + 'CancelAnimationFrame' ] || window[ vendors[ x ] + 'CancelRequestAnimationFrame' ];
if( !window.requestAnimationFrame ) {
window.requestAnimationFrame = function( callback, element ) {
var currTime = new Date().getTime();
var timeToCall = Math.max( 0, 16 - ( currTime - lastTime ) );
var id = window.setTimeout(
callback( currTime + timeToCall );
lastTime = currTime + timeToCall;
if( !window.cancelAnimationFrame ) {
window.cancelAnimationFrame = function( id ) {
(function(){ 'use strict';
function hasClass( elem, className ) {
return new RegExp( ' ' + className + ' ' ).test( ' ' + elem.className + ' ' );
function addClass( elem, className ) {
if( !hasClass(elem, className ) ) {
elem.className += ' ' + className;
function removeClass( elem, className ) {
var newClass = ' ' + elem.className.replace( /[\t\r\n]/g, ' ' ) + ' ';
if( hasClass( elem, className ) ) {
while( newClass.indexOf(' ' + className + ' ' ) >= 0 ) {
newClass = newClass.replace( ' ' + className + ' ', ' ' );
elem.className = newClass.replace( /^\s+|\s+$/g, '' );
function toggleClass( elem, className ) {
var newClass = ' ' + elem.className.replace( /[\t\r\n]/g, ' ' ) + ' ';
if( hasClass(elem, className ) ) {
while( newClass.indexOf( ' ' + className + ' ' ) >= 0 ) {
newClass = newClass.replace( ' ' + className + ' ' , ' ' );
elem.className = newClass.replace( /^\s+|\s+$/g, '' );
elem.className += ' ' + className;
(function(){ 'use strict';
g.mathProps = 'E LN10 LN2 LOG2E LOG10E PI SQRT1_2 SQRT2 abs acos asin atan ceil cos exp floor log round sin sqrt tan atan2 pow max min'.split( ' ' );
for ( var i = 0; i < g.mathProps.length; i++ ) {
g[ g.mathProps[ i ] ] = g.m[ g.mathProps[ i ] ];
g.isset = function( prop ) {
return typeof prop != 'undefined';
if( g.isset( g.config ) && g.config.debug && window.console ){
console.log( Array.prototype.slice.call( arguments ) );
(function(){ 'use strict';
g.Group.prototype.add = function( item ) {
this.collection.push( item );
g.Group.prototype.remove = function( index ) {
if( index < this.length ) {
this.collection.splice( index, 1 );
g.Group.prototype.empty = function() {
this.collection.length = 0;
g.Group.prototype.each = function( action, asc ) {
for( i = 0; i < this.length; i++ ) {
this.collection[ i ][ action ]( i );
this.collection[ i ][ action ]( i );
(function(){ 'use strict';
g.util.rand = function( min, max ) {
return g.m.random() * ( max - min ) + min;
g.util.randInt = function( min, max ) {
return g.m.floor( g.m.random() * ( max - min + 1) ) + min;
(function(){ 'use strict';
g.addState = function( state ) {
g.states[ state.name ] = state;
g.setState = function( name ) {
g.states[ g.state ].exit();
g.states[ g.state ].init();
g.currentState = function() {
return g.states[ g.state ];
(function(){ 'use strict';
g.Time.prototype.reset = function() {
g.Time.prototype.update = function() {
this.delta = this.now - this.last;
this.ndelta = Math.min( Math.max( this.delta / ( 1000 / 60 ), 0.0001 ), 10 );
this.elapsed += this.delta;
this.nelapsed += this.ndelta;
(function(){ 'use strict';
g.Grid = function( cols, rows ) {
for( var x = 0; x < cols; x++ ) {
for( var y = 0; y < rows; y++ ) {
this.tiles[ x ].push( 'empty' );
g.Grid.prototype.get = function( x, y ) {
return this.tiles[ x ][ y ];
g.Grid.prototype.set = function( x, y, val ) {
this.tiles[ x ][ y ] = val;
(function(){ 'use strict';
g.BoardTile = function( opt ) {
this.parentState = opt.parentState;
this.parentGroup = opt.parentGroup;
this.elem = document.createElement( 'div' );
this.elem.style.position = 'absolute';
this.elem.className = 'tile';
this.parentState.stageElem.appendChild( this.elem );
g.BoardTile.prototype.update = function() {
for( var k in this.classes ) {
if( this.classes[ k ] ) {
if( this.parentState.food.tile.col == this.col || this.parentState.food.tile.row == this.row ) {
if( this.col < this.parentState.food.tile.col ) {
if( this.col > this.parentState.food.tile.col ) {
if( this.row > this.parentState.food.tile.row ) {
if( this.row < this.parentState.food.tile.row ) {
if( this.parentState.food.eaten ) {
g.BoardTile.prototype.updateDimensions = function() {
this.x = this.col * this.parentState.tileWidth;
this.y = this.row * this.parentState.tileHeight;
this.w = this.parentState.tileWidth - this.parentState.spacing;
this.h = this.parentState.tileHeight - this.parentState.spacing;
this.elem.style.left = this.x + 'px';
this.elem.style.top = this.y + 'px';
this.elem.style.width = this.w + 'px';
this.elem.style.height = this.h + 'px';
g.BoardTile.prototype.render = function() {
for( var k in this.classes ) {
if( this.classes[ k ] ) {
this.elem.className = 'tile ' + classString;
(function(){ 'use strict';
g.SnakeTile = function( opt ) {
this.parentState = opt.parentState;
this.parentGroup = opt.parentGroup;
this.borderRadiusAmount = 0;
this.elem = document.createElement( 'div' );
this.elem.style.position = 'absolute';
this.parentState.stageElem.appendChild( this.elem );
g.SnakeTile.prototype.update = function( i ) {
this.x = this.col * this.parentState.tileWidth;
this.y = this.row * this.parentState.tileHeight;
this.blur = this.parentState.dimAvg * 0.03 + Math.sin( this.parentState.time.elapsed / 200 ) * this.parentState.dimAvg * 0.015;
if( this.parentState.snake.dir == 'n' ) {
this.borderRadius = this.borderRadiusAmount + '% ' + this.borderRadiusAmount + '% 0 0';
} else if( this.parentState.snake.dir == 's' ) {
this.borderRadius = '0 0 ' + this.borderRadiusAmount + '% ' + this.borderRadiusAmount + '%';
} else if( this.parentState.snake.dir == 'e' ) {
this.borderRadius = '0 ' + this.borderRadiusAmount + '% ' + this.borderRadiusAmount + '% 0';
} else if( this.parentState.snake.dir == 'w' ) {
this.borderRadius = this.borderRadiusAmount + '% 0 0 ' + this.borderRadiusAmount + '%';
this.alpha = 1 - ( i / this.parentState.snake.tiles.length ) * 0.6;
this.rotation = ( this.parentState.snake.justAteTick / this.parentState.snake.justAteTickMax ) * 90;
this.scale = 1 + ( this.parentState.snake.justAteTick / this.parentState.snake.justAteTickMax ) * 1;
g.SnakeTile.prototype.updateDimensions = function() {
this.w = this.parentState.tileWidth - this.parentState.spacing;
this.h = this.parentState.tileHeight - this.parentState.spacing;
g.SnakeTile.prototype.render = function( i ) {
this.elem.style.left = this.x + 'px';
this.elem.style.top = this.y + 'px';
this.elem.style.width = this.w + 'px';
this.elem.style.height = this.h + 'px';
this.elem.style.backgroundColor = 'rgba(255, 255, 255, ' + this.alpha + ')';
this.elem.style.boxShadow = '0 0 ' + this.blur + 'px #fff';
this.elem.style.borderRadius = this.borderRadius;
(function(){ 'use strict';
g.FoodTile = function( opt ) {
this.parentState = opt.parentState;
this.parentGroup = opt.parentGroup;
this.elem = document.createElement( 'div' );
this.elem.style.position = 'absolute';
this.parentState.stageElem.appendChild( this.elem );
g.FoodTile.prototype.update = function() {
this.x = this.col * this.parentState.tileWidth;
this.y = this.row * this.parentState.tileHeight;
this.blur = this.parentState.dimAvg * 0.03 + Math.sin( this.parentState.time.elapsed / 200 ) * this.parentState.dimAvg * 0.015;
this.scale = 0.8 + Math.sin( this.parentState.time.elapsed / 200 ) * 0.2;
if( this.parentState.food.birthTick || this.parentState.food.deathTick ) {
if( this.parentState.food.birthTick ) {
this.opacity = 1 - ( this.parentState.food.birthTick / 1 ) * 1;
this.opacity = ( this.parentState.food.deathTick / 1 ) * 1;
g.FoodTile.prototype.updateDimensions = function() {
this.w = this.parentState.tileWidth - this.parentState.spacing;
this.h = this.parentState.tileHeight - this.parentState.spacing;
g.FoodTile.prototype.render = function() {
this.elem.style.left = this.x + 'px';
this.elem.style.top = this.y + 'px';
this.elem.style.width = this.w + 'px';
this.elem.style.height = this.h + 'px';
this.elem.style[ 'transform' ] = 'translateZ(0) scale(' + this.scale + ')';
this.elem.style.backgroundColor = 'hsla(' + this.hue + ', 100%, 60%, 1)';
this.elem.style.boxShadow = '0 0 ' + this.blur + 'px hsla(' + this.hue + ', 100%, 60%, 1)';
this.elem.style.opacity = this.opacity;
(function(){ 'use strict';
g.Snake = function( opt ) {
this.parentState = opt.parentState;
for( var i = 0; i < 5; i++ ) {
this.tiles.push( new g.SnakeTile({
parentState: this.parentState,
x: ( 8 - i ) * opt.parentState.tileWidth,
y: 3 * opt.parentState.tileHeight,
w: opt.parentState.tileWidth - opt.parentState.spacing,
h: opt.parentState.tileHeight - opt.parentState.spacing
this.updateTickMax = this.updateTick;
this.updateTickLimit = 3;
this.updateTickChange = 0.2;
this.justAteTickChange = 0.05;
var i = this.tiles.length;
this.parentState.grid.set( this.tiles[ i ].col, this.tiles[ i ].row, 'snake' );
g.Snake.prototype.updateDimensions = function() {
var i = this.tiles.length;
this.tiles[ i ].updateDimensions();
g.Snake.prototype.update = function() {
if( this.parentState.keys.up ) {
if( this.dir != 's' && this.dir != 'n' && this.currDir != 's' && this.currDir != 'n' ) {
} else if( this.parentState.keys.down) {
if( this.dir != 'n' && this.dir != 's' && this.currDir != 'n' && this.currDir != 's' ) {
} else if( this.parentState.keys.right ) {
if( this.dir != 'w' && this.dir != 'e' && this.currDir != 'w' && this.currDir != 'e' ) {
} else if( this.parentState.keys.left ) {
if( this.dir != 'e' && this.dir != 'w' && this.currDir != 'e' && this.currDir != 'w' ) {
this.parentState.keys.up = 0;
this.parentState.keys.down = 0;
this.parentState.keys.right = 0;
this.parentState.keys.left = 0;
this.updateTick += this.parentState.time.ndelta;
if( this.updateTick >= this.updateTickMax ) {
this.updateTick = ( this.updateTick - this.updateTickMax );
this.tiles.unshift( new g.SnakeTile({
parentState: this.parentState,
col: this.tiles[ 0 ].col,
row: this.tiles[ 0 ].row,
x: this.tiles[ 0 ].col * this.parentState.tileWidth,
y: this.tiles[ 0 ].row * this.parentState.tileHeight,
w: this.parentState.tileWidth - this.parentState.spacing,
h: this.parentState.tileHeight - this.parentState.spacing
this.last = this.tiles.pop();
this.parentState.stageElem.removeChild( this.last.elem );
this.parentState.boardTiles.collection[ this.last.col + ( this.last.row * this.parentState.cols ) ].classes.pressed = 2;
var i = this.tiles.length;
this.parentState.grid.set( this.tiles[ i ].col, this.tiles[ i ].row, 'snake' );
this.parentState.grid.set( this.last.col, this.last.row, 'empty' );
this.tiles[ 0 ].row -= 1;
} else if( this.dir == 's' ) {
this.tiles[ 0 ].row += 1;
} else if( this.dir == 'w' ) {
this.tiles[ 0 ].col -= 1;
} else if( this.dir == 'e' ) {
this.tiles[ 0 ].col += 1;
if( this.tiles[ 0 ].col >= this.parentState.cols ) {
if( this.tiles[ 0 ].col < 0 ) {
this.tiles[ 0 ].col = this.parentState.cols - 1;
if( this.tiles[ 0 ].row >= this.parentState.rows ) {
if( this.tiles[ 0 ].row < 0 ) {
this.tiles[ 0 ].row = this.parentState.rows - 1;
if( this.parentState.grid.get( this.tiles[ 0 ].col, this.tiles[ 0 ].row ) == 'snake' ) {
clearTimeout( this.foodCreateTimeout );
if( this.parentState.grid.get( this.tiles[ 0 ].col, this.tiles[ 0 ].row ) == 'food' ) {
this.tiles.push( new g.SnakeTile({
parentState: this.parentState,
x: this.last.col * this.parentState.tileWidth,
y: this.last.row * this.parentState.tileHeight,
w: this.parentState.tileWidth - this.parentState.spacing,
h: this.parentState.tileHeight - this.parentState.spacing
if( this.updateTickMax - this.updateTickChange > this.updateTickLimit ) {
this.updateTickMax -= this.updateTickChange;
this.parentState.score++;
this.parentState.scoreElem.innerHTML = this.parentState.score;
this.justAteTick = this.justAteTickMax;
this.parentState.food.eaten = 1;
this.parentState.stageElem.removeChild( this.parentState.food.tile.elem );
this.foodCreateTimeout = setTimeout( function() {
_this.parentState.food = new g.Food({
parentState: _this.parentState
var i = this.tiles.length;
this.tiles[ i ].update( i );
if( this.justAteTick > 0 ) {
this.justAteTick -= this.justAteTickChange;
} else if( this.justAteTick < 0 ) {
g.Snake.prototype.render = function() {
var i = this.tiles.length;
this.tiles[ i ].render( i );
(function(){ 'use strict';
g.Food = function( opt ) {
this.parentState = opt.parentState;
this.tile = new g.FoodTile({
parentState: this.parentState,
w: opt.parentState.tileWidth - opt.parentState.spacing,
h: opt.parentState.tileHeight - opt.parentState.spacing
this.birthTickChange = 0.025;
this.deathTickChange = 0.05;
g.Food.prototype.reset = function() {
for( var x = 0; x < this.parentState.cols; x++) {
for( var y = 0; y < this.parentState.rows; y++) {
var tile = this.parentState.grid.get( x, y );
empty.push( { x: x, y: y } );
var newTile = empty[ g.util.randInt( 0, empty.length - 1 ) ];
this.tile.col = newTile.x;
this.tile.row = newTile.y;
g.Food.prototype.updateDimensions = function() {
this.tile.updateDimensions();
g.Food.prototype.update = function() {
if( this.birthTick > 0 ) {
this.birthTick -= this.birthTickChange;
} else if( this.birthTick < 0 ) {
this.parentState.grid.set( this.tile.col, this.tile.row, 'food' );
g.Food.prototype.render = function() {
(function(){ 'use strict';
StatePlay.prototype.init = function() {
this.scoreElem = document.querySelector( '.score' );
this.stageElem = document.querySelector( '.stage' );
this.boardTiles = new g.Group();
this.foodCreateTimeout = null;
this.scoreElem.innerHTML = this.score;
this.time = new g.Time();
if( this.winWidth < this.winHeight ) {
this.rows = this.dimLong;
this.cols = this.dimShort;
this.rows = this.dimShort;
this.cols = this.dimLong;
this.grid = new g.Grid( this.cols, this.rows );
this.snake = new g.Snake({
StatePlay.prototype.getDimensions = function() {
this.winWidth = window.innerWidth;
this.winHeight = window.innerHeight;
this.activeWidth = this.winWidth - ( this.winWidth * this.padding );
this.activeHeight = this.winHeight - ( this.winHeight * this.padding );
StatePlay.prototype.resize = function() {
var _this = g.currentState();
_this.stageRatio = _this.rows / _this.cols;
if( _this.activeWidth > _this.activeHeight / _this.stageRatio ) {
_this.stageHeight = _this.activeHeight;
_this.stageElem.style.height = _this.stageHeight + 'px';
_this.stageWidth = Math.floor( _this.stageHeight /_this.stageRatio );
_this.stageElem.style.width = _this.stageWidth + 'px';
_this.stageWidth = _this.activeWidth;
_this.stageElem.style.width = _this.stageWidth + 'px';
_this.stageHeight = Math.floor( _this.stageWidth * _this.stageRatio );
_this.stageElem.style.height = _this.stageHeight + 'px';
_this.tileWidth = ~~( _this.stageWidth / _this.cols );
_this.tileHeight = ~~( _this.stageHeight / _this.rows );
_this.dimAvg = ( _this.activeWidth + _this.activeHeight ) / 2;
_this.spacing = Math.max( 1, ~~( _this.dimAvg * 0.0025 ) );
_this.stageElem.style.marginTop = ( -_this.stageElem.offsetHeight / 2 ) + _this.headerHeight / 2 + 'px';
_this.boardTiles.each( 'updateDimensions' );
_this.snake !== undefined && _this.snake.updateDimensions();
_this.food !== undefined && _this.food.updateDimensions();
StatePlay.prototype.createBoardTiles = function() {
for( var y = 0; y < this.rows; y++ ) {
for( var x = 0; x < this.cols; x++ ) {
this.boardTiles.add( new g.BoardTile({
parentGroup: this.boardTiles,
w: this.tileWidth - this.spacing,
h: this.tileHeight - this.spacing
StatePlay.prototype.upOn = function() { g.currentState().keys.up = 1; }
StatePlay.prototype.downOn = function() { g.currentState().keys.down = 1; }
StatePlay.prototype.rightOn = function() { g.currentState().keys.right = 1; }
StatePlay.prototype.leftOn = function() { g.currentState().keys.left = 1; }
StatePlay.prototype.upOff = function() { g.currentState().keys.up = 0; }
StatePlay.prototype.downOff = function() { g.currentState().keys.down = 0; }
StatePlay.prototype.rightOff = function() { g.currentState().keys.right = 0; }
StatePlay.prototype.leftOff = function() { g.currentState().keys.left = 0; }
StatePlay.prototype.keydown = function( e ) {
var e = ( e.keyCode ? e.keyCode : e.which ),
_this = g.currentState();
if( e === 38 || e === 87 ) { _this.upOn(); }
if( e === 39 || e === 68 ) { _this.rightOn(); }
if( e === 40 || e === 83 ) { _this.downOn(); }
if( e === 37 || e === 65 ) { _this.leftOn(); }
StatePlay.prototype.bindEvents = function() {
var _this = g.currentState();
window.addEventListener( 'keydown', _this.keydown, false );
window.addEventListener( 'resize', _this.resize, false );
StatePlay.prototype.step = function() {
this.boardTiles.each( 'update' );
this.boardTiles.each( 'render' );
StatePlay.prototype.exit = function() {
window.removeEventListener( 'keydown', this.keydown, false );
window.removeEventListener( 'resize', this.resize, false );
this.stageElem.innerHTML = '';
g.addState( new StatePlay() );
(function(){ 'use strict';
debug: window.location.hash == '#debug' ? 1 : 0,
g.setState( g.config.state );
requestAnimationFrame( g.step );
g.states[ g.state ].step();
window.addEventListener( 'load', g.step, false );