add: new tictactoe game

add: testing to tictactoe

rename: tictactoe

add: tictactoe steps

refactor: tictactoe const names

refactor: tictactoe loop

add: tictactoe bad switch example (fallthrough)

refactor: tictactoe loop skin

remove: tictactoe changable skin

refactor: tictactoe all

remove: tictactoe unnecessary base dir

add: tictactoe slices

add: tictactoe slices

remove: tictactoe fallthrough

rename: tictactoe slices 10 -> 09

update: loops skin tictactoe

add: tictactoe randomization

add: tictactoe infinite loop and labeled break

refactor: tictactoe rand and infinite loop

add: tictactoe buggy winning algo

add: tictactoe more tests

rename: tictactoe wrongPlay to wrongMove

add: tictactoe even more tests

fix: tictactoe

rename: tictactoe waitForInput to wait

add: tictactoe os.args gameSpeed

remove: tictactoe unnecessary files

rename: tictactoe game messages

refactor: tictactoe main loop

add: types and arrays
This commit is contained in:
Inanc Gumus
2019-07-27 18:16:17 +03:00
parent e191567e1f
commit eb8f9987a8
97 changed files with 3457 additions and 3 deletions

View File

@@ -0,0 +1,26 @@
package main
import "fmt"
// printBoard prints the board
func printBoard() {
fmt.Printf("%s\n\n", banner)
fmt.Printf("%s\n", sepHeader)
for i, step := 0, 3; i < len(cells); i += step {
fmt.Print(sepCell)
for j := 0; j < step; j++ {
move := cells[i+j]
fmt.Printf(" %s %s", move, sepCell)
}
fmt.Println()
if lastLine := (i + step); lastLine != len(cells) {
fmt.Println(sepLine)
}
}
fmt.Printf("%s\n", sepFooter)
}

View File

@@ -0,0 +1,41 @@
package main
// Examples are normally used for showing how to use your package.
// But you can also use them as output testing.
func ExamplePrintBoard() {
printBoard()
// Output:
// ~~~~~~~~~~~~~~~
// TIC~TAC~TOE
// ~~~~~~~~~~~~~~~
//
// /---+---+---\
// | | | |
// +---+---+---+
// | | | |
// +---+---+---+
// | | | |
// \---+---+---/
}
func ExamplePrintBoardCells() {
cells[0] = player1
cells[4] = player2
cells[8] = player1
printBoard()
// Output:
// ~~~~~~~~~~~~~~~
// TIC~TAC~TOE
// ~~~~~~~~~~~~~~~
//
// /---+---+---\
// | X | | |
// +---+---+---+
// | | O | |
// +---+---+---+
// | | | X |
// \---+---+---/
}

View File

@@ -0,0 +1,55 @@
package main
// -------------------------------------------------
// IS THERE A WINNER? OR IS IT A TIE?
// -------------------------------------------------
// /---+---+---\
// | 0 | 1 | 2 |
// +---+---+---+
// | 3 | 4 | 5 |
// +---+---+---+
// | 6 | 7 | 8 |
// \---+---+---/
func checkWinOrTie() {
// intentional bug: tie shouldn't happen before winning detection
if tie = turn == maxTurns; tie {
// if tie don't check for the winning
return
}
// loop over all the players
for i := 1; i <= 2; i++ {
// check for the next player
p := player2
if i == 1 {
p = player1
}
/* check horizontals */
hor := (cells[0] == p && cells[1] == p && cells[2] == p) ||
(cells[3] == p && cells[4] == p && cells[5] == p) ||
(cells[6] == p && cells[7] == p && cells[8] == p)
/* check verticals */
ver := (cells[0] == p && cells[3] == p && cells[6] == p) ||
(cells[1] == p && cells[4] == p && cells[7] == p) ||
(cells[2] == p && cells[5] == p && cells[8] == p)
/* check diagonals */
diag := (cells[0] == p && cells[4] == p && cells[8] == p) ||
(cells[2] == p && cells[4] == p && cells[6] == p)
// any winner?
if hor || ver || diag {
won = true
// this player wins
player = p
// there is a winner so don't check for tie!
return
}
}
}

View File

