Created
November 23, 2016 06:31
-
-
Save anonymous/15d7f2ecbace23a4b7de37ef8db0cfcd to your computer and use it in GitHub Desktop.
Tic-Tac-Toe // source https://jsbin.com/qototad
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<!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'>♥</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> |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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 */ | |
} |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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