brunch

You can make anything
by writing

C.S.Lewis

by 팔이오 Jun 19. 2020

[React Project] tic tac tok 게임

프로젝트 연습

이 프로젝트는 리액트 공식문서 (https://ko.reactjs.org/tutorial/tutorial.html#storing-a-history-of-moves) 를 보고 따라 만든 리액트 연습용 프로젝트 입니다.


1. 프로젝트 순서

cmd 창 open

프로젝트를 생성할 폴더로 이동 :  cd 폴더경로

npx create-react-app 프로젝트명 혹은 npm init react-app 프로젝트명

프로젝트 폴더로 이동 : cd 프로젝트명

프로젝트 폴더 내부에 있는 src 폴더로 이동 : cd src

src 내부에 있는 모든 파일들을 삭제 : (linux/mac : rm -f *), (window : del *)

다시 프로젝트 폴더로 이동 : cd 프로젝트명

코드 에디터 열기

이 문서의 2단계를 실행 후 npm start 명령어 입력

코드 작성하기


2. CSS / JS 파일 추가

src 폴더 안에 아래의 두 파일 추가


- index.css

body {

  font: 14px "Century Gothic", Futura, sans-serif;

  margin: 20px;

}

ol, ul {

  padding-left: 30px;

}

.board-row:after {

  clear: both;

  content: "";

  display: table;

}

.status {

  margin-bottom: 10px;

}

.square {

  background: #fff;

  border: 1px solid #999;

  float: left;

  font-size: 24px;

  font-weight: bold;

  line-height: 34px;

  height: 34px;

  margin-right: -1px;

  margin-top: -1px;

  padding: 0;

  text-align: center;

  width: 34px;

}

.square:focus {

  outline: none;

}

.kbd-navigation .square:focus {

  background: #ddd;

}

.game {

  display: flex;

  flex-direction: row;

}

.game-info {

  margin-left: 20px;

}



- index.js

import React from 'react'; 

import ReactDOM from 'react-dom'; 

import './index.css';


class Square extends React.Component {

  render() {

    return (

      <button className="square">

        {/* TODO */}

      </button>

    );

  }

}

class Board extends React.Component {

  renderSquare(i) {

    return <Square />;

  }

  render() {

    const status = 'Next player: X';

    return (

      <div>

        <div className="status">{status}</div>

        <div className="board-row">

          {this.renderSquare(0)}

          {this.renderSquare(1)}

          {this.renderSquare(2)}

        </div>

        <div className="board-row">

          {this.renderSquare(3)}

          {this.renderSquare(4)}

          {this.renderSquare(5)}

        </div>

        <div className="board-row">

          {this.renderSquare(6)}

          {this.renderSquare(7)}

          {this.renderSquare(8)}

        </div>

      </div>

    );

  }

}

class Game extends React.Component {

  render() {

    return (

      <div className="game">

        <div className="game-board">

          <Board />

        </div>

        <div className="game-info">

          <div>{/* status */}</div>

          <ol>{/* TODO */}</ol>

        </div>

      </div>

    );

  }

}

// ========================================

ReactDOM.render(

  <Game />,

  document.getElementById('root')

);



3. tic tac tok 게임 (설명은 주석으로 되어있음)

- index.js 를 아래와 같이 수정한다.


import React from 'react'; // import 실제사용할별명 from 패키지경로및패키지명

import ReactDOM from 'react-dom';

import './index.css';


// 리액트에서는 JSX를 사용한다. Javascript를 확장한 것으로

// js 내부에서 html 코드를 작성할 수 있다.

// html 코드 내에서 {} 중괄호를 사용하여 자바스크립트 표현식, 변수 등을 표시할 수 있다. 


  //함수 컴포넌트 정의

  function Square(props) {


      //Game -> Board -> Square 로 넘겨준 props.onClick을 실행함

      return (

        <button className="square" onClick={props.onClick}>

            {props.value}

        </button>


        // 클래스 컴포넌트 정의인 경우 onClick prop의 값은 {() => this.props.onClick()} 이어야 함.

        // 괄호가 사라진 것에 대해 주의하기

      );

  }


  //클래스 컴포넌트 정의

  class Board extends React.Component {


      renderSquare(i) { //prop 정의를 한 Square 컴포넌트를 반환한다

        //Board에서 onClick 이벤트로 Game 컴포넌트에서 넘겨받은 이벤트 핸들러를 Square 컴포넌트에 넘겨주고 있음

        return (<Square 

            value={this.props.squares[i]}

            onClick={() => this.props.onClick(i)} />

        );



        //화살표 함수의 경우 다음과 같이 쓸 수 있다

        // () => this.props.onClick(i) == function() {return this.props.onClick(i)}

        // 인자가 없을 경우 : () => n // 괄호 생략 불가능

        // 인자가 하나일 경우 : param => n 혹은 (param) => n // 괄호 생략 가능

        // 인자가 여러개일 경우 : (param1, param2) => n // 괄호 생략 불가능

      }




    render() {

      return (

        // this (Board 클래스) 의 renderSquare 메서드를 실행한다.

        //renderSquare(n) => 이때 n값은 배열(Board.props.history[n].squares)에 매칭/이력 추가시킬 인덱스 값

        <div>

          <div className="board-row">

            {this.renderSquare(0)}

            {this.renderSquare(1)}

            {this.renderSquare(2)}

          </div>

          <div className="board-row">

            {this.renderSquare(3)}

            {this.renderSquare(4)}

            {this.renderSquare(5)}

          </div>

          <div className="board-row">

            {this.renderSquare(6)}

            {this.renderSquare(7)}

            {this.renderSquare(8)}

          </div>

        </div>

      );

    }

}


class Game extends React.Component {


    //Square 컴포넌트 (3*3 각각의 사각형 박스) 를 클릭했을 때 발생하는 이벤트

    handleClick(i) {

        //slice의 매개변수는 (시작 인덱스, 자를 인덱스 + 1) 로 표기해야 원하는 배열이 추출된다 (표기한 end 인덱스의 값은 포함 안하기 때문)

        const history = this.state.history.slice(0, this.state.stepNumber + 1); // 현재 판까지의 게임 기록을 가져옴 (특정 기록으로 이동 후 그 상태로 플레이 한다면 미래 기록을 날려야 하기 때문)

        const current = history[history.length - 1];

        const squares = current.squares.slice(); 


        if(calculateWinner(squares) || squares[i]){ // 현재 판에서 누가 이겼거나 이미 값이 채워져 있는 경우에는 handleClick 메서드를 더이상 실행시키지 않는다.

            return;

        }


        squares[i] = this.state.xIsNext ? 'X' : 'O'; // 클릭할 때마다 O or X 토글 시켜 띄우도록


        this.setState({

            history : history.concat([{ // 기존 배열에 추가하는 것이 아닌 똑같이 생긴 새 배열에 원소를 추가하는 방식을 더 권장함

                squares : squares,

            }]),

            stepNumber : history.length,

            xIsNext : !this.state.xIsNext

        });

    }


    //게임의 타임테이블 버튼을 누를 때 발생하는 이벤트 핸들러

    jumpTo(step) { // stepNumber를 업데이트 하는 역할

        this.setState({

            stepNumber : step,

            xIsNext : (step % 2) === 0, // stepNumber가 짝수일 때마다 xIsNext를 true로 설정 == 짝수일 때마다 클릭하면 X가 뜸

        })

    }


    constructor(props) {

        super(props); // 리액트에서의 생성자 정의 시 super(props) 호출 필수

        this.state = { // 현재 저장값 모음

            history : [{ 

                squares : Array(9).fill(null),

            }],

            stepNumber : 0, // 현재 진행중인 게임 단계(사용자가 선택한 게임 이력의 인덱스)

            xIsNext : true,

        }

    }


    render() {


      const history = this.state.history; // 게임 이력 데이터

      const current = history[this.state.stepNumber]; // 선택된 기록의 플레이 상태를 가지고 옴

      const winner = calculateWinner(current.squares); //누가 승자인지 체크


      const moves = history.map((step, move) => { // 게임 이력 리스트의 길이만큼 버튼 생성

            const desc = move ?

            'Go to move #' + move :

            'Go to game start';


            return ( 


                // db와 같이 사용할 경우 button에 key prop 를 추천한다. 어떠한 이벤트가 작용했을 경우

                // 리스트의 상태가 변한다면 사람은 단번에 알아보지만, 리액트는 컴퓨터 프로그램으로 사람이 의도한 바를 알 수 없다.

                // key prop 를 지정하면 어떤 컴포넌트의 상태를 업데이트(삭제, 추가 등) 하기에 용이하기 때문이다

                // 지정하지 않을 경우 기본값으로 인덱스 값이 할당되지만 경고가 뜬다.

                // key={i} 를 사용할 경우 경고는 안 뜨지만 동일한 문제가 생기므로 권장하지 않는다.

                // 새로 랜더링 했을 경우 기존에 있던 리스트에 포함된 키가 리랜더링을 한 새로운 리스트에는 그 키가 없을 경우 

                // 해당 컴포넌트를 삭제하고, 존재하지 않는 키를 가지고 있다면 컴포넌트를 새로 생성한다.

                // 동적 리스트를 랜더링 할 경우 적절한 키를 할당하는 것을 권장한다.

                // db에 마땅한 값이 없을 경우 데이터 재구성을 고려해야한다.


                <li key={move}>

                    <button onClick={() => this.jumpTo(move)}>{desc}</button>

                </li>

            )

      });


      let status;


      if(winner) { // 승자가 있을 경우

        status = 'Winner: ' + winner;

      } else { // 승자가 없을 경우

        status = 'Next player: '+(this.state.xIsNext ? 'X' : 'O');

      }


      return ( // 현재 상태의 플레이 기록과 handleClick 메서드를 onClick 핸들러로 Board 컴포넌트에 넘겨준다.

        <div className="game">

          <div className="game-board">

            <Board squares={current.squares}

            onClick={(i) => this.handleClick(i)} />

          </div>

          <div className="game-info">

            <div>{status}</div>

            <ol>{moves}</ol>

          </div>

        </div>

      );

    }

  }


  // ========================================


  ReactDOM.render(

    <Game />,

    document.getElementById('root')

  );


  //승자체크 함수

  function calculateWinner(squares) {

    const lines = [ // 이길 수 있는 경우의 수

      [0, 1, 2],

      [3, 4, 5],

      [6, 7, 8],

      [0, 3, 6],

      [1, 4, 7],

      [2, 5, 8],

      [0, 4, 8],

      [2, 4, 6],

    ];

    for (let i = 0; i < lines.length; i++) {

      const [a, b, c] = lines[i]; // 만약 i가 0이라면 => a = 0, b = 1, c = 2

      if (squares[a] && squares[a] === squares[b] && squares[a] === squares[c]) { // 빈 값이 아니고 해당 배열 인덱스에 해당되는 값이 일치한다면 승자(이때 배열값은 X 혹은 O 혹은 null 값이다.)

        return squares[a];

      }

    }

    return null;

  }

브런치는 최신 브라우저에 최적화 되어있습니다. IE chrome safari