@@ -0,0 +1,20 @@
package main
import (
"math/rand"
"time"
)
// init is another special function
// Go calls it before the main function
func init() {
rand.Seed(time.Now().UnixNano())
initCells()
}
// initCells initialize the played cells to empty
func initCells() {
for i := range cells {
cells[i] = emptyCell
}
}

View File

@@ -0,0 +1,124 @@
package main
import "fmt"
/*
~ TICTACTOE GAME IN GO ~
+ This example uses the very basics of the Go language.
+ The goal is learning all the basics.
*/
const maxTurns = 9
var (
won, tie bool // is there any winner or a tie?
turn int // total valid turns played
player = player1 // current player
cells [maxTurns]string // used to draw the board: contains the players' moves
lastPos int // last played position
wrongMove bool // was the last move wrong?
)
//
// VERSION 2: HELPER FUNC
//
func main() {
printBoard()
wait()
for nextTurn() {
wait()
}
// loop forever until the game ends (tie or win)
// for {
// // wait()
// if !nextTurn() {
// break
// }
// }
}
func nextTurn() bool {
play()
printBoard()
fmt.Printf("\n>>> PLAYER %q PLAYS to %d\n", player, lastPos+1)
// the switch below is about winning and tie conditions.
// so it is good have checkWinOrTie() as a simple statement.
// totally optional.
switch checkWinOrTie(); {
default:
switchPlayer()
printStatus()
case wrongMove:
fmt.Printf(">>> CELL IS OCCUPIED: PLAY AGAIN!\n")
wrongMove = false // reset for the next turn
case won, tie:
if won {
fmt.Println(">>> WINNER:", player)
} else {
fmt.Println(">>> TIE!")
}
return false
}
return true
}
//
// VERSION 1: LABELED BREAK
//
// func main() {
// printBoard()
// // this a label: it can be used by break, continue and goto
// theGameLoop:
// // loop forever until game ends (tie or win)
// for {
// // wait()
// play()
// printBoard()
// fmt.Printf("\n>>> PLAYER %q PLAYS to %d\n", player, lastPos+1)
// // the switch below is about winning and tie conditions.
// // so it is good have checkWinOrTie() as a simple statement.
// // totally optional.
// switch checkWinOrTie(); {
// default:
// printStatus()
// case wrongMove:
// fmt.Printf(">>> CELL IS OCCUPIED: PLAY AGAIN!\n")
// wrongMove = false // reset for the next turn
// case won, tie:
// if won {
// fmt.Println(">>> WINNER:", player)
// } else {
// fmt.Println(">>> TIE!")
// }
// break theGameLoop
// }
// }
// }
func wait() {
fmt.Println()
fmt.Scanln()
}
// printStatus prints the current status of the game
// it cannot access to the names (vars, consts, etc) inside any other func
func printStatus() {
fmt.Println()
progress := (1 - (float64(turn) / maxTurns)) * 100
fmt.Printf("Current Turn : %d\n", turn)
fmt.Printf("Is there a winner : %t\n", won)
fmt.Printf("Turns left : %.1f%%\n", progress)
}

View File

@@ -0,0 +1,37 @@
package main
import "math/rand"
// play plays the game for the current player.
// + registers the player's move in the board.
// if the move is valid:
// + increases the turn.
func play() {
// pick a random move (very intelligent AI!)
// it can play to the same position!
lastPos = rand.Intn(maxTurns)
// is it a valid move?
if cells[lastPos] != emptyCell {
wrongMove = true
// skip the rest of the function from running
return
}
// register the move: put the player's sign on the board
cells[lastPos] = player
// increment the current turns
turn++
}
// switchPlayer switches to the next player
func switchPlayer() {
// switch the player
if player == player1 {
player = player2
} else {
player = player1
}
}

View File

@@ -0,0 +1,17 @@
package main
const (
banner = `
~~~~~~~~~~~~~~~
TIC~TAC~TOE
~~~~~~~~~~~~~~~`
// skin options :-)
player1, player2 = "X", "O"
emptyCell = " "
sepHeader = `/---+---+---\`
sepLine = `+---+---+---+`
sepFooter = `\---+---+---/`
sepCell = "|"
)