-
-
Save sadasant/91f68703cc59533ea88e7744604ce92a 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 id="viewport" name="viewport" content="width=device-width, initial-scale=1"> | |
<title>Tic-Tac-Toe</title> | |
</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> | |
</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: 41.5vh; | |
height: 90.3vh; | |
display: block; | |
margin-left: auto; | |
margin-right: auto; | |
margin-top: 5.1vh; | |
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: 6.2vh; | |
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: 2.58vh; | |
color: #999; | |
margin-top: -4px; | |
margin-bottom: 4px; | |
transition: opacity 0.3s; | |
} | |
.wrapper #intro #restart { | |
background: none; | |
color: #e88; | |
border: none; | |
font-weight: bold; | |
text-transform: uppercase; | |
font-size: 2.58vh; | |
display: none; | |
cursor: pointer; | |
text-align: center; | |
width: 100%; | |
margin-top: -0.75vh; | |
} | |
.wrapper #intro #restart.visible { | |
display: block; | |
} | |
/* tic-tac-toe blocks. | |
* People play here. | |
*/ | |
.wrapper #tic-tac-toe div { | |
display: block; | |
float: left; | |
width: 12.99vh; | |
height: 12.8vh; | |
margin-left: 0.6vh; | |
margin-top: 0.7vh; | |
font-size: 11.35vh; | |
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: 6.19vh 6.19vh; | |
display: block; | |
width: 6.19vh; | |
height: 6.19vh; | |
content: ''; | |
opacity: 0; | |
margin-left: 3.5vh; | |
margin-top: 3vh; | |
animation: click 5s 0s infinite; | |
} | |
@keyframes click { | |
0% { | |
opacity: 0; | |
} | |
3% { | |
opacity: 0.5; | |
} | |
15% { | |
margin-left: 3.5vh; | |
margin-top: 3vh; | |
} | |
20% { | |
margin-left: 3.4vh; | |
margin-top: 2.8vh; | |
} | |
25% { | |
margin-left: 3.5vh; | |
margin-top: 3vh; | |
} | |
30% { | |
margin-left: 3.4vh; | |
margin-top: 2.8vh; | |
} | |
35% { | |
margin-left: 3.5vh; | |
margin-top: 3vh; | |
} | |
47% { | |
opacity: 0.5; | |
} | |
50% { | |
opacity: 0; | |
} | |
} | |
/* | |
* Winner highlight | |
*/ | |
.wrapper #winner { | |
display: none; | |
float: left; | |
margin-top: -41.2vh; | |
margin-bottom: -113px; | |
margin-left: -2.6vh; | |
width: 46.5vh; | |
height: 35vh; | |
font-size: 11.35vh; | |
text-align: center; | |
text-transform: uppercase; | |
padding-top: 8.8vh; | |
background: rgba(245, 245, 245, .8); | |
} | |
/* The author. | |
* Hi mom! | |
*/ | |
.wrapper .author { | |
font-size: 1.55vh; | |
color: #999; | |
float: right; | |
margin-right: 5px; | |
margin-top: 1.2vh; | |
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([message], show, 500) | |
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) | |
} | |
}) | |
}, 0) | |
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