To write code that is functional, reusable, chainable.
To use Typescript, which constraints the way I use functions and variables, thus reducing runtime errors.
- Uncover mechanics and functionality of the app at a high level. Figure out what functions we would need, and how could we efficiently address every winning scenario. This functionality is mostly inside of tic-tac-toe-functions.ts.
- Test our functions to make sure they are working as we expect them to. (i.e. when we rotate de board, does it check for winners?).
- Document our code, via Typescript and JS DOC. Using VS Code, when you hover on a function, or a type or a const, a chunk of relevant information will display (which could be types added by typescript, or documentation added by the developer) that gives whoever is working on that file more information on how to use it and what it needs. JS DOC can translate its text to markdown, and could become a useful wiki page. (For instance, Rambda, on its website, uses JS DOC to display the documentation of their website).
This is where we started building the functionality of the game, creating types, and pseudo coding the functions that our app would need. This file uses soultrain library the most. Probably the most used function is the pipeline helper function, which behaves as the proposed pipeline operator in JS. WHen using the pipeline operator, the output of a function becomes the input of a new one, something similar to the pipe operator in unix. Typescript is a superset of javascript. It can progressively be added to a project. By default, every type in javaScript (objects, arrays, strings) has an 'any' type. When you indicate types, you are expressing to your function that it takes a specific type of parameter, and that it returns a specific type. This practice prevents runtime errors, since it immediately hightlights the text when something is missing. It also aids in documentation, as it helps other developers visualize better how a function works and what it will return.
Here is an example of one of the functions in this project using typescript:
export const flatBoardToBoard = curry((cellsPerRow: number, flatBoard: FlatBoard): Board =>
chunk(cellsPerRow, flatBoard) as Board
)
This function will take a flatboard and make it into a Board, which is an array with three arrays inside of it, each of which representing a row [[cell, cell, cell], [cell, cell, cell], [cell, cell, cell] ]. If we were to use Board incorrectly, this error would be immediately flagged. Using the curry helper function, we add types to our parameters: cellsPerRow and flatBoard. It will return a Board. So, if we were to pass as a parameter, instead of a FlatBoard, some other kind of array, the function will be highlighted until the error is fixed.
Throughout this file, I documented every function using JS DOC. This becomes very useful since it adds a snippet of info on the characteristics of the function, as well as example on usage. I find this info to be very valuable espcecially when you have to maintain websites over time.
We used React Stateless Components. The two files that handle the UI are index.tsx and TicTacToe.tsx. We have avoided using redux or a similar library to manage this game, since it did not need it at this stage of the game.
We start reading this app from index.tsx. At the bottom of this file there is an init() function that initially renders the game. In this file we have our parent component App. There is an interface that has restricted our App to take two values: board and player, with their respective types that were defined on tic-tac-toe-functions.ts. The app component uses a Maybe monad, which contains either a Just or a nothing (it is sometimes helpful to see the Maybe monad as a conditional statement...it is usually described as a box that contains either a just or nothing, and it has a flatmap/chain operator). The chain method will check if there are any winners. If there aren't it will return nothing, if there are, it will return the winner. If there is a winner, the PlayerNumber winner will display, the GameBoard will re render, but now it will be disabled ( disable={true} ), and a reset button will reset the board by calling the init function onClick. If there is no winner, the map operator will return nothing. Join or value (which is meant to return this.value) will first check that there are no empty squares. If the condition evaluates to true, it will be a tie game, if not, the board will re render.
This file has 2 stateless components. GameBoard and GameBoardRow. As far as types, we define an interface for both of these components, and we also create a type for our RenderCallBack. GameBoard has the 4 parameters described on the interface. If one of those parameters were removed, it will be immediately flagged what type is missing. On game board we are going to map our board, already defined as [ [0,0,0], [0,0,0], 0,0,0] ], and we are going to map it to our component GameBoardRow which is defined below. GameBoardRow will map on each row, and first check if the form is disabled or not. If it is not, then the game is still going on (disable = false). Our board will rerender with player's move and the player is changed.