Skip to content

Instantly share code, notes, and snippets.

Created November 23, 2016 06:31
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 1 You must be signed in to fork a gist
  • Save anonymous/15d7f2ecbace23a4b7de37ef8db0cfcd to your computer and use it in GitHub Desktop.
Save anonymous/15d7f2ecbace23a4b7de37ef8db0cfcd to your computer and use it in GitHub Desktop.
Tic-Tac-Toe // source https://jsbin.com/qototad
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=600, initial-scale=1">
<title>Tic-Tac-Toe</title>
<style id="jsbin-css">
html {
background: #f5f5f5;
}
/* The wrapper
* wraps the app
*/
.wrapper {
width: 80.3vw;
height: 90.3vw;
display: block;
margin-left: auto;
margin-right: auto;
margin-top: 10vw;
font-family: sans-serif;
}
/* The wrapper's intro.
* Here goes the title
* and the controls.
*/
.wrapper #intro {
text-align: center;
}
.wrapper #intro .title {
font-size: 12vw;
font-weight: bold;
display: inline-block;
}
.wrapper #intro .title .x,
.wrapper #intro .title .o {
opacity: 0;
float: left;
}
.wrapper #intro .title .x {
color: #e88;
}
.wrapper #intro .title .o {
color: #8be;
}
.wrapper #intro .message {
opacity: 0;
font-size: 5vw;
color: #999;
margin-top: -4px;
margin-bottom: 4px;
}
.wrapper #intro #restart {
background: none;
color: #e88;
border: none;
font-weight: bold;
text-transform: uppercase;
font-size: 5vw;
display: none;
cursor: pointer;
text-align: center;
width: 100%;
margin-top: -5px;
}
.wrapper #intro #restart.visible {
display: block;
}
/* tic-tac-toe blocks.
* People play here.
*/
.wrapper #tic-tac-toe div {
display: block;
float: left;
width: 25vw;
height: 25vw;
margin-left: 1.2vw;
margin-top: 1.2vw;
font-size: 22vw;
font-weight: bold;
text-align: center;
background: white;
color: #888;
cursor: pointer;
opacity: 0;
}
.wrapper #tic-tac-toe div.x {
color: #e88;
cursor: default;
}
.wrapper #tic-tac-toe div.o {
color: #8be;
cursor: default;
}
.wrapper #tic-tac-toe div.yellow {
background: #ffb;
}
.wrapper #tic-tac-toe div.showCursor:before {
background-image: url();
background-size: 12vw 12vw;
display: block;
width: 12vw;
height: 12vw;
content: '';
opacity: 0;
margin-left: 6.5vw;
margin-top: 6vw;
animation: click 5s 1s infinite;
}
@keyframes click {
0% {
opacity: 0;
}
3% {
opacity: 0.5;
}
15% {
margin-left: 6.5vw;
margin-top: 6vw;
}
20% {
margin-left: 6vw;
margin-top: 5.5vw;
}
25% {
margin-left: 6.5vw;
margin-top: 6vw;
}
30% {
margin-left: 6vw;
margin-top: 5.5vw;
}
35% {
margin-left: 6.5vw;
margin-top: 6vw;
}
47% {
opacity: 0.5;
}
50% {
opacity: 0;
}
}
/*
* Winner highlight
*/
.wrapper #winner {
display: none;
float: left;
margin-top: -80vw;
margin-bottom: -113px;
margin-left: -5vw;
width: 90vw;
height: 90vw;
font-size: 22vw;
text-align: center;
text-transform: uppercase;
padding-top: 17vw;
background: rgba(245, 245, 245, .8);
}
/* The author.
* Hi mom!
*/
.wrapper .author {
font-size: 3vw;
color: #999;
float: right;
margin-right: 5px;
margin-top: 2vw;
margin-bottom: 5vw;
}
.wrapper .author .hearts {
color: #e88;
}
.wrapper .author a {
text-decoration: none;
color: #8be;
}
.wrapper .author a:hover {
text-decoration: underline;
}
/*
* Misc - taken from elsewhere
* Sources:
* .noselect - https://stackoverflow.com/questions/826782/how-to-disable-text-selection-highlighting-using-css
*/
.noselect, .wrapper #tic-tac-toe div {
-webkit-touch-callout: none; /* iOS Safari */
-webkit-user-select: none; /* Chrome/Safari/Opera */
-khtml-user-select: none; /* Konqueror */
-moz-user-select: none; /* Firefox */
-ms-user-select: none; /* Internet Explorer/Edge */
user-select: none; /* Non-prefixed version, currently
not supported by any browser */
}
</style>
</head>
<body>
<div class='wrapper'>
<div id='intro'>
<div class='title'>
<div class='x'>Tic-</div>
<div class='o'>Tac-</div>
<div class='x'>Toe!</div>
</div>
<div class='message'>Click in the board</div>
<button id='restart'>Restart</button>
</div>
<div id='tic-tac-toe' class='.noselect'>
<div class=''></div>
<div class=''></div>
<div class=''></div>
<div class=''></div>
<div class=''></div>
<div class=''></div>
<div class=''></div>
<div class=''></div>
<div class=''></div>
</div>
<div id='winner'>
<!--
PLAYER
<b>X</b>
WINS!
-->
</div>
<div class='author'>
Made with
<span class='hearts'>&hearts;</span>
by
<a href='http://twitter.com/sadasant' target='_blank'>
@sadasant
</a>
</div>
</div>
<script id="jsbin-javascript">
window.onload = function() {
// Makes arrays
function makeArray(obj) {
return Array.prototype.slice.call(obj)
}
// Vanilla jQuery for the lazy
var $ = function() {
var args = makeArray(arguments);
var r = document.querySelectorAll.apply(document, args);
return r.length === 1 ? r[0] : r;
}
// Vanilla class functions
function addClass(el, className) {
removeClass(el, className)
el.className += ' ' + className
}
function removeClass(el, className) {
el.className = el.className.replace(className, '')
}
// Our main variables
var title = $('.wrapper #intro .title');
var message = $('.wrapper #intro .message');
var tictactoe = $('.wrapper #tic-tac-toe');
var blocks = $('.wrapper #tic-tac-toe div');
var restart = $('#restart');
var winner = $('#winner');
// Animated intro!
intro()
function intro() {
function show(el) {
el.style.opacity = 1;
}
function hide(el) {
el.style.opacity = 0;
}
var delayer = new Delayer();
var titleParts = makeArray(title.children);
var blocksArray = makeArray(blocks);
// Flushing
message.style.display = 'block'
winner.style.display = 'none'
hide(message)
removeClass(restart, 'visible')
titleParts.map(hide)
blocksArray.map(function(el) {
el.className = '';
el.innerText = '';
hide(el)
})
delayer.add(titleParts, show, 500)
delayer.add(blocksArray, show, 100)
delayer.add([blocks[0]], function(first) {
addClass(first, 'showCursor')
blocksArray.map(function(el) {
el.onclick = function() {
removeClass(first, 'showCursor')
message.style.display = 'none'
first.onclick = undefined
startGame(el)
}
})
}, 100)
delayer.add([message], show, 500)
delayer.run()
}
// Main game logic
var startsX = true;
function startGame(el) {
var player1 = {
str: startsX ? 'x' : 'o'
}
var player2 = {
str: !startsX ? 'x' : 'o',
}
var goesP1 = true;
var turns = 0;
addClass(restart, 'visible')
restart.onclick = function() {
intro()
}
var blocksArray = makeArray(blocks);
blocksArray.map(function(el) {
el.onclick = function() {
if (el.innerText) return
var player = goesP1 ? player1 : player2;
el.innerText = player.str.toUpperCase();
addClass(el, player.str);
goesP1 = !goesP1;
turns++
if (turns === 9) {
setTimeout(tie, 125);
} else
if (turns >= 5) {
setTimeout(function() {
checkWinner(player, el);
}, 125)
}
}
})
function checkWinner(player, el) {
var B = blocksArray;
var i = B.indexOf(el);
var str = player.str.toUpperCase()
function eqAt(a, b, c) {
return B[a].innerText === B[b].innerText && B[a].innerText === B[c].innerText;
}
var col = i % 3;
var row = i - col;
if (eqAt(row, row + 1, row + 2)) return won(player, row, row + 1, row + 2);
if (eqAt(col, col + 3, col + 6)) return won(player, col, col + 3, col + 6);
if (eqAt(0, 4, 8)) return won(player, 0, 4, 8)
if (eqAt(2, 4, 6)) return won(player, 2, 4, 6)
}
function won(player, a, b, c) {
addClass(blocksArray[a], 'yellow')
addClass(blocksArray[b], 'yellow')
addClass(blocksArray[c], 'yellow')
winner.style.display = 'block';
winner.style.color = player.str === 'x' ? '#e88' : '#8be';
winner.innerHTML = 'player <b>' + player.str + '</b> wins!'
blocksArray.map(function(el) {
el.onclick = undefined
})
}
function tie() {
winner.style.display = 'block';
winner.style.color = '#999';
winner.innerHTML = 'IT WAS<br>A TIE'
}
el.onclick()
}
// Delayer object
// adds animation *frames* into
// a strack, then replays them
function Delayer() {
var stack = []
this.add = function(arr, func, time) {
if (!(arr && arr.length)) {
throw 'Delayer.add error: invalid first argument, must be an array'
}
arr.map(function(el) {
stack.push({
el: el,
func: func,
timeout: time
})
})
}
var that = this
this.run = function() {
if (!stack.length) return
var current = stack.shift()
setTimeout(function() {
current.func(current.el)
that.run()
}, current.timeout)
}
}
}
</script>
<script id="jsbin-source-css" type="text/css">html {
background: #f5f5f5;
}
/* The wrapper
* wraps the app
*/
.wrapper {
width: 80.3vw;
height: 90.3vw;
display: block;
margin-left: auto;
margin-right: auto;
margin-top: 10vw;
font-family: sans-serif;
}
/* The wrapper's intro.
* Here goes the title
* and the controls.
*/
.wrapper #intro {
text-align: center;
}
.wrapper #intro .title {
font-size: 12vw;
font-weight: bold;
display: inline-block;
}
.wrapper #intro .title .x,
.wrapper #intro .title .o {
opacity: 0;
float: left;
}
.wrapper #intro .title .x {
color: #e88;
}
.wrapper #intro .title .o {
color: #8be;
}
.wrapper #intro .message {
opacity: 0;
font-size: 5vw;
color: #999;
margin-top: -4px;
margin-bottom: 4px;
}
.wrapper #intro #restart {
background: none;
color: #e88;
border: none;
font-weight: bold;
text-transform: uppercase;
font-size: 5vw;
display: none;
cursor: pointer;
text-align: center;
width: 100%;
margin-top: -5px;
}
.wrapper #intro #restart.visible {
display: block;
}
/* tic-tac-toe blocks.
* People play here.
*/
.wrapper #tic-tac-toe div {
display: block;
float: left;
width: 25vw;
height: 25vw;
margin-left: 1.2vw;
margin-top: 1.2vw;
font-size: 22vw;
font-weight: bold;
text-align: center;
background: white;
color: #888;
cursor: pointer;
opacity: 0;
}
.wrapper #tic-tac-toe div.x {
color: #e88;
cursor: default;
}
.wrapper #tic-tac-toe div.o {
color: #8be;
cursor: default;
}
.wrapper #tic-tac-toe div.yellow {
background: #ffb;
}
.wrapper #tic-tac-toe div.showCursor:before {
background-image: url();
background-size: 12vw 12vw;
display: block;
width: 12vw;
height: 12vw;
content: '';
opacity: 0;
margin-left: 6.5vw;
margin-top: 6vw;
animation: click 5s 1s infinite;
}
@keyframes click {
0% {
opacity: 0;
}
3% {
opacity: 0.5;
}
15% {
margin-left: 6.5vw;
margin-top: 6vw;
}
20% {
margin-left: 6vw;
margin-top: 5.5vw;
}
25% {
margin-left: 6.5vw;
margin-top: 6vw;
}
30% {
margin-left: 6vw;
margin-top: 5.5vw;
}
35% {
margin-left: 6.5vw;
margin-top: 6vw;
}
47% {
opacity: 0.5;
}
50% {
opacity: 0;
}
}
/*
* Winner highlight
*/
.wrapper #winner {
display: none;
float: left;
margin-top: -80vw;
margin-bottom: -113px;
margin-left: -5vw;
width: 90vw;
height: 90vw;
font-size: 22vw;
text-align: center;
text-transform: uppercase;
padding-top: 17vw;
background: rgba(245, 245, 245, .8);
}
/* The author.
* Hi mom!
*/
.wrapper .author {
font-size: 3vw;
color: #999;
float: right;
margin-right: 5px;
margin-top: 2vw;
margin-bottom: 5vw;
}
.wrapper .author .hearts {
color: #e88;
}
.wrapper .author a {
text-decoration: none;
color: #8be;
}
.wrapper .author a:hover {
text-decoration: underline;
}
/*
* Misc - taken from elsewhere
* Sources:
* .noselect - https://stackoverflow.com/questions/826782/how-to-disable-text-selection-highlighting-using-css
*/
.noselect, .wrapper #tic-tac-toe div {
-webkit-touch-callout: none; /* iOS Safari */
-webkit-user-select: none; /* Chrome/Safari/Opera */
-khtml-user-select: none; /* Konqueror */
-moz-user-select: none; /* Firefox */
-ms-user-select: none; /* Internet Explorer/Edge */
user-select: none; /* Non-prefixed version, currently
not supported by any browser */
}</script>
<script id="jsbin-source-javascript" type="text/javascript">window.onload = function() {
// Makes arrays
function makeArray(obj) {
return Array.prototype.slice.call(obj)
}
// Vanilla jQuery for the lazy
var $ = function() {
var args = makeArray(arguments);
var r = document.querySelectorAll.apply(document, args);
return r.length === 1 ? r[0] : r;
}
// Vanilla class functions
function addClass(el, className) {
removeClass(el, className)
el.className += ' ' + className
}
function removeClass(el, className) {
el.className = el.className.replace(className, '')
}
// Our main variables
var title = $('.wrapper #intro .title');
var message = $('.wrapper #intro .message');
var tictactoe = $('.wrapper #tic-tac-toe');
var blocks = $('.wrapper #tic-tac-toe div');
var restart = $('#restart');
var winner = $('#winner');
// Animated intro!
intro()
function intro() {
function show(el) {
el.style.opacity = 1;
}
function hide(el) {
el.style.opacity = 0;
}
var delayer = new Delayer();
var titleParts = makeArray(title.children);
var blocksArray = makeArray(blocks);
// Flushing
message.style.display = 'block'
winner.style.display = 'none'
hide(message)
removeClass(restart, 'visible')
titleParts.map(hide)
blocksArray.map(function(el) {
el.className = '';
el.innerText = '';
hide(el)
})
delayer.add(titleParts, show, 500)
delayer.add(blocksArray, show, 100)
delayer.add([blocks[0]], function(first) {
addClass(first, 'showCursor')
blocksArray.map(function(el) {
el.onclick = function() {
removeClass(first, 'showCursor')
message.style.display = 'none'
first.onclick = undefined
startGame(el)
}
})
}, 100)
delayer.add([message], show, 500)
delayer.run()
}
// Main game logic
var startsX = true;
function startGame(el) {
var player1 = {
str: startsX ? 'x' : 'o'
}
var player2 = {
str: !startsX ? 'x' : 'o',
}
var goesP1 = true;
var turns = 0;
addClass(restart, 'visible')
restart.onclick = function() {
intro()
}
var blocksArray = makeArray(blocks);
blocksArray.map(function(el) {
el.onclick = function() {
if (el.innerText) return
var player = goesP1 ? player1 : player2;
el.innerText = player.str.toUpperCase();
addClass(el, player.str);
goesP1 = !goesP1;
turns++
if (turns === 9) {
setTimeout(tie, 125);
} else
if (turns >= 5) {
setTimeout(function() {
checkWinner(player, el);
}, 125)
}
}
})
function checkWinner(player, el) {
var B = blocksArray;
var i = B.indexOf(el);
var str = player.str.toUpperCase()
function eqAt(a, b, c) {
return B[a].innerText === B[b].innerText && B[a].innerText === B[c].innerText;
}
var col = i % 3;
var row = i - col;
if (eqAt(row, row + 1, row + 2)) return won(player, row, row + 1, row + 2);
if (eqAt(col, col + 3, col + 6)) return won(player, col, col + 3, col + 6);
if (eqAt(0, 4, 8)) return won(player, 0, 4, 8)
if (eqAt(2, 4, 6)) return won(player, 2, 4, 6)
}
function won(player, a, b, c) {
addClass(blocksArray[a], 'yellow')
addClass(blocksArray[b], 'yellow')
addClass(blocksArray[c], 'yellow')
winner.style.display = 'block';
winner.style.color = player.str === 'x' ? '#e88' : '#8be';
winner.innerHTML = 'player <b>' + player.str + '</b> wins!'
blocksArray.map(function(el) {
el.onclick = undefined
})
}
function tie() {
winner.style.display = 'block';
winner.style.color = '#999';
winner.innerHTML = 'IT WAS<br>A TIE'
}
el.onclick()
}
// Delayer object
// adds animation *frames* into
// a strack, then replays them
function Delayer() {
var stack = []
this.add = function(arr, func, time) {
if (!(arr && arr.length)) {
throw 'Delayer.add error: invalid first argument, must be an array'
}
arr.map(function(el) {
stack.push({
el: el,
func: func,
timeout: time
})
})
}
var that = this
this.run = function() {
if (!stack.length) return
var current = stack.shift()
setTimeout(function() {
current.func(current.el)
that.run()
}, current.timeout)
}
}
}</script></body>
</html>
html {
background: #f5f5f5;
}
/* The wrapper
* wraps the app
*/
.wrapper {
width: 80.3vw;
height: 90.3vw;
display: block;
margin-left: auto;
margin-right: auto;
margin-top: 10vw;
font-family: sans-serif;
}
/* The wrapper's intro.
* Here goes the title
* and the controls.
*/
.wrapper #intro {
text-align: center;
}
.wrapper #intro .title {
font-size: 12vw;
font-weight: bold;
display: inline-block;
}
.wrapper #intro .title .x,
.wrapper #intro .title .o {
opacity: 0;
float: left;
}
.wrapper #intro .title .x {
color: #e88;
}
.wrapper #intro .title .o {
color: #8be;
}
.wrapper #intro .message {
opacity: 0;
font-size: 5vw;
color: #999;
margin-top: -4px;
margin-bottom: 4px;
}
.wrapper #intro #restart {
background: none;
color: #e88;
border: none;
font-weight: bold;
text-transform: uppercase;
font-size: 5vw;
display: none;
cursor: pointer;
text-align: center;
width: 100%;
margin-top: -5px;
}
.wrapper #intro #restart.visible {
display: block;
}
/* tic-tac-toe blocks.
* People play here.
*/
.wrapper #tic-tac-toe div {
display: block;
float: left;
width: 25vw;
height: 25vw;
margin-left: 1.2vw;
margin-top: 1.2vw;
font-size: 22vw;
font-weight: bold;
text-align: center;
background: white;
color: #888;
cursor: pointer;
opacity: 0;
}
.wrapper #tic-tac-toe div.x {
color: #e88;
cursor: default;
}
.wrapper #tic-tac-toe div.o {
color: #8be;
cursor: default;
}
.wrapper #tic-tac-toe div.yellow {
background: #ffb;
}
.wrapper #tic-tac-toe div.showCursor:before {
background-image: url();
background-size: 12vw 12vw;
display: block;
width: 12vw;
height: 12vw;
content: '';
opacity: 0;
margin-left: 6.5vw;
margin-top: 6vw;
animation: click 5s 1s infinite;
}
@keyframes click {
0% {
opacity: 0;
}
3% {
opacity: 0.5;
}
15% {
margin-left: 6.5vw;
margin-top: 6vw;
}
20% {
margin-left: 6vw;
margin-top: 5.5vw;
}
25% {
margin-left: 6.5vw;
margin-top: 6vw;
}
30% {
margin-left: 6vw;
margin-top: 5.5vw;
}
35% {
margin-left: 6.5vw;
margin-top: 6vw;
}
47% {
opacity: 0.5;
}
50% {
opacity: 0;
}
}
/*
* Winner highlight
*/
.wrapper #winner {
display: none;
float: left;
margin-top: -80vw;
margin-bottom: -113px;
margin-left: -5vw;
width: 90vw;
height: 90vw;
font-size: 22vw;
text-align: center;
text-transform: uppercase;
padding-top: 17vw;
background: rgba(245, 245, 245, .8);
}
/* The author.
* Hi mom!
*/
.wrapper .author {
font-size: 3vw;
color: #999;
float: right;
margin-right: 5px;
margin-top: 2vw;
margin-bottom: 5vw;
}
.wrapper .author .hearts {
color: #e88;
}
.wrapper .author a {
text-decoration: none;
color: #8be;
}
.wrapper .author a:hover {
text-decoration: underline;
}
/*
* Misc - taken from elsewhere
* Sources:
* .noselect - https://stackoverflow.com/questions/826782/how-to-disable-text-selection-highlighting-using-css
*/
.noselect, .wrapper #tic-tac-toe div {
-webkit-touch-callout: none; /* iOS Safari */
-webkit-user-select: none; /* Chrome/Safari/Opera */
-khtml-user-select: none; /* Konqueror */
-moz-user-select: none; /* Firefox */
-ms-user-select: none; /* Internet Explorer/Edge */
user-select: none; /* Non-prefixed version, currently
not supported by any browser */
}
window.onload = function() {
// Makes arrays
function makeArray(obj) {
return Array.prototype.slice.call(obj)
}
// Vanilla jQuery for the lazy
var $ = function() {
var args = makeArray(arguments);
var r = document.querySelectorAll.apply(document, args);
return r.length === 1 ? r[0] : r;
}
// Vanilla class functions
function addClass(el, className) {
removeClass(el, className)
el.className += ' ' + className
}
function removeClass(el, className) {
el.className = el.className.replace(className, '')
}
// Our main variables
var title = $('.wrapper #intro .title');
var message = $('.wrapper #intro .message');
var tictactoe = $('.wrapper #tic-tac-toe');
var blocks = $('.wrapper #tic-tac-toe div');
var restart = $('#restart');
var winner = $('#winner');
// Animated intro!
intro()
function intro() {
function show(el) {
el.style.opacity = 1;
}
function hide(el) {
el.style.opacity = 0;
}
var delayer = new Delayer();
var titleParts = makeArray(title.children);
var blocksArray = makeArray(blocks);
// Flushing
message.style.display = 'block'
winner.style.display = 'none'
hide(message)
removeClass(restart, 'visible')
titleParts.map(hide)
blocksArray.map(function(el) {
el.className = '';
el.innerText = '';
hide(el)
})
delayer.add(titleParts, show, 500)
delayer.add(blocksArray, show, 100)
delayer.add([blocks[0]], function(first) {
addClass(first, 'showCursor')
blocksArray.map(function(el) {
el.onclick = function() {
removeClass(first, 'showCursor')
message.style.display = 'none'
first.onclick = undefined
startGame(el)
}
})
}, 100)
delayer.add([message], show, 500)
delayer.run()
}
// Main game logic
var startsX = true;
function startGame(el) {
var player1 = {
str: startsX ? 'x' : 'o'
}
var player2 = {
str: !startsX ? 'x' : 'o',
}
var goesP1 = true;
var turns = 0;
addClass(restart, 'visible')
restart.onclick = function() {
intro()
}
var blocksArray = makeArray(blocks);
blocksArray.map(function(el) {
el.onclick = function() {
if (el.innerText) return
var player = goesP1 ? player1 : player2;
el.innerText = player.str.toUpperCase();
addClass(el, player.str);
goesP1 = !goesP1;
turns++
if (turns === 9) {
setTimeout(tie, 125);
} else
if (turns >= 5) {
setTimeout(function() {
checkWinner(player, el);
}, 125)
}
}
})
function checkWinner(player, el) {
var B = blocksArray;
var i = B.indexOf(el);
var str = player.str.toUpperCase()
function eqAt(a, b, c) {
return B[a].innerText === B[b].innerText && B[a].innerText === B[c].innerText;
}
var col = i % 3;
var row = i - col;
if (eqAt(row, row + 1, row + 2)) return won(player, row, row + 1, row + 2);
if (eqAt(col, col + 3, col + 6)) return won(player, col, col + 3, col + 6);
if (eqAt(0, 4, 8)) return won(player, 0, 4, 8)
if (eqAt(2, 4, 6)) return won(player, 2, 4, 6)
}
function won(player, a, b, c) {
addClass(blocksArray[a], 'yellow')
addClass(blocksArray[b], 'yellow')
addClass(blocksArray[c], 'yellow')
winner.style.display = 'block';
winner.style.color = player.str === 'x' ? '#e88' : '#8be';
winner.innerHTML = 'player <b>' + player.str + '</b> wins!'
blocksArray.map(function(el) {
el.onclick = undefined
})
}
function tie() {
winner.style.display = 'block';
winner.style.color = '#999';
winner.innerHTML = 'IT WAS<br>A TIE'
}
el.onclick()
}
// Delayer object
// adds animation *frames* into
// a strack, then replays them
function Delayer() {
var stack = []
this.add = function(arr, func, time) {
if (!(arr && arr.length)) {
throw 'Delayer.add error: invalid first argument, must be an array'
}
arr.map(function(el) {
stack.push({
el: el,
func: func,
timeout: time
})
})
}
var that = this
this.run = function() {
if (!stack.length) return
var current = stack.shift()
setTimeout(function() {
current.func(current.el)
that.run()
}, current.timeout)
}
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment