<!doctype html> <html lang="zh"> <head> <title>纯HTML和CSS生成的贪食蛇游戏</title> <meta charset="UTF-8" /> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <meta name="viewport" content="width=device-width, initial-scale=1"> <style> /*General*/ * { box-sizing: border-box; font-family: 'microsoft yahei',Arial,sans-serif; } html, body { background-color: #000; height: 100%; } body { background: #222; background: radial-gradient(#333, #111); background-position: center center; background-repeat: no-repeat; background-size: cover; color: #fff; font: 100%/1.5 sans-serif; overflow: hidden; } /*Score*/ .score { color: rgba(255, 255, 255, 0.5); font-size: 16px; font-weight: bold; padding-top: 5px; text-align: center; } /*Stage*/ .stage { bottom: 0; left: 0; margin: auto; position: absolute; right: 0; top: 0; z-index: 2; } /*Tiles*/ .tile { background: rgba(0, 0, 0, 0.15); position: absolute; transition-property: background, box-shadow, opacity, transform ; transform: translateZ(0); transition-duration: 3000ms; } .tile:before { bottom: 0; content: ''; height: 0; left: 0; margin: auto; opacity: 0; position: absolute; right: 0; top: 0; width: 0; transition: opacity 300ms; } .tile.path:before { opacity: 1; } .tile.up:before { border-bottom: 4px inset rgba(255, 255, 255, 0.15); border-left: 4px solid transparent; border-right: 4px solid transparent; } .tile.down:before { border-top: 4px inset rgba(255, 255, 255, 0.15); border-left: 4px solid transparent; border-right: 4px solid transparent; } .tile.left:before { border-right: 4px inset rgba(255, 255, 255, 0.15); border-top: 4px solid transparent; border-bottom: 4px solid transparent; } .tile.right:before { border-left: 4px inset rgba(255, 255, 255, 0.15); border-top: 4px solid transparent; border-bottom: 4px solid transparent; } @media (max-width: 900px), (max-height: 900px) { .tile.up:before, .tile.down:before, .tile.left:before, .tile.right:before { border-width: 3px; } } @media (max-width: 500px), (max-height: 500px) { .tile.up:before, .tile.down:before, .tile.left:before, .tile.right:before { border-width: 2px; } } .tile.pressed { background: rgba(0, 0, 0, 0.3); box-shadow: inset 0 0 10px rgba(0, 0, 0, 0.6); transition-duration: 0ms; } </style> </head> <body> <div>分数:<span class="score">0</span></div> <div class="stage"></div> <script src="prefixfree.min.js"></script> <script src="modernizr.min.js"></script> <script> /*Polyfill*/ (function(){ 'use strict'; /*Request Animation Frame*/ var lastTime = 0; 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( function() { callback( currTime + timeToCall ); }, timeToCall ); lastTime = currTime + timeToCall; return id; } } if( !window.cancelAnimationFrame ) { window.cancelAnimationFrame = function( id ) { clearTimeout( id ); } } })(); /*DOM Manipulation*/ (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, '' ); } else { elem.className += ' ' + className; } }; })(); /*Core*/ g = {}; (function(){ 'use strict'; /*Math*/ g.m = Math; 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.m.TWO_PI = g.m.PI * 2; /*Miscellaneous*/ g.isset = function( prop ) { return typeof prop != 'undefined'; }; g.log = function() { if( g.isset( g.config ) && g.config.debug && window.console ){ console.log( Array.prototype.slice.call( arguments ) ); } }; })(); /*Group*/ (function(){ 'use strict'; g.Group = function() { this.collection = []; this.length = 0; }; g.Group.prototype.add = function( item ) { this.collection.push( item ); this.length++; }; g.Group.prototype.remove = function( index ) { if( index < this.length ) { this.collection.splice( index, 1 ); this.length--; } }; g.Group.prototype.empty = function() { this.collection.length = 0; this.length = 0; }; g.Group.prototype.each = function( action, asc ) { var asc = asc || 0, i; if( asc ) { for( i = 0; i < this.length; i++ ) { this.collection[ i ][ action ]( i ); } } else { i = this.length; while( i-- ) { this.collection[ i ][ action ]( i ); } } }; })(); /*Utilities*/ (function(){ 'use strict'; g.util = {}; /*Random*/ 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; }; }()); /*State*/ (function(){ 'use strict'; g.states = {}; g.addState = function( state ) { g.states[ state.name ] = state; }; g.setState = function( name ) { if( g.state ) { g.states[ g.state ].exit(); } g.state = name; g.states[ g.state ].init(); }; g.currentState = function() { return g.states[ g.state ]; }; }()); /*Time*/ (function(){ 'use strict'; g.Time = function() { this.reset(); } g.Time.prototype.reset = function() { this.now = Date.now(); this.last = Date.now(); this.delta = 60; this.ndelta = 1; this.elapsed = 0; this.nelapsed = 0; this.tick = 0; }; g.Time.prototype.update = function() { this.now = Date.now(); 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; this.last = this.now; this.tick++; }; })(); /*Grid Entity*/ (function(){ 'use strict'; g.Grid = function( cols, rows ) { this.cols = cols; this.rows = rows; this.tiles = []; for( var x = 0; x < cols; x++ ) { this.tiles[ 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; }; })(); /*Board Tile Entity*/ (function(){ 'use strict'; g.BoardTile = function( opt ) { this.parentState = opt.parentState; this.parentGroup = opt.parentGroup; this.col = opt.col; this.row = opt.row; this.x = opt.x; this.y = opt.y; this.z = 0; this.w = opt.w; this.h = opt.h; this.elem = document.createElement( 'div' ); this.elem.style.position = 'absolute'; this.elem.className = 'tile'; this.parentState.stageElem.appendChild( this.elem ); this.classes = { pressed: 0, path: 0, up: 0, down: 0, left: 0, right: 0 } this.updateDimensions(); }; g.BoardTile.prototype.update = function() { for( var k in this.classes ) { if( this.classes[ k ] ) { this.classes[ k ]--; } } if( this.parentState.food.tile.col == this.col || this.parentState.food.tile.row == this.row ) { this.classes.path = 1; if( this.col < this.parentState.food.tile.col ) { this.classes.right = 1; } else { this.classes.right = 0; } if( this.col > this.parentState.food.tile.col ) { this.classes.left = 1; } else { this.classes.left = 0; } if( this.row > this.parentState.food.tile.row ) { this.classes.up = 1; } else { this.classes.up = 0; } if( this.row < this.parentState.food.tile.row ) { this.classes.down = 1; } else { this.classes.down = 0; } } else { this.classes.path = 0; } if( this.parentState.food.eaten ) { this.classes.path = 0; } }; 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() { var classString = ''; for( var k in this.classes ) { if( this.classes[ k ] ) { classString += k + ' '; } } this.elem.className = 'tile ' + classString; }; })(); /*Snake Tile Entity*/ (function(){ 'use strict'; g.SnakeTile = function( opt ) { this.parentState = opt.parentState; this.parentGroup = opt.parentGroup; this.col = opt.col; this.row = opt.row; this.x = opt.x; this.y = opt.y; this.w = opt.w; this.h = opt.h; this.color = null; this.scale = 1; this.rotation = 0; this.blur = 0; this.alpha = 1; this.borderRadius = 0; 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; if( i == 0 ) { this.color = '#fff'; 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 + '%'; } } else { this.color = '#fff'; this.blur = 0; this.borderRadius = '0'; } 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; }; })(); /*Food Tile Entity*/ (function(){ 'use strict'; g.FoodTile = function( opt ) { this.parentState = opt.parentState; this.parentGroup = opt.parentGroup; this.col = opt.col; this.row = opt.row; this.x = opt.x; this.y = opt.y; this.w = opt.w; this.h = opt.h; this.blur = 0; this.scale = 1; this.hue = 100; this.opacity = 0; 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; } else { this.opacity = ( this.parentState.food.deathTick / 1 ) * 1; } } else { this.opacity = 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[ Modernizr.prefixed( '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; }; })(); /*Snake Entity*/ (function(){ 'use strict'; g.Snake = function( opt ) { this.parentState = opt.parentState; this.dir = 'e', this.currDir = this.dir; this.tiles = []; for( var i = 0; i < 5; i++ ) { this.tiles.push( new g.SnakeTile({ parentState: this.parentState, parentGroup: this.tiles, col: 8 - i, row: 3, 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.last = 0; this.updateTick = 10; this.updateTickMax = this.updateTick; this.updateTickLimit = 3; this.updateTickChange = 0.2; this.deathFlag = 0; this.justAteTick = 0; this.justAteTickMax = 1; this.justAteTickChange = 0.05; // sync data grid of the play state var i = this.tiles.length; while( i-- ) { this.parentState.grid.set( this.tiles[ i ].col, this.tiles[ i ].row, 'snake' ); } }; g.Snake.prototype.updateDimensions = function() { var i = this.tiles.length; while( i-- ) { 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' ) { this.dir = 'n'; } } else if( this.parentState.keys.down) { if( this.dir != 'n' && this.dir != 's' && this.currDir != 'n' && this.currDir != 's' ) { this.dir = 's'; } } else if( this.parentState.keys.right ) { if( this.dir != 'w' && this.dir != 'e' && this.currDir != 'w' && this.currDir != 'e' ) { this.dir = 'e'; } } else if( this.parentState.keys.left ) { if( this.dir != 'e' && this.dir != 'w' && this.currDir != 'e' && this.currDir != 'w' ) { this.dir = '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 ) { // reset the update timer to 0, or whatever leftover there is this.updateTick = ( this.updateTick - this.updateTickMax ); // rotate snake block array this.tiles.unshift( new g.SnakeTile({ parentState: this.parentState, parentGroup: this.tiles, 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; // sync data grid of the play state var i = this.tiles.length; while( i-- ) { this.parentState.grid.set( this.tiles[ i ].col, this.tiles[ i ].row, 'snake' ); } this.parentState.grid.set( this.last.col, this.last.row, 'empty' ); // move the snake's head if ( this.dir == 'n' ) { this.currDir = 'n'; this.tiles[ 0 ].row -= 1; } else if( this.dir == 's' ) { this.currDir = 's'; this.tiles[ 0 ].row += 1; } else if( this.dir == 'w' ) { this.currDir = 'w'; this.tiles[ 0 ].col -= 1; } else if( this.dir == 'e' ) { this.currDir = 'e'; this.tiles[ 0 ].col += 1; } // wrap walls this.wallFlag = false; if( this.tiles[ 0 ].col >= this.parentState.cols ) { this.tiles[ 0 ].col = 0; this.wallFlag = true; } if( this.tiles[ 0 ].col < 0 ) { this.tiles[ 0 ].col = this.parentState.cols - 1; this.wallFlag = true; } if( this.tiles[ 0 ].row >= this.parentState.rows ) { this.tiles[ 0 ].row = 0; this.wallFlag = true; } if( this.tiles[ 0 ].row < 0 ) { this.tiles[ 0 ].row = this.parentState.rows - 1; this.wallFlag = true; } // check death by eating self if( this.parentState.grid.get( this.tiles[ 0 ].col, this.tiles[ 0 ].row ) == 'snake' ) { this.deathFlag = 1; clearTimeout( this.foodCreateTimeout ); } // check eating of food if( this.parentState.grid.get( this.tiles[ 0 ].col, this.tiles[ 0 ].row ) == 'food' ) { this.tiles.push( new g.SnakeTile({ parentState: this.parentState, parentGroup: this.tiles, col: this.last.col, row: this.last.row, 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 ); var _this = this; this.foodCreateTimeout = setTimeout( function() { _this.parentState.food = new g.Food({ parentState: _this.parentState }); }, 300); } // check death by eating self if( this.deathFlag ) { g.setState( 'play' ); } } // update individual snake tiles var i = this.tiles.length; while( i-- ) { this.tiles[ i ].update( i ); } if( this.justAteTick > 0 ) { this.justAteTick -= this.justAteTickChange; } else if( this.justAteTick < 0 ) { this.justAteTick = 0; } }; g.Snake.prototype.render = function() { // render individual snake tiles var i = this.tiles.length; while( i-- ) { this.tiles[ i ].render( i ); } }; })(); /*Food Entity*/ (function(){ 'use strict'; g.Food = function( opt ) { this.parentState = opt.parentState; this.tile = new g.FoodTile({ parentState: this.parentState, col: 0, row: 0, x: 0, y: 0, w: opt.parentState.tileWidth - opt.parentState.spacing, h: opt.parentState.tileHeight - opt.parentState.spacing }); this.reset(); this.eaten = 0; this.birthTick = 1; this.deathTick = 0; this.birthTickChange = 0.025; this.deathTickChange = 0.05; }; g.Food.prototype.reset = function() { var empty = []; 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 ); if( tile == 'empty' ) { 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() { // update food tile this.tile.update(); if( this.birthTick > 0 ) { this.birthTick -= this.birthTickChange; } else if( this.birthTick < 0 ) { this.birthTick = 0; } // sync data grid of the play state this.parentState.grid.set( this.tile.col, this.tile.row, 'food' ); }; g.Food.prototype.render = function() { this.tile.render(); }; })(); /*================================================ Play State ================================================*/ (function(){ 'use strict'; function StatePlay() { this.name = 'play'; } StatePlay.prototype.init = function() { this.scoreElem = document.querySelector( '.score' ); this.stageElem = document.querySelector( '.stage' ); this.dimLong = 28; this.dimShort = 16; this.padding = 0.25; this.boardTiles = new g.Group(); this.keys = {}; this.foodCreateTimeout = null; this.score = 0; this.scoreElem.innerHTML = this.score; this.time = new g.Time(); this.getDimensions(); if( this.winWidth < this.winHeight ) { this.rows = this.dimLong; this.cols = this.dimShort; } else { this.rows = this.dimShort; this.cols = this.dimLong; } this.spacing = 1; this.grid = new g.Grid( this.cols, this.rows ); this.resize(); this.createBoardTiles(); this.bindEvents(); this.snake = new g.Snake({ parentState: this }); this.food = new g.Food({ parentState: this }); }; 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.getDimensions(); _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'; } else { _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({ parentState: this, parentGroup: this.boardTiles, col: x, row: y, x: x * this.tileWidth, y: y * this.tileHeight, 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 ) { e.preventDefault(); 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' ); this.snake.update(); this.snake.render(); this.food.update(); this.food.render(); this.time.update(); }; StatePlay.prototype.exit = function() { window.removeEventListener( 'keydown', this.keydown, false ); window.removeEventListener( 'resize', this.resize, false ); this.stageElem.innerHTML = ''; this.grid.tiles = null; this.time = null; }; g.addState( new StatePlay() ); })(); /*================================================ Game ================================================*/ (function(){ 'use strict'; g.config = { title: 'Snakely', debug: window.location.hash == '#debug' ? 1 : 0, state: 'play' }; g.setState( g.config.state ); g.time = new g.Time(); g.step = function() { requestAnimationFrame( g.step ); g.states[ g.state ].step(); g.time.update(); }; window.addEventListener( 'load', g.step, false ); })(); </script> </body> </html>