2015-01-30 01:10:49 +00:00
|
|
|
/**
|
2015-03-26 18:24:15 +00:00
|
|
|
* The examples provided by Facebook are for non-commercial testing and
|
|
|
|
* evaluation purposes only.
|
2015-03-23 22:07:33 +00:00
|
|
|
*
|
2015-03-26 18:24:15 +00:00
|
|
|
* Facebook reserves all rights not expressly granted.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
|
|
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NON INFRINGEMENT. IN NO EVENT SHALL
|
|
|
|
* FACEBOOK BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
|
|
|
|
* AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
|
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
2015-01-30 01:10:49 +00:00
|
|
|
*
|
|
|
|
* @providesModule TicTacToeApp
|
|
|
|
* @flow
|
|
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
|
2016-04-09 03:36:40 +00:00
|
|
|
var React = require('react');
|
|
|
|
var ReactNative = require('react-native');
|
2015-01-30 01:10:49 +00:00
|
|
|
var {
|
2015-02-19 14:57:05 +00:00
|
|
|
AppRegistry,
|
2015-01-30 01:10:49 +00:00
|
|
|
StyleSheet,
|
|
|
|
Text,
|
|
|
|
TouchableHighlight,
|
|
|
|
View,
|
2016-04-09 03:36:40 +00:00
|
|
|
} = ReactNative;
|
2015-01-30 01:10:49 +00:00
|
|
|
|
|
|
|
class Board {
|
|
|
|
grid: Array<Array<number>>;
|
|
|
|
turn: number;
|
|
|
|
|
|
|
|
constructor() {
|
|
|
|
var size = 3;
|
|
|
|
var grid = Array(size);
|
|
|
|
for (var i = 0; i < size; i++) {
|
|
|
|
var row = Array(size);
|
|
|
|
for (var j = 0; j < size; j++) {
|
|
|
|
row[j] = 0;
|
|
|
|
}
|
|
|
|
grid[i] = row;
|
|
|
|
}
|
|
|
|
this.grid = grid;
|
|
|
|
|
|
|
|
this.turn = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
mark(row: number, col: number, player: number): Board {
|
|
|
|
this.grid[row][col] = player;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2015-02-08 01:23:25 +00:00
|
|
|
hasMark(row: number, col: number): boolean {
|
2015-01-30 01:10:49 +00:00
|
|
|
return this.grid[row][col] !== 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
winner(): ?number {
|
|
|
|
for (var i = 0; i < 3; i++) {
|
|
|
|
if (this.grid[i][0] !== 0 && this.grid[i][0] === this.grid[i][1] &&
|
|
|
|
this.grid[i][0] === this.grid[i][2]) {
|
|
|
|
return this.grid[i][0];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (var i = 0; i < 3; i++) {
|
|
|
|
if (this.grid[0][i] !== 0 && this.grid[0][i] === this.grid[1][i] &&
|
|
|
|
this.grid[0][i] === this.grid[2][i]) {
|
|
|
|
return this.grid[0][i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.grid[0][0] !== 0 && this.grid[0][0] === this.grid[1][1] &&
|
|
|
|
this.grid[0][0] === this.grid[2][2]) {
|
|
|
|
return this.grid[0][0];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.grid[0][2] !== 0 && this.grid[0][2] === this.grid[1][1] &&
|
|
|
|
this.grid[0][2] === this.grid[2][0]) {
|
|
|
|
return this.grid[0][2];
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
tie(): boolean {
|
|
|
|
for (var i = 0; i < 3; i++) {
|
|
|
|
for (var j = 0; j < 3; j++) {
|
|
|
|
if (this.grid[i][j] === 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return this.winner() === null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-26 08:00:02 +00:00
|
|
|
class Cell extends React.Component {
|
|
|
|
cellStyle = () => {
|
2015-01-30 01:10:49 +00:00
|
|
|
switch (this.props.player) {
|
|
|
|
case 1:
|
|
|
|
return styles.cellX;
|
|
|
|
case 2:
|
|
|
|
return styles.cellO;
|
|
|
|
default:
|
|
|
|
return null;
|
|
|
|
}
|
2016-07-26 08:00:02 +00:00
|
|
|
};
|
2015-01-30 01:10:49 +00:00
|
|
|
|
2016-07-26 08:00:02 +00:00
|
|
|
textStyle = () => {
|
2015-01-30 01:10:49 +00:00
|
|
|
switch (this.props.player) {
|
|
|
|
case 1:
|
|
|
|
return styles.cellTextX;
|
|
|
|
case 2:
|
|
|
|
return styles.cellTextO;
|
|
|
|
default:
|
|
|
|
return {};
|
|
|
|
}
|
2016-07-26 08:00:02 +00:00
|
|
|
};
|
2015-01-30 01:10:49 +00:00
|
|
|
|
2016-07-26 08:00:02 +00:00
|
|
|
textContents = () => {
|
2015-01-30 01:10:49 +00:00
|
|
|
switch (this.props.player) {
|
|
|
|
case 1:
|
|
|
|
return 'X';
|
|
|
|
case 2:
|
|
|
|
return 'O';
|
|
|
|
default:
|
|
|
|
return '';
|
|
|
|
}
|
2016-07-26 08:00:02 +00:00
|
|
|
};
|
2015-01-30 01:10:49 +00:00
|
|
|
|
|
|
|
render() {
|
|
|
|
return (
|
2015-02-08 01:23:25 +00:00
|
|
|
<TouchableHighlight
|
|
|
|
onPress={this.props.onPress}
|
|
|
|
underlayColor="transparent"
|
|
|
|
activeOpacity={0.5}>
|
2015-01-30 01:10:49 +00:00
|
|
|
<View style={[styles.cell, this.cellStyle()]}>
|
2015-07-06 11:03:12 +00:00
|
|
|
<Text style={[styles.cellText, this.textStyle()]}>
|
|
|
|
{this.textContents()}
|
|
|
|
</Text>
|
2015-01-30 01:10:49 +00:00
|
|
|
</View>
|
|
|
|
</TouchableHighlight>
|
|
|
|
);
|
|
|
|
}
|
2016-07-26 08:00:02 +00:00
|
|
|
}
|
2015-01-30 01:10:49 +00:00
|
|
|
|
2016-07-26 08:00:02 +00:00
|
|
|
class GameEndOverlay extends React.Component {
|
2015-01-30 01:10:49 +00:00
|
|
|
render() {
|
|
|
|
var board = this.props.board;
|
|
|
|
|
|
|
|
var tie = board.tie();
|
|
|
|
var winner = board.winner();
|
|
|
|
if (!winner && !tie) {
|
|
|
|
return <View />;
|
|
|
|
}
|
|
|
|
|
|
|
|
var message;
|
|
|
|
if (tie) {
|
|
|
|
message = 'It\'s a tie!';
|
|
|
|
} else {
|
|
|
|
message = (winner === 1 ? 'X' : 'O') + ' wins!';
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<View style={styles.overlay}>
|
|
|
|
<Text style={styles.overlayMessage}>{message}</Text>
|
2015-02-08 01:23:25 +00:00
|
|
|
<TouchableHighlight
|
|
|
|
onPress={this.props.onRestart}
|
|
|
|
underlayColor="transparent"
|
|
|
|
activeOpacity={0.5}>
|
2015-01-30 01:10:49 +00:00
|
|
|
<View style={styles.newGame}>
|
|
|
|
<Text style={styles.newGameText}>New Game</Text>
|
|
|
|
</View>
|
|
|
|
</TouchableHighlight>
|
|
|
|
</View>
|
|
|
|
);
|
|
|
|
}
|
2016-07-26 08:00:02 +00:00
|
|
|
}
|
2015-01-30 01:10:49 +00:00
|
|
|
|
|
|
|
var TicTacToeApp = React.createClass({
|
|
|
|
getInitialState() {
|
|
|
|
return { board: new Board(), player: 1 };
|
|
|
|
},
|
|
|
|
|
|
|
|
restartGame() {
|
|
|
|
this.setState(this.getInitialState());
|
|
|
|
},
|
|
|
|
|
|
|
|
nextPlayer(): number {
|
|
|
|
return this.state.player === 1 ? 2 : 1;
|
|
|
|
},
|
|
|
|
|
|
|
|
handleCellPress(row: number, col: number) {
|
2015-02-08 01:23:25 +00:00
|
|
|
if (this.state.board.hasMark(row, col)) {
|
2015-01-30 01:10:49 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.setState({
|
|
|
|
board: this.state.board.mark(row, col, this.state.player),
|
|
|
|
player: this.nextPlayer(),
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
render() {
|
|
|
|
var rows = this.state.board.grid.map((cells, row) =>
|
|
|
|
<View key={'row' + row} style={styles.row}>
|
|
|
|
{cells.map((player, col) =>
|
|
|
|
<Cell
|
|
|
|
key={'cell' + col}
|
|
|
|
player={player}
|
|
|
|
onPress={this.handleCellPress.bind(this, row, col)}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</View>
|
|
|
|
);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<View style={styles.container}>
|
|
|
|
<Text style={styles.title}>EXTREME T3</Text>
|
|
|
|
<View style={styles.board}>
|
|
|
|
{rows}
|
|
|
|
</View>
|
2015-02-08 01:23:25 +00:00
|
|
|
<GameEndOverlay
|
|
|
|
board={this.state.board}
|
|
|
|
onRestart={this.restartGame}
|
|
|
|
/>
|
2015-01-30 01:10:49 +00:00
|
|
|
</View>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
var styles = StyleSheet.create({
|
|
|
|
container: {
|
|
|
|
flex: 1,
|
|
|
|
justifyContent: 'center',
|
|
|
|
alignItems: 'center',
|
|
|
|
backgroundColor: 'white'
|
|
|
|
},
|
|
|
|
title: {
|
|
|
|
fontFamily: 'Chalkduster',
|
|
|
|
fontSize: 39,
|
|
|
|
marginBottom: 20,
|
|
|
|
},
|
|
|
|
board: {
|
|
|
|
padding: 5,
|
|
|
|
backgroundColor: '#47525d',
|
|
|
|
borderRadius: 10,
|
|
|
|
},
|
|
|
|
row: {
|
|
|
|
flexDirection: 'row',
|
|
|
|
},
|
|
|
|
|
|
|
|
// CELL
|
|
|
|
|
|
|
|
cell: {
|
|
|
|
width: 80,
|
|
|
|
height: 80,
|
|
|
|
borderRadius: 5,
|
|
|
|
backgroundColor: '#7b8994',
|
|
|
|
margin: 5,
|
|
|
|
flex: 1,
|
|
|
|
justifyContent: 'center',
|
|
|
|
alignItems: 'center',
|
|
|
|
},
|
|
|
|
cellX: {
|
|
|
|
backgroundColor: '#72d0eb',
|
|
|
|
},
|
|
|
|
cellO: {
|
|
|
|
backgroundColor: '#7ebd26',
|
|
|
|
},
|
|
|
|
|
|
|
|
// CELL TEXT
|
|
|
|
|
|
|
|
cellText: {
|
|
|
|
fontSize: 50,
|
|
|
|
fontFamily: 'AvenirNext-Bold',
|
|
|
|
},
|
|
|
|
cellTextX: {
|
|
|
|
color: '#19a9e5',
|
|
|
|
},
|
|
|
|
cellTextO: {
|
|
|
|
color: '#b9dc2f',
|
|
|
|
},
|
|
|
|
|
|
|
|
// GAME OVER
|
|
|
|
|
|
|
|
overlay: {
|
|
|
|
position: 'absolute',
|
|
|
|
top: 0,
|
|
|
|
bottom: 0,
|
|
|
|
left: 0,
|
|
|
|
right: 0,
|
|
|
|
backgroundColor: 'rgba(221, 221, 221, 0.5)',
|
|
|
|
flex: 1,
|
|
|
|
flexDirection: 'column',
|
|
|
|
justifyContent: 'center',
|
|
|
|
alignItems: 'center',
|
|
|
|
},
|
|
|
|
overlayMessage: {
|
|
|
|
fontSize: 40,
|
|
|
|
marginBottom: 20,
|
|
|
|
marginLeft: 20,
|
|
|
|
marginRight: 20,
|
|
|
|
fontFamily: 'AvenirNext-DemiBold',
|
|
|
|
textAlign: 'center',
|
|
|
|
},
|
|
|
|
newGame: {
|
2016-02-11 16:31:45 +00:00
|
|
|
backgroundColor: '#887765',
|
2015-01-30 01:10:49 +00:00
|
|
|
padding: 20,
|
|
|
|
borderRadius: 5,
|
|
|
|
},
|
|
|
|
newGameText: {
|
|
|
|
color: 'white',
|
|
|
|
fontSize: 20,
|
|
|
|
fontFamily: 'AvenirNext-DemiBold',
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2015-02-19 14:57:05 +00:00
|
|
|
AppRegistry.registerComponent('TicTacToeApp', () => TicTacToeApp);
|
2015-01-30 01:10:49 +00:00
|
|
|
|
|
|
|
module.exports = TicTacToeApp;
|