mazda-custom-application-sdk/apps/app.tetris/tetris.js

437 lines
14 KiB
JavaScript
Raw Normal View History

2016-03-15 05:47:00 +00:00
// jQuery Tetris plug-in
// by Alexander Gyoshev (http://blog.gyoshev.net/)
// licensed under a Creative Commons Attribution-ShareAlike 3.0 Unported License. (http://creativecommons.org/licenses/by-sa/3.0/)
// Modified for use with Mazda Infotainment System
(function($) {
var extend = $.extend,
proxy = $.proxy,
keys = {
left: 37,
up: 38,
right: 39,
down: 40
},
// jQuery plug-in
tetris = $.fn.tetris = function(options) {
options = extend($.fn.tetris.defaults, options);
return this.each(function() {
var $this = $(this), instance;
if (!$this.data("tetris")) {
instance = new impl(this, options);
$this.data("tetris", instance);
}
});
},
// Tetris implementation
impl = tetris.implementation = function(element, options) {
var $element = $(element),
that = this;
extend(that, {
element: element,
$element: $element,
frozen: {}
}, options);
that.currentTile = that.generateTile();
$element
.css({
width: that.cols * that.tileSize,
height: that.rows * that.tileSize
})
.bind({
repaint: proxy(that.repaint, that),
tick: proxy(that.tick, that),
tileDrop: proxy(that.tileDrop, that),
rowCompleted: proxy(that.rowCompleted, that)
/// TODO: handle gameOver event with dignity
})
.trigger('repaint');
};
tetris.defaults = {
rows: 22,
cols: 10,
tileSize: 16
};
impl.prototype = {
handle: function(eventId) {
if(this.isGameOver) {
if(eventId == "selectStart") {
this.restartGame();
}
} else {
switch(eventId) {
case "upStart":
case "ccw":
case "cw":
this.rotate();
break;
case "leftStart":
this.move(-1);
break;
case "rightStart":
this.move(1);
break;
case "downStart":
this.down();
break;
}
}
},
tick: function() {
this.down();
this.$element.trigger('repaint');
},
tileDrop: function() {
var that = this;
that.freeze(that.currentTile);
that.$element.find('.current').remove();
that.currentTile = that.generateTile();
if (!that.isValidLocation(that.currentTile.shape)) {
this.gameover();
}
},
rowCompleted: function(e, rowStart) {
var that = this,
i,
cols = that.cols,
tileSize = that.tileSize;
that.$element.find('.frozen')
.filter(function() {
var index = $(this).data('index');
return index - (index % cols) == rowStart;
})
.remove()
.end()
.filter(function() {
var index = $(this).data('index');
if (index - (index % cols) < rowStart)
return index - (index % cols) < rowStart;
})
.css('top', function() {
return parseInt($(this).css('top')) + tileSize;
})
.each(function() {
var t = $(this);
t.data('index', t.data('index') + cols);
});
for (i = rowStart; i < rowStart + cols; i++) {
delete that.frozen[i];
}
for (i = rowStart-1; i >= 0; i--) {
if (that.frozen[i]) {
that.frozen[i + cols] = true;
delete that.frozen[i];
}
}
},
isValidLocation: function(location) {
var i, j,
cols = this.cols,
maxStageIndex = cols * this.rows;
for (i = 0; i < location.length; i++) {
if (location[i] < 0 || location[i] >= maxStageIndex
|| this.frozen[location[i]]) {
return false;
}
for (j = 0; j < i; j++) {
if (((location[i] % cols == 0) && (location[j] % cols == cols - 1))
|| ((location[i] % cols == cols - 1) && (location[j] % cols == 0)))
return false;
}
}
return true;
},
move: function(modifier) {
var that = this,
i,
cols = that.cols,
shape = that.currentTile.shape,
newLocation = $.map(shape, function(x) {
return x + modifier;
}),
hitsEdge = false;
for (i = 0; i < shape.length; i++) {
if ((modifier < 0 && shape[i] % cols == 0)
|| (modifier > 0 && shape[i] % cols == cols - 1)) {
hitsEdge = true;
}
}
if (!hitsEdge && that.isValidLocation(newLocation)) {
that.currentTile.shape = newLocation;
that.$element.trigger('repaint');
}
},
rotate: function() {
var that = this,
currentTile = that.currentTile,
newLocation = currentTile.shape.slice(),
rotation;
if (currentTile.shapeStates) {
rotation = currentTile.shapeStates[currentTile.shapeStateIndex];
newLocation = $.map(newLocation, function(x, index) { return x + rotation[index]; });
} else if (currentTile.shapeRotation) {
newLocation = currentTile.shapeRotation(newLocation);
}
if (that.isValidLocation(newLocation)) {
currentTile.shape = newLocation;
if (currentTile.shapeStates) {
currentTile.shapeStateIndex = (++currentTile.shapeStateIndex) % currentTile.shapeStates.length;
}
}
that.$element.trigger('repaint');
},
down: function() {
var that = this,
cols = that.cols,
maxStageIndex = cols * that.rows,
shape = that.currentTile.shape,
newLocation = $.map(shape, function(x) { return x + cols; });
if (that.isValidLocation(newLocation)) {
that.currentTile.shape = newLocation;
that.$element.trigger('repaint');
} else {
that.$element.trigger('tileDrop');
}
},
generateTile: function(type) {
// build shape cache
var cols = this.cols,
center = Math.floor(cols/2) + cols,
direction = [-cols, +1, +cols, -1];
function squareRotation(shape) {
var directions = [-cols-1, -cols, -cols+1,
-1, 0, +1,
+cols-1, +cols, +cols+1],
rotation = [-cols+1, +1, +cols+1,
-cols , 0, +cols,
-cols-1, -1, +cols-1],
center = shape[0];
return $.map(shape, function(coord) {
for (var i = 0; i < directions.length; i++) {
if (coord == center + directions[i]) {
return center + rotation[i];
}
}
});
}
if (!this.tileCache) {
/// TODO: allow extensibility for custom tiles
/// TODO: move this somewhere else
this.tileCache = [
{
type: 'O',
shape: [ center, center+1, center+direction[0], center+direction[0]+1 ]
},
{
type: 'J',
shape: [ center, center-1, center+1, center-1+direction[0] ],
shapeRotation: squareRotation
},
{
type: 'L',
shape: [ center, center-1, center+1, center+1+direction[0] ],
shapeRotation: squareRotation
},
{
type: 'I',
shape: [ center-1, center, center+1, center+2 ],
shapeStates: [
[+2-cols, +1, +cols, +2*cols-1],
[+1+2*cols, +cols, -1, -2-cols],
[-2+cols, -1, -cols, -2*cols+1],
[-1-2*cols, -cols, +1, +2+cols]
],
shapeStateIndex: 0
},
{
type: 'S',
shape: [ center, center-1, center+direction[0], center+direction[0]+1 ],
shapeRotation: squareRotation
},
{
type: 'Z',
shape: [ center, center+1, center+direction[0], center+direction[0]-1 ],
shapeRotation: squareRotation
},
{
type: 'T',
shape: [ center, center-1, center+1, center+direction[0] ],
shapeRotation: squareRotation
}
];
}
if (typeof type != 'undefined') {
for (var i = 0; i < this.tileCache.length; i++) {
if (this.tileCache[i].type == type) {
tileIndex = i;
break;
}
}
} else {
// Random Generator using Knuth shuffle (http://tetris.wikia.com/wiki/Random_Generator)
if (!this.randomBag || this.randomBag.length == 0) {
var tilesCount = this.tileCache.length;
this.randomBag = [];
for (var i = 0; i < tilesCount; i++) {
this.randomBag[i] = i;
}
for (var i = tilesCount - 1; i > 0; i--) {
var rand = Math.floor(Math.random() * i),
tmp = this.randomBag[rand];
this.randomBag[rand] = this.randomBag[i];
this.randomBag[i] = tmp;
}
}
tileIndex = this.randomBag.shift();
}
return extend({}, this.tileCache[tileIndex], { shapeLocation: squareRotation });
},
freeze: function(tile) {
var frozenTilesHtml = [],
shape = tile.shape,
tileSize = this.tileSize,
cols = this.cols,
rowsToCheck = [];
for (var i = 0; i < shape.length; i++) {
if ($.inArray(shape[i] - (shape[i] % cols), rowsToCheck) === -1) {
rowsToCheck.push(shape[i] - (shape[i] % cols));
}
this.frozen[shape[i]] = true;
frozenTilesHtml.push('<div class="tile frozen type-' + tile.type + '" />');
}
$(frozenTilesHtml.join(''))
.each(function(i) {
$(this).css({
left: (shape[i] % cols) * tileSize,
top: Math.floor(shape[i] / cols) * tileSize
})
.data('index', shape[i]);
})
.appendTo(this.element);
while (rowsToCheck.length) {
var rowStart = rowsToCheck.shift(),
broken = false;
for (var i = rowStart; i < rowStart + cols; i++) {
if (!this.frozen[i])
broken = true;
}
if (!broken) {
this.$element.trigger('rowCompleted', rowStart);
}
}
},
repaint: function() {
var cols = this.cols,
tileSize = this.tileSize,
shape = this.currentTile.shape,
currentTile = this.$element.find('.current');
if (currentTile.length == 0) {
// render new tile
var currentTileHtml = [];
for (var h = 0; h < shape.length; h++) {
currentTileHtml.push('<div class="tile current type-' + this.currentTile.type + '" />');
}
currentTile = this.$element.append(currentTileHtml.join('')).find('.current');
}
// position shape
for (var i = 0; i < shape.length; i++) {
currentTile.eq(i).css({
left: (shape[i] % cols) * tileSize,
top: Math.floor(shape[i] / cols) * tileSize
});
}
},
start: function() {
if(this.isStarted) return;
this.isStarted = true;
var $element = this.$element;
if (!this.isValidLocation(this.currentTile.shape)) {
this.gameover();
}
/// TODO: improve timer
this.timer = setInterval(function() {
$element.trigger('tick');
}, 600);
},
pause: function() {
this.isStarted = false;
if (this.timer) {
window.clearInterval(this.timer);
this.timer = null;
}
},
gameover: function() {
this.isStarted = false;
this.isGameOver = true;
this.$element.trigger('gameOver');
},
restartGame: function() {
this.$element.empty();
this.$element.trigger('restartGame');
this.isStarted = false;
this.isGameOver = false;
this.frozen = {};
this.start();
}
};
})(jQuery);