This project was bootstrapped with Create React App.
This is a template to quickly initialise a react project with, what I believe to be, best practices in mind. The technologies initialised are: React, TypeScript, ESlint, Prettier, Emotion CSS-in-JS, and Cypress automated testing. This will preconfigure ESLint, Prettier, and a tsconfig file, and also provides instruction on how to alter it to suite your needs. It also provides the useful VSCode extensions for this template. A .env
file and a cypress.env.json
file are also created, and automatically added to a .gitignore
to ensure sensitive info is not committed.
Below is a list of the VSCode extensions recommended to get the most out of this template. Note, the ones with asterixis (*) I consider required, or their equivalent if using a different IDE/editor.
- Bracket Pair Colorizer (*)
- ESLint (*)
- GitLens (*)
- JavaScript and Typescript Nightly (*)
- Prettier (*)
- vscode-styled-components (*)
- ES7 React/Redux/GraphQL snippets
- JavaScript (ES6) code snippets
- Redux DevTools
- Sass/Less/Stylus/Pug/Jade/Typesciprt/Javascript Compile hero process
- Simple React Snippets
This project currently uses Emotion for CSS. This allows the creation of css stylings in the same file as Components to keep things scoped nicely.
The following comment is required at the top of every file which will use css-in-js, as this tells React to grab the JSX creation from emotion instead of React.createElement.
/** @jsxImportSource @emotion/react */
This means we can create styles like this:
import React from "react";
import { css } from "@emotion/react";
const hoverColor = "green";
const styling = css`
color: red;
&:hover {
color: ${hoverColor};
font-size: 20px;
}
`;
Note, this will have syntax highlighting and autocomplete for the css when the above extensions are installed.
We then add the stylings as such using the css
property we imported.
const Another: React.FC = (): JSX.Element => {
return (
<div css={styling}>
<h2>Another thing here.</h2>
</div>
);
};
This uses Yarn as a package manager. If yarn has never been used, install it first (https://classic.yarnpkg.com/en/docs/install/#windows-stable).
yarn add -D <package>
for dev dependenciesyarn global add <package>
for global depsyarn install
for installing the projectyarn <script>
to run a script in package.json.
This project uses ESLint as a linter, and Prettier as a code formatter. The .eslintrc.js
file handles the settings for linting, currently it just uses the default "react-app" settings. However, this can be adjusted to adhere to different rules.
module.exports = {
extends: ["react-app", "react-app/jest"],
rules: {
quotes: ["error", "single"],
},
};
The above the project would need single quotes for all strings rather than double, and would throw an error if doing so. If we wanted instead a warning (could still run), we would specify to "warn".
module.exports = {
extends: ["react-app", "react-app/jest"],
rules: {
quotes: ["warn", "single"],
},
};
The rules can be found on eslint.org. To make ESLint ignore some files, we can use the ignorePatterns
option in the .eslintrc.js
file to give some files to ignore:
module.exports = {
extends: ["react-app", "react-app/jest"],
ignorePatterns: ["src/**/__tests/"],
};
The above tells ESLint not to run on any file in a folder called __tests__
that is in any subdirectory (or a direct child of) src
.
module.exports = {
extends: ["react-app", "react-app/jest"],
ignorePatterns: ["**/*.{js,jsx}"],
};
The above tells ESLint not to run on any file in any directory which has the extension of .js
or .jsx
.
Finally, we add TypeScript support via a couple plugins that are already installed, and should be installed with a yarn install
command.
module.exports = {
extends: [
"react-app",
"react-app/jest",
"eslint:recommended",
"plugin:@typescript-eslint/eslint-recommended",
"plugin:@typescript-eslint/recommended",
],
};
Can also run yarn lint
to lint everything in src
. This will show any warnings or errors that won't be autosolved by prettier (see below).
The .prettierrc.js
file contains the formatting that prettier will automatically adhere to.
module.exports = {
tabWidth: 4,
semi: true,
singleQuote: false,
useTabs: true,
};
These rules again can be found online. To get prettier and eslint to work, ensure the following properties are in the VSCode user settings.
"editor.defaultFormatter": "esbenp.prettier-vscode",
"editor.formatOnSave": true,
"eslint.validate": [
"javascript",
"javascriptreact",
{ "language": "typescript", "autoFix": true },
{ "language": "typescriptreact", "autoFix": true }
],
"javascript.updateImportsOnFileMove.enabled": "always",
This will cause prettier to format code on every save. This means that, if the Prettier settings and the ESLint/tsconfig are setup correctly, prettier should always cause no warnings to appear due to ESLint. But they probably still will as is the software life ¯\_(ツ)_/¯.
This project using automated testing using Cypress. This can be run from the terminal yarn cypress open
or by running the cy:dev
script with yarn cy:dev
. This opens the cypress test runner. The tests are from the cypress/integration
folder. Each can be run, or all can be run at once. To get intellisense on cypress commands, add /// <reference types="Cypress" />
to the top of each cypress test file (spec.js
). You can also write these tests in TypeScript if you want, they are in JavaScript to highlight that you can use both TS and JS in the same project without issues.
Before running the tests, ensure that the dev server is running on localhost:3000 in a separate terminal window. TO provide sensitive information in a secure way when doing automated testing (passwords etc), we use a cypress.env.json
file, in which we store our env variables for cypress testing (note: we can also use a normal .env
file and prefix our cypress env vars with CYPRESS_
too).
{
"adminUsername": "admin",
"adminPassword": "password444"
}
// OR in normal .env file
CYPRESS_adminUsername=admin
CYPRESS_adminPassword=password444
Then, in a cypress test, we would write:
describe("First test", () => {
it("Admin login works", () => {
cy.visit("http://localhost:3000");
cy.get("#login-username").type(Cypress.env(`adminUsername`));
cy.get("#login-password").type(Cypress.env(`adminPassword`));
});
});
To run these tests in headless mode, use yarn cy:run
- this is, by default, set to run with no config file (--config-file false), however this can be changed after running yarn cy:dev
once to generate a cypress.json
file, then pointing the config file at this location.
This will get cypress to select the elements on the page with a id
property of login-username/login-password
respectively, and type in each of the values found in the env variables.
NOTE - if you are using this via a Linux shell, you must install the following dependencies (as root user) before cypress will work (see https://docs.cypress.io/guides/guides/continuous-integration.html#Advanced-setup):
apt-get install libgtk2.0-0 libgtk-3-0 libgbm-dev libnotify-dev libgconf-2-4 libnss3 libxss1 libasound2 libxtst6 xauth xvfb
If this fails, you may need to update your current packages.
sudo apt-get update
If an issue with installing xvfb
still occurs after the above, try installing it again:
sudo apt-get install xvfb
Right now, Cypress seems to struggle functioning correctly on WSL1 distros, therefore it is recommended to upgrade to WSL2 to use Cypress correctly.