diff --git a/src/blockrain.jquery.libs.js b/src/blockrain.jquery.libs.js deleted file mode 100644 index 21eb819..0000000 --- a/src/blockrain.jquery.libs.js +++ /dev/null @@ -1,2 +0,0 @@ -// jQuery Widget -(function(e){"function"==typeof define&&define.amd?define(["jquery"],e):e(jQuery)})(function(e){var t=0,i=Array.prototype.slice;e.cleanData=function(t){return function(i){var s,n,a;for(a=0;null!=(n=i[a]);a++)try{s=e._data(n,"events"),s&&s.remove&&e(n).triggerHandler("remove")}catch(o){}t(i)}}(e.cleanData),e.widget=function(t,i,s){var n,a,o,r,h={},l=t.split(".")[0];return t=t.split(".")[1],n=l+"-"+t,s||(s=i,i=e.Widget),e.expr[":"][n.toLowerCase()]=function(t){return!!e.data(t,n)},e[l]=e[l]||{},a=e[l][t],o=e[l][t]=function(e,t){return this._createWidget?(arguments.length&&this._createWidget(e,t),void 0):new o(e,t)},e.extend(o,a,{version:s.version,_proto:e.extend({},s),_childConstructors:[]}),r=new i,r.options=e.widget.extend({},r.options),e.each(s,function(t,s){return e.isFunction(s)?(h[t]=function(){var e=function(){return i.prototype[t].apply(this,arguments)},n=function(e){return i.prototype[t].apply(this,e)};return function(){var t,i=this._super,a=this._superApply;return this._super=e,this._superApply=n,t=s.apply(this,arguments),this._super=i,this._superApply=a,t}}(),void 0):(h[t]=s,void 0)}),o.prototype=e.widget.extend(r,{widgetEventPrefix:a?r.widgetEventPrefix||t:t},h,{constructor:o,namespace:l,widgetName:t,widgetFullName:n}),a?(e.each(a._childConstructors,function(t,i){var s=i.prototype;e.widget(s.namespace+"."+s.widgetName,o,i._proto)}),delete a._childConstructors):i._childConstructors.push(o),e.widget.bridge(t,o),o},e.widget.extend=function(t){for(var s,n,a=i.call(arguments,1),o=0,r=a.length;r>o;o++)for(s in a[o])n=a[o][s],a[o].hasOwnProperty(s)&&void 0!==n&&(t[s]=e.isPlainObject(n)?e.isPlainObject(t[s])?e.widget.extend({},t[s],n):e.widget.extend({},n):n);return t},e.widget.bridge=function(t,s){var n=s.prototype.widgetFullName||t;e.fn[t]=function(a){var o="string"==typeof a,r=i.call(arguments,1),h=this;return a=!o&&r.length?e.widget.extend.apply(null,[a].concat(r)):a,o?this.each(function(){var i,s=e.data(this,n);return"instance"===a?(h=s,!1):s?e.isFunction(s[a])&&"_"!==a.charAt(0)?(i=s[a].apply(s,r),i!==s&&void 0!==i?(h=i&&i.jquery?h.pushStack(i.get()):i,!1):void 0):e.error("no such method '"+a+"' for "+t+" widget instance"):e.error("cannot call methods on "+t+" prior to initialization; "+"attempted to call method '"+a+"'")}):this.each(function(){var t=e.data(this,n);t?(t.option(a||{}),t._init&&t._init()):e.data(this,n,new s(a,this))}),h}},e.Widget=function(){},e.Widget._childConstructors=[],e.Widget.prototype={widgetName:"widget",widgetEventPrefix:"",defaultElement:"
",options:{disabled:!1,create:null},_createWidget:function(i,s){s=e(s||this.defaultElement||this)[0],this.element=e(s),this.uuid=t++,this.eventNamespace="."+this.widgetName+this.uuid,this.bindings=e(),this.hoverable=e(),this.focusable=e(),s!==this&&(e.data(s,this.widgetFullName,this),this._on(!0,this.element,{remove:function(e){e.target===s&&this.destroy()}}),this.document=e(s.style?s.ownerDocument:s.document||s),this.window=e(this.document[0].defaultView||this.document[0].parentWindow)),this.options=e.widget.extend({},this.options,this._getCreateOptions(),i),this._create(),this._trigger("create",null,this._getCreateEventData()),this._init()},_getCreateOptions:e.noop,_getCreateEventData:e.noop,_create:e.noop,_init:e.noop,destroy:function(){this._destroy(),this.element.unbind(this.eventNamespace).removeData(this.widgetFullName).removeData(e.camelCase(this.widgetFullName)),this.widget().unbind(this.eventNamespace).removeAttr("aria-disabled").removeClass(this.widgetFullName+"-disabled "+"ui-state-disabled"),this.bindings.unbind(this.eventNamespace),this.hoverable.removeClass("ui-state-hover"),this.focusable.removeClass("ui-state-focus")},_destroy:e.noop,widget:function(){return this.element},option:function(t,i){var s,n,a,o=t;if(0===arguments.length)return e.widget.extend({},this.options);if("string"==typeof t)if(o={},s=t.split("."),t=s.shift(),s.length){for(n=o[t]=e.widget.extend({},this.options[t]),a=0;s.length-1>a;a++)n[s[a]]=n[s[a]]||{},n=n[s[a]];if(t=s.pop(),1===arguments.length)return void 0===n[t]?null:n[t];n[t]=i}else{if(1===arguments.length)return void 0===this.options[t]?null:this.options[t];o[t]=i}return this._setOptions(o),this},_setOptions:function(e){var t;for(t in e)this._setOption(t,e[t]);return this},_setOption:function(e,t){return this.options[e]=t,"disabled"===e&&(this.widget().toggleClass(this.widgetFullName+"-disabled",!!t),t&&(this.hoverable.removeClass("ui-state-hover"),this.focusable.removeClass("ui-state-focus"))),this},enable:function(){return this._setOptions({disabled:!1})},disable:function(){return this._setOptions({disabled:!0})},_on:function(t,i,s){var n,a=this;"boolean"!=typeof t&&(s=i,i=t,t=!1),s?(i=n=e(i),this.bindings=this.bindings.add(i)):(s=i,i=this.element,n=this.widget()),e.each(s,function(s,o){function r(){return t||a.options.disabled!==!0&&!e(this).hasClass("ui-state-disabled")?("string"==typeof o?a[o]:o).apply(a,arguments):void 0}"string"!=typeof o&&(r.guid=o.guid=o.guid||r.guid||e.guid++);var h=s.match(/^([\w:-]*)\s*(.*)$/),l=h[1]+a.eventNamespace,u=h[2];u?n.delegate(u,l,r):i.bind(l,r)})},_off:function(t,i){i=(i||"").split(" ").join(this.eventNamespace+" ")+this.eventNamespace,t.unbind(i).undelegate(i),this.bindings=e(this.bindings.not(t).get()),this.focusable=e(this.focusable.not(t).get()),this.hoverable=e(this.hoverable.not(t).get())},_delay:function(e,t){function i(){return("string"==typeof e?s[e]:e).apply(s,arguments)}var s=this;return setTimeout(i,t||0)},_hoverable:function(t){this.hoverable=this.hoverable.add(t),this._on(t,{mouseenter:function(t){e(t.currentTarget).addClass("ui-state-hover")},mouseleave:function(t){e(t.currentTarget).removeClass("ui-state-hover")}})},_focusable:function(t){this.focusable=this.focusable.add(t),this._on(t,{focusin:function(t){e(t.currentTarget).addClass("ui-state-focus")},focusout:function(t){e(t.currentTarget).removeClass("ui-state-focus")}})},_trigger:function(t,i,s){var n,a,o=this.options[t];if(s=s||{},i=e.Event(i),i.type=(t===this.widgetEventPrefix?t:this.widgetEventPrefix+t).toLowerCase(),i.target=this.element[0],a=i.originalEvent)for(n in a)n in i||(i[n]=a[n]);return this.element.trigger(i,s),!(e.isFunction(o)&&o.apply(this.element[0],[i].concat(s))===!1||i.isDefaultPrevented())}},e.each({show:"fadeIn",hide:"fadeOut"},function(t,i){e.Widget.prototype["_"+t]=function(s,n,a){"string"==typeof n&&(n={effect:n});var o,r=n?n===!0||"number"==typeof n?i:n.effect||i:t;n=n||{},"number"==typeof n&&(n={duration:n}),o=!e.isEmptyObject(n),n.complete=a,n.delay&&s.delay(n.delay),o&&e.effects&&e.effects.effect[r]?s[t](n):r!==t&&s[r]?s[r](n.duration,n.easing,a):s.queue(function(i){e(this)[t](),a&&a.call(s[0]),i()})}}),e.widget}); \ No newline at end of file diff --git a/src/blockrain.jquery.src.js b/src/blockrain.jquery.src.js deleted file mode 100644 index 7fa74e0..0000000 --- a/src/blockrain.jquery.src.js +++ /dev/null @@ -1,1578 +0,0 @@ -((function ( $ ) { - - "use strict"; - - $.widget('aerolab.blockrain', { - - options: { - autoplay: false, // Let a bot play the game - autoplayRestart: true, // Restart the game automatically once a bot loses - showFieldOnStart: true, // Show a bunch of random blocks on the start screen (it looks nice) - theme: null, // The theme name or a theme object - blockWidth: 10, // How many blocks wide the field is (The standard is 10 blocks) - autoBlockWidth: false, // The blockWidth is dinamically calculated based on the autoBlockSize. Disabled blockWidth. Useful for responsive backgrounds - autoBlockSize: 24, // The max size of a block for autowidth mode - difficulty: 'normal', // Difficulty (normal|nice|evil). - speed: 20, // The speed of the game. The higher, the faster the pieces go. - asdwKeys: true, // Enable ASDW keys - - // Copy - playText: 'Let\'s play some Tetris', - playButtonText: 'Play', - gameOverText: 'Game Over', - restartButtonText: 'Play Again', - scoreText: 'Score', - - // Basic Callbacks - onStart: function(){}, - onRestart: function(){}, - onGameOver: function(score){}, - - // When a block is placed - onPlaced: function(){}, - // When a line is made. Returns the number of lines, score assigned and total score - onLine: function(lines, scoreIncrement, score){} - }, - - - /** - * Start/Restart Game - */ - start: function() { - this._doStart(); - this.options.onStart.call(this.element); - }, - - restart: function() { - this._doStart(); - this.options.onRestart.call(this.element); - }, - - gameover: function() { - this.showGameOverMessage(); - this._board.gameover = true; - this.options.onGameOver.call(this.element, this._filled.score); - }, - - _doStart: function() { - this._filled.clearAll(); - this._filled._resetScore(); - this._board.cur = this._board.nextShape(); - this._board.started = true; - this._board.gameover = false; - this._board.dropDelay = 5; - this._board.render(true); - this._board.animate(); - - this._$start.fadeOut(150); - this._$gameover.fadeOut(150); - this._$score.fadeIn(150); - }, - - - pause: function() { - this._board.paused = true; - }, - - resume: function() { - this._board.paused = false; - }, - - autoplay: function(enable) { - if( typeof enable !== 'boolean' ){ enable = true; } - - // On autoplay, start the game right away - this.options.autoplay = enable; - if( enable && ! this._board.started ) { - this._doStart(); - } - this._setupControls( ! enable ); - this._setupTouchControls( ! enable ); - }, - - controls: function(enable) { - if( typeof enable !== 'boolean' ){ enable = true; } - this._setupControls(enable); - }, - - touchControls: function(enable) { - if( typeof enable !== 'boolean' ){ enable = true; } - this._setupTouchControls(enable); - }, - - score: function(newScore) { - if( typeof newScore !== 'undefined' && parseInt(newScore) >= 0 ) { - this._filled.score = parseInt(newScore); - this._$scoreText.text(this._filled_score); - } - return this._filled.score; - }, - - freesquares: function() { - return this._filled.getFreeSpaces(); - }, - - showStartMessage: function() { - this._$start.show(); - }, - - showGameOverMessage: function() { - this._$gameover.show(); - }, - - /** - * Update the sizes of the renderer (this makes the game responsive) - */ - updateSizes: function() { - - this._PIXEL_WIDTH = this.element.innerWidth(); - this._PIXEL_HEIGHT = this.element.innerHeight(); - - this._BLOCK_WIDTH = this.options.blockWidth; - this._BLOCK_HEIGHT = Math.floor(this.element.innerHeight() / this.element.innerWidth() * this._BLOCK_WIDTH); - - this._block_size = Math.floor(this._PIXEL_WIDTH / this._BLOCK_WIDTH); - this._border_width = 2; - - // Recalculate the pixel width and height so the canvas always has the best possible size - this._PIXEL_WIDTH = this._block_size * this._BLOCK_WIDTH; - this._PIXEL_HEIGHT = this._block_size * this._BLOCK_HEIGHT; - - this._$canvas .attr('width', this._PIXEL_WIDTH) - .attr('height', this._PIXEL_HEIGHT); - }, - - - theme: function(newTheme){ - - if( typeof newTheme === 'undefined' ) { - return this.options.theme || this._theme; - } - - // Setup the theme properly - if( typeof newTheme === 'string' ) { - this.options.theme = newTheme; - this._theme = $.extend(true, {}, BlockrainThemes[newTheme]); - } - else { - this.options.theme = null; - this._theme = newTheme; - } - - if( typeof this._theme === 'undefined' || this._theme === null ) { - this._theme = $.extend(true, {}, BlockrainThemes['retro']); - this.options.theme = 'retro'; - } - - if( isNaN(parseInt(this._theme.strokeWidth)) || typeof parseInt(this._theme.strokeWidth) !== 'number' ) { - this._theme.strokeWidth = 2; - } - - // Load the image assets - this._preloadThemeAssets(); - - if( this._board !== null ) { - if( typeof this._theme.background === 'string' ) { - this._$canvas.css('background-color', this._theme.background); - } - this._board.render(); - } - }, - - - // Theme - _theme: { - - }, - - - // UI Elements - _$game: null, - _$canvas: null, - _$gameholder: null, - _$start: null, - _$gameover: null, - _$score: null, - _$scoreText: null, - - - // Canvas - _canvas: null, - _ctx: null, - - - // Initialization - _create: function() { - - var game = this; - - this.theme(this.options.theme); - - this._createHolder(); - this._createUI(); - - this._refreshBlockSizes(); - - this.updateSizes(); - - $(window).resize(function(){ - //game.updateSizes(); - }); - - this._SetupShapeFactory(); - this._SetupFilled(); - this._SetupInfo(); - this._SetupBoard(); - - this._info.init(); - this._board.init(); - - var renderLoop = function(){ - requestAnimationFrame(renderLoop); - game._board.render(); - }; - renderLoop(); - - if( this.options.autoplay ) { - this.autoplay(true); - this._setupTouchControls(false); - } else { - this._setupControls(true); - this._setupTouchControls(false); - } - - }, - - _checkCollisions: function(x, y, blocks, checkDownOnly) { - // x & y should be aspirational values - var i = 0, len = blocks.length, a, b; - for (; i= this._BLOCK_HEIGHT || this._filled.check(a, b)) { - return true; - } else if (!checkDownOnly && a < 0 || a >= this._BLOCK_WIDTH) { - return true; - } - } - return false; - }, - - - _board: null, - _info: null, - _filled: null, - - - /** - * Draws the background - */ - _drawBackground: function() { - - if( typeof this._theme.background !== 'string' ) { - return; - } - - if( this._theme.backgroundGrid instanceof Image ) { - - // Not loaded - if( this._theme.backgroundGrid.width === 0 || this._theme.backgroundGrid.height === 0 ){ return; } - - this._ctx.globalAlpha = 1.0; - - for( var x=0; x maxx) { maxx = blocks[i]; } - if (blocks[i+1] < miny) { miny = blocks[i+1]; } - if (blocks[i+1] > maxy) { maxy = blocks[i+1]; } - } - return { - left: minx, - right: maxx, - top: miny, - bottom: maxy, - width: maxx - minx, - height: maxy - miny - }; - } - }); - - return this.init(); - }; - - this._shapeFactory = { - line: function() { - return new Shape(game, game._shapes.line, false, 'line'); - }, - square: function() { - return new Shape(game, game._shapes.square, false, 'square'); - }, - arrow: function() { - return new Shape(game, game._shapes.arrow, false, 'arrow'); - }, - leftHook: function() { - return new Shape(game, game._shapes.leftHook, false, 'leftHook'); - }, - rightHook: function() { - return new Shape(game, game._shapes.rightHook, false, 'rightHook'); - }, - leftZag: function() { - return new Shape(game, game._shapes.leftZag, false, 'leftZag'); - }, - rightZag: function() { - return new Shape(game, game._shapes.rightZag, false, 'rightZag'); - } - }; - }, - - - _SetupFilled: function() { - var game = this; - if( this._filled !== null ){ return; } - - this._filled = { - data: new Array(game._BLOCK_WIDTH * game._BLOCK_HEIGHT), - score: 0, - toClear: {}, - check: function(x, y) { - return this.data[this.asIndex(x, y)]; - }, - add: function(x, y, blockType, blockVariation, blockIndex, blockOrientation) { - if (x >= 0 && x < game._BLOCK_WIDTH && y >= 0 && y < game._BLOCK_HEIGHT) { - this.data[this.asIndex(x, y)] = { - blockType: blockType, - blockVariation: blockVariation, - blockIndex: blockIndex, - blockOrientation: blockOrientation - }; - } - }, - getFreeSpaces: function() { - var count = 0; - for( var i=0; i=0; i--) { - this.data[i] = (i >= game._BLOCK_WIDTH ? this.data[i-game._BLOCK_WIDTH] : undefined); - } - }, - checkForClears: function() { - var startLines = game._board.lines; - var rows = [], i, len, count, mod; - - for (i=0, len=this.data.length; i 1 ) { - game._board.dropDelay *= 0.9; - } - } - - var clearedLines = game._board.lines - startLines; - this._updateScore(clearedLines); - }, - _updateScore: function(numLines) { - if( numLines <= 0 ) { return; } - var scores = [0,400,1000,3000,12000]; - if( numLines >= scores.length ){ numLines = scores.length-1 } - - this.score += scores[numLines]; - game._$scoreText.text(this.score); - - game.options.onLine.call(game.element, numLines, scores[numLines], this.score); - }, - _resetScore: function() { - this.score = 0; - game._$scoreText.text(this.score); - }, - draw: function() { - for (var i=0, len=this.data.length, row, color; i= this.dropDelay) || - (game.options.autoplay) || - (this.holding.drop && (now - this.holding.drop) >= this.holdingThreshold) ) { - drop = true; - moved = true; - this.dropCount = 0; - } - - // Move Left by holding - if( this.holding.left && (now - this.holding.left) >= this.holdingThreshold ) { - moved = true; - this.cur.moveLeft(); - } - - // Move Right by holding - if( this.holding.right && (now - this.holding.right) >= this.holdingThreshold ) { - moved = true; - this.cur.moveRight(); - } - - // Test for a collision, add the piece to the filled blocks and fetch the next one - if (drop) { - var cur = this.cur, x = cur.x, y = cur.y, blocks = cur.getBlocks(); - if (game._checkCollisions(x, y+1, blocks, true)) { - drop = false; - var blockIndex = 0; - for (var i=0; i 0) { - return blockTheme[0]; - } else { - return null; - } - } else { - return blockTheme; - } - } - - if( typeof falling !== 'boolean' ){ falling = true; } - if( falling ) { - if( typeof game._theme.primary === 'string' && game._theme.primary !== '' ) { - return game._theme.primary; - } else if( typeof game._theme.blocks !== 'undefined' && game._theme.blocks !== null ) { - return getBlockVariation(game._theme.blocks[blockType], blockVariation); - } else { - return getBlockVariation(game._theme.complexBlocks[blockType], blockVariation); - } - } else { - if( typeof game._theme.secondary === 'string' && game._theme.secondary !== '' ) { - return game._theme.secondary; - } else if( typeof game._theme.blocks !== 'undefined' && game._theme.blocks !== null ) { - return getBlockVariation(game._theme.blocks[blockType], blockVariation); - } else { - return getBlockVariation(game._theme.complexBlocks[blockType], blockVariation); - } - } - } - - }; - - game._niceShapes = game._getNiceShapes(); - }, - - // Utility Functions - _randInt: function(a, b) { return a + Math.floor(Math.random() * (1 + b - a)); }, - _randSign: function() { return this._randInt(0, 1) * 2 - 1; }, - _randChoice: function(choices) { return choices[this._randInt(0, choices.length-1)]; }, - - - /** - * Find base64 encoded images and load them as image objects, which can be used by the canvas renderer - */ - _preloadThemeAssets: function() { - - var game = this; - - var hexColorcheck = new RegExp('^#[A-F0-9+]{3,6}', 'i'); - var base64check = new RegExp('^data:image/(png|gif|jpg);base64,', 'i'); - - var handleAssetLoad = function() { - // Rerender the board as soon as an asset loads - if( game._board ) { - game._board.render(true); - } - }; - - var loadAsset = function(src) { - var plainSrc = src; - if( ! hexColorcheck.test( plainSrc ) ) { - // It's an image - src = new Image(); - src.src = plainSrc; - src.onload = handleAssetLoad; - } else { - // It's a color - src = plainSrc; - } - return src; - }; - - var startAssetLoad = function(block) { - // Assets can be an array of variation so they can change color/design randomly - if( $.isArray(block) && block.length > 0 ) { - for( var i=0; i
'); - this._$gameholder.css('position', 'relative').css('width', '100%').css('height', '100%'); - - this.element.html('').append(this._$gameholder); - - // Create the game canvas and context - this._$canvas = $(''); - if( typeof this._theme.background === 'string' ) { - this._$canvas.css('background-color', this._theme.background); - } - this._$gameholder.append(this._$canvas); - - this._canvas = this._$canvas.get(0); - this._ctx = this._canvas.getContext('2d'); - - }, - - - _createUI: function() { - - var game = this; - - // Score - game._$score = $( - '
'+ - '
'+ - '
'+ this.options.scoreText +'
'+ - '
0
'+ - '
'+ - '
').hide(); - game._$scoreText = game._$score.find('.blockrain-score-num'); - game._$gameholder.append(game._$score); - - // Create the start menu - game._$start = $( - '
'+ - '
'+ - '
'+ this.options.playText +'
'+ - ''+ this.options.playButtonText +''+ - '
'+ - '
').hide(); - game._$gameholder.append(game._$start); - - game._$start.find('.blockrain-start-btn').click(function(event){ - event.preventDefault(); - game.start(); - }); - - // Create the game over menu - game._$gameover = $( - '
'+ - '
'+ - '
'+ this.options.gameOverText +'
'+ - ''+ this.options.restartButtonText +''+ - '
'+ - '
').hide(); - game._$gameover.find('.blockrain-game-over-btn').click(function(event){ - event.preventDefault(); - game.restart(); - }); - game._$gameholder.append(game._$gameover); - - this._createControls(); - }, - - - _createControls: function() { - - var game = this; - - game._$touchLeft = $('').appendTo(game._$gameholder); - game._$touchRight = $('').appendTo(game._$gameholder); - game._$touchRotateRight = $('').appendTo(game._$gameholder); - game._$touchRotateLeft = $('').appendTo(game._$gameholder); - game._$touchDrop = $('').appendTo(game._$gameholder); - - }, - - - _refreshBlockSizes: function() { - - if( this.options.autoBlockWidth ) { - this.options.blockWidth = Math.ceil( this.element.width() / this.options.autoBlockSize ); - } - - }, - - - _getNiceShapes: function() { - /* - * Things I need for this to work... - * - ability to test each shape with this._filled data - * - maybe give empty spots scores? and try to maximize the score? - */ - - var game = this; - - var shapes = {}, - attr; - - for( var attr in this._shapeFactory ) { - shapes[attr] = this._shapeFactory[attr](); - } - - function scoreBlocks(possibles, blocks, x, y, filled, width, height) { - var i, len=blocks.length, score=0, bottoms = {}, tx, ty, overlaps; - - // base score - for (i=0; i best_score_for_shape) { - best_score_for_shape = score; - best_orientation_for_shape = i; - best_x_for_shape = x; - } - break; - } - } - } - } - - if ((evil && best_score_for_shape < best_score) || - (!evil && best_score_for_shape > best_score)) { - best_shape = shape; - best_score = best_score_for_shape; - best_orientation = best_orientation_for_shape; - best_x = best_x_for_shape; - } - } - - best_shape.best_orientation = best_orientation; - best_shape.best_x = best_x; - - return best_shape; - }; - - func.no_preview = true; - return func; - }, - - - _randomShapes: function() { - // Todo: The shapefuncs should be cached. - var shapeFuncs = []; - $.each(this._shapeFactory, function(k,v) { shapeFuncs.push(v); }); - - return this._randChoice(shapeFuncs); - }, - - - /** - * Controls - */ - _setupControls: function(enable) { - - var game = this; - - var moveLeft = function(start) { - if( ! start ) { game._board.holding.left = null; return; } - if( ! game._board.holding.left ) { - game._board.cur.moveLeft(); - game._board.holding.left = Date.now(); - game._board.holding.right = null; - } - } - var moveRight = function(start) { - if( ! start ) { game._board.holding.right = null; return; } - if( ! game._board.holding.right ) { - game._board.cur.moveRight(); - game._board.holding.right = Date.now(); - game._board.holding.left = null; - } - } - var drop = function(start) { - if( ! start ) { game._board.holding.drop = null; return; } - if( ! game._board.holding.drop ) { - game._board.cur.drop(); - game._board.holding.drop = Date.now(); - } - } - var rotateLeft = function() { - game._board.cur.rotate('left'); - } - var rotateRight = function() { - game._board.cur.rotate('right'); - } - - // Handlers: These are used to be able to bind/unbind controls - var handleKeyDown = function(evt) { - if( ! game._board.cur ) { return true; } - var caught = false; - - caught = true; - if (game.options.asdwKeys) { - switch(evt.keyCode) { - case 65: /*a*/ moveLeft(true); break; - case 68: /*d*/ moveRight(true); break; - case 83: /*s*/ drop(true); break; - case 87: /*w*/ game._board.cur.rotate('right'); break; - } - } - switch(evt.keyCode) { - case 37: /*left*/ moveLeft(true); break; - case 39: /*right*/ moveRight(true); break; - case 40: /*down*/ drop(true); break; - case 38: /*up*/ game._board.cur.rotate('right'); break; - case 88: /*x*/ game._board.cur.rotate('right'); break; - case 90: /*z*/ game._board.cur.rotate('left'); break; - default: caught = false; - } - if (caught) evt.preventDefault(); - return !caught; - }; - - - var handleKeyUp = function(evt) { - if( ! game._board.cur ) { return true; } - var caught = false; - - caught = true; - if (game.options.asdwKeys) { - switch(evt.keyCode) { - case 65: /*a*/ moveLeft(false); break; - case 68: /*d*/ moveRight(false); break; - case 83: /*s*/ drop(false); break; - } - } - switch(evt.keyCode) { - case 37: /*left*/ moveLeft(false); break; - case 39: /*right*/ moveRight(false); break; - case 40: /*down*/ drop(false); break; - default: caught = false; - } - if (caught) evt.preventDefault(); - return !caught; - }; - - function isStopKey(evt) { - var cfg = { - stopKeys: {37:1, 38:1, 39:1, 40:1} - }; - - var isStop = (cfg.stopKeys[evt.keyCode] || (cfg.moreStopKeys && cfg.moreStopKeys[evt.keyCode])); - if (isStop) evt.preventDefault(); - return isStop; - } - - function getKey(evt) { return 'safekeypress.' + evt.keyCode; } - - function keydown(evt) { - var key = getKey(evt); - $.data(this, key, ($.data(this, key) || 0) - 1); - return handleKeyDown.call(this, evt); - } - - function keyup(evt) { - $.data(this, getKey(evt), 0); - handleKeyUp.call(this, evt); - return isStopKey(evt); - } - - // Unbind everything by default - // Use event namespacing so we don't ruin other keypress events - $(document) .unbind('keydown.blockrain') - .unbind('keyup.blockrain'); - - if( ! game.options.autoplay ) { - if( enable ) { - $(document) .bind('keydown.blockrain', keydown) - .bind('keyup.blockrain', keyup); - } - } - }, - - - _setupTouchControls: function(enable) { - - var game = this; - - // Movements can be held for faster movement - var moveLeft = function(event){ - event.preventDefault(); - game._board.cur.moveLeft(); - game._board.holding.left = Date.now(); - game._board.holding.right = null; - game._board.holding.drop = null; - }; - var moveRight = function(event){ - event.preventDefault(); - game._board.cur.moveRight(); - game._board.holding.right = Date.now(); - game._board.holding.left = null; - game._board.holding.drop = null; - }; - var drop = function(event){ - event.preventDefault(); - game._board.cur.drop(); - game._board.holding.drop = Date.now(); - }; - var endMoveLeft = function(event){ - event.preventDefault(); - game._board.holding.left = null; - }; - var endMoveRight = function(event){ - event.preventDefault(); - game._board.holding.right = null; - }; - var endDrop = function(event){ - event.preventDefault(); - game._board.holding.drop = null; - }; - - // Rotations can't be held - var rotateLeft = function(event){ - event.preventDefault(); - game._board.cur.rotate('left'); - }; - var rotateRight = function(event){ - event.preventDefault(); - game._board.cur.rotate('right'); - }; - - // Unbind everything by default - game._$touchLeft.unbind('touchstart touchend click'); - game._$touchRight.unbind('touchstart touchend click'); - game._$touchRotateLeft.unbind('touchstart touchend click'); - game._$touchRotateRight.unbind('touchstart touchend click'); - game._$touchDrop.unbind('touchstart touchend click'); - - if( ! game.options.autoplay && enable ) { - game._$touchLeft.show().bind('touchstart click', moveLeft).bind('touchend', endMoveLeft); - game._$touchRight.show().bind('touchstart click', moveRight).bind('touchend', endMoveRight); - game._$touchDrop.show().bind('touchstart click', drop).bind('touchend', endDrop); - game._$touchRotateLeft.show().bind('touchstart click', rotateLeft); - game._$touchRotateRight.show().bind('touchstart click', rotateRight); - } else { - game._$touchLeft.hide(); - game._$touchRight.hide(); - game._$touchRotateLeft.hide(); - game._$touchRotateRight.hide(); - game._$touchDrop.hide(); - } - - } - - }); - -})(jQuery)); \ No newline at end of file diff --git a/src/index.js b/src/index.js new file mode 100644 index 0000000..e6ff363 --- /dev/null +++ b/src/index.js @@ -0,0 +1,1443 @@ +import $ from 'jquery'; + +import Shape from './Shape'; +import BlockrainThemes from './themes'; + + + +const defaultOptions = { + autoplay: false, // Let a bot play the game + autoplayRestart: true, // Restart the game automatically once a bot loses + showFieldOnStart: true, // Show a bunch of random blocks on the start screen (it looks nice) + theme: null, // The theme name or a theme object + blockWidth: 10, // How many blocks wide the field is (The standard is 10 blocks) + autoBlockWidth: false, // The blockWidth is dinamically calculated based on the autoBlockSize. Disabled blockWidth. Useful for responsive backgrounds + autoBlockSize: 24, // The max size of a block for autowidth mode + difficulty: 'normal', // Difficulty (normal|nice|evil). + speed: 20, // The speed of the game. The higher, the faster the pieces go. + asdwKeys: true, // Enable ASDW keys + + // Copy + playText: 'Let\'s play some Tetris', + playButtonText: 'Play', + gameOverText: 'Game Over', + restartButtonText: 'Play Again', + scoreText: 'Score', + + // Basic Callbacks + onStart: function() {}, + onRestart: function() {}, + onGameOver: function(score) {}, + + // When a block is placed + onPlaced: function() {}, + // When a line is made. Returns the number of lines, score assigned and total score + onLine: function(lines, scoreIncrement, score) {} +} + +class Blockrain { + + constructor($el, options) { + this.element = $el; + this.options = $.extend({}, defaultOptions, options); + + // Theme + this._theme = {}; + + // UI Elements + this._$game = null; + this._$canvas = null; + this._$gameholder = null; + this._$start = null; + this._$gameover = null; + this._$score = null; + this._$scoreText = null; + + // Canvas + this._canvas = null; + this._ctx = null; + + this._board = null; + this._info = null; + this._filled = null; + + /** + * Shapes + */ + _shapeFactory = null; + + /** + * The shapes have a reference point (the dot) and always rotate left. + * Keep in mind that the blocks should keep in the same relative position when rotating, + * to allow for custom per-block themes. + */ + _shapes = { + /* + * X + * O XOXX + * X + * X + * . . + */ + line: [ + [ 0, -1, 0, -2, 0, -3, 0, -4], + [ 2, -2, 1, -2, 0, -2, -1, -2], + [ 0, -4, 0, -3, 0, -2, 0, -1], + [-1, -2, 0, -2, 1, -2, 2, -2] + ], + /* + * XX + * XX + */ + square: [ + [0, 0, 1, 0, 0, -1, 1, -1], + [1, 0, 1, -1, 0, 0, 0, -1], + [1, -1, 0, -1, 1, 0, 0, 0], + [0, -1, 0, 0, 1, -1, 1, 0] + ], + /* + * X X X + * XOX XO XOX OX + * . .X .X .X + */ + arrow: [ + [0, -1, 1, -1, 2, -1, 1, -2], + [1, 0, 1, -1, 1, -2, 0, -1], + [2, -1, 1, -1, 0, -1, 1, 0], + [1, -2, 1, -1, 1, 0, 2, -1] + ], + /* + * X X XX + * O XOX O XOX + * .XX . .X X + */ + rightHook: [ + [2, 0, 1, 0, 1, -1, 1, -2], + [2, -2, 2, -1, 1, -1, 0, -1], + [0, -2, 1, -2, 1, -1, 1, 0], + [0, 0, 0, -1, 1, -1, 2, -1] + ], + /* + * X XX X + * O XOX O XOX + * XX . X .X . + */ + leftHook: [ + [0, 0, 1, 0, 1, -1, 1, -2], + [2, 0, 2, -1, 1, -1, 0, -1], + [2, -2, 1, -2, 1, -1, 1, 0], + [0, -2, 0, -1, 1, -1, 2, -1] + ], + /* + * X XX + * XO OX + * X . + */ + leftZag: [ + [0, 0, 0, -1, 1, -1, 1, -2], + [2, -1, 1, -1, 1, -2, 0, -2], + [1, -2, 1, -1, 0, -1, 0, 0], + [0, -2, 1, -2, 1, -1, 2, -1] + ], + /* + * X + * XO OX + * .X XX + */ + rightZag: [ + [1, 0, 1, -1, 0, -1, 0, -2], + [2, -1, 1, -1, 1, 0, 0, 0], + [0, -2, 0, -1, 1, -1, 1, 0], + [0, 0, 1, 0, 1, -1, 2, -1] + ] + }; + + } + + /** + * Start/Restart Game + */ + start() { + this._doStart(); + this.options.onStart.call(this.element); + } + + restart() { + this._doStart(); + this.options.onRestart.call(this.element); + } + + gameover() { + this.showGameOverMessage(); + this._board.gameover = true; + this.options.onGameOver.call(this.element, this._filled.score); + } + + _doStart() { + this._filled.clearAll(); + this._filled._resetScore(); + this._board.cur = this._board.nextShape(); + this._board.started = true; + this._board.gameover = false; + this._board.dropDelay = 5; + this._board.render(true); + this._board.animate(); + + this._$start.fadeOut(150); + this._$gameover.fadeOut(150); + this._$score.fadeIn(150); + } + + pause() { + this._board.paused = true; + } + + resume() { + this._board.paused = false; + } + + autoplay(enable=true) { + // On autoplay, start the game right away + this.options.autoplay = enable; + if (enable && !this._board.started) { + this._doStart(); + } + this._setupControls(!enable); + this._setupTouchControls(!enable); + } + + controls(enable=true) { + this._setupControls(enable); + } + + touchControls(enable=true) { + this._setupTouchControls(enable); + } + + score(newScore) { + if (typeof newScore !== 'undefined' && parseInt(newScore) >= 0) { + this._filled.score = parseInt(newScore); + this._$scoreText.text(this._filled_score); + } + return this._filled.score; + } + + freesquares() { + return this._filled.getFreeSpaces(); + } + + showStartMessage() { + this._$start.show(); + } + + showGameOverMessage() { + this._$gameover.show(); + } + + /** + * Update the sizes of the renderer (this makes the game responsive) + */ + updateSizes() { + this._PIXEL_WIDTH = this.element.innerWidth(); + this._PIXEL_HEIGHT = this.element.innerHeight(); + + this._BLOCK_WIDTH = this.options.blockWidth; + this._BLOCK_HEIGHT = Math.floor(this.element.innerHeight() / this.element.innerWidth() * this._BLOCK_WIDTH); + + this._block_size = Math.floor(this._PIXEL_WIDTH / this._BLOCK_WIDTH); + this._border_width = 2; + + // Recalculate the pixel width and height so the canvas always has the best possible size + this._PIXEL_WIDTH = this._block_size * this._BLOCK_WIDTH; + this._PIXEL_HEIGHT = this._block_size * this._BLOCK_HEIGHT; + + this._$canvas .attr('width', this._PIXEL_WIDTH) + .attr('height', this._PIXEL_HEIGHT); + } + + theme(newTheme) { + if (typeof newTheme === 'undefined') { + return this.options.theme || this._theme; + } + + // Setup the theme properly + if (typeof newTheme === 'string') { + this.options.theme = newTheme; + this._theme = $.extend(true, {}, BlockrainThemes[newTheme]); + } + else { + this.options.theme = null; + this._theme = newTheme; + } + + if (typeof this._theme === 'undefined' || this._theme === null) { + this._theme = $.extend(true, {}, BlockrainThemes['retro']); + this.options.theme = 'retro'; + } + + if (isNaN(parseInt(this._theme.strokeWidth)) || typeof parseInt(this._theme.strokeWidth) !== 'number') { + this._theme.strokeWidth = 2; + } + + // Load the image assets + this._preloadThemeAssets(); + + if (this._board !== null) { + if (typeof this._theme.background === 'string') { + this._$canvas.css('background-color', this._theme.background); + } + this._board.render(); + } + } + + // Initialization + _create() { + let game = this; + + this.theme(this.options.theme); + + this._createHolder(); + this._createUI(); + + this._refreshBlockSizes(); + + this.updateSizes(); + + // $(window).resize(() => this.updateSizes); + + this._SetupShapeFactory(); + this._SetupFilled(); + this._SetupInfo(); + this._SetupBoard(); + + this._info.init(); + this._board.init(); + + const renderLoop = function() { + requestAnimationFrame(renderLoop); + game._board.render(); + }; + renderLoop(); + + if (this.options.autoplay) { + this.autoplay(true); + this._setupTouchControls(false); + } else { + this._setupControls(true); + this._setupTouchControls(false); + } + } + + _checkCollisions(x, y, blocks, checkDownOnly) { + // x & y should be aspirational values + let i = 0, len = blocks.length, a, b; + for (; i= this._BLOCK_HEIGHT || this._filled.check(a, b)) { + return true; + } else if (!checkDownOnly && a < 0 || a >= this._BLOCK_WIDTH) { + return true; + } + } + return false; + } + + /** + * Draws the background + */ + _drawBackground() { + if (typeof this._theme.background !== 'string') { + return; + } + + if (this._theme.backgroundGrid instanceof Image) { + + // Not loaded + if (this._theme.backgroundGrid.width === 0 || this._theme.backgroundGrid.height === 0) { + return; + } + + this._ctx.globalAlpha = 1.0; + + for (let x = 0; x < this._BLOCK_WIDTH; x++) { + for (let y = 0; y < this._BLOCK_HEIGHT; y++) { + let cx = x * this._block_size; + let cy = y * this._block_size; + + this._ctx.drawImage(this._theme.backgroundGrid, + 0, 0, this._theme.backgroundGrid.width, this._theme.backgroundGrid.height, + cx, cy, this._block_size, this._block_size); + } + } + + } else if (typeof this._theme.backgroundGrid === 'string') { + + let borderWidth = this._theme.strokeWidth; + let borderDistance = Math.round(this._block_size * 0.23); + let squareDistance = Math.round(this._block_size * 0.30); + + this._ctx.globalAlpha = 1.0; + this._ctx.fillStyle = this._theme.backgroundGrid; + + for (let x = 0; x < this._BLOCK_WIDTH; x++) { + for (let y = 0; y < this._BLOCK_HEIGHT; y++) { + let cx = x * this._block_size; + let cy = y * this._block_size; + + this._ctx.fillRect(cx+borderWidth, cy+borderWidth, this._block_size-borderWidth * 2, this._block_size-borderWidth * 2); + } + } + + } + + this._ctx.globalAlpha = 1.0; + } + + // TODO: move setups into constructor? + + _SetupShapeFactory() { + let game = this; + if (this._shapeFactory !== null) { return; } + + this._shapeFactory = { + line: function() { + return new Shape(game, game._shapes.line, false, 'line'); + }, + square: function() { + return new Shape(game, game._shapes.square, false, 'square'); + }, + arrow: function() { + return new Shape(game, game._shapes.arrow, false, 'arrow'); + }, + leftHook: function() { + return new Shape(game, game._shapes.leftHook, false, 'leftHook'); + }, + rightHook: function() { + return new Shape(game, game._shapes.rightHook, false, 'rightHook'); + }, + leftZag: function() { + return new Shape(game, game._shapes.leftZag, false, 'leftZag'); + }, + rightZag: function() { + return new Shape(game, game._shapes.rightZag, false, 'rightZag'); + } + }; + } + + _SetupFilled() { + let game = this; + if (this._filled !== null) { return; } + + this._filled = { + data: new Array(game._BLOCK_WIDTH * game._BLOCK_HEIGHT), + score: 0, + toClear: {}, + check: function(x, y) { + return this.data[this.asIndex(x, y)]; + }, + add: function(x, y, blockType, blockVariation, blockIndex, blockOrientation) { + if (x >= 0 && x < game._BLOCK_WIDTH && y >= 0 && y < game._BLOCK_HEIGHT) { + this.data[this.asIndex(x, y)] = { + blockType: blockType, + blockVariation: blockVariation, + blockIndex: blockIndex, + blockOrientation: blockOrientation + }; + } + }, + getFreeSpaces: function() { + let count = 0; + for (let i = 0; i < this.data.length; i++) { + count += (this.data[i] ? 1 : 0); + } + }, + asIndex: function(x, y) { + return x + y * game._BLOCK_WIDTH; + }, + asX: function(index) { + return index % game._BLOCK_WIDTH; + }, + asY: function(index) { + return Math.floor(index / game._BLOCK_WIDTH); + }, + clearAll: function() { + this.data = new Array(game._BLOCK_WIDTH * game._BLOCK_HEIGHT); + }, + _popRow: function(row_to_pop) { + for (let i = game._BLOCK_WIDTH * (row_to_pop + 1) - 1; i >= 0; i--) { + this.data[i] = i >= game._BLOCK_WIDTH ? this.data[i - game._BLOCK_WIDTH] : void 0; + } + }, + checkForClears: function() { + let startLines = game._board.lines; + let rows = [], i, len, count, mod; + + for (let i = 0, len = this.data.length; i < len; i++) { + mod = this.asX(i); + if (mod == 0) count = 0; + if (this.data[i] && typeof this.data[i] !== 'undefined' && typeof this.data[i].blockType === 'string') { + count += 1; + } + if (mod == game._BLOCK_WIDTH - 1 && count == game._BLOCK_WIDTH) { + rows.push(this.asY(i)); + } + } + + for (let i = 0, len = rows.length; i < len; i++) { + this._popRow(rows[i]); + game._board.lines++; + if (game._board.lines % 10 == 0 && game._board.dropDelay > 1) { + game._board.dropDelay *= 0.9; + } + } + + let clearedLines = game._board.lines - startLines; + this._updateScore(clearedLines); + }, + _updateScore: function(numLines) { + if (numLines <= 0) { return; } + let scores = [0, 400, 1000, 3000, 12000]; + if (numLines >= scores.length) { + numLines = scores.length - 1; + } + + this.score += scores[numLines]; + game._$scoreText.text(this.score); + + game.options.onLine.call(game.element, numLines, scores[numLines], this.score); + }, + _resetScore: function() { + this.score = 0; + game._$scoreText.text(this.score); + }, + draw: function() { + for (let i = 0, len = this.data.length, row, color; i= this.dropDelay) || + (game.options.autoplay) || + (this.holding.drop && (now - this.holding.drop) >= this.holdingThreshold)) { + drop = true; + moved = true; + this.dropCount = 0; + } + + // Move Left by holding + if (this.holding.left && (now - this.holding.left) >= this.holdingThreshold) { + moved = true; + this.cur.moveLeft(); + } + + // Move Right by holding + if (this.holding.right && (now - this.holding.right) >= this.holdingThreshold) { + moved = true; + this.cur.moveRight(); + } + + // Test for a collision, add the piece to the filled blocks and fetch the next one + if (drop) { + let cur = this.cur, x = cur.x, y = cur.y, blocks = cur.getBlocks(); + if (game._checkCollisions(x, y+1, blocks, true)) { + drop = false; + let blockIndex = 0; + for (let i = 0; i < cur.blocksLen; i += 2) { + game._filled.add(x + blocks[i], y + blocks[i+1], cur.blockType, cur.blockVariation, blockIndex, cur.orientation); + if (y + blocks[i] < 0) { + gameOver = true; + } + blockIndex++; + } + game._filled.checkForClears(); + this.cur = this.nextShape(); + this.renderChanged = true; + + // Stop holding drop (and any other buttons). Just in case the controls get sticky. + this.holding.left = null; + this.holding.right = null; + this.holding.drop = null; + + game.options.onPlaced.call(game.element); + } + } + } + + // Drop + if (drop) { + moved = true; + this.cur.y++; + } + + if (drop || moved) { + this.renderChanged = true; + } + + if (gameOver) { + this.gameover = true; + + game.gameover(); + + if (game.options.autoplay && game.options.autoplayRestart) { + // On autoplay, restart the game automatically + game.restart(); + } + this.renderChanged = true; + + } else { + + // Update the speed + this.animateDelay = 1000 / game.options.speed; + + this.animateTimeoutId = window.setTimeout(function() { + game._board.animate(); + }, this.animateDelay); + + } + }, + + createRandomBoard: function() { + let start = [], blockTypes = [], i, ilen, j, jlen, blockType; + + // Draw a random blockrain screen + blockTypes = Object.keys(game._shapeFactory); + + for (let i=0, ilen=game._BLOCK_WIDTH; i 0) { + return blockTheme[0]; + } else { + return null; + } + } else { + return blockTheme; + } + } + + if (typeof falling !== 'boolean') { falling = true; } + if (falling) { + if (typeof game._theme.primary === 'string' && game._theme.primary !== '') { + return game._theme.primary; + } else if (typeof game._theme.blocks !== 'undefined' && game._theme.blocks !== null) { + return getBlockVariation(game._theme.blocks[blockType], blockVariation); + } else { + return getBlockVariation(game._theme.complexBlocks[blockType], blockVariation); + } + } else { + if (typeof game._theme.secondary === 'string' && game._theme.secondary !== '') { + return game._theme.secondary; + } else if (typeof game._theme.blocks !== 'undefined' && game._theme.blocks !== null) { + return getBlockVariation(game._theme.blocks[blockType], blockVariation); + } else { + return getBlockVariation(game._theme.complexBlocks[blockType], blockVariation); + } + } + } + + }; + + game._niceShapes = game._getNiceShapes(); + } + + // Utility Functions + _randInt(a, b) { + return a + Math.floor(Math.random() * (1 + b - a)); + } + + _randSign() { + return this._randInt(0, 1) * 2 - 1; + } + + _randChoice(choices) { + return choices[this._randInt(0, choices.length-1)]; + } + + /** + * Find base64 encoded images and load them as image objects, which can be used by the canvas renderer + */ + _preloadThemeAssets() { + let game = this; + let hexColorcheck = new RegExp('^#[A-F0-9+]{3,6}', 'i'); + let base64check = new RegExp('^data:image/(png|gif|jpg);base64,', 'i'); + + const handleAssetLoad = function() { + // Rerender the board as soon as an asset loads + if (game._board) { + game._board.render(true); + } + }; + + const loadAsset = function(src) { + let plainSrc = src; + if (!hexColorcheck.test(plainSrc)) { + // It's an image + src = new Image(); + src.src = plainSrc; + src.onload = handleAssetLoad; + } else { + // It's a color + src = plainSrc; + } + return src; + }; + + const startAssetLoad = function(block) { + // Assets can be an array of variation so they can change color/design randomly + if (Array.isArray(block) && block.length > 0) { + for (let i = 0; i < block.length; i++) { + block[i] = loadAsset(block[i]); + } + } else if (typeof block === 'string') { + block = loadAsset(block); + } + return block; + }; + + // TODO: use same loop instead of 2. DRY + if (typeof this._theme.complexBlocks !== 'undefined') { + let keys = Object.keys(this._theme.complexBlocks); + // Load the complexBlocks + for (let i = 0; i < keys.length; i++) { + this._theme.complexBlocks[keys[i]] = startAssetLoad(this._theme.complexBlocks[keys[i]]); + } + } else if (typeof this._theme.blocks !== 'undefined') { + let keys = Object.keys(this._theme.blocks); + // Load the blocks + for (let i = 0; i < keys.length; i++) { + this._theme.blocks[keys[i]] = startAssetLoad(this._theme.blocks[keys[i]]); + } + } + + // Load the bg + if (typeof this._theme.backgroundGrid !== 'undefined' && + typeof this._theme.backgroundGrid === 'string' && + !hexColorcheck.test(this._theme.backgroundGrid)) { + let src = this._theme.backgroundGrid; + this._theme.backgroundGrid = new Image(); + this._theme.backgroundGrid.src = src; + this._theme.backgroundGrid.onload = handleAssetLoad; + } + } + + _createHolder() { + // Create the main holder (it holds all the ui elements, the original element is just the wrapper) + this._$gameholder = $('
'); + this._$gameholder.css('position', 'relative').css('width', '100%').css('height', '100%'); + + this.element.html('').append(this._$gameholder); + + // Create the game canvas and context + this._$canvas = $(''); + if (typeof this._theme.background === 'string') { + this._$canvas.css('background-color', this._theme.background); + } + this._$gameholder.append(this._$canvas); + + this._canvas = this._$canvas.get(0); + this._ctx = this._canvas.getContext('2d'); + } + + _createUI() { + // Score + this._$score = $( + '
'+ + '
'+ + '
'+ this.options.scoreText +'
'+ + '
0
'+ + '
'+ + '
').hide(); + this._$scoreText = this._$score.find('.blockrain-score-num'); + this._$gameholder.append(this._$score); + + // Create the start menu + this._$start = $( + '
'+ + ''+ + '
').hide(); + this._$gameholder.append(this._$start); + + this._$start.find('.blockrain-start-btn').click((e) => { + e.preventDefault(); + this.start(); + }); + + // Create the game over menu + this._$gameover = $( + '
'+ + '
'+ + '
'+ this.options.gameOverText +'
'+ + ''+ this.options.restartButtonText +''+ + '
'+ + '
').hide(); + this._$gameover.find('.blockrain-game-over-btn').click((e) => { + e.preventDefault(); + this.restart(); + }); + this._$gameholder.append(this._$gameover); + + this._createControls(); + } + + _createControls() { + this._$touchLeft = $('').appendTo(this._$thisholder); + this._$touchRight = $('').appendTo(this._$thisholder); + this._$touchRotateRight = $('').appendTo(this._$thisholder); + this._$touchRotateLeft = $('').appendTo(this._$thisholder); + this._$touchDrop = $('').appendTo(this._$gameholder); + } + + _refreshBlockSizes() { + if (this.options.autoBlockWidth) { + this.options.blockWidth = Math.ceil(this.element.width() / this.options.autoBlockSize); + } + } + + /* + * Things I need for this to work... + * - ability to test each shape with this._filled data + * - maybe give empty spots scores? and try to maximize the score? + */ + _getNiceShapes() { + let game = this; + + let shapes = {}, + attr; + + for (let attr in this._shapeFactory) { + shapes[attr] = this._shapeFactory[attr](); + } + + const scoreBlocks = function scoreBlocks(possibles, blocks, x, y, filled, width, height) { + let i, len=blocks.length, score=0, bottoms = {}, tx, ty, overlaps; + + // base score + for (let i = 0; i < len; i += 2) { + score += possibles[game._filled.asIndex(x + blocks[i], y + blocks[i+1])] || 0; + } + + // overlap score -- //TODO - don't count overlaps if cleared? + for (let i = 0; i < len; i += 2) { + tx = blocks[i]; + ty = blocks[i+1]; + if (typeof bottoms[tx] === 'undefined' || bottoms[tx] < ty) { + bottoms[tx] = ty; + } + } + overlaps = 0; + for (let tx in bottoms) { + tx = parseInt(tx); + for (let ty = bottoms[tx]+1, i = 0; y + ty < height; ty++, i++) { + if (!game._filled.check(x + tx, y + ty)) { + overlaps += i == 0 ? 2 : 1; //TODO-score better + //if (i == 0) overlaps += 1; + break; + } + } + } + + score = score - overlaps; + + return score; + } + + const resetShapes = function resetShapes() { + for (let attr in shapes) { + shapes[attr].x = 0; + shapes[attr].y = -1; + } + } + + //TODO -- evil mode needs to realize that overlap is bad... + const niceShapes = function niceShapes(filled, checkCollisions, width, height, mode, _one_shape) { + if (!_one_shape) resetShapes(); + + let possibles = new Array(width * height), + evil = mode == 'evil', + x, y, py, + attr, shape, i, blocks, bounds, + score, best_shape, best_score = (evil ? 1 : -1) * 999, best_orientation, best_x, + best_score_for_shape, best_orientation_for_shape, best_x_for_shape; + + for (let x = 0; x < width; x++) { + for (let y = 0; y <= height; y++) { + if (y == height || filled.check(x, y)) { + for (let py = y - 4; py < y; py++) { + possibles[filled.asIndex(x, py)] = py; //TODO - figure out better scoring? + } + break; + } + } + } + + // for each shape... + let opts = typeof _one_shape === 'undefined' ? shapes : {cur: _one_shape}; //BOO + for (let attr in opts) { //TODO - check in random order to prevent later shapes from winning + shape = opts[attr]; + best_score_for_shape = -999; + + // for each orientation... + for (let i = 0; i < (shape.symmetrical ? 2 : 4); i++) { //TODO - only look at unique orientations + blocks = shape.getBlocks(i); + bounds = shape.getBounds(blocks); + + // try each possible position... + for (let x = -bounds.left; x < width - bounds.width; x++) { + for (let y = -1; y < height - bounds.bottom; y++) { + if (game._checkCollisions(x, y + 1, blocks, true)) { + // collision + score = scoreBlocks(possibles, blocks, x, y, filled, width, height); + if (score > best_score_for_shape) { + best_score_for_shape = score; + best_orientation_for_shape = i; + best_x_for_shape = x; + } + break; + } + } + } + } + + if ((evil && best_score_for_shape < best_score) || + (!evil && best_score_for_shape > best_score)) { + best_shape = shape; + best_score = best_score_for_shape; + best_orientation = best_orientation_for_shape; + best_x = best_x_for_shape; + } + } + + best_shape.best_orientation = best_orientation; + best_shape.best_x = best_x; + + return best_shape; + }; + + niceShapes.no_preview = true; + return niceShapes; + } + + _randomShapes() { + // Todo: The shapefuncs should be cached. + const shapeFuncs = Object + .keys(this._shapeFactory) + .map((k) => { return this._shapeFactory[k]}) + ; + + return this._randChoice(shapeFuncs); + } + + /** + * Controls + * TODO: DRY this up!!! + */ + _setupControls(enable) { + let game = this; + + const moveLeft = function(start) { + if (!start) { + game._board.holding.left = null; + return; + } + if (!game._board.holding.left) { + game._board.cur.moveLeft(); + game._board.holding.left = Date.now(); + game._board.holding.right = null; + } + } + const moveRight = function(start) { + if (!start) { + game._board.holding.right = null; + return; + } + if (!game._board.holding.right) { + game._board.cur.moveRight(); + game._board.holding.right = Date.now(); + game._board.holding.left = null; + } + } + const drop = function(start) { + if (!start) { + game._board.holding.drop = null; + return; + } + if (!game._board.holding.drop) { + game._board.cur.drop(); + game._board.holding.drop = Date.now(); + } + } + const rotateLeft = function() { + game._board.cur.rotate('left'); + } + const rotateRight = function() { + game._board.cur.rotate('right'); + } + + // Handlers: These are used to be able to bind/unbind controls + const handleKeyDown = function(evt) { + if (!game._board.cur) { return true; } + let caught = true; + + if (game.options.asdwKeys) { + switch(evt.keyCode) { + case 65: /*a*/ moveLeft(true); break; + case 68: /*d*/ moveRight(true); break; + case 83: /*s*/ drop(true); break; + case 87: /*w*/ rotateRight(); break; + } + } + switch(evt.keyCode) { + case 37: /*left*/ moveLeft(true); break; + case 39: /*right*/ moveRight(true); break; + case 40: /*down*/ drop(true); break; + case 38: /*up*/ rotateRight(); break; + case 88: /*x*/ rotateRight(); break; + case 90: /*z*/ rotateLeft(); break; + default: caught = false; + } + if (caught) evt.preventDefault(); + return !caught; + }; + + const handleKeyUp = function(evt) { + if (!game._board.cur) { return true; } + let caught = true; + + if (game.options.asdwKeys) { + switch(evt.keyCode) { + case 65: /*a*/ moveLeft(false); break; + case 68: /*d*/ moveRight(false); break; + case 83: /*s*/ drop(false); break; + } + } + switch(evt.keyCode) { + case 37: /*left*/ moveLeft(false); break; + case 39: /*right*/ moveRight(false); break; + case 40: /*down*/ drop(false); break; + default: caught = false; + } + if (caught) evt.preventDefault(); + return !caught; + }; + + const isStopKey = function(evt) { + let cfg = { + stopKeys: {37:1, 38:1, 39:1, 40:1} + }; + + let isStop = (cfg.stopKeys[evt.keyCode] || (cfg.moreStopKeys && cfg.moreStopKeys[evt.keyCode])); + if (isStop) evt.preventDefault(); + return isStop; + } + + const getKey = function(evt) { return 'safekeypress.' + evt.keyCode; } + + const keydown = function(evt) { + let key = getKey(evt); + // TODO: before it was $.data(this) ... check this was the $el + $.data(game.element, key, ($.data(game.element, key) || 0) - 1); + return handleKeyDown(evt); + } + + const keyup = function(evt) { + $.data(this, getKey(evt), 0); + handleKeyUp.call(this, evt); + return isStopKey(evt); + } + + // Unbind everything by default + // Use event namespacing so we don't ruin other keypress events + $(document).unbind('keydown.blockrain') + .unbind('keyup.blockrain'); + + if (!game.options.autoplay && enable) { + $(document).bind('keydown.blockrain', keydown) + .bind('keyup.blockrain', keyup); + } + } + + _setupTouchControls(enable) { + let game = this; + + // Movements can be held for faster movement + const moveLeft = function(event) { + event.preventDefault(); + game._board.cur.moveLeft(); + game._board.holding.left = Date.now(); + game._board.holding.right = null; + game._board.holding.drop = null; + }; + const moveRight = function(event) { + event.preventDefault(); + game._board.cur.moveRight(); + game._board.holding.right = Date.now(); + game._board.holding.left = null; + game._board.holding.drop = null; + }; + const drop = function(event) { + event.preventDefault(); + game._board.cur.drop(); + game._board.holding.drop = Date.now(); + }; + const endMoveLeft = function(event) { + event.preventDefault(); + game._board.holding.left = null; + }; + const endMoveRight = function(event) { + event.preventDefault(); + game._board.holding.right = null; + }; + const endDrop = function(event) { + event.preventDefault(); + game._board.holding.drop = null; + }; + + // Rotations can't be held + const rotateLeft = function(event) { + event.preventDefault(); + game._board.cur.rotate('left'); + }; + const rotateRight = function(event) { + event.preventDefault(); + game._board.cur.rotate('right'); + }; + + // Unbind everything by default + game._$touchLeft.unbind('touchstart touchend click'); + game._$touchRight.unbind('touchstart touchend click'); + game._$touchRotateLeft.unbind('touchstart touchend click'); + game._$touchRotateRight.unbind('touchstart touchend click'); + game._$touchDrop.unbind('touchstart touchend click'); + + if (!game.options.autoplay && enable) { + game._$touchLeft.show().bind('touchstart click', moveLeft).bind('touchend', endMoveLeft); + game._$touchRight.show().bind('touchstart click', moveRight).bind('touchend', endMoveRight); + game._$touchDrop.show().bind('touchstart click', drop).bind('touchend', endDrop); + game._$touchRotateLeft.show().bind('touchstart click', rotateLeft); + game._$touchRotateRight.show().bind('touchstart click', rotateRight); + } else { + game._$touchLeft.hide(); + game._$touchRight.hide(); + game._$touchRotateLeft.hide(); + game._$touchRotateRight.hide(); + game._$touchDrop.hide(); + } + } + +} diff --git a/src/shape.js b/src/shape.js new file mode 100644 index 0000000..ab34f78 --- /dev/null +++ b/src/shape.js @@ -0,0 +1,95 @@ + +export default class Shape { + + constructor(game, orientations, symmetrical, blockType) { + this.x = 0; + this.y = 0; + // TODO: remove dep to game, current dependency is the wrong way + this.game = game; + this.symmetrical = symmetrical; + this.blockType = blockType; + this.blockVariation = null; + this.blocksLen = orientations[0].length; + this.orientations = orientations; + this.orientation = 0; // 4 possible + } + + init() { + this.orientation = 0; + this.x = Math.floor(this.game._BLOCK_WIDTH / 2) - 1; + this.y = -1; + + return this; + } + + rotate(direction) { + let orientation = (this.orientation + (direction === 'left' ? 1 : -1) + 4) % 4; + + //TODO - when past limit - auto shift and remember that too! + if (!this.game._checkCollisions(this.x, this.y, this.getBlocks(orientation))) { + this.orientation = orientation; + this.game._board.renderChanged = true; + } + } + + moveRight() { + if (!this.game._checkCollisions(this.x + 1, this.y, this.getBlocks())) { + this.x++; + this.game._board.renderChanged = true; + } + } + + moveLeft() { + if (!this.game._checkCollisions(this.x - 1, this.y, this.getBlocks())) { + this.x--; + this.game._board.renderChanged = true; + } + } + + drop() { + if (!this.game._checkCollisions(this.x, this.y + 1, this.getBlocks())) { + this.y++; + // Reset the drop count, as we dropped the block sooner + this.game._board.dropCount = -1; + this.game._board.animate(); + this.game._board.renderChanged = true; + } + } + + getBlocks(orientation) { // optional param + return this.orientations[typeof orientation !== 'undefined' ? orientation : this.orientation]; + } + + draw(_x, _y, _orientation) { + let blocks = this.getBlocks(_orientation), + x = typeof _x === 'undefined' ? this.x : _x, + y = typeof _y === 'undefined' ? this.y : _y, + i = 0, + index = 0; + + for (; i < this.blocksLen; i += 2) { + this.game._board.drawBlock(x + blocks[i], y + blocks[i+1], this.blockType, this.blockletiation, index, this.orientation, true); + index++; + } + } + + getBounds(_blocks) { // _blocks can be an array of blocks, an orientation index, or undefined + let blocks = Array.isArray(_blocks) ? _blocks : this.getBlocks(_blocks), + i=0, len=blocks.length, minx=999, maxx=-999, miny=999, maxy=-999; + for (; i < len; i += 2) { + if (blocks[i] < minx) { minx = blocks[i]; } + if (blocks[i] > maxx) { maxx = blocks[i]; } + if (blocks[i+1] < miny) { miny = blocks[i+1]; } + if (blocks[i+1] > maxy) { maxy = blocks[i+1]; } + } + return { + left: minx, + right: maxx, + top: miny, + bottom: maxy, + width: maxx - minx, + height: maxy - miny + }; + } + +} diff --git a/src/blockrain.jquery.themes.js b/src/themes.js similarity index 99% rename from src/blockrain.jquery.themes.js rename to src/themes.js index 9124c37..c92d425 100644 --- a/src/blockrain.jquery.themes.js +++ b/src/themes.js @@ -1,8 +1,8 @@ /** * Themes. You can add more custom themes to this object. */ -window.BlockrainThemes = { - 'custom': { +export let BlockrainThemes = { + custom: { background: '#040304', backgroundGrid: '', complexBlocks: { @@ -15,7 +15,7 @@ window.BlockrainThemes = { leftZag: 'assets/blocks/custom/leftZag.png' } }, - 'candy': { + candy: { background: '#040304', backgroundGrid: '', blocks: { @@ -28,7 +28,7 @@ window.BlockrainThemes = { leftZag: '' } }, - 'modern': { + modern: { background: '#000000', backgroundGrid: '', primary: null, @@ -44,7 +44,7 @@ window.BlockrainThemes = { leftZag: '#fa1e1e' } }, - 'retro': { + retro: { background: '#000000', backgroundGrid: '', primary: null, @@ -61,7 +61,7 @@ window.BlockrainThemes = { leftZag: '#fa1e1e' } }, - 'monochrome': { + monochrome: { background: '#000000', backgroundGrid: '', primary: '#ffffff', @@ -69,12 +69,12 @@ window.BlockrainThemes = { stroke: '#000000', innerStroke: '#000000' }, - 'aerolab': { + aerolab: { background: '#ffffff', primary: '#ff7b00', secondary: '#000000' }, - 'gameboy': { + gameboy: { background: '#C4CFA1', primary: null, secondary: null, @@ -91,7 +91,7 @@ window.BlockrainThemes = { leftZag: '#595F45' } }, - 'vim': { + vim: { background: '#000000', backgroundGrid: '', primary: '#C2FFAE', @@ -99,5 +99,5 @@ window.BlockrainThemes = { stroke: '#000000', strokeWidth: 3, innerStroke: null - }, -}; \ No newline at end of file + } +